2021-07-07 10:08:53 +00:00
|
|
|
/*
|
|
|
|
Copyright 2017 - 2021 The Matrix.org Foundation C.I.C.
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2023-04-03 08:26:55 +00:00
|
|
|
import React, { createRef } from "react";
|
2021-07-07 10:08:53 +00:00
|
|
|
import {
|
|
|
|
AuthType,
|
|
|
|
IAuthData,
|
|
|
|
IAuthDict,
|
|
|
|
IInputs,
|
|
|
|
InteractiveAuth,
|
|
|
|
IStageStatus,
|
|
|
|
} from "matrix-js-sdk/src/interactive-auth";
|
|
|
|
import { MatrixClient } from "matrix-js-sdk/src/client";
|
2021-10-22 22:23:32 +00:00
|
|
|
import { logger } from "matrix-js-sdk/src/logger";
|
2021-07-07 10:08:53 +00:00
|
|
|
|
|
|
|
import getEntryComponentForLoginType, { IStageComponent } from "../views/auth/InteractiveAuthEntryComponents";
|
|
|
|
import Spinner from "../views/elements/Spinner";
|
|
|
|
|
|
|
|
export const ERROR_USER_CANCELLED = new Error("User cancelled auth session");
|
|
|
|
|
2023-07-04 13:49:27 +00:00
|
|
|
type InteractiveAuthCallbackSuccess<T> = (
|
2022-05-20 17:14:17 +00:00
|
|
|
success: true,
|
2023-07-04 13:49:27 +00:00
|
|
|
response: T,
|
2022-05-20 17:14:17 +00:00
|
|
|
extra?: { emailSid?: string; clientSecret?: string },
|
|
|
|
) => void;
|
|
|
|
type InteractiveAuthCallbackFailure = (success: false, response: IAuthData | Error) => void;
|
2023-07-04 13:49:27 +00:00
|
|
|
export type InteractiveAuthCallback<T> = InteractiveAuthCallbackSuccess<T> & InteractiveAuthCallbackFailure;
|
2022-05-20 17:14:17 +00:00
|
|
|
|
2023-04-03 08:26:55 +00:00
|
|
|
export interface InteractiveAuthProps<T> {
|
2021-07-07 10:08:53 +00:00
|
|
|
// matrix client to use for UI auth requests
|
|
|
|
matrixClient: MatrixClient;
|
|
|
|
// response from initial request. If not supplied, will do a request on mount.
|
2021-08-10 10:03:03 +00:00
|
|
|
authData?: IAuthData;
|
2021-07-07 10:08:53 +00:00
|
|
|
// Inputs provided by the user to the auth process
|
|
|
|
// and used by various stages. As passed to js-sdk
|
|
|
|
// interactive-auth
|
|
|
|
inputs?: IInputs;
|
|
|
|
sessionId?: string;
|
|
|
|
clientSecret?: string;
|
|
|
|
emailSid?: string;
|
|
|
|
// If true, poll to see if the auth flow has been completed out-of-band
|
|
|
|
poll?: boolean;
|
|
|
|
// If true, components will be told that the 'Continue' button
|
|
|
|
// is managed by some other party and should not be managed by
|
|
|
|
// the component itself.
|
|
|
|
continueIsManaged?: boolean;
|
|
|
|
// continueText and continueKind are passed straight through to the AuthEntryComponent.
|
|
|
|
continueText?: string;
|
|
|
|
continueKind?: string;
|
|
|
|
// callback
|
2023-07-04 13:49:27 +00:00
|
|
|
makeRequest(auth: IAuthDict | null): Promise<T>;
|
2021-07-07 10:08:53 +00:00
|
|
|
// callback called when the auth process has finished,
|
|
|
|
// successfully or unsuccessfully.
|
|
|
|
// @param {boolean} status True if the operation requiring
|
|
|
|
// auth was completed successfully, false if canceled.
|
|
|
|
// @param {object} result The result of the authenticated call
|
|
|
|
// if successful, otherwise the error object.
|
|
|
|
// @param {object} extra Additional information about the UI Auth
|
|
|
|
// process:
|
|
|
|
// * emailSid {string} If email auth was performed, the sid of
|
|
|
|
// the auth session.
|
|
|
|
// * clientSecret {string} The client secret used in auth
|
|
|
|
// sessions with the ID server.
|
2023-07-04 13:49:27 +00:00
|
|
|
onAuthFinished: InteractiveAuthCallback<T>;
|
2021-07-07 10:08:53 +00:00
|
|
|
// As js-sdk interactive-auth
|
2021-08-10 10:03:03 +00:00
|
|
|
requestEmailToken?(email: string, secret: string, attempt: number, session: string): Promise<{ sid: string }>;
|
2021-07-07 10:08:53 +00:00
|
|
|
// Called when the stage changes, or the stage's phase changes. First
|
|
|
|
// argument is the stage, second is the phase. Some stages do not have
|
|
|
|
// phases and will be counted as 0 (numeric).
|
2023-03-07 10:45:55 +00:00
|
|
|
onStagePhaseChange?(stage: AuthType | null, phase: number): void;
|
2021-07-07 10:08:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
interface IState {
|
|
|
|
authStage?: AuthType;
|
|
|
|
stageState?: IStageStatus;
|
|
|
|
busy: boolean;
|
|
|
|
errorText?: string;
|
2021-10-11 13:43:55 +00:00
|
|
|
errorCode?: string;
|
2021-07-07 10:08:53 +00:00
|
|
|
submitButtonEnabled: boolean;
|
|
|
|
}
|
|
|
|
|
2023-04-03 08:26:55 +00:00
|
|
|
export default class InteractiveAuthComponent<T> extends React.Component<InteractiveAuthProps<T>, IState> {
|
2023-07-04 13:49:27 +00:00
|
|
|
private readonly authLogic: InteractiveAuth<T>;
|
2023-02-24 15:28:40 +00:00
|
|
|
private readonly intervalId: number | null = null;
|
2021-07-07 10:08:53 +00:00
|
|
|
private readonly stageComponent = createRef<IStageComponent>();
|
|
|
|
|
|
|
|
private unmounted = false;
|
|
|
|
|
2023-04-03 08:26:55 +00:00
|
|
|
public constructor(props: InteractiveAuthProps<T>) {
|
2021-07-07 10:08:53 +00:00
|
|
|
super(props);
|
|
|
|
|
|
|
|
this.state = {
|
|
|
|
busy: false,
|
|
|
|
submitButtonEnabled: false,
|
|
|
|
};
|
|
|
|
|
2023-07-04 13:49:27 +00:00
|
|
|
this.authLogic = new InteractiveAuth<T>({
|
2021-07-07 10:08:53 +00:00
|
|
|
authData: this.props.authData,
|
|
|
|
doRequest: this.requestCallback,
|
|
|
|
busyChanged: this.onBusyChanged,
|
|
|
|
inputs: this.props.inputs,
|
|
|
|
stateUpdated: this.authStateUpdated,
|
|
|
|
matrixClient: this.props.matrixClient,
|
|
|
|
sessionId: this.props.sessionId,
|
|
|
|
clientSecret: this.props.clientSecret,
|
|
|
|
emailSid: this.props.emailSid,
|
|
|
|
requestEmailToken: this.requestEmailToken,
|
2023-05-24 11:02:32 +00:00
|
|
|
supportedStages: [
|
|
|
|
AuthType.Password,
|
|
|
|
AuthType.Recaptcha,
|
|
|
|
AuthType.Email,
|
|
|
|
AuthType.Msisdn,
|
|
|
|
AuthType.Terms,
|
|
|
|
AuthType.RegistrationToken,
|
|
|
|
AuthType.UnstableRegistrationToken,
|
|
|
|
AuthType.Sso,
|
|
|
|
AuthType.SsoUnstable,
|
|
|
|
],
|
2021-07-07 10:08:53 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
if (this.props.poll) {
|
2022-11-30 11:32:56 +00:00
|
|
|
this.intervalId = window.setInterval(() => {
|
2021-07-07 10:08:53 +00:00
|
|
|
this.authLogic.poll();
|
|
|
|
}, 2000);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-12 13:25:14 +00:00
|
|
|
public componentDidMount(): void {
|
2021-07-07 10:08:53 +00:00
|
|
|
this.authLogic
|
|
|
|
.attemptAuth()
|
|
|
|
.then((result) => {
|
|
|
|
const extra = {
|
|
|
|
emailSid: this.authLogic.getEmailSid(),
|
|
|
|
clientSecret: this.authLogic.getClientSecret(),
|
|
|
|
};
|
|
|
|
this.props.onAuthFinished(true, result, extra);
|
|
|
|
})
|
|
|
|
.catch((error) => {
|
|
|
|
this.props.onAuthFinished(false, error);
|
2021-10-15 14:30:53 +00:00
|
|
|
logger.error("Error during user-interactive auth:", error);
|
2021-07-07 10:08:53 +00:00
|
|
|
if (this.unmounted) {
|
|
|
|
return;
|
|
|
|
}
|
2022-12-12 11:24:14 +00:00
|
|
|
|
2021-07-07 10:08:53 +00:00
|
|
|
const msg = error.message || error.toString();
|
|
|
|
this.setState({
|
|
|
|
errorText: msg,
|
2021-10-11 13:43:55 +00:00
|
|
|
errorCode: error.errcode,
|
2022-12-12 11:24:14 +00:00
|
|
|
});
|
2021-07-07 10:08:53 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-01-12 13:25:14 +00:00
|
|
|
public componentWillUnmount(): void {
|
2021-07-07 10:08:53 +00:00
|
|
|
this.unmounted = true;
|
|
|
|
|
2021-08-10 10:03:03 +00:00
|
|
|
if (this.intervalId !== null) {
|
|
|
|
clearInterval(this.intervalId);
|
2021-07-07 10:08:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private requestEmailToken = async (
|
|
|
|
email: string,
|
|
|
|
secret: string,
|
|
|
|
attempt: number,
|
|
|
|
session: string,
|
|
|
|
): Promise<{ sid: string }> => {
|
|
|
|
this.setState({
|
|
|
|
busy: true,
|
|
|
|
});
|
|
|
|
try {
|
2023-03-07 10:45:55 +00:00
|
|
|
// We know this method only gets called on flows where requestEmailToken is passed but types don't
|
|
|
|
return await this.props.requestEmailToken!(email, secret, attempt, session);
|
2021-07-07 10:08:53 +00:00
|
|
|
} finally {
|
|
|
|
this.setState({
|
|
|
|
busy: false,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
private authStateUpdated = (stageType: AuthType, stageState: IStageStatus): void => {
|
|
|
|
const oldStage = this.state.authStage;
|
|
|
|
this.setState(
|
|
|
|
{
|
|
|
|
busy: false,
|
|
|
|
authStage: stageType,
|
|
|
|
stageState: stageState,
|
|
|
|
errorText: stageState.error,
|
2021-10-11 13:43:55 +00:00
|
|
|
errorCode: stageState.errcode,
|
2021-07-07 10:08:53 +00:00
|
|
|
},
|
|
|
|
() => {
|
|
|
|
if (oldStage !== stageType) {
|
|
|
|
this.setFocus();
|
|
|
|
} else if (!stageState.error) {
|
|
|
|
this.stageComponent.current?.attemptFailed?.();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
2023-07-04 13:49:27 +00:00
|
|
|
private requestCallback = (auth: IAuthDict | null, background: boolean): Promise<T> => {
|
2021-07-07 10:08:53 +00:00
|
|
|
// This wrapper just exists because the js-sdk passes a second
|
|
|
|
// 'busy' param for backwards compat. This throws the tests off
|
|
|
|
// so discard it here.
|
|
|
|
return this.props.makeRequest(auth);
|
|
|
|
};
|
|
|
|
|
|
|
|
private onBusyChanged = (busy: boolean): void => {
|
|
|
|
// if we've started doing stuff, reset the error messages
|
2023-05-07 21:12:45 +00:00
|
|
|
// The JS SDK eagerly reports itself as "not busy" right after any
|
|
|
|
// immediate work has completed, but that's not really what we want at
|
|
|
|
// the UI layer, so we ignore this signal and show a spinner until
|
|
|
|
// there's a new screen to show the user. This is implemented by setting
|
|
|
|
// `busy: false` in `authStateUpdated`.
|
|
|
|
// See also https://github.com/vector-im/element-web/issues/12546
|
2021-07-07 10:08:53 +00:00
|
|
|
if (busy) {
|
|
|
|
this.setState({
|
|
|
|
busy: true,
|
2023-02-24 15:28:40 +00:00
|
|
|
errorText: undefined,
|
|
|
|
errorCode: undefined,
|
2021-07-07 10:08:53 +00:00
|
|
|
});
|
|
|
|
}
|
2023-05-07 21:12:45 +00:00
|
|
|
|
|
|
|
// authStateUpdated is not called during sso flows
|
|
|
|
if (!busy && (this.state.authStage === AuthType.Sso || this.state.authStage === AuthType.SsoUnstable)) {
|
|
|
|
this.setState({ busy });
|
|
|
|
}
|
2021-07-07 10:08:53 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
private setFocus(): void {
|
|
|
|
this.stageComponent.current?.focus?.();
|
|
|
|
}
|
|
|
|
|
|
|
|
private submitAuthDict = (authData: IAuthDict): void => {
|
|
|
|
this.authLogic.submitAuthDict(authData);
|
|
|
|
};
|
|
|
|
|
|
|
|
private onPhaseChange = (newPhase: number): void => {
|
2023-03-07 10:45:55 +00:00
|
|
|
this.props.onStagePhaseChange?.(this.state.authStage ?? null, newPhase || 0);
|
2021-07-07 10:08:53 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
private onStageCancel = (): void => {
|
|
|
|
this.props.onAuthFinished(false, ERROR_USER_CANCELLED);
|
|
|
|
};
|
|
|
|
|
2021-10-11 13:43:55 +00:00
|
|
|
private onAuthStageFailed = (e: Error): void => {
|
|
|
|
this.props.onAuthFinished(false, e);
|
|
|
|
};
|
|
|
|
|
|
|
|
private setEmailSid = (sid: string): void => {
|
|
|
|
this.authLogic.setEmailSid(sid);
|
|
|
|
};
|
|
|
|
|
2023-02-13 17:01:43 +00:00
|
|
|
public render(): React.ReactNode {
|
2021-07-07 10:08:53 +00:00
|
|
|
const stage = this.state.authStage;
|
|
|
|
if (!stage) {
|
|
|
|
if (this.state.busy) {
|
|
|
|
return <Spinner />;
|
|
|
|
} else {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const StageComponent = getEntryComponentForLoginType(stage);
|
|
|
|
return (
|
|
|
|
<StageComponent
|
|
|
|
ref={this.stageComponent as any}
|
|
|
|
loginType={stage}
|
|
|
|
matrixClient={this.props.matrixClient}
|
|
|
|
authSessionId={this.authLogic.getSessionId()}
|
|
|
|
clientSecret={this.authLogic.getClientSecret()}
|
|
|
|
stageParams={this.authLogic.getStageParams(stage)}
|
|
|
|
submitAuthDict={this.submitAuthDict}
|
2021-10-11 13:43:55 +00:00
|
|
|
errorText={this.state.errorText}
|
|
|
|
errorCode={this.state.errorCode}
|
2021-07-07 10:08:53 +00:00
|
|
|
busy={this.state.busy}
|
|
|
|
inputs={this.props.inputs}
|
|
|
|
stageState={this.state.stageState}
|
|
|
|
fail={this.onAuthStageFailed}
|
|
|
|
setEmailSid={this.setEmailSid}
|
|
|
|
showContinue={!this.props.continueIsManaged}
|
|
|
|
onPhaseChange={this.onPhaseChange}
|
2022-05-13 14:10:22 +00:00
|
|
|
requestEmailToken={this.authLogic.requestEmailToken}
|
2021-07-07 10:08:53 +00:00
|
|
|
continueText={this.props.continueText}
|
|
|
|
continueKind={this.props.continueKind}
|
|
|
|
onCancel={this.onStageCancel}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|