-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathinvoices.go
158 lines (141 loc) · 6.46 KB
/
invoices.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
package payd
import (
"context"
"time"
"github.com/libsv/go-bt/v2"
validator "github.com/theflyingcodr/govalidator"
"gopkg.in/guregu/null.v3"
)
// InvoiceState enforces invoice states.
type InvoiceState string
// contains states that an invocie can have.
const (
StateInvoicePending InvoiceState = "pending"
StateInvoicePaid InvoiceState = "paid"
StateInvoiceRefunded InvoiceState = "refunded"
StateInvoiceDeleted InvoiceState = "deleted"
)
func (i InvoiceState) String() string {
return string(i)
}
// Invoice identifies a single payment request from this payd wallet,
// it states the amount, id and optional refund address. This indicate
// we are requesting n satoshis in payment.
type Invoice struct {
// ID is a unique identifier for an invoice and can be used
// to lookup a single invoice.
ID string `json:"id" db:"invoice_id"`
// Reference is an identifier that can be used to link the
// PayD invoice with an external system.
Reference null.String `json:"reference" db:"payment_reference"`
// Description is an optional text field that can have some further info
// like 'invoice for oranges'.
Description null.String `json:"description" db:"description"`
// Satoshis is the total amount this invoice is to pay.
Satoshis uint64 `json:"satoshis" db:"satoshis"`
// ExpiresAt is an optional param that can be passed to set an expiration
// date on an invoice, after which, payments will not be accepted.
ExpiresAt null.Time `json:"expiresAt" db:"expires_at"`
// PaymentReceivedAt will be set when this invoice has been paid and
// states when the payment was received in UTC time.
PaymentReceivedAt null.Time `json:"paymentReceivedAt" db:"payment_received_at"`
// RefundTo is an optional paymail address that can be used to refund the
// customer if required.
RefundTo null.String `json:"refundTo" db:"refund_to"`
// RefundedAt if this payment has been refunded, this date will be set
// to the UTC time of the refund.
RefundedAt null.Time `json:"refundedAt" db:"refunded_at"`
// State is the current status of the invoice.
State InvoiceState `json:"state" db:"state" enums:"pending,paid,refunded,deleted"`
// SPVRequired if true will mean this invoice requires a valid spvenvelope otherwise a rawTX will suffice.
SPVRequired bool `json:"-" db:"spv_required"`
MetaData
}
// InvoiceCreate is used to create a new invoice.
type InvoiceCreate struct {
InvoiceID string `json:"-" db:"invoice_id"`
// Satoshis is the total amount this invoice is to pay.
Satoshis uint64 `json:"satoshis" db:"satoshis"`
// Reference is an identifier that can be used to link the
// payd invoice with an external system.
// MaxLength is 32 characters.
Reference null.String `json:"reference" db:"payment_reference" swaggertype:"primitive,string"`
// Description is an optional text field that can have some further info
// like 'invoice for oranges'.
// MaxLength is 1024 characters.
Description null.String `json:"description" db:"description" swaggertype:"primitive,string"`
// CreatedAt is the timestamp when the invoice was created.
CreatedAt time.Time `json:"-" db:"created_at"`
// ExpiresAt is an optional param that can be passed to set an expiration
// date on an invoice, after which, payments will not be accepted.
ExpiresAt null.Time `json:"expiresAt" db:"expires_at"`
// SPVRequired if true will mean this invoice requires a valid spvenvelope otherwise a rawTX will suffice.
SPVRequired bool `json:"-" db:"spv_required"`
}
// Validate will check that InvoiceCreate params match expectations.
func (i InvoiceCreate) Validate(svc TimestampService) error {
return validator.New().
Validate("satoshis", validator.MinUInt64(i.Satoshis, bt.DustLimit)).
Validate("description", validator.StrLength(i.Description.ValueOrZero(), 0, 1024)).
Validate("paymentReference", validator.StrLength(i.Reference.ValueOrZero(), 0, 32)).
Validate("expiresAt", validator.DateAfter(i.ExpiresAt.Time.UTC(), svc.NowUTC())).
Err()
}
// InvoiceUpdatePaid can be used to update an invoice after it has been created.
type InvoiceUpdatePaid struct {
PaymentReceivedAt time.Time `db:"payment_received_at"`
RefundTo string `db:"refund_to"`
}
// InvoiceUpdateRefunded can be used to update an invoice state to refunded.
type InvoiceUpdateRefunded struct {
// RefundTo will set an invoice as refunded.
RefundTo null.String `db:"refund_to"`
// RefundedAt if this payment has been refunded, this date will be set
// to the UTC time of the refund.
RefundedAt null.Time `json:"refundedAt" db:"refunded_at"`
}
// InvoiceUpdateArgs are used to identify the invoice to update.
type InvoiceUpdateArgs struct {
InvoiceID string
}
// InvoiceArgs contains argument/s to return a single invoice.
type InvoiceArgs struct {
InvoiceID string `param:"invoiceID" db:"invoice_id"`
}
// Validate will check that invoice arguments match expectations.
func (i *InvoiceArgs) Validate() error {
return validator.New().
Validate("invoiceID", validator.StrLength(i.InvoiceID, 1, 30)).
Err()
}
// InvoiceService defines a service for managing invoices.
type InvoiceService interface {
Invoice(ctx context.Context, args InvoiceArgs) (*Invoice, error)
Invoices(ctx context.Context) ([]Invoice, error)
InvoicesPending(ctx context.Context) ([]Invoice, error)
Create(ctx context.Context, req InvoiceCreate) (*Invoice, error)
Delete(ctx context.Context, args InvoiceArgs) error
}
// InvoiceReaderWriter can be implemented to support storing and retrieval of invoices.
type InvoiceReaderWriter interface {
InvoiceWriter
InvoiceReader
}
// InvoiceWriter defines a data store used to write invoice data.
type InvoiceWriter interface {
// Create will persist a new Invoice in the data store.
InvoiceCreate(ctx context.Context, req InvoiceCreate) (*Invoice, error)
// Update will update an invoice matching the provided args with the requested changes.
InvoiceUpdate(ctx context.Context, args InvoiceUpdateArgs, req InvoiceUpdatePaid) (*Invoice, error)
// Delete will remove an invoice from the data store, depending on implementation this could
// be a hard or soft delete.
InvoiceDelete(ctx context.Context, args InvoiceArgs) error
}
// InvoiceReader defines a data store used to read invoice data.
type InvoiceReader interface {
// Invoice will return an invoice that matches the provided args.
Invoice(ctx context.Context, args InvoiceArgs) (*Invoice, error)
// Invoices returns all currently stored invoices TODO: update to support search args
Invoices(ctx context.Context) ([]Invoice, error)
InvoicesPending(ctx context.Context) ([]Invoice, error)
}