Factor out crypto setup process into a store
To make components pure and avoid react 18 dev mode problems due to components making requests when mounted.
This commit is contained in:
parent
c659afa8db
commit
e7e7331558
6 changed files with 165 additions and 61 deletions
2
src/@types/global.d.ts
vendored
2
src/@types/global.d.ts
vendored
|
@ -44,6 +44,7 @@ import { IConfigOptions } from "../IConfigOptions";
|
||||||
import { MatrixDispatcher } from "../dispatcher/dispatcher";
|
import { MatrixDispatcher } from "../dispatcher/dispatcher";
|
||||||
import { DeepReadonly } from "./common";
|
import { DeepReadonly } from "./common";
|
||||||
import MatrixChat from "../components/structures/MatrixChat";
|
import MatrixChat from "../components/structures/MatrixChat";
|
||||||
|
import { InitialCryptoSetupStore } from "../stores/InitialCryptoSetupStore";
|
||||||
|
|
||||||
/* eslint-disable @typescript-eslint/naming-convention */
|
/* eslint-disable @typescript-eslint/naming-convention */
|
||||||
|
|
||||||
|
@ -117,6 +118,7 @@ declare global {
|
||||||
mxPerformanceEntryNames: any;
|
mxPerformanceEntryNames: any;
|
||||||
mxUIStore: UIStore;
|
mxUIStore: UIStore;
|
||||||
mxSetupEncryptionStore?: SetupEncryptionStore;
|
mxSetupEncryptionStore?: SetupEncryptionStore;
|
||||||
|
mxInitialCryptoStore?: InitialCryptoSetupStore;
|
||||||
mxRoomScrollStateStore?: RoomScrollStateStore;
|
mxRoomScrollStateStore?: RoomScrollStateStore;
|
||||||
mxActiveWidgetStore?: ActiveWidgetStore;
|
mxActiveWidgetStore?: ActiveWidgetStore;
|
||||||
mxOnRecaptchaLoaded?: () => void;
|
mxOnRecaptchaLoaded?: () => void;
|
||||||
|
|
|
@ -132,6 +132,7 @@ import { SessionLockStolenView } from "./auth/SessionLockStolenView";
|
||||||
import { ConfirmSessionLockTheftView } from "./auth/ConfirmSessionLockTheftView";
|
import { ConfirmSessionLockTheftView } from "./auth/ConfirmSessionLockTheftView";
|
||||||
import { LoginSplashView } from "./auth/LoginSplashView";
|
import { LoginSplashView } from "./auth/LoginSplashView";
|
||||||
import { cleanUpDraftsIfRequired } from "../../DraftCleaner";
|
import { cleanUpDraftsIfRequired } from "../../DraftCleaner";
|
||||||
|
import { InitialCryptoSetupStore } from "../../stores/InitialCryptoSetupStore";
|
||||||
|
|
||||||
// legacy export
|
// legacy export
|
||||||
export { default as Views } from "../../Views";
|
export { default as Views } from "../../Views";
|
||||||
|
@ -428,6 +429,12 @@ export default class MatrixChat extends React.PureComponent<IProps, IState> {
|
||||||
!(await shouldSkipSetupEncryption(cli))
|
!(await shouldSkipSetupEncryption(cli))
|
||||||
) {
|
) {
|
||||||
// if cross-signing is not yet set up, do so now if possible.
|
// if cross-signing is not yet set up, do so now if possible.
|
||||||
|
InitialCryptoSetupStore.sharedInstance().startInitialCryptoSetup(
|
||||||
|
cli,
|
||||||
|
Boolean(this.tokenLogin),
|
||||||
|
this.stores,
|
||||||
|
this.onCompleteSecurityE2eSetupFinished,
|
||||||
|
);
|
||||||
this.setStateForNewView({ view: Views.E2E_SETUP });
|
this.setStateForNewView({ view: Views.E2E_SETUP });
|
||||||
} else {
|
} else {
|
||||||
this.onLoggedIn();
|
this.onLoggedIn();
|
||||||
|
@ -2073,14 +2080,7 @@ export default class MatrixChat extends React.PureComponent<IProps, IState> {
|
||||||
} else if (this.state.view === Views.COMPLETE_SECURITY) {
|
} else if (this.state.view === Views.COMPLETE_SECURITY) {
|
||||||
view = <CompleteSecurity onFinished={this.onCompleteSecurityE2eSetupFinished} />;
|
view = <CompleteSecurity onFinished={this.onCompleteSecurityE2eSetupFinished} />;
|
||||||
} else if (this.state.view === Views.E2E_SETUP) {
|
} else if (this.state.view === Views.E2E_SETUP) {
|
||||||
view = (
|
view = <E2eSetup onFinished={this.onCompleteSecurityE2eSetupFinished} />;
|
||||||
<E2eSetup
|
|
||||||
matrixClient={MatrixClientPeg.safeGet()}
|
|
||||||
onFinished={this.onCompleteSecurityE2eSetupFinished}
|
|
||||||
accountPassword={this.stores.accountPasswordStore.getPassword()}
|
|
||||||
tokenLogin={!!this.tokenLogin}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
} else if (this.state.view === Views.LOGGED_IN) {
|
} else if (this.state.view === Views.LOGGED_IN) {
|
||||||
// `ready` and `view==LOGGED_IN` may be set before `page_type` (because the
|
// `ready` and `view==LOGGED_IN` may be set before `page_type` (because the
|
||||||
// latter is set via the dispatcher). If we don't yet have a `page_type`,
|
// latter is set via the dispatcher). If we don't yet have a `page_type`,
|
||||||
|
|
|
@ -7,17 +7,13 @@ Please see LICENSE files in the repository root for full details.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import React from "react";
|
import React from "react";
|
||||||
import { MatrixClient } from "matrix-js-sdk/src/matrix";
|
|
||||||
|
|
||||||
import AuthPage from "../../views/auth/AuthPage";
|
import AuthPage from "../../views/auth/AuthPage";
|
||||||
import CompleteSecurityBody from "../../views/auth/CompleteSecurityBody";
|
import CompleteSecurityBody from "../../views/auth/CompleteSecurityBody";
|
||||||
import { InitialCryptoSetupDialog } from "../../views/dialogs/security/InitialCryptoSetupDialog";
|
import { InitialCryptoSetupDialog } from "../../views/dialogs/security/InitialCryptoSetupDialog";
|
||||||
|
|
||||||
interface IProps {
|
interface IProps {
|
||||||
matrixClient: MatrixClient;
|
|
||||||
onFinished: () => void;
|
onFinished: () => void;
|
||||||
accountPassword?: string;
|
|
||||||
tokenLogin: boolean;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export default class E2eSetup extends React.Component<IProps> {
|
export default class E2eSetup extends React.Component<IProps> {
|
||||||
|
@ -25,12 +21,7 @@ export default class E2eSetup extends React.Component<IProps> {
|
||||||
return (
|
return (
|
||||||
<AuthPage>
|
<AuthPage>
|
||||||
<CompleteSecurityBody>
|
<CompleteSecurityBody>
|
||||||
<InitialCryptoSetupDialog
|
<InitialCryptoSetupDialog onFinished={this.props.onFinished} />
|
||||||
matrixClient={this.props.matrixClient}
|
|
||||||
onFinished={this.props.onFinished}
|
|
||||||
accountPassword={this.props.accountPassword}
|
|
||||||
tokenLogin={this.props.tokenLogin}
|
|
||||||
/>
|
|
||||||
</CompleteSecurityBody>
|
</CompleteSecurityBody>
|
||||||
</AuthPage>
|
</AuthPage>
|
||||||
);
|
);
|
||||||
|
|
|
@ -7,20 +7,15 @@ SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
|
||||||
Please see LICENSE files in the repository root for full details.
|
Please see LICENSE files in the repository root for full details.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import React, { useCallback, useEffect, useState } from "react";
|
import React, { useCallback } from "react";
|
||||||
import { logger } from "matrix-js-sdk/src/logger";
|
|
||||||
import { MatrixClient } from "matrix-js-sdk/src/matrix";
|
|
||||||
|
|
||||||
import { _t } from "../../../../languageHandler";
|
import { _t } from "../../../../languageHandler";
|
||||||
import DialogButtons from "../../elements/DialogButtons";
|
import DialogButtons from "../../elements/DialogButtons";
|
||||||
import BaseDialog from "../BaseDialog";
|
import BaseDialog from "../BaseDialog";
|
||||||
import Spinner from "../../elements/Spinner";
|
import Spinner from "../../elements/Spinner";
|
||||||
import { createCrossSigning } from "../../../../CreateCrossSigning";
|
import { InitialCryptoSetupStore, useInitialCryptoSetupStatus } from "../../../../stores/InitialCryptoSetupStore";
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
matrixClient: MatrixClient;
|
|
||||||
accountPassword?: string;
|
|
||||||
tokenLogin: boolean;
|
|
||||||
onFinished: (success?: boolean) => void;
|
onFinished: (success?: boolean) => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -29,54 +24,27 @@ interface Props {
|
||||||
* In most cases, only a spinner is shown, but for more
|
* In most cases, only a spinner is shown, but for more
|
||||||
* complex auth like SSO, the user may need to complete some steps to proceed.
|
* complex auth like SSO, the user may need to complete some steps to proceed.
|
||||||
*/
|
*/
|
||||||
export const InitialCryptoSetupDialog: React.FC<Props> = ({
|
export const InitialCryptoSetupDialog: React.FC<Props> = ({ onFinished }) => {
|
||||||
matrixClient,
|
const onRetryClick = useCallback(() => {
|
||||||
accountPassword,
|
InitialCryptoSetupStore.sharedInstance().retry();
|
||||||
tokenLogin,
|
}, []);
|
||||||
onFinished,
|
|
||||||
}) => {
|
|
||||||
const [error, setError] = useState(false);
|
|
||||||
|
|
||||||
const doSetup = useCallback(async () => {
|
const onCancelClick = useCallback(() => {
|
||||||
const cryptoApi = matrixClient.getCrypto();
|
|
||||||
if (!cryptoApi) return;
|
|
||||||
|
|
||||||
setError(false);
|
|
||||||
|
|
||||||
try {
|
|
||||||
await createCrossSigning(matrixClient, tokenLogin, accountPassword);
|
|
||||||
|
|
||||||
onFinished(true);
|
|
||||||
} catch (e) {
|
|
||||||
if (tokenLogin) {
|
|
||||||
// ignore any failures, we are relying on grace period here
|
|
||||||
onFinished(false);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
setError(true);
|
|
||||||
logger.error("Error bootstrapping cross-signing", e);
|
|
||||||
}
|
|
||||||
}, [matrixClient, tokenLogin, accountPassword, onFinished]);
|
|
||||||
|
|
||||||
const onCancel = useCallback(() => {
|
|
||||||
onFinished(false);
|
onFinished(false);
|
||||||
}, [onFinished]);
|
}, [onFinished]);
|
||||||
|
|
||||||
useEffect(() => {
|
const status = useInitialCryptoSetupStatus(InitialCryptoSetupStore.sharedInstance());
|
||||||
doSetup();
|
|
||||||
}, [doSetup]);
|
|
||||||
|
|
||||||
let content;
|
let content;
|
||||||
if (error) {
|
if (status === "error") {
|
||||||
content = (
|
content = (
|
||||||
<div>
|
<div>
|
||||||
<p>{_t("encryption|unable_to_setup_keys_error")}</p>
|
<p>{_t("encryption|unable_to_setup_keys_error")}</p>
|
||||||
<div className="mx_Dialog_buttons">
|
<div className="mx_Dialog_buttons">
|
||||||
<DialogButtons
|
<DialogButtons
|
||||||
primaryButton={_t("action|retry")}
|
primaryButton={_t("action|retry")}
|
||||||
onPrimaryButtonClick={doSetup}
|
onPrimaryButtonClick={onRetryClick}
|
||||||
onCancel={onCancel}
|
onCancel={onCancelClick}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
138
src/stores/InitialCryptoSetupStore.ts
Normal file
138
src/stores/InitialCryptoSetupStore.ts
Normal file
|
@ -0,0 +1,138 @@
|
||||||
|
/*
|
||||||
|
Copyright 2024 New Vector Ltd.
|
||||||
|
|
||||||
|
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
|
||||||
|
Please see LICENSE files in the repository root for full details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import EventEmitter from "events";
|
||||||
|
import { MatrixClient } from "matrix-js-sdk/src/matrix";
|
||||||
|
import { logger } from "matrix-js-sdk/src/logger";
|
||||||
|
import { useEffect, useState } from "react";
|
||||||
|
|
||||||
|
import { createCrossSigning } from "../CreateCrossSigning";
|
||||||
|
import { SdkContextClass } from "../contexts/SDKContext";
|
||||||
|
|
||||||
|
type Status = "in_progress" | "complete" | "error" | undefined;
|
||||||
|
|
||||||
|
export const useInitialCryptoSetupStatus = (store: InitialCryptoSetupStore): Status => {
|
||||||
|
const [status, setStatus] = useState<Status>(store.getStatus());
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const update = (): void => {
|
||||||
|
setStatus(store.getStatus());
|
||||||
|
};
|
||||||
|
|
||||||
|
store.on("update", update);
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
store.off("update", update);
|
||||||
|
};
|
||||||
|
}, [store]);
|
||||||
|
|
||||||
|
return status;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Logic for setting up crypto state that's done immediately after
|
||||||
|
* a user registers. Should be transparent to the user, not requiring
|
||||||
|
* interaction in most cases.
|
||||||
|
* As distinct from SetupEncryptionStore which is for setting up
|
||||||
|
* 4S or verifying the device, will always require interaction
|
||||||
|
* from the user in some form.
|
||||||
|
*/
|
||||||
|
export class InitialCryptoSetupStore extends EventEmitter {
|
||||||
|
private status: Status = undefined;
|
||||||
|
|
||||||
|
private client?: MatrixClient;
|
||||||
|
private isTokenLogin?: boolean;
|
||||||
|
private stores?: SdkContextClass;
|
||||||
|
private onFinished?: (success: boolean) => void;
|
||||||
|
|
||||||
|
public static sharedInstance(): InitialCryptoSetupStore {
|
||||||
|
if (!window.mxInitialCryptoStore) window.mxInitialCryptoStore = new InitialCryptoSetupStore();
|
||||||
|
return window.mxInitialCryptoStore;
|
||||||
|
}
|
||||||
|
|
||||||
|
public getStatus(): Status {
|
||||||
|
return this.status;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Start the initial crypto setup process.
|
||||||
|
*
|
||||||
|
* @param {MatrixClient} client The client to use for the setup
|
||||||
|
* @param {boolean} isTokenLogin True if the user logged in via a token login, otherwise false
|
||||||
|
* @param {SdkContextClass} stores The stores to use for the setup
|
||||||
|
*/
|
||||||
|
public startInitialCryptoSetup(
|
||||||
|
client: MatrixClient,
|
||||||
|
isTokenLogin: boolean,
|
||||||
|
stores: SdkContextClass,
|
||||||
|
onFinished: (success: boolean) => void,
|
||||||
|
): void {
|
||||||
|
this.client = client;
|
||||||
|
this.isTokenLogin = isTokenLogin;
|
||||||
|
this.stores = stores;
|
||||||
|
this.onFinished = onFinished;
|
||||||
|
|
||||||
|
this.doSetup().catch(() => logger.error("Initial crypto setup failed"));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retry the initial crypto setup process.
|
||||||
|
*
|
||||||
|
* If no crypto setup is currently in process, this will return false.
|
||||||
|
*
|
||||||
|
* @returns {boolean} True if a retry was initiated, otherwise false
|
||||||
|
*/
|
||||||
|
public retry(): boolean {
|
||||||
|
if (this.client === undefined || this.isTokenLogin === undefined || this.stores == undefined) return false;
|
||||||
|
|
||||||
|
this.doSetup().catch(() => logger.error("Initial crypto setup failed"));
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
private reset(): void {
|
||||||
|
this.client = undefined;
|
||||||
|
this.isTokenLogin = undefined;
|
||||||
|
this.stores = undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
private async doSetup(): Promise<void> {
|
||||||
|
if (this.client === undefined || this.isTokenLogin === undefined || this.stores == undefined) {
|
||||||
|
throw new Error("No setup is in progress");
|
||||||
|
}
|
||||||
|
|
||||||
|
const cryptoApi = this.client.getCrypto();
|
||||||
|
if (!cryptoApi) throw new Error("No crypto module found!");
|
||||||
|
|
||||||
|
this.status = "in_progress";
|
||||||
|
this.emit("update");
|
||||||
|
|
||||||
|
try {
|
||||||
|
await createCrossSigning(this.client, this.isTokenLogin, this.stores.accountPasswordStore.getPassword());
|
||||||
|
|
||||||
|
this.reset();
|
||||||
|
|
||||||
|
this.status = "complete";
|
||||||
|
this.emit("update");
|
||||||
|
this.onFinished?.(true);
|
||||||
|
} catch (e) {
|
||||||
|
if (this.isTokenLogin) {
|
||||||
|
// ignore any failures, we are relying on grace period here
|
||||||
|
this.reset();
|
||||||
|
|
||||||
|
this.status = "complete";
|
||||||
|
this.emit("update");
|
||||||
|
this.onFinished?.(true);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
logger.error("Error bootstrapping cross-signing", e);
|
||||||
|
this.status = "error";
|
||||||
|
this.emit("update");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -33,6 +33,11 @@ export enum Phase {
|
||||||
ConfirmReset = 6,
|
ConfirmReset = 6,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Logic for setting up 4S and/or verifying the user's device: a process requiring
|
||||||
|
* ongoing interaction with the user, as distinct from InitialCryptoSetupStore which
|
||||||
|
* a (usually) non-interactive process that happens immediately after registration.
|
||||||
|
*/
|
||||||
export class SetupEncryptionStore extends EventEmitter {
|
export class SetupEncryptionStore extends EventEmitter {
|
||||||
private started?: boolean;
|
private started?: boolean;
|
||||||
public phase?: Phase;
|
public phase?: Phase;
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue