wallet-core/packages/taler-wallet-webextension/src/cta/Payment/test.ts

525 lines
15 KiB
TypeScript
Raw Normal View History

2022-04-21 19:23:53 +02:00
/*
This file is part of GNU Taler
2022-06-06 17:05:26 +02:00
(C) 2022 Taler Systems S.A.
2022-04-21 19:23:53 +02:00
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/>
*/
/**
*
* @author Sebastian Javier Marchano (sebasjm)
*/
2022-06-06 05:09:25 +02:00
import {
Amounts,
ConfirmPayResult,
2022-06-06 05:09:25 +02:00
ConfirmPayResultType,
NotificationType,
PreparePayResultInsufficientBalance,
2022-10-25 17:23:08 +02:00
PreparePayResultPaymentPossible,
PreparePayResultType,
2022-06-06 05:09:25 +02:00
} from "@gnu-taler/taler-util";
2022-10-25 17:23:08 +02:00
import { WalletApiOperation } from "@gnu-taler/taler-wallet-core";
2022-04-21 19:23:53 +02:00
import { expect } from "chai";
import { tests } from "../../../../web-util/src/index.browser.js";
import { ErrorAlert, useAlertContext } from "../../context/alert.js";
import { nullFunction } from "../../mui/handlers.js";
2022-10-25 17:23:08 +02:00
import { createWalletApiMock } from "../../test-utils.js";
2022-07-31 01:55:41 +02:00
import { useComponentState } from "./state.js";
2022-04-21 19:23:53 +02:00
2022-07-31 01:55:41 +02:00
describe("Payment CTA states", () => {
2022-04-21 19:23:53 +02:00
it("should tell the user that the URI is missing", async () => {
const { handler, TestingContext } = createWalletApiMock();
2022-10-25 17:23:08 +02:00
const props = {
talerPayUri: undefined,
cancel: nullFunction,
goToWalletManualWithdraw: nullFunction,
onSuccess: nullFunction,
};
2022-04-21 19:23:53 +02:00
2022-12-15 21:12:03 +01:00
const hookBehavior = await tests.hookBehaveLikeThis(
useComponentState,
props,
[
({ status, error }) => {
expect(status).equals("loading");
expect(error).undefined;
},
({ status, error }) => {
2023-01-09 12:38:48 +01:00
expect(status).equals("error");
2022-12-15 21:12:03 +01:00
if (error === undefined) expect.fail();
2023-01-09 12:38:48 +01:00
// expect(error.hasError).true;
// expect(error.operational).false;
2022-12-15 21:12:03 +01:00
},
],
TestingContext,
);
2022-04-21 19:23:53 +02:00
2022-12-15 21:12:03 +01:00
expect(hookBehavior).deep.equal({ result: "ok" });
expect(handler.getCallingQueueState()).eq("empty");
2022-04-21 19:23:53 +02:00
});
it("should response with no balance", async () => {
const { handler, TestingContext } = createWalletApiMock();
2022-10-25 17:23:08 +02:00
const props = {
talerPayUri: "taller://pay",
cancel: nullFunction,
goToWalletManualWithdraw: nullFunction,
onSuccess: nullFunction,
};
2022-10-25 17:23:08 +02:00
handler.addWalletCallResponse(
WalletApiOperation.PreparePayForUri,
undefined,
{
status: PreparePayResultType.InsufficientBalance,
amountRaw: "USD:10",
} as PreparePayResultInsufficientBalance,
);
handler.addWalletCallResponse(
WalletApiOperation.GetBalances,
{},
{ balances: [] },
);
2022-10-25 17:23:08 +02:00
2022-12-15 21:12:03 +01:00
const hookBehavior = await tests.hookBehaveLikeThis(
useComponentState,
props,
[
({ status, error }) => {
expect(status).equals("loading");
expect(error).undefined;
},
(state) => {
if (state.status !== "no-balance-for-currency") {
expect(state).eq({});
return;
}
expect(state.balance).undefined;
expect(state.amount).deep.equal(Amounts.parseOrThrow("USD:10"));
},
],
TestingContext,
);
2022-04-21 19:23:53 +02:00
2022-12-15 21:12:03 +01:00
expect(hookBehavior).deep.equal({ result: "ok" });
expect(handler.getCallingQueueState()).eq("empty");
2022-04-21 19:23:53 +02:00
});
it("should not be able to pay if there is no enough balance", async () => {
const { handler, TestingContext } = createWalletApiMock();
2022-10-25 17:23:08 +02:00
const props = {
talerPayUri: "taller://pay",
cancel: nullFunction,
goToWalletManualWithdraw: nullFunction,
onSuccess: nullFunction,
};
handler.addWalletCallResponse(
WalletApiOperation.PreparePayForUri,
undefined,
{
status: PreparePayResultType.InsufficientBalance,
amountRaw: "USD:10",
} as PreparePayResultInsufficientBalance,
);
handler.addWalletCallResponse(
WalletApiOperation.GetBalances,
{},
{
balances: [
{
available: "USD:5",
hasPendingTransactions: false,
pendingIncoming: "USD:0",
pendingOutgoing: "USD:0",
requiresUserInput: false,
},
],
},
);
2022-10-25 17:23:08 +02:00
2022-12-15 21:12:03 +01:00
const hookBehavior = await tests.hookBehaveLikeThis(
useComponentState,
props,
[
({ status, error }) => {
expect(status).equals("loading");
expect(error).undefined;
},
(state) => {
if (state.status !== "no-enough-balance") expect.fail();
expect(state.balance).deep.equal(Amounts.parseOrThrow("USD:5"));
expect(state.amount).deep.equal(Amounts.parseOrThrow("USD:10"));
},
],
TestingContext,
);
2022-04-21 19:23:53 +02:00
2022-12-15 21:12:03 +01:00
expect(hookBehavior).deep.equal({ result: "ok" });
expect(handler.getCallingQueueState()).eq("empty");
2022-04-21 19:23:53 +02:00
});
it("should be able to pay (without fee)", async () => {
const { handler, TestingContext } = createWalletApiMock();
2022-10-25 17:23:08 +02:00
const props = {
talerPayUri: "taller://pay",
cancel: nullFunction,
goToWalletManualWithdraw: nullFunction,
onSuccess: nullFunction,
};
handler.addWalletCallResponse(
WalletApiOperation.PreparePayForUri,
undefined,
{
status: PreparePayResultType.PaymentPossible,
amountRaw: "USD:10",
amountEffective: "USD:10",
} as PreparePayResultPaymentPossible,
);
handler.addWalletCallResponse(
WalletApiOperation.GetBalances,
{},
{
balances: [
{
available: "USD:15",
hasPendingTransactions: false,
pendingIncoming: "USD:0",
pendingOutgoing: "USD:0",
requiresUserInput: false,
},
],
},
);
2022-12-15 21:12:03 +01:00
const hookBehavior = await tests.hookBehaveLikeThis(
useComponentState,
props,
[
({ status, error }) => {
expect(status).equals("loading");
expect(error).undefined;
},
(state) => {
if (state.status !== "ready") {
expect(state).eq({});
return;
}
expect(state.balance).deep.equal(Amounts.parseOrThrow("USD:15"));
expect(state.amount).deep.equal(Amounts.parseOrThrow("USD:10"));
expect(state.payHandler.onClick).not.undefined;
},
],
TestingContext,
);
2022-12-15 21:12:03 +01:00
expect(hookBehavior).deep.equal({ result: "ok" });
expect(handler.getCallingQueueState()).eq("empty");
2022-04-21 19:23:53 +02:00
});
it("should be able to pay (with fee)", async () => {
const { handler, TestingContext } = createWalletApiMock();
2022-10-25 17:23:08 +02:00
const props = {
talerPayUri: "taller://pay",
cancel: nullFunction,
goToWalletManualWithdraw: nullFunction,
onSuccess: nullFunction,
};
handler.addWalletCallResponse(
WalletApiOperation.PreparePayForUri,
undefined,
{
status: PreparePayResultType.PaymentPossible,
amountRaw: "USD:9",
amountEffective: "USD:10",
} as PreparePayResultPaymentPossible,
);
handler.addWalletCallResponse(
WalletApiOperation.GetBalances,
{},
{
balances: [
{
available: "USD:15",
hasPendingTransactions: false,
pendingIncoming: "USD:0",
pendingOutgoing: "USD:0",
requiresUserInput: false,
},
],
},
);
2022-12-15 21:12:03 +01:00
const hookBehavior = await tests.hookBehaveLikeThis(
useComponentState,
props,
[
({ status, error }) => {
expect(status).equals("loading");
expect(error).undefined;
},
(state) => {
if (state.status !== "ready") expect.fail();
expect(state.balance).deep.equal(Amounts.parseOrThrow("USD:15"));
expect(state.amount).deep.equal(Amounts.parseOrThrow("USD:9"));
expect(state.payHandler.onClick).not.undefined;
},
],
TestingContext,
);
2022-12-15 21:12:03 +01:00
expect(hookBehavior).deep.equal({ result: "ok" });
expect(handler.getCallingQueueState()).eq("empty");
2022-04-21 19:23:53 +02:00
});
it("should get confirmation done after pay successfully", async () => {
const { handler, TestingContext } = createWalletApiMock();
2022-10-25 17:23:08 +02:00
const props = {
talerPayUri: "taller://pay",
cancel: nullFunction,
goToWalletManualWithdraw: nullFunction,
onSuccess: nullFunction,
};
handler.addWalletCallResponse(
WalletApiOperation.PreparePayForUri,
undefined,
{
status: PreparePayResultType.PaymentPossible,
amountRaw: "USD:9",
amountEffective: "USD:10",
} as PreparePayResultPaymentPossible,
);
handler.addWalletCallResponse(
WalletApiOperation.GetBalances,
{},
{
balances: [
{
available: "USD:15",
hasPendingTransactions: false,
pendingIncoming: "USD:0",
pendingOutgoing: "USD:0",
requiresUserInput: false,
},
],
},
);
2022-10-25 17:23:08 +02:00
handler.addWalletCallResponse(WalletApiOperation.ConfirmPay, undefined, {
type: ConfirmPayResultType.Done,
contractTerms: {},
} as ConfirmPayResult);
2022-10-25 17:23:08 +02:00
2022-12-15 21:12:03 +01:00
const hookBehavior = await tests.hookBehaveLikeThis(
useComponentState,
props,
[
({ status, error }) => {
expect(status).equals("loading");
expect(error).undefined;
},
(state) => {
if (state.status !== "ready") {
expect(state).eq({});
return;
}
expect(state.balance).deep.equal(Amounts.parseOrThrow("USD:15"));
expect(state.amount).deep.equal(Amounts.parseOrThrow("USD:9"));
if (state.payHandler.onClick === undefined) expect.fail();
state.payHandler.onClick();
},
],
TestingContext,
);
2022-12-15 21:12:03 +01:00
expect(hookBehavior).deep.equal({ result: "ok" });
expect(handler.getCallingQueueState()).eq("empty");
2022-04-21 19:23:53 +02:00
});
it("should not stay in ready state after pay with error", async () => {
const { handler, TestingContext } = createWalletApiMock();
2022-10-25 17:23:08 +02:00
const props = {
talerPayUri: "taller://pay",
cancel: nullFunction,
goToWalletManualWithdraw: nullFunction,
onSuccess: nullFunction,
};
handler.addWalletCallResponse(
WalletApiOperation.PreparePayForUri,
undefined,
{
status: PreparePayResultType.PaymentPossible,
amountRaw: "USD:9",
amountEffective: "USD:10",
} as PreparePayResultPaymentPossible,
);
handler.addWalletCallResponse(
WalletApiOperation.GetBalances,
{},
{
balances: [
{
available: "USD:15",
hasPendingTransactions: false,
pendingIncoming: "USD:0",
pendingOutgoing: "USD:0",
requiresUserInput: false,
},
],
},
);
2022-10-25 17:23:08 +02:00
handler.addWalletCallResponse(WalletApiOperation.ConfirmPay, undefined, {
type: ConfirmPayResultType.Pending,
lastError: { code: 1 },
} as ConfirmPayResult);
2022-10-25 17:23:08 +02:00
2022-12-15 21:12:03 +01:00
const hookBehavior = await tests.hookBehaveLikeThis(
() => {
const state = useComponentState(props);
// const { alerts } = useAlertContext();
return { ...state, alerts: {} };
},
{},
2022-12-15 21:12:03 +01:00
[
({ status, error }) => {
expect(status).equals("loading");
expect(error).undefined;
},
(state) => {
if (state.status !== "ready") expect.fail();
expect(state.balance).deep.equal(Amounts.parseOrThrow("USD:15"));
expect(state.amount).deep.equal(Amounts.parseOrThrow("USD:9"));
// expect(r.totalFees).deep.equal(Amounts.parseOrThrow("USD:1"));
if (state.payHandler.onClick === undefined) expect.fail();
state.payHandler.onClick();
},
// (state) => {
// if (state.status !== "ready") expect.fail();
// expect(state.balance).deep.equal(Amounts.parseOrThrow("USD:15"));
// expect(state.amount).deep.equal(Amounts.parseOrThrow("USD:9"));
// // FIXME: check that the error is pushed to the alertContext
// // expect(state.alerts.length).eq(1);
// // const alert = state.alerts[0]
// // if (alert.type !== "error") expect.fail();
// // expect(alert.cause.errorDetail.payResult).deep.equal({
// // type: ConfirmPayResultType.Pending,
// // lastError: { code: 1 },
// // });
// },
2022-12-15 21:12:03 +01:00
],
TestingContext,
);
2022-10-25 17:23:08 +02:00
2022-12-15 21:12:03 +01:00
expect(hookBehavior).deep.equal({ result: "ok" });
expect(handler.getCallingQueueState()).eq("empty");
2022-04-21 19:23:53 +02:00
});
it("should update balance if a coins is withdraw", async () => {
const { handler, TestingContext } = createWalletApiMock();
2022-10-25 17:23:08 +02:00
const props = {
talerPayUri: "taller://pay",
cancel: nullFunction,
goToWalletManualWithdraw: nullFunction,
onSuccess: nullFunction,
};
2022-04-21 19:23:53 +02:00
handler.addWalletCallResponse(
WalletApiOperation.PreparePayForUri,
undefined,
{
status: PreparePayResultType.PaymentPossible,
amountRaw: "USD:9",
amountEffective: "USD:10",
} as PreparePayResultPaymentPossible,
);
handler.addWalletCallResponse(
WalletApiOperation.GetBalances,
{},
{
balances: [
{
available: "USD:10",
hasPendingTransactions: false,
pendingIncoming: "USD:0",
pendingOutgoing: "USD:0",
requiresUserInput: false,
},
],
},
);
handler.addWalletCallResponse(
WalletApiOperation.PreparePayForUri,
undefined,
{
status: PreparePayResultType.PaymentPossible,
amountRaw: "USD:9",
amountEffective: "USD:10",
} as PreparePayResultPaymentPossible,
);
handler.addWalletCallResponse(
WalletApiOperation.GetBalances,
{},
{
balances: [
{
available: "USD:15",
hasPendingTransactions: false,
pendingIncoming: "USD:0",
pendingOutgoing: "USD:0",
requiresUserInput: false,
},
],
},
);
2022-10-25 17:23:08 +02:00
2022-12-15 21:12:03 +01:00
const hookBehavior = await tests.hookBehaveLikeThis(
useComponentState,
props,
[
({ status, error }) => {
expect(status).equals("loading");
expect(error).undefined;
},
(state) => {
if (state.status !== "ready") expect.fail();
expect(state.balance).deep.equal(Amounts.parseOrThrow("USD:10"));
expect(state.amount).deep.equal(Amounts.parseOrThrow("USD:9"));
// expect(r.totalFees).deep.equal(Amounts.parseOrThrow("USD:1"));
expect(state.payHandler.onClick).not.undefined;
handler.notifyEventFromWallet(NotificationType.CoinWithdrawn);
},
(state) => {
if (state.status !== "ready") expect.fail();
expect(state.balance).deep.equal(Amounts.parseOrThrow("USD:15"));
expect(state.amount).deep.equal(Amounts.parseOrThrow("USD:9"));
// expect(r.totalFees).deep.equal(Amounts.parseOrThrow("USD:1"));
expect(state.payHandler.onClick).not.undefined;
},
],
TestingContext,
);
2022-12-15 21:12:03 +01:00
expect(hookBehavior).deep.equal({ result: "ok" });
expect(handler.getCallingQueueState()).eq("empty");
2022-04-21 19:23:53 +02:00
});
2022-06-06 05:09:25 +02:00
});