2019-10-31 16:56:54 +00:00
|
|
|
// Copyright (C) 2019 Storj Labs, Inc.
|
|
|
|
// See LICENSE for copying information.
|
|
|
|
|
|
|
|
package stripecoinpayments
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"time"
|
|
|
|
|
2021-06-22 01:09:56 +01:00
|
|
|
"github.com/stripe/stripe-go/v72"
|
2022-12-21 20:27:29 +00:00
|
|
|
"github.com/zeebo/errs"
|
2019-10-31 16:56:54 +00:00
|
|
|
|
2020-03-30 10:08:50 +01:00
|
|
|
"storj.io/common/uuid"
|
2019-10-31 16:56:54 +00:00
|
|
|
"storj.io/storj/satellite/payments"
|
|
|
|
)
|
|
|
|
|
|
|
|
// invoices is an implementation of payments.Invoices.
|
2019-11-05 13:16:02 +00:00
|
|
|
//
|
2020-01-29 00:57:15 +00:00
|
|
|
// architecture: Service
|
2019-10-31 16:56:54 +00:00
|
|
|
type invoices struct {
|
|
|
|
service *Service
|
|
|
|
}
|
|
|
|
|
2022-12-21 20:27:29 +00:00
|
|
|
// AttemptPayOverdueInvoices attempts to pay a user's open, overdue invoices.
|
|
|
|
func (invoices *invoices) AttemptPayOverdueInvoices(ctx context.Context, userID uuid.UUID) (err error) {
|
|
|
|
customerID, err := invoices.service.db.Customers().GetCustomerID(ctx, userID)
|
|
|
|
if err != nil {
|
|
|
|
return Error.Wrap(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
params := &stripe.InvoiceListParams{
|
|
|
|
Customer: &customerID,
|
|
|
|
Status: stripe.String(string(stripe.InvoiceStatusOpen)),
|
|
|
|
DueDateRange: &stripe.RangeQueryParams{LesserThan: time.Now().Unix()},
|
|
|
|
}
|
|
|
|
|
|
|
|
var errGrp errs.Group
|
|
|
|
|
|
|
|
invoicesIterator := invoices.service.stripeClient.Invoices().List(params)
|
|
|
|
for invoicesIterator.Next() {
|
|
|
|
stripeInvoice := invoicesIterator.Invoice()
|
|
|
|
|
|
|
|
params := &stripe.InvoicePayParams{}
|
|
|
|
invResponse, err := invoices.service.stripeClient.Invoices().Pay(stripeInvoice.ID, params)
|
|
|
|
if err != nil {
|
|
|
|
errGrp.Add(Error.New("unable to pay invoice %s: %w", stripeInvoice.ID, err))
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if invResponse != nil && invResponse.Status != stripe.InvoiceStatusPaid {
|
|
|
|
errGrp.Add(Error.New("invoice not paid after payment triggered %s", stripeInvoice.ID))
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if err = invoicesIterator.Err(); err != nil {
|
|
|
|
return Error.Wrap(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return errGrp.Err()
|
|
|
|
}
|
|
|
|
|
2021-10-18 23:18:18 +01:00
|
|
|
// List returns a list of invoices for a given payment account.
|
|
|
|
func (invoices *invoices) List(ctx context.Context, userID uuid.UUID) (invoicesList []payments.Invoice, err error) {
|
|
|
|
defer mon.Task()(&ctx, userID)(&err)
|
|
|
|
|
|
|
|
customerID, err := invoices.service.db.Customers().GetCustomerID(ctx, userID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, Error.Wrap(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
params := &stripe.InvoiceListParams{
|
|
|
|
Customer: &customerID,
|
|
|
|
}
|
|
|
|
|
|
|
|
invoicesIterator := invoices.service.stripeClient.Invoices().List(params)
|
|
|
|
for invoicesIterator.Next() {
|
|
|
|
stripeInvoice := invoicesIterator.Invoice()
|
|
|
|
|
|
|
|
total := stripeInvoice.Total
|
|
|
|
for _, line := range stripeInvoice.Lines.Data {
|
|
|
|
// If amount is negative, this is a coupon or a credit line item.
|
|
|
|
// Add them to the total.
|
|
|
|
if line.Amount < 0 {
|
|
|
|
total -= line.Amount
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
invoicesList = append(invoicesList, payments.Invoice{
|
|
|
|
ID: stripeInvoice.ID,
|
|
|
|
Description: stripeInvoice.Description,
|
|
|
|
Amount: total,
|
|
|
|
Status: string(stripeInvoice.Status),
|
|
|
|
Link: stripeInvoice.InvoicePDF,
|
|
|
|
Start: time.Unix(stripeInvoice.PeriodStart, 0),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
if err = invoicesIterator.Err(); err != nil {
|
|
|
|
return nil, Error.Wrap(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return invoicesList, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ListWithDiscounts returns a list of invoices and coupon usages for a given payment account.
|
|
|
|
func (invoices *invoices) ListWithDiscounts(ctx context.Context, userID uuid.UUID) (invoicesList []payments.Invoice, couponUsages []payments.CouponUsage, err error) {
|
2019-10-31 16:56:54 +00:00
|
|
|
defer mon.Task()(&ctx, userID)(&err)
|
|
|
|
|
2019-11-05 13:16:02 +00:00
|
|
|
customerID, err := invoices.service.db.Customers().GetCustomerID(ctx, userID)
|
2019-10-31 16:56:54 +00:00
|
|
|
if err != nil {
|
2021-08-27 01:51:26 +01:00
|
|
|
return nil, nil, Error.Wrap(err)
|
2019-10-31 16:56:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
params := &stripe.InvoiceListParams{
|
|
|
|
Customer: &customerID,
|
|
|
|
}
|
2021-10-22 15:16:27 +01:00
|
|
|
params.AddExpand("data.total_discount_amounts.discount")
|
2019-10-31 16:56:54 +00:00
|
|
|
|
2020-05-15 09:46:41 +01:00
|
|
|
invoicesIterator := invoices.service.stripeClient.Invoices().List(params)
|
2019-10-31 16:56:54 +00:00
|
|
|
for invoicesIterator.Next() {
|
|
|
|
stripeInvoice := invoicesIterator.Invoice()
|
|
|
|
|
2020-05-08 12:14:55 +01:00
|
|
|
total := stripeInvoice.Total
|
|
|
|
for _, line := range stripeInvoice.Lines.Data {
|
|
|
|
// If amount is negative, this is a coupon or a credit line item.
|
|
|
|
// Add them to the total.
|
|
|
|
if line.Amount < 0 {
|
|
|
|
total -= line.Amount
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-31 16:56:54 +00:00
|
|
|
invoicesList = append(invoicesList, payments.Invoice{
|
|
|
|
ID: stripeInvoice.ID,
|
|
|
|
Description: stripeInvoice.Description,
|
2020-05-08 12:14:55 +01:00
|
|
|
Amount: total,
|
2019-10-31 16:56:54 +00:00
|
|
|
Status: string(stripeInvoice.Status),
|
|
|
|
Link: stripeInvoice.InvoicePDF,
|
|
|
|
Start: time.Unix(stripeInvoice.PeriodStart, 0),
|
|
|
|
})
|
2021-08-27 01:51:26 +01:00
|
|
|
|
|
|
|
for _, dcAmt := range stripeInvoice.TotalDiscountAmounts {
|
|
|
|
if dcAmt == nil {
|
|
|
|
return nil, nil, Error.New("discount amount is nil")
|
|
|
|
}
|
|
|
|
|
|
|
|
dc := dcAmt.Discount
|
|
|
|
|
|
|
|
coupon, err := stripeDiscountToPaymentsCoupon(dc)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, Error.Wrap(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
usage := payments.CouponUsage{
|
|
|
|
Coupon: *coupon,
|
|
|
|
Amount: dcAmt.Amount,
|
|
|
|
PeriodStart: time.Unix(stripeInvoice.PeriodStart, 0),
|
|
|
|
PeriodEnd: time.Unix(stripeInvoice.PeriodEnd, 0),
|
|
|
|
}
|
|
|
|
|
|
|
|
if dc.PromotionCode != nil {
|
|
|
|
usage.Coupon.PromoCode = dc.PromotionCode.Code
|
|
|
|
}
|
|
|
|
|
|
|
|
couponUsages = append(couponUsages, usage)
|
|
|
|
}
|
2019-10-31 16:56:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if err = invoicesIterator.Err(); err != nil {
|
2021-08-27 01:51:26 +01:00
|
|
|
return nil, nil, Error.Wrap(err)
|
2019-10-31 16:56:54 +00:00
|
|
|
}
|
|
|
|
|
2021-08-27 01:51:26 +01:00
|
|
|
return invoicesList, couponUsages, nil
|
2019-10-31 16:56:54 +00:00
|
|
|
}
|
2020-07-06 22:31:40 +01:00
|
|
|
|
|
|
|
// CheckPendingItems returns if pending invoice items for a given payment account exist.
|
|
|
|
func (invoices *invoices) CheckPendingItems(ctx context.Context, userID uuid.UUID) (existingItems bool, err error) {
|
|
|
|
defer mon.Task()(&ctx, userID)(&err)
|
|
|
|
|
|
|
|
customerID, err := invoices.service.db.Customers().GetCustomerID(ctx, userID)
|
|
|
|
if err != nil {
|
|
|
|
return false, Error.Wrap(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
params := &stripe.InvoiceItemListParams{
|
|
|
|
Customer: &customerID,
|
|
|
|
Pending: stripe.Bool(true),
|
|
|
|
}
|
|
|
|
|
|
|
|
itemIterator := invoices.service.stripeClient.InvoiceItems().List(params)
|
|
|
|
for itemIterator.Next() {
|
|
|
|
item := itemIterator.InvoiceItem()
|
|
|
|
if item != nil {
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if err = itemIterator.Err(); err != nil {
|
|
|
|
return false, Error.Wrap(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return false, nil
|
|
|
|
}
|