From 11fa3397053c16cfcbf594c1389a75eaad94a40e Mon Sep 17 00:00:00 2001 From: Florian Dold Date: Wed, 12 Aug 2020 16:32:07 +0530 Subject: fix preparePay bug and add integration test for it --- packages/taler-integrationtests/src/harness.ts | 11 ++- .../src/test-payment-idempotency.ts | 103 +++++++++++++++++++++ .../taler-integrationtests/src/test-payment.ts | 8 +- packages/taler-wallet-core/src/operations/pay.ts | 8 +- .../taler-wallet-core/src/types/walletTypes.ts | 9 ++ 5 files changed, 130 insertions(+), 9 deletions(-) create mode 100644 packages/taler-integrationtests/src/test-payment-idempotency.ts diff --git a/packages/taler-integrationtests/src/harness.ts b/packages/taler-integrationtests/src/harness.ts index ecb0758da..e8a0941d2 100644 --- a/packages/taler-integrationtests/src/harness.ts +++ b/packages/taler-integrationtests/src/harness.ts @@ -41,6 +41,9 @@ import { CoreApiResponse, PreparePayResult, PreparePayRequest, + codecForPreparePayResultPaymentPossible, + codecForPreparePayResult, + OperationFailedError, } from "taler-wallet-core"; import { URL } from "url"; import axios from "axios"; @@ -1111,7 +1114,7 @@ export class WalletCli { async apiRequest( request: string, - payload: Record, + payload: unknown, ): Promise { const wdb = this.globalTestState.testDir + "/walletdb.json"; const resp = await sh( @@ -1144,6 +1147,10 @@ export class WalletCli { } async preparePay(req: PreparePayRequest): Promise { - throw Error("not implemented"); + const resp = await this.apiRequest("preparePay", req); + if (resp.type === "response") { + return codecForPreparePayResult().decode(resp.result); + } + throw new OperationFailedError(resp.error); } } diff --git a/packages/taler-integrationtests/src/test-payment-idempotency.ts b/packages/taler-integrationtests/src/test-payment-idempotency.ts new file mode 100644 index 000000000..4d6727715 --- /dev/null +++ b/packages/taler-integrationtests/src/test-payment-idempotency.ts @@ -0,0 +1,103 @@ +/* + This file is part of GNU Taler + (C) 2020 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 + */ + +/** + * Imports. + */ +import { runTest, GlobalTestState } from "./harness"; +import { createSimpleTestkudosEnvironment, withdrawViaBank } from "./helpers"; +import { PreparePayResultType } from "taler-wallet-core"; + +/** + * Test the wallet-core payment API, especially that repeated operations + * return the expected result. + */ +runTest(async (t: GlobalTestState) => { + // Set up test environment + + const { + wallet, + bank, + exchange, + merchant, + } = await createSimpleTestkudosEnvironment(t); + + // Withdraw digital cash into the wallet. + + await withdrawViaBank(t, { wallet, bank, exchange, amount: "TESTKUDOS:20" }); + + // Set up order. + + const orderResp = await merchant.createOrder("default", { + order: { + summary: "Buy me!", + amount: "TESTKUDOS:5", + fulfillment_url: "taler://fulfillment-success/thx", + }, + }); + + let orderStatus = await merchant.queryPrivateOrderStatus( + "default", + orderResp.order_id, + ); + + t.assertTrue(orderStatus.order_status === "unpaid"); + + const talerPayUri = orderStatus.taler_pay_uri; + + // Make wallet pay for the order + + const preparePayResult = await wallet.preparePay({ + talerPayUri: orderStatus.taler_pay_uri, + }); + + const preparePayResultRep = await wallet.preparePay({ + talerPayUri: orderStatus.taler_pay_uri, + }); + + t.assertTrue( + preparePayResult.status === PreparePayResultType.PaymentPossible, + ); + t.assertTrue( + preparePayResultRep.status === PreparePayResultType.PaymentPossible, + ); + + const proposalId = preparePayResult.proposalId; + + const r2 = await wallet.apiRequest("confirmPay", { + // FIXME: should be validated, don't cast! + proposalId: proposalId, + }); + t.assertTrue(r2.type === "response"); + + // Check if payment was successful. + + orderStatus = await merchant.queryPrivateOrderStatus( + "default", + orderResp.order_id, + ); + + t.assertTrue(orderStatus.order_status === "paid"); + + const preparePayResultAfter = await wallet.preparePay({ + talerPayUri, + }); + + t.assertTrue(preparePayResultAfter.status === PreparePayResultType.AlreadyConfirmed); + t.assertTrue(preparePayResultAfter.paid === true); + + await t.shutdown(); +}); diff --git a/packages/taler-integrationtests/src/test-payment.ts b/packages/taler-integrationtests/src/test-payment.ts index 3fd879580..77645909c 100644 --- a/packages/taler-integrationtests/src/test-payment.ts +++ b/packages/taler-integrationtests/src/test-payment.ts @@ -19,6 +19,7 @@ */ import { runTest, GlobalTestState } from "./harness"; import { createSimpleTestkudosEnvironment, withdrawViaBank } from "./helpers"; +import { PreparePayResultType } from "taler-wallet-core"; /** * Run test for basic, bank-integrated withdrawal. @@ -56,14 +57,15 @@ runTest(async (t: GlobalTestState) => { // Make wallet pay for the order - const r1 = await wallet.apiRequest("preparePay", { + const preparePayResult = await wallet.preparePay({ talerPayUri: orderStatus.taler_pay_uri, }); - t.assertTrue(r1.type === "response"); + + t.assertTrue(preparePayResult.status === PreparePayResultType.PaymentPossible); const r2 = await wallet.apiRequest("confirmPay", { // FIXME: should be validated, don't cast! - proposalId: (r1.result as any).proposalId, + proposalId: preparePayResult.proposalId, }); t.assertTrue(r2.type === "response"); diff --git a/packages/taler-wallet-core/src/operations/pay.ts b/packages/taler-wallet-core/src/operations/pay.ts index db5a56d18..0576f7eab 100644 --- a/packages/taler-wallet-core/src/operations/pay.ts +++ b/packages/taler-wallet-core/src/operations/pay.ts @@ -980,17 +980,17 @@ export async function preparePayForUri( amountRaw: Amounts.stringify(purchase.contractData.amount), amountEffective: Amounts.stringify(purchase.payCostInfo.totalCost), }; - } else if (purchase.paymentSubmitPending) { + } else { + const paid = !purchase.paymentSubmitPending; return { status: PreparePayResultType.AlreadyConfirmed, contractTerms: JSON.parse(purchase.contractTermsRaw), - paid: false, + paid, amountRaw: Amounts.stringify(purchase.contractData.amount), amountEffective: Amounts.stringify(purchase.payCostInfo.totalCost), + ...(paid ? { nextUrl: purchase.contractData.orderId } : {}), }; } - // FIXME: we don't handle aborted payments correctly here. - throw Error("BUG: invariant violation (purchase status)"); } /** diff --git a/packages/taler-wallet-core/src/types/walletTypes.ts b/packages/taler-wallet-core/src/types/walletTypes.ts index 7a648dd56..ec57e7d2a 100644 --- a/packages/taler-wallet-core/src/types/walletTypes.ts +++ b/packages/taler-wallet-core/src/types/walletTypes.ts @@ -48,6 +48,7 @@ import { codecForBoolean, codecForConstString, codecForAny, + buildCodecForUnion, } from "../util/codec"; import { AmountString, codecForContractTerms } from "./talerTypes"; import { TransactionError } from "./transactions"; @@ -399,6 +400,14 @@ export const codecForPreparePayResultAlreadyConfirmed = (): Codec< .property("contractTerms", codecForAny()) .build("PreparePayResultAlreadyConfirmed"); +export const codecForPreparePayResult = (): Codec => + buildCodecForUnion() + .discriminateOn("status") + .alternative(PreparePayResultType.AlreadyConfirmed, codecForPreparePayResultAlreadyConfirmed()) + .alternative(PreparePayResultType.InsufficientBalance, codecForPreparePayResultInsufficientBalance()) + .alternative(PreparePayResultType.PaymentPossible, codecForPreparePayResultPaymentPossible()) + .build("PreparePayResult"); + export type PreparePayResult = | PreparePayResultInsufficientBalance | PreparePayResultAlreadyConfirmed -- cgit v1.2.3