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

499 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 {
2022-10-25 17:23:08 +02:00
Amounts, ConfirmPayResult,
2022-06-06 05:09:25 +02:00
ConfirmPayResultType,
2022-10-25 17:23:08 +02:00
NotificationType, PreparePayResultInsufficientBalance,
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";
2022-10-25 17:23:08 +02:00
import { mountHook, nullFunction } from "../../test-utils.js";
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 () => {
2022-10-25 17:23:08 +02:00
const { handler, mock } = createWalletApiMock();
const props = {
talerPayUri: undefined,
cancel: nullFunction,
goToWalletManualWithdraw: nullFunction,
onSuccess: async () => {
null;
},
}
const { pullLastResultOrThrow, waitForStateUpdate, assertNoPendingUpdate } =
2022-06-06 05:09:25 +02:00
mountHook(() =>
2022-10-25 17:23:08 +02:00
useComponentState(props, mock),
2022-06-06 05:09:25 +02:00
);
2022-04-21 19:23:53 +02:00
{
2022-10-25 17:23:08 +02:00
const { status, error } = pullLastResultOrThrow();
2022-06-06 05:09:25 +02:00
expect(status).equals("loading");
2022-07-31 01:55:41 +02:00
expect(error).undefined;
2022-04-21 19:23:53 +02:00
}
2022-10-25 17:23:08 +02:00
expect(await waitForStateUpdate()).true;
2022-04-21 19:23:53 +02:00
{
2022-10-25 17:23:08 +02:00
const { status, error } = pullLastResultOrThrow();
2022-04-21 19:23:53 +02:00
2022-07-31 01:55:41 +02:00
expect(status).equals("loading-uri");
if (error === undefined) expect.fail();
expect(error.hasError).true;
expect(error.operational).false;
2022-04-21 19:23:53 +02:00
}
2022-06-06 05:09:25 +02:00
await assertNoPendingUpdate();
2022-10-25 17:23:08 +02:00
expect(handler.getCallingQueueState()).eq("empty")
2022-04-21 19:23:53 +02:00
});
it("should response with no balance", async () => {
2022-10-25 17:23:08 +02:00
const { handler, mock } = createWalletApiMock();
const props = {
talerPayUri: "taller://pay",
cancel: nullFunction,
goToWalletManualWithdraw: nullFunction,
onSuccess: async () => {
null;
},
}
handler.addWalletCallResponse(WalletApiOperation.PreparePayForUri, undefined, {
status: PreparePayResultType.InsufficientBalance,
amountRaw: "USD:10",
} as PreparePayResultInsufficientBalance)
handler.addWalletCallResponse(WalletApiOperation.GetBalances, {}, { balances: [] })
const { pullLastResultOrThrow, waitForStateUpdate, assertNoPendingUpdate } =
2022-06-06 05:09:25 +02:00
mountHook(() =>
2022-10-25 17:23:08 +02:00
useComponentState(props, mock),
2022-06-06 05:09:25 +02:00
);
2022-04-21 19:23:53 +02:00
{
2022-10-25 17:23:08 +02:00
const { status, error } = pullLastResultOrThrow();
2022-06-06 05:09:25 +02:00
expect(status).equals("loading");
2022-07-31 01:55:41 +02:00
expect(error).undefined;
2022-04-21 19:23:53 +02:00
}
2022-10-25 17:23:08 +02:00
expect(await waitForStateUpdate()).true;
2022-04-21 19:23:53 +02:00
{
2022-10-25 17:23:08 +02:00
const r = pullLastResultOrThrow();
if (r.status !== "no-balance-for-currency") {
expect(r).eq({})
return;
}
2022-04-21 19:23:53 +02:00
expect(r.balance).undefined;
2022-06-06 05:09:25 +02:00
expect(r.amount).deep.equal(Amounts.parseOrThrow("USD:10"));
2022-04-21 19:23:53 +02:00
}
2022-06-06 05:09:25 +02:00
await assertNoPendingUpdate();
2022-10-25 17:23:08 +02:00
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 () => {
2022-10-25 17:23:08 +02:00
const { handler, mock } = createWalletApiMock();
const props = {
talerPayUri: "taller://pay",
cancel: nullFunction,
goToWalletManualWithdraw: nullFunction,
onSuccess: async () => {
null;
},
}
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,
}]
})
const { pullLastResultOrThrow, waitForStateUpdate, assertNoPendingUpdate } =
2022-06-06 05:09:25 +02:00
mountHook(() =>
2022-10-25 17:23:08 +02:00
useComponentState(props, mock),
2022-06-06 05:09:25 +02:00
);
2022-04-21 19:23:53 +02:00
{
2022-10-25 17:23:08 +02:00
const { status, error } = pullLastResultOrThrow();
2022-06-06 05:09:25 +02:00
expect(status).equals("loading");
2022-07-31 01:55:41 +02:00
expect(error).undefined;
2022-04-21 19:23:53 +02:00
}
2022-10-25 17:23:08 +02:00
expect(await waitForStateUpdate()).true;
2022-04-21 19:23:53 +02:00
{
2022-10-25 17:23:08 +02:00
const r = pullLastResultOrThrow();
2022-07-31 01:55:41 +02:00
if (r.status !== "no-enough-balance") expect.fail();
2022-06-06 05:09:25 +02:00
expect(r.balance).deep.equal(Amounts.parseOrThrow("USD:5"));
expect(r.amount).deep.equal(Amounts.parseOrThrow("USD:10"));
2022-04-21 19:23:53 +02:00
}
2022-06-06 05:09:25 +02:00
await assertNoPendingUpdate();
2022-10-25 17:23:08 +02:00
expect(handler.getCallingQueueState()).eq("empty")
2022-04-21 19:23:53 +02:00
});
it("should be able to pay (without fee)", async () => {
2022-10-25 17:23:08 +02:00
const { handler, mock } = createWalletApiMock();
const props = {
talerPayUri: "taller://pay",
cancel: nullFunction,
goToWalletManualWithdraw: nullFunction,
onSuccess: async () => {
null;
},
}
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,
}]
})
const { pullLastResultOrThrow, waitForStateUpdate, assertNoPendingUpdate } =
2022-06-06 05:09:25 +02:00
mountHook(() =>
2022-10-25 17:23:08 +02:00
useComponentState(props, mock),
2022-06-06 05:09:25 +02:00
);
2022-04-21 19:23:53 +02:00
{
2022-10-25 17:23:08 +02:00
const { status, error } = pullLastResultOrThrow();
2022-06-06 05:09:25 +02:00
expect(status).equals("loading");
2022-07-31 01:55:41 +02:00
expect(error).undefined;
2022-04-21 19:23:53 +02:00
}
2022-10-25 17:23:08 +02:00
expect(await waitForStateUpdate()).true;
2022-04-21 19:23:53 +02:00
{
2022-10-25 17:23:08 +02:00
const r = pullLastResultOrThrow();
if (r.status !== "ready") {
expect(r).eq({})
return
}
2022-06-06 05:09:25 +02:00
expect(r.balance).deep.equal(Amounts.parseOrThrow("USD:15"));
expect(r.amount).deep.equal(Amounts.parseOrThrow("USD:10"));
2022-04-21 19:23:53 +02:00
expect(r.payHandler.onClick).not.undefined;
}
2022-06-06 05:09:25 +02:00
await assertNoPendingUpdate();
2022-10-25 17:23:08 +02:00
expect(handler.getCallingQueueState()).eq("empty")
2022-04-21 19:23:53 +02:00
});
it("should be able to pay (with fee)", async () => {
2022-10-25 17:23:08 +02:00
const { handler, mock } = createWalletApiMock();
const props = {
talerPayUri: "taller://pay",
cancel: nullFunction,
goToWalletManualWithdraw: nullFunction,
onSuccess: async () => {
null;
},
}
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,
}]
})
const { pullLastResultOrThrow, waitForStateUpdate, assertNoPendingUpdate } =
2022-06-06 05:09:25 +02:00
mountHook(() =>
2022-09-16 19:29:35 +02:00
useComponentState(
2022-10-25 17:23:08 +02:00
props,
mock
2022-09-16 19:29:35 +02:00
),
2022-06-06 05:09:25 +02:00
);
2022-04-21 19:23:53 +02:00
{
2022-10-25 17:23:08 +02:00
const { status, error } = pullLastResultOrThrow();
2022-06-06 05:09:25 +02:00
expect(status).equals("loading");
2022-07-31 01:55:41 +02:00
expect(error).undefined;
2022-04-21 19:23:53 +02:00
}
2022-10-25 17:23:08 +02:00
expect(await waitForStateUpdate()).true;
2022-04-21 19:23:53 +02:00
{
2022-10-25 17:23:08 +02:00
const r = pullLastResultOrThrow();
2022-06-06 05:09:25 +02:00
if (r.status !== "ready") expect.fail();
expect(r.balance).deep.equal(Amounts.parseOrThrow("USD:15"));
expect(r.amount).deep.equal(Amounts.parseOrThrow("USD:9"));
2022-04-21 19:23:53 +02:00
expect(r.payHandler.onClick).not.undefined;
}
2022-06-06 05:09:25 +02:00
await assertNoPendingUpdate();
2022-10-25 17:23:08 +02:00
expect(handler.getCallingQueueState()).eq("empty")
2022-04-21 19:23:53 +02:00
});
it("should get confirmation done after pay successfully", async () => {
2022-10-25 17:23:08 +02:00
const { handler, mock } = createWalletApiMock();
const props = {
talerPayUri: "taller://pay",
cancel: nullFunction,
goToWalletManualWithdraw: nullFunction,
onSuccess: async () => {
null;
},
}
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,
}]
})
handler.addWalletCallResponse(WalletApiOperation.ConfirmPay, undefined, {
type: ConfirmPayResultType.Done,
contractTerms: {},
} as ConfirmPayResult)
const { pullLastResultOrThrow, waitForStateUpdate, assertNoPendingUpdate } =
2022-06-06 05:09:25 +02:00
mountHook(() =>
2022-09-16 19:29:35 +02:00
useComponentState(
2022-10-25 17:23:08 +02:00
props, mock
2022-09-16 19:29:35 +02:00
),
2022-06-06 05:09:25 +02:00
);
2022-04-21 19:23:53 +02:00
{
2022-10-25 17:23:08 +02:00
const { status, error } = pullLastResultOrThrow();
2022-06-06 05:09:25 +02:00
expect(status).equals("loading");
2022-07-31 01:55:41 +02:00
expect(error).undefined;
2022-04-21 19:23:53 +02:00
}
2022-10-25 17:23:08 +02:00
expect(await waitForStateUpdate()).true;
2022-04-21 19:23:53 +02:00
{
2022-10-25 17:23:08 +02:00
const r = pullLastResultOrThrow();
if (r.status !== "ready") {
expect(r).eq({})
return;
}
2022-06-06 05:09:25 +02:00
expect(r.balance).deep.equal(Amounts.parseOrThrow("USD:15"));
expect(r.amount).deep.equal(Amounts.parseOrThrow("USD:9"));
2022-04-21 19:23:53 +02:00
if (r.payHandler.onClick === undefined) expect.fail();
2022-06-06 05:09:25 +02:00
r.payHandler.onClick();
2022-04-21 19:23:53 +02:00
}
2022-06-06 05:09:25 +02:00
await assertNoPendingUpdate();
2022-10-25 17:23:08 +02:00
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 () => {
2022-10-25 17:23:08 +02:00
const { handler, mock } = createWalletApiMock();
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,
}]
})
handler.addWalletCallResponse(WalletApiOperation.ConfirmPay, undefined, {
type: ConfirmPayResultType.Pending,
lastError: { code: 1 },
} as ConfirmPayResult)
const { pullLastResultOrThrow, waitForStateUpdate, assertNoPendingUpdate } =
2022-06-06 05:09:25 +02:00
mountHook(() =>
2022-09-16 19:29:35 +02:00
useComponentState(
2022-10-25 17:23:08 +02:00
props, mock
2022-09-16 19:29:35 +02:00
),
2022-06-06 05:09:25 +02:00
);
2022-04-21 19:23:53 +02:00
{
2022-10-25 17:23:08 +02:00
const { status, error } = pullLastResultOrThrow();
2022-06-06 05:09:25 +02:00
expect(status).equals("loading");
2022-07-31 01:55:41 +02:00
expect(error).undefined;
2022-04-21 19:23:53 +02:00
}
2022-10-25 17:23:08 +02:00
expect(await waitForStateUpdate()).true
2022-04-21 19:23:53 +02:00
{
2022-10-25 17:23:08 +02:00
const r = pullLastResultOrThrow();
2022-06-06 05:09:25 +02:00
if (r.status !== "ready") expect.fail();
expect(r.balance).deep.equal(Amounts.parseOrThrow("USD:15"));
expect(r.amount).deep.equal(Amounts.parseOrThrow("USD:9"));
// expect(r.totalFees).deep.equal(Amounts.parseOrThrow("USD:1"));
2022-04-21 19:23:53 +02:00
if (r.payHandler.onClick === undefined) expect.fail();
2022-06-06 05:09:25 +02:00
r.payHandler.onClick();
2022-04-21 19:23:53 +02:00
}
2022-10-25 17:23:08 +02:00
expect(await waitForStateUpdate()).true
2022-04-21 19:23:53 +02:00
{
2022-10-25 17:23:08 +02:00
const r = pullLastResultOrThrow();
2022-06-06 05:09:25 +02:00
if (r.status !== "ready") expect.fail();
expect(r.balance).deep.equal(Amounts.parseOrThrow("USD:15"));
expect(r.amount).deep.equal(Amounts.parseOrThrow("USD:9"));
// expect(r.totalFees).deep.equal(Amounts.parseOrThrow("USD:1"));
2022-04-21 19:23:53 +02:00
expect(r.payHandler.onClick).undefined;
if (r.payHandler.error === undefined) expect.fail();
//FIXME: error message here is bad
2022-06-06 05:09:25 +02:00
expect(r.payHandler.error.errorDetail.hint).eq(
"could not confirm payment",
);
2022-04-21 19:23:53 +02:00
expect(r.payHandler.error.errorDetail.payResult).deep.equal({
type: ConfirmPayResultType.Pending,
2022-06-06 05:09:25 +02:00
lastError: { code: 1 },
});
2022-04-21 19:23:53 +02:00
}
2022-06-06 05:09:25 +02:00
await assertNoPendingUpdate();
2022-10-25 17:23:08 +02:00
expect(handler.getCallingQueueState()).eq("empty")
2022-04-21 19:23:53 +02:00
});
it("should update balance if a coins is withdraw", async () => {
2022-10-25 17:23:08 +02:00
const { handler, mock } = createWalletApiMock();
const props = {
talerPayUri: "taller://pay",
cancel: nullFunction,
goToWalletManualWithdraw: nullFunction,
onSuccess: async () => {
null;
},
2022-04-21 19:23:53 +02:00
}
2022-10-25 17:23:08 +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,
}]
})
const { pullLastResultOrThrow, waitForStateUpdate, assertNoPendingUpdate } =
2022-06-06 05:09:25 +02:00
mountHook(() =>
2022-09-16 19:29:35 +02:00
useComponentState(
2022-10-25 17:23:08 +02:00
props, mock
2022-09-16 19:29:35 +02:00
),
2022-06-06 05:09:25 +02:00
);
2022-04-21 19:23:53 +02:00
{
2022-10-25 17:23:08 +02:00
const { status, error } = pullLastResultOrThrow();
2022-06-06 05:09:25 +02:00
expect(status).equals("loading");
2022-07-31 01:55:41 +02:00
expect(error).undefined;
2022-04-21 19:23:53 +02:00
}
2022-10-25 17:23:08 +02:00
expect(await waitForStateUpdate()).true;
2022-04-21 19:23:53 +02:00
{
2022-10-25 17:23:08 +02:00
const r = pullLastResultOrThrow();
if (r.status !== "ready") {
expect(r).eq({})
return
}
2022-06-06 05:09:25 +02:00
expect(r.balance).deep.equal(Amounts.parseOrThrow("USD:10"));
expect(r.amount).deep.equal(Amounts.parseOrThrow("USD:9"));
// expect(r.totalFees).deep.equal(Amounts.parseOrThrow("USD:1"));
2022-04-21 19:23:53 +02:00
expect(r.payHandler.onClick).not.undefined;
2022-10-25 17:23:08 +02:00
handler.notifyEventFromWallet(NotificationType.CoinWithdrawn);
2022-04-21 19:23:53 +02:00
}
2022-10-25 17:23:08 +02:00
expect(await waitForStateUpdate()).true;
2022-04-21 19:23:53 +02:00
{
2022-10-25 17:23:08 +02:00
const r = pullLastResultOrThrow();
if (r.status !== "ready") {
expect(r).eq({})
return
}
2022-06-06 05:09:25 +02:00
expect(r.balance).deep.equal(Amounts.parseOrThrow("USD:15"));
expect(r.amount).deep.equal(Amounts.parseOrThrow("USD:9"));
// expect(r.totalFees).deep.equal(Amounts.parseOrThrow("USD:1"));
2022-04-21 19:23:53 +02:00
expect(r.payHandler.onClick).not.undefined;
}
2022-06-06 05:09:25 +02:00
await assertNoPendingUpdate();
2022-10-25 17:23:08 +02:00
expect(handler.getCallingQueueState()).eq("empty")
2022-04-21 19:23:53 +02:00
});
2022-06-06 05:09:25 +02:00
});