import React from 'react'; import classNames from 'classnames'; import { LocalizerType } from '../../types/Util'; import { SessionInput } from './SessionInput'; import { SessionButton, SessionButtonType } from './SessionButton'; import { trigger } from '../../shims/events'; import { SessionHtmlRenderer } from './SessionHTMLRenderer'; interface Props { i18n: LocalizerType; } enum SignInMode { Default, UsingSeed, LinkingDevice, } enum SignUpMode { Default, SessionIDShown, } enum TabType { Create, SignIn, } interface State { selectedTab: TabType; signInMode: SignInMode; signUpMode: SignUpMode; displayName: string; password: string; validatePassword: string; passwordErrorString: string; passwordFieldsMatch: boolean; mnemonicSeed: string; hexGeneratedPubKey: string; primaryDevicePubKey: string; } const Tab = ({ isSelected, label, onSelect, type, }: { isSelected: boolean; label: string; onSelect?: (event: TabType) => void; type: TabType; }) => { const handleClick = onSelect ? () => { onSelect(type); } : undefined; return (
{label}
); }; export class RegistrationTabs extends React.Component { private readonly accountManager: any; constructor(props: any) { super(props); this.onSeedChanged = this.onSeedChanged.bind(this); this.onDisplayNameChanged = this.onDisplayNameChanged.bind(this); this.onPasswordChanged = this.onPasswordChanged.bind(this); this.onPasswordVerifyChanged = this.onPasswordVerifyChanged.bind(this); this.onSignUpGenerateSessionIDClick = this.onSignUpGenerateSessionIDClick.bind( this ); this.onSignUpGetStartedClick = this.onSignUpGetStartedClick.bind(this); this.onSecondDeviceSessionIDChanged = this.onSecondDeviceSessionIDChanged.bind( this ); this.onSecondaryDeviceRegistered = this.onSecondaryDeviceRegistered.bind( this ); this.state = { selectedTab: TabType.Create, signInMode: SignInMode.Default, signUpMode: SignUpMode.Default, displayName: '', password: '', validatePassword: '', passwordErrorString: '', passwordFieldsMatch: false, mnemonicSeed: '', hexGeneratedPubKey: '', primaryDevicePubKey: '', }; this.accountManager = window.getAccountManager(); // Clean status in case the app closed unexpectedly window.textsecure.storage.remove('secondaryDeviceStatus'); } public render() { this.generateMnemonicAndKeyPair().ignore(); return this.renderTabs(); } private async generateMnemonicAndKeyPair() { if (this.state.mnemonicSeed === '') { const language = 'english'; const mnemonic = await this.accountManager.generateMnemonic(language); let seedHex = window.mnemonic.mn_decode(mnemonic, language); // handle shorter than 32 bytes seeds const privKeyHexLength = 32 * 2; if (seedHex.length !== privKeyHexLength) { seedHex = seedHex.concat(seedHex); seedHex = seedHex.substring(0, privKeyHexLength); } const privKeyHex = window.mnemonic.sc_reduce32(seedHex); const privKey = window.dcodeIO.ByteBuffer.wrap( privKeyHex, 'hex' ).toArrayBuffer(); const keyPair = await window.libsignal.Curve.async.createKeyPair(privKey); const hexGeneratedPubKey = Buffer.from(keyPair.pubKey).toString('hex'); this.setState({ mnemonicSeed: mnemonic, hexGeneratedPubKey, // our 'frontend' sessionID }); } } private renderTabs() { const { selectedTab } = this.state; const { i18n } = this.props; const createAccount = i18n('createAccount'); const signIn = i18n('signIn'); const isCreateSelected = selectedTab === TabType.Create; const isSignInSelected = selectedTab === TabType.SignIn; return (
{this.renderSections()}
); } private readonly handleTabSelect = (tabType: TabType): void => { if (tabType !== TabType.SignIn) { this.cancelSecondaryDevice().ignore(); } this.setState({ selectedTab: tabType, signInMode: SignInMode.Default, signUpMode: SignUpMode.Default, }); }; private onSeedChanged(val: string) { this.setState({ mnemonicSeed: val }); } private onDisplayNameChanged(val: string) { const sanitizedName = this.sanitiseNameInput(val); this.setState({ displayName: sanitizedName }); } private onPasswordChanged(val: string) { this.setState({ password: val }); this.onValidatePassword(); // FIXME add bubbles or something to help the user know what he did wrong } private onPasswordVerifyChanged(val: string) { this.setState({ validatePassword: val }); } private renderSections() { const { selectedTab } = this.state; if (selectedTab === TabType.Create) { return this.renderSignUp(); } return this.renderSignIn(); } private renderSignUp() { const { signUpMode } = this.state; const { i18n } = this.props; if (signUpMode === SignUpMode.Default) { return (
{this.renderSignUpHeader()} {this.renderSignUpButton()}
); } else { return (
{this.renderSignUpHeader()}
{i18n('yourUniqueSessionID')}
{this.renderEnterSessionID(false, this.state.hexGeneratedPubKey)} {this.renderSignUpButton()} {this.getRenderTermsConditionAgreement()}
); } } private getRenderTermsConditionAgreement() { const { selectedTab, signInMode, signUpMode } = this.state; if (selectedTab === TabType.Create) { return signUpMode !== SignUpMode.Default ? this.renderTermsConditionAgreement() : null; } else { return signInMode !== SignInMode.Default ? this.renderTermsConditionAgreement() : null; } } private renderSignUpHeader() { const allUsersAreRandomly = this.props.i18n('allUsersAreRandomly...'); return
{allUsersAreRandomly}
; } private renderSignUpButton() { const { signUpMode } = this.state; const { i18n } = this.props; let buttonType: any; let buttonText: string; if (signUpMode !== SignUpMode.Default) { buttonType = SessionButtonType.FullGreen; buttonText = i18n('getStarted'); } else { buttonType = SessionButtonType.Green; buttonText = i18n('generateSessionID'); } return ( { if (signUpMode === SignUpMode.Default) { this.onSignUpGenerateSessionIDClick().ignore(); } else { this.onSignUpGetStartedClick(); } }} buttonType={buttonType} text={buttonText} /> ); } private async onSignUpGenerateSessionIDClick() { this.setState({ signUpMode: SignUpMode.SessionIDShown, }); } private onSignUpGetStartedClick() { this.setState({ selectedTab: TabType.SignIn, signInMode: SignInMode.UsingSeed, }); } private renderSignIn() { return (
{this.renderRegistrationContent()} {this.renderSignInButtons()} {this.getRenderTermsConditionAgreement()}
); } private renderRegistrationContent() { const { signInMode } = this.state; const { i18n } = this.props; if (signInMode === SignInMode.UsingSeed) { return (
{ this.onSeedChanged(val); }} /> { this.onDisplayNameChanged(val); }} /> { this.onPasswordChanged(val); }} /> { this.onPasswordVerifyChanged(val); }} />
); } else if (signInMode === SignInMode.LinkingDevice) { return (
{i18n('devicePairingHeader')}
{this.renderEnterSessionID(true)}
); } else { return null; } } private renderEnterSessionID(contentEditable: boolean, text?: string) { const { i18n } = this.props; const enterSessionIDHere = i18n('enterSessionIDHere'); return (
{ if (contentEditable) { this.onSecondDeviceSessionIDChanged(e); } }} > {text}
); } private onSecondDeviceSessionIDChanged(e: any) { e.preventDefault(); const hexEncodedPubKey = e.target.innerHTML; this.setState({ primaryDevicePubKey: hexEncodedPubKey, }); } private renderSignInButtons() { const { signInMode } = this.state; const { i18n } = this.props; const or = i18n('or'); if (signInMode === SignInMode.Default) { return (
{this.renderRestoreUsingSeedButton(SessionButtonType.Green)}
{or}
{this.renderLinkDeviceToExistingAccountButton()}
); } if (signInMode === SignInMode.LinkingDevice) { return (
{this.renderContinueYourSessionButton()}
{or}
{this.renderRestoreUsingSeedButton(SessionButtonType.White)}
); } return (
{this.renderContinueYourSessionButton()}
{or}
{this.renderLinkDeviceToExistingAccountButton()}
); } private renderTermsConditionAgreement() { const { i18n } = this.props; // FIXME link to our Terms and Conditions and privacy statement return (
); } private renderContinueYourSessionButton() { return ( { if (this.state.signInMode === SignInMode.UsingSeed) { this.register('english').ignore(); } else { this.registerSecondaryDevice().ignore(); } }} buttonType={SessionButtonType.FullGreen} text={this.props.i18n('continueYourSession')} /> ); } private renderRestoreUsingSeedButton(buttonType: SessionButtonType) { return ( { this.cancelSecondaryDevice().ignore(); this.setState({ signInMode: SignInMode.UsingSeed, primaryDevicePubKey: '', mnemonicSeed: '', displayName: '', signUpMode: SignUpMode.Default, }); }} buttonType={buttonType} text={this.props.i18n('restoreUsingSeed')} /> ); } private renderLinkDeviceToExistingAccountButton() { return ( { this.setState({ signInMode: SignInMode.LinkingDevice, mnemonicSeed: '', displayName: '', signUpMode: SignUpMode.Default, }); }} buttonType={SessionButtonType.White} text={this.props.i18n('linkDeviceToExistingAccount')} /> ); } private trim(value: string) { return value ? value.trim() : value; } private validatePassword() { const input = this.trim(this.state.password); const confirmationInput = this.trim(this.state.validatePassword); // If user hasn't set a value then skip if (!input && !confirmationInput) { return null; } const error = window.passwordUtil.validatePassword(input, this.props.i18n); if (error) { return error; } if (input !== confirmationInput) { return "Password don't match"; } return null; } private onValidatePassword() { const passwordValidation = this.validatePassword(); if (passwordValidation) { this.setState({ passwordErrorString: passwordValidation }); } else { // Show green box around inputs that match const input = this.trim(this.state.password); const confirmationInput = this.trim(this.state.validatePassword); const passwordFieldsMatch = input !== undefined && input === confirmationInput; this.setState({ passwordErrorString: '', passwordFieldsMatch, }); } } private sanitiseNameInput(val: string) { return val.trim().replace(window.displayNameRegex, ''); } private async resetRegistration() { await window.Signal.Data.removeAllIdentityKeys(); await window.Signal.Data.removeAllPrivateConversations(); window.Whisper.Registration.remove(); // Do not remove all items since they are only set // at startup. window.textsecure.storage.remove('identityKey'); window.textsecure.storage.remove('secondaryDeviceStatus'); window.ConversationController.reset(); await window.ConversationController.load(); window.Whisper.RotateSignedPreKeyListener.stop(window.Whisper.events); } private async register(language: string) { const { password, mnemonicSeed, displayName } = this.state; // Make sure the password is valid if (this.validatePassword()) { //this.showToast(i18n('invalidPassword')); return; } if (!mnemonicSeed) { return; } if (!displayName) { return; } // Ensure we clear the secondary device registration status window.textsecure.storage.remove('secondaryDeviceStatus'); try { await this.resetRegistration(); await window.setPassword(password); await this.accountManager.registerSingleDevice( mnemonicSeed, language, displayName ); trigger('openInbox'); } catch (e) { if (typeof e === 'string') { //this.showToast(e); } //this.log(e); } } private async cancelSecondaryDevice() { window.Whisper.events.off( 'secondaryDeviceRegistration', this.onSecondaryDeviceRegistered ); await this.resetRegistration(); } private async registerSecondaryDevice() { // tslint:disable-next-line: no-backbone-get-set-outside-model if (window.textsecure.storage.get('secondaryDeviceStatus') === 'ongoing') { return; } await this.resetRegistration(); window.textsecure.storage.put('secondaryDeviceStatus', 'ongoing'); const primaryPubKey = this.state.primaryDevicePubKey; // Ensure only one listener window.Whisper.events.off( 'secondaryDeviceRegistration', this.onSecondaryDeviceRegistered ); window.Whisper.events.once( 'secondaryDeviceRegistration', this.onSecondaryDeviceRegistered ); const onError = async (error: any) => { window.console.error(error); await this.resetRegistration(); }; const c = new window.Whisper.Conversation({ id: primaryPubKey, type: 'private', }); const validationError = c.validateNumber(); if (validationError) { onError('Invalid public key').ignore(); return; } try { const fakeMnemonic = this.state.mnemonicSeed; await this.accountManager.registerSingleDevice( fakeMnemonic, 'english', null ); await this.accountManager.requestPairing(primaryPubKey); const pubkey = window.textsecure.storage.user.getNumber(); const words = window.mnemonic.pubkey_to_secret_words(pubkey); window.console.log('pubkey_to_secret_words'); window.console.log(`Here is your secret:\n${words}`); } catch (e) { window.console.log(e); //onError(e); } } private async onSecondaryDeviceRegistered() { // Ensure the left menu is updated trigger('userChanged', { isSecondaryDevice: true }); // will re-run the background initialisation trigger('registration_done'); trigger('openInbox'); } }