chore: migrate redux module encryption to typescript
This commit is contained in:
parent
9d9553b075
commit
2c2635e250
|
@ -1,35 +0,0 @@
|
||||||
import * as types from './actionsTypes';
|
|
||||||
|
|
||||||
export function encryptionInit() {
|
|
||||||
return {
|
|
||||||
type: types.ENCRYPTION.INIT
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
export function encryptionStop() {
|
|
||||||
return {
|
|
||||||
type: types.ENCRYPTION.STOP
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
export function encryptionSet(enabled = false, banner = null) {
|
|
||||||
return {
|
|
||||||
type: types.ENCRYPTION.SET,
|
|
||||||
enabled,
|
|
||||||
banner
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
export function encryptionSetBanner(banner) {
|
|
||||||
return {
|
|
||||||
type: types.ENCRYPTION.SET_BANNER,
|
|
||||||
banner
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
export function encryptionDecodeKey(password) {
|
|
||||||
return {
|
|
||||||
type: types.ENCRYPTION.DECODE_KEY,
|
|
||||||
password
|
|
||||||
};
|
|
||||||
}
|
|
|
@ -0,0 +1,51 @@
|
||||||
|
import { Action } from 'redux';
|
||||||
|
|
||||||
|
import { ENCRYPTION } from './actionsTypes';
|
||||||
|
|
||||||
|
export interface IEncryptionSet extends Action {
|
||||||
|
enabled: boolean;
|
||||||
|
banner: any;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface IEncryptionSetBanner extends Action {
|
||||||
|
banner: any;
|
||||||
|
}
|
||||||
|
export interface IEncryptionDecodeKey extends Action {
|
||||||
|
password: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export type TActionEncryption = IEncryptionSet & IEncryptionSetBanner & IEncryptionDecodeKey;
|
||||||
|
|
||||||
|
export function encryptionInit(): Action {
|
||||||
|
return {
|
||||||
|
type: ENCRYPTION.INIT
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function encryptionStop(): Action {
|
||||||
|
return {
|
||||||
|
type: ENCRYPTION.STOP
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function encryptionSet(enabled = false, banner: any = null): IEncryptionSet {
|
||||||
|
return {
|
||||||
|
type: ENCRYPTION.SET,
|
||||||
|
enabled,
|
||||||
|
banner
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function encryptionSetBanner(banner: any = null): IEncryptionSetBanner {
|
||||||
|
return {
|
||||||
|
type: ENCRYPTION.SET_BANNER,
|
||||||
|
banner
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function encryptionDecodeKey(password: string): IEncryptionDecodeKey {
|
||||||
|
return {
|
||||||
|
type: ENCRYPTION.DECODE_KEY,
|
||||||
|
password
|
||||||
|
};
|
||||||
|
}
|
|
@ -1,7 +1,9 @@
|
||||||
import { TActionSelectedUsers } from '../../actions/selectedUsers';
|
|
||||||
import { TActionActiveUsers } from '../../actions/activeUsers';
|
import { TActionActiveUsers } from '../../actions/activeUsers';
|
||||||
|
import { TActionEncryption } from '../../actions/encryption';
|
||||||
|
import { TActionSelectedUsers } from '../../actions/selectedUsers';
|
||||||
// REDUCERS
|
// REDUCERS
|
||||||
import { IActiveUsers } from '../../reducers/activeUsers';
|
import { IActiveUsers } from '../../reducers/activeUsers';
|
||||||
|
import { IEncryption } from '../../reducers/encryption';
|
||||||
import { ISelectedUsers } from '../../reducers/selectedUsers';
|
import { ISelectedUsers } from '../../reducers/selectedUsers';
|
||||||
|
|
||||||
export interface IApplicationState {
|
export interface IApplicationState {
|
||||||
|
@ -23,9 +25,9 @@ export interface IApplicationState {
|
||||||
createDiscussion: any;
|
createDiscussion: any;
|
||||||
inquiry: any;
|
inquiry: any;
|
||||||
enterpriseModules: any;
|
enterpriseModules: any;
|
||||||
encryption: any;
|
encryption: IEncryption;
|
||||||
permissions: any;
|
permissions: any;
|
||||||
roles: any;
|
roles: any;
|
||||||
}
|
}
|
||||||
|
|
||||||
export type TApplicationActions = TActionActiveUsers & TActionSelectedUsers;
|
export type TApplicationActions = TActionActiveUsers & TActionSelectedUsers & TActionEncryption;
|
||||||
|
|
|
@ -0,0 +1,28 @@
|
||||||
|
import { encryptionSet, encryptionInit, encryptionSetBanner } from '../actions/encryption';
|
||||||
|
import { mockedStore } from './mockedStore';
|
||||||
|
import { initialState } from './encryption';
|
||||||
|
|
||||||
|
describe('test encryption reducer', () => {
|
||||||
|
it('should return initial state', () => {
|
||||||
|
const state = mockedStore.getState().encryption;
|
||||||
|
expect(state).toEqual(initialState);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return modified store after encryptionSet', () => {
|
||||||
|
mockedStore.dispatch(encryptionSet(true, true));
|
||||||
|
const state = mockedStore.getState().encryption;
|
||||||
|
expect(state).toEqual({ banner: true, enabled: true });
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return empty store after encryptionInit', () => {
|
||||||
|
mockedStore.dispatch(encryptionInit());
|
||||||
|
const state = mockedStore.getState().encryption;
|
||||||
|
expect(state).toEqual({ banner: null, enabled: false });
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return initial state after encryptionSetBanner', () => {
|
||||||
|
mockedStore.dispatch(encryptionSetBanner(true));
|
||||||
|
const state = mockedStore.getState().encryption;
|
||||||
|
expect(state).toEqual({ banner: true, enabled: false });
|
||||||
|
});
|
||||||
|
});
|
|
@ -1,11 +1,18 @@
|
||||||
import { ENCRYPTION } from '../actions/actionsTypes';
|
import { ENCRYPTION } from '../actions/actionsTypes';
|
||||||
|
import { TApplicationActions } from '../definitions';
|
||||||
|
|
||||||
const initialState = {
|
export interface IEncryption {
|
||||||
|
enabled: boolean;
|
||||||
|
// TODO
|
||||||
|
banner: any;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const initialState: IEncryption = {
|
||||||
enabled: false,
|
enabled: false,
|
||||||
banner: null
|
banner: null
|
||||||
};
|
};
|
||||||
|
|
||||||
export default function encryption(state = initialState, action) {
|
export default function encryption(state = initialState, action: TApplicationActions): IEncryption {
|
||||||
switch (action.type) {
|
switch (action.type) {
|
||||||
case ENCRYPTION.SET:
|
case ENCRYPTION.SET:
|
||||||
return {
|
return {
|
|
@ -1,23 +1,23 @@
|
||||||
|
import { StackNavigationOptions } from '@react-navigation/stack';
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { ScrollView, StyleSheet, Text } from 'react-native';
|
import { ScrollView, StyleSheet, Text } from 'react-native';
|
||||||
import { connect } from 'react-redux';
|
import { connect } from 'react-redux';
|
||||||
import { Dispatch } from 'redux';
|
|
||||||
import { StackNavigationOptions, StackNavigationProp } from '@react-navigation/stack';
|
|
||||||
|
|
||||||
import I18n from '../i18n';
|
import { encryptionDecodeKey } from '../actions/encryption';
|
||||||
import { withTheme } from '../theme';
|
|
||||||
import Button from '../containers/Button';
|
|
||||||
import { themes } from '../constants/colors';
|
import { themes } from '../constants/colors';
|
||||||
import TextInput from '../containers/TextInput';
|
import Button from '../containers/Button';
|
||||||
import SafeAreaView from '../containers/SafeAreaView';
|
|
||||||
import * as HeaderButton from '../containers/HeaderButton';
|
import * as HeaderButton from '../containers/HeaderButton';
|
||||||
import { encryptionDecodeKey as encryptionDecodeKeyAction } from '../actions/encryption';
|
import SafeAreaView from '../containers/SafeAreaView';
|
||||||
import scrollPersistTaps from '../utils/scrollPersistTaps';
|
|
||||||
import KeyboardView from '../presentation/KeyboardView';
|
|
||||||
import StatusBar from '../containers/StatusBar';
|
import StatusBar from '../containers/StatusBar';
|
||||||
import { events, logEvent } from '../utils/log';
|
import TextInput from '../containers/TextInput';
|
||||||
import sharedStyles from './Styles';
|
import { IBaseScreen } from '../definitions';
|
||||||
|
import I18n from '../i18n';
|
||||||
|
import KeyboardView from '../presentation/KeyboardView';
|
||||||
import { E2EEnterYourPasswordStackParamList } from '../stacks/types';
|
import { E2EEnterYourPasswordStackParamList } from '../stacks/types';
|
||||||
|
import { withTheme } from '../theme';
|
||||||
|
import { events, logEvent } from '../utils/log';
|
||||||
|
import scrollPersistTaps from '../utils/scrollPersistTaps';
|
||||||
|
import sharedStyles from './Styles';
|
||||||
|
|
||||||
const styles = StyleSheet.create({
|
const styles = StyleSheet.create({
|
||||||
container: {
|
container: {
|
||||||
|
@ -34,21 +34,17 @@ interface IE2EEnterYourPasswordViewState {
|
||||||
password: string;
|
password: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
interface IE2EEnterYourPasswordViewProps {
|
type TE2EEnterYourPasswordViewProps = IBaseScreen<E2EEnterYourPasswordStackParamList, 'E2EEnterYourPasswordView'>;
|
||||||
encryptionDecodeKey: (password: string) => void;
|
|
||||||
theme: string;
|
|
||||||
navigation: StackNavigationProp<E2EEnterYourPasswordStackParamList, 'E2EEnterYourPasswordView'>;
|
|
||||||
}
|
|
||||||
|
|
||||||
class E2EEnterYourPasswordView extends React.Component<IE2EEnterYourPasswordViewProps, IE2EEnterYourPasswordViewState> {
|
class E2EEnterYourPasswordView extends React.Component<TE2EEnterYourPasswordViewProps, IE2EEnterYourPasswordViewState> {
|
||||||
private passwordInput?: TextInput;
|
private passwordInput?: TextInput;
|
||||||
|
|
||||||
static navigationOptions = ({ navigation }: Pick<IE2EEnterYourPasswordViewProps, 'navigation'>): StackNavigationOptions => ({
|
static navigationOptions = ({ navigation }: Pick<TE2EEnterYourPasswordViewProps, 'navigation'>): StackNavigationOptions => ({
|
||||||
headerLeft: () => <HeaderButton.CloseModal navigation={navigation} testID='e2e-enter-your-password-view-close' />,
|
headerLeft: () => <HeaderButton.CloseModal navigation={navigation} testID='e2e-enter-your-password-view-close' />,
|
||||||
title: I18n.t('Enter_Your_E2E_Password')
|
title: I18n.t('Enter_Your_E2E_Password')
|
||||||
});
|
});
|
||||||
|
|
||||||
constructor(props: IE2EEnterYourPasswordViewProps) {
|
constructor(props: TE2EEnterYourPasswordViewProps) {
|
||||||
super(props);
|
super(props);
|
||||||
this.state = {
|
this.state = {
|
||||||
password: ''
|
password: ''
|
||||||
|
@ -58,8 +54,8 @@ class E2EEnterYourPasswordView extends React.Component<IE2EEnterYourPasswordView
|
||||||
submit = () => {
|
submit = () => {
|
||||||
logEvent(events.E2E_ENTER_PW_SUBMIT);
|
logEvent(events.E2E_ENTER_PW_SUBMIT);
|
||||||
const { password } = this.state;
|
const { password } = this.state;
|
||||||
const { encryptionDecodeKey } = this.props;
|
const { dispatch } = this.props;
|
||||||
encryptionDecodeKey(password);
|
dispatch(encryptionDecodeKey(password));
|
||||||
};
|
};
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
|
@ -109,7 +105,4 @@ class E2EEnterYourPasswordView extends React.Component<IE2EEnterYourPasswordView
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const mapDispatchToProps = (dispatch: Dispatch) => ({
|
export default connect(null)(withTheme(E2EEnterYourPasswordView));
|
||||||
encryptionDecodeKey: (password: string) => dispatch(encryptionDecodeKeyAction(password))
|
|
||||||
});
|
|
||||||
export default connect(null, mapDispatchToProps)(withTheme(E2EEnterYourPasswordView));
|
|
||||||
|
|
|
@ -1,25 +1,24 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { StackNavigationProp } from '@react-navigation/stack';
|
|
||||||
import { Dispatch } from 'redux';
|
|
||||||
import { connect } from 'react-redux';
|
|
||||||
import { Clipboard, ScrollView, StyleSheet, Text, View } from 'react-native';
|
import { Clipboard, ScrollView, StyleSheet, Text, View } from 'react-native';
|
||||||
|
import { connect } from 'react-redux';
|
||||||
|
|
||||||
import { encryptionSetBanner as encryptionSetBannerAction } from '../actions/encryption';
|
import { encryptionSetBanner } from '../actions/encryption';
|
||||||
import { E2E_RANDOM_PASSWORD_KEY } from '../lib/encryption/constants';
|
import { themes } from '../constants/colors';
|
||||||
|
import Button from '../containers/Button';
|
||||||
import * as HeaderButton from '../containers/HeaderButton';
|
import * as HeaderButton from '../containers/HeaderButton';
|
||||||
import scrollPersistTaps from '../utils/scrollPersistTaps';
|
|
||||||
import SafeAreaView from '../containers/SafeAreaView';
|
import SafeAreaView from '../containers/SafeAreaView';
|
||||||
import UserPreferences from '../lib/userPreferences';
|
|
||||||
import { events, logEvent } from '../utils/log';
|
|
||||||
import StatusBar from '../containers/StatusBar';
|
import StatusBar from '../containers/StatusBar';
|
||||||
import { LISTENER } from '../containers/Toast';
|
import { LISTENER } from '../containers/Toast';
|
||||||
import { themes } from '../constants/colors';
|
import { IApplicationState, IBaseScreen } from '../definitions';
|
||||||
import EventEmitter from '../utils/events';
|
|
||||||
import Button from '../containers/Button';
|
|
||||||
import { withTheme } from '../theme';
|
|
||||||
import I18n from '../i18n';
|
import I18n from '../i18n';
|
||||||
import sharedStyles from './Styles';
|
import { E2E_RANDOM_PASSWORD_KEY } from '../lib/encryption/constants';
|
||||||
|
import UserPreferences from '../lib/userPreferences';
|
||||||
import { E2ESaveYourPasswordStackParamList } from '../stacks/types';
|
import { E2ESaveYourPasswordStackParamList } from '../stacks/types';
|
||||||
|
import { withTheme } from '../theme';
|
||||||
|
import EventEmitter from '../utils/events';
|
||||||
|
import { events, logEvent } from '../utils/log';
|
||||||
|
import scrollPersistTaps from '../utils/scrollPersistTaps';
|
||||||
|
import sharedStyles from './Styles';
|
||||||
|
|
||||||
const styles = StyleSheet.create({
|
const styles = StyleSheet.create({
|
||||||
container: {
|
container: {
|
||||||
|
@ -59,11 +58,8 @@ interface IE2ESaveYourPasswordViewState {
|
||||||
password: string;
|
password: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
interface IE2ESaveYourPasswordViewProps {
|
interface IE2ESaveYourPasswordViewProps extends IBaseScreen<E2ESaveYourPasswordStackParamList, 'E2ESaveYourPasswordView'> {
|
||||||
server: string;
|
server: string;
|
||||||
navigation: StackNavigationProp<E2ESaveYourPasswordStackParamList, 'E2ESaveYourPasswordView'>;
|
|
||||||
encryptionSetBanner(): void;
|
|
||||||
theme: string;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
class E2ESaveYourPasswordView extends React.Component<IE2ESaveYourPasswordViewProps, IE2ESaveYourPasswordViewState> {
|
class E2ESaveYourPasswordView extends React.Component<IE2ESaveYourPasswordViewProps, IE2ESaveYourPasswordViewState> {
|
||||||
|
@ -103,11 +99,11 @@ class E2ESaveYourPasswordView extends React.Component<IE2ESaveYourPasswordViewPr
|
||||||
|
|
||||||
onSaved = async () => {
|
onSaved = async () => {
|
||||||
logEvent(events.E2E_SAVE_PW_SAVED);
|
logEvent(events.E2E_SAVE_PW_SAVED);
|
||||||
const { navigation, server, encryptionSetBanner } = this.props;
|
const { navigation, server, dispatch } = this.props;
|
||||||
// Remove stored password
|
// Remove stored password
|
||||||
await UserPreferences.removeItem(`${server}-${E2E_RANDOM_PASSWORD_KEY}`);
|
await UserPreferences.removeItem(`${server}-${E2E_RANDOM_PASSWORD_KEY}`);
|
||||||
// Hide encryption banner
|
// Hide encryption banner
|
||||||
encryptionSetBanner();
|
dispatch(encryptionSetBanner());
|
||||||
navigation.pop();
|
navigation.pop();
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -173,10 +169,8 @@ class E2ESaveYourPasswordView extends React.Component<IE2ESaveYourPasswordViewPr
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const mapStateToProps = (state: any) => ({
|
const mapStateToProps = (state: IApplicationState) => ({
|
||||||
server: state.server.server
|
server: state.server.server
|
||||||
});
|
});
|
||||||
const mapDispatchToProps = (dispatch: Dispatch) => ({
|
|
||||||
encryptionSetBanner: () => dispatch(encryptionSetBannerAction())
|
export default connect(mapStateToProps)(withTheme(E2ESaveYourPasswordView));
|
||||||
});
|
|
||||||
export default connect(mapStateToProps, mapDispatchToProps)(withTheme(E2ESaveYourPasswordView));
|
|
||||||
|
|
Loading…
Reference in New Issue