From 2e56a22f6b192600e8ec662b01ad436cbd1a5e65 Mon Sep 17 00:00:00 2001 From: Florian Dold Date: Tue, 11 Aug 2020 17:55:45 +0530 Subject: [PATCH] refund response --- .../src/operations/refund.ts | 72 ++++++++++++++++++- 1 file changed, 69 insertions(+), 3 deletions(-) diff --git a/packages/taler-wallet-core/src/operations/refund.ts b/packages/taler-wallet-core/src/operations/refund.ts index 2b6ee97ae..d7aaf7272 100644 --- a/packages/taler-wallet-core/src/operations/refund.ts +++ b/packages/taler-wallet-core/src/operations/refund.ts @@ -41,12 +41,13 @@ import { import { NotificationType } from "../types/notifications"; import { parseRefundUri } from "../util/taleruri"; import { createRefreshGroup, getTotalRefreshCost } from "./refresh"; -import { Amounts } from "../util/amounts"; +import { Amounts, AmountJson } from "../util/amounts"; import { MerchantCoinRefundStatus, MerchantCoinRefundSuccessStatus, MerchantCoinRefundFailureStatus, codecForMerchantOrderStatusPaid, + AmountString, } from "../types/talerTypes"; import { guardOperationException } from "./errors"; import { getTimestampNow } from "../util/time"; @@ -302,6 +303,30 @@ async function acceptRefunds( }); } +/** + * Summary of the refund status of a purchase. + */ +export interface RefundSummary { + pendingAtExchange: boolean; + amountEffectivePaid: AmountJson; + amountRefundGranted: AmountJson; + amountRefundGone: AmountJson; +} + +export interface ApplyRefundResponse { + contractTermsHash: string; + + proposalId: string; + + amountEffectivePaid: AmountString; + + amountRefundGranted: AmountString; + + amountRefundGone: AmountString; + + pendingAtExchange: boolean; +} + /** * Accept a refund, return the contract hash for the contract * that was involved in the refund. @@ -309,7 +334,7 @@ async function acceptRefunds( export async function applyRefund( ws: InternalWalletState, talerRefundUri: string, -): Promise<{ contractTermsHash: string; proposalId: string }> { +): Promise { const parseResult = parseRefundUri(talerRefundUri); logger.trace("applying refund", parseResult); @@ -318,7 +343,7 @@ export async function applyRefund( throw Error("invalid refund URI"); } - const purchase = await ws.db.getIndexed(Stores.purchases.orderIdIndex, [ + let purchase = await ws.db.getIndexed(Stores.purchases.orderIdIndex, [ parseResult.merchantBaseUrl, parseResult.orderId, ]); @@ -355,9 +380,50 @@ export async function applyRefund( await processPurchaseQueryRefund(ws, proposalId); } + purchase = await ws.db.get(Stores.purchases, proposalId); + + if (!purchase) { + throw Error("purchase no longer exists"); + } + + const p = purchase; + + let amountRefundGranted = Amounts.getZero( + purchase.contractData.amount.currency, + ); + let amountRefundGone = Amounts.getZero(purchase.contractData.amount.currency); + + let pendingAtExchange = false; + + Object.keys(purchase.refunds).forEach((rk) => { + const refund = p.refunds[rk]; + if (refund.type === RefundState.Pending) { + pendingAtExchange = true; + } + if ( + refund.type === RefundState.Applied || + refund.type === RefundState.Pending + ) { + amountRefundGranted = Amounts.add( + amountRefundGranted, + Amounts.sub( + refund.refundAmount, + refund.refundFee, + refund.totalRefreshCostBound, + ).amount, + ).amount; + } else { + amountRefundGone = Amounts.add(amountRefundGone, refund.refundAmount).amount; + } + }); + return { contractTermsHash: purchase.contractData.contractTermsHash, proposalId: purchase.proposalId, + amountEffectivePaid: Amounts.stringify(purchase.payCostInfo.totalCost), + amountRefundGone: Amounts.stringify(amountRefundGone), + amountRefundGranted: Amounts.stringify(amountRefundGranted), + pendingAtExchange, }; }