2019-10-23 18:33:24 +01:00
|
|
|
// Copyright (C) 2019 Storj Labs, Inc.
|
|
|
|
// See LICENSE for copying information.
|
|
|
|
|
|
|
|
import { ErrorUnauthorized } from '@/api/errors/ErrorUnauthorized';
|
2019-11-25 12:59:41 +00:00
|
|
|
import { BillingHistoryItem, CreditCard, PaymentsApi, ProjectCharge, TokenDeposit } from '@/types/payments';
|
2019-10-23 18:33:24 +01:00
|
|
|
import { HttpClient } from '@/utils/httpClient';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* PaymentsHttpApi is a http implementation of Payments API.
|
|
|
|
* Exposes all payments-related functionality
|
|
|
|
*/
|
|
|
|
export class PaymentsHttpApi implements PaymentsApi {
|
|
|
|
private readonly client: HttpClient = new HttpClient();
|
|
|
|
private readonly ROOT_PATH: string = '/api/v0/payments';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get account balance
|
|
|
|
*
|
|
|
|
* @returns balance in cents
|
|
|
|
* @throws Error
|
|
|
|
*/
|
|
|
|
public async getBalance(): Promise<number> {
|
|
|
|
const path = `${this.ROOT_PATH}/account/balance`;
|
|
|
|
const response = await this.client.get(path);
|
|
|
|
|
|
|
|
if (response.ok) {
|
|
|
|
return await response.json();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (response.status === 401) {
|
|
|
|
throw new ErrorUnauthorized();
|
|
|
|
}
|
|
|
|
|
|
|
|
throw new Error('can not get balance');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Try to set up a payment account
|
|
|
|
*
|
|
|
|
* @throws Error
|
|
|
|
*/
|
|
|
|
public async setupAccount(): Promise<void> {
|
|
|
|
const path = `${this.ROOT_PATH}/account`;
|
|
|
|
const response = await this.client.post(path, null);
|
|
|
|
|
|
|
|
if (response.ok) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (response.status === 401) {
|
|
|
|
throw new ErrorUnauthorized();
|
|
|
|
}
|
|
|
|
|
|
|
|
throw new Error('can not setup account');
|
|
|
|
}
|
|
|
|
|
2019-11-20 13:46:22 +00:00
|
|
|
/**
|
|
|
|
* projectsCharges returns how much money current user will be charged for each project which he owns.
|
|
|
|
*/
|
|
|
|
public async projectsCharges(): Promise<ProjectCharge[]> {
|
2019-11-15 14:27:44 +00:00
|
|
|
const path = `${this.ROOT_PATH}/account/charges`;
|
|
|
|
const response = await this.client.get(path);
|
|
|
|
|
|
|
|
if (!response.ok) {
|
|
|
|
if (response.status === 401) {
|
|
|
|
throw new ErrorUnauthorized();
|
|
|
|
}
|
|
|
|
|
|
|
|
throw new Error('can not get projects charges');
|
|
|
|
}
|
|
|
|
|
|
|
|
const charges = await response.json();
|
2019-11-20 13:46:22 +00:00
|
|
|
if (charges) {
|
|
|
|
return charges.map(charge =>
|
|
|
|
new ProjectCharge(
|
|
|
|
charge.projectId,
|
|
|
|
charge.storage,
|
|
|
|
charge.egress,
|
|
|
|
charge.objectCount),
|
|
|
|
);
|
|
|
|
}
|
2019-11-15 14:27:44 +00:00
|
|
|
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
2019-10-23 18:33:24 +01:00
|
|
|
/**
|
|
|
|
* Add credit card
|
|
|
|
* @param token - stripe token used to add a credit card as a payment method
|
|
|
|
* @throws Error
|
|
|
|
*/
|
|
|
|
public async addCreditCard(token: string): Promise<void> {
|
|
|
|
const path = `${this.ROOT_PATH}/cards`;
|
|
|
|
const response = await this.client.post(path, token);
|
|
|
|
|
2019-10-29 14:24:16 +00:00
|
|
|
if (response.ok) {
|
2019-10-23 18:33:24 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (response.status === 401) {
|
|
|
|
throw new ErrorUnauthorized();
|
|
|
|
}
|
|
|
|
|
|
|
|
throw new Error('can not add credit card');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Detach credit card from payment account.
|
|
|
|
* @param cardId
|
|
|
|
* @throws Error
|
|
|
|
*/
|
|
|
|
public async removeCreditCard(cardId: string): Promise<void> {
|
|
|
|
const path = `${this.ROOT_PATH}/cards/${cardId}`;
|
|
|
|
const response = await this.client.delete(path);
|
|
|
|
|
|
|
|
if (response.ok) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (response.status === 401) {
|
|
|
|
throw new ErrorUnauthorized();
|
|
|
|
}
|
|
|
|
|
|
|
|
throw new Error('can not remove credit card');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get list of user`s credit cards
|
|
|
|
*
|
|
|
|
* @returns list of credit cards
|
|
|
|
* @throws Error
|
|
|
|
*/
|
|
|
|
public async listCreditCards(): Promise<CreditCard[]> {
|
|
|
|
const path = `${this.ROOT_PATH}/cards`;
|
|
|
|
const response = await this.client.get(path);
|
|
|
|
|
|
|
|
if (!response.ok) {
|
|
|
|
if (response.status === 401) {
|
|
|
|
throw new ErrorUnauthorized();
|
|
|
|
}
|
|
|
|
throw new Error('can not list credit cards');
|
|
|
|
}
|
|
|
|
|
|
|
|
const creditCards = await response.json();
|
|
|
|
|
|
|
|
if (creditCards) {
|
|
|
|
return creditCards.map(card => new CreditCard(card.id, card.expMonth, card.expYear, card.brand, card.last4, card.isDefault));
|
|
|
|
}
|
|
|
|
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Make credit card default
|
|
|
|
* @param cardId
|
|
|
|
* @throws Error
|
|
|
|
*/
|
|
|
|
public async makeCreditCardDefault(cardId: string): Promise<void> {
|
|
|
|
const path = `${this.ROOT_PATH}/cards`;
|
|
|
|
const response = await this.client.patch(path, cardId);
|
|
|
|
|
|
|
|
if (response.ok) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (response.status === 401) {
|
|
|
|
throw new ErrorUnauthorized();
|
|
|
|
}
|
|
|
|
|
|
|
|
throw new Error('can not make credit card default');
|
|
|
|
}
|
2019-10-31 16:56:54 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a list of invoices, transactions and all others billing history items for payment account.
|
|
|
|
*
|
|
|
|
* @returns list of billing history items
|
|
|
|
* @throws Error
|
|
|
|
*/
|
|
|
|
public async billingHistory(): Promise<BillingHistoryItem[]> {
|
|
|
|
const path = `${this.ROOT_PATH}/billing-history`;
|
|
|
|
const response = await this.client.get(path);
|
|
|
|
|
|
|
|
if (!response.ok) {
|
|
|
|
if (response.status === 401) {
|
|
|
|
throw new ErrorUnauthorized();
|
|
|
|
}
|
|
|
|
throw new Error('can not list billing history');
|
|
|
|
}
|
|
|
|
|
|
|
|
const billingHistoryItems = await response.json();
|
|
|
|
|
|
|
|
if (billingHistoryItems) {
|
|
|
|
return billingHistoryItems.map(item =>
|
|
|
|
new BillingHistoryItem(
|
|
|
|
item.id,
|
|
|
|
item.description,
|
|
|
|
item.amount,
|
2019-11-25 12:59:41 +00:00
|
|
|
item.tokenReceived,
|
2019-10-31 16:56:54 +00:00
|
|
|
item.status,
|
|
|
|
item.link,
|
|
|
|
new Date(item.start),
|
|
|
|
new Date(item.end),
|
|
|
|
item.type),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return [];
|
|
|
|
}
|
2019-11-25 12:59:41 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* makeTokenDeposit process coin payments
|
|
|
|
* @param amount
|
|
|
|
* @throws Error
|
|
|
|
*/
|
|
|
|
public async makeTokenDeposit(amount: number): Promise<TokenDeposit> {
|
|
|
|
const path = `${this.ROOT_PATH}/tokens/deposit`;
|
|
|
|
const response = await this.client.post(path, JSON.stringify({ amount }));
|
|
|
|
|
|
|
|
if (!response.ok) {
|
|
|
|
if (response.status === 401) {
|
|
|
|
throw new ErrorUnauthorized();
|
|
|
|
}
|
|
|
|
|
|
|
|
throw new Error('can not process coin payment');
|
|
|
|
}
|
|
|
|
|
|
|
|
const result = await response.json();
|
|
|
|
|
2019-12-12 15:18:47 +00:00
|
|
|
return new TokenDeposit(result.amount, result.address, result.link);
|
2019-11-25 12:59:41 +00:00
|
|
|
}
|
2019-10-23 18:33:24 +01:00
|
|
|
}
|