wallet-core/packages/taler-wallet-core/src/bank-api-client.ts

501 lines
13 KiB
TypeScript
Raw Normal View History

/*
This file is part of GNU Taler
(C) 2022 Taler Systems S.A.
GNU Taler is free software; you can redistribute it and/or modify it under the
terms of the GNU General Public License as published by the Free Software
Foundation; either version 3, or (at your option) any later version.
GNU Taler is distributed in the hope that it will be useful, but WITHOUT ANY
WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
A PARTICULAR PURPOSE. See the GNU General Public License for more details.
You should have received a copy of the GNU General Public License along with
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
*/
/**
* Client for the Taler (demo-)bank.
*/
/**
* Imports.
*/
import {
AmountString,
2022-11-11 20:52:45 +01:00
base64FromArrayBuffer,
buildCodecForObject,
Codec,
2022-03-18 15:32:41 +01:00
codecForAny,
codecForString,
encodeCrock,
2023-05-11 13:04:09 +02:00
generateIban,
getRandomBytes,
2022-03-15 17:51:05 +01:00
j2s,
Logger,
2022-11-11 20:52:45 +01:00
stringToBytes,
2023-02-15 23:32:42 +01:00
TalerError,
TalerErrorCode,
} from "@gnu-taler/taler-util";
2023-02-20 16:37:05 +01:00
import {
2023-04-22 22:17:08 +02:00
checkSuccessResponseOrThrow,
createPlatformHttpLib,
2023-02-20 16:37:05 +01:00
HttpRequestLibrary,
readSuccessResponseJsonOrThrow,
} from "@gnu-taler/taler-util/http";
2022-03-15 17:51:05 +01:00
const logger = new Logger("bank-api-client.ts");
export enum CreditDebitIndicator {
Credit = "credit",
Debit = "debit",
}
export interface BankAccountBalanceResponse {
balance: {
amount: AmountString;
credit_debit_indicator: CreditDebitIndicator;
};
}
export interface BankServiceHandle {
2022-08-23 22:30:05 +02:00
readonly bankAccessApiBaseUrl: string;
readonly http: HttpRequestLibrary;
}
export interface BankUser {
username: string;
password: string;
accountPaytoUri: string;
}
export interface WithdrawalOperationInfo {
withdrawal_id: string;
taler_withdraw_uri: string;
}
/**
* FIXME: Rename, this is not part of the integration test harness anymore.
*/
export interface HarnessExchangeBankAccount {
accountName: string;
accountPassword: string;
accountPaytoUri: string;
wireGatewayApiBaseUrl: string;
}
/**
* Helper function to generate the "Authorization" HTTP header.
*/
function makeBasicAuthHeader(username: string, password: string): string {
const auth = `${username}:${password}`;
2022-11-11 20:52:45 +01:00
const authEncoded: string = base64FromArrayBuffer(stringToBytes(auth));
return `Basic ${authEncoded}`;
}
const codecForWithdrawalOperationInfo = (): Codec<WithdrawalOperationInfo> =>
buildCodecForObject<WithdrawalOperationInfo>()
.property("withdrawal_id", codecForString())
.property("taler_withdraw_uri", codecForString())
.build("WithdrawalOperationInfo");
2023-04-22 22:17:08 +02:00
/**
* @deprecated Use BankAccessApiClient or WireGatewayApi
*/
export namespace BankApi {
2022-08-23 22:30:05 +02:00
// FIXME: Move to BankAccessApi?!
export async function registerAccount(
bank: BankServiceHandle,
username: string,
password: string,
2023-05-11 13:04:09 +02:00
options: {
iban?: string;
},
): Promise<BankUser> {
2022-08-23 22:30:05 +02:00
const url = new URL("testing/register", bank.bankAccessApiBaseUrl);
2023-05-11 13:04:09 +02:00
const resp = await bank.http.postJson(url.href, {
username,
password,
2023-05-11 13:05:49 +02:00
iban: options?.iban,
2023-05-11 13:04:09 +02:00
});
let paytoUri = `payto://x-taler-bank/localhost/${username}`;
2022-08-25 23:35:29 +02:00
if (resp.status !== 200 && resp.status !== 202 && resp.status !== 204) {
2022-03-18 15:32:41 +01:00
logger.error(`${j2s(await resp.json())}`);
throw TalerError.fromDetail(
TalerErrorCode.GENERIC_UNEXPECTED_REQUEST_ERROR,
{
httpStatusCode: resp.status,
},
);
2022-03-18 15:32:41 +01:00
}
try {
// Pybank has no body, thus this might throw.
const respJson = await resp.json();
// LibEuFin demobank returns payto URI in response
if (respJson.paytoUri) {
paytoUri = respJson.paytoUri;
}
} catch (e) {
// Do nothing
}
return {
password,
username,
accountPaytoUri: paytoUri,
};
}
2022-08-23 22:30:05 +02:00
// FIXME: Move to BankAccessApi?!
export async function createRandomBankUser(
bank: BankServiceHandle,
): Promise<BankUser> {
const username = "user-" + encodeCrock(getRandomBytes(10)).toLowerCase();
const password = "pw-" + encodeCrock(getRandomBytes(10)).toLowerCase();
2023-05-11 13:04:09 +02:00
// FIXME: This is just a temporary workaround, because demobank is running out of short IBANs
const iban = generateIban("DE", 15);
return await registerAccount(bank, username, password, {
iban,
});
}
export async function confirmWithdrawalOperation(
bank: BankServiceHandle,
bankUser: BankUser,
wopi: WithdrawalOperationInfo,
): Promise<void> {
const url = new URL(
`accounts/${bankUser.username}/withdrawals/${wopi.withdrawal_id}/confirm`,
2022-08-23 22:30:05 +02:00
bank.bankAccessApiBaseUrl,
);
2022-08-23 22:30:05 +02:00
logger.info(`confirming withdrawal operation via ${url.href}`);
const resp = await bank.http.postJson(
url.href,
{},
{
headers: {
Authorization: makeBasicAuthHeader(
bankUser.username,
bankUser.password,
),
},
},
);
2022-08-23 22:30:05 +02:00
logger.info(`response status ${resp.status}`);
const respJson = await readSuccessResponseJsonOrThrow(resp, codecForAny());
2022-08-23 22:30:05 +02:00
// FIXME: We don't check the status here!
}
export async function abortWithdrawalOperation(
bank: BankServiceHandle,
bankUser: BankUser,
wopi: WithdrawalOperationInfo,
): Promise<void> {
const url = new URL(
`accounts/${bankUser.username}/withdrawals/${wopi.withdrawal_id}/abort`,
2022-08-25 23:35:29 +02:00
bank.bankAccessApiBaseUrl,
);
const resp = await bank.http.postJson(
url.href,
{},
{
headers: {
Authorization: makeBasicAuthHeader(
bankUser.username,
bankUser.password,
),
},
},
);
await readSuccessResponseJsonOrThrow(resp, codecForAny());
}
}
2023-04-22 22:17:08 +02:00
/**
* @deprecated use BankAccessApiClient
*/
export namespace BankAccessApi {
export async function getAccountBalance(
bank: BankServiceHandle,
bankUser: BankUser,
): Promise<BankAccountBalanceResponse> {
2022-08-23 22:30:05 +02:00
const url = new URL(
`accounts/${bankUser.username}`,
bank.bankAccessApiBaseUrl,
);
const resp = await bank.http.get(url.href, {
headers: {
Authorization: makeBasicAuthHeader(
bankUser.username,
bankUser.password,
),
},
});
return await resp.json();
}
export async function createWithdrawalOperation(
bank: BankServiceHandle,
bankUser: BankUser,
amount: string,
): Promise<WithdrawalOperationInfo> {
const url = new URL(
`accounts/${bankUser.username}/withdrawals`,
2022-08-23 22:30:05 +02:00
bank.bankAccessApiBaseUrl,
);
const resp = await bank.http.postJson(
url.href,
{
amount,
},
{
headers: {
Authorization: makeBasicAuthHeader(
bankUser.username,
bankUser.password,
),
},
},
);
return readSuccessResponseJsonOrThrow(
resp,
codecForWithdrawalOperationInfo(),
);
}
}
export interface BankAccessApiClientArgs {
baseUrl: string;
2023-08-24 18:29:54 +02:00
auth?: { username: string; password: string };
2023-04-24 17:53:20 +02:00
enableThrottling?: boolean;
allowHttp?: boolean;
}
export interface BankAccessApiCreateTransactionRequest {
amount: AmountString;
paytoUri: string;
}
2023-04-22 22:17:08 +02:00
export class WireGatewayApiClientArgs {
accountName: string;
accountPassword: string;
wireGatewayApiBaseUrl: string;
2023-04-24 17:53:20 +02:00
enableThrottling?: boolean;
allowHttp?: boolean;
2023-04-22 22:17:08 +02:00
}
2023-04-24 17:53:20 +02:00
/**
* This API look like it belongs to harness
* but it will be nice to have in utils to be used by others
*/
2023-04-22 22:17:08 +02:00
export class WireGatewayApiClient {
2023-04-24 17:53:20 +02:00
httpLib;
2023-04-22 22:17:08 +02:00
2023-04-24 17:53:20 +02:00
constructor(private args: WireGatewayApiClientArgs) {
this.httpLib = createPlatformHttpLib({
enableThrottling: !!args.enableThrottling,
allowHttp: !!args.allowHttp,
});
}
2023-04-22 22:17:08 +02:00
async adminAddIncoming(params: {
amount: string;
reservePub: string;
debitAccountPayto: string;
}): Promise<void> {
2023-04-24 17:53:20 +02:00
let url = new URL(`admin/add-incoming`, this.args.wireGatewayApiBaseUrl);
2023-04-22 22:17:08 +02:00
const resp = await this.httpLib.fetch(url.href, {
method: "POST",
body: {
amount: params.amount,
reserve_pub: params.reservePub,
debit_account: params.debitAccountPayto,
},
headers: {
Authorization: makeBasicAuthHeader(
this.args.accountName,
this.args.accountPassword,
),
},
});
logger.info(`add-incoming response status: ${resp.status}`);
await checkSuccessResponseOrThrow(resp);
}
}
2023-04-24 17:53:20 +02:00
/**
* This API look like it belongs to harness
* but it will be nice to have in utils to be used by others
*/
export class BankAccessApiClient {
2023-08-23 14:40:23 +02:00
httpLib: HttpRequestLibrary;
2023-04-24 17:53:20 +02:00
constructor(private args: BankAccessApiClientArgs) {
this.httpLib = createPlatformHttpLib({
enableThrottling: !!args.enableThrottling,
allowHttp: !!args.allowHttp,
});
}
2023-08-24 18:29:54 +02:00
setAuth(auth: { username: string; password: string }) {
this.args.auth = auth;
}
private makeAuthHeader(): Record<string, string> {
if (!this.args.auth) {
return {};
}
const authHeaderValue = makeBasicAuthHeader(
this.args.auth.username,
this.args.auth.password,
);
return {
Authorization: authHeaderValue,
};
}
async getTransactions(username: string): Promise<void> {
const auth = this.args.auth;
const reqUrl = new URL(
2023-08-24 18:29:54 +02:00
`accounts/${username}/transactions`,
this.args.baseUrl,
);
const resp = await this.httpLib.fetch(reqUrl.href, {
method: "GET",
headers: {
2023-08-24 18:29:54 +02:00
...this.makeAuthHeader(),
},
});
const res = await readSuccessResponseJsonOrThrow(resp, codecForAny());
logger.info(`result: ${j2s(res)}`);
}
async createTransaction(
2023-08-24 18:29:54 +02:00
username: string,
req: BankAccessApiCreateTransactionRequest,
): Promise<any> {
const reqUrl = new URL(
2023-08-24 18:29:54 +02:00
`accounts/${username}/transactions`,
this.args.baseUrl,
);
2023-08-24 18:29:54 +02:00
const resp = await this.httpLib.fetch(reqUrl.href, {
method: "POST",
body: req,
2023-08-24 18:29:54 +02:00
headers: this.makeAuthHeader(),
});
return await readSuccessResponseJsonOrThrow(resp, codecForAny());
}
2023-08-24 18:29:54 +02:00
async registerAccount(
username: string,
password: string,
options: {
iban?: string;
},
): Promise<BankUser> {
const url = new URL("testing/register", this.args.baseUrl);
const resp = await this.httpLib.fetch(url.href, {
method: "POST",
body: {
username,
password,
iban: options?.iban,
},
});
let paytoUri = `payto://x-taler-bank/localhost/${username}`;
if (resp.status !== 200 && resp.status !== 202 && resp.status !== 204) {
logger.error(`${j2s(await resp.json())}`);
throw TalerError.fromDetail(
TalerErrorCode.GENERIC_UNEXPECTED_REQUEST_ERROR,
{
httpStatusCode: resp.status,
},
);
}
try {
// Pybank has no body, thus this might throw.
const respJson = await resp.json();
// LibEuFin demobank returns payto URI in response
if (respJson.paytoUri) {
paytoUri = respJson.paytoUri;
}
} catch (e) {
// Do nothing
}
return {
password,
username,
accountPaytoUri: paytoUri,
};
}
async createRandomBankUser(): Promise<BankUser> {
const username = "user-" + encodeCrock(getRandomBytes(10)).toLowerCase();
const password = "pw-" + encodeCrock(getRandomBytes(10)).toLowerCase();
// FIXME: This is just a temporary workaround, because demobank is running out of short IBANs
const iban = generateIban("DE", 15);
return await this.registerAccount(username, password, {
iban,
});
}
async createWithdrawalOperation(
user: string,
amount: string,
): Promise<WithdrawalOperationInfo> {
const url = new URL(`accounts/${user}/withdrawals`, this.args.baseUrl);
const resp = await this.httpLib.fetch(url.href, {
method: "POST",
body: {
amount,
},
headers: this.makeAuthHeader(),
});
return readSuccessResponseJsonOrThrow(
resp,
codecForWithdrawalOperationInfo(),
);
}
async confirmWithdrawalOperation(
username: string,
wopi: WithdrawalOperationInfo,
): Promise<void> {
const url = new URL(
`accounts/${username}/withdrawals/${wopi.withdrawal_id}/confirm`,
this.args.baseUrl,
);
logger.info(`confirming withdrawal operation via ${url.href}`);
const resp = await this.httpLib.fetch(url.href, {
method: "POST",
body: {},
headers: this.makeAuthHeader(),
});
logger.info(`response status ${resp.status}`);
const respJson = await readSuccessResponseJsonOrThrow(resp, codecForAny());
// FIXME: We don't check the status here!
}
async abortWithdrawalOperation(
accountName: string,
wopi: WithdrawalOperationInfo,
): Promise<void> {
const url = new URL(
`accounts/${accountName}/withdrawals/${wopi.withdrawal_id}/abort`,
this.args.baseUrl,
);
const resp = await this.httpLib.fetch(url.href, {
method: "POST",
body: {},
headers: this.makeAuthHeader(),
});
await readSuccessResponseJsonOrThrow(resp, codecForAny());
}
}