2019-10-17 15:04:50 +01:00
|
|
|
// Copyright (C) 2019 Storj Labs, Inc.
|
|
|
|
// See LICENSE for copying information.
|
|
|
|
|
|
|
|
package stripecoinpayments
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2020-07-22 18:41:24 +01:00
|
|
|
"encoding/json"
|
2020-05-28 12:31:02 +01:00
|
|
|
"strconv"
|
2020-07-22 18:41:24 +01:00
|
|
|
"strings"
|
2020-05-28 12:31:02 +01:00
|
|
|
"time"
|
|
|
|
|
2021-06-22 01:09:56 +01:00
|
|
|
"github.com/stripe/stripe-go/v72"
|
2020-07-22 18:41:24 +01:00
|
|
|
"go.uber.org/zap"
|
2019-10-17 15:04:50 +01:00
|
|
|
|
2020-03-30 10:08:50 +01:00
|
|
|
"storj.io/common/uuid"
|
2019-10-17 15:04:50 +01:00
|
|
|
"storj.io/storj/satellite/payments"
|
|
|
|
"storj.io/storj/satellite/payments/coinpayments"
|
|
|
|
)
|
|
|
|
|
2020-05-28 12:31:02 +01:00
|
|
|
const (
|
|
|
|
// StripeDepositTransactionDescription is the description for Stripe
|
|
|
|
// balance transactions representing STORJ deposits.
|
|
|
|
StripeDepositTransactionDescription = "STORJ deposit"
|
|
|
|
|
|
|
|
// StripeDepositBonusTransactionDescription is the description for Stripe
|
|
|
|
// balance transactions representing bonuses received for STORJ deposits.
|
|
|
|
StripeDepositBonusTransactionDescription = "STORJ deposit bonus"
|
|
|
|
)
|
|
|
|
|
2019-10-17 15:04:50 +01:00
|
|
|
// ensure that storjTokens implements payments.StorjTokens.
|
|
|
|
var _ payments.StorjTokens = (*storjTokens)(nil)
|
|
|
|
|
|
|
|
// storjTokens implements payments.StorjTokens.
|
2020-01-29 00:57:15 +00:00
|
|
|
//
|
|
|
|
// architecture: Service
|
2019-10-17 15:04:50 +01:00
|
|
|
type storjTokens struct {
|
|
|
|
service *Service
|
|
|
|
}
|
|
|
|
|
2019-11-12 11:14:34 +00:00
|
|
|
// ListTransactionInfos fetches all transactions from the database for specified user, reconstructing checkout link.
|
|
|
|
func (tokens *storjTokens) ListTransactionInfos(ctx context.Context, userID uuid.UUID) (_ []payments.TransactionInfo, err error) {
|
|
|
|
defer mon.Task()(&ctx, userID)(&err)
|
|
|
|
|
|
|
|
txs, err := tokens.service.db.Transactions().ListAccount(ctx, userID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, Error.Wrap(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
var infos []payments.TransactionInfo
|
|
|
|
for _, tx := range txs {
|
|
|
|
link := coinpayments.GetCheckoutURL(tx.Key, tx.ID)
|
|
|
|
|
|
|
|
var status payments.TransactionStatus
|
|
|
|
switch tx.Status {
|
|
|
|
case coinpayments.StatusPending:
|
|
|
|
status = payments.TransactionStatusPending
|
|
|
|
case coinpayments.StatusReceived:
|
|
|
|
status = payments.TransactionStatusPaid
|
|
|
|
case coinpayments.StatusCancelled:
|
|
|
|
status = payments.TransactionStatusCancelled
|
|
|
|
default:
|
|
|
|
// unknown
|
|
|
|
status = payments.TransactionStatus(tx.Status.String())
|
|
|
|
}
|
|
|
|
|
2019-11-21 13:23:16 +00:00
|
|
|
rate, err := tokens.service.db.Transactions().GetLockedRate(ctx, tx.ID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2019-11-12 11:14:34 +00:00
|
|
|
infos = append(infos,
|
|
|
|
payments.TransactionInfo{
|
2019-11-21 13:23:16 +00:00
|
|
|
ID: []byte(tx.ID),
|
satellite/payments: specialized type for monetary amounts
Why: big.Float is not an ideal type for dealing with monetary amounts,
because no matter how high the precision, some non-integer decimal
values can not be represented exactly in base-2 floating point. Also,
storing gob-encoded big.Float values in the database makes it very hard
to use those values in meaningful queries, making it difficult to do
any sort of analysis on billing.
For better accuracy, then, we can just represent monetary values as
integers (in whatever base units are appropriate for the currency). For
example, STORJ tokens or Bitcoins can not be split into pieces smaller
than 10^-8, so we can store amounts of STORJ or BTC with precision
simply by moving the decimal point 8 digits to the right. For USD values
(assuming we don't want to deal with fractional cents), we can move the
decimal point 2 digits to the right.
To make it easier and less error-prone to deal with the math involved, I
introduce here a new type, monetary.Amount, instances of which have an
associated value _and_ a currency.
Change-Id: I03395d52f0e2473cf301361f6033722b54640265
2021-08-10 23:29:50 +01:00
|
|
|
Amount: tx.Amount,
|
|
|
|
Received: tx.Received,
|
|
|
|
AmountCents: convertToCents(rate, tx.Amount),
|
|
|
|
ReceivedCents: convertToCents(rate, tx.Received),
|
2019-11-21 13:23:16 +00:00
|
|
|
Address: tx.Address,
|
|
|
|
Status: status,
|
|
|
|
Link: link,
|
|
|
|
ExpiresAt: tx.CreatedAt.Add(tx.Timeout),
|
|
|
|
CreatedAt: tx.CreatedAt,
|
2019-11-12 11:14:34 +00:00
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
return infos, nil
|
|
|
|
}
|
2020-05-28 12:31:02 +01:00
|
|
|
|
|
|
|
// ListDepositBonuses returns all deposit bonuses from Stripe associated with user.
|
|
|
|
func (tokens *storjTokens) ListDepositBonuses(ctx context.Context, userID uuid.UUID) (_ []payments.DepositBonus, err error) {
|
|
|
|
defer mon.Task()(&ctx, userID)(&err)
|
|
|
|
|
|
|
|
cusID, err := tokens.service.db.Customers().GetCustomerID(ctx, userID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var bonuses []payments.DepositBonus
|
2020-07-22 18:41:24 +01:00
|
|
|
|
2023-03-14 02:59:24 +00:00
|
|
|
params := &stripe.CustomerParams{Params: stripe.Params{Context: ctx}}
|
|
|
|
customer, err := tokens.service.stripeClient.Customers().Get(cusID, params)
|
2020-07-22 18:41:24 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
for key, value := range customer.Metadata {
|
|
|
|
if !strings.HasPrefix(key, "credit_") {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
var credit payments.Credit
|
|
|
|
err = json.Unmarshal([]byte(value), &credit)
|
|
|
|
if err != nil {
|
|
|
|
tokens.service.log.Error("Error unmarshaling credit history from Stripe metadata",
|
|
|
|
zap.String("Customer ID", cusID),
|
|
|
|
zap.String("Metadata Key", key),
|
|
|
|
zap.String("Metadata Value", value),
|
|
|
|
zap.Error(err),
|
|
|
|
)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
bonuses = append(bonuses,
|
|
|
|
payments.DepositBonus{
|
|
|
|
TransactionID: payments.TransactionID(credit.TransactionID),
|
|
|
|
AmountCents: credit.Amount,
|
|
|
|
Percentage: 10,
|
|
|
|
CreatedAt: credit.Created,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2023-03-14 02:59:24 +00:00
|
|
|
it := tokens.service.stripeClient.CustomerBalanceTransactions().List(&stripe.CustomerBalanceTransactionListParams{
|
|
|
|
ListParams: stripe.ListParams{Context: ctx},
|
|
|
|
Customer: stripe.String(cusID),
|
|
|
|
})
|
2020-05-28 12:31:02 +01:00
|
|
|
for it.Next() {
|
|
|
|
tx := it.CustomerBalanceTransaction()
|
|
|
|
|
|
|
|
if tx.Type != stripe.CustomerBalanceTransactionTypeAdjustment {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if tx.Description != StripeDepositBonusTransactionDescription {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
percentage := int64(10)
|
|
|
|
percentageStr, ok := tx.Metadata["percentage"]
|
|
|
|
if ok {
|
|
|
|
percentage, err = strconv.ParseInt(percentageStr, 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bonuses = append(bonuses,
|
|
|
|
payments.DepositBonus{
|
|
|
|
TransactionID: []byte(tx.Metadata["txID"]),
|
|
|
|
AmountCents: -tx.Amount,
|
|
|
|
Percentage: percentage,
|
|
|
|
CreatedAt: time.Unix(tx.Created, 0),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
return bonuses, nil
|
|
|
|
}
|