From 46c805b4d7bbace654d6f9962aac3fe7fb638bc7 Mon Sep 17 00:00:00 2001 From: wrd Date: Fri, 20 Dec 2024 16:04:41 +0100 Subject: [PATCH 1/3] Get server information and round-robin calls to servers when exporting Close #8129 --- packages/tuta-wasm-loader/lib/WasmHandler.ts | 2 + .../worker/facades/lazy/MailExportFacade.ts | 27 ++++- .../api/worker/rest/EntityRestClient.ts | 4 + .../native/main/MailExportController.ts | 23 +++- .../workerUtils/worker/WorkerLocator.ts | 2 +- .../worker/facades/MailExportFacadeTest.ts | 21 ++-- .../api/worker/rest/EntityRestClientTest.ts | 12 ++ .../native/main/MailExportControllerTest.ts | 107 ++++++++++++------ 8 files changed, 146 insertions(+), 52 deletions(-) diff --git a/packages/tuta-wasm-loader/lib/WasmHandler.ts b/packages/tuta-wasm-loader/lib/WasmHandler.ts index 50cd2b8be0c3..71b73b87dac5 100644 --- a/packages/tuta-wasm-loader/lib/WasmHandler.ts +++ b/packages/tuta-wasm-loader/lib/WasmHandler.ts @@ -19,6 +19,7 @@ async function generateWasm(command: string, options: WasmGeneratorOptions) { ...process.env, ...options.env, }, + maxBuffer: Infinity, cwd: options.workingDir ?? process.cwd(), }) promise.child.stdout?.on("data", (data) => { @@ -37,6 +38,7 @@ async function generateWasmFallback(wasmFilePath: string, options: WasmGenerator ...process.env, ...options.env, }, + maxBuffer: Infinity, }) return result.stdout } diff --git a/src/common/api/worker/facades/lazy/MailExportFacade.ts b/src/common/api/worker/facades/lazy/MailExportFacade.ts index f3740b6dd89d..f6c457bab174 100644 --- a/src/common/api/worker/facades/lazy/MailExportFacade.ts +++ b/src/common/api/worker/facades/lazy/MailExportFacade.ts @@ -10,6 +10,9 @@ import { MailExportTokenFacade } from "./MailExportTokenFacade.js" import { assertNotNull, isNotNull } from "@tutao/tutanota-utils" import { NotFoundError } from "../../../common/error/RestError" import { elementIdPart } from "../../../common/utils/EntityUtils" +import { BlobAccessTokenFacade } from "../BlobAccessTokenFacade" +import { BlobServerUrl } from "../../../entities/storage/TypeRefs" +import { Group } from "../../../entities/sys/TypeRefs" assertWorkerOrNode() @@ -29,11 +32,20 @@ export class MailExportFacade { private readonly bulkMailLoader: BulkMailLoader, private readonly blobFacade: BlobFacade, private readonly cryptoFacade: CryptoFacade, + private readonly blobAccessTokenFacade: BlobAccessTokenFacade, ) {} - async loadFixedNumberOfMailsWithCache(mailListId: Id, startId: Id): Promise { + /** + * Returns a list of servers that can be used to request data from. + */ + async getExportServers(group: Group): Promise { + const blobServerAccessInfo = await this.blobAccessTokenFacade.requestWriteToken(ArchiveDataType.Attachments, group._id) + return blobServerAccessInfo.servers + } + + async loadFixedNumberOfMailsWithCache(mailListId: Id, startId: Id, baseUrl: string): Promise { return this.mailExportTokenFacade.loadWithToken((token) => - this.bulkMailLoader.loadFixedNumberOfMailsWithCache(mailListId, startId, this.options(token)), + this.bulkMailLoader.loadFixedNumberOfMailsWithCache(mailListId, startId, { baseUrl, ...this.options(token) }), ) } @@ -41,16 +53,19 @@ export class MailExportFacade { return this.mailExportTokenFacade.loadWithToken((token) => this.bulkMailLoader.loadMailDetails(mails, this.options(token))) } - async loadAttachments(mails: readonly Mail[]): Promise { - return this.mailExportTokenFacade.loadWithToken((token) => this.bulkMailLoader.loadAttachments(mails, this.options(token))) + async loadAttachments(mails: readonly Mail[], baseUrl: string): Promise { + return this.mailExportTokenFacade.loadWithToken((token) => this.bulkMailLoader.loadAttachments(mails, { baseUrl, ...this.options(token) })) } - async loadAttachmentData(mail: Mail, attachments: readonly TutanotaFile[]): Promise { + async loadAttachmentData(mail: Mail, attachments: readonly TutanotaFile[], baseUrl: string): Promise { const attachmentsWithKeys = await this.cryptoFacade.enforceSessionKeyUpdateIfNeeded(mail, attachments) const downloads = await this.mailExportTokenFacade.loadWithToken((token) => { const referencingInstances = attachmentsWithKeys.map(createReferencingInstance) - return this.blobFacade.downloadAndDecryptBlobsOfMultipleInstances(ArchiveDataType.Attachments, referencingInstances, this.options(token)) + return this.blobFacade.downloadAndDecryptBlobsOfMultipleInstances(ArchiveDataType.Attachments, referencingInstances, { + baseUrl, + ...this.options(token), + }) }) const attachmentData = Array.from(downloads.entries()).map(([fileId, bytes]) => { diff --git a/src/common/api/worker/rest/EntityRestClient.ts b/src/common/api/worker/rest/EntityRestClient.ts index 95ced5ff6ac4..25dfa007818e 100644 --- a/src/common/api/worker/rest/EntityRestClient.ts +++ b/src/common/api/worker/rest/EntityRestClient.ts @@ -94,6 +94,7 @@ export interface EntityRestClientLoadOptions { ownerKeyProvider?: OwnerKeyProvider /** Defaults to {@link CacheMode.ReadAndWrite }*/ cacheMode?: CacheMode + baseUrl?: string } export interface OwnerEncSessionKeyProvider { @@ -204,6 +205,7 @@ export class EntityRestClient implements EntityRestInterface { queryParams, headers, responseType: MediaType.Json, + baseUrl: opts.baseUrl, }) const entity = JSON.parse(json) const migratedEntity = await this._crypto.applyMigrations(typeRef, entity) @@ -258,6 +260,7 @@ export class EntityRestClient implements EntityRestInterface { queryParams, headers, responseType: MediaType.Json, + baseUrl: opts.baseUrl, }) return this._handleLoadMultipleResult(typeRef, JSON.parse(json)) } @@ -285,6 +288,7 @@ export class EntityRestClient implements EntityRestInterface { queryParams, headers, responseType: MediaType.Json, + baseUrl: opts.baseUrl, }) } return this._handleLoadMultipleResult(typeRef, JSON.parse(json), ownerEncSessionKeyProvider) diff --git a/src/mail-app/native/main/MailExportController.ts b/src/mail-app/native/main/MailExportController.ts index 57bbc3ed2d08..9c117749afb9 100644 --- a/src/mail-app/native/main/MailExportController.ts +++ b/src/mail-app/native/main/MailExportController.ts @@ -15,6 +15,7 @@ import type { TranslationText } from "../../../common/misc/LanguageViewModel" import { SuspensionError } from "../../../common/api/common/error/SuspensionError" import { Scheduler } from "../../../common/api/common/utils/Scheduler" import { ExportError, ExportErrorReason } from "../../../common/api/common/error/ExportError" +import { BlobServerUrl } from "../../../common/api/entities/storage/TypeRefs" import { assertMainOrNode } from "../../../common/api/common/Env" assertMainOrNode() @@ -37,6 +38,8 @@ const TAG = "MailboxExport" export class MailExportController { private _state: Stream = stream({ type: "idle" }) private _lastExport: Date | null = null + private servers?: BlobServerUrl[] + private serverCount: number = 0 get lastExport(): Date | null { return this._lastExport @@ -145,6 +148,7 @@ export class MailExportController { private async runExport(mailboxDetail: MailboxDetail, mailBags: MailBag[], mailId: Id) { const startTime = assertNotNull(this._lastExport) + this.servers = await this.mailExportFacade.getExportServers(mailboxDetail.mailGroup) for (const mailBag of mailBags) { await this.exportMailBag(mailBag, mailId) if (this._state().type !== "exporting" || this._lastExport !== startTime) { @@ -160,18 +164,16 @@ export class MailExportController { } private async exportMailBag(mailBag: MailBag, startId: Id): Promise { - console.log(TAG, `Exporting mail bag: ${mailBag._id} ${startId}`) let currentStartId = startId while (true) { try { - const downloadedMails = await this.mailExportFacade.loadFixedNumberOfMailsWithCache(mailBag.mails, currentStartId) + const downloadedMails = await this.mailExportFacade.loadFixedNumberOfMailsWithCache(mailBag.mails, currentStartId, this.getServerUrl()) if (downloadedMails.length === 0) { break } const downloadedMailDetails = await this.mailExportFacade.loadMailDetails(downloadedMails) - const attachmentInfo = await this.mailExportFacade.loadAttachments(downloadedMails) - + const attachmentInfo = await this.mailExportFacade.loadAttachments(downloadedMails, this.getServerUrl()) for (const { mail, mailDetails } of downloadedMailDetails) { if (this._state().type !== "exporting") { return @@ -179,7 +181,7 @@ export class MailExportController { const mailAttachmentInfo = mail.attachments .map((attachmentId) => attachmentInfo.find((attachment) => isSameId(attachment._id, attachmentId))) .filter(isNotNull) - const attachments = await this.mailExportFacade.loadAttachmentData(mail, mailAttachmentInfo) + const attachments = await this.mailExportFacade.loadAttachmentData(mail, mailAttachmentInfo, this.getServerUrl()) const { makeMailBundle } = await import("../../mail/export/Bundler.js") const mailBundle = makeMailBundle(this.sanitizer, mail, mailDetails, attachments) @@ -223,4 +225,15 @@ export class MailExportController { } } } + + private getServerUrl(): string { + if (this.servers) { + this.serverCount += 1 + if (this.serverCount >= this.servers.length) { + this.serverCount = 0 + } + return this.servers[this.serverCount].url + } + throw new Error("No servers") + } } diff --git a/src/mail-app/workerUtils/worker/WorkerLocator.ts b/src/mail-app/workerUtils/worker/WorkerLocator.ts index 5337671fdf96..a9602c4a6e13 100644 --- a/src/mail-app/workerUtils/worker/WorkerLocator.ts +++ b/src/mail-app/workerUtils/worker/WorkerLocator.ts @@ -531,7 +531,7 @@ export async function initLocator(worker: WorkerImpl, browserData: BrowserData) const { MailExportFacade } = await import("../../../common/api/worker/facades/lazy/MailExportFacade.js") const { MailExportTokenFacade } = await import("../../../common/api/worker/facades/lazy/MailExportTokenFacade.js") const mailExportTokenFacade = new MailExportTokenFacade(locator.serviceExecutor) - return new MailExportFacade(mailExportTokenFacade, await locator.bulkMailLoader(), await locator.blob(), locator.crypto) + return new MailExportFacade(mailExportTokenFacade, await locator.bulkMailLoader(), await locator.blob(), locator.crypto, locator.blobAccessToken) }) } diff --git a/test/tests/api/worker/facades/MailExportFacadeTest.ts b/test/tests/api/worker/facades/MailExportFacadeTest.ts index 91cc6819549b..0110a72fffb1 100644 --- a/test/tests/api/worker/facades/MailExportFacadeTest.ts +++ b/test/tests/api/worker/facades/MailExportFacadeTest.ts @@ -4,11 +4,12 @@ import { MailExportTokenFacade } from "../../../../../src/common/api/worker/faca import { BulkMailLoader } from "../../../../../src/mail-app/workerUtils/index/BulkMailLoader.js" import { BlobFacade } from "../../../../../src/common/api/worker/facades/lazy/BlobFacade.js" import { CryptoFacade } from "../../../../../src/common/api/worker/crypto/CryptoFacade.js" -import { object, when } from "testdouble" +import { instance, object, when } from "testdouble" import { createTestEntity } from "../../../TestUtils.js" import { FileTypeRef, MailDetailsTypeRef, MailTypeRef } from "../../../../../src/common/api/entities/tutanota/TypeRefs.js" import { ArchiveDataType } from "../../../../../src/common/api/common/TutanotaConstants" import { createReferencingInstance } from "../../../../../src/common/api/common/utils/BlobUtils" +import { BlobAccessTokenFacade } from "../../../../../src/common/api/worker/facades/BlobAccessTokenFacade" o.spec("MailExportFacade", () => { const token = "my token" @@ -23,6 +24,7 @@ o.spec("MailExportFacade", () => { let bulkMailLoader!: BulkMailLoader let blobFacade!: BlobFacade let cryptoFacade!: CryptoFacade + let blobAccessTokenFacade!: BlobAccessTokenFacade o.beforeEach(() => { tokenFacade = { @@ -31,13 +33,17 @@ o.spec("MailExportFacade", () => { bulkMailLoader = object() blobFacade = object() cryptoFacade = object() - facade = new MailExportFacade(tokenFacade, bulkMailLoader, blobFacade, cryptoFacade) + blobAccessTokenFacade = instance(BlobAccessTokenFacade) + facade = new MailExportFacade(tokenFacade, bulkMailLoader, blobFacade, cryptoFacade, blobAccessTokenFacade) }) o.test("loadFixedNumberOfMailsWithCache", async () => { - when(bulkMailLoader.loadFixedNumberOfMailsWithCache("mailListId", "startId", { extraHeaders: tokenHeaders })).thenResolve([mail1, mail2]) + when(bulkMailLoader.loadFixedNumberOfMailsWithCache("mailListId", "startId", { baseUrl: "baseUrl", extraHeaders: tokenHeaders })).thenResolve([ + mail1, + mail2, + ]) - const result = await facade.loadFixedNumberOfMailsWithCache("mailListId", "startId") + const result = await facade.loadFixedNumberOfMailsWithCache("mailListId", "startId", "baseUrl") o(result).deepEquals([mail1, mail2]) }) @@ -56,9 +62,9 @@ o.spec("MailExportFacade", () => { o.test("loadAttachments", async () => { const expected = [createTestEntity(FileTypeRef), createTestEntity(FileTypeRef)] - when(bulkMailLoader.loadAttachments([mail1, mail2], { extraHeaders: tokenHeaders })).thenResolve(expected) + when(bulkMailLoader.loadAttachments([mail1, mail2], { baseUrl: "baseUrl", extraHeaders: tokenHeaders })).thenResolve(expected) - const result = await facade.loadAttachments([mail1, mail2]) + const result = await facade.loadAttachments([mail1, mail2], "baseUrl") o(result).deepEquals(expected) }) @@ -77,6 +83,7 @@ o.spec("MailExportFacade", () => { ArchiveDataType.Attachments, [createReferencingInstance(mailAttachments[0]), createReferencingInstance(mailAttachments[1])], { + baseUrl: "baseUrl", extraHeaders: tokenHeaders, }, ), @@ -87,7 +94,7 @@ o.spec("MailExportFacade", () => { ]), ) - const result = await facade.loadAttachmentData(mail1, mailAttachments) + const result = await facade.loadAttachmentData(mail1, mailAttachments, "baseUrl") o(result).deepEquals([ { _type: "DataFile", name: "mail1", mimeType: "img/png", data: dataByteMail1, cid: "12345", size: 3, id: ["attachment", "id1"] }, diff --git a/test/tests/api/worker/rest/EntityRestClientTest.ts b/test/tests/api/worker/rest/EntityRestClientTest.ts index 445c7f128e5e..6e274efe5763 100644 --- a/test/tests/api/worker/rest/EntityRestClientTest.ts +++ b/test/tests/api/worker/rest/EntityRestClientTest.ts @@ -125,6 +125,7 @@ o.spec("EntityRestClient", function () { headers: { ...authHeader, v: String(tutanotaModelInfo.version) }, responseType: MediaType.Json, queryParams: undefined, + baseUrl: undefined, }), ).thenResolve(JSON.stringify({ instance: "calendar" })) @@ -139,6 +140,7 @@ o.spec("EntityRestClient", function () { headers: { ...authHeader, v: String(sysModelInfo.version) }, responseType: MediaType.Json, queryParams: undefined, + baseUrl: undefined, }), ).thenResolve(JSON.stringify({ instance: "customer" })) @@ -154,6 +156,7 @@ o.spec("EntityRestClient", function () { headers: { ...authHeader, v: String(tutanotaModelInfo.version), baz: "quux" }, responseType: MediaType.Json, queryParams: { foo: "bar" }, + baseUrl: undefined, }), ).thenResolve(JSON.stringify({ instance: "calendar" })) @@ -174,6 +177,7 @@ o.spec("EntityRestClient", function () { headers: { ...authHeader, v: String(tutanotaModelInfo.version) }, responseType: MediaType.Json, queryParams: undefined, + baseUrl: undefined, }), ).thenResolve(JSON.stringify({ _ownerEncSessionKey: "some key" })) @@ -201,6 +205,7 @@ o.spec("EntityRestClient", function () { headers: { ...authHeader, v: String(tutanotaModelInfo.version) }, queryParams: { start: startId, count: String(count), reverse: String(false) }, responseType: MediaType.Json, + baseUrl: undefined, }), ).thenResolve(JSON.stringify([{ instance: 1 }, { instance: 2 }])) @@ -229,6 +234,7 @@ o.spec("EntityRestClient", function () { headers: { ...authHeader, v: String(sysModelInfo.version) }, queryParams: { ids: "0,1,2,3,4" }, responseType: MediaType.Json, + baseUrl: undefined, }), ).thenResolve(JSON.stringify([{ instance: 1 }, { instance: 2 }])) @@ -250,6 +256,7 @@ o.spec("EntityRestClient", function () { headers: { ...authHeader, v: String(sysModelInfo.version) }, queryParams: { ids: ids.join(",") }, responseType: MediaType.Json, + baseUrl: undefined, }), { times: 1 }, ).thenResolve(JSON.stringify([{ instance: 1 }, { instance: 2 }])) @@ -271,6 +278,7 @@ o.spec("EntityRestClient", function () { headers: { ...authHeader, v: String(sysModelInfo.version) }, queryParams: { ids: countFrom(0, 100).join(",") }, responseType: MediaType.Json, + baseUrl: undefined, }), { times: 1 }, ).thenResolve(JSON.stringify([{ instance: 1 }])) @@ -280,6 +288,7 @@ o.spec("EntityRestClient", function () { headers: { ...authHeader, v: String(sysModelInfo.version) }, queryParams: { ids: "100" }, responseType: MediaType.Json, + baseUrl: undefined, }), { times: 1 }, ).thenResolve(JSON.stringify([{ instance: 2 }])) @@ -299,6 +308,7 @@ o.spec("EntityRestClient", function () { headers: { ...authHeader, v: String(sysModelInfo.version) }, queryParams: { ids: countFrom(0, 100).join(",") }, responseType: MediaType.Json, + baseUrl: undefined, }), { times: 1 }, ).thenResolve(JSON.stringify([{ instance: 1 }])) @@ -308,6 +318,7 @@ o.spec("EntityRestClient", function () { headers: { ...authHeader, v: String(sysModelInfo.version) }, queryParams: { ids: countFrom(100, 100).join(",") }, responseType: MediaType.Json, + baseUrl: undefined, }), { times: 1 }, ).thenResolve(JSON.stringify([{ instance: 2 }])) @@ -317,6 +328,7 @@ o.spec("EntityRestClient", function () { headers: { ...authHeader, v: String(sysModelInfo.version) }, queryParams: { ids: countFrom(200, 11).join(",") }, responseType: MediaType.Json, + baseUrl: undefined, }), { times: 1 }, ).thenResolve(JSON.stringify([{ instance: 3 }])) diff --git a/test/tests/native/main/MailExportControllerTest.ts b/test/tests/native/main/MailExportControllerTest.ts index 9c92831ad0e8..446daa29fa32 100644 --- a/test/tests/native/main/MailExportControllerTest.ts +++ b/test/tests/native/main/MailExportControllerTest.ts @@ -24,11 +24,12 @@ import { GENERATED_MAX_ID, getElementId } from "../../../../src/common/api/commo import { assertNotNull } from "@tutao/tutanota-utils" import { createDataFile } from "../../../../src/common/api/common/DataFile.js" import { makeMailBundle } from "../../../../src/mail-app/mail/export/Bundler.js" -import { MailboxExportState } from "../../../../src/common/desktop/export/MailboxExportPersistence.js" import { MailExportFacade } from "../../../../src/common/api/worker/facades/lazy/MailExportFacade.js" -import { SuspensionError } from "../../../../src/common/api/common/error/SuspensionError" -import { spy } from "@tutao/tutanota-test-utils" +import { BlobServerUrlTypeRef } from "../../../../src/common/api/entities/storage/TypeRefs" import { ExportError, ExportErrorReason } from "../../../../src/common/api/common/error/ExportError" +import { MailboxExportState } from "../../../../src/common/desktop/export/MailboxExportPersistence" +import { spy } from "@tutao/tutanota-test-utils" +import { SuspensionError } from "../../../../src/common/api/common/error/SuspensionError" o.spec("MailExportController", function () { const userId = "userId" @@ -50,10 +51,16 @@ o.spec("MailExportController", function () { userController = { userId: userId } as Partial as UserController mailboxDetail = { mailbox: createTestEntity(MailBoxTypeRef, { - currentMailBag: createTestEntity(MailBagTypeRef), - archivedMailBags: [createTestEntity(MailBagTypeRef), createTestEntity(MailBagTypeRef)], + _id: "mailboxId", + currentMailBag: createTestEntity(MailBagTypeRef, { _id: "currentMailBagId", mails: "currentMailList" }), + archivedMailBags: [ + createTestEntity(MailBagTypeRef, { _id: "archivedMailBagId1", mails: "archivedMailList1" }), + createTestEntity(MailBagTypeRef, { _id: "archivedMailBagId2", mails: "archivedMailList2" }), + ], + }), + mailGroup: createTestEntity(GroupTypeRef, { + _id: "mailGroupId", }), - mailGroup: createTestEntity(GroupTypeRef), mailGroupInfo: createTestEntity(GroupInfoTypeRef), mailboxGroupRoot: createTestEntity(MailboxGroupRootTypeRef), } @@ -68,7 +75,7 @@ o.spec("MailExportController", function () { controller = new MailExportController(mailExportFacade, sanitizer, exportFacade, logins, mailboxModel, scheduler) }) - function prepareMailData(mailBag: MailBag, startId: Id) { + function prepareMailData(mailBag: MailBag, startId: Id, num: number) { const mailDetails = createTestEntity(MailDetailsTypeRef, { sentDate: new Date(42), body: createTestEntity(BodyTypeRef, { @@ -78,7 +85,7 @@ o.spec("MailExportController", function () { }) const attachmentInfo = createTestEntity(FileTypeRef, { _id: ["fileListId", "fileId"] }) const mail = createTestEntity(MailTypeRef, { - _id: ["mailListId", startId + "_1"], + _id: ["mailListId", startId + `_${num}`], attachments: [attachmentInfo._id], receivedDate: new Date(43), sender: createTestEntity(MailAddressTypeRef, { @@ -88,27 +95,33 @@ o.spec("MailExportController", function () { }) const attachmentData = new Uint8Array([1, 2, 3]) const dataFile = createDataFile("test", "application/octet-stream", attachmentData) - when(mailExportFacade.loadFixedNumberOfMailsWithCache(mailBag.mails, startId)).thenResolve([mail]) + when(mailExportFacade.loadFixedNumberOfMailsWithCache(mailBag.mails, startId, matchers.anything())).thenResolve([mail]) when(mailExportFacade.loadMailDetails([mail])).thenResolve([{ mail, mailDetails }]) - when(mailExportFacade.loadAttachments([mail])).thenResolve([attachmentInfo]) - when(mailExportFacade.loadAttachmentData(mail, [attachmentInfo])).thenResolve([dataFile]) + when(mailExportFacade.loadAttachments([mail], matchers.anything())).thenResolve([attachmentInfo]) + when(mailExportFacade.loadAttachmentData(mail, [attachmentInfo], matchers.anything())).thenResolve([dataFile]) const mailBundle = makeMailBundle(sanitizer, mail, mailDetails, [dataFile]) return { mail, mailBundle, mailDetails } } o.spec("startExport", function () { - o.test("it updates the initial state", function () { - controller.startExport(mailboxDetail) + o.test("it updates the initial state", async function () { + when(mailExportFacade.getExportServers(mailboxDetail.mailGroup)).thenResolve([createTestEntity(BlobServerUrlTypeRef, { url: "baseUrl" })]) + when(mailExportFacade.loadFixedNumberOfMailsWithCache(matchers.anything(), matchers.anything(), matchers.anything())).thenResolve([]) + await controller.startExport(mailboxDetail) verify( - exportFacade.startMailboxExport(userId, mailboxDetail.mailbox._id, assertNotNull(mailboxDetail.mailbox.currentMailBag).mails, GENERATED_MAX_ID), + exportFacade.startMailboxExport(userId, mailboxDetail.mailbox._id, assertNotNull(mailboxDetail.mailbox.currentMailBag)._id, GENERATED_MAX_ID), ) }) o.test("it runs the export", async function () { + when(mailExportFacade.getExportServers(mailboxDetail.mailGroup)).thenResolve([createTestEntity(BlobServerUrlTypeRef, { url: "baseUrl" })]) + const mailBag = assertNotNull(mailboxDetail.mailbox.currentMailBag) - const { mail, mailBundle } = prepareMailData(mailBag, GENERATED_MAX_ID) - when(mailExportFacade.loadFixedNumberOfMailsWithCache(mailBag.mails, getElementId(mail))).thenResolve([]) + const { mail, mailBundle } = prepareMailData(mailBag, GENERATED_MAX_ID, 1) + prepareMailData(assertNotNull(mailboxDetail.mailbox.archivedMailBags[0]), GENERATED_MAX_ID, 2) + prepareMailData(assertNotNull(mailboxDetail.mailbox.archivedMailBags[1]), GENERATED_MAX_ID, 3) + when(mailExportFacade.loadFixedNumberOfMailsWithCache(matchers.anything(), matchers.not(GENERATED_MAX_ID), matchers.anything())).thenResolve([]) await controller.startExport(mailboxDetail) @@ -117,7 +130,7 @@ o.spec("MailExportController", function () { o.test("it sets state to locked when a LockedForUser ExportError is thrown", async function () { when( - exportFacade.startMailboxExport(userId, mailboxDetail.mailbox._id, assertNotNull(mailboxDetail.mailbox.currentMailBag).mails, GENERATED_MAX_ID), + exportFacade.startMailboxExport(userId, mailboxDetail.mailbox._id, assertNotNull(mailboxDetail.mailbox.currentMailBag)._id, GENERATED_MAX_ID), ).thenReject(new ExportError("message", ExportErrorReason.LockedForUser)) await controller.startExport(mailboxDetail) o(controller.state().type).equals("locked") @@ -135,7 +148,7 @@ o.spec("MailExportController", function () { o.test("when persisted state is running it runs the export", async function () { const initialMailId = "initialMailId" const mailBag = assertNotNull(mailboxDetail.mailbox.currentMailBag) - const { mail, mailBundle } = prepareMailData(mailBag, initialMailId) + const { mail, mailBundle } = prepareMailData(mailBag, initialMailId, 1) const persistedState: MailboxExportState = { type: "running", mailboxId: mailboxDetail.mailbox._id, @@ -146,7 +159,9 @@ o.spec("MailExportController", function () { exportDirectoryPath: "directory", } when(exportFacade.getMailboxExportState(userId)).thenResolve(persistedState) - when(mailExportFacade.loadFixedNumberOfMailsWithCache(mailBag.mails, getElementId(mail))).thenResolve([]) + when(mailExportFacade.getExportServers(mailboxDetail.mailGroup)).thenResolve([createTestEntity(BlobServerUrlTypeRef, { url: "baseUrl" })]) + when(mailExportFacade.loadFixedNumberOfMailsWithCache(mailBag.mails, matchers.not(initialMailId), matchers.anything())).thenResolve([]) + when(mailExportFacade.loadFixedNumberOfMailsWithCache(matchers.not(mailBag.mails), matchers.anything(), matchers.anything())).thenResolve([]) await controller.resumeIfNeeded() @@ -166,19 +181,25 @@ o.spec("MailExportController", function () { }) o.spec("cancelExport", function () { - o.test("canceling resets the state", function () { - controller.startExport(mailboxDetail) - controller.cancelExport() + o.test("canceling resets the state", async function () { + when(mailExportFacade.getExportServers(mailboxDetail.mailGroup)).thenResolve([createTestEntity(BlobServerUrlTypeRef, { url: "baseUrl" })]) + when(mailExportFacade.loadFixedNumberOfMailsWithCache(matchers.anything(), matchers.anything(), matchers.anything())).thenResolve([]) + const startPromise = controller.startExport(mailboxDetail) + const cancelPromise = controller.cancelExport() o(controller.state().type).equals("idle") + + await Promise.all([startPromise, cancelPromise]) }) }) o.spec("export loop", function () { o.test("it continues to load mail list", async function () { const mailBag = assertNotNull(mailboxDetail.mailbox.currentMailBag) - const { mail: mail1 } = prepareMailData(mailBag, GENERATED_MAX_ID) - const { mail: mail2, mailBundle: mailBundle2 } = prepareMailData(mailBag, getElementId(mail1)) - when(mailExportFacade.loadFixedNumberOfMailsWithCache(mailBag.mails, getElementId(mail2))).thenResolve([]) + const { mail: mail1 } = prepareMailData(mailBag, GENERATED_MAX_ID, 1) + const { mail: mail2, mailBundle: mailBundle2 } = prepareMailData(mailBag, getElementId(mail1), 2) + when(mailExportFacade.loadFixedNumberOfMailsWithCache(mailBag.mails, getElementId(mail2), matchers.anything())).thenResolve([]) + when(mailExportFacade.loadFixedNumberOfMailsWithCache(matchers.not(mailBag.mails), matchers.anything(), matchers.anything())).thenResolve([]) + when(mailExportFacade.getExportServers(mailboxDetail.mailGroup)).thenResolve([createTestEntity(BlobServerUrlTypeRef, { url: "baseUrl" })]) await controller.startExport(mailboxDetail) @@ -188,14 +209,15 @@ o.spec("MailExportController", function () { o.test("it loops over mail bags", async function () { const currentMailBag = assertNotNull(mailboxDetail.mailbox.currentMailBag) - const { mail: mail1, mailBundle: mailBundle1 } = prepareMailData(currentMailBag, GENERATED_MAX_ID) - when(mailExportFacade.loadFixedNumberOfMailsWithCache(currentMailBag.mails, getElementId(mail1))).thenResolve([]) + const { mail: mail1, mailBundle: mailBundle1 } = prepareMailData(currentMailBag, GENERATED_MAX_ID, 1) + when(mailExportFacade.loadFixedNumberOfMailsWithCache(currentMailBag.mails, getElementId(mail1), "baseUrl")).thenResolve([]) const archivedMailBag1 = mailboxDetail.mailbox.archivedMailBags[0] - const { mail: mail2, mailBundle: mailBundle2 } = prepareMailData(archivedMailBag1, GENERATED_MAX_ID) - when(mailExportFacade.loadFixedNumberOfMailsWithCache(archivedMailBag1.mails, getElementId(mail2))).thenResolve([]) + const { mail: mail2, mailBundle: mailBundle2 } = prepareMailData(archivedMailBag1, GENERATED_MAX_ID, 2) + when(mailExportFacade.loadFixedNumberOfMailsWithCache(archivedMailBag1.mails, getElementId(mail2), "baseUrl")).thenResolve([]) const archivedMailBag2 = mailboxDetail.mailbox.archivedMailBags[1] - const { mail: mail3, mailBundle: mailBundle3 } = prepareMailData(archivedMailBag2, GENERATED_MAX_ID) - when(mailExportFacade.loadFixedNumberOfMailsWithCache(archivedMailBag2.mails, getElementId(mail3))).thenResolve([]) + const { mail: mail3, mailBundle: mailBundle3 } = prepareMailData(archivedMailBag2, GENERATED_MAX_ID, 3) + when(mailExportFacade.loadFixedNumberOfMailsWithCache(archivedMailBag2.mails, getElementId(mail3), "baseUrl")).thenResolve([]) + when(mailExportFacade.getExportServers(mailboxDetail.mailGroup)).thenResolve([{ _id: "id", url: "baseUrl", _type: BlobServerUrlTypeRef }]) await controller.startExport(mailboxDetail) @@ -204,12 +226,31 @@ o.spec("MailExportController", function () { verify(exportFacade.saveMailboxExport(mailBundle3, userId, archivedMailBag2._id, getElementId(mail3))) verify(exportFacade.endMailboxExport(userId)) }) + + o.test("it loops over servers", async function () { + when(mailExportFacade.getExportServers(mailboxDetail.mailGroup)).thenResolve([ + { _id: "id", url: "baseUrl1", _type: BlobServerUrlTypeRef }, + { _id: "id", url: "baseUrl2", _type: BlobServerUrlTypeRef }, + { _id: "id", url: "baseUrl3", _type: BlobServerUrlTypeRef }, + ]) + const currentMailBag = assertNotNull(mailboxDetail.mailbox.currentMailBag) + const { mail: mail1, mailBundle: mailBundle1, mailDetails: mailDetails1 } = prepareMailData(currentMailBag, GENERATED_MAX_ID, 1) + when(mailExportFacade.loadFixedNumberOfMailsWithCache(currentMailBag.mails, getElementId(mail1), matchers.anything())).thenResolve([]) + when(mailExportFacade.loadFixedNumberOfMailsWithCache(matchers.not(currentMailBag.mails), matchers.anything(), matchers.anything())).thenResolve([]) + + await controller.startExport(mailboxDetail) + + verify(mailExportFacade.loadFixedNumberOfMailsWithCache(currentMailBag.mails, GENERATED_MAX_ID, "baseUrl2")) + verify(mailExportFacade.loadAttachments([mail1], "baseUrl3")) + verify(mailExportFacade.loadAttachmentData(mail1, matchers.anything(), "baseUrl1")) + }) }) o.spec("handle errors", function () { o.test("SuspensionError", async () => { + when(mailExportFacade.getExportServers(mailboxDetail.mailGroup)).thenResolve([{ _id: "id", url: "baseUrl", _type: BlobServerUrlTypeRef }]) let wasThrown = false - when(mailExportFacade.loadFixedNumberOfMailsWithCache(matchers.anything(), matchers.anything())).thenDo(() => { + when(mailExportFacade.loadFixedNumberOfMailsWithCache(matchers.anything(), matchers.anything(), matchers.anything())).thenDo(() => { if (wasThrown) { return Promise.resolve([]) } else { @@ -218,7 +259,7 @@ o.spec("MailExportController", function () { } }) await controller.startExport(mailboxDetail) - verify(mailExportFacade.loadFixedNumberOfMailsWithCache(matchers.anything(), matchers.anything()), { times: 3 + 1 }) + verify(mailExportFacade.loadFixedNumberOfMailsWithCache(matchers.anything(), matchers.anything(), matchers.anything()), { times: 3 + 1 }) o(wasThrown).equals(true) }) }) From b3268496da6db9bddb357c2732e52b1c980fd4de Mon Sep 17 00:00:00 2001 From: ivk Date: Fri, 20 Dec 2024 18:06:19 +0100 Subject: [PATCH 2/3] Don't use baseUrl for requesting blob access tokens for files Co-authored-by: paw --- .../api/worker/facades/lazy/MailExportFacade.ts | 3 +-- src/mail-app/native/main/MailExportController.ts | 12 ++++++------ 2 files changed, 7 insertions(+), 8 deletions(-) diff --git a/src/common/api/worker/facades/lazy/MailExportFacade.ts b/src/common/api/worker/facades/lazy/MailExportFacade.ts index f6c457bab174..e3bf810be279 100644 --- a/src/common/api/worker/facades/lazy/MailExportFacade.ts +++ b/src/common/api/worker/facades/lazy/MailExportFacade.ts @@ -57,13 +57,12 @@ export class MailExportFacade { return this.mailExportTokenFacade.loadWithToken((token) => this.bulkMailLoader.loadAttachments(mails, { baseUrl, ...this.options(token) })) } - async loadAttachmentData(mail: Mail, attachments: readonly TutanotaFile[], baseUrl: string): Promise { + async loadAttachmentData(mail: Mail, attachments: readonly TutanotaFile[]): Promise { const attachmentsWithKeys = await this.cryptoFacade.enforceSessionKeyUpdateIfNeeded(mail, attachments) const downloads = await this.mailExportTokenFacade.loadWithToken((token) => { const referencingInstances = attachmentsWithKeys.map(createReferencingInstance) return this.blobFacade.downloadAndDecryptBlobsOfMultipleInstances(ArchiveDataType.Attachments, referencingInstances, { - baseUrl, ...this.options(token), }) }) diff --git a/src/mail-app/native/main/MailExportController.ts b/src/mail-app/native/main/MailExportController.ts index 9c117749afb9..b53055dcdbd8 100644 --- a/src/mail-app/native/main/MailExportController.ts +++ b/src/mail-app/native/main/MailExportController.ts @@ -39,7 +39,7 @@ export class MailExportController { private _state: Stream = stream({ type: "idle" }) private _lastExport: Date | null = null private servers?: BlobServerUrl[] - private serverCount: number = 0 + private serverIndex: number = 0 get lastExport(): Date | null { return this._lastExport @@ -181,7 +181,7 @@ export class MailExportController { const mailAttachmentInfo = mail.attachments .map((attachmentId) => attachmentInfo.find((attachment) => isSameId(attachment._id, attachmentId))) .filter(isNotNull) - const attachments = await this.mailExportFacade.loadAttachmentData(mail, mailAttachmentInfo, this.getServerUrl()) + const attachments = await this.mailExportFacade.loadAttachmentData(mail, mailAttachmentInfo) const { makeMailBundle } = await import("../../mail/export/Bundler.js") const mailBundle = makeMailBundle(this.sanitizer, mail, mailDetails, attachments) @@ -228,11 +228,11 @@ export class MailExportController { private getServerUrl(): string { if (this.servers) { - this.serverCount += 1 - if (this.serverCount >= this.servers.length) { - this.serverCount = 0 + this.serverIndex += 1 + if (this.serverIndex >= this.servers.length) { + this.serverIndex = 0 } - return this.servers[this.serverCount].url + return this.servers[this.serverIndex].url } throw new Error("No servers") } From 144c51d28996753d4eba10413e0c63876d5561f5 Mon Sep 17 00:00:00 2001 From: ivk Date: Fri, 20 Dec 2024 18:12:58 +0100 Subject: [PATCH 3/3] Fix parsing multiple blob response Co-authored-by: paw --- .../api/worker/facades/lazy/BlobFacade.ts | 23 ++++++++--- .../api/worker/facades/BlobFacadeTest.ts | 40 ++++++++++++++----- 2 files changed, 46 insertions(+), 17 deletions(-) diff --git a/src/common/api/worker/facades/lazy/BlobFacade.ts b/src/common/api/worker/facades/lazy/BlobFacade.ts index 8047cbb39870..809eb1bf2d6c 100644 --- a/src/common/api/worker/facades/lazy/BlobFacade.ts +++ b/src/common/api/worker/facades/lazy/BlobFacade.ts @@ -27,7 +27,7 @@ import type { AesApp } from "../../../../native/worker/AesApp.js" import { InstanceMapper } from "../../crypto/InstanceMapper.js" import { Blob, BlobReferenceTokenWrapper, createBlobReferenceTokenWrapper } from "../../../entities/sys/TypeRefs.js" import { FileReference } from "../../../common/utils/FileUtils.js" -import { handleRestError, NotFoundError } from "../../../common/error/RestError.js" +import { handleRestError } from "../../../common/error/RestError.js" import { ProgrammingError } from "../../../common/error/ProgrammingError.js" import { IServiceExecutor } from "../../../common/ServiceRequest.js" import { BlobGetInTypeRef, BlobPostOut, BlobPostOutTypeRef, BlobServerAccessInfo, createBlobGetIn, createBlobId } from "../../../entities/storage/TypeRefs.js" @@ -454,23 +454,34 @@ export class BlobFacade { /** * Deserializes a list of BlobWrappers that are in the following binary format - * element [ blobId ] [ blobHash ] [blobSize] [blob] [ . . . ] [ blobNId ] [ blobNHash ] [blobNSize] [blobN] - * bytes 9 6 4 blobSize 9 6 4 blobSize + * element [ #blobs ] [ blobId ] [ blobHash ] [blobSize] [blob] [ . . . ] [ blobNId ] [ blobNHash ] [blobNSize] [blobN] + * bytes 4 9 6 4 blobSize 9 6 4 blobSize * * @return a map from blobId to the binary data */ export function parseMultipleBlobsResponse(concatBinaryData: Uint8Array): Map { - let offset = 0 const dataView = new DataView(concatBinaryData.buffer) const result = new Map() + const blobCount = dataView.getInt32(0) + if (blobCount <= 0) { + throw new Error(`Invalid blob count: ${blobCount}`) + } + let offset = 4 while (offset < concatBinaryData.length) { const blobIdBytes = concatBinaryData.slice(offset, offset + 9) const blobId = base64ToBase64Ext(uint8ArrayToBase64(blobIdBytes)) const blobSize = dataView.getInt32(offset + 15) - const contents = concatBinaryData.slice(offset + 19, offset + 19 + blobSize) + const dataStartOffset = offset + 19 + if (blobSize < 0 || dataStartOffset + blobSize > concatBinaryData.length) { + throw new Error(`Invalid blob size: ${blobSize}. Remaining length: ${concatBinaryData.length - dataStartOffset}`) + } + const contents = concatBinaryData.slice(dataStartOffset, dataStartOffset + blobSize) result.set(blobId, contents) - offset += 9 + 6 + 4 + blobSize + offset = dataStartOffset + blobSize + } + if (blobCount !== result.size) { + throw new Error(`Parsed wrong number of blobs: ${blobCount}. Expected: ${result.size}`) } return result } diff --git a/test/tests/api/worker/facades/BlobFacadeTest.ts b/test/tests/api/worker/facades/BlobFacadeTest.ts index 859adbfd0f9e..b233ac4e2940 100644 --- a/test/tests/api/worker/facades/BlobFacadeTest.ts +++ b/test/tests/api/worker/facades/BlobFacadeTest.ts @@ -182,6 +182,8 @@ o.spec("BlobFacade test", function () { // data size is 65 (16 data block, 16 iv, 32 hmac, 1 byte for mac marking) const blobSizeBinary = new Uint8Array([0, 0, 0, 65]) const blobResponse = concat( + // number of blobs + new Uint8Array([0, 0, 0, 1]), // blob id base64ToUint8Array(base64ExtToBase64(blobId)), // blob hash @@ -230,6 +232,8 @@ o.spec("BlobFacade test", function () { // data size is 65 (16 data block, 16 iv, 32 hmac, 1 byte for mac marking) const blobSizeBinary = new Uint8Array([0, 0, 0, 65]) const blobResponse = concat( + // number of blobs + new Uint8Array([0, 0, 0, 2]), // blob id base64ToUint8Array(base64ExtToBase64(blobId1)), // blob hash @@ -374,6 +378,8 @@ o.spec("BlobFacade test", function () { // data size is 65 (16 data block, 16 iv, 32 hmac, 1 byte for mac marking) const blobSizeBinary = new Uint8Array([0, 0, 0, 65]) const blobResponse = concat( + // number of blobs + new Uint8Array([0, 0, 0, 3]), // blob id base64ToUint8Array(base64ExtToBase64(blobId1)), // blob hash @@ -463,6 +469,8 @@ o.spec("BlobFacade test", function () { // data size is 65 (16 data block, 16 iv, 32 hmac, 1 byte for mac marking) const blobSizeBinary = new Uint8Array([0, 0, 0, 65]) const blobResponse1 = concat( + // number of blobs + new Uint8Array([0, 0, 0, 2]), // blob id base64ToUint8Array(base64ExtToBase64(blobId1)), // blob hash @@ -482,6 +490,8 @@ o.spec("BlobFacade test", function () { ) const blobResponse2 = concat( + // number of blobs + new Uint8Array([0, 0, 0, 1]), //blodId base64ToUint8Array(base64ExtToBase64(blobId3)), // blob hash @@ -553,6 +563,8 @@ o.spec("BlobFacade test", function () { // data size is 65 (16 data block, 16 iv, 32 hmac, 1 byte for mac marking) const blobSizeBinary = new Uint8Array([0, 0, 0, 65]) const blobResponse = concat( + // number of blobs + new Uint8Array([0, 0, 0, 2]), // blob id base64ToUint8Array(base64ExtToBase64(blobId1)), // blob hash @@ -583,13 +595,15 @@ o.spec("BlobFacade test", function () { // Blob id OETv4XP----0 hash [3, -112, 88, -58, -14, -64] bytes [1, 2, 3] // Blob id OETv4XS----0 hash [113, -110, 56, 92, 60, 6] bytes [1, 2, 3, 4, 5, 6] const binaryData = new Int8Array([ - // blob id 1 [0-8] + // number of blobs [0-3] 2 + 0, 0, 0, 2, + // blob id 1 [4-12] 100, -9, -69, 22, 38, -128, 0, 0, 1, - // blob hash 1 [9-14] + // blob hash 1 [13-18] 3, -112, 88, -58, -14, -64, - // blob size 1 [15-18] + // blob size 1 [19-22] 0, 0, 0, 3, - // blob data 1 [19-21] + // blob data 1 [23-25] 1, 2, 3, // blob id 2 100, -9, -69, 22, 39, 64, 0, 0, 1, @@ -613,13 +627,15 @@ o.spec("BlobFacade test", function () { o.test("parses one blob", function () { // Blob id OETv4XP----0 hash [3, -112, 88, -58, -14, -64] bytes [1, 2, 3] const binaryData = new Int8Array([ - // blob id 1 [0-8] + // number of blobs [0-3] + 0, 0, 0, 1, + // blob id 1 [4-12] 100, -9, -69, 22, 38, -128, 0, 0, 1, - // blob hash 1 [9-14] + // blob hash 1 [13-18] 3, -112, 88, -58, -14, -64, - // blob size 1 [15-18] + // blob size 1 [19-22] 0, 0, 0, 3, - // blob data 1 [19-21] + // blob data 1 [23-25] 1, 2, 3, ]) @@ -632,11 +648,13 @@ o.spec("BlobFacade test", function () { const blobDataNumbers = Array(384).fill(1) const binaryData = new Int8Array( [ - // blob id 1 [0-8] + // number of blobs [0-3] + 0, 0, 0, 1, + // blob id 1 [4-12] 100, -9, -69, 22, 38, -128, 0, 0, 1, - // blob hash 1 [9-14] + // blob hash 1 [13-18] 3, -112, 88, -58, -14, -64, - // blob size 1 [15-18] 384 + // blob size 1 [19-22] 0, 0, 1, 128, ].concat(blobDataNumbers), )