2019-10-17 15:42:18 +01:00
|
|
|
// Copyright (C) 2019 Storj Labs, Inc.
|
|
|
|
// See LICENSE for copying information.
|
|
|
|
|
|
|
|
package consoleapi
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
2019-10-23 18:33:24 +01:00
|
|
|
"io/ioutil"
|
2019-10-17 15:42:18 +01:00
|
|
|
"net/http"
|
|
|
|
|
2019-10-23 18:33:24 +01:00
|
|
|
"github.com/gorilla/mux"
|
|
|
|
"github.com/zeebo/errs"
|
2019-10-17 15:42:18 +01:00
|
|
|
"go.uber.org/zap"
|
|
|
|
monkit "gopkg.in/spacemonkeygo/monkit.v2"
|
|
|
|
|
|
|
|
"storj.io/storj/satellite/console"
|
|
|
|
)
|
|
|
|
|
2019-10-23 18:33:24 +01:00
|
|
|
var (
|
|
|
|
// ErrPaymentsAPI - console payments api error type.
|
|
|
|
ErrPaymentsAPI = errs.Class("console payments api error")
|
|
|
|
mon = monkit.Package()
|
|
|
|
)
|
2019-10-17 15:42:18 +01:00
|
|
|
|
|
|
|
// Payments is an api controller that exposes all payment related functionality
|
|
|
|
type Payments struct {
|
|
|
|
log *zap.Logger
|
|
|
|
service *console.Service
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewPayments is a constructor for api payments controller.
|
|
|
|
func NewPayments(log *zap.Logger, service *console.Service) *Payments {
|
|
|
|
return &Payments{
|
|
|
|
log: log,
|
|
|
|
service: service,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetupAccount creates a payment account for the user.
|
|
|
|
func (p *Payments) SetupAccount(w http.ResponseWriter, r *http.Request) {
|
|
|
|
ctx := r.Context()
|
|
|
|
var err error
|
|
|
|
defer mon.Task()(&ctx)(&err)
|
|
|
|
|
2019-10-23 18:33:24 +01:00
|
|
|
err = p.service.Payments().SetupAccount(ctx)
|
|
|
|
if err != nil {
|
|
|
|
if console.ErrUnauthorized.Has(err) {
|
|
|
|
p.serveJSONError(w, http.StatusUnauthorized, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
p.serveJSONError(w, http.StatusInternalServerError, err)
|
2019-10-17 15:42:18 +01:00
|
|
|
return
|
|
|
|
}
|
2019-10-23 18:33:24 +01:00
|
|
|
}
|
2019-10-17 15:42:18 +01:00
|
|
|
|
2019-10-23 18:33:24 +01:00
|
|
|
// AccountBalance returns an integer amount in cents that represents the current balance of payment account.
|
|
|
|
func (p *Payments) AccountBalance(w http.ResponseWriter, r *http.Request) {
|
|
|
|
ctx := r.Context()
|
|
|
|
var err error
|
|
|
|
defer mon.Task()(&ctx)(&err)
|
2019-10-17 15:42:18 +01:00
|
|
|
|
2019-10-23 18:33:24 +01:00
|
|
|
balance, err := p.service.Payments().AccountBalance(ctx)
|
2019-10-17 15:42:18 +01:00
|
|
|
if err != nil {
|
2019-10-23 18:33:24 +01:00
|
|
|
if console.ErrUnauthorized.Has(err) {
|
|
|
|
p.serveJSONError(w, http.StatusUnauthorized, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-10-17 15:42:18 +01:00
|
|
|
p.serveJSONError(w, http.StatusInternalServerError, err)
|
2019-10-23 18:33:24 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
err = json.NewEncoder(w).Encode(balance)
|
|
|
|
if err != nil {
|
|
|
|
p.log.Error("failed to write json balance response", zap.Error(ErrPaymentsAPI.Wrap(err)))
|
2019-10-17 15:42:18 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-23 18:33:24 +01:00
|
|
|
// AddCreditCard is used to save new credit card and attach it to payment account.
|
|
|
|
func (p *Payments) AddCreditCard(w http.ResponseWriter, r *http.Request) {
|
2019-10-17 15:42:18 +01:00
|
|
|
ctx := r.Context()
|
|
|
|
var err error
|
|
|
|
defer mon.Task()(&ctx)(&err)
|
|
|
|
|
2019-10-23 18:33:24 +01:00
|
|
|
bodyBytes, err := ioutil.ReadAll(r.Body)
|
|
|
|
if err != nil {
|
|
|
|
p.serveJSONError(w, http.StatusBadRequest, err)
|
2019-10-17 15:42:18 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-10-23 18:33:24 +01:00
|
|
|
token := string(bodyBytes)
|
2019-10-17 15:42:18 +01:00
|
|
|
|
2019-10-23 18:33:24 +01:00
|
|
|
err = p.service.Payments().AddCreditCard(ctx, token)
|
2019-10-17 15:42:18 +01:00
|
|
|
if err != nil {
|
2019-10-23 18:33:24 +01:00
|
|
|
if console.ErrUnauthorized.Has(err) {
|
|
|
|
p.serveJSONError(w, http.StatusUnauthorized, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-10-17 15:42:18 +01:00
|
|
|
p.serveJSONError(w, http.StatusInternalServerError, err)
|
|
|
|
return
|
|
|
|
}
|
2019-10-23 18:33:24 +01:00
|
|
|
}
|
2019-10-17 15:42:18 +01:00
|
|
|
|
2019-10-23 18:33:24 +01:00
|
|
|
// ListCreditCards returns a list of credit cards for a given payment account.
|
|
|
|
func (p *Payments) ListCreditCards(w http.ResponseWriter, r *http.Request) {
|
|
|
|
ctx := r.Context()
|
|
|
|
var err error
|
|
|
|
defer mon.Task()(&ctx)(&err)
|
|
|
|
|
|
|
|
cards, err := p.service.Payments().ListCreditCards(ctx)
|
|
|
|
if err != nil {
|
|
|
|
if console.ErrUnauthorized.Has(err) {
|
|
|
|
p.serveJSONError(w, http.StatusUnauthorized, err)
|
|
|
|
return
|
|
|
|
}
|
2019-10-17 15:42:18 +01:00
|
|
|
|
2019-10-23 18:33:24 +01:00
|
|
|
p.serveJSONError(w, http.StatusInternalServerError, err)
|
|
|
|
return
|
|
|
|
}
|
2019-10-17 15:42:18 +01:00
|
|
|
|
2019-10-23 18:33:24 +01:00
|
|
|
err = json.NewEncoder(w).Encode(cards)
|
2019-10-17 15:42:18 +01:00
|
|
|
if err != nil {
|
2019-10-23 18:33:24 +01:00
|
|
|
p.log.Error("failed to write json list cards response", zap.Error(ErrPaymentsAPI.Wrap(err)))
|
2019-10-17 15:42:18 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-23 18:33:24 +01:00
|
|
|
// MakeCreditCardDefault makes a credit card default payment method.
|
|
|
|
func (p *Payments) MakeCreditCardDefault(w http.ResponseWriter, r *http.Request) {
|
2019-10-17 15:42:18 +01:00
|
|
|
ctx := r.Context()
|
|
|
|
var err error
|
|
|
|
defer mon.Task()(&ctx)(&err)
|
|
|
|
|
2019-10-23 18:33:24 +01:00
|
|
|
cardID, err := ioutil.ReadAll(r.Body)
|
|
|
|
if err != nil {
|
|
|
|
p.serveJSONError(w, http.StatusBadRequest, err)
|
2019-10-17 15:42:18 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-10-23 18:33:24 +01:00
|
|
|
err = p.service.Payments().MakeCreditCardDefault(ctx, string(cardID))
|
|
|
|
if err != nil {
|
|
|
|
if console.ErrUnauthorized.Has(err) {
|
|
|
|
p.serveJSONError(w, http.StatusUnauthorized, err)
|
|
|
|
return
|
|
|
|
}
|
2019-10-17 15:42:18 +01:00
|
|
|
|
2019-10-23 18:33:24 +01:00
|
|
|
p.serveJSONError(w, http.StatusInternalServerError, err)
|
|
|
|
return
|
2019-10-17 15:42:18 +01:00
|
|
|
}
|
2019-10-23 18:33:24 +01:00
|
|
|
}
|
2019-10-17 15:42:18 +01:00
|
|
|
|
2019-10-23 18:33:24 +01:00
|
|
|
// RemoveCreditCard is used to detach a credit card from payment account.
|
|
|
|
func (p *Payments) RemoveCreditCard(w http.ResponseWriter, r *http.Request) {
|
|
|
|
ctx := r.Context()
|
|
|
|
var err error
|
|
|
|
defer mon.Task()(&ctx)(&err)
|
2019-10-17 15:42:18 +01:00
|
|
|
|
2019-10-23 18:33:24 +01:00
|
|
|
vars := mux.Vars(r)
|
|
|
|
cardID := vars["cardId"]
|
|
|
|
|
|
|
|
if cardID == "" {
|
2019-10-17 15:42:18 +01:00
|
|
|
p.serveJSONError(w, http.StatusBadRequest, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-10-23 18:33:24 +01:00
|
|
|
err = p.service.Payments().RemoveCreditCard(ctx, cardID)
|
2019-10-17 15:42:18 +01:00
|
|
|
if err != nil {
|
2019-10-23 18:33:24 +01:00
|
|
|
if console.ErrUnauthorized.Has(err) {
|
|
|
|
p.serveJSONError(w, http.StatusUnauthorized, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-10-17 15:42:18 +01:00
|
|
|
p.serveJSONError(w, http.StatusInternalServerError, err)
|
2019-10-23 18:33:24 +01:00
|
|
|
return
|
2019-10-17 15:42:18 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-31 16:56:54 +00:00
|
|
|
// BillingHistory returns a list of invoices, transactions and all others billing history items for payment account.
|
|
|
|
func (p *Payments) BillingHistory(w http.ResponseWriter, r *http.Request) {
|
|
|
|
ctx := r.Context()
|
|
|
|
var err error
|
|
|
|
defer mon.Task()(&ctx)(&err)
|
|
|
|
|
|
|
|
billingHistory, err := p.service.Payments().BillingHistory(ctx)
|
|
|
|
if err != nil {
|
|
|
|
if console.ErrUnauthorized.Has(err) {
|
|
|
|
p.serveJSONError(w, http.StatusUnauthorized, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
p.serveJSONError(w, http.StatusInternalServerError, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
err = json.NewEncoder(w).Encode(billingHistory)
|
|
|
|
if err != nil {
|
|
|
|
p.log.Error("failed to write json billing history response", zap.Error(ErrPaymentsAPI.Wrap(err)))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-17 15:42:18 +01:00
|
|
|
// serveJSONError writes JSON error to response output stream.
|
|
|
|
func (p *Payments) serveJSONError(w http.ResponseWriter, status int, err error) {
|
|
|
|
w.WriteHeader(status)
|
|
|
|
|
|
|
|
var response struct {
|
|
|
|
Error string `json:"error"`
|
|
|
|
}
|
|
|
|
|
|
|
|
response.Error = err.Error()
|
|
|
|
|
|
|
|
err = json.NewEncoder(w).Encode(response)
|
|
|
|
if err != nil {
|
2019-10-23 18:33:24 +01:00
|
|
|
p.log.Error("failed to write json error response", zap.Error(ErrPaymentsAPI.Wrap(err)))
|
2019-10-17 15:42:18 +01:00
|
|
|
}
|
|
|
|
}
|