You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
	
	
		
			350 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			TypeScript
		
	
		
		
			
		
	
	
			350 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			TypeScript
		
	
| 
											2 years ago
										 | /* eslint-disable @typescript-eslint/no-misused-promises */ | ||
| 
											2 years ago
										 | 
 | ||
| 
											2 years ago
										 | import autoBind from 'auto-bind'; | ||
| 
											2 years ago
										 | import { Component } from 'react'; | ||
| 
											3 years ago
										 | import { Data } from '../../data/data'; | ||
| 
											2 years ago
										 | import { ToastUtils } from '../../session/utils'; | ||
| 
											4 years ago
										 | import { sessionPassword } from '../../state/ducks/modalDialog'; | ||
| 
											4 years ago
										 | import { LocalizerKeys } from '../../types/LocalizerKeys'; | ||
| 
											2 years ago
										 | import type { PasswordAction } from '../../types/ReduxTypes'; | ||
| 
											3 years ago
										 | import { assertUnreachable } from '../../types/sqlSharedTypes'; | ||
| 
											2 years ago
										 | import { matchesHash, validatePassword } from '../../util/passwordUtils'; | ||
|  | import { SessionWrapperModal } from '../SessionWrapperModal'; | ||
|  | import { SessionButton, SessionButtonColor, SessionButtonType } from '../basic/SessionButton'; | ||
|  | import { SpacerSM } from '../basic/Text'; | ||
| 
											4 years ago
										 | 
 | ||
| 
											6 years ago
										 | interface Props { | ||
| 
											4 years ago
										 |   passwordAction: PasswordAction; | ||
|  |   onOk: () => void; | ||
| 
											6 years ago
										 | } | ||
|  | 
 | ||
|  | interface State { | ||
|  |   error: string | null; | ||
| 
											5 years ago
										 |   currentPasswordEntered: string | null; | ||
|  |   currentPasswordConfirmEntered: string | null; | ||
| 
											5 years ago
										 |   currentPasswordRetypeEntered: string | null; | ||
| 
											6 years ago
										 | } | ||
|  | 
 | ||
| 
											1 year ago
										 | export class SessionSetPasswordDialog extends Component<Props, State> { | ||
| 
											5 years ago
										 |   private passportInput: HTMLInputElement | null = null; | ||
| 
											6 years ago
										 | 
 | ||
| 
											6 years ago
										 |   constructor(props: any) { | ||
|  |     super(props); | ||
|  | 
 | ||
|  |     this.state = { | ||
|  |       error: null, | ||
| 
											5 years ago
										 |       currentPasswordEntered: null, | ||
|  |       currentPasswordConfirmEntered: null, | ||
| 
											5 years ago
										 |       currentPasswordRetypeEntered: null, | ||
| 
											6 years ago
										 |     }; | ||
| 
											6 years ago
										 | 
 | ||
| 
											4 years ago
										 |     autoBind(this); | ||
| 
											6 years ago
										 |   } | ||
|  | 
 | ||
|  |   public componentDidMount() { | ||
| 
											3 years ago
										 |     document.addEventListener('keyup', this.onEnterPressed); | ||
|  | 
 | ||
| 
											6 years ago
										 |     setTimeout(() => { | ||
| 
											2 years ago
										 |       this.passportInput?.focus(); | ||
| 
											5 years ago
										 |     }, 1); | ||
| 
											6 years ago
										 |   } | ||
|  | 
 | ||
| 
											3 years ago
										 |   public componentWillUnmount() { | ||
|  |     document.removeEventListener('keyup', this.onEnterPressed); | ||
|  |   } | ||
|  | 
 | ||
| 
											6 years ago
										 |   public render() { | ||
| 
											4 years ago
										 |     const { passwordAction } = this.props; | ||
| 
											3 years ago
										 |     let placeholders: Array<string> = []; | ||
|  |     switch (passwordAction) { | ||
|  |       case 'change': | ||
|  |         placeholders = [ | ||
|  |           window.i18n('typeInOldPassword'), | ||
|  |           window.i18n('enterNewPassword'), | ||
|  |           window.i18n('confirmNewPassword'), | ||
|  |         ]; | ||
|  |         break; | ||
|  |       case 'remove': | ||
|  |         placeholders = [window.i18n('enterPassword')]; | ||
|  |         break; | ||
| 
											3 years ago
										 |       case 'enter': | ||
|  |         placeholders = [window.i18n('enterPassword')]; | ||
|  |         break; | ||
| 
											3 years ago
										 |       default: | ||
|  |         placeholders = [window.i18n('createPassword'), window.i18n('confirmPassword')]; | ||
|  |     } | ||
| 
											6 years ago
										 | 
 | ||
| 
											3 years ago
										 |     const confirmButtonText = | ||
|  |       passwordAction === 'remove' ? window.i18n('remove') : window.i18n('done'); | ||
| 
											4 years ago
										 |     // do this separately so typescript's compiler likes it
 | ||
|  |     const localizedKeyAction: LocalizerKeys = | ||
|  |       passwordAction === 'change' | ||
|  |         ? 'changePassword' | ||
|  |         : passwordAction === 'remove' | ||
| 
											2 years ago
										 |           ? 'removePassword' | ||
|  |           : passwordAction === 'enter' | ||
|  |             ? 'passwordViewTitle' | ||
|  |             : 'setPassword'; | ||
| 
											6 years ago
										 | 
 | ||
|  |     return ( | ||
| 
											4 years ago
										 |       <SessionWrapperModal title={window.i18n(localizedKeyAction)} onClose={this.closeDialog}> | ||
| 
											4 years ago
										 |         <SpacerSM /> | ||
| 
											6 years ago
										 | 
 | ||
|  |         <div className="session-modal__input-group"> | ||
|  |           <input | ||
|  |             type="password" | ||
|  |             id="password-modal-input" | ||
| 
											5 years ago
										 |             ref={input => { | ||
|  |               this.passportInput = input; | ||
|  |             }} | ||
| 
											6 years ago
										 |             placeholder={placeholders[0]} | ||
| 
											3 years ago
										 |             onChange={this.onPasswordInput} | ||
|  |             onPaste={this.onPasswordInput} | ||
| 
											3 years ago
										 |             data-testid="password-input" | ||
| 
											6 years ago
										 |           /> | ||
| 
											3 years ago
										 |           {passwordAction !== 'enter' && passwordAction !== 'remove' && ( | ||
| 
											6 years ago
										 |             <input | ||
|  |               type="password" | ||
| 
											6 years ago
										 |               id="password-modal-input-confirm" | ||
|  |               placeholder={placeholders[1]} | ||
| 
											3 years ago
										 |               onChange={this.onPasswordConfirmInput} | ||
|  |               onPaste={this.onPasswordConfirmInput} | ||
| 
											3 years ago
										 |               data-testid="password-input-confirm" | ||
| 
											6 years ago
										 |             /> | ||
| 
											6 years ago
										 |           )} | ||
| 
											4 years ago
										 |           {passwordAction === 'change' && ( | ||
| 
											5 years ago
										 |             <input | ||
|  |               type="password" | ||
|  |               id="password-modal-input-reconfirm" | ||
|  |               placeholder={placeholders[2]} | ||
| 
											3 years ago
										 |               onPaste={this.onPasswordRetypeInput} | ||
|  |               onChange={this.onPasswordRetypeInput} | ||
| 
											3 years ago
										 |               data-testid="password-input-reconfirm" | ||
| 
											5 years ago
										 |             /> | ||
|  |           )} | ||
| 
											6 years ago
										 |         </div> | ||
|  | 
 | ||
| 
											4 years ago
										 |         <SpacerSM /> | ||
| 
											6 years ago
										 | 
 | ||
|  |         <div className="session-modal__button-group"> | ||
| 
											3 years ago
										 |           <SessionButton | ||
| 
											3 years ago
										 |             text={confirmButtonText} | ||
| 
											3 years ago
										 |             buttonColor={passwordAction === 'remove' ? SessionButtonColor.Danger : undefined} | ||
|  |             buttonType={SessionButtonType.Simple} | ||
| 
											5 years ago
										 |             onClick={this.setPassword} | ||
| 
											6 years ago
										 |           /> | ||
| 
											3 years ago
										 |           {passwordAction !== 'enter' && ( | ||
|  |             <SessionButton | ||
|  |               text={window.i18n('cancel')} | ||
|  |               buttonColor={passwordAction !== 'remove' ? SessionButtonColor.Danger : undefined} | ||
|  |               buttonType={SessionButtonType.Simple} | ||
|  |               onClick={this.closeDialog} | ||
|  |             /> | ||
|  |           )} | ||
| 
											6 years ago
										 |         </div> | ||
| 
											4 years ago
										 |       </SessionWrapperModal> | ||
| 
											6 years ago
										 |     ); | ||
|  |   } | ||
| 
											6 years ago
										 | 
 | ||
| 
											6 years ago
										 |   public async validatePasswordHash(password: string | null) { | ||
|  |     // Check if the password matches the hash we have stored
 | ||
| 
											3 years ago
										 |     const hash = await Data.getPasswordHash(); | ||
| 
											4 years ago
										 |     if (hash && !matchesHash(password, hash)) { | ||
| 
											6 years ago
										 |       return false; | ||
|  |     } | ||
| 
											6 years ago
										 | 
 | ||
| 
											6 years ago
										 |     return true; | ||
|  |   } | ||
|  | 
 | ||
|  |   private showError() { | ||
| 
											3 years ago
										 |     if (this.state.error) { | ||
|  |       ToastUtils.pushToastError('enterPasswordErrorToast', this.state.error); | ||
|  |     } | ||
| 
											6 years ago
										 |   } | ||
|  | 
 | ||
| 
											5 years ago
										 |   /** | ||
|  |    * Returns false and set the state error field in the input is not a valid password | ||
|  |    * or returns true | ||
|  |    */ | ||
|  |   private validatePassword(firstPassword: string) { | ||
| 
											5 years ago
										 |     // if user did not fill the first password field, we can't do anything
 | ||
| 
											4 years ago
										 |     const errorFirstInput = validatePassword(firstPassword); | ||
| 
											5 years ago
										 |     if (errorFirstInput !== null) { | ||
| 
											6 years ago
										 |       this.setState({ | ||
| 
											5 years ago
										 |         error: errorFirstInput, | ||
| 
											6 years ago
										 |       }); | ||
| 
											3 years ago
										 |       this.showError(); | ||
| 
											5 years ago
										 |       return false; | ||
|  |     } | ||
|  |     return true; | ||
|  |   } | ||
|  | 
 | ||
| 
											5 years ago
										 |   private async handleActionSet(enteredPassword: string, enteredPasswordConfirm: string) { | ||
| 
											5 years ago
										 |     // be sure both password are valid
 | ||
|  |     if (!this.validatePassword(enteredPassword)) { | ||
| 
											6 years ago
										 |       return; | ||
|  |     } | ||
| 
											5 years ago
										 |     // no need to validate second password. we just need to check that enteredPassword is valid, and that both password matches
 | ||
| 
											6 years ago
										 | 
 | ||
| 
											5 years ago
										 |     if (enteredPassword !== enteredPasswordConfirm) { | ||
|  |       this.setState({ | ||
|  |         error: window.i18n('setPasswordInvalid'), | ||
|  |       }); | ||
| 
											3 years ago
										 |       this.showError(); | ||
| 
											5 years ago
										 |       return; | ||
| 
											5 years ago
										 |     } | ||
| 
											5 years ago
										 |     await window.setPassword(enteredPassword, null); | ||
|  |     ToastUtils.pushToastSuccess( | ||
|  |       'setPasswordSuccessToast', | ||
|  |       window.i18n('setPasswordTitle'), | ||
| 
											3 years ago
										 |       window.i18n('setPasswordToastDescription') | ||
| 
											5 years ago
										 |     ); | ||
| 
											5 years ago
										 | 
 | ||
| 
											4 years ago
										 |     this.props.onOk(); | ||
| 
											5 years ago
										 |     this.closeDialog(); | ||
|  |   } | ||
| 
											6 years ago
										 | 
 | ||
| 
											5 years ago
										 |   private async handleActionChange( | ||
|  |     oldPassword: string, | ||
|  |     newPassword: string, | ||
|  |     newConfirmedPassword: string | ||
|  |   ) { | ||
| 
											5 years ago
										 |     // We don't validate oldPassword on change: this is validate on the validatePasswordHash below
 | ||
|  |     // we only validate the newPassword here
 | ||
|  |     if (!this.validatePassword(newPassword)) { | ||
|  |       return; | ||
| 
											5 years ago
										 |     } | ||
| 
											5 years ago
										 | 
 | ||
|  |     // Check the retyped password matches the new password
 | ||
|  |     if (newPassword !== newConfirmedPassword) { | ||
|  |       this.setState({ | ||
| 
											5 years ago
										 |         error: window.i18n('passwordsDoNotMatch'), | ||
|  |       }); | ||
| 
											3 years ago
										 |       this.showError(); | ||
| 
											5 years ago
										 |       return; | ||
| 
											5 years ago
										 |     } | ||
|  | 
 | ||
| 
											5 years ago
										 |     const isValidWithStoredInDB = Boolean(await this.validatePasswordHash(oldPassword)); | ||
| 
											5 years ago
										 |     if (!isValidWithStoredInDB) { | ||
| 
											6 years ago
										 |       this.setState({ | ||
| 
											5 years ago
										 |         error: window.i18n('changePasswordInvalid'), | ||
| 
											6 years ago
										 |       }); | ||
| 
											3 years ago
										 |       this.showError(); | ||
| 
											6 years ago
										 |       return; | ||
|  |     } | ||
|  |     await window.setPassword(newPassword, oldPassword); | ||
|  | 
 | ||
| 
											5 years ago
										 |     ToastUtils.pushToastSuccess( | ||
|  |       'setPasswordSuccessToast', | ||
|  |       window.i18n('changePasswordTitle'), | ||
| 
											3 years ago
										 |       window.i18n('changePasswordToastDescription') | ||
| 
											5 years ago
										 |     ); | ||
| 
											6 years ago
										 | 
 | ||
| 
											4 years ago
										 |     this.props.onOk(); | ||
| 
											6 years ago
										 |     this.closeDialog(); | ||
|  |   } | ||
|  | 
 | ||
| 
											5 years ago
										 |   private async handleActionRemove(oldPassword: string) { | ||
|  |     // We don't validate oldPassword on change: this is validate on the validatePasswordHash below
 | ||
| 
											5 years ago
										 |     const isValidWithStoredInDB = Boolean(await this.validatePasswordHash(oldPassword)); | ||
| 
											5 years ago
										 |     if (!isValidWithStoredInDB) { | ||
|  |       this.setState({ | ||
|  |         error: window.i18n('removePasswordInvalid'), | ||
|  |       }); | ||
| 
											3 years ago
										 |       this.showError(); | ||
| 
											5 years ago
										 |       return; | ||
| 
											6 years ago
										 |     } | ||
| 
											5 years ago
										 |     await window.setPassword(null, oldPassword); | ||
|  | 
 | ||
|  |     ToastUtils.pushToastWarning( | ||
|  |       'setPasswordSuccessToast', | ||
|  |       window.i18n('removePasswordTitle'), | ||
|  |       window.i18n('removePasswordToastDescription') | ||
|  |     ); | ||
|  | 
 | ||
| 
											4 years ago
										 |     this.props.onOk(); | ||
| 
											5 years ago
										 |     this.closeDialog(); | ||
| 
											6 years ago
										 |   } | ||
| 
											6 years ago
										 | 
 | ||
| 
											3 years ago
										 |   private async onEnterPressed(event: any) { | ||
|  |     if (event.key === 'Enter') { | ||
|  |       event.stopPropagation(); | ||
| 
											2 years ago
										 |       await this.setPassword(); | ||
| 
											3 years ago
										 |     } | ||
|  |   } | ||
|  | 
 | ||
| 
											3 years ago
										 |   private async handleActionEnter(enteredPassword: string) { | ||
|  |     // be sure the password is valid
 | ||
|  |     if (!this.validatePassword(enteredPassword)) { | ||
|  |       return; | ||
|  |     } | ||
|  | 
 | ||
|  |     const isValidWithStoredInDB = Boolean(await this.validatePasswordHash(enteredPassword)); | ||
|  |     if (!isValidWithStoredInDB) { | ||
|  |       this.setState({ | ||
|  |         error: window.i18n('invalidPassword'), | ||
|  |       }); | ||
|  |       this.showError(); | ||
|  |       return; | ||
|  |     } | ||
|  | 
 | ||
|  |     this.props.onOk(); | ||
|  |     this.closeDialog(); | ||
|  |   } | ||
|  | 
 | ||
| 
											5 years ago
										 |   private async setPassword() { | ||
| 
											4 years ago
										 |     const { passwordAction } = this.props; | ||
| 
											2 years ago
										 |     const { currentPasswordEntered, currentPasswordConfirmEntered, currentPasswordRetypeEntered } = | ||
|  |       this.state; | ||
| 
											5 years ago
										 | 
 | ||
|  |     // Trim leading / trailing whitespace for UX
 | ||
|  |     const firstPasswordEntered = (currentPasswordEntered || '').trim(); | ||
|  |     const secondPasswordEntered = (currentPasswordConfirmEntered || '').trim(); | ||
| 
											5 years ago
										 |     const thirdPasswordEntered = (currentPasswordRetypeEntered || '').trim(); | ||
| 
											5 years ago
										 | 
 | ||
| 
											4 years ago
										 |     switch (passwordAction) { | ||
|  |       case 'set': { | ||
| 
											5 years ago
										 |         await this.handleActionSet(firstPasswordEntered, secondPasswordEntered); | ||
|  |         return; | ||
|  |       } | ||
| 
											4 years ago
										 |       case 'change': { | ||
| 
											5 years ago
										 |         await this.handleActionChange( | ||
|  |           firstPasswordEntered, | ||
| 
											5 years ago
										 |           secondPasswordEntered, | ||
|  |           thirdPasswordEntered | ||
| 
											5 years ago
										 |         ); | ||
|  |         return; | ||
|  |       } | ||
| 
											4 years ago
										 |       case 'remove': { | ||
| 
											5 years ago
										 |         await this.handleActionRemove(firstPasswordEntered); | ||
|  |         return; | ||
|  |       } | ||
| 
											3 years ago
										 |       case 'enter': { | ||
|  |         await this.handleActionEnter(firstPasswordEntered); | ||
|  |         return; | ||
|  |       } | ||
| 
											5 years ago
										 |       default: | ||
| 
											3 years ago
										 |         assertUnreachable(passwordAction, 'passwordAction'); | ||
| 
											6 years ago
										 |     } | ||
| 
											5 years ago
										 |   } | ||
| 
											6 years ago
										 | 
 | ||
| 
											5 years ago
										 |   private closeDialog() { | ||
| 
											4 years ago
										 |     window.inboxStore?.dispatch(sessionPassword(null)); | ||
| 
											6 years ago
										 |   } | ||
|  | 
 | ||
| 
											3 years ago
										 |   private onPasswordInput(event: any) { | ||
| 
											5 years ago
										 |     const currentPasswordEntered = event.target.value; | ||
|  |     this.setState({ currentPasswordEntered }); | ||
| 
											5 years ago
										 |   } | ||
| 
											6 years ago
										 | 
 | ||
| 
											3 years ago
										 |   private onPasswordConfirmInput(event: any) { | ||
| 
											5 years ago
										 |     const currentPasswordConfirmEntered = event.target.value; | ||
|  |     this.setState({ currentPasswordConfirmEntered }); | ||
| 
											6 years ago
										 |   } | ||
| 
											5 years ago
										 | 
 | ||
| 
											3 years ago
										 |   private onPasswordRetypeInput(event: any) { | ||
| 
											5 years ago
										 |     const currentPasswordRetypeEntered = event.target.value; | ||
|  |     this.setState({ currentPasswordRetypeEntered }); | ||
|  |   } | ||
| 
											6 years ago
										 | } |