// Copyright 2020 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import { assert } from 'chai'; import * as sinon from 'sinon'; import { v4 as generateUuid } from 'uuid'; import { times } from 'lodash'; import type { ReadonlyDeep } from 'type-fest'; import { reducer as rootReducer } from '../../../state/reducer'; import { noopAction } from '../../../state/ducks/noop'; import { ComposerStep, ConversationVerificationState, OneTimeModalState, } from '../../../state/ducks/conversationsEnums'; import type { CancelVerificationDataByConversationActionType, ConversationMessageType, ConversationType, ConversationsStateType, MessageType, TargetedConversationChangedActionType, ToggleConversationInChooseMembersActionType, MessageChangedActionType, ConversationsUpdatedActionType, } from '../../../state/ducks/conversations'; import { TARGETED_CONVERSATION_CHANGED, actions, cancelConversationVerification, clearCancelledConversationVerification, getConversationCallMode, getEmptyState, reducer, updateConversationLookups, } from '../../../state/ducks/conversations'; import { ReadStatus } from '../../../messages/MessageReadStatus'; import type { SingleServePromiseIdString } from '../../../services/singleServePromise'; import { CallMode } from '../../../types/CallDisposition'; import { type AciString, type PniString, generateAci, getAciFromPrefix, } from '../../../types/ServiceId'; import { generateStoryDistributionId } from '../../../types/StoryDistributionId'; import { getDefaultConversation, getDefaultConversationWithServiceId, getDefaultGroup, } from '../../../test-both/helpers/getDefaultConversation'; import { getDefaultAvatars } from '../../../types/Avatar'; import { defaultStartDirectConversationComposerState, defaultChooseGroupMembersComposerState, defaultSetGroupMetadataComposerState, } from '../../../test-both/helpers/defaultComposerStates'; import { updateRemoteConfig } from '../../../test-both/helpers/RemoteConfigStub'; import type { ShowSendAnywayDialogActionType } from '../../../state/ducks/globalModals'; import { SHOW_SEND_ANYWAY_DIALOG } from '../../../state/ducks/globalModals'; import type { StoryDistributionListsActionType } from '../../../state/ducks/storyDistributionLists'; import { DELETE_LIST, HIDE_MY_STORIES_FROM, MODIFY_LIST, VIEWERS_CHANGED, } from '../../../state/ducks/storyDistributionLists'; import { MY_STORY_ID } from '../../../types/Stories'; import type { ReadonlyMessageAttributesType } from '../../../model-types.d'; import { strictAssert } from '../../../util/assert'; const { clearGroupCreationError, clearInvitedServiceIdsForNewlyCreatedGroup, closeContactSpoofingReview, closeMaximumGroupSizeModal, closeRecommendedGroupSizeModal, conversationStoppedByMissingVerification, createGroup, discardMessages, repairNewestMessage, repairOldestMessage, resetAllChatColors, reviewConversationNameCollision, setComposeGroupAvatar, setComposeGroupName, setComposeSearchTerm, setPreJoinConversation, showArchivedConversations, showChooseGroupMembers, showConversation, showInbox, startComposing, startSettingGroupMetadata, toggleConversationInChooseMembers, } = actions; // can't use messageChanged action creator because it's a ThunkAction function messageChanged( messageId: string, conversationId: string, data: ReadonlyMessageAttributesType ): ReadonlyDeep { return { type: 'MESSAGE_CHANGED', payload: { id: messageId, conversationId, data, }, }; } describe('both/state/ducks/conversations', () => { const LIST_ID_1 = generateStoryDistributionId(); const LIST_ID_2 = generateStoryDistributionId(); const SERVICE_ID_1 = generateAci(); const SERVICE_ID_2 = generateAci(); const SERVICE_ID_3 = generateAci(); const SERVICE_ID_4 = generateAci(); const getEmptyRootState = () => rootReducer(undefined, noopAction()); let sinonSandbox: sinon.SinonSandbox; let createGroupStub: sinon.SinonStub; beforeEach(async () => { await window.ConversationController.load(); sinonSandbox = sinon.createSandbox(); sinonSandbox.stub(window.Whisper.events, 'trigger'); createGroupStub = sinon.stub(); }); afterEach(() => { sinonSandbox.restore(); }); describe('helpers', () => { describe('getConversationCallMode', () => { const fakeConversation: ConversationType = getDefaultConversation(); const fakeGroup: ConversationType = getDefaultGroup(); it("returns null if you've left the conversation", () => { assert.strictEqual( getConversationCallMode({ ...fakeConversation, left: true, }), null ); }); it("returns null if you've blocked the other person", () => { assert.strictEqual( getConversationCallMode({ ...fakeConversation, isBlocked: true, }), null ); }); it("returns null if you haven't accepted message requests", () => { assert.strictEqual( getConversationCallMode({ ...fakeConversation, acceptedMessageRequest: false, }), null ); }); it('returns null if the conversation is Note to Self', () => { assert.strictEqual( getConversationCallMode({ ...fakeConversation, isMe: true, }), null ); }); it('returns null for v1 groups', () => { assert.strictEqual( getConversationCallMode({ ...fakeGroup, groupVersion: 1, }), null ); assert.strictEqual( getConversationCallMode({ ...fakeGroup, groupVersion: undefined, }), null ); }); it('returns CallMode.Direct if the conversation is a normal direct conversation', () => { assert.strictEqual( getConversationCallMode(fakeConversation), CallMode.Direct ); }); it('returns CallMode.Group if the conversation is a v2 group', () => { assert.strictEqual( getConversationCallMode({ ...fakeGroup, groupVersion: 2, }), CallMode.Group ); }); }); describe('updateConversationLookups', () => { it('does not change lookups if no conversations provided', () => { const state = getEmptyState(); const result = updateConversationLookups(undefined, undefined, state); assert.strictEqual( state.conversationsByE164, result.conversationsByE164 ); assert.strictEqual( state.conversationsByServiceId, result.conversationsByServiceId ); assert.strictEqual( state.conversationsByGroupId, result.conversationsByGroupId ); }); it('adds and removes e164-only contact', () => { const removed = getDefaultConversation({ id: 'id-removed', e164: 'e164-removed', serviceId: undefined, }); const state = { ...getEmptyState(), conversationsByE164: { 'e164-removed': removed, }, }; const added = getDefaultConversation({ id: 'id-added', e164: 'e164-added', serviceId: undefined, }); const expected = { 'e164-added': added, }; const actual = updateConversationLookups([added], [removed], state); assert.deepEqual(actual.conversationsByE164, expected); assert.strictEqual( state.conversationsByServiceId, actual.conversationsByServiceId ); assert.strictEqual( state.conversationsByGroupId, actual.conversationsByGroupId ); }); it('adds and removes uuid-only contact', () => { const removed = getDefaultConversationWithServiceId({ id: 'id-removed', e164: undefined, }); const state = { ...getEmptyState(), conversationsByServiceId: { [removed.serviceId]: removed, }, }; const added = getDefaultConversationWithServiceId({ id: 'id-added', e164: undefined, }); const expected = { [added.serviceId]: added, }; const actual = updateConversationLookups([added], [removed], state); assert.strictEqual( state.conversationsByE164, actual.conversationsByE164 ); assert.deepEqual(actual.conversationsByServiceId, expected); assert.strictEqual( state.conversationsByGroupId, actual.conversationsByGroupId ); }); it('adds and removes groupId-only contact', () => { const removed = getDefaultConversation({ id: 'id-removed', groupId: 'groupId-removed', e164: undefined, serviceId: undefined, }); const state: ConversationsStateType = { ...getEmptyState(), conversationsByGroupId: { 'groupId-removed': removed, }, }; const added = getDefaultConversation({ id: 'id-added', groupId: 'groupId-added', e164: undefined, serviceId: undefined, }); const expected = { 'groupId-added': added, }; const actual = updateConversationLookups([added], [removed], state); assert.strictEqual( state.conversationsByE164, actual.conversationsByE164 ); assert.strictEqual( state.conversationsByServiceId, actual.conversationsByServiceId ); assert.deepEqual(actual.conversationsByGroupId, expected); }); it('adds and removes multiple conversations', () => { const removed = getDefaultConversation({ id: 'id-removed', groupId: 'groupId-removed', e164: 'e164-removed', serviceId: 'serviceId-removed' as unknown as AciString, pni: 'pni-removed' as unknown as PniString, username: 'username-removed', }); const stable = getDefaultConversation({ id: 'id-stable', groupId: 'groupId-stable', e164: 'e164-stable', serviceId: 'serviceId-stable' as unknown as AciString, pni: 'pni-stable' as unknown as PniString, username: 'username-stable', }); const state: ConversationsStateType = { ...getEmptyState(), conversationsByServiceId: { 'serviceId-removed': removed, 'serviceId-stable': stable, 'pni-removed': removed, 'pni-stable': stable, }, conversationsByE164: { 'e164-removed': removed, 'e164-stable': stable, }, conversationsByGroupId: { 'groupId-removed': removed, 'groupId-stable': stable, }, conversationsByUsername: { 'username-removed': removed, 'username-stable': stable, }, }; const added1 = getDefaultConversation({ id: 'id-added1', groupId: 'groupId-added1', e164: 'e164-added1', serviceId: 'serviceId-added1' as unknown as AciString, pni: 'pni-added1' as unknown as PniString, username: 'username-added1', }); const added2 = getDefaultConversation({ id: 'id-added2', groupId: 'groupId-added2', e164: undefined, serviceId: undefined, pni: undefined, username: undefined, }); const actual = { ...state, ...updateConversationLookups([added1, added2], [removed], state), }; const expected = { ...getEmptyState(), conversationsByServiceId: { 'serviceId-added1': added1, 'pni-added1': added1, 'serviceId-stable': stable, 'pni-stable': stable, }, conversationsByE164: { 'e164-added1': added1, 'e164-stable': stable, }, conversationsByGroupId: { 'groupId-added1': added1, 'groupId-stable': stable, 'groupId-added2': added2, }, conversationsByUsername: { 'username-added1': added1, 'username-stable': stable, }, }; assert.deepEqual(actual, expected); }); }); }); describe('reducer', () => { const time = Date.now(); const previousTime = time - 1; const conversationId = 'conversation-guid-1'; const messageId = 'message-guid-1'; const messageIdTwo = 'message-guid-2'; const messageIdThree = 'message-guid-3'; const sourceServiceId = generateAci(); function getDefaultMessage(id: string): MessageType { return { attachments: [], conversationId, id, received_at: previousTime, sent_at: previousTime, source: 'source', sourceServiceId, timestamp: previousTime, type: 'incoming' as const, readStatus: ReadStatus.Read, }; } function getDefaultConversationMessage(): ConversationMessageType { return { messageChangeCounter: 0, messageIds: [], metrics: { totalUnseen: 0, }, scrollToMessageCounter: 0, }; } describe('showConversation', () => { it('does not select a conversation if it does not exist', () => { const state = { ...getEmptyState(), }; const dispatch = sinon.spy(); showConversation({ conversationId: 'abc123' })( dispatch, getEmptyRootState, null ); const action = dispatch.getCall(0).args[0]; const nextState = reducer(state, action); assert.isUndefined(nextState.selectedConversationId); assert.isUndefined(nextState.targetedMessage); }); it('selects a conversation id', () => { const conversation = getDefaultConversation({ id: 'abc123', }); const state = { ...getEmptyState(), conversationLookup: { [conversation.id]: conversation, }, }; const dispatch = sinon.spy(); showConversation({ conversationId: 'abc123' })( dispatch, getEmptyRootState, null ); const action = dispatch.getCall(0).args[0]; const nextState = reducer(state, action); assert.equal(nextState.selectedConversationId, 'abc123'); assert.isUndefined(nextState.targetedMessage); }); it('selects a conversation and a message', () => { const conversation = getDefaultConversation({ id: 'abc123', }); const state = { ...getEmptyState(), conversationLookup: { [conversation.id]: conversation, }, }; const dispatch = sinon.spy(); showConversation({ conversationId: 'abc123', messageId: 'xyz987', })(dispatch, getEmptyRootState, null); const action = dispatch.getCall(0).args[0]; const nextState = reducer(state, action); assert.equal(nextState.selectedConversationId, 'abc123'); assert.equal(nextState.targetedMessage, 'xyz987'); }); describe('showConversation switchToAssociatedView=true', () => { let action: TargetedConversationChangedActionType; beforeEach(() => { const dispatch = sinon.spy(); showConversation({ conversationId: 'fake-conversation-id', switchToAssociatedView: true, })(dispatch, getEmptyRootState, null); [action] = dispatch.getCall(0).args; }); it('shows the inbox if the conversation is not archived', () => { const conversation = getDefaultConversation({ id: 'fake-conversation-id', }); const state = { ...getEmptyState(), conversationLookup: { [conversation.id]: conversation, }, }; const result = reducer(state, action); assert.isUndefined(result.composer); assert.isFalse(result.showArchived); }); it('shows the archive if the conversation is archived', () => { const conversation = getDefaultConversation({ id: 'fake-conversation-id', isArchived: true, }); const state = { ...getEmptyState(), conversationLookup: { [conversation.id]: conversation, }, }; const result = reducer(state, action); assert.isUndefined(result.composer); assert.isTrue(result.showArchived); }); }); }); describe('CLEAR_GROUP_CREATION_ERROR', () => { it('clears the group creation error', () => { const state = { ...getEmptyState(), composer: { ...defaultSetGroupMetadataComposerState, hasError: true as const, }, }; const action = clearGroupCreationError(); const result = reducer(state, action); assert( result.composer?.step === ComposerStep.SetGroupMetadata && result.composer.hasError === false ); }); }); describe('CLEAR_INVITED_UUIDS_FOR_NEWLY_CREATED_GROUP', () => { it('clears the list of invited conversation UUIDs', () => { const state = { ...getEmptyState(), invitedServiceIdsForNewlyCreatedGroup: [generateAci(), generateAci()], }; const action = clearInvitedServiceIdsForNewlyCreatedGroup(); const result = reducer(state, action); assert.isUndefined(result.invitedServiceIdsForNewlyCreatedGroup); }); }); describe('CLOSE_CONTACT_SPOOFING_REVIEW', () => { it('closes the contact spoofing review modal if it was open', () => { const state = { ...getEmptyState(), hasContactSpoofingReview: true, }; const action = closeContactSpoofingReview(); const actual = reducer(state, action); assert.isFalse(actual.hasContactSpoofingReview); }); it("does nothing if the modal wasn't already open", () => { const state = getEmptyState(); const action = closeContactSpoofingReview(); const actual = reducer(state, action); assert.deepEqual(actual, state); }); }); describe('CLOSE_MAXIMUM_GROUP_SIZE_MODAL', () => { it('closes the maximum group size modal if it was open', () => { const state = { ...getEmptyState(), composer: { ...defaultChooseGroupMembersComposerState, maximumGroupSizeModalState: OneTimeModalState.Showing, }, }; const action = closeMaximumGroupSizeModal(); const result = reducer(state, action); assert( result.composer?.step === ComposerStep.ChooseGroupMembers && result.composer.maximumGroupSizeModalState === OneTimeModalState.Shown, 'Expected the modal to be closed' ); }); it('does nothing if the maximum group size modal was never shown', () => { const state = { ...getEmptyState(), composer: defaultChooseGroupMembersComposerState, }; const action = closeMaximumGroupSizeModal(); const result = reducer(state, action); assert.strictEqual(result, state); }); it('does nothing if the maximum group size modal already closed', () => { const state = { ...getEmptyState(), composer: { ...defaultChooseGroupMembersComposerState, maximumGroupSizeModalState: OneTimeModalState.Shown, }, }; const action = closeMaximumGroupSizeModal(); const result = reducer(state, action); assert.strictEqual(result, state); }); }); describe('CLOSE_RECOMMENDED_GROUP_SIZE_MODAL', () => { it('closes the recommended group size modal if it was open', () => { const state = { ...getEmptyState(), composer: { ...defaultChooseGroupMembersComposerState, recommendedGroupSizeModalState: OneTimeModalState.Showing, }, }; const action = closeRecommendedGroupSizeModal(); const result = reducer(state, action); assert( result.composer?.step === ComposerStep.ChooseGroupMembers && result.composer.recommendedGroupSizeModalState === OneTimeModalState.Shown, 'Expected the modal to be closed' ); }); it('does nothing if the recommended group size modal was never shown', () => { const state = { ...getEmptyState(), composer: defaultChooseGroupMembersComposerState, }; const action = closeRecommendedGroupSizeModal(); const result = reducer(state, action); assert.strictEqual(result, state); }); it('does nothing if the recommended group size modal already closed', () => { const state = { ...getEmptyState(), composer: { ...defaultChooseGroupMembersComposerState, recommendedGroupSizeModalState: OneTimeModalState.Shown, }, }; const action = closeRecommendedGroupSizeModal(); const result = reducer(state, action); assert.strictEqual(result, state); }); }); describe('createGroup', () => { const conversationsState = { ...getEmptyState(), composer: { ...defaultSetGroupMetadataComposerState, selectedConversationIds: ['abc123'], groupName: 'Foo Bar Group', groupAvatar: new Uint8Array([1, 2, 3]), }, }; it('immediately dispatches a CREATE_GROUP_PENDING action, which puts the composer in a loading state', () => { const dispatch = sinon.spy(); createGroup()( dispatch, () => ({ ...getEmptyRootState(), conversations: conversationsState, }), null ); sinon.assert.calledOnce(dispatch); sinon.assert.calledWith(dispatch, { type: 'CREATE_GROUP_PENDING' }); const action = dispatch.getCall(0).args[0]; const result = reducer(conversationsState, action); assert( result.composer?.step === ComposerStep.SetGroupMetadata && result.composer.isCreating && !result.composer.hasError ); }); it('calls groups.createGroupV2', async () => { await createGroup(createGroupStub)( sinon.spy(), () => ({ ...getEmptyRootState(), conversations: conversationsState, }), null ); sinon.assert.calledOnce(createGroupStub); sinon.assert.calledWith(createGroupStub, { name: 'Foo Bar Group', avatar: new Uint8Array([1, 2, 3]), avatars: [], expireTimer: 0, conversationIds: ['abc123'], }); }); it("trims the group's title before calling groups.createGroupV2", async () => { await createGroup(createGroupStub)( sinon.spy(), () => ({ ...getEmptyRootState(), conversations: { ...conversationsState, composer: { ...conversationsState.composer, groupName: ' To Trim \t', }, }, }), null ); sinon.assert.calledWith( createGroupStub, sinon.match({ name: 'To Trim' }) ); }); it('dispatches a CREATE_GROUP_REJECTED action if group creation fails, which marks the state with an error', async () => { createGroupStub.rejects(new Error('uh oh')); const dispatch = sinon.spy(); const createGroupPromise = createGroup(createGroupStub)( dispatch, () => ({ ...getEmptyRootState(), conversations: conversationsState, }), null ); const pendingAction = dispatch.getCall(0).args[0]; const stateAfterPending = reducer(conversationsState, pendingAction); await createGroupPromise; sinon.assert.calledTwice(dispatch); sinon.assert.calledWith(dispatch, { type: 'CREATE_GROUP_REJECTED' }); const rejectedAction = dispatch.getCall(1).args[0]; const result = reducer(stateAfterPending, rejectedAction); assert( result.composer?.step === ComposerStep.SetGroupMetadata && !result.composer.isCreating && result.composer.hasError ); }); it("when rejecting, does nothing to the left pane if it's no longer in this composer state", async () => { createGroupStub.rejects(new Error('uh oh')); const dispatch = sinon.spy(); const createGroupPromise = createGroup(createGroupStub)( dispatch, () => ({ ...getEmptyRootState(), conversations: conversationsState, }), null ); await createGroupPromise; const state = getEmptyState(); const rejectedAction = dispatch.getCall(1).args[0]; const result = reducer(state, rejectedAction); assert.strictEqual(result, state); }); it('dispatches a CREATE_GROUP_FULFILLED event (which updates the newly-created conversation IDs), triggers a showConversation event and switches to the associated conversation on success', async () => { const abc = getAciFromPrefix('abc'); createGroupStub.resolves({ id: '9876', get: (key: string) => { if (key !== 'pendingMembersV2') { throw new Error('This getter is not set up for this test'); } return [{ serviceId: abc }]; }, }); const dispatch = sinon.spy(); await createGroup(createGroupStub)( dispatch, () => ({ ...getEmptyRootState(), conversations: conversationsState, }), null ); sinon.assert.calledWith(dispatch, { type: 'CREATE_GROUP_FULFILLED', payload: { invitedServiceIds: [abc] }, }); sinon.assert.calledWith(dispatch, { type: TARGETED_CONVERSATION_CHANGED, payload: { conversationId: '9876', messageId: undefined, switchToAssociatedView: true, }, }); const fulfilledAction = dispatch.getCall(1).args[0]; const result = reducer(conversationsState, fulfilledAction); assert.deepEqual(result.invitedServiceIdsForNewlyCreatedGroup, [abc]); }); }); describe('CONVERSATION_STOPPED_BY_MISSING_VERIFICATION', () => { it('adds to state, removing duplicates', () => { const first = reducer( getEmptyState(), conversationStoppedByMissingVerification({ conversationId: 'convo A', untrustedServiceIds: [SERVICE_ID_1], }) ); const second = reducer( first, conversationStoppedByMissingVerification({ conversationId: 'convo A', untrustedServiceIds: [SERVICE_ID_2], }) ); const third = reducer( second, conversationStoppedByMissingVerification({ conversationId: 'convo A', untrustedServiceIds: [SERVICE_ID_1, SERVICE_ID_3], }) ); assert.deepStrictEqual(third.verificationDataByConversation, { 'convo A': { type: ConversationVerificationState.PendingVerification, serviceIdsNeedingVerification: [ SERVICE_ID_1, SERVICE_ID_2, SERVICE_ID_3, ], }, }); }); it('stomps on VerificationCancelled state', () => { const state: ConversationsStateType = { ...getEmptyState(), verificationDataByConversation: { 'convo A': { type: ConversationVerificationState.VerificationCancelled, canceledAt: Date.now(), }, }, }; const actual = reducer( state, conversationStoppedByMissingVerification({ conversationId: 'convo A', untrustedServiceIds: [SERVICE_ID_1, SERVICE_ID_2], }) ); assert.deepStrictEqual(actual.verificationDataByConversation, { 'convo A': { type: ConversationVerificationState.PendingVerification, serviceIdsNeedingVerification: [SERVICE_ID_1, SERVICE_ID_2], }, }); }); }); describe('SHOW_SEND_ANYWAY_DIALOG', () => { it('adds nothing to existing empty state', () => { const state = getEmptyState(); const action: ShowSendAnywayDialogActionType = { type: SHOW_SEND_ANYWAY_DIALOG, payload: { untrustedByConversation: {}, promiseUuid: generateUuid() as SingleServePromiseIdString, source: undefined, }, }; const actual = reducer(state, action); assert.deepStrictEqual(actual.verificationDataByConversation, {}); }); it('adds multiple conversations and distribution lists to empty list', () => { const state = getEmptyState(); const action: ShowSendAnywayDialogActionType = { type: SHOW_SEND_ANYWAY_DIALOG, payload: { untrustedByConversation: { [LIST_ID_1]: { serviceIds: [SERVICE_ID_1, SERVICE_ID_2], byDistributionId: { [LIST_ID_1]: { serviceIds: [SERVICE_ID_1, SERVICE_ID_3], }, [LIST_ID_2]: { serviceIds: [SERVICE_ID_2, SERVICE_ID_4], }, }, }, }, promiseUuid: generateUuid() as SingleServePromiseIdString, source: undefined, }, }; const actual = reducer(state, action); assert.deepStrictEqual(actual.verificationDataByConversation, { [LIST_ID_1]: { type: ConversationVerificationState.PendingVerification, serviceIdsNeedingVerification: [SERVICE_ID_1, SERVICE_ID_2], byDistributionId: { [LIST_ID_1]: { serviceIdsNeedingVerification: [SERVICE_ID_1, SERVICE_ID_3], }, [LIST_ID_2]: { serviceIdsNeedingVerification: [SERVICE_ID_2, SERVICE_ID_4], }, }, }, }); }); it('adds and de-dupes in multiple conversations and distribution lists', () => { const state: ConversationsStateType = { ...getEmptyState(), verificationDataByConversation: { [LIST_ID_1]: { type: ConversationVerificationState.PendingVerification, serviceIdsNeedingVerification: [SERVICE_ID_1], byDistributionId: { [LIST_ID_1]: { serviceIdsNeedingVerification: [SERVICE_ID_1], }, }, }, }, }; const action: ShowSendAnywayDialogActionType = { type: SHOW_SEND_ANYWAY_DIALOG, payload: { untrustedByConversation: { [LIST_ID_1]: { serviceIds: [SERVICE_ID_1, SERVICE_ID_2], byDistributionId: { [LIST_ID_1]: { serviceIds: [SERVICE_ID_1, SERVICE_ID_3], }, [LIST_ID_2]: { serviceIds: [SERVICE_ID_2, SERVICE_ID_4], }, }, }, }, promiseUuid: generateUuid() as SingleServePromiseIdString, source: undefined, }, }; const actual = reducer(state, action); assert.deepStrictEqual(actual.verificationDataByConversation, { [LIST_ID_1]: { type: ConversationVerificationState.PendingVerification, serviceIdsNeedingVerification: [SERVICE_ID_1, SERVICE_ID_2], byDistributionId: { [LIST_ID_1]: { serviceIdsNeedingVerification: [SERVICE_ID_1, SERVICE_ID_3], }, [LIST_ID_2]: { serviceIdsNeedingVerification: [SERVICE_ID_2, SERVICE_ID_4], }, }, }, }); }); }); describe('CANCEL_CONVERSATION_PENDING_VERIFICATION', () => { function getAction( timestamp: number, conversationsState: ConversationsStateType ): CancelVerificationDataByConversationActionType { const dispatch = sinon.spy(); cancelConversationVerification(timestamp)( dispatch, () => ({ ...getEmptyRootState(), conversations: conversationsState, }), null ); return dispatch.getCall(0).args[0]; } it('replaces existing PendingVerification state', () => { const now = Date.now(); const state: ConversationsStateType = { ...getEmptyState(), verificationDataByConversation: { 'convo A': { type: ConversationVerificationState.PendingVerification, serviceIdsNeedingVerification: [SERVICE_ID_1, SERVICE_ID_2], }, }, }; const action = getAction(now, state); const actual = reducer(state, action); assert.deepStrictEqual(actual.verificationDataByConversation, { 'convo A': { type: ConversationVerificationState.VerificationCancelled, canceledAt: now, }, }); }); it('updates timestamp for existing VerificationCancelled state', () => { const now = Date.now(); const state: ConversationsStateType = { ...getEmptyState(), verificationDataByConversation: { 'convo A': { type: ConversationVerificationState.VerificationCancelled, canceledAt: now - 1, }, }, }; const action = getAction(now, state); const actual = reducer(state, action); assert.deepStrictEqual(actual.verificationDataByConversation, { 'convo A': { type: ConversationVerificationState.VerificationCancelled, canceledAt: now, }, }); }); it('uses newest timestamp when updating existing VerificationCancelled state', () => { const now = Date.now(); const state: ConversationsStateType = { ...getEmptyState(), verificationDataByConversation: { 'convo A': { type: ConversationVerificationState.VerificationCancelled, canceledAt: now, }, }, }; const action = getAction(now, state); const actual = reducer(state, action); assert.deepStrictEqual(actual.verificationDataByConversation, { 'convo A': { type: ConversationVerificationState.VerificationCancelled, canceledAt: now, }, }); }); it('does nothing if no existing state', () => { const state: ConversationsStateType = getEmptyState(); const action = getAction(Date.now(), state); const actual = reducer(state, action); assert.strictEqual(actual, state); }); }); describe('CANCEL_CONVERSATION_PENDING_VERIFICATION', () => { it('removes existing VerificationCancelled state', () => { const now = Date.now(); const state: ConversationsStateType = { ...getEmptyState(), verificationDataByConversation: { 'convo A': { type: ConversationVerificationState.VerificationCancelled, canceledAt: now, }, }, }; const actual = reducer( state, clearCancelledConversationVerification('convo A') ); assert.deepStrictEqual(actual.verificationDataByConversation, {}); }); it('leaves existing PendingVerification state', () => { const state: ConversationsStateType = { ...getEmptyState(), verificationDataByConversation: { 'convo A': { type: ConversationVerificationState.PendingVerification, serviceIdsNeedingVerification: [SERVICE_ID_1, SERVICE_ID_2], }, }, }; const actual = reducer( state, clearCancelledConversationVerification('convo A') ); assert.deepStrictEqual(actual, state); }); it('does nothing with empty state', () => { const state: ConversationsStateType = getEmptyState(); const actual = reducer( state, clearCancelledConversationVerification('convo A') ); assert.deepStrictEqual(actual, state); }); }); describe('REPAIR_NEWEST_MESSAGE', () => { it('updates newest', () => { const action = repairNewestMessage(conversationId); const state: ConversationsStateType = { ...getEmptyState(), messagesLookup: { [messageId]: { ...getDefaultMessage(messageId), received_at: time, sent_at: time, }, }, messagesByConversation: { [conversationId]: { ...getDefaultConversationMessage(), messageIds: [messageIdThree, messageIdTwo, messageId], metrics: { totalUnseen: 0, }, }, }, }; const expected: ConversationsStateType = { ...getEmptyState(), messagesLookup: { [messageId]: { ...getDefaultMessage(messageId), received_at: time, sent_at: time, }, }, messagesByConversation: { [conversationId]: { ...getDefaultConversationMessage(), messageIds: [messageIdThree, messageIdTwo, messageId], metrics: { totalUnseen: 0, newest: { id: messageId, received_at: time, sent_at: time, }, }, }, }, }; const actual = reducer(state, action); assert.deepEqual(actual, expected); }); it('clears newest', () => { const action = repairNewestMessage(conversationId); const state: ConversationsStateType = { ...getEmptyState(), messagesLookup: { [messageId]: { ...getDefaultMessage(messageId), received_at: time, }, }, messagesByConversation: { [conversationId]: { ...getDefaultConversationMessage(), messageIds: [], metrics: { totalUnseen: 0, newest: { id: messageId, received_at: time, }, }, }, }, }; const expected: ConversationsStateType = { ...getEmptyState(), messagesLookup: { [messageId]: { ...getDefaultMessage(messageId), received_at: time, }, }, messagesByConversation: { [conversationId]: { ...getDefaultConversationMessage(), messageIds: [], metrics: { newest: undefined, totalUnseen: 0, }, }, }, }; const actual = reducer(state, action); assert.deepEqual(actual, expected); }); it('returns state if conversation not present', () => { const action = repairNewestMessage(conversationId); const state: ConversationsStateType = getEmptyState(); const actual = reducer(state, action); assert.equal(actual, state); }); }); describe('REPAIR_OLDEST_MESSAGE', () => { it('updates oldest', () => { const action = repairOldestMessage(conversationId); const state: ConversationsStateType = { ...getEmptyState(), messagesLookup: { [messageId]: { ...getDefaultMessage(messageId), received_at: time, sent_at: time, }, }, messagesByConversation: { [conversationId]: { ...getDefaultConversationMessage(), messageIds: [messageId, messageIdTwo, messageIdThree], metrics: { totalUnseen: 0, }, }, }, }; const expected: ConversationsStateType = { ...getEmptyState(), messagesLookup: { [messageId]: { ...getDefaultMessage(messageId), received_at: time, sent_at: time, }, }, messagesByConversation: { [conversationId]: { ...getDefaultConversationMessage(), messageIds: [messageId, messageIdTwo, messageIdThree], metrics: { totalUnseen: 0, oldest: { id: messageId, received_at: time, sent_at: time, }, }, }, }, }; const actual = reducer(state, action); assert.deepEqual(actual, expected); }); it('clears oldest', () => { const action = repairOldestMessage(conversationId); const state: ConversationsStateType = { ...getEmptyState(), messagesLookup: { [messageId]: { ...getDefaultMessage(messageId), received_at: time, }, }, messagesByConversation: { [conversationId]: { ...getDefaultConversationMessage(), messageIds: [], metrics: { totalUnseen: 0, oldest: { id: messageId, received_at: time, }, }, }, }, }; const expected: ConversationsStateType = { ...getEmptyState(), messagesLookup: { [messageId]: { ...getDefaultMessage(messageId), received_at: time, }, }, messagesByConversation: { [conversationId]: { ...getDefaultConversationMessage(), messageIds: [], metrics: { oldest: undefined, totalUnseen: 0, }, }, }, }; const actual = reducer(state, action); assert.deepEqual(actual, expected); }); it('returns state if conversation not present', () => { const action = repairOldestMessage(conversationId); const state: ConversationsStateType = getEmptyState(); const actual = reducer(state, action); assert.equal(actual, state); }); }); describe('REVIEW_CONVERSATION_NAME_COLLISION', () => { it('starts reviewing a name collision', () => { const state = getEmptyState(); const action = reviewConversationNameCollision(); const actual = reducer(state, action); assert.isTrue(actual.hasContactSpoofingReview); }); }); describe('SET_COMPOSE_GROUP_AVATAR', () => { it("can clear the composer's group avatar", () => { const state = { ...getEmptyState(), composer: { ...defaultSetGroupMetadataComposerState, groupAvatar: new Uint8Array(2), }, }; const action = setComposeGroupAvatar(undefined); const result = reducer(state, action); assert( result.composer?.step === ComposerStep.SetGroupMetadata && result.composer.groupAvatar === undefined ); }); it("can set the composer's group avatar", () => { const avatar = new Uint8Array([1, 2, 3]); const state = { ...getEmptyState(), composer: defaultSetGroupMetadataComposerState, }; const action = setComposeGroupAvatar(avatar); const result = reducer(state, action); assert( result.composer?.step === ComposerStep.SetGroupMetadata && result.composer.groupAvatar === avatar ); }); }); describe('SET_COMPOSE_GROUP_NAME', () => { it("can set the composer's group name", () => { const state = { ...getEmptyState(), composer: defaultSetGroupMetadataComposerState, }; const action = setComposeGroupName('bing bong'); const result = reducer(state, action); assert( result.composer?.step === ComposerStep.SetGroupMetadata && result.composer.groupName === 'bing bong' ); }); }); describe('SET_COMPOSE_SEARCH_TERM', () => { it('updates the contact search term', () => { const state = { ...getEmptyState(), composer: defaultStartDirectConversationComposerState, }; const result = reducer(state, setComposeSearchTerm('foo bar')); assert.deepEqual(result.composer, { ...defaultStartDirectConversationComposerState, searchTerm: 'foo bar', }); }); }); describe('DISCARD_MESSAGES', () => { const startState: ConversationsStateType = { ...getEmptyState(), messagesLookup: { [messageId]: getDefaultMessage(messageId), [messageIdTwo]: getDefaultMessage(messageIdTwo), [messageIdThree]: getDefaultMessage(messageIdThree), }, messagesByConversation: { [conversationId]: { messageChangeCounter: 0, metrics: { totalUnseen: 0, }, scrollToMessageCounter: 0, messageIds: [messageId, messageIdTwo, messageIdThree], }, }, }; it('eliminates older messages', () => { const toDiscard = { conversationId, numberToKeepAtBottom: 2, }; const state = reducer(startState, discardMessages(toDiscard)); assert.deepEqual( state.messagesByConversation[conversationId]?.messageIds, [messageIdTwo, messageIdThree] ); }); it('eliminates newer messages', () => { const toDiscard = { conversationId, numberToKeepAtTop: 2, }; const state = reducer(startState, discardMessages(toDiscard)); assert.deepEqual( state.messagesByConversation[conversationId]?.messageIds, [messageId, messageIdTwo] ); }); }); describe('SET_PRE_JOIN_CONVERSATION', () => { const startState = { ...getEmptyState(), }; it('starts with empty value', () => { assert.isUndefined(startState.preJoinConversation); }); it('sets value as provided', () => { const preJoinConversation = { title: 'Pre-join group!', memberCount: 4, approvalRequired: false, }; const stateWithData = reducer( startState, setPreJoinConversation(preJoinConversation) ); assert.deepEqual( stateWithData.preJoinConversation, preJoinConversation ); const resetState = reducer( stateWithData, setPreJoinConversation(undefined) ); assert.isUndefined(resetState.preJoinConversation); }); }); describe('MESSAGE_CHANGED', () => { const startState: ConversationsStateType = { ...getEmptyState(), conversationLookup: { [conversationId]: { ...getDefaultConversation(), id: conversationId, groupVersion: 2, groupId: 'dGhpc2lzYWdyb3VwaWR0aGlzaXNhZ3JvdXBpZHRoaXM=', }, }, messagesByConversation: { [conversationId]: { messageChangeCounter: 0, messageIds: [messageId, messageIdTwo, messageIdThree], metrics: { totalUnseen: 0, }, scrollToMessageCounter: 0, }, }, messagesLookup: { [messageId]: { ...getDefaultMessage(messageId), displayLimit: undefined, }, [messageIdTwo]: { ...getDefaultMessage(messageIdTwo), displayLimit: undefined, }, [messageIdThree]: { ...getDefaultMessage(messageIdThree), displayLimit: undefined, }, }, }; const changedMessage = { ...getDefaultMessage(messageId), body: 'changed', displayLimit: undefined, isSpoilerExpanded: undefined, }; it('updates message data', () => { const state = reducer( startState, messageChanged(messageId, conversationId, changedMessage) ); assert.deepEqual(state.messagesLookup[messageId], changedMessage); assert.strictEqual( state.messagesByConversation[conversationId]?.messageChangeCounter, 0 ); }); it('does not update lookup if it is a story reply', () => { const state = reducer( startState, messageChanged(messageId, conversationId, { ...changedMessage, storyId: 'story-id', }) ); assert.deepEqual( state.messagesLookup[messageId], startState.messagesLookup[messageId] ); assert.strictEqual( state.messagesByConversation[conversationId]?.messageChangeCounter, 0 ); }); }); describe('SHOW_ARCHIVED_CONVERSATIONS', () => { it('is a no-op when already at the archive', () => { const state = { ...getEmptyState(), showArchived: true, }; const action = showArchivedConversations(); const result = reducer(state, action); assert.isTrue(result.showArchived); assert.isUndefined(result.composer); }); it('switches from the inbox to the archive', () => { const state = getEmptyState(); const action = showArchivedConversations(); const result = reducer(state, action); assert.isTrue(result.showArchived); assert.isUndefined(result.composer); }); it('switches from the composer to the archive', () => { const state = { ...getEmptyState(), composer: defaultStartDirectConversationComposerState, }; const action = showArchivedConversations(); const result = reducer(state, action); assert.isTrue(result.showArchived); assert.isUndefined(result.composer); }); }); describe('SHOW_INBOX', () => { it('is a no-op when already at the inbox', () => { const state = getEmptyState(); const action = showInbox(); const result = reducer(state, action); assert.isFalse(result.showArchived); assert.isUndefined(result.composer); }); it('switches from the archive to the inbox', () => { const state = { ...getEmptyState(), showArchived: true, }; const action = showInbox(); const result = reducer(state, action); assert.isFalse(result.showArchived); assert.isUndefined(result.composer); }); it('switches from the composer to the inbox', () => { const state = { ...getEmptyState(), composer: defaultStartDirectConversationComposerState, }; const action = showInbox(); const result = reducer(state, action); assert.isFalse(result.showArchived); assert.isUndefined(result.composer); }); }); describe('START_COMPOSING', () => { it('does nothing if on the first step of the composer', () => { const state = { ...getEmptyState(), composer: defaultStartDirectConversationComposerState, }; const action = startComposing(); const result = reducer(state, action); assert.isFalse(result.showArchived); assert.deepEqual( result.composer, defaultStartDirectConversationComposerState ); }); it('if on the second step of the composer, goes back to the first step, clearing the search term', () => { const state = { ...getEmptyState(), composer: { ...defaultChooseGroupMembersComposerState, searchTerm: 'to be cleared', }, }; const action = startComposing(); const result = reducer(state, action); assert.isFalse(result.showArchived); assert.deepEqual( result.composer, defaultStartDirectConversationComposerState ); }); it('if on the third step of the composer, goes back to the first step, clearing everything', () => { const state = { ...getEmptyState(), composer: defaultSetGroupMetadataComposerState, }; const action = startComposing(); const result = reducer(state, action); assert.isFalse(result.showArchived); assert.deepEqual( result.composer, defaultStartDirectConversationComposerState ); }); it('switches from the inbox to the composer', () => { const state = getEmptyState(); const action = startComposing(); const result = reducer(state, action); assert.isFalse(result.showArchived); assert.deepEqual( result.composer, defaultStartDirectConversationComposerState ); }); it('switches from the archive to the inbox', () => { const state = { ...getEmptyState(), showArchived: true, }; const action = startComposing(); const result = reducer(state, action); assert.isFalse(result.showArchived); assert.deepEqual( result.composer, defaultStartDirectConversationComposerState ); }); }); describe('SHOW_CHOOSE_GROUP_MEMBERS', () => { it('switches to the second step of the composer if on the first step', () => { const state = { ...getEmptyState(), composer: { ...defaultStartDirectConversationComposerState, searchTerm: 'to be cleared', }, }; const action = showChooseGroupMembers(); const result = reducer(state, action); assert.isFalse(result.showArchived); assert.deepEqual(result.composer, { ...defaultChooseGroupMembersComposerState, userAvatarData: getDefaultAvatars(true), }); }); it('does nothing if already on the second step of the composer', () => { const state = { ...getEmptyState(), composer: defaultChooseGroupMembersComposerState, }; const action = showChooseGroupMembers(); const result = reducer(state, action); assert.strictEqual(result, state); }); it('returns to the second step if on the third step of the composer', () => { const state = { ...getEmptyState(), composer: { ...defaultSetGroupMetadataComposerState, groupName: 'Foo Bar Group', groupAvatar: new Uint8Array([4, 2]), }, }; const action = showChooseGroupMembers(); const result = reducer(state, action); assert.isFalse(result.showArchived); assert.deepEqual(result.composer, { ...defaultChooseGroupMembersComposerState, groupName: 'Foo Bar Group', groupAvatar: new Uint8Array([4, 2]), }); }); it('switches from the inbox to the second step of the composer', () => { const state = getEmptyState(); const action = showChooseGroupMembers(); const result = reducer(state, action); assert.isFalse(result.showArchived); assert.deepEqual(result.composer, { ...defaultChooseGroupMembersComposerState, userAvatarData: getDefaultAvatars(true), }); }); it('switches from the archive to the second step of the composer', () => { const state = { ...getEmptyState(), showArchived: true, }; const action = showChooseGroupMembers(); const result = reducer(state, action); assert.isFalse(result.showArchived); assert.deepEqual(result.composer, { ...defaultChooseGroupMembersComposerState, userAvatarData: getDefaultAvatars(true), }); }); }); describe('START_SETTING_GROUP_METADATA', () => { it('moves from the second to the third step of the composer', () => { const state = { ...getEmptyState(), composer: { ...defaultChooseGroupMembersComposerState, selectedConversationIds: ['abc', 'def'], }, }; const action = startSettingGroupMetadata(); const result = reducer(state, action); assert.deepEqual(result.composer, { ...defaultSetGroupMetadataComposerState, selectedConversationIds: ['abc', 'def'], }); }); it('maintains state when going from the second to third steps of the composer, if the second step already had some data (likely from a previous visit)', () => { const state = { ...getEmptyState(), composer: { ...defaultChooseGroupMembersComposerState, searchTerm: 'foo bar', selectedConversationIds: ['abc', 'def'], groupName: 'Foo Bar Group', groupAvatar: new Uint8Array([6, 9]), }, }; const action = startSettingGroupMetadata(); const result = reducer(state, action); assert.deepEqual(result.composer, { ...defaultSetGroupMetadataComposerState, selectedConversationIds: ['abc', 'def'], groupName: 'Foo Bar Group', groupAvatar: new Uint8Array([6, 9]), }); }); it('does nothing if already on the third step of the composer', () => { const state = { ...getEmptyState(), composer: defaultSetGroupMetadataComposerState, }; const action = startSettingGroupMetadata(); const result = reducer(state, action); assert.strictEqual(result, state); }); }); describe('TOGGLE_CONVERSATION_IN_CHOOSE_MEMBERS', () => { function getAction( id: string, conversationsState: ConversationsStateType ): ToggleConversationInChooseMembersActionType { const dispatch = sinon.spy(); toggleConversationInChooseMembers(id)( dispatch, () => ({ ...getEmptyRootState(), conversations: conversationsState, }), null ); return dispatch.getCall(0).args[0]; } beforeEach(async () => { await updateRemoteConfig([ { name: 'global.groupsv2.maxGroupSize', value: '22', enabled: true }, { name: 'global.groupsv2.groupSizeHardLimit', value: '33', enabled: true, }, ]); }); it('adds conversation IDs to the list', () => { const zero = { ...getEmptyState(), composer: defaultChooseGroupMembersComposerState, }; const one = reducer(zero, getAction('abc', zero)); const two = reducer(one, getAction('def', one)); assert.deepEqual(two.composer, { ...defaultChooseGroupMembersComposerState, selectedConversationIds: ['abc', 'def'], }); }); it('removes conversation IDs from the list', () => { const state = { ...getEmptyState(), composer: { ...defaultChooseGroupMembersComposerState, selectedConversationIds: ['abc', 'def'], }, }; const action = getAction('abc', state); const result = reducer(state, action); assert.deepEqual(result.composer, { ...defaultChooseGroupMembersComposerState, selectedConversationIds: ['def'], }); }); it('shows the recommended group size modal when first crossing the maximum recommended group size', () => { const oldSelectedConversationIds = times(21, () => generateUuid()); const newUuid = generateUuid(); const state = { ...getEmptyState(), composer: { ...defaultChooseGroupMembersComposerState, selectedConversationIds: oldSelectedConversationIds, }, }; const action = getAction(newUuid, state); const result = reducer(state, action); assert.deepEqual(result.composer, { ...defaultChooseGroupMembersComposerState, selectedConversationIds: [...oldSelectedConversationIds, newUuid], recommendedGroupSizeModalState: OneTimeModalState.Showing, }); }); it("doesn't show the recommended group size modal twice", () => { const oldSelectedConversationIds = times(21, () => generateUuid()); const newUuid = generateUuid(); const state = { ...getEmptyState(), composer: { ...defaultChooseGroupMembersComposerState, selectedConversationIds: oldSelectedConversationIds, recommendedGroupSizeModalState: OneTimeModalState.Shown, }, }; const action = getAction(newUuid, state); const result = reducer(state, action); assert.deepEqual(result.composer, { ...defaultChooseGroupMembersComposerState, selectedConversationIds: [...oldSelectedConversationIds, newUuid], recommendedGroupSizeModalState: OneTimeModalState.Shown, }); }); it('defaults the maximum recommended size to 151', async () => { for (const value of [null, 'xyz']) { // eslint-disable-next-line no-await-in-loop await updateRemoteConfig([ { name: 'global.groupsv2.maxGroupSize', value, enabled: true, }, { name: 'global.groupsv2.groupSizeHardLimit', value: '33', enabled: true, }, ]); const state = { ...getEmptyState(), composer: defaultChooseGroupMembersComposerState, }; const action = getAction(generateUuid(), state); assert.strictEqual(action.payload.maxRecommendedGroupSize, 151); } }); it('shows the maximum group size modal when first reaching the maximum group size', () => { const oldSelectedConversationIds = times(31, () => generateUuid()); const newUuid = generateUuid(); const state = { ...getEmptyState(), composer: { ...defaultChooseGroupMembersComposerState, selectedConversationIds: oldSelectedConversationIds, recommendedGroupSizeModalState: OneTimeModalState.Shown, maximumGroupSizeModalState: OneTimeModalState.NeverShown, }, }; const action = getAction(newUuid, state); const result = reducer(state, action); assert.deepEqual(result.composer, { ...defaultChooseGroupMembersComposerState, selectedConversationIds: [...oldSelectedConversationIds, newUuid], recommendedGroupSizeModalState: OneTimeModalState.Shown, maximumGroupSizeModalState: OneTimeModalState.Showing, }); }); it("doesn't show the maximum group size modal twice", () => { const oldSelectedConversationIds = times(31, () => generateUuid()); const newUuid = generateUuid(); const state = { ...getEmptyState(), composer: { ...defaultChooseGroupMembersComposerState, selectedConversationIds: oldSelectedConversationIds, recommendedGroupSizeModalState: OneTimeModalState.Shown, maximumGroupSizeModalState: OneTimeModalState.Shown, }, }; const action = getAction(newUuid, state); const result = reducer(state, action); assert.deepEqual(result.composer, { ...defaultChooseGroupMembersComposerState, selectedConversationIds: [...oldSelectedConversationIds, newUuid], recommendedGroupSizeModalState: OneTimeModalState.Shown, maximumGroupSizeModalState: OneTimeModalState.Shown, }); }); it('cannot select more than the maximum number of conversations', () => { const state = { ...getEmptyState(), composer: { ...defaultChooseGroupMembersComposerState, selectedConversationIds: times(1000, () => generateUuid()), }, }; const action = getAction(generateUuid(), state); const result = reducer(state, action); assert.deepEqual(result, state); }); it('defaults the maximum group size to 1001 if the recommended maximum is smaller', async () => { for (const value of [null, 'xyz']) { // eslint-disable-next-line no-await-in-loop await updateRemoteConfig([ { name: 'global.groupsv2.maxGroupSize', value: '2', enabled: true }, { name: 'global.groupsv2.groupSizeHardLimit', value, enabled: true, }, ]); const state = { ...getEmptyState(), composer: defaultChooseGroupMembersComposerState, }; const action = getAction(generateUuid(), state); assert.strictEqual(action.payload.maxGroupSize, 1001); } }); it('defaults the maximum group size to (recommended maximum + 1) if the recommended maximum is more than 1001', async () => { await updateRemoteConfig([ { name: 'global.groupsv2.maxGroupSize', value: '1234', enabled: true, }, { name: 'global.groupsv2.groupSizeHardLimit', value: '2', enabled: true, }, ]); const state = { ...getEmptyState(), composer: defaultChooseGroupMembersComposerState, }; const action = getAction(generateUuid(), state); assert.strictEqual(action.payload.maxGroupSize, 1235); }); }); describe('COLORS_CHANGED', () => { const abc = getDefaultConversationWithServiceId({ id: 'abc', conversationColor: 'wintergreen', }); const def = getDefaultConversationWithServiceId({ id: 'def', conversationColor: 'infrared', }); const ghi = getDefaultConversation({ id: 'ghi', e164: 'ghi', conversationColor: 'ember', }); const jkl = getDefaultConversation({ id: 'jkl', groupId: 'jkl', conversationColor: 'plum', }); const getState = () => ({ ...getEmptyRootState(), conversations: { ...getEmptyState(), conversationLookup: { abc, def, ghi, jkl, }, conversationsByServiceId: { abc, def, }, conversationsByE164: { ghi, }, conversationsByGroupId: { jkl, }, }, }); it('resetAllChatColors', async () => { const dispatch = sinon.spy(); await resetAllChatColors()(dispatch, getState, null); const [action] = dispatch.getCall(0).args; const nextState = reducer(getState().conversations, action); sinon.assert.calledOnce(dispatch); assert.isUndefined(nextState.conversationLookup.abc.conversationColor); assert.isUndefined(nextState.conversationLookup.def.conversationColor); assert.isUndefined(nextState.conversationLookup.ghi.conversationColor); assert.isUndefined(nextState.conversationLookup.jkl.conversationColor); assert.isUndefined( nextState.conversationsByServiceId[abc.serviceId].conversationColor ); assert.isUndefined( nextState.conversationsByServiceId[def.serviceId].conversationColor ); assert.isUndefined(nextState.conversationsByE164.ghi.conversationColor); assert.isUndefined( nextState.conversationsByGroupId.jkl.conversationColor ); await window.storage.remove('defaultConversationColor'); }); }); // When distribution lists change describe('VIEWERS_CHANGED', () => { const state: ConversationsStateType = { ...getEmptyState(), verificationDataByConversation: { convo1: { type: ConversationVerificationState.PendingVerification, serviceIdsNeedingVerification: [], byDistributionId: { [LIST_ID_1]: { serviceIdsNeedingVerification: [ SERVICE_ID_1, SERVICE_ID_2, SERVICE_ID_3, ], }, [LIST_ID_2]: { serviceIdsNeedingVerification: [SERVICE_ID_3], }, }, }, }, }; it('removes uuids now missing from the list', async () => { const action: StoryDistributionListsActionType = { type: VIEWERS_CHANGED, payload: { listId: LIST_ID_1, memberServiceIds: [SERVICE_ID_1, SERVICE_ID_2], }, }; const actual = reducer(state, action); assert.deepEqual(actual.verificationDataByConversation, { convo1: { type: ConversationVerificationState.PendingVerification, serviceIdsNeedingVerification: [], byDistributionId: { [LIST_ID_1]: { serviceIdsNeedingVerification: [SERVICE_ID_1, SERVICE_ID_2], }, [LIST_ID_2]: { serviceIdsNeedingVerification: [SERVICE_ID_3], }, }, }, }); }); it('removes now-empty list', async () => { const action: StoryDistributionListsActionType = { type: VIEWERS_CHANGED, payload: { listId: LIST_ID_1, memberServiceIds: [], }, }; const actual = reducer(state, action); assert.deepEqual(actual.verificationDataByConversation, { convo1: { type: ConversationVerificationState.PendingVerification, serviceIdsNeedingVerification: [], byDistributionId: { [LIST_ID_2]: { serviceIdsNeedingVerification: [SERVICE_ID_3], }, }, }, }); }); }); describe('HIDE_MY_STORIES_FROM', () => { const state: ConversationsStateType = { ...getEmptyState(), verificationDataByConversation: { convo1: { type: ConversationVerificationState.PendingVerification, serviceIdsNeedingVerification: [], byDistributionId: { [MY_STORY_ID]: { serviceIdsNeedingVerification: [ SERVICE_ID_1, SERVICE_ID_2, SERVICE_ID_3, ], }, [LIST_ID_2]: { serviceIdsNeedingVerification: [SERVICE_ID_3], }, }, }, }, }; it('removes now hidden uuids', async () => { const action: StoryDistributionListsActionType = { type: HIDE_MY_STORIES_FROM, payload: [SERVICE_ID_1, SERVICE_ID_2], }; const actual = reducer(state, action); assert.deepEqual(actual.verificationDataByConversation, { convo1: { type: ConversationVerificationState.PendingVerification, serviceIdsNeedingVerification: [], byDistributionId: { [MY_STORY_ID]: { serviceIdsNeedingVerification: [SERVICE_ID_3], }, [LIST_ID_2]: { serviceIdsNeedingVerification: [SERVICE_ID_3], }, }, }, }); }); it('eliminates list if all items removed', async () => { const action: StoryDistributionListsActionType = { type: HIDE_MY_STORIES_FROM, payload: [SERVICE_ID_1, SERVICE_ID_2, SERVICE_ID_3], }; const actual = reducer(state, action); assert.deepEqual(actual.verificationDataByConversation, { convo1: { type: ConversationVerificationState.PendingVerification, serviceIdsNeedingVerification: [], byDistributionId: { [LIST_ID_2]: { serviceIdsNeedingVerification: [SERVICE_ID_3], }, }, }, }); }); }); describe('DELETE_LIST', () => { const state: ConversationsStateType = { ...getEmptyState(), verificationDataByConversation: { convo1: { type: ConversationVerificationState.PendingVerification, serviceIdsNeedingVerification: [], byDistributionId: { [LIST_ID_1]: { serviceIdsNeedingVerification: [ SERVICE_ID_1, SERVICE_ID_2, SERVICE_ID_3, ], }, [LIST_ID_2]: { serviceIdsNeedingVerification: [SERVICE_ID_3], }, }, }, }, }; it('eliminates deleted list entirely', async () => { const action: StoryDistributionListsActionType = { type: DELETE_LIST, payload: { deletedAtTimestamp: Date.now(), listId: LIST_ID_1, }, }; const actual = reducer(state, action); assert.deepEqual(actual.verificationDataByConversation, { convo1: { type: ConversationVerificationState.PendingVerification, serviceIdsNeedingVerification: [], byDistributionId: { [LIST_ID_2]: { serviceIdsNeedingVerification: [SERVICE_ID_3], }, }, }, }); }); it('deletes parent conversation if no other lists, no top-level uuids', async () => { const starting: ConversationsStateType = { ...getEmptyState(), verificationDataByConversation: { convo1: { type: ConversationVerificationState.PendingVerification, serviceIdsNeedingVerification: [], byDistributionId: { [LIST_ID_1]: { serviceIdsNeedingVerification: [ SERVICE_ID_1, SERVICE_ID_2, SERVICE_ID_3, ], }, }, }, }, }; const action: StoryDistributionListsActionType = { type: DELETE_LIST, payload: { deletedAtTimestamp: Date.now(), listId: LIST_ID_1, }, }; const actual = reducer(starting, action); assert.deepEqual(actual.verificationDataByConversation, {}); }); it('deletes byDistributionId if top-level list does have uuids', async () => { const starting: ConversationsStateType = { ...getEmptyState(), verificationDataByConversation: { convo1: { type: ConversationVerificationState.PendingVerification, serviceIdsNeedingVerification: [SERVICE_ID_1], byDistributionId: { [LIST_ID_1]: { serviceIdsNeedingVerification: [ SERVICE_ID_1, SERVICE_ID_2, SERVICE_ID_3, ], }, }, }, }, }; const action: StoryDistributionListsActionType = { type: DELETE_LIST, payload: { deletedAtTimestamp: Date.now(), listId: LIST_ID_1, }, }; const actual = reducer(starting, action); assert.deepEqual(actual.verificationDataByConversation, { convo1: { type: ConversationVerificationState.PendingVerification, serviceIdsNeedingVerification: [SERVICE_ID_1], }, }); }); }); describe('MODIFY_LIST', () => { const state: ConversationsStateType = { ...getEmptyState(), verificationDataByConversation: { convo1: { type: ConversationVerificationState.PendingVerification, serviceIdsNeedingVerification: [], byDistributionId: { [LIST_ID_1]: { serviceIdsNeedingVerification: [ SERVICE_ID_1, SERVICE_ID_2, SERVICE_ID_3, ], }, [LIST_ID_2]: { serviceIdsNeedingVerification: [SERVICE_ID_3], }, }, }, }, }; it('removes toRemove uuids for isBlockList = false', async () => { const action: StoryDistributionListsActionType = { type: MODIFY_LIST, payload: { id: LIST_ID_1, name: 'list1', allowsReplies: true, isBlockList: false, membersToAdd: [SERVICE_ID_2, SERVICE_ID_4], membersToRemove: [SERVICE_ID_3], }, }; const actual = reducer(state, action); assert.deepEqual(actual.verificationDataByConversation, { convo1: { type: ConversationVerificationState.PendingVerification, serviceIdsNeedingVerification: [], byDistributionId: { [LIST_ID_1]: { serviceIdsNeedingVerification: [SERVICE_ID_1, SERVICE_ID_2], }, [LIST_ID_2]: { serviceIdsNeedingVerification: [SERVICE_ID_3], }, }, }, }); }); it('removes toAdd uuids for isBlocklist = true', async () => { const action: StoryDistributionListsActionType = { type: MODIFY_LIST, payload: { id: LIST_ID_1, name: 'list1', allowsReplies: true, isBlockList: true, membersToAdd: [SERVICE_ID_2, SERVICE_ID_1], membersToRemove: [SERVICE_ID_3], }, }; const actual = reducer(state, action); assert.deepEqual(actual.verificationDataByConversation, { convo1: { type: ConversationVerificationState.PendingVerification, serviceIdsNeedingVerification: [], byDistributionId: { [LIST_ID_1]: { serviceIdsNeedingVerification: [SERVICE_ID_3], }, [LIST_ID_2]: { serviceIdsNeedingVerification: [SERVICE_ID_3], }, }, }, }); }); }); describe('CONVERSATIONS_UPDATED', () => { it('adds and updates multiple conversations', () => { const conversation1 = getDefaultConversation(); const conversation2 = getDefaultConversation(); const newConversation = getDefaultConversation(); strictAssert(conversation1.serviceId, 'must exist'); strictAssert(conversation1.e164, 'must exist'); strictAssert(conversation2.serviceId, 'must exist'); strictAssert(conversation2.e164, 'must exist'); strictAssert(newConversation.serviceId, 'must exist'); strictAssert(newConversation.e164, 'must exist'); const state = { ...getEmptyState(), conversationLookup: { [conversation1.id]: conversation1, [conversation2.id]: conversation2, }, conversationsByE164: { [conversation1.e164]: conversation1, [conversation2.e164]: conversation2, }, conversationsByServiceId: { [conversation1.serviceId]: conversation1, [conversation2.serviceId]: conversation2, }, }; const updatedConversation1 = { ...conversation1, e164: undefined, title: 'new title', }; const updatedConversation2 = { ...conversation2, active_at: 12345, }; const updatedConversation2Again = { ...conversation2, active_at: 98765, }; const action: ConversationsUpdatedActionType = { type: 'CONVERSATIONS_UPDATED', payload: { data: [ updatedConversation1, updatedConversation2, newConversation, updatedConversation2Again, ], }, }; const actual = reducer(state, action); const expected: ConversationsStateType = { ...state, conversationLookup: { [conversation1.id]: updatedConversation1, [conversation2.id]: updatedConversation2Again, [newConversation.id]: newConversation, }, conversationsByE164: { [conversation2.e164]: updatedConversation2Again, [newConversation.e164]: newConversation, }, conversationsByServiceId: { [conversation1.serviceId]: updatedConversation1, [conversation2.serviceId]: updatedConversation2Again, [newConversation.serviceId]: newConversation, }, }; assert.deepEqual(actual, expected); }); it('updates root state if conversation is selected', () => { const conversation1 = getDefaultConversation({ isArchived: true }); const conversation2 = getDefaultConversation(); strictAssert(conversation1.serviceId, 'must exist'); strictAssert(conversation1.e164, 'must exist'); strictAssert(conversation2.serviceId, 'must exist'); strictAssert(conversation2.e164, 'must exist'); const state: ConversationsStateType = { ...getEmptyState(), selectedConversationId: conversation1.id, showArchived: true, conversationLookup: { [conversation1.id]: conversation1, [conversation2.id]: conversation2, }, conversationsByE164: { [conversation1.e164]: conversation1, [conversation2.e164]: conversation2, }, conversationsByServiceId: { [conversation1.serviceId]: conversation1, [conversation2.serviceId]: conversation2, }, }; const updatedConversation1 = { ...conversation1, isArchived: false, }; const updatedConversation2 = { ...conversation2, active_at: 12345, }; const action: ConversationsUpdatedActionType = { type: 'CONVERSATIONS_UPDATED', payload: { data: [updatedConversation1, updatedConversation2], }, }; const actual = reducer(state, action); const expected: ConversationsStateType = { ...state, showArchived: false, conversationLookup: { [conversation1.id]: updatedConversation1, [conversation2.id]: updatedConversation2, }, conversationsByE164: { [conversation1.e164]: updatedConversation1, [conversation2.e164]: updatedConversation2, }, conversationsByServiceId: { [conversation1.serviceId]: updatedConversation1, [conversation2.serviceId]: updatedConversation2, }, }; assert.deepEqual(actual, expected); }); }); }); });