2075 lines
64 KiB
TypeScript
2075 lines
64 KiB
TypeScript
// Copyright 2020-2022 Signal Messenger, LLC
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
import { assert } from 'chai';
|
|
import * as sinon from 'sinon';
|
|
import { cloneDeep, noop } from 'lodash';
|
|
import type { StateType as RootStateType } from '../../../state/reducer';
|
|
import { reducer as rootReducer } from '../../../state/reducer';
|
|
import { noopAction } from '../../../state/ducks/noop';
|
|
import type {
|
|
CallingStateType,
|
|
GroupCallStateChangeActionType,
|
|
} from '../../../state/ducks/calling';
|
|
import {
|
|
actions,
|
|
getActiveCall,
|
|
getEmptyState,
|
|
isAnybodyElseInGroupCall,
|
|
reducer,
|
|
} from '../../../state/ducks/calling';
|
|
import { calling as callingService } from '../../../services/calling';
|
|
import {
|
|
CallMode,
|
|
CallState,
|
|
GroupCallConnectionState,
|
|
GroupCallJoinState,
|
|
} from '../../../types/Calling';
|
|
import { UUID } from '../../../types/UUID';
|
|
import { getDefaultConversation } from '../../../test-both/helpers/getDefaultConversation';
|
|
import type { UnwrapPromise } from '../../../types/Util';
|
|
|
|
describe('calling duck', () => {
|
|
const stateWithDirectCall: CallingStateType = {
|
|
...getEmptyState(),
|
|
callsByConversation: {
|
|
'fake-direct-call-conversation-id': {
|
|
callMode: CallMode.Direct as CallMode.Direct,
|
|
conversationId: 'fake-direct-call-conversation-id',
|
|
callState: CallState.Accepted,
|
|
isIncoming: false,
|
|
isVideoCall: false,
|
|
hasRemoteVideo: false,
|
|
},
|
|
},
|
|
};
|
|
|
|
const stateWithActiveDirectCall = {
|
|
...stateWithDirectCall,
|
|
activeCallState: {
|
|
conversationId: 'fake-direct-call-conversation-id',
|
|
hasLocalAudio: true,
|
|
hasLocalVideo: false,
|
|
amISpeaking: false,
|
|
isInSpeakerView: false,
|
|
showParticipantsList: false,
|
|
safetyNumberChangedUuids: [],
|
|
outgoingRing: true,
|
|
pip: false,
|
|
settingsDialogOpen: false,
|
|
},
|
|
};
|
|
|
|
const stateWithIncomingDirectCall = {
|
|
...getEmptyState(),
|
|
callsByConversation: {
|
|
'fake-direct-call-conversation-id': {
|
|
callMode: CallMode.Direct as CallMode.Direct,
|
|
conversationId: 'fake-direct-call-conversation-id',
|
|
callState: CallState.Ringing,
|
|
isIncoming: true,
|
|
isVideoCall: false,
|
|
hasRemoteVideo: false,
|
|
},
|
|
},
|
|
};
|
|
|
|
const creatorUuid = UUID.generate().toString();
|
|
const differentCreatorUuid = UUID.generate().toString();
|
|
const remoteUuid = UUID.generate().toString();
|
|
const ringerUuid = UUID.generate().toString();
|
|
|
|
const stateWithGroupCall = {
|
|
...getEmptyState(),
|
|
callsByConversation: {
|
|
'fake-group-call-conversation-id': {
|
|
callMode: CallMode.Group as CallMode.Group,
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
connectionState: GroupCallConnectionState.Connected,
|
|
joinState: GroupCallJoinState.NotJoined,
|
|
peekInfo: {
|
|
uuids: [creatorUuid],
|
|
creatorUuid,
|
|
eraId: 'xyz',
|
|
maxDevices: 16,
|
|
deviceCount: 1,
|
|
},
|
|
remoteParticipants: [
|
|
{
|
|
uuid: remoteUuid,
|
|
demuxId: 123,
|
|
hasRemoteAudio: true,
|
|
hasRemoteVideo: true,
|
|
presenting: false,
|
|
sharingScreen: false,
|
|
videoAspectRatio: 4 / 3,
|
|
},
|
|
],
|
|
},
|
|
},
|
|
};
|
|
|
|
const stateWithIncomingGroupCall = {
|
|
...stateWithGroupCall,
|
|
callsByConversation: {
|
|
...stateWithGroupCall.callsByConversation,
|
|
'fake-group-call-conversation-id': {
|
|
...stateWithGroupCall.callsByConversation[
|
|
'fake-group-call-conversation-id'
|
|
],
|
|
ringId: BigInt(123),
|
|
ringerUuid: UUID.generate().toString(),
|
|
},
|
|
},
|
|
};
|
|
|
|
const stateWithActiveGroupCall = {
|
|
...stateWithGroupCall,
|
|
activeCallState: {
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
hasLocalAudio: true,
|
|
hasLocalVideo: false,
|
|
amISpeaking: false,
|
|
isInSpeakerView: false,
|
|
showParticipantsList: false,
|
|
safetyNumberChangedUuids: [],
|
|
outgoingRing: false,
|
|
pip: false,
|
|
settingsDialogOpen: false,
|
|
},
|
|
};
|
|
|
|
const ourUuid = UUID.generate().toString();
|
|
|
|
const getEmptyRootState = () => {
|
|
const rootState = rootReducer(undefined, noopAction());
|
|
return {
|
|
...rootState,
|
|
user: {
|
|
...rootState.user,
|
|
ourUuid,
|
|
},
|
|
};
|
|
};
|
|
|
|
beforeEach(function beforeEach() {
|
|
this.sandbox = sinon.createSandbox();
|
|
});
|
|
|
|
afterEach(function afterEach() {
|
|
this.sandbox.restore();
|
|
});
|
|
|
|
describe('actions', () => {
|
|
describe('getPresentingSources', () => {
|
|
beforeEach(function beforeEach() {
|
|
this.callingServiceGetPresentingSources = this.sandbox
|
|
.stub(callingService, 'getPresentingSources')
|
|
.resolves([
|
|
{
|
|
id: 'foo.bar',
|
|
name: 'Foo Bar',
|
|
thumbnail: 'xyz',
|
|
},
|
|
]);
|
|
});
|
|
|
|
it('retrieves sources from the calling service', async function test() {
|
|
const { getPresentingSources } = actions;
|
|
const dispatch = sinon.spy();
|
|
await getPresentingSources()(dispatch, getEmptyRootState, null);
|
|
|
|
sinon.assert.calledOnce(this.callingServiceGetPresentingSources);
|
|
});
|
|
|
|
it('dispatches SET_PRESENTING_SOURCES', async function test() {
|
|
const { getPresentingSources } = actions;
|
|
const dispatch = sinon.spy();
|
|
await getPresentingSources()(dispatch, getEmptyRootState, null);
|
|
|
|
sinon.assert.calledOnce(dispatch);
|
|
sinon.assert.calledWith(dispatch, {
|
|
type: 'calling/SET_PRESENTING_SOURCES',
|
|
payload: [
|
|
{
|
|
id: 'foo.bar',
|
|
name: 'Foo Bar',
|
|
thumbnail: 'xyz',
|
|
},
|
|
],
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('remoteSharingScreenChange', () => {
|
|
it("updates whether someone's screen is being shared", () => {
|
|
const { remoteSharingScreenChange } = actions;
|
|
|
|
const payload = {
|
|
conversationId: 'fake-direct-call-conversation-id',
|
|
isSharingScreen: true,
|
|
};
|
|
|
|
const state = {
|
|
...stateWithActiveDirectCall,
|
|
};
|
|
const nextState = reducer(state, remoteSharingScreenChange(payload));
|
|
|
|
const expectedState = {
|
|
...stateWithActiveDirectCall,
|
|
callsByConversation: {
|
|
'fake-direct-call-conversation-id': {
|
|
...stateWithActiveDirectCall.callsByConversation[
|
|
'fake-direct-call-conversation-id'
|
|
],
|
|
isSharingScreen: true,
|
|
},
|
|
},
|
|
};
|
|
|
|
assert.deepEqual(nextState, expectedState);
|
|
});
|
|
});
|
|
|
|
describe('setPresenting', () => {
|
|
beforeEach(function beforeEach() {
|
|
this.callingServiceSetPresenting = this.sandbox.stub(
|
|
callingService,
|
|
'setPresenting'
|
|
);
|
|
});
|
|
|
|
it('calls setPresenting on the calling service', function test() {
|
|
const { setPresenting } = actions;
|
|
const dispatch = sinon.spy();
|
|
const presentedSource = {
|
|
id: 'window:786',
|
|
name: 'Application',
|
|
};
|
|
const getState = () => ({
|
|
...getEmptyRootState(),
|
|
calling: {
|
|
...stateWithActiveGroupCall,
|
|
},
|
|
});
|
|
|
|
setPresenting(presentedSource)(dispatch, getState, null);
|
|
|
|
sinon.assert.calledOnce(this.callingServiceSetPresenting);
|
|
sinon.assert.calledWith(
|
|
this.callingServiceSetPresenting,
|
|
'fake-group-call-conversation-id',
|
|
false,
|
|
presentedSource
|
|
);
|
|
});
|
|
|
|
it('dispatches SET_PRESENTING', () => {
|
|
const { setPresenting } = actions;
|
|
const dispatch = sinon.spy();
|
|
const presentedSource = {
|
|
id: 'window:786',
|
|
name: 'Application',
|
|
};
|
|
const getState = () => ({
|
|
...getEmptyRootState(),
|
|
calling: {
|
|
...stateWithActiveGroupCall,
|
|
},
|
|
});
|
|
|
|
setPresenting(presentedSource)(dispatch, getState, null);
|
|
|
|
sinon.assert.calledOnce(dispatch);
|
|
sinon.assert.calledWith(dispatch, {
|
|
type: 'calling/SET_PRESENTING',
|
|
payload: presentedSource,
|
|
});
|
|
});
|
|
|
|
it('turns off presenting when no value is passed in', () => {
|
|
const dispatch = sinon.spy();
|
|
const { setPresenting } = actions;
|
|
const presentedSource = {
|
|
id: 'window:786',
|
|
name: 'Application',
|
|
};
|
|
|
|
const getState = () => ({
|
|
...getEmptyRootState(),
|
|
calling: {
|
|
...stateWithActiveGroupCall,
|
|
},
|
|
});
|
|
|
|
setPresenting(presentedSource)(dispatch, getState, null);
|
|
|
|
const action = dispatch.getCall(0).args[0];
|
|
|
|
const nextState = reducer(getState().calling, action);
|
|
|
|
assert.isDefined(nextState.activeCallState);
|
|
assert.equal(
|
|
nextState.activeCallState?.presentingSource,
|
|
presentedSource
|
|
);
|
|
assert.isUndefined(
|
|
nextState.activeCallState?.presentingSourcesAvailable
|
|
);
|
|
});
|
|
|
|
it('sets the presenting value when one is passed in', () => {
|
|
const dispatch = sinon.spy();
|
|
const { setPresenting } = actions;
|
|
|
|
const getState = () => ({
|
|
...getEmptyRootState(),
|
|
calling: {
|
|
...stateWithActiveGroupCall,
|
|
},
|
|
});
|
|
|
|
setPresenting()(dispatch, getState, null);
|
|
|
|
const action = dispatch.getCall(0).args[0];
|
|
|
|
const nextState = reducer(getState().calling, action);
|
|
|
|
assert.isDefined(nextState.activeCallState);
|
|
assert.isUndefined(nextState.activeCallState?.presentingSource);
|
|
assert.isUndefined(
|
|
nextState.activeCallState?.presentingSourcesAvailable
|
|
);
|
|
});
|
|
});
|
|
|
|
describe('acceptCall', () => {
|
|
const { acceptCall } = actions;
|
|
|
|
beforeEach(function beforeEach() {
|
|
this.callingServiceAccept = this.sandbox
|
|
.stub(callingService, 'acceptDirectCall')
|
|
.resolves();
|
|
this.callingServiceJoin = this.sandbox
|
|
.stub(callingService, 'joinGroupCall')
|
|
.resolves();
|
|
});
|
|
|
|
describe('accepting a direct call', () => {
|
|
const getState = () => ({
|
|
...getEmptyRootState(),
|
|
calling: stateWithIncomingDirectCall,
|
|
});
|
|
|
|
it('dispatches an ACCEPT_CALL_PENDING action', async () => {
|
|
const dispatch = sinon.spy();
|
|
|
|
await acceptCall({
|
|
conversationId: 'fake-direct-call-conversation-id',
|
|
asVideoCall: true,
|
|
})(dispatch, getState, null);
|
|
|
|
sinon.assert.calledOnce(dispatch);
|
|
sinon.assert.calledWith(dispatch, {
|
|
type: 'calling/ACCEPT_CALL_PENDING',
|
|
payload: {
|
|
conversationId: 'fake-direct-call-conversation-id',
|
|
asVideoCall: true,
|
|
},
|
|
});
|
|
|
|
await acceptCall({
|
|
conversationId: 'fake-direct-call-conversation-id',
|
|
asVideoCall: false,
|
|
})(dispatch, getState, null);
|
|
|
|
sinon.assert.calledTwice(dispatch);
|
|
sinon.assert.calledWith(dispatch, {
|
|
type: 'calling/ACCEPT_CALL_PENDING',
|
|
payload: {
|
|
conversationId: 'fake-direct-call-conversation-id',
|
|
asVideoCall: false,
|
|
},
|
|
});
|
|
});
|
|
|
|
it('asks the calling service to accept the call', async function test() {
|
|
const dispatch = sinon.spy();
|
|
|
|
await acceptCall({
|
|
conversationId: 'fake-direct-call-conversation-id',
|
|
asVideoCall: true,
|
|
})(dispatch, getState, null);
|
|
|
|
sinon.assert.calledOnce(this.callingServiceAccept);
|
|
sinon.assert.calledWith(
|
|
this.callingServiceAccept,
|
|
'fake-direct-call-conversation-id',
|
|
true
|
|
);
|
|
|
|
await acceptCall({
|
|
conversationId: 'fake-direct-call-conversation-id',
|
|
asVideoCall: false,
|
|
})(dispatch, getState, null);
|
|
|
|
sinon.assert.calledTwice(this.callingServiceAccept);
|
|
sinon.assert.calledWith(
|
|
this.callingServiceAccept,
|
|
'fake-direct-call-conversation-id',
|
|
false
|
|
);
|
|
});
|
|
|
|
it('updates the active call state with ACCEPT_CALL_PENDING', async () => {
|
|
const dispatch = sinon.spy();
|
|
await acceptCall({
|
|
conversationId: 'fake-direct-call-conversation-id',
|
|
asVideoCall: true,
|
|
})(dispatch, getState, null);
|
|
const action = dispatch.getCall(0).args[0];
|
|
|
|
const result = reducer(stateWithIncomingDirectCall, action);
|
|
|
|
assert.deepEqual(result.activeCallState, {
|
|
conversationId: 'fake-direct-call-conversation-id',
|
|
hasLocalAudio: true,
|
|
hasLocalVideo: true,
|
|
amISpeaking: false,
|
|
isInSpeakerView: false,
|
|
showParticipantsList: false,
|
|
safetyNumberChangedUuids: [],
|
|
outgoingRing: false,
|
|
pip: false,
|
|
settingsDialogOpen: false,
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('accepting a group call', () => {
|
|
const getState = () => ({
|
|
...getEmptyRootState(),
|
|
calling: stateWithIncomingGroupCall,
|
|
});
|
|
|
|
it('dispatches an ACCEPT_CALL_PENDING action', async () => {
|
|
const dispatch = sinon.spy();
|
|
|
|
await acceptCall({
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
asVideoCall: true,
|
|
})(dispatch, getState, null);
|
|
|
|
sinon.assert.calledOnce(dispatch);
|
|
sinon.assert.calledWith(dispatch, {
|
|
type: 'calling/ACCEPT_CALL_PENDING',
|
|
payload: {
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
asVideoCall: true,
|
|
},
|
|
});
|
|
|
|
await acceptCall({
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
asVideoCall: false,
|
|
})(dispatch, getState, null);
|
|
|
|
sinon.assert.calledTwice(dispatch);
|
|
sinon.assert.calledWith(dispatch, {
|
|
type: 'calling/ACCEPT_CALL_PENDING',
|
|
payload: {
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
asVideoCall: false,
|
|
},
|
|
});
|
|
});
|
|
|
|
it('asks the calling service to join the call', async function test() {
|
|
const dispatch = sinon.spy();
|
|
|
|
await acceptCall({
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
asVideoCall: true,
|
|
})(dispatch, getState, null);
|
|
|
|
sinon.assert.calledOnce(this.callingServiceJoin);
|
|
sinon.assert.calledWith(
|
|
this.callingServiceJoin,
|
|
'fake-group-call-conversation-id',
|
|
true,
|
|
true
|
|
);
|
|
|
|
await acceptCall({
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
asVideoCall: false,
|
|
})(dispatch, getState, null);
|
|
|
|
sinon.assert.calledTwice(this.callingServiceJoin);
|
|
sinon.assert.calledWith(
|
|
this.callingServiceJoin,
|
|
'fake-group-call-conversation-id',
|
|
true,
|
|
false
|
|
);
|
|
});
|
|
|
|
it('updates the active call state with ACCEPT_CALL_PENDING', async () => {
|
|
const dispatch = sinon.spy();
|
|
await acceptCall({
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
asVideoCall: true,
|
|
})(dispatch, getState, null);
|
|
const action = dispatch.getCall(0).args[0];
|
|
|
|
const result = reducer(stateWithIncomingGroupCall, action);
|
|
|
|
assert.deepEqual(result.activeCallState, {
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
hasLocalAudio: true,
|
|
hasLocalVideo: true,
|
|
amISpeaking: false,
|
|
isInSpeakerView: false,
|
|
showParticipantsList: false,
|
|
safetyNumberChangedUuids: [],
|
|
outgoingRing: false,
|
|
pip: false,
|
|
settingsDialogOpen: false,
|
|
});
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('cancelCall', () => {
|
|
const { cancelCall } = actions;
|
|
|
|
beforeEach(function beforeEach() {
|
|
this.callingServiceStopCallingLobby = this.sandbox.stub(
|
|
callingService,
|
|
'stopCallingLobby'
|
|
);
|
|
});
|
|
|
|
it('stops the calling lobby for that conversation', function test() {
|
|
cancelCall({ conversationId: '123' });
|
|
|
|
sinon.assert.calledOnce(this.callingServiceStopCallingLobby);
|
|
sinon.assert.calledWith(this.callingServiceStopCallingLobby, '123');
|
|
});
|
|
|
|
it('completely removes an active direct call from the state', () => {
|
|
const result = reducer(
|
|
stateWithActiveDirectCall,
|
|
cancelCall({ conversationId: 'fake-direct-call-conversation-id' })
|
|
);
|
|
|
|
assert.notProperty(
|
|
result.callsByConversation,
|
|
'fake-direct-call-conversation-id'
|
|
);
|
|
assert.isUndefined(result.activeCallState);
|
|
});
|
|
|
|
it('removes the active group call, but leaves it in the state', () => {
|
|
const result = reducer(
|
|
stateWithActiveGroupCall,
|
|
cancelCall({ conversationId: 'fake-group-call-conversation-id' })
|
|
);
|
|
|
|
assert.property(
|
|
result.callsByConversation,
|
|
'fake-group-call-conversation-id'
|
|
);
|
|
assert.isUndefined(result.activeCallState);
|
|
});
|
|
});
|
|
|
|
describe('cancelIncomingGroupCallRing', () => {
|
|
const { cancelIncomingGroupCallRing } = actions;
|
|
|
|
it('does nothing if there is no associated group call', () => {
|
|
const state = getEmptyState();
|
|
const action = cancelIncomingGroupCallRing({
|
|
conversationId: 'garbage',
|
|
ringId: BigInt(1),
|
|
});
|
|
|
|
const result = reducer(state, action);
|
|
|
|
assert.strictEqual(result, state);
|
|
});
|
|
|
|
it("does nothing if the ring to cancel isn't the same one", () => {
|
|
const action = cancelIncomingGroupCallRing({
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
ringId: BigInt(999),
|
|
});
|
|
|
|
const result = reducer(stateWithIncomingGroupCall, action);
|
|
|
|
assert.strictEqual(result, stateWithIncomingGroupCall);
|
|
});
|
|
|
|
it('removes the ring state, but not the call', () => {
|
|
const action = cancelIncomingGroupCallRing({
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
ringId: BigInt(123),
|
|
});
|
|
|
|
const result = reducer(stateWithIncomingGroupCall, action);
|
|
const call =
|
|
result.callsByConversation['fake-group-call-conversation-id'];
|
|
// It'd be nice to do this with an assert, but Chai doesn't understand it.
|
|
if (call?.callMode !== CallMode.Group) {
|
|
throw new Error('Expected to find a group call');
|
|
}
|
|
|
|
assert.isUndefined(call.ringId);
|
|
assert.isUndefined(call.ringerUuid);
|
|
});
|
|
});
|
|
|
|
describe('declineCall', () => {
|
|
const { declineCall } = actions;
|
|
|
|
let declineDirectCall: sinon.SinonStub;
|
|
let declineGroupCall: sinon.SinonStub;
|
|
|
|
beforeEach(function beforeEach() {
|
|
declineDirectCall = this.sandbox.stub(
|
|
callingService,
|
|
'declineDirectCall'
|
|
);
|
|
declineGroupCall = this.sandbox.stub(
|
|
callingService,
|
|
'declineGroupCall'
|
|
);
|
|
});
|
|
|
|
describe('declining a direct call', () => {
|
|
const getState = () => ({
|
|
...getEmptyRootState(),
|
|
calling: stateWithIncomingDirectCall,
|
|
});
|
|
|
|
it('dispatches a DECLINE_DIRECT_CALL action', () => {
|
|
const dispatch = sinon.spy();
|
|
|
|
declineCall({ conversationId: 'fake-direct-call-conversation-id' })(
|
|
dispatch,
|
|
getState,
|
|
null
|
|
);
|
|
|
|
sinon.assert.calledOnce(dispatch);
|
|
sinon.assert.calledWith(dispatch, {
|
|
type: 'calling/DECLINE_DIRECT_CALL',
|
|
payload: {
|
|
conversationId: 'fake-direct-call-conversation-id',
|
|
},
|
|
});
|
|
});
|
|
|
|
it('asks the calling service to decline the call', () => {
|
|
const dispatch = sinon.spy();
|
|
|
|
declineCall({ conversationId: 'fake-direct-call-conversation-id' })(
|
|
dispatch,
|
|
getState,
|
|
null
|
|
);
|
|
|
|
sinon.assert.calledOnce(declineDirectCall);
|
|
sinon.assert.calledWith(
|
|
declineDirectCall,
|
|
'fake-direct-call-conversation-id'
|
|
);
|
|
});
|
|
|
|
it('removes the call from the state', () => {
|
|
const dispatch = sinon.spy();
|
|
declineCall({ conversationId: 'fake-direct-call-conversation-id' })(
|
|
dispatch,
|
|
getState,
|
|
null
|
|
);
|
|
const action = dispatch.getCall(0).args[0];
|
|
|
|
const result = reducer(stateWithIncomingGroupCall, action);
|
|
|
|
assert.notProperty(
|
|
result.callsByConversation,
|
|
'fake-direct-call-conversation-id'
|
|
);
|
|
});
|
|
});
|
|
|
|
describe('declining a group call', () => {
|
|
const getState = () => ({
|
|
...getEmptyRootState(),
|
|
calling: stateWithIncomingGroupCall,
|
|
});
|
|
|
|
it('dispatches a CANCEL_INCOMING_GROUP_CALL_RING action', () => {
|
|
const dispatch = sinon.spy();
|
|
|
|
declineCall({ conversationId: 'fake-group-call-conversation-id' })(
|
|
dispatch,
|
|
getState,
|
|
null
|
|
);
|
|
|
|
sinon.assert.calledOnce(dispatch);
|
|
sinon.assert.calledWith(dispatch, {
|
|
type: 'calling/CANCEL_INCOMING_GROUP_CALL_RING',
|
|
payload: {
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
ringId: BigInt(123),
|
|
},
|
|
});
|
|
});
|
|
|
|
it('asks the calling service to decline the call', () => {
|
|
const dispatch = sinon.spy();
|
|
|
|
declineCall({ conversationId: 'fake-group-call-conversation-id' })(
|
|
dispatch,
|
|
getState,
|
|
null
|
|
);
|
|
|
|
sinon.assert.calledOnce(declineGroupCall);
|
|
sinon.assert.calledWith(
|
|
declineGroupCall,
|
|
'fake-group-call-conversation-id',
|
|
BigInt(123)
|
|
);
|
|
});
|
|
|
|
// NOTE: The state effects of this action are tested with
|
|
// `cancelIncomingGroupCallRing`.
|
|
});
|
|
});
|
|
|
|
describe('groupCallAudioLevelsChange', () => {
|
|
const { groupCallAudioLevelsChange } = actions;
|
|
|
|
const remoteDeviceStates = [
|
|
{ audioLevel: 0.3, demuxId: 1 },
|
|
{ audioLevel: 0.4, demuxId: 2 },
|
|
{ audioLevel: 0.5, demuxId: 3 },
|
|
{ audioLevel: 0.2, demuxId: 7 },
|
|
{ audioLevel: 0.1, demuxId: 8 },
|
|
{ audioLevel: 0, demuxId: 9 },
|
|
];
|
|
|
|
it("does nothing if there's no relevant call", () => {
|
|
const action = groupCallAudioLevelsChange({
|
|
audioLevelForSpeaking: 0.25,
|
|
conversationId: 'garbage',
|
|
localAudioLevel: 1,
|
|
remoteDeviceStates,
|
|
});
|
|
|
|
const result = reducer(stateWithActiveGroupCall, action);
|
|
|
|
assert.strictEqual(result, stateWithActiveGroupCall);
|
|
});
|
|
|
|
it('does nothing if the state change would be a no-op', () => {
|
|
const state = {
|
|
...stateWithActiveGroupCall,
|
|
callsByConversation: {
|
|
'fake-group-call-conversation-id': {
|
|
...stateWithActiveGroupCall.callsByConversation[
|
|
'fake-group-call-conversation-id'
|
|
],
|
|
speakingDemuxIds: new Set([3, 2, 1]),
|
|
},
|
|
},
|
|
};
|
|
const action = groupCallAudioLevelsChange({
|
|
audioLevelForSpeaking: 0.25,
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
localAudioLevel: 0.1,
|
|
remoteDeviceStates,
|
|
});
|
|
|
|
const result = reducer(state, action);
|
|
|
|
assert.strictEqual(result, state);
|
|
});
|
|
|
|
it('updates the set of speaking participants, including yourself', () => {
|
|
const action = groupCallAudioLevelsChange({
|
|
audioLevelForSpeaking: 0.25,
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
localAudioLevel: 0.8,
|
|
remoteDeviceStates,
|
|
});
|
|
const result = reducer(stateWithActiveGroupCall, action);
|
|
|
|
assert.isTrue(result.activeCallState?.amISpeaking);
|
|
|
|
const call =
|
|
result.callsByConversation['fake-group-call-conversation-id'];
|
|
if (call?.callMode !== CallMode.Group) {
|
|
throw new Error('Expected a group call to be found');
|
|
}
|
|
assert.deepStrictEqual(call.speakingDemuxIds, new Set([1, 2, 3]));
|
|
});
|
|
});
|
|
|
|
describe('groupCallStateChange', () => {
|
|
const { groupCallStateChange } = actions;
|
|
|
|
function getAction(
|
|
...args: Parameters<typeof groupCallStateChange>
|
|
): GroupCallStateChangeActionType {
|
|
const dispatch = sinon.spy();
|
|
|
|
groupCallStateChange(...args)(dispatch, getEmptyRootState, null);
|
|
|
|
return dispatch.getCall(0).args[0];
|
|
}
|
|
|
|
it('saves a new call to the map of conversations', () => {
|
|
const result = reducer(
|
|
getEmptyState(),
|
|
getAction({
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
connectionState: GroupCallConnectionState.Connected,
|
|
joinState: GroupCallJoinState.Joining,
|
|
hasLocalAudio: true,
|
|
hasLocalVideo: false,
|
|
peekInfo: {
|
|
uuids: [creatorUuid],
|
|
creatorUuid,
|
|
eraId: 'xyz',
|
|
maxDevices: 16,
|
|
deviceCount: 1,
|
|
},
|
|
remoteParticipants: [
|
|
{
|
|
uuid: remoteUuid,
|
|
demuxId: 123,
|
|
hasRemoteAudio: true,
|
|
hasRemoteVideo: true,
|
|
presenting: false,
|
|
sharingScreen: false,
|
|
videoAspectRatio: 4 / 3,
|
|
},
|
|
],
|
|
})
|
|
);
|
|
|
|
assert.deepEqual(
|
|
result.callsByConversation['fake-group-call-conversation-id'],
|
|
{
|
|
callMode: CallMode.Group,
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
connectionState: GroupCallConnectionState.Connected,
|
|
joinState: GroupCallJoinState.Joining,
|
|
peekInfo: {
|
|
uuids: [creatorUuid],
|
|
creatorUuid,
|
|
eraId: 'xyz',
|
|
maxDevices: 16,
|
|
deviceCount: 1,
|
|
},
|
|
remoteParticipants: [
|
|
{
|
|
uuid: remoteUuid,
|
|
demuxId: 123,
|
|
hasRemoteAudio: true,
|
|
hasRemoteVideo: true,
|
|
presenting: false,
|
|
sharingScreen: false,
|
|
videoAspectRatio: 4 / 3,
|
|
},
|
|
],
|
|
}
|
|
);
|
|
});
|
|
|
|
it('updates a call in the map of conversations', () => {
|
|
const result = reducer(
|
|
stateWithGroupCall,
|
|
getAction({
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
connectionState: GroupCallConnectionState.Connected,
|
|
joinState: GroupCallJoinState.Joined,
|
|
hasLocalAudio: true,
|
|
hasLocalVideo: false,
|
|
peekInfo: {
|
|
uuids: ['1b9e4d42-1f56-45c5-b6f4-d1be5a54fefa'],
|
|
maxDevices: 16,
|
|
deviceCount: 1,
|
|
},
|
|
remoteParticipants: [
|
|
{
|
|
uuid: remoteUuid,
|
|
demuxId: 456,
|
|
hasRemoteAudio: false,
|
|
hasRemoteVideo: true,
|
|
presenting: false,
|
|
sharingScreen: false,
|
|
videoAspectRatio: 16 / 9,
|
|
},
|
|
],
|
|
})
|
|
);
|
|
|
|
assert.deepEqual(
|
|
result.callsByConversation['fake-group-call-conversation-id'],
|
|
{
|
|
callMode: CallMode.Group,
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
connectionState: GroupCallConnectionState.Connected,
|
|
joinState: GroupCallJoinState.Joined,
|
|
peekInfo: {
|
|
uuids: ['1b9e4d42-1f56-45c5-b6f4-d1be5a54fefa'],
|
|
maxDevices: 16,
|
|
deviceCount: 1,
|
|
},
|
|
remoteParticipants: [
|
|
{
|
|
uuid: remoteUuid,
|
|
demuxId: 456,
|
|
hasRemoteAudio: false,
|
|
hasRemoteVideo: true,
|
|
presenting: false,
|
|
sharingScreen: false,
|
|
videoAspectRatio: 16 / 9,
|
|
},
|
|
],
|
|
}
|
|
);
|
|
});
|
|
|
|
it("keeps the existing ring state if you haven't joined the call", () => {
|
|
const state = {
|
|
...stateWithGroupCall,
|
|
callsByConversation: {
|
|
...stateWithGroupCall.callsByConversation,
|
|
'fake-group-call-conversation-id': {
|
|
...stateWithGroupCall.callsByConversation[
|
|
'fake-group-call-conversation-id'
|
|
],
|
|
ringId: BigInt(456),
|
|
ringerUuid,
|
|
},
|
|
},
|
|
};
|
|
const result = reducer(
|
|
state,
|
|
getAction({
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
connectionState: GroupCallConnectionState.Connected,
|
|
joinState: GroupCallJoinState.NotJoined,
|
|
hasLocalAudio: true,
|
|
hasLocalVideo: false,
|
|
peekInfo: {
|
|
uuids: ['1b9e4d42-1f56-45c5-b6f4-d1be5a54fefa'],
|
|
maxDevices: 16,
|
|
deviceCount: 1,
|
|
},
|
|
remoteParticipants: [
|
|
{
|
|
uuid: remoteUuid,
|
|
demuxId: 456,
|
|
hasRemoteAudio: false,
|
|
hasRemoteVideo: true,
|
|
presenting: false,
|
|
sharingScreen: false,
|
|
videoAspectRatio: 16 / 9,
|
|
},
|
|
],
|
|
})
|
|
);
|
|
|
|
assert.include(
|
|
result.callsByConversation['fake-group-call-conversation-id'],
|
|
{
|
|
callMode: CallMode.Group,
|
|
ringId: BigInt(456),
|
|
ringerUuid,
|
|
}
|
|
);
|
|
});
|
|
|
|
it("removes the ring state if you've joined the call", () => {
|
|
const state = {
|
|
...stateWithGroupCall,
|
|
callsByConversation: {
|
|
...stateWithGroupCall.callsByConversation,
|
|
'fake-group-call-conversation-id': {
|
|
...stateWithGroupCall.callsByConversation[
|
|
'fake-group-call-conversation-id'
|
|
],
|
|
ringId: BigInt(456),
|
|
ringerUuid,
|
|
},
|
|
},
|
|
};
|
|
const result = reducer(
|
|
state,
|
|
getAction({
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
connectionState: GroupCallConnectionState.Connected,
|
|
joinState: GroupCallJoinState.Joined,
|
|
hasLocalAudio: true,
|
|
hasLocalVideo: false,
|
|
peekInfo: {
|
|
uuids: ['1b9e4d42-1f56-45c5-b6f4-d1be5a54fefa'],
|
|
maxDevices: 16,
|
|
deviceCount: 1,
|
|
},
|
|
remoteParticipants: [
|
|
{
|
|
uuid: remoteUuid,
|
|
demuxId: 456,
|
|
hasRemoteAudio: false,
|
|
hasRemoteVideo: true,
|
|
presenting: false,
|
|
sharingScreen: false,
|
|
videoAspectRatio: 16 / 9,
|
|
},
|
|
],
|
|
})
|
|
);
|
|
|
|
assert.notProperty(
|
|
result.callsByConversation['fake-group-call-conversation-id'],
|
|
'ringId'
|
|
);
|
|
assert.notProperty(
|
|
result.callsByConversation['fake-group-call-conversation-id'],
|
|
'ringerUuid'
|
|
);
|
|
});
|
|
|
|
it("if no call is active, doesn't touch the active call state", () => {
|
|
const result = reducer(
|
|
stateWithGroupCall,
|
|
getAction({
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
connectionState: GroupCallConnectionState.Connected,
|
|
joinState: GroupCallJoinState.Joined,
|
|
hasLocalAudio: true,
|
|
hasLocalVideo: false,
|
|
peekInfo: {
|
|
uuids: ['1b9e4d42-1f56-45c5-b6f4-d1be5a54fefa'],
|
|
maxDevices: 16,
|
|
deviceCount: 1,
|
|
},
|
|
remoteParticipants: [
|
|
{
|
|
uuid: remoteUuid,
|
|
demuxId: 456,
|
|
hasRemoteAudio: false,
|
|
hasRemoteVideo: true,
|
|
presenting: false,
|
|
sharingScreen: false,
|
|
videoAspectRatio: 16 / 9,
|
|
},
|
|
],
|
|
})
|
|
);
|
|
|
|
assert.isUndefined(result.activeCallState);
|
|
});
|
|
|
|
it("if the call is not active, doesn't touch the active call state", () => {
|
|
const result = reducer(
|
|
stateWithActiveGroupCall,
|
|
getAction({
|
|
conversationId: 'another-fake-conversation-id',
|
|
connectionState: GroupCallConnectionState.Connected,
|
|
joinState: GroupCallJoinState.Joined,
|
|
hasLocalAudio: true,
|
|
hasLocalVideo: true,
|
|
peekInfo: {
|
|
uuids: ['1b9e4d42-1f56-45c5-b6f4-d1be5a54fefa'],
|
|
maxDevices: 16,
|
|
deviceCount: 1,
|
|
},
|
|
remoteParticipants: [
|
|
{
|
|
uuid: remoteUuid,
|
|
demuxId: 456,
|
|
hasRemoteAudio: false,
|
|
hasRemoteVideo: true,
|
|
presenting: false,
|
|
sharingScreen: false,
|
|
videoAspectRatio: 16 / 9,
|
|
},
|
|
],
|
|
})
|
|
);
|
|
|
|
assert.deepEqual(result.activeCallState, {
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
hasLocalAudio: true,
|
|
hasLocalVideo: false,
|
|
amISpeaking: false,
|
|
isInSpeakerView: false,
|
|
showParticipantsList: false,
|
|
safetyNumberChangedUuids: [],
|
|
outgoingRing: false,
|
|
pip: false,
|
|
settingsDialogOpen: false,
|
|
});
|
|
});
|
|
|
|
it('if the call is active, updates the active call state', () => {
|
|
const result = reducer(
|
|
stateWithActiveGroupCall,
|
|
getAction({
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
connectionState: GroupCallConnectionState.Connected,
|
|
joinState: GroupCallJoinState.Joined,
|
|
hasLocalAudio: true,
|
|
hasLocalVideo: true,
|
|
peekInfo: {
|
|
uuids: ['1b9e4d42-1f56-45c5-b6f4-d1be5a54fefa'],
|
|
maxDevices: 16,
|
|
deviceCount: 1,
|
|
},
|
|
remoteParticipants: [
|
|
{
|
|
uuid: remoteUuid,
|
|
demuxId: 456,
|
|
hasRemoteAudio: false,
|
|
hasRemoteVideo: true,
|
|
presenting: false,
|
|
sharingScreen: false,
|
|
videoAspectRatio: 16 / 9,
|
|
},
|
|
],
|
|
})
|
|
);
|
|
|
|
assert.strictEqual(
|
|
result.activeCallState?.conversationId,
|
|
'fake-group-call-conversation-id'
|
|
);
|
|
assert.isTrue(result.activeCallState?.hasLocalAudio);
|
|
assert.isTrue(result.activeCallState?.hasLocalVideo);
|
|
});
|
|
|
|
it("doesn't stop ringing if nobody is in the call", () => {
|
|
const state = {
|
|
...stateWithActiveGroupCall,
|
|
activeCallState: {
|
|
...stateWithActiveGroupCall.activeCallState,
|
|
outgoingRing: true,
|
|
},
|
|
};
|
|
const result = reducer(
|
|
state,
|
|
getAction({
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
connectionState: GroupCallConnectionState.Connected,
|
|
joinState: GroupCallJoinState.Joined,
|
|
hasLocalAudio: true,
|
|
hasLocalVideo: true,
|
|
peekInfo: {
|
|
uuids: [],
|
|
maxDevices: 16,
|
|
deviceCount: 0,
|
|
},
|
|
remoteParticipants: [],
|
|
})
|
|
);
|
|
|
|
assert.isTrue(result.activeCallState?.outgoingRing);
|
|
});
|
|
|
|
it('stops ringing if someone enters the call', () => {
|
|
const state = {
|
|
...stateWithActiveGroupCall,
|
|
activeCallState: {
|
|
...stateWithActiveGroupCall.activeCallState,
|
|
outgoingRing: true,
|
|
},
|
|
};
|
|
const result = reducer(
|
|
state,
|
|
getAction({
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
connectionState: GroupCallConnectionState.Connected,
|
|
joinState: GroupCallJoinState.Joined,
|
|
hasLocalAudio: true,
|
|
hasLocalVideo: true,
|
|
peekInfo: {
|
|
uuids: ['1b9e4d42-1f56-45c5-b6f4-d1be5a54fefa'],
|
|
maxDevices: 16,
|
|
deviceCount: 1,
|
|
},
|
|
remoteParticipants: [],
|
|
})
|
|
);
|
|
|
|
assert.isFalse(result.activeCallState?.outgoingRing);
|
|
});
|
|
});
|
|
|
|
describe('peekNotConnectedGroupCall', () => {
|
|
const { peekNotConnectedGroupCall } = actions;
|
|
|
|
beforeEach(function beforeEach() {
|
|
this.callingServicePeekGroupCall = this.sandbox.stub(
|
|
callingService,
|
|
'peekGroupCall'
|
|
);
|
|
this.callingServiceUpdateCallHistoryForGroupCall = this.sandbox.stub(
|
|
callingService,
|
|
'updateCallHistoryForGroupCall'
|
|
);
|
|
this.clock = this.sandbox.useFakeTimers();
|
|
});
|
|
|
|
describe('thunk', () => {
|
|
function noopTest(connectionState: GroupCallConnectionState) {
|
|
return async function test(this: Mocha.Context) {
|
|
const dispatch = sinon.spy();
|
|
|
|
await peekNotConnectedGroupCall({
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
})(
|
|
dispatch,
|
|
() => ({
|
|
...getEmptyRootState(),
|
|
calling: {
|
|
...stateWithGroupCall,
|
|
callsByConversation: {
|
|
'fake-group-call-conversation-id': {
|
|
...stateWithGroupCall.callsByConversation[
|
|
'fake-group-call-conversation-id'
|
|
],
|
|
connectionState,
|
|
},
|
|
},
|
|
},
|
|
}),
|
|
null
|
|
);
|
|
|
|
sinon.assert.notCalled(dispatch);
|
|
sinon.assert.notCalled(this.callingServicePeekGroupCall);
|
|
};
|
|
}
|
|
|
|
it(
|
|
'no-ops if trying to peek at a connecting group call',
|
|
noopTest(GroupCallConnectionState.Connecting)
|
|
);
|
|
|
|
it(
|
|
'no-ops if trying to peek at a connected group call',
|
|
noopTest(GroupCallConnectionState.Connected)
|
|
);
|
|
|
|
it(
|
|
'no-ops if trying to peek at a reconnecting group call',
|
|
noopTest(GroupCallConnectionState.Reconnecting)
|
|
);
|
|
|
|
// These tests are incomplete.
|
|
});
|
|
});
|
|
|
|
describe('returnToActiveCall', () => {
|
|
const { returnToActiveCall } = actions;
|
|
|
|
it('does nothing if not in PiP mode', () => {
|
|
const result = reducer(stateWithActiveDirectCall, returnToActiveCall());
|
|
|
|
assert.deepEqual(result, stateWithActiveDirectCall);
|
|
});
|
|
|
|
it('closes the PiP', () => {
|
|
const state = {
|
|
...stateWithActiveDirectCall,
|
|
activeCallState: {
|
|
...stateWithActiveDirectCall.activeCallState,
|
|
pip: true,
|
|
},
|
|
};
|
|
const result = reducer(state, returnToActiveCall());
|
|
|
|
assert.deepEqual(result, stateWithActiveDirectCall);
|
|
});
|
|
});
|
|
|
|
describe('receiveIncomingGroupCall', () => {
|
|
const { receiveIncomingGroupCall } = actions;
|
|
|
|
it('does nothing if the call was already ringing', () => {
|
|
const action = receiveIncomingGroupCall({
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
ringId: BigInt(456),
|
|
ringerUuid,
|
|
});
|
|
const result = reducer(stateWithIncomingGroupCall, action);
|
|
|
|
assert.strictEqual(result, stateWithIncomingGroupCall);
|
|
});
|
|
|
|
it('does nothing if the call was already joined', () => {
|
|
const state = {
|
|
...stateWithGroupCall,
|
|
callsByConversation: {
|
|
...stateWithGroupCall.callsByConversation,
|
|
'fake-group-call-conversation-id': {
|
|
...stateWithGroupCall.callsByConversation[
|
|
'fake-group-call-conversation-id'
|
|
],
|
|
joinState: GroupCallJoinState.Joined,
|
|
},
|
|
},
|
|
};
|
|
const action = receiveIncomingGroupCall({
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
ringId: BigInt(456),
|
|
ringerUuid,
|
|
});
|
|
const result = reducer(state, action);
|
|
|
|
assert.strictEqual(result, state);
|
|
});
|
|
|
|
it('creates a new group call if one did not exist', () => {
|
|
const action = receiveIncomingGroupCall({
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
ringId: BigInt(456),
|
|
ringerUuid,
|
|
});
|
|
const result = reducer(getEmptyState(), action);
|
|
|
|
assert.deepEqual(
|
|
result.callsByConversation['fake-group-call-conversation-id'],
|
|
{
|
|
callMode: CallMode.Group,
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
connectionState: GroupCallConnectionState.NotConnected,
|
|
joinState: GroupCallJoinState.NotJoined,
|
|
peekInfo: {
|
|
uuids: [],
|
|
maxDevices: Infinity,
|
|
deviceCount: 0,
|
|
},
|
|
remoteParticipants: [],
|
|
ringId: BigInt(456),
|
|
ringerUuid,
|
|
}
|
|
);
|
|
});
|
|
|
|
it('attaches ring state to an existing call', () => {
|
|
const action = receiveIncomingGroupCall({
|
|
conversationId: 'fake-group-call-conversation-id',
|
|
ringId: BigInt(456),
|
|
ringerUuid,
|
|
});
|
|
const result = reducer(stateWithGroupCall, action);
|
|
|
|
assert.include(
|
|
result.callsByConversation['fake-group-call-conversation-id'],
|
|
{
|
|
ringId: BigInt(456),
|
|
ringerUuid,
|
|
}
|
|
);
|
|
});
|
|
});
|
|
|
|
describe('setLocalAudio', () => {
|
|
const { setLocalAudio } = actions;
|
|
|
|
beforeEach(function beforeEach() {
|
|
this.callingServiceSetOutgoingAudio = this.sandbox.stub(
|
|
callingService,
|
|
'setOutgoingAudio'
|
|
);
|
|
});
|
|
|
|
it('dispatches a SET_LOCAL_AUDIO_FULFILLED action', () => {
|
|
const dispatch = sinon.spy();
|
|
|
|
setLocalAudio({ enabled: true })(
|
|
dispatch,
|
|
() => ({
|
|
...getEmptyRootState(),
|
|
calling: stateWithActiveDirectCall,
|
|
}),
|
|
null
|
|
);
|
|
|
|
sinon.assert.calledOnce(dispatch);
|
|
sinon.assert.calledWith(dispatch, {
|
|
type: 'calling/SET_LOCAL_AUDIO_FULFILLED',
|
|
payload: { enabled: true },
|
|
});
|
|
});
|
|
|
|
it('updates the outgoing audio for the active call', function test() {
|
|
const dispatch = sinon.spy();
|
|
|
|
setLocalAudio({ enabled: false })(
|
|
dispatch,
|
|
() => ({
|
|
...getEmptyRootState(),
|
|
calling: stateWithActiveDirectCall,
|
|
}),
|
|
null
|
|
);
|
|
|
|
sinon.assert.calledOnce(this.callingServiceSetOutgoingAudio);
|
|
sinon.assert.calledWith(
|
|
this.callingServiceSetOutgoingAudio,
|
|
'fake-direct-call-conversation-id',
|
|
false
|
|
);
|
|
|
|
setLocalAudio({ enabled: true })(
|
|
dispatch,
|
|
() => ({
|
|
...getEmptyRootState(),
|
|
calling: stateWithActiveDirectCall,
|
|
}),
|
|
null
|
|
);
|
|
|
|
sinon.assert.calledTwice(this.callingServiceSetOutgoingAudio);
|
|
sinon.assert.calledWith(
|
|
this.callingServiceSetOutgoingAudio,
|
|
'fake-direct-call-conversation-id',
|
|
true
|
|
);
|
|
});
|
|
|
|
it('updates the local audio state with SET_LOCAL_AUDIO_FULFILLED', () => {
|
|
const dispatch = sinon.spy();
|
|
setLocalAudio({ enabled: false })(
|
|
dispatch,
|
|
() => ({
|
|
...getEmptyRootState(),
|
|
calling: stateWithActiveDirectCall,
|
|
}),
|
|
null
|
|
);
|
|
const action = dispatch.getCall(0).args[0];
|
|
|
|
const result = reducer(stateWithActiveDirectCall, action);
|
|
|
|
assert.isFalse(result.activeCallState?.hasLocalAudio);
|
|
});
|
|
});
|
|
|
|
describe('setOutgoingRing', () => {
|
|
const { setOutgoingRing } = actions;
|
|
|
|
it('enables a desire to ring', () => {
|
|
const action = setOutgoingRing(true);
|
|
const result = reducer(stateWithActiveGroupCall, action);
|
|
|
|
assert.isTrue(result.activeCallState?.outgoingRing);
|
|
});
|
|
|
|
it('disables a desire to ring', () => {
|
|
const action = setOutgoingRing(false);
|
|
const result = reducer(stateWithActiveDirectCall, action);
|
|
|
|
assert.isFalse(result.activeCallState?.outgoingRing);
|
|
});
|
|
});
|
|
|
|
describe('startCallingLobby', () => {
|
|
const { startCallingLobby } = actions;
|
|
|
|
let rootState: RootStateType;
|
|
let startCallingLobbyStub: sinon.SinonStub;
|
|
|
|
beforeEach(function beforeEach() {
|
|
startCallingLobbyStub = this.sandbox
|
|
.stub(callingService, 'startCallingLobby')
|
|
.resolves();
|
|
|
|
const emptyRootState = getEmptyRootState();
|
|
rootState = {
|
|
...emptyRootState,
|
|
conversations: {
|
|
...emptyRootState.conversations,
|
|
conversationLookup: {
|
|
'fake-conversation-id': getDefaultConversation(),
|
|
},
|
|
},
|
|
};
|
|
});
|
|
|
|
describe('thunk', () => {
|
|
it('asks the calling service to start the lobby', async () => {
|
|
await startCallingLobby({
|
|
conversationId: 'fake-conversation-id',
|
|
isVideoCall: true,
|
|
})(noop, () => rootState, null);
|
|
|
|
sinon.assert.calledOnce(startCallingLobbyStub);
|
|
});
|
|
|
|
it('requests audio by default', async () => {
|
|
await startCallingLobby({
|
|
conversationId: 'fake-conversation-id',
|
|
isVideoCall: true,
|
|
})(noop, () => rootState, null);
|
|
|
|
sinon.assert.calledWithMatch(startCallingLobbyStub, {
|
|
hasLocalAudio: true,
|
|
});
|
|
});
|
|
|
|
it("doesn't request audio if the group call already has 8 devices", async () => {
|
|
await startCallingLobby({
|
|
conversationId: 'fake-conversation-id',
|
|
isVideoCall: true,
|
|
})(
|
|
noop,
|
|
() => {
|
|
const callingState = cloneDeep(stateWithGroupCall);
|
|
callingState.callsByConversation[
|
|
'fake-group-call-conversation-id'
|
|
].peekInfo.deviceCount = 8;
|
|
return { ...rootState, calling: callingState };
|
|
},
|
|
null
|
|
);
|
|
|
|
sinon.assert.calledWithMatch(startCallingLobbyStub, {
|
|
hasLocalVideo: true,
|
|
});
|
|
});
|
|
|
|
it('requests video when starting a video call', async () => {
|
|
await startCallingLobby({
|
|
conversationId: 'fake-conversation-id',
|
|
isVideoCall: true,
|
|
})(noop, () => rootState, null);
|
|
|
|
sinon.assert.calledWithMatch(startCallingLobbyStub, {
|
|
hasLocalVideo: true,
|
|
});
|
|
});
|
|
|
|
it("doesn't request video when not a video call", async () => {
|
|
await startCallingLobby({
|
|
conversationId: 'fake-conversation-id',
|
|
isVideoCall: false,
|
|
})(noop, () => rootState, null);
|
|
|
|
sinon.assert.calledWithMatch(startCallingLobbyStub, {
|
|
hasLocalVideo: false,
|
|
});
|
|
});
|
|
|
|
it('dispatches an action if the calling lobby returns something', async () => {
|
|
startCallingLobbyStub.resolves({
|
|
callMode: CallMode.Direct,
|
|
hasLocalAudio: true,
|
|
hasLocalVideo: true,
|
|
});
|
|
|
|
const dispatch = sinon.stub();
|
|
|
|
await startCallingLobby({
|
|
conversationId: 'fake-conversation-id',
|
|
isVideoCall: true,
|
|
})(dispatch, () => rootState, null);
|
|
|
|
sinon.assert.calledOnce(dispatch);
|
|
});
|
|
|
|
it("doesn't dispatch an action if the calling lobby returns nothing", async () => {
|
|
const dispatch = sinon.stub();
|
|
|
|
await startCallingLobby({
|
|
conversationId: 'fake-conversation-id',
|
|
isVideoCall: true,
|
|
})(dispatch, () => rootState, null);
|
|
|
|
sinon.assert.notCalled(dispatch);
|
|
});
|
|
});
|
|
|
|
describe('action', () => {
|
|
const getState = async (
|
|
callingState: CallingStateType,
|
|
callingServiceResult: UnwrapPromise<
|
|
ReturnType<typeof callingService.startCallingLobby>
|
|
>,
|
|
conversationId = 'fake-conversation-id'
|
|
): Promise<CallingStateType> => {
|
|
startCallingLobbyStub.resolves(callingServiceResult);
|
|
|
|
const dispatch = sinon.stub();
|
|
|
|
await startCallingLobby({
|
|
conversationId,
|
|
isVideoCall: true,
|
|
})(dispatch, () => ({ ...rootState, calling: callingState }), null);
|
|
|
|
const action = dispatch.getCall(0).args[0];
|
|
|
|
return reducer(callingState, action);
|
|
};
|
|
|
|
it('saves a direct call and makes it active', async () => {
|
|
const result = await getState(getEmptyState(), {
|
|
callMode: CallMode.Direct as const,
|
|
hasLocalAudio: true,
|
|
hasLocalVideo: true,
|
|
});
|
|
|
|
assert.deepEqual(result.callsByConversation['fake-conversation-id'], {
|
|
callMode: CallMode.Direct,
|
|
conversationId: 'fake-conversation-id',
|
|
isIncoming: false,
|
|
isVideoCall: true,
|
|
});
|
|
assert.deepEqual(result.activeCallState, {
|
|
conversationId: 'fake-conversation-id',
|
|
hasLocalAudio: true,
|
|
hasLocalVideo: true,
|
|
amISpeaking: false,
|
|
isInSpeakerView: false,
|
|
showParticipantsList: false,
|
|
safetyNumberChangedUuids: [],
|
|
pip: false,
|
|
settingsDialogOpen: false,
|
|
outgoingRing: true,
|
|
});
|
|
});
|
|
|
|
it('saves a group call and makes it active', async () => {
|
|
const result = await getState(getEmptyState(), {
|
|
callMode: CallMode.Group,
|
|
hasLocalAudio: true,
|
|
hasLocalVideo: true,
|
|
connectionState: GroupCallConnectionState.Connected,
|
|
joinState: GroupCallJoinState.NotJoined,
|
|
peekInfo: {
|
|
uuids: [creatorUuid],
|
|
creatorUuid,
|
|
eraId: 'xyz',
|
|
maxDevices: 16,
|
|
deviceCount: 1,
|
|
},
|
|
remoteParticipants: [
|
|
{
|
|
uuid: remoteUuid,
|
|
demuxId: 123,
|
|
hasRemoteAudio: true,
|
|
hasRemoteVideo: true,
|
|
presenting: false,
|
|
sharingScreen: false,
|
|
videoAspectRatio: 4 / 3,
|
|
},
|
|
],
|
|
});
|
|
|
|
assert.deepEqual(result.callsByConversation['fake-conversation-id'], {
|
|
callMode: CallMode.Group,
|
|
conversationId: 'fake-conversation-id',
|
|
connectionState: GroupCallConnectionState.Connected,
|
|
joinState: GroupCallJoinState.NotJoined,
|
|
peekInfo: {
|
|
uuids: [creatorUuid],
|
|
creatorUuid,
|
|
eraId: 'xyz',
|
|
maxDevices: 16,
|
|
deviceCount: 1,
|
|
},
|
|
remoteParticipants: [
|
|
{
|
|
uuid: remoteUuid,
|
|
demuxId: 123,
|
|
hasRemoteAudio: true,
|
|
hasRemoteVideo: true,
|
|
presenting: false,
|
|
sharingScreen: false,
|
|
videoAspectRatio: 4 / 3,
|
|
},
|
|
],
|
|
});
|
|
assert.deepEqual(
|
|
result.activeCallState?.conversationId,
|
|
'fake-conversation-id'
|
|
);
|
|
assert.isFalse(result.activeCallState?.outgoingRing);
|
|
});
|
|
|
|
it('chooses fallback peek info if none is sent and there is no existing call', async () => {
|
|
const result = await getState(getEmptyState(), {
|
|
callMode: CallMode.Group,
|
|
hasLocalAudio: true,
|
|
hasLocalVideo: true,
|
|
connectionState: GroupCallConnectionState.Connected,
|
|
joinState: GroupCallJoinState.NotJoined,
|
|
peekInfo: undefined,
|
|
remoteParticipants: [],
|
|
});
|
|
|
|
const call = result.callsByConversation['fake-conversation-id'];
|
|
assert.deepEqual(call?.callMode === CallMode.Group && call.peekInfo, {
|
|
uuids: [],
|
|
maxDevices: Infinity,
|
|
deviceCount: 0,
|
|
});
|
|
});
|
|
|
|
it("doesn't overwrite an existing group call's peek info if none was sent", async () => {
|
|
const result = await getState(stateWithGroupCall, {
|
|
callMode: CallMode.Group,
|
|
hasLocalAudio: true,
|
|
hasLocalVideo: true,
|
|
connectionState: GroupCallConnectionState.Connected,
|
|
joinState: GroupCallJoinState.NotJoined,
|
|
peekInfo: undefined,
|
|
remoteParticipants: [
|
|
{
|
|
uuid: remoteUuid,
|
|
demuxId: 123,
|
|
hasRemoteAudio: true,
|
|
hasRemoteVideo: true,
|
|
presenting: false,
|
|
sharingScreen: false,
|
|
videoAspectRatio: 4 / 3,
|
|
},
|
|
],
|
|
});
|
|
|
|
const call =
|
|
result.callsByConversation['fake-group-call-conversation-id'];
|
|
assert.deepEqual(call?.callMode === CallMode.Group && call.peekInfo, {
|
|
uuids: [creatorUuid],
|
|
creatorUuid,
|
|
eraId: 'xyz',
|
|
maxDevices: 16,
|
|
deviceCount: 1,
|
|
});
|
|
});
|
|
|
|
it("can overwrite an existing group call's peek info", async () => {
|
|
const state = {
|
|
...getEmptyState(),
|
|
callsByConversation: {
|
|
'fake-conversation-id': {
|
|
...stateWithGroupCall.callsByConversation[
|
|
'fake-group-call-conversation-id'
|
|
],
|
|
conversationId: 'fake-conversation-id',
|
|
},
|
|
},
|
|
};
|
|
|
|
const result = await getState(state, {
|
|
callMode: CallMode.Group,
|
|
hasLocalAudio: true,
|
|
hasLocalVideo: true,
|
|
connectionState: GroupCallConnectionState.Connected,
|
|
joinState: GroupCallJoinState.NotJoined,
|
|
peekInfo: {
|
|
uuids: [differentCreatorUuid],
|
|
creatorUuid: differentCreatorUuid,
|
|
eraId: 'abc',
|
|
maxDevices: 5,
|
|
deviceCount: 1,
|
|
},
|
|
remoteParticipants: [
|
|
{
|
|
uuid: remoteUuid,
|
|
demuxId: 123,
|
|
hasRemoteAudio: true,
|
|
hasRemoteVideo: true,
|
|
presenting: false,
|
|
sharingScreen: false,
|
|
videoAspectRatio: 4 / 3,
|
|
},
|
|
],
|
|
});
|
|
|
|
const call = result.callsByConversation['fake-conversation-id'];
|
|
assert.deepEqual(call?.callMode === CallMode.Group && call.peekInfo, {
|
|
uuids: [differentCreatorUuid],
|
|
creatorUuid: differentCreatorUuid,
|
|
eraId: 'abc',
|
|
maxDevices: 5,
|
|
deviceCount: 1,
|
|
});
|
|
});
|
|
|
|
it("doesn't overwrite an existing group call's ring state if it was set previously", async () => {
|
|
const result = await getState(
|
|
{
|
|
...stateWithGroupCall,
|
|
callsByConversation: {
|
|
'fake-group-call-conversation-id': {
|
|
...stateWithGroupCall.callsByConversation[
|
|
'fake-group-call-conversation-id'
|
|
],
|
|
ringId: BigInt(987),
|
|
ringerUuid,
|
|
},
|
|
},
|
|
},
|
|
{
|
|
callMode: CallMode.Group,
|
|
hasLocalAudio: true,
|
|
hasLocalVideo: true,
|
|
connectionState: GroupCallConnectionState.Connected,
|
|
joinState: GroupCallJoinState.NotJoined,
|
|
peekInfo: undefined,
|
|
remoteParticipants: [
|
|
{
|
|
uuid: remoteUuid,
|
|
demuxId: 123,
|
|
hasRemoteAudio: true,
|
|
hasRemoteVideo: true,
|
|
presenting: false,
|
|
sharingScreen: false,
|
|
videoAspectRatio: 4 / 3,
|
|
},
|
|
],
|
|
}
|
|
);
|
|
const call =
|
|
result.callsByConversation['fake-group-call-conversation-id'];
|
|
// It'd be nice to do this with an assert, but Chai doesn't understand it.
|
|
if (call?.callMode !== CallMode.Group) {
|
|
throw new Error('Expected to find a group call');
|
|
}
|
|
|
|
assert.strictEqual(call.ringId, BigInt(987));
|
|
assert.strictEqual(call.ringerUuid, ringerUuid);
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('startCall', () => {
|
|
const { startCall } = actions;
|
|
|
|
beforeEach(function beforeEach() {
|
|
this.callingStartOutgoingDirectCall = this.sandbox.stub(
|
|
callingService,
|
|
'startOutgoingDirectCall'
|
|
);
|
|
this.callingJoinGroupCall = this.sandbox
|
|
.stub(callingService, 'joinGroupCall')
|
|
.resolves();
|
|
});
|
|
|
|
it('asks the calling service to start an outgoing direct call', async function test() {
|
|
const dispatch = sinon.spy();
|
|
await startCall({
|
|
callMode: CallMode.Direct,
|
|
conversationId: '123',
|
|
hasLocalAudio: true,
|
|
hasLocalVideo: false,
|
|
})(dispatch, getEmptyRootState, null);
|
|
|
|
sinon.assert.calledOnce(this.callingStartOutgoingDirectCall);
|
|
sinon.assert.calledWith(
|
|
this.callingStartOutgoingDirectCall,
|
|
'123',
|
|
true,
|
|
false
|
|
);
|
|
|
|
sinon.assert.notCalled(this.callingJoinGroupCall);
|
|
});
|
|
|
|
it('asks the calling service to join a group call', async function test() {
|
|
const dispatch = sinon.spy();
|
|
await startCall({
|
|
callMode: CallMode.Group,
|
|
conversationId: '123',
|
|
hasLocalAudio: true,
|
|
hasLocalVideo: false,
|
|
})(dispatch, getEmptyRootState, null);
|
|
|
|
sinon.assert.calledOnce(this.callingJoinGroupCall);
|
|
sinon.assert.calledWith(this.callingJoinGroupCall, '123', true, false);
|
|
|
|
sinon.assert.notCalled(this.callingStartOutgoingDirectCall);
|
|
});
|
|
|
|
it('saves direct calls and makes them active', async () => {
|
|
const dispatch = sinon.spy();
|
|
await startCall({
|
|
callMode: CallMode.Direct,
|
|
conversationId: 'fake-conversation-id',
|
|
hasLocalAudio: true,
|
|
hasLocalVideo: false,
|
|
})(dispatch, getEmptyRootState, null);
|
|
const action = dispatch.getCall(0).args[0];
|
|
|
|
const result = reducer(getEmptyState(), action);
|
|
|
|
assert.deepEqual(result.callsByConversation['fake-conversation-id'], {
|
|
callMode: CallMode.Direct,
|
|
conversationId: 'fake-conversation-id',
|
|
callState: CallState.Prering,
|
|
isIncoming: false,
|
|
isVideoCall: false,
|
|
});
|
|
assert.deepEqual(result.activeCallState, {
|
|
conversationId: 'fake-conversation-id',
|
|
hasLocalAudio: true,
|
|
hasLocalVideo: false,
|
|
amISpeaking: false,
|
|
isInSpeakerView: false,
|
|
showParticipantsList: false,
|
|
safetyNumberChangedUuids: [],
|
|
pip: false,
|
|
settingsDialogOpen: false,
|
|
outgoingRing: true,
|
|
});
|
|
});
|
|
|
|
it("doesn't dispatch any actions for group calls", () => {
|
|
const dispatch = sinon.spy();
|
|
startCall({
|
|
callMode: CallMode.Group,
|
|
conversationId: '123',
|
|
hasLocalAudio: true,
|
|
hasLocalVideo: false,
|
|
})(dispatch, getEmptyRootState, null);
|
|
|
|
sinon.assert.notCalled(dispatch);
|
|
});
|
|
});
|
|
|
|
describe('toggleSettings', () => {
|
|
const { toggleSettings } = actions;
|
|
|
|
it('toggles the settings dialog', () => {
|
|
const afterOneToggle = reducer(
|
|
stateWithActiveDirectCall,
|
|
toggleSettings()
|
|
);
|
|
const afterTwoToggles = reducer(afterOneToggle, toggleSettings());
|
|
const afterThreeToggles = reducer(afterTwoToggles, toggleSettings());
|
|
|
|
assert.isTrue(afterOneToggle.activeCallState?.settingsDialogOpen);
|
|
assert.isFalse(afterTwoToggles.activeCallState?.settingsDialogOpen);
|
|
assert.isTrue(afterThreeToggles.activeCallState?.settingsDialogOpen);
|
|
});
|
|
});
|
|
|
|
describe('toggleParticipants', () => {
|
|
const { toggleParticipants } = actions;
|
|
|
|
it('toggles the participants list', () => {
|
|
const afterOneToggle = reducer(
|
|
stateWithActiveDirectCall,
|
|
toggleParticipants()
|
|
);
|
|
const afterTwoToggles = reducer(afterOneToggle, toggleParticipants());
|
|
const afterThreeToggles = reducer(
|
|
afterTwoToggles,
|
|
toggleParticipants()
|
|
);
|
|
|
|
assert.isTrue(afterOneToggle.activeCallState?.showParticipantsList);
|
|
assert.isFalse(afterTwoToggles.activeCallState?.showParticipantsList);
|
|
assert.isTrue(afterThreeToggles.activeCallState?.showParticipantsList);
|
|
});
|
|
});
|
|
|
|
describe('togglePip', () => {
|
|
const { togglePip } = actions;
|
|
|
|
it('toggles the PiP', () => {
|
|
const afterOneToggle = reducer(stateWithActiveDirectCall, togglePip());
|
|
const afterTwoToggles = reducer(afterOneToggle, togglePip());
|
|
const afterThreeToggles = reducer(afterTwoToggles, togglePip());
|
|
|
|
assert.isTrue(afterOneToggle.activeCallState?.pip);
|
|
assert.isFalse(afterTwoToggles.activeCallState?.pip);
|
|
assert.isTrue(afterThreeToggles.activeCallState?.pip);
|
|
});
|
|
});
|
|
|
|
describe('toggleSpeakerView', () => {
|
|
const { toggleSpeakerView } = actions;
|
|
|
|
it('toggles speaker view', () => {
|
|
const afterOneToggle = reducer(
|
|
stateWithActiveGroupCall,
|
|
toggleSpeakerView()
|
|
);
|
|
const afterTwoToggles = reducer(afterOneToggle, toggleSpeakerView());
|
|
const afterThreeToggles = reducer(afterTwoToggles, toggleSpeakerView());
|
|
|
|
assert.isTrue(afterOneToggle.activeCallState?.isInSpeakerView);
|
|
assert.isFalse(afterTwoToggles.activeCallState?.isInSpeakerView);
|
|
assert.isTrue(afterThreeToggles.activeCallState?.isInSpeakerView);
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('helpers', () => {
|
|
describe('getActiveCall', () => {
|
|
it('returns undefined if there are no calls', () => {
|
|
assert.isUndefined(getActiveCall(getEmptyState()));
|
|
});
|
|
|
|
it('returns undefined if there is no active call', () => {
|
|
assert.isUndefined(getActiveCall(stateWithDirectCall));
|
|
});
|
|
|
|
it('returns the active call', () => {
|
|
assert.deepEqual(getActiveCall(stateWithActiveDirectCall), {
|
|
callMode: CallMode.Direct,
|
|
conversationId: 'fake-direct-call-conversation-id',
|
|
callState: CallState.Accepted,
|
|
isIncoming: false,
|
|
isVideoCall: false,
|
|
hasRemoteVideo: false,
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('isAnybodyElseInGroupCall', () => {
|
|
it('returns false with no peek info', () => {
|
|
assert.isFalse(isAnybodyElseInGroupCall(undefined, remoteUuid));
|
|
});
|
|
|
|
it('returns false if the peek info has no participants', () => {
|
|
assert.isFalse(isAnybodyElseInGroupCall({ uuids: [] }, remoteUuid));
|
|
});
|
|
|
|
it('returns false if the peek info has one participant, you', () => {
|
|
assert.isFalse(
|
|
isAnybodyElseInGroupCall({ uuids: [creatorUuid] }, creatorUuid)
|
|
);
|
|
});
|
|
|
|
it('returns true if the peek info has one participant, someone else', () => {
|
|
assert.isTrue(
|
|
isAnybodyElseInGroupCall({ uuids: [creatorUuid] }, remoteUuid)
|
|
);
|
|
});
|
|
|
|
it('returns true if the peek info has two participants, you and someone else', () => {
|
|
assert.isTrue(
|
|
isAnybodyElseInGroupCall(
|
|
{ uuids: [creatorUuid, remoteUuid] },
|
|
remoteUuid
|
|
)
|
|
);
|
|
});
|
|
});
|
|
});
|
|
});
|