diff --git a/packages/taler-wallet-cli/tsconfig.json b/packages/taler-wallet-cli/tsconfig.json index 5ded91d7a..945161176 100644 --- a/packages/taler-wallet-cli/tsconfig.json +++ b/packages/taler-wallet-cli/tsconfig.json @@ -2,7 +2,7 @@ "compileOnSave": true, "compilerOptions": { "composite": true, - "target": "ES6", + "target": "ES2018", "module": "ESNext", "moduleResolution": "node", "sourceMap": true, diff --git a/packages/taler-wallet-core/src/operations/backup/index.ts b/packages/taler-wallet-core/src/operations/backup/index.ts index 3799741ee..b3b98fe1b 100644 --- a/packages/taler-wallet-core/src/operations/backup/index.ts +++ b/packages/taler-wallet-core/src/operations/backup/index.ts @@ -242,11 +242,7 @@ function deriveBlobSecret(bc: WalletBackupConfState): Uint8Array { } interface BackupForProviderArgs { - backupConfig: WalletBackupConfState; - provider: BackupProviderRecord; - currentBackupHash: ArrayBuffer; - encBackup: ArrayBuffer; - backupJson: WalletBackupContentV1; + backupProviderBaseUrl: string; /** * Should we attempt one more upload after trying @@ -267,13 +263,22 @@ async function runBackupCycleForProvider( ws: InternalWalletState, args: BackupForProviderArgs, ): Promise { - const { - backupConfig, - provider, - currentBackupHash, - encBackup, - backupJson, - } = args; + const provider = await ws.db + .mktx((x) => ({ backupProviders: x.backupProviders })) + .runReadOnly(async (tx) => { + return tx.backupProviders.get(args.backupProviderBaseUrl); + }); + + if (!provider) { + logger.warn("provider disappeared"); + return; + } + + const backupJson = await exportBackup(ws); + const backupConfig = await provideBackupState(ws); + const encBackup = await encryptBackup(backupConfig, backupJson); + const currentBackupHash = hash(encBackup); + const accountKeyPair = deriveAccountKeyPair(backupConfig, provider.baseUrl); const newHash = encodeCrock(currentBackupHash); @@ -301,11 +306,11 @@ async function runBackupCycleForProvider( headers: { "content-type": "application/octet-stream", "sync-signature": syncSig, - "if-none-match": encodeCrock(currentBackupHash), + "if-none-match": newHash, ...(provider.lastBackupHash ? { - "if-match": provider.lastBackupHash, - } + "if-match": provider.lastBackupHash, + } : {}), }, }); @@ -366,7 +371,11 @@ async function runBackupCycleForProvider( provRec.currentPaymentProposalId = proposalId; // FIXME: allocate error code for this! await tx.backupProviders.put(provRec); - await incrementBackupRetryInTx(tx, args.provider.baseUrl, undefined); + await incrementBackupRetryInTx( + tx, + args.backupProviderBaseUrl, + undefined, + ); }); if (doPay) { @@ -418,6 +427,7 @@ async function runBackupCycleForProvider( .runReadWrite(async (tx) => { const prov = await tx.backupProvider.get(provider.baseUrl); if (!prov) { + logger.warn("backup provider not found anymore"); return; } prov.lastBackupHash = encodeCrock(hash(backupEnc)); @@ -446,7 +456,7 @@ async function runBackupCycleForProvider( await ws.db .mktx((x) => ({ backupProviders: x.backupProviders })) .runReadWrite(async (tx) => { - incrementBackupRetryInTx(tx, args.provider.baseUrl, err); + incrementBackupRetryInTx(tx, args.backupProviderBaseUrl, err); }); } @@ -504,17 +514,8 @@ export async function processBackupForProvider( incrementBackupRetry(ws, backupProviderBaseUrl, err); const run = async () => { - const backupJson = await exportBackup(ws); - const backupConfig = await provideBackupState(ws); - const encBackup = await encryptBackup(backupConfig, backupJson); - const currentBackupHash = hash(encBackup); - await runBackupCycleForProvider(ws, { - provider, - backupJson, - backupConfig, - encBackup, - currentBackupHash, + backupProviderBaseUrl: provider.baseUrl, retryAfterPayment: true, }); }; @@ -531,16 +532,20 @@ export const codecForRemoveBackupProvider = (): Codec { - await ws.db.mktx(({ backupProviders }) => ({ backupProviders })) +export async function removeBackupProvider( + ws: InternalWalletState, + req: RemoveBackupProviderRequest, +): Promise { + await ws.db + .mktx(({ backupProviders }) => ({ backupProviders })) .runReadWrite(async (tx) => { - await tx.backupProviders.delete(req.provider) - }) + await tx.backupProviders.delete(req.provider); + }); } export interface RunBackupCycleRequest { /** - * List of providers to backup or empty for all known providers. + * List of providers to backup or empty for all known providers. */ providers?: Array; } @@ -557,28 +562,25 @@ export const codecForRunBackupCycle = (): Codec => * 2. Download, verify and import backups from connected sync accounts. * 3. Upload the updated backup blob. */ -export async function runBackupCycle(ws: InternalWalletState, req: RunBackupCycleRequest): Promise { +export async function runBackupCycle( + ws: InternalWalletState, + req: RunBackupCycleRequest, +): Promise { const providers = await ws.db .mktx((x) => ({ backupProviders: x.backupProviders })) .runReadOnly(async (tx) => { if (req.providers) { - const rs = await Promise.all(req.providers.map(id => tx.backupProviders.get(id))) - return rs.filter(notEmpty) + const rs = await Promise.all( + req.providers.map((id) => tx.backupProviders.get(id)), + ); + return rs.filter(notEmpty); } - return await tx.backupProviders.iter(req.providers).toArray(); + return await tx.backupProviders.iter().toArray(); }); - const backupJson = await exportBackup(ws); - const backupConfig = await provideBackupState(ws); - const encBackup = await encryptBackup(backupConfig, backupJson); - const currentBackupHash = hash(encBackup); for (const provider of providers) { await runBackupCycleForProvider(ws, { - provider, - backupJson, - backupConfig, - encBackup, - currentBackupHash, + backupProviderBaseUrl: provider.baseUrl, retryAfterPayment: true, }); } @@ -645,7 +647,7 @@ export async function addBackupProvider( return; } }); - const termsUrl = new URL("terms", canonUrl); + const termsUrl = new URL("config", canonUrl); const resp = await ws.http.get(termsUrl.href); const terms = await readSuccessResponseJsonOrThrow( resp, @@ -680,7 +682,7 @@ export async function addBackupProvider( }); } -export async function restoreFromRecoverySecret(): Promise { } +export async function restoreFromRecoverySecret(): Promise {} /** * Information about one provider. @@ -907,7 +909,7 @@ async function backupRecoveryTheirs( if (!existingProv) { await tx.backupProviders.put({ baseUrl: prov.url, - name: 'not-defined', + name: "not-defined", paymentProposalIds: [], state: { tag: BackupProviderStateTag.Ready,