From ce331167c49b61b48ee9a2a7ebb0a25522d67f8f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Lloren=C3=A7=20Muntaner?= Date: Mon, 5 Aug 2024 15:28:48 +0200 Subject: [PATCH] Use new warning page and change the rendering logic (#2551) * Use new warning page with new showing logic * Use a helper to be tested * Add missing tests * Improve test description * Improve comment * Refactor helper * Improve comment * Improve comments * Fix type --- .../src/flows/recovery/recoveryWizard.test.ts | 204 ++++++++++++++++++ .../src/flows/recovery/recoveryWizard.ts | 136 ++++++++++-- .../src/repositories/identityMetadata.test.ts | 141 +++++++++++- .../src/repositories/identityMetadata.ts | 95 +++++--- src/frontend/src/utils/utils.ts | 7 + .../pages/addDeviceWarningOnePasskey.astro | 2 +- 6 files changed, 533 insertions(+), 52 deletions(-) create mode 100644 src/frontend/src/flows/recovery/recoveryWizard.test.ts diff --git a/src/frontend/src/flows/recovery/recoveryWizard.test.ts b/src/frontend/src/flows/recovery/recoveryWizard.test.ts new file mode 100644 index 0000000000..2f686bbff4 --- /dev/null +++ b/src/frontend/src/flows/recovery/recoveryWizard.test.ts @@ -0,0 +1,204 @@ +import { + AnchorCredentials, + CredentialId, + PublicKey, + WebAuthnCredential, +} from "$generated/internet_identity_types"; +import { PinIdentityMaterial } from "../pin/idb"; +import { getDevicesStatus } from "./recoveryWizard"; + +const ONE_WEEK_MILLIS = 7 * 24 * 60 * 60 * 1000; +const nowInMillis = 1722259851155; +const moreThanAWeekAgo = nowInMillis - ONE_WEEK_MILLIS - 1; +const lessThanAWeekAgo = nowInMillis - 1; + +const pinIdentityMaterial: PinIdentityMaterial = + {} as unknown as PinIdentityMaterial; + +const noCredentials: AnchorCredentials = { + credentials: [], + recovery_credentials: [], + recovery_phrases: [], +}; + +const device: WebAuthnCredential = { + pubkey: [] as PublicKey, + credential_id: [] as CredentialId, +}; + +const oneDeviceOnly: AnchorCredentials = { + credentials: [device], + recovery_credentials: [], + recovery_phrases: [], +}; + +const oneRecoveryDeviceOnly: AnchorCredentials = { + credentials: [], + recovery_credentials: [device], + recovery_phrases: [], +}; + +const oneDeviceAndPhrase: AnchorCredentials = { + credentials: [device], + recovery_credentials: [], + recovery_phrases: [[] as PublicKey], +}; + +const twoDevices: AnchorCredentials = { + credentials: [device, { ...device }], + recovery_credentials: [], + recovery_phrases: [[] as PublicKey], +}; + +const threeDevices: AnchorCredentials = { + credentials: [device, { ...device }, { ...device }], + recovery_credentials: [], + recovery_phrases: [[] as PublicKey], +}; + +const oneNormalOneRecovery: AnchorCredentials = { + credentials: [device], + recovery_credentials: [device], + recovery_phrases: [[] as PublicKey], +}; + +test("getDevicesStatus returns 'pin-only' for user with pin and has seen recovery longer than a week ago", () => { + expect( + getDevicesStatus({ + credentials: noCredentials, + identityMetadata: { + recoveryPageShownTimestampMillis: moreThanAWeekAgo, + }, + pinIdentityMaterial, + nowInMillis, + }) + ).toBe("pin-only"); +}); + +test("getDevicesStatus returns 'one-device' for user with one passkey and has seen recovery longer than a week ago", () => { + expect( + getDevicesStatus({ + credentials: oneDeviceOnly, + identityMetadata: { + recoveryPageShownTimestampMillis: moreThanAWeekAgo, + }, + pinIdentityMaterial: undefined, + nowInMillis, + }) + ).toBe("one-device"); +}); + +test("getDevicesStatus returns true for user with one passkey and empty identity metadata", () => { + expect( + getDevicesStatus({ + credentials: oneDeviceOnly, + identityMetadata: {}, + pinIdentityMaterial: undefined, + nowInMillis, + }) + ).toBe("one-device"); +}); + +test("getDevicesStatus returns 'one-device' for user with one recovery device and has seen recovery longer than a week ago", () => { + expect( + getDevicesStatus({ + credentials: oneRecoveryDeviceOnly, + identityMetadata: { + recoveryPageShownTimestampMillis: moreThanAWeekAgo, + }, + pinIdentityMaterial: undefined, + nowInMillis, + }) + ).toBe("one-device"); +}); + +test("getDevicesStatus returns 'one-device' for user with one device and a recovery phrase", () => { + expect( + getDevicesStatus({ + credentials: oneDeviceAndPhrase, + identityMetadata: { + recoveryPageShownTimestampMillis: moreThanAWeekAgo, + }, + pinIdentityMaterial: undefined, + nowInMillis, + }) + ).toBe("one-device"); +}); + +test("getDevicesStatus returns 'no-warning' for user with pin that has disabled the warning", () => { + expect( + getDevicesStatus({ + credentials: noCredentials, + identityMetadata: { + recoveryPageShownTimestampMillis: moreThanAWeekAgo, + doNotShowRecoveryPageRequestTimestampMillis: moreThanAWeekAgo, + }, + pinIdentityMaterial, + nowInMillis, + }) + ).toBe("no-warning"); +}); + +test("getDevicesStatus returns 'no-warning' for user with one device that has disabled the warning", () => { + expect( + getDevicesStatus({ + credentials: oneDeviceOnly, + identityMetadata: { + recoveryPageShownTimestampMillis: moreThanAWeekAgo, + doNotShowRecoveryPageRequestTimestampMillis: moreThanAWeekAgo, + }, + pinIdentityMaterial: undefined, + nowInMillis, + }) + ).toBe("no-warning"); +}); + +test("getDevicesStatus returns 'no-warning' for user with two devices", () => { + expect( + getDevicesStatus({ + credentials: twoDevices, + identityMetadata: { + recoveryPageShownTimestampMillis: moreThanAWeekAgo, + }, + pinIdentityMaterial: undefined, + nowInMillis, + }) + ).toBe("no-warning"); +}); + +test("getDevicesStatus returns 'no-warning' for user with one normal device and a recovery device", () => { + expect( + getDevicesStatus({ + credentials: oneNormalOneRecovery, + identityMetadata: { + recoveryPageShownTimestampMillis: moreThanAWeekAgo, + }, + pinIdentityMaterial: undefined, + nowInMillis, + }) + ).toBe("no-warning"); +}); + +test("getDevicesStatus returns 'no-warning' for user with more than two devices and empty identity metadata", () => { + expect( + getDevicesStatus({ + credentials: threeDevices, + identityMetadata: {}, + pinIdentityMaterial: undefined, + nowInMillis, + }) + ).toBe("no-warning"); +}); + +test("getDevicesStatus returns 'no-warning' for user with pin and has seen recovery less than a week ago", () => { + expect( + getDevicesStatus({ + credentials: noCredentials, + identityMetadata: { + recoveryPageShownTimestampMillis: lessThanAWeekAgo, + }, + pinIdentityMaterial, + nowInMillis, + }) + ).toBe("no-warning"); +}); diff --git a/src/frontend/src/flows/recovery/recoveryWizard.ts b/src/frontend/src/flows/recovery/recoveryWizard.ts index e4fad90808..3af05989be 100644 --- a/src/frontend/src/flows/recovery/recoveryWizard.ts +++ b/src/frontend/src/flows/recovery/recoveryWizard.ts @@ -4,9 +4,16 @@ import { renderPage } from "$src/utils/lit-html"; import { TemplateResult } from "lit-html"; import { AuthenticatedConnection } from "$src/utils/iiConnection"; -import { setupRecovery } from "./setupRecovery"; +import { AnchorCredentials } from "$generated/internet_identity_types"; import { infoScreenTemplate } from "$src/components/infoScreen"; +import { IdentityMetadata } from "$src/repositories/identityMetadata"; +import { isNullish } from "@dfinity/utils"; +import { addDevice } from "../addDevice/manage/addDevice"; +import { + PinIdentityMaterial, + idbRetrievePinIdentityMaterial, +} from "../pin/idb"; import copyJson from "./recoveryWizard.json"; /* Phrase creation kick-off screen */ @@ -80,7 +87,7 @@ export const addPhrase = ({ ); }; -type DeviceStatus = "pin-only" | "one-passkey"; +type DeviceStatus = "pin-only" | "one-device"; const addDeviceWarningTemplate = ({ ok, @@ -102,7 +109,7 @@ const addDeviceWarningTemplate = ({ copy.paragraph_add_device_pin_only, copy.add_device_title_pin_only, ], - "one-passkey": [ + "one-device": [ copy.paragraph_add_device_one_passkey, copy.add_device_title_one_passkey, ], @@ -127,6 +134,82 @@ const addDeviceWarningTemplate = ({ // TODO: Create the `addDeviceWarning` page and use it in `recoveryWizard` function. export const addDeviceWarningPage = renderPage(addDeviceWarningTemplate); +// Prompt the user to create a recovery phrase +export const addDeviceWarning = ({ + status, +}: { + status: DeviceStatus; +}): Promise<{ action: "remind-later" | "do-not-remind" | "add-device" }> => { + return new Promise((resolve) => + addDeviceWarningPage({ + i18n: new I18n(), + ok: () => resolve({ action: "add-device" }), + remindLater: () => resolve({ action: "remind-later" }), + doNotRemindAgain: () => resolve({ action: "do-not-remind" }), + status, + }) + ); +}; + +/** + * Helper to encapsulate the logic of when and which recovery warning page to show. + * + * Three conditions must be met for the warning page to be shown: + * * Not having seen the recovery page in the last week + * (on registration, the user is not shown the page, but set it as seen to not bother during the onboarding) + * * The user has at most one device. + * (a phrase and pin are not considered a device, only normal devices or recovery devices) + * * The user has not disabled the warning. + * (users can choose to not see the warning again by clicking "do not remind" button) + * + * When the warning page is shown, two different messages could be displayed: + * * User has only the pin authentication method. + * * User has only one device. + * + * @param params {Object} + * @param params.credentials {AnchorCredentials} + * @param params.identityMetadata {IdentityMetadata | undefined} + * @param params.pinIdentityMaterial {PinIdentityMaterial | undefined} + * @param params.nowInMillis {number} + * @returns {DeviceStatus | "no-warning"} + */ +// Exported for testing +export const getDevicesStatus = ({ + credentials, + identityMetadata, + pinIdentityMaterial, + nowInMillis, +}: { + credentials: AnchorCredentials; + identityMetadata: IdentityMetadata | undefined; + pinIdentityMaterial: PinIdentityMaterial | undefined; + nowInMillis: number; +}): DeviceStatus | "no-warning" => { + const ONE_WEEK_MILLIS = 7 * 24 * 60 * 60 * 1000; + const oneWeekAgoTimestamp = nowInMillis - ONE_WEEK_MILLIS; + const hasNotSeenRecoveryPageLastWeek = + (identityMetadata?.recoveryPageShownTimestampMillis ?? 0) < + oneWeekAgoTimestamp; + const showWarningPageEnabled = isNullish( + identityMetadata?.doNotShowRecoveryPageRequestTimestampMillis + ); + const totalDevicesCount = + credentials.credentials.length + credentials.recovery_credentials.length; + if ( + totalDevicesCount <= 1 && + hasNotSeenRecoveryPageLastWeek && + showWarningPageEnabled + ) { + if (totalDevicesCount === 0 && !pinIdentityMaterial) { + // This should never happen because it means that the user has no devices and no pin. + // But we still handle it to avoid a crash assuming there was an error retrieving the pin material. + return "pin-only"; + } + return totalDevicesCount === 0 ? "pin-only" : "one-device"; + } + return "no-warning"; +}; + // TODO: Add e2e test https://dfinity.atlassian.net/browse/GIX-2600 export const recoveryWizard = async ( userNumber: bigint, @@ -134,29 +217,42 @@ export const recoveryWizard = async ( ): Promise => { // Here, if the user doesn't have any recovery device, we prompt them to add // one. - const [recoveries, identityMetadata] = await withLoader(() => - Promise.all([ - connection.lookupRecovery(userNumber), - connection.getIdentityMetadata(), - ]) + const [credentials, identityMetadata, pinIdentityMaterial] = await withLoader( + () => + Promise.all([ + connection.lookupCredentials(userNumber), + connection.getIdentityMetadata(), + idbRetrievePinIdentityMaterial({ + userNumber, + }), + ]) ); - - const ONE_WEEK_MILLIS = 7 * 24 * 60 * 60 * 1000; const nowInMillis = Date.now(); - const oneWeekAgoTimestamp = nowInMillis - ONE_WEEK_MILLIS; - const hasNotSeenRecoveryPageLastWeek = - (identityMetadata?.recoveryPageShownTimestampMillis ?? 0) < - oneWeekAgoTimestamp; - if (recoveries.length === 0 && hasNotSeenRecoveryPageLastWeek) { + + const devicesStatus = getDevicesStatus({ + credentials, + identityMetadata, + pinIdentityMaterial, + nowInMillis, + }); + + if (devicesStatus !== "no-warning") { // `await` here doesn't add any waiting time beacause we already got the metadata earlier. await connection.updateIdentityMetadata({ recoveryPageShownTimestampMillis: nowInMillis, }); - const doAdd = await addPhrase({ intent: "securityReminder" }); - if (doAdd !== "cancel") { - doAdd satisfies "ok"; - - await setupRecovery({ userNumber, connection }); + const userChoice = await addDeviceWarning({ + status: devicesStatus, + }); + if (userChoice.action === "add-device") { + await addDevice({ userNumber, connection }); + } + if (userChoice.action === "do-not-remind") { + // `await` here doesn't add any waiting time beacause we already got the metadata earlier. + await connection.updateIdentityMetadata({ + doNotShowRecoveryPageRequestTimestampMillis: nowInMillis, + }); } + // Do nothing if `"remind-later"`. } }; diff --git a/src/frontend/src/repositories/identityMetadata.test.ts b/src/frontend/src/repositories/identityMetadata.test.ts index 18940fbe23..1402bd2d7e 100644 --- a/src/frontend/src/repositories/identityMetadata.test.ts +++ b/src/frontend/src/repositories/identityMetadata.test.ts @@ -1,19 +1,26 @@ import { MetadataMapV2 } from "$generated/internet_identity_types"; import { + DO_NOT_SHOW_RECOVERY_PAGE_REQUEST_TIMESTAMP_MILLIS, IdentityMetadata, IdentityMetadataRepository, RECOVERY_PAGE_SHOW_TIMESTAMP_MILLIS, } from "./identityMetadata"; const recoveryPageShownTimestampMillis = 1234567890; +const doNotShowRecoveryPageRequestTimestampMillis = 3456789012; const mockRawMetadata: MetadataMapV2 = [ [ RECOVERY_PAGE_SHOW_TIMESTAMP_MILLIS, { String: String(recoveryPageShownTimestampMillis) }, ], + [ + DO_NOT_SHOW_RECOVERY_PAGE_REQUEST_TIMESTAMP_MILLIS, + { String: String(doNotShowRecoveryPageRequestTimestampMillis) }, + ], ]; const mockIdentityMetadata: IdentityMetadata = { recoveryPageShownTimestampMillis, + doNotShowRecoveryPageRequestTimestampMillis, }; const getterMockSuccess = vi.fn().mockResolvedValue(mockRawMetadata); @@ -66,12 +73,81 @@ test("IdentityMetadataRepository returns undefined without raising an error if f expect(console.warn).toHaveBeenCalledTimes(1); }); +test("IdentityMetadataRepository changes partial data in memory", async () => { + const instance = IdentityMetadataRepository.init({ + getter: getterMockSuccess, + setter: setterMockSuccess, + }); + + const newRecoveryPageShownTimestampMillis = 9876543210; + await instance.updateMetadata({ + recoveryPageShownTimestampMillis: newRecoveryPageShownTimestampMillis, + }); + + expect(await instance.getMetadata()).toEqual({ + recoveryPageShownTimestampMillis: newRecoveryPageShownTimestampMillis, + doNotShowRecoveryPageRequestTimestampMillis: + doNotShowRecoveryPageRequestTimestampMillis, + }); +}); + test("IdentityMetadataRepository changes data in memory", async () => { const instance = IdentityMetadataRepository.init({ getter: getterMockSuccess, setter: setterMockSuccess, }); + const newRecoveryPageShownTimestampMillis = 9876543210; + const newDoNotShowRecoveryPageRequestTimestampMillis = 1234567890; + await instance.updateMetadata({ + recoveryPageShownTimestampMillis: newRecoveryPageShownTimestampMillis, + doNotShowRecoveryPageRequestTimestampMillis: + newDoNotShowRecoveryPageRequestTimestampMillis, + }); + + expect(await instance.getMetadata()).toEqual({ + recoveryPageShownTimestampMillis: newRecoveryPageShownTimestampMillis, + doNotShowRecoveryPageRequestTimestampMillis: + newDoNotShowRecoveryPageRequestTimestampMillis, + }); +}); + +test("IdentityMetadataRepository sets data from partial data in memory", async () => { + const partialMetadata: MetadataMapV2 = [ + [ + RECOVERY_PAGE_SHOW_TIMESTAMP_MILLIS, + { String: String(recoveryPageShownTimestampMillis) }, + ], + ]; + const instance = IdentityMetadataRepository.init({ + getter: vi.fn().mockResolvedValue(partialMetadata), + setter: setterMockSuccess, + }); + + expect(await instance.getMetadata()).toEqual({ + recoveryPageShownTimestampMillis: recoveryPageShownTimestampMillis, + }); + + const newDoNotShowRecoveryPageRequestTimestampMillis = 1234567890; + await instance.updateMetadata({ + doNotShowRecoveryPageRequestTimestampMillis: + newDoNotShowRecoveryPageRequestTimestampMillis, + }); + + expect(await instance.getMetadata()).toEqual({ + recoveryPageShownTimestampMillis: recoveryPageShownTimestampMillis, + doNotShowRecoveryPageRequestTimestampMillis: + newDoNotShowRecoveryPageRequestTimestampMillis, + }); +}); + +test("IdentityMetadataRepository sets partial data in memory", async () => { + const noMetadata: MetadataMapV2 = []; + const instance = IdentityMetadataRepository.init({ + getter: vi.fn().mockResolvedValue(noMetadata), + setter: setterMockSuccess, + }); + const newRecoveryPageShownTimestampMillis = 9876543210; await instance.updateMetadata({ recoveryPageShownTimestampMillis: newRecoveryPageShownTimestampMillis, @@ -90,12 +166,17 @@ test("IdentityMetadataRepository sets data in memory", async () => { }); const newRecoveryPageShownTimestampMillis = 9876543210; + const newDoNotShowRecoveryPageRequestTimestampMillis = 1234567890; await instance.updateMetadata({ recoveryPageShownTimestampMillis: newRecoveryPageShownTimestampMillis, + doNotShowRecoveryPageRequestTimestampMillis: + newDoNotShowRecoveryPageRequestTimestampMillis, }); expect(await instance.getMetadata()).toEqual({ recoveryPageShownTimestampMillis: newRecoveryPageShownTimestampMillis, + doNotShowRecoveryPageRequestTimestampMillis: + newDoNotShowRecoveryPageRequestTimestampMillis, }); }); @@ -106,8 +187,11 @@ test("IdentityMetadataRepository commits updated metadata to canister", async () }); const newRecoveryPageShownTimestampMillis = 9876543210; + const newDoNotShowRecoveryPageRequestTimestampMillis = 1234567890; await instance.updateMetadata({ recoveryPageShownTimestampMillis: newRecoveryPageShownTimestampMillis, + doNotShowRecoveryPageRequestTimestampMillis: + newDoNotShowRecoveryPageRequestTimestampMillis, }); expect(setterMockSuccess).not.toHaveBeenCalled(); @@ -119,6 +203,10 @@ test("IdentityMetadataRepository commits updated metadata to canister", async () RECOVERY_PAGE_SHOW_TIMESTAMP_MILLIS, { String: String(newRecoveryPageShownTimestampMillis) }, ], + [ + DO_NOT_SHOW_RECOVERY_PAGE_REQUEST_TIMESTAMP_MILLIS, + { String: String(newDoNotShowRecoveryPageRequestTimestampMillis) }, + ], ]); }); @@ -141,8 +229,11 @@ test("IdentityMetadataRepository doesn't raise an error if committing fails", as }); const newRecoveryPageShownTimestampMillis = 9876543210; + const newDoNotShowRecoveryPageRequestTimestampMillis = 1234567890; const newMetadata = { recoveryPageShownTimestampMillis: newRecoveryPageShownTimestampMillis, + doNotShowRecoveryPageRequestTimestampMillis: + newDoNotShowRecoveryPageRequestTimestampMillis, }; await instance.updateMetadata(newMetadata); @@ -156,6 +247,10 @@ test("IdentityMetadataRepository doesn't raise an error if committing fails", as RECOVERY_PAGE_SHOW_TIMESTAMP_MILLIS, { String: String(newRecoveryPageShownTimestampMillis) }, ], + [ + DO_NOT_SHOW_RECOVERY_PAGE_REQUEST_TIMESTAMP_MILLIS, + { String: String(newDoNotShowRecoveryPageRequestTimestampMillis) }, + ], ]); // But the value in memory is not lost. @@ -190,10 +285,54 @@ test("IdentityMetadataRepository commits additional metadata to canister after u expect(setterMockSuccess).toHaveBeenCalledTimes(1); expect(setterMockSuccess).toHaveBeenCalledWith([ - anotherMetadataEntry, [ RECOVERY_PAGE_SHOW_TIMESTAMP_MILLIS, { String: String(newRecoveryPageShownTimestampMillis) }, ], + anotherMetadataEntry, + ]); +}); + +test("IdentityMetadataRepository commits from initial partial data after adding more partial data", async () => { + const partialMetadata: MetadataMapV2 = [ + [ + RECOVERY_PAGE_SHOW_TIMESTAMP_MILLIS, + { String: String(recoveryPageShownTimestampMillis) }, + ], + ]; + const instance = IdentityMetadataRepository.init({ + getter: vi.fn().mockResolvedValue(partialMetadata), + setter: setterMockSuccess, + }); + + expect(await instance.getMetadata()).toEqual({ + recoveryPageShownTimestampMillis: recoveryPageShownTimestampMillis, + }); + + const newDoNotShowRecoveryPageRequestTimestampMillis = 1234567890; + await instance.updateMetadata({ + doNotShowRecoveryPageRequestTimestampMillis: + newDoNotShowRecoveryPageRequestTimestampMillis, + }); + + expect(await instance.getMetadata()).toEqual({ + recoveryPageShownTimestampMillis: recoveryPageShownTimestampMillis, + doNotShowRecoveryPageRequestTimestampMillis: + newDoNotShowRecoveryPageRequestTimestampMillis, + }); + + expect(setterMockSuccess).not.toHaveBeenCalled(); + await instance.commitMetadata(); + + expect(setterMockSuccess).toHaveBeenCalledTimes(1); + expect(setterMockSuccess).toHaveBeenCalledWith([ + [ + RECOVERY_PAGE_SHOW_TIMESTAMP_MILLIS, + { String: String(recoveryPageShownTimestampMillis) }, + ], + [ + DO_NOT_SHOW_RECOVERY_PAGE_REQUEST_TIMESTAMP_MILLIS, + { String: String(newDoNotShowRecoveryPageRequestTimestampMillis) }, + ], ]); }); diff --git a/src/frontend/src/repositories/identityMetadata.ts b/src/frontend/src/repositories/identityMetadata.ts index 9bc40199e3..754824c5d7 100644 --- a/src/frontend/src/repositories/identityMetadata.ts +++ b/src/frontend/src/repositories/identityMetadata.ts @@ -1,30 +1,73 @@ import { MetadataMapV2 } from "$generated/internet_identity_types"; +import { isValidKey } from "$src/utils/utils"; export type IdentityMetadata = { recoveryPageShownTimestampMillis?: number; + doNotShowRecoveryPageRequestTimestampMillis?: number; }; export const RECOVERY_PAGE_SHOW_TIMESTAMP_MILLIS = "recoveryPageShownTimestampMillis"; +export const DO_NOT_SHOW_RECOVERY_PAGE_REQUEST_TIMESTAMP_MILLIS = + "doNotShowRecoveryPageRequestTimestampMillis"; +const NUMBER_KEYS: Array = [ + RECOVERY_PAGE_SHOW_TIMESTAMP_MILLIS, + DO_NOT_SHOW_RECOVERY_PAGE_REQUEST_TIMESTAMP_MILLIS, +]; const convertMetadata = (rawMetadata: MetadataMapV2): IdentityMetadata => { - const recoveryPageEntry = rawMetadata.find( - ([key]) => key === RECOVERY_PAGE_SHOW_TIMESTAMP_MILLIS - ); - if (recoveryPageEntry === undefined) { - return {}; - } - const stringValue = recoveryPageEntry[1]; - if ("String" in stringValue) { - const recoveryPageShownTimestampMillis = Number(stringValue.String); - if (isNaN(recoveryPageShownTimestampMillis)) { - return {}; + return rawMetadata.reduce((acc, [key, value]) => { + if (isValidKey(key, NUMBER_KEYS)) { + if (!("String" in value)) { + return acc; + } + const stringValue = value.String; + const numberValue = Number(stringValue); + if (!isNaN(numberValue)) { + // We need to cast the key because TS is not smart enough to + acc[key] = numberValue; + } } - return { - recoveryPageShownTimestampMillis, - }; - } - return {}; + return acc; + }, {} as IdentityMetadata); +}; + +const updateMetadataMapV2 = ({ + metadataMap, + partialIdentityMetadata, +}: { + metadataMap: MetadataMapV2; + partialIdentityMetadata: Partial; +}): MetadataMapV2 => { + // Convert the partialIdentityMetadata into the format of MetadataMapV2 + const identityMetadataEntries: MetadataMapV2 = Object.entries( + partialIdentityMetadata + ).map(([key, value]) => { + if (typeof value === "number") { + return [key, { String: value.toString() }] as [ + string, + { String: string } + ]; + } + return [key, { String: value as string }] as [string, { String: string }]; + }); + + // Update or add entries in metadataMap + const updatedMetadataMap: MetadataMapV2 = metadataMap.map(([key, value]) => { + const updatedEntry = identityMetadataEntries.find( + ([identityKey]) => identityKey === key + ); + return updatedEntry ? updatedEntry : [key, value]; + }); + + // Add new entries that were not in the original metadataMap + identityMetadataEntries.forEach(([identityKey, identityValue]) => { + if (!metadataMap.some(([key]) => key === identityKey)) { + updatedMetadataMap.push([identityKey, identityValue]); + } + }); + + return updatedMetadataMap; }; type MetadataGetter = () => Promise; @@ -116,8 +159,8 @@ export class IdentityMetadataRepository { let currentWait = 0; const MAX_WAIT_MILLIS = 10_000; const ONE_WAIT_MILLIS = 1_000; - while (this.rawMetadata === "loading" || currentWait < MAX_WAIT_MILLIS) { - await new Promise((resolve) => setTimeout(resolve, 100)); + while (this.rawMetadata === "loading" && currentWait < MAX_WAIT_MILLIS) { + await new Promise((resolve) => setTimeout(resolve, ONE_WAIT_MILLIS)); currentWait += ONE_WAIT_MILLIS; } }; @@ -155,19 +198,11 @@ export class IdentityMetadataRepository { ): Promise => { await this.waitUntilMetadataIsLoaded(); if (this.metadataIsLoaded(this.rawMetadata)) { - let updatedMetadata: MetadataMapV2 = [...this.rawMetadata]; this.updatedMetadata = true; - updatedMetadata = updatedMetadata - .filter(([key]) => key !== RECOVERY_PAGE_SHOW_TIMESTAMP_MILLIS) - .concat([ - [ - RECOVERY_PAGE_SHOW_TIMESTAMP_MILLIS, - { - String: String(partialMetadata.recoveryPageShownTimestampMillis), - }, - ], - ]); - this.rawMetadata = updatedMetadata; + this.rawMetadata = updateMetadataMapV2({ + metadataMap: this.rawMetadata, + partialIdentityMetadata: partialMetadata, + }); } // Do nothing if the metadata is not loaded. }; diff --git a/src/frontend/src/utils/utils.ts b/src/frontend/src/utils/utils.ts index 7351193d0a..95d6e21f00 100644 --- a/src/frontend/src/utils/utils.ts +++ b/src/frontend/src/utils/utils.ts @@ -353,3 +353,10 @@ export type OmitParams any, A extends string> = ( // Zip two arrays together export const zip = (a: A[], b: B[]): [A, B][] => Array.from(Array(Math.min(b.length, a.length)), (_, i) => [a[i], b[i]]); + +export const isValidKey = ( + key: string | number | symbol, + keys: Array +): key is keyof T => { + return keys.includes(key as keyof T); +}; diff --git a/src/showcase/src/pages/addDeviceWarningOnePasskey.astro b/src/showcase/src/pages/addDeviceWarningOnePasskey.astro index 6079f1d049..0936d519de 100644 --- a/src/showcase/src/pages/addDeviceWarningOnePasskey.astro +++ b/src/showcase/src/pages/addDeviceWarningOnePasskey.astro @@ -13,7 +13,7 @@ import Screen from "../layouts/Screen.astro"; remindLater: () => toast.info("Remind later"), doNotRemindAgain: () => toast.info("Do not remind again"), i18n, - status: "one-passkey", + status: "one-device", });