223 lines
5.5 KiB
TypeScript
223 lines
5.5 KiB
TypeScript
// Copyright 2020 Signal Messenger, LLC
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
import type { ReadonlyDeep } from 'type-fest';
|
|
import type { ThunkAction } from 'redux-thunk';
|
|
|
|
import { generateSecurityNumberBlock } from '../../util/safetyNumber';
|
|
import type { ConversationType } from './conversations';
|
|
import {
|
|
reloadProfiles,
|
|
toggleVerification,
|
|
} from '../../shims/contactVerification';
|
|
import * as log from '../../logging/log';
|
|
import * as Errors from '../../types/errors';
|
|
import type { StateType as RootStateType } from '../reducer';
|
|
import { getSecurityNumberIdentifierType } from '../selectors/items';
|
|
|
|
export type SafetyNumberContactType = ReadonlyDeep<{
|
|
safetyNumber: string;
|
|
safetyNumberChanged?: boolean;
|
|
verificationDisabled: boolean;
|
|
}>;
|
|
|
|
export type SafetyNumberStateType = ReadonlyDeep<{
|
|
contacts: {
|
|
[key: string]: SafetyNumberContactType;
|
|
};
|
|
}>;
|
|
|
|
const GENERATE_FULFILLED = 'safetyNumber/GENERATE_FULFILLED';
|
|
const TOGGLE_VERIFIED_FULFILLED = 'safetyNumber/TOGGLE_VERIFIED_FULFILLED';
|
|
const TOGGLE_VERIFIED_PENDING = 'safetyNumber/TOGGLE_VERIFIED_PENDING';
|
|
|
|
type GenerateFulfilledActionType = ReadonlyDeep<{
|
|
type: 'safetyNumber/GENERATE_FULFILLED';
|
|
payload: {
|
|
contact: ConversationType;
|
|
safetyNumber: string;
|
|
};
|
|
}>;
|
|
|
|
type ToggleVerifiedPendingActionType = ReadonlyDeep<{
|
|
type: 'safetyNumber/TOGGLE_VERIFIED_PENDING';
|
|
payload: {
|
|
contact: ConversationType;
|
|
};
|
|
}>;
|
|
|
|
type ToggleVerifiedFulfilledActionType = ReadonlyDeep<{
|
|
type: 'safetyNumber/TOGGLE_VERIFIED_FULFILLED';
|
|
payload: {
|
|
contact: ConversationType;
|
|
safetyNumber?: string;
|
|
safetyNumberChanged?: boolean;
|
|
};
|
|
}>;
|
|
|
|
export type SafetyNumberActionType = ReadonlyDeep<
|
|
| GenerateFulfilledActionType
|
|
| ToggleVerifiedPendingActionType
|
|
| ToggleVerifiedFulfilledActionType
|
|
>;
|
|
|
|
function generate(
|
|
contact: ConversationType
|
|
): ThunkAction<void, RootStateType, unknown, GenerateFulfilledActionType> {
|
|
return async (dispatch, getState) => {
|
|
try {
|
|
const securityNumberBlock = await generateSecurityNumberBlock(
|
|
contact,
|
|
getSecurityNumberIdentifierType(getState(), { now: Date.now() })
|
|
);
|
|
dispatch({
|
|
type: GENERATE_FULFILLED,
|
|
payload: {
|
|
contact,
|
|
safetyNumber: securityNumberBlock.join(' '),
|
|
},
|
|
});
|
|
} catch (error) {
|
|
log.error(
|
|
'failed to generate security number:',
|
|
Errors.toLogFormat(error)
|
|
);
|
|
}
|
|
};
|
|
}
|
|
|
|
function toggleVerified(
|
|
contact: ConversationType
|
|
): ThunkAction<
|
|
void,
|
|
RootStateType,
|
|
unknown,
|
|
ToggleVerifiedPendingActionType | ToggleVerifiedFulfilledActionType
|
|
> {
|
|
return async (dispatch, getState) => {
|
|
dispatch({
|
|
type: TOGGLE_VERIFIED_PENDING,
|
|
payload: {
|
|
contact,
|
|
},
|
|
});
|
|
|
|
try {
|
|
await alterVerification(contact);
|
|
|
|
dispatch({
|
|
type: TOGGLE_VERIFIED_FULFILLED,
|
|
payload: {
|
|
contact,
|
|
},
|
|
});
|
|
} catch (err) {
|
|
if (err.name === 'OutgoingIdentityKeyError') {
|
|
await reloadProfiles(contact.id);
|
|
const securityNumberBlock = await generateSecurityNumberBlock(
|
|
contact,
|
|
getSecurityNumberIdentifierType(getState(), { now: Date.now() })
|
|
);
|
|
|
|
dispatch({
|
|
type: TOGGLE_VERIFIED_FULFILLED,
|
|
payload: {
|
|
contact,
|
|
safetyNumber: securityNumberBlock.join(' '),
|
|
safetyNumberChanged: true,
|
|
},
|
|
});
|
|
}
|
|
}
|
|
};
|
|
}
|
|
|
|
async function alterVerification(contact: ConversationType): Promise<void> {
|
|
try {
|
|
await toggleVerification(contact.id);
|
|
} catch (result) {
|
|
if (result instanceof Error) {
|
|
if (result.name === 'OutgoingIdentityKeyError') {
|
|
throw result;
|
|
} else {
|
|
log.error('failed to toggle verified:', Errors.toLogFormat(result));
|
|
}
|
|
} else {
|
|
const keyError = result.errors.find(
|
|
(error: Error) => error.name === 'OutgoingIdentityKeyError'
|
|
);
|
|
if (keyError) {
|
|
throw keyError;
|
|
} else {
|
|
result.errors.forEach((error: Error) => {
|
|
log.error('failed to toggle verified:', Errors.toLogFormat(error));
|
|
});
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
export const actions = {
|
|
generateSafetyNumber: generate,
|
|
toggleVerified,
|
|
};
|
|
|
|
export function getEmptyState(): SafetyNumberStateType {
|
|
return {
|
|
contacts: {},
|
|
};
|
|
}
|
|
|
|
export function reducer(
|
|
state: Readonly<SafetyNumberStateType> = getEmptyState(),
|
|
action: Readonly<SafetyNumberActionType>
|
|
): SafetyNumberStateType {
|
|
if (action.type === TOGGLE_VERIFIED_PENDING) {
|
|
const { contact } = action.payload;
|
|
const { id } = contact;
|
|
const record = state.contacts[id];
|
|
return {
|
|
contacts: {
|
|
...state.contacts,
|
|
[id]: {
|
|
...record,
|
|
safetyNumberChanged: false,
|
|
verificationDisabled: true,
|
|
},
|
|
},
|
|
};
|
|
}
|
|
|
|
if (action.type === TOGGLE_VERIFIED_FULFILLED) {
|
|
const { contact, ...restProps } = action.payload;
|
|
const { id } = contact;
|
|
const record = state.contacts[id];
|
|
return {
|
|
contacts: {
|
|
...state.contacts,
|
|
[id]: {
|
|
...record,
|
|
...restProps,
|
|
verificationDisabled: false,
|
|
},
|
|
},
|
|
};
|
|
}
|
|
|
|
if (action.type === GENERATE_FULFILLED) {
|
|
const { contact, safetyNumber } = action.payload;
|
|
const { id } = contact;
|
|
const record = state.contacts[id];
|
|
return {
|
|
contacts: {
|
|
...state.contacts,
|
|
[id]: {
|
|
...record,
|
|
safetyNumber,
|
|
},
|
|
},
|
|
};
|
|
}
|
|
|
|
return state;
|
|
}
|