/* This file is part of TALER (C) 2016 GNUnet e.V. 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. 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 TALER; see the file COPYING. If not, see */ /** * Interface to the wallet through WebExtension messaging. */ /** * Imports. */ import { AcceptExchangeTosRequest, AcceptManualWithdrawalResult, AcceptTipRequest, AcceptWithdrawalResponse, AddExchangeRequest, AmountString, ApplyRefundResponse, BalancesResponse, ConfirmPayResult, CoreApiResponse, CreateDepositGroupRequest, CreateDepositGroupResponse, DeleteTransactionRequest, ExchangesListRespose, GetExchangeTosResult, GetExchangeWithdrawalInfo, GetFeeForDepositRequest, GetWithdrawalDetailsForUriRequest, KnownBankAccounts, NotificationType, PreparePayResult, PrepareTipRequest, PrepareTipResult, RetryTransactionRequest, SetWalletDeviceIdRequest, TransactionsResponse, WalletDiagnostics, WithdrawUriInfoResponse } from "@gnu-taler/taler-util"; import { AddBackupProviderRequest, BackupInfo, OperationFailedError, PendingOperationsResponse, RemoveBackupProviderRequest } from "@gnu-taler/taler-wallet-core"; import { DepositFee } from "@gnu-taler/taler-wallet-core/src/operations/deposits"; import { ExchangeWithdrawDetails } from "@gnu-taler/taler-wallet-core/src/operations/withdraw"; import { MessageFromBackend } from "./wxBackend"; export interface ExtendedPermissionsResponse { newValue: boolean; } /** * Response with information about available version upgrades. */ export interface UpgradeResponse { /** * Is a reset required because of a new DB version * that can't be automatically upgraded? */ dbResetRequired: boolean; /** * Current database version. */ currentDbVersion: string; /** * Old db version (if applicable). */ oldDbVersion: string; } async function callBackend(operation: string, payload: any): Promise { return new Promise((resolve, reject) => { // eslint-disable-next-line no-undef chrome.runtime.sendMessage({ operation, payload, id: "(none)" }, (resp) => { // eslint-disable-next-line no-undef if (chrome.runtime.lastError) { console.log("Error calling backend"); reject( new Error( `Error contacting backend: ${chrome.runtime.lastError.message}`, ), ); } console.log("got response", resp); const r = resp as CoreApiResponse; if (r.type === "error") { reject(new OperationFailedError(r.error)); return; } resolve(r.result); }); }); } /** * Start refreshing a coin. */ export function refresh(coinPub: string): Promise { return callBackend("refresh-coin", { coinPub }); } /** * Pay for a proposal. */ export function confirmPay( proposalId: string, sessionId: string | undefined, ): Promise { return callBackend("confirmPay", { proposalId, sessionId }); } /** * Check upgrade information */ export function checkUpgrade(): Promise { return callBackend("check-upgrade", {}); } /** * Reset database */ export function resetDb(): Promise { return callBackend("reset-db", {}); } export function getFeeForDeposit(depositPaytoUri: string, amount: AmountString): Promise { return callBackend("getFeeForDeposit", { depositPaytoUri, amount } as GetFeeForDepositRequest); } export function createDepositGroup(depositPaytoUri: string, amount: AmountString): Promise { return callBackend("createDepositGroup", { depositPaytoUri, amount } as CreateDepositGroupRequest); } /** * Get balances for all currencies/exchanges. */ export function getBalance(): Promise { return callBackend("getBalances", {}); } /** * Retrieve the full event history for this wallet. */ export function getTransactions(): Promise { return callBackend("getTransactions", {}); } interface CurrencyInfo { name: string; baseUrl: string; pub: string; } interface ListOfKnownCurrencies { auditors: CurrencyInfo[]; exchanges: CurrencyInfo[]; } /** * Get a list of currencies from known auditors and exchanges */ export function listKnownCurrencies(): Promise { return callBackend("listCurrencies", {}).then((result) => { console.log("result list", result); const auditors = result.trustedAuditors.map( (a: Record) => ({ name: a.currency, baseUrl: a.auditorBaseUrl, pub: a.auditorPub, }), ); const exchanges = result.trustedExchanges.map( (a: Record) => ({ name: a.currency, baseUrl: a.exchangeBaseUrl, pub: a.exchangeMasterPub, }), ); return { auditors, exchanges }; }); } export function listExchanges(): Promise { return callBackend("listExchanges", {}); } export function listKnownBankAccounts(currency?: string): Promise { return callBackend("listKnownBankAccounts", { currency }); } /** * Get information about the current state of wallet backups. */ export function getBackupInfo(): Promise { return callBackend("getBackupInfo", {}); } /** * Add a backup provider and activate it */ export function addBackupProvider( backupProviderBaseUrl: string, name: string, ): Promise { return callBackend("addBackupProvider", { backupProviderBaseUrl, activate: true, name, } as AddBackupProviderRequest); } export function setWalletDeviceId(walletDeviceId: string): Promise { return callBackend("setWalletDeviceId", { walletDeviceId, } as SetWalletDeviceIdRequest); } export function syncAllProviders(): Promise { return callBackend("runBackupCycle", {}); } export function syncOneProvider(url: string): Promise { return callBackend("runBackupCycle", { providers: [url] }); } export function removeProvider(url: string): Promise { return callBackend("removeBackupProvider", { provider: url, } as RemoveBackupProviderRequest); } export function extendedProvider(url: string): Promise { return callBackend("extendBackupProvider", { provider: url }); } /** * Retry a transaction * @param transactionId * @returns */ export function retryTransaction(transactionId: string): Promise { return callBackend("retryTransaction", { transactionId, } as RetryTransactionRequest); } /** * Permanently delete a transaction from the transaction list */ export function deleteTransaction(transactionId: string): Promise { return callBackend("deleteTransaction", { transactionId, } as DeleteTransactionRequest); } /** * Download a refund and accept it. */ export function applyRefund( talerRefundUri: string, ): Promise { return callBackend("applyRefund", { talerRefundUri }); } /** * Get details about a pay operation. */ export function preparePay(talerPayUri: string): Promise { return callBackend("preparePay", { talerPayUri }); } /** * Get details about a withdraw operation. */ export function acceptWithdrawal( talerWithdrawUri: string, selectedExchange: string, ): Promise { return callBackend("acceptBankIntegratedWithdrawal", { talerWithdrawUri, exchangeBaseUrl: selectedExchange, }); } /** * Create a reserve into the exchange that expect the amount indicated * @param exchangeBaseUrl * @param amount * @returns */ export function acceptManualWithdrawal( exchangeBaseUrl: string, amount: string, ): Promise { return callBackend("acceptManualWithdrawal", { amount, exchangeBaseUrl, }); } export function setExchangeTosAccepted( exchangeBaseUrl: string, etag: string | undefined, ): Promise { return callBackend("setExchangeTosAccepted", { exchangeBaseUrl, etag, } as AcceptExchangeTosRequest); } /** * Get diagnostics information */ export function getDiagnostics(): Promise { return callBackend("wxGetDiagnostics", {}); } /** * Get diagnostics information */ export function setExtendedPermissions( value: boolean, ): Promise { return callBackend("wxSetExtendedPermissions", { value }); } /** * Get diagnostics information */ export function getExtendedPermissions(): Promise { return callBackend("wxGetExtendedPermissions", {}); } /** * Get diagnostics information */ export function getWithdrawalDetailsForUri( req: GetWithdrawalDetailsForUriRequest, ): Promise { return callBackend("getWithdrawalDetailsForUri", req); } /** * Get diagnostics information */ export function getExchangeWithdrawalInfo( req: GetExchangeWithdrawalInfo, ): Promise { return callBackend("getExchangeWithdrawalInfo", req); } export function getExchangeTos( exchangeBaseUrl: string, acceptedFormat: string[], ): Promise { return callBackend("getExchangeTos", { exchangeBaseUrl, acceptedFormat, }); } export function getPendingOperations(): Promise { return callBackend("getPendingOperations", {}); } export function addExchange(req: AddExchangeRequest): Promise { return callBackend("addExchange", req); } export function prepareTip(req: PrepareTipRequest): Promise { return callBackend("prepareTip", req); } export function acceptTip(req: AcceptTipRequest): Promise { return callBackend("acceptTip", req); } export function exportDB(): Promise { return callBackend("exportDb", {}); } export function onUpdateNotification(messageTypes: Array, doCallback: () => void): () => void { // eslint-disable-next-line no-undef const port = chrome.runtime.connect({ name: "notifications" }); const listener = (message: MessageFromBackend): void => { const shouldNotify = messageTypes.includes(message.type) console.log("Notification arrived, should notify?", shouldNotify, message.type, messageTypes) if (shouldNotify) { doCallback(); } }; port.onMessage.addListener(listener); return () => { port.onMessage.removeListener(listener); }; }