@webex/calling 3.11.0 → 3.12.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/CallHistory/CallHistory.js +596 -0
- package/dist/CallHistory/CallHistory.js.map +1 -0
- package/dist/CallHistory/CallHistory.test.js +916 -0
- package/dist/CallHistory/CallHistory.test.js.map +1 -0
- package/dist/CallHistory/callHistoryFixtures.js +650 -0
- package/dist/CallHistory/callHistoryFixtures.js.map +1 -0
- package/dist/CallHistory/constants.js +38 -0
- package/dist/CallHistory/constants.js.map +1 -0
- package/dist/CallHistory/types.js +7 -0
- package/dist/CallHistory/types.js.map +1 -0
- package/dist/CallSettings/CallSettings.js +311 -0
- package/dist/CallSettings/CallSettings.js.map +1 -0
- package/dist/CallSettings/CallSettings.test.js +122 -0
- package/dist/CallSettings/CallSettings.test.js.map +1 -0
- package/dist/CallSettings/UcmBackendConnector.js +308 -0
- package/dist/CallSettings/UcmBackendConnector.js.map +1 -0
- package/dist/CallSettings/UcmBackendConnector.test.js +411 -0
- package/dist/CallSettings/UcmBackendConnector.test.js.map +1 -0
- package/dist/CallSettings/WxCallBackendConnector.js +597 -0
- package/dist/CallSettings/WxCallBackendConnector.js.map +1 -0
- package/dist/CallSettings/WxCallBackendConnector.test.js +904 -0
- package/dist/CallSettings/WxCallBackendConnector.test.js.map +1 -0
- package/dist/CallSettings/constants.js +31 -0
- package/dist/CallSettings/constants.js.map +1 -0
- package/dist/CallSettings/testFixtures.js +68 -0
- package/dist/CallSettings/testFixtures.js.map +1 -0
- package/dist/CallSettings/types.js +7 -0
- package/dist/CallSettings/types.js.map +1 -0
- package/dist/CallingClient/CallingClient.js +1119 -0
- package/dist/CallingClient/CallingClient.js.map +1 -0
- package/dist/CallingClient/CallingClient.test.js +1266 -0
- package/dist/CallingClient/CallingClient.test.js.map +1 -0
- package/dist/CallingClient/callRecordFixtures.js +101 -0
- package/dist/CallingClient/callRecordFixtures.js.map +1 -0
- package/dist/CallingClient/calling/CallerId/index.js +275 -0
- package/dist/CallingClient/calling/CallerId/index.js.map +1 -0
- package/dist/CallingClient/calling/CallerId/index.test.js +275 -0
- package/dist/CallingClient/calling/CallerId/index.test.js.map +1 -0
- package/dist/CallingClient/calling/CallerId/types.js +7 -0
- package/dist/CallingClient/calling/CallerId/types.js.map +1 -0
- package/dist/CallingClient/calling/call.js +3486 -0
- package/dist/CallingClient/calling/call.js.map +1 -0
- package/dist/CallingClient/calling/call.test.js +3612 -0
- package/dist/CallingClient/calling/call.test.js.map +1 -0
- package/dist/CallingClient/calling/callManager.js +460 -0
- package/dist/CallingClient/calling/callManager.js.map +1 -0
- package/dist/CallingClient/calling/callManager.test.js +741 -0
- package/dist/CallingClient/calling/callManager.test.js.map +1 -0
- package/dist/CallingClient/calling/index.js +30 -0
- package/dist/CallingClient/calling/index.js.map +1 -0
- package/dist/CallingClient/calling/types.js +74 -0
- package/dist/CallingClient/calling/types.js.map +1 -0
- package/dist/CallingClient/callingClientFixtures.js +143 -0
- package/dist/CallingClient/callingClientFixtures.js.map +1 -0
- package/dist/CallingClient/constants.js +243 -0
- package/dist/CallingClient/constants.js.map +1 -0
- package/dist/CallingClient/line/index.js +332 -0
- package/dist/CallingClient/line/index.js.map +1 -0
- package/dist/CallingClient/line/line.test.js +327 -0
- package/dist/CallingClient/line/line.test.js.map +1 -0
- package/dist/CallingClient/line/types.js +21 -0
- package/dist/CallingClient/line/types.js.map +1 -0
- package/dist/CallingClient/registration/index.js +19 -0
- package/dist/CallingClient/registration/index.js.map +1 -0
- package/dist/CallingClient/registration/register.js +1686 -0
- package/dist/CallingClient/registration/register.js.map +1 -0
- package/dist/CallingClient/registration/register.test.js +1885 -0
- package/dist/CallingClient/registration/register.test.js.map +1 -0
- package/dist/CallingClient/registration/registerFixtures.js +36 -0
- package/dist/CallingClient/registration/registerFixtures.js.map +1 -0
- package/dist/CallingClient/registration/types.js +7 -0
- package/dist/CallingClient/registration/types.js.map +1 -0
- package/dist/CallingClient/registration/webWorker.js +130 -0
- package/dist/CallingClient/registration/webWorker.js.map +1 -0
- package/dist/CallingClient/registration/webWorker.test.js +303 -0
- package/dist/CallingClient/registration/webWorker.test.js.map +1 -0
- package/dist/CallingClient/registration/webWorkerStr.js +15 -0
- package/dist/CallingClient/registration/webWorkerStr.js.map +1 -0
- package/dist/CallingClient/types.js +7 -0
- package/dist/CallingClient/types.js.map +1 -0
- package/dist/CallingClient/windowsChromiumIceWarmupUtils.js +142 -0
- package/dist/CallingClient/windowsChromiumIceWarmupUtils.js.map +1 -0
- package/dist/Contacts/ContactsClient.js +1206 -0
- package/dist/Contacts/ContactsClient.js.map +1 -0
- package/dist/Contacts/ContactsClient.test.js +1004 -0
- package/dist/Contacts/ContactsClient.test.js.map +1 -0
- package/dist/Contacts/constants.js +40 -0
- package/dist/Contacts/constants.js.map +1 -0
- package/dist/Contacts/contactFixtures.js +430 -0
- package/dist/Contacts/contactFixtures.js.map +1 -0
- package/dist/Contacts/types.js +43 -0
- package/dist/Contacts/types.js.map +1 -0
- package/dist/Errors/catalog/CallError.js +89 -0
- package/dist/Errors/catalog/CallError.js.map +1 -0
- package/dist/Errors/catalog/CallingDeviceError.js +83 -0
- package/dist/Errors/catalog/CallingDeviceError.js.map +1 -0
- package/dist/Errors/catalog/ExtendedError.js +42 -0
- package/dist/Errors/catalog/ExtendedError.js.map +1 -0
- package/dist/Errors/catalog/LineError.js +85 -0
- package/dist/Errors/catalog/LineError.js.map +1 -0
- package/dist/Errors/index.js +28 -0
- package/dist/Errors/index.js.map +1 -0
- package/dist/Errors/types.js +59 -0
- package/dist/Errors/types.js.map +1 -0
- package/dist/Events/impl/index.js +79 -0
- package/dist/Events/impl/index.js.map +1 -0
- package/dist/Events/types.js +107 -0
- package/dist/Events/types.js.map +1 -0
- package/dist/Logger/index.js +228 -0
- package/dist/Logger/index.js.map +1 -0
- package/dist/Logger/index.test.js +87 -0
- package/dist/Logger/index.test.js.map +1 -0
- package/dist/Logger/types.js +34 -0
- package/dist/Logger/types.js.map +1 -0
- package/dist/Metrics/index.js +534 -0
- package/dist/Metrics/index.js.map +1 -0
- package/dist/Metrics/index.test.js +463 -0
- package/dist/Metrics/index.test.js.map +1 -0
- package/dist/Metrics/types.js +64 -0
- package/dist/Metrics/types.js.map +1 -0
- package/dist/SDKConnector/index.js +102 -0
- package/dist/SDKConnector/index.js.map +1 -0
- package/dist/SDKConnector/index.test.js +9 -0
- package/dist/SDKConnector/index.test.js.map +1 -0
- package/dist/SDKConnector/types.js +7 -0
- package/dist/SDKConnector/types.js.map +1 -0
- package/dist/SDKConnector/utils.js +39 -0
- package/dist/SDKConnector/utils.js.map +1 -0
- package/dist/SDKConnector/utils.test.js +9 -0
- package/dist/SDKConnector/utils.test.js.map +1 -0
- package/dist/Voicemail/BroadworksBackendConnector.js +699 -0
- package/dist/Voicemail/BroadworksBackendConnector.js.map +1 -0
- package/dist/Voicemail/BroadworksBackendConnector.test.js +820 -0
- package/dist/Voicemail/BroadworksBackendConnector.test.js.map +1 -0
- package/dist/Voicemail/UcmBackendConnector.js +628 -0
- package/dist/Voicemail/UcmBackendConnector.js.map +1 -0
- package/dist/Voicemail/UcmBackendConnector.test.js +738 -0
- package/dist/Voicemail/UcmBackendConnector.test.js.map +1 -0
- package/dist/Voicemail/Voicemail.js +472 -0
- package/dist/Voicemail/Voicemail.js.map +1 -0
- package/dist/Voicemail/Voicemail.test.js +391 -0
- package/dist/Voicemail/Voicemail.test.js.map +1 -0
- package/dist/Voicemail/WxCallBackendConnector.js +657 -0
- package/dist/Voicemail/WxCallBackendConnector.js.map +1 -0
- package/dist/Voicemail/WxCallBackendConnector.test.js +1225 -0
- package/dist/Voicemail/WxCallBackendConnector.test.js.map +1 -0
- package/dist/Voicemail/constants.js +61 -0
- package/dist/Voicemail/constants.js.map +1 -0
- package/dist/Voicemail/types.js +7 -0
- package/dist/Voicemail/types.js.map +1 -0
- package/dist/Voicemail/voicemailFixture.js +524 -0
- package/dist/Voicemail/voicemailFixture.js.map +1 -0
- package/dist/api.js +157 -0
- package/dist/api.js.map +1 -0
- package/dist/common/Utils.js +1483 -0
- package/dist/common/Utils.js.map +1 -0
- package/dist/common/Utils.test.js +1989 -0
- package/dist/common/Utils.test.js.map +1 -0
- package/dist/common/constants.js +62 -0
- package/dist/common/constants.js.map +1 -0
- package/dist/common/index.js +19 -0
- package/dist/common/index.js.map +1 -0
- package/dist/common/testUtil.js +983 -0
- package/dist/common/testUtil.js.map +1 -0
- package/dist/common/types.js +75 -0
- package/dist/common/types.js.map +1 -0
- package/dist/index.js +321 -0
- package/dist/index.js.map +1 -0
- package/dist/module/CallHistory/CallHistory.js +28 -17
- package/dist/module/CallingClient/CallingClient.js +43 -1
- package/dist/module/CallingClient/calling/call.js +3 -2
- package/dist/module/CallingClient/constants.js +2 -0
- package/dist/module/CallingClient/registration/register.js +8 -0
- package/dist/types/CallHistory/CallHistory.d.ts.map +1 -1
- package/dist/types/CallingClient/CallingClient.d.ts +2 -1
- package/dist/types/CallingClient/CallingClient.d.ts.map +1 -1
- package/dist/types/CallingClient/calling/call.d.ts.map +1 -1
- package/dist/types/CallingClient/constants.d.ts +2 -0
- package/dist/types/CallingClient/constants.d.ts.map +1 -1
- package/dist/types/CallingClient/registration/register.d.ts +2 -1
- package/dist/types/CallingClient/registration/register.d.ts.map +1 -1
- package/dist/types/CallingClient/registration/types.d.ts +2 -1
- package/dist/types/CallingClient/registration/types.d.ts.map +1 -1
- package/dist/types/CallingClient/types.d.ts +2 -1
- package/dist/types/CallingClient/types.d.ts.map +1 -1
- package/dist/types/common/types.d.ts +4 -0
- package/dist/types/common/types.d.ts.map +1 -1
- package/package.json +4 -4
|
@@ -0,0 +1,3612 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
var _typeof = require("@babel/runtime-corejs2/helpers/typeof");
|
|
4
|
+
var _WeakMap = require("@babel/runtime-corejs2/core-js/weak-map");
|
|
5
|
+
var _Object$defineProperty = require("@babel/runtime-corejs2/core-js/object/define-property");
|
|
6
|
+
var _Object$getOwnPropertyDescriptor = require("@babel/runtime-corejs2/core-js/object/get-own-property-descriptor");
|
|
7
|
+
var _interopRequireDefault = require("@babel/runtime-corejs2/helpers/interopRequireDefault");
|
|
8
|
+
var _regenerator = _interopRequireDefault(require("@babel/runtime-corejs2/regenerator"));
|
|
9
|
+
var _keys = _interopRequireDefault(require("@babel/runtime-corejs2/core-js/object/keys"));
|
|
10
|
+
var _promise = _interopRequireDefault(require("@babel/runtime-corejs2/core-js/promise"));
|
|
11
|
+
var _now = _interopRequireDefault(require("@babel/runtime-corejs2/core-js/date/now"));
|
|
12
|
+
var _stringify = _interopRequireDefault(require("@babel/runtime-corejs2/core-js/json/stringify"));
|
|
13
|
+
var _asyncToGenerator2 = _interopRequireDefault(require("@babel/runtime-corejs2/helpers/asyncToGenerator"));
|
|
14
|
+
var InternalMediaCoreModule = _interopRequireWildcard(require("@webex/internal-media-core"));
|
|
15
|
+
var _mediaHelpers = require("@webex/media-helpers");
|
|
16
|
+
var _types = require("../../Errors/types");
|
|
17
|
+
var _Utils = _interopRequireWildcard(require("../../common/Utils"));
|
|
18
|
+
var Utils = _Utils;
|
|
19
|
+
var _types2 = require("../../Events/types");
|
|
20
|
+
var _constants = require("../constants");
|
|
21
|
+
var _types3 = require("../../common/types");
|
|
22
|
+
var _types4 = require("../../Metrics/types");
|
|
23
|
+
var _call = require("./call");
|
|
24
|
+
var _types5 = require("./types");
|
|
25
|
+
var _testUtil = require("../../common/testUtil");
|
|
26
|
+
var _callManager = require("./callManager");
|
|
27
|
+
var _Logger = _interopRequireDefault(require("../../Logger"));
|
|
28
|
+
var _Errors = require("../../Errors");
|
|
29
|
+
var _constants2 = require("../../common/constants");
|
|
30
|
+
function _interopRequireWildcard(e, t) { if ("function" == typeof _WeakMap) var r = new _WeakMap(), n = new _WeakMap(); return (_interopRequireWildcard = function _interopRequireWildcard(e, t) { if (!t && e && e.__esModule) return e; var o, i, f = { __proto__: null, default: e }; if (null === e || "object" != _typeof(e) && "function" != typeof e) return f; if (o = t ? n : r) { if (o.has(e)) return o.get(e); o.set(e, f); } for (var _t2 in e) "default" !== _t2 && {}.hasOwnProperty.call(e, _t2) && ((i = (o = _Object$defineProperty) && _Object$getOwnPropertyDescriptor(e, _t2)) && (i.get || i.set) ? o(f, _t2, i) : f[_t2] = e[_t2]); return f; })(e, t); }
|
|
31
|
+
/* eslint-disable @typescript-eslint/no-explicit-any */
|
|
32
|
+
/* eslint-disable dot-notation */
|
|
33
|
+
/* eslint-disable @typescript-eslint/no-shadow */
|
|
34
|
+
|
|
35
|
+
jest.mock('@webex/internal-media-core');
|
|
36
|
+
var uploadLogsSpy = jest.spyOn(Utils, 'uploadLogs').mockResolvedValue(undefined);
|
|
37
|
+
var webex = (0, _testUtil.getTestUtilsWebex)();
|
|
38
|
+
var mockInternalMediaCoreModule = InternalMediaCoreModule;
|
|
39
|
+
var defaultServiceIndicator = _types3.ServiceIndicator.CALLING;
|
|
40
|
+
var activeUrl = 'FakeActiveUrl';
|
|
41
|
+
var mockLineId = 'e4e8ee2a-a154-4e52-8f11-ef4cde2dce72';
|
|
42
|
+
describe('Call Tests', function () {
|
|
43
|
+
var deviceId = '55dfb53f-bed2-36da-8e85-cee7f02aa68e';
|
|
44
|
+
var dest = {
|
|
45
|
+
type: _types3.CallType.URI,
|
|
46
|
+
address: 'tel:5003'
|
|
47
|
+
};
|
|
48
|
+
var dummyMidCallEvent = {
|
|
49
|
+
eventType: 'callInfo',
|
|
50
|
+
eventData: {
|
|
51
|
+
callerId: {
|
|
52
|
+
from: '"Bob Marley" <sip:5010@207.182.171.130;user=phone>;tag=888068389-1654853820619-'
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
};
|
|
56
|
+
var callManager;
|
|
57
|
+
var deleteCallFromCollection = jest.fn();
|
|
58
|
+
var disconnectStats = {
|
|
59
|
+
'rtp-rxstat': {
|
|
60
|
+
Dur: 53.77999999999787,
|
|
61
|
+
LostPkt: 11,
|
|
62
|
+
LatePkt: 0,
|
|
63
|
+
Oct: 41156,
|
|
64
|
+
Pkt: 2379,
|
|
65
|
+
AvgJit: 0.041720656289440466,
|
|
66
|
+
VQMetrics: {
|
|
67
|
+
maxJitter: 0,
|
|
68
|
+
VoPktSizeMs: 20,
|
|
69
|
+
VoOneWayDelayMs: 0.26116666666666666,
|
|
70
|
+
hwType: 'OS X 10.15.7 64-bit/Chrome-105.0.0.0',
|
|
71
|
+
networkType: 'wifi',
|
|
72
|
+
VoRxCodec: 'opus'
|
|
73
|
+
}
|
|
74
|
+
},
|
|
75
|
+
'rtp-txstat': {
|
|
76
|
+
Dur: 53.77999999999787,
|
|
77
|
+
Pkt: 2410,
|
|
78
|
+
Oct: 41156,
|
|
79
|
+
VQMetrics: {
|
|
80
|
+
VoTxCodec: 'opus',
|
|
81
|
+
rtpBitRate: 32000
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
};
|
|
85
|
+
var parseMediaQualityStatisticsMock = jest.spyOn(Utils, 'parseMediaQualityStatistics').mockReturnValue(disconnectStats);
|
|
86
|
+
var mockTrack = {
|
|
87
|
+
enabled: false
|
|
88
|
+
};
|
|
89
|
+
var mockEffect = {
|
|
90
|
+
isEnabled: true,
|
|
91
|
+
effectTrack: mockTrack,
|
|
92
|
+
on: jest.fn(),
|
|
93
|
+
off: jest.fn()
|
|
94
|
+
};
|
|
95
|
+
var roapMediaConnectionConfig = {
|
|
96
|
+
skipInactiveTransceivers: true,
|
|
97
|
+
iceServers: [],
|
|
98
|
+
iceCandidatesTimeout: _constants.ICE_CANDIDATES_TIMEOUT,
|
|
99
|
+
sdpMunging: {
|
|
100
|
+
convertPort9to0: true,
|
|
101
|
+
addContentSlides: false,
|
|
102
|
+
copyClineToSessionLevel: true
|
|
103
|
+
}
|
|
104
|
+
};
|
|
105
|
+
var roapMediaConnectionOptions = {
|
|
106
|
+
localTracks: {
|
|
107
|
+
audio: mockTrack
|
|
108
|
+
},
|
|
109
|
+
direction: {
|
|
110
|
+
audio: 'sendrecv',
|
|
111
|
+
video: 'inactive',
|
|
112
|
+
screenShareVideo: 'inactive'
|
|
113
|
+
}
|
|
114
|
+
};
|
|
115
|
+
afterEach(function () {
|
|
116
|
+
webex.request = jest.fn();
|
|
117
|
+
});
|
|
118
|
+
beforeEach(function () {
|
|
119
|
+
callManager = (0, _callManager.getCallManager)(webex, defaultServiceIndicator);
|
|
120
|
+
});
|
|
121
|
+
it('create call object', function () {
|
|
122
|
+
webex.request.mockReturnValueOnce({
|
|
123
|
+
statusCode: 200,
|
|
124
|
+
body: {
|
|
125
|
+
device: {
|
|
126
|
+
deviceId: '8a67806f-fc4d-446b-a131-31e71ea5b010',
|
|
127
|
+
correlationId: '8a67806f-fc4d-446b-a131-31e71ea5b011'
|
|
128
|
+
},
|
|
129
|
+
callId: '8a67806f-fc4d-446b-a131-31e71ea5b020',
|
|
130
|
+
callData: {
|
|
131
|
+
callState: _types5.MobiusCallState.PROCEEDING
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
});
|
|
135
|
+
var call = (0, _call.createCall)(activeUrl, webex, _types3.CallDirection.OUTBOUND, deviceId, mockLineId, deleteCallFromCollection, defaultServiceIndicator, dest);
|
|
136
|
+
expect(call).toBeTruthy();
|
|
137
|
+
});
|
|
138
|
+
it('should log a warning when sending a digit fails', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee() {
|
|
139
|
+
var tone, errorMessage, mockMediaConnection, callManager, call, realMediaConnection, warnSpy, infoSpy;
|
|
140
|
+
return _regenerator.default.wrap(function (_context) {
|
|
141
|
+
while (1) switch (_context.prev = _context.next) {
|
|
142
|
+
case 0:
|
|
143
|
+
tone = '1';
|
|
144
|
+
errorMessage = 'Failed to send digit'; // Mock the mediaConnection object
|
|
145
|
+
mockMediaConnection = {
|
|
146
|
+
insertDTMF: jest.fn(function () {
|
|
147
|
+
throw new Error(errorMessage);
|
|
148
|
+
})
|
|
149
|
+
};
|
|
150
|
+
callManager = (0, _callManager.getCallManager)(webex, defaultServiceIndicator);
|
|
151
|
+
call = callManager.createCall(_types3.CallDirection.OUTBOUND, deviceId, mockLineId, dest);
|
|
152
|
+
realMediaConnection = call.mediaConnection; // Set the mock mediaConnection object
|
|
153
|
+
call.mediaConnection = mockMediaConnection;
|
|
154
|
+
|
|
155
|
+
// Spy on the log methods
|
|
156
|
+
warnSpy = jest.spyOn(_Logger.default, 'warn');
|
|
157
|
+
infoSpy = jest.spyOn(_Logger.default, 'info'); // Call the sendDigit method
|
|
158
|
+
call.sendDigit(tone);
|
|
159
|
+
|
|
160
|
+
// Expect log.info to be called with the digit being sent
|
|
161
|
+
expect(infoSpy).toHaveBeenCalledWith("".concat(_constants2.METHOD_START_MESSAGE, " with: ").concat(tone), {
|
|
162
|
+
file: 'call',
|
|
163
|
+
method: 'sendDigit'
|
|
164
|
+
});
|
|
165
|
+
|
|
166
|
+
// Expect the log.warn method to be called with the error message
|
|
167
|
+
expect(warnSpy).toHaveBeenCalledWith("Unable to send digit on call: ".concat(errorMessage), {
|
|
168
|
+
file: 'call',
|
|
169
|
+
method: 'sendDigit'
|
|
170
|
+
});
|
|
171
|
+
|
|
172
|
+
// Restore the real mediaConnection object
|
|
173
|
+
call.mediaConnection = realMediaConnection;
|
|
174
|
+
call.end();
|
|
175
|
+
_context.next = 1;
|
|
176
|
+
return (0, _Utils.waitForMsecs)(50);
|
|
177
|
+
case 1:
|
|
178
|
+
// Need to add a small delay for Promise and callback to finish.
|
|
179
|
+
|
|
180
|
+
/* After call ends, call manager should have 0 record */
|
|
181
|
+
expect((0, _keys.default)(callManager.getActiveCalls()).length).toBe(0);
|
|
182
|
+
case 2:
|
|
183
|
+
case "end":
|
|
184
|
+
return _context.stop();
|
|
185
|
+
}
|
|
186
|
+
}, _callee);
|
|
187
|
+
})));
|
|
188
|
+
it('delete call object when ending the call', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee2() {
|
|
189
|
+
var callManager, mockStream, localAudioStream, call, infoSpy;
|
|
190
|
+
return _regenerator.default.wrap(function (_context2) {
|
|
191
|
+
while (1) switch (_context2.prev = _context2.next) {
|
|
192
|
+
case 0:
|
|
193
|
+
webex.request.mockReturnValue({
|
|
194
|
+
statusCode: 200,
|
|
195
|
+
body: {
|
|
196
|
+
device: {
|
|
197
|
+
deviceId: '8a67806f-fc4d-446b-a131-31e71ea5b010',
|
|
198
|
+
correlationId: '8a67806f-fc4d-446b-a131-31e71ea5b011'
|
|
199
|
+
},
|
|
200
|
+
callId: '8a67806f-fc4d-446b-a131-31e71ea5b020'
|
|
201
|
+
}
|
|
202
|
+
});
|
|
203
|
+
callManager = (0, _callManager.getCallManager)(webex, defaultServiceIndicator);
|
|
204
|
+
mockStream = {
|
|
205
|
+
on: jest.fn(),
|
|
206
|
+
setUserMuted: jest.fn()
|
|
207
|
+
};
|
|
208
|
+
localAudioStream = mockStream;
|
|
209
|
+
call = callManager.createCall(_types3.CallDirection.OUTBOUND, deviceId, mockLineId, dest);
|
|
210
|
+
infoSpy = jest.spyOn(_Logger.default, 'info');
|
|
211
|
+
expect(call).toBeTruthy();
|
|
212
|
+
/* After creation , call manager should have 1 record */
|
|
213
|
+
expect((0, _keys.default)(callManager.getActiveCalls()).length).toBe(1);
|
|
214
|
+
call.mute(localAudioStream);
|
|
215
|
+
expect(call.isMuted()).toEqual(true);
|
|
216
|
+
expect(mockStream.setUserMuted).toBeCalledOnceWith(true);
|
|
217
|
+
expect(infoSpy).toHaveBeenCalledWith('invoking with: user mute', {
|
|
218
|
+
file: 'call',
|
|
219
|
+
method: 'mute'
|
|
220
|
+
});
|
|
221
|
+
call.mute(localAudioStream);
|
|
222
|
+
expect(mockStream.setUserMuted).toBeCalledWith(false);
|
|
223
|
+
expect(call.isMuted()).toEqual(false);
|
|
224
|
+
call.end();
|
|
225
|
+
expect(infoSpy).toHaveBeenCalledWith(_constants2.METHOD_START_MESSAGE, {
|
|
226
|
+
file: 'call',
|
|
227
|
+
method: 'end'
|
|
228
|
+
});
|
|
229
|
+
_context2.next = 1;
|
|
230
|
+
return (0, _Utils.waitForMsecs)(50);
|
|
231
|
+
case 1:
|
|
232
|
+
// Need to add a small delay for Promise and callback to finish.
|
|
233
|
+
expect(parseMediaQualityStatisticsMock).toHaveBeenCalledTimes(1);
|
|
234
|
+
expect(webex.request.mock.calls[0][0].body.metrics).toStrictEqual(disconnectStats);
|
|
235
|
+
expect(call.getDisconnectReason().code).toBe(_types5.DisconnectCode.NORMAL);
|
|
236
|
+
expect(call.getDisconnectReason().cause).toBe(_types5.DisconnectCause.NORMAL);
|
|
237
|
+
|
|
238
|
+
/* After call ends, call manager should have 0 record */
|
|
239
|
+
expect((0, _keys.default)(callManager.getActiveCalls()).length).toBe(0);
|
|
240
|
+
case 2:
|
|
241
|
+
case "end":
|
|
242
|
+
return _context2.stop();
|
|
243
|
+
}
|
|
244
|
+
}, _callee2);
|
|
245
|
+
})));
|
|
246
|
+
it('Check whether media requests succeed or not', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee3() {
|
|
247
|
+
var mediaResponse, call, response;
|
|
248
|
+
return _regenerator.default.wrap(function (_context3) {
|
|
249
|
+
while (1) switch (_context3.prev = _context3.next) {
|
|
250
|
+
case 0:
|
|
251
|
+
webex.request.mockReturnValue({
|
|
252
|
+
statusCode: 200,
|
|
253
|
+
body: {
|
|
254
|
+
device: {
|
|
255
|
+
deviceId: '8a67806f-fc4d-446b-a131-31e71ea5b010',
|
|
256
|
+
correlationId: '8a67806f-fc4d-446b-a131-31e71ea5b011'
|
|
257
|
+
},
|
|
258
|
+
callId: '8a67806f-fc4d-446b-a131-31e71ea5b020'
|
|
259
|
+
}
|
|
260
|
+
});
|
|
261
|
+
mediaResponse = {
|
|
262
|
+
statusCode: 200,
|
|
263
|
+
body: {
|
|
264
|
+
device: {
|
|
265
|
+
deviceId: '8a67806f-fc4d-446b-a131-31e71ea5b010',
|
|
266
|
+
correlationId: '8a67806f-fc4d-446b-a131-31e71ea5b011'
|
|
267
|
+
},
|
|
268
|
+
callId: '8a67806f-fc4d-446b-a131-31e71ea5b020'
|
|
269
|
+
}
|
|
270
|
+
};
|
|
271
|
+
call = callManager.createCall(_types3.CallDirection.INBOUND, deviceId, mockLineId, dest);
|
|
272
|
+
_context3.next = 1;
|
|
273
|
+
return call['postMedia']({});
|
|
274
|
+
case 1:
|
|
275
|
+
response = _context3.sent;
|
|
276
|
+
expect(response.body).toStrictEqual(mediaResponse.body);
|
|
277
|
+
case 2:
|
|
278
|
+
case "end":
|
|
279
|
+
return _context3.stop();
|
|
280
|
+
}
|
|
281
|
+
}, _callee3);
|
|
282
|
+
})));
|
|
283
|
+
it('check whether callerId midcall event is serviced or not', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee4() {
|
|
284
|
+
var call;
|
|
285
|
+
return _regenerator.default.wrap(function (_context4) {
|
|
286
|
+
while (1) switch (_context4.prev = _context4.next) {
|
|
287
|
+
case 0:
|
|
288
|
+
call = callManager.createCall(_types3.CallDirection.OUTBOUND, deviceId, mockLineId, dest);
|
|
289
|
+
call.handleMidCallEvent(dummyMidCallEvent);
|
|
290
|
+
_context4.next = 1;
|
|
291
|
+
return (0, _Utils.waitForMsecs)(50);
|
|
292
|
+
case 1:
|
|
293
|
+
expect(call.getCallerInfo().name).toStrictEqual('Bob Marley');
|
|
294
|
+
expect(call.getCallerInfo().num).toStrictEqual('5010');
|
|
295
|
+
expect(call.getCallerInfo().avatarSrc).toBeFalsy();
|
|
296
|
+
case 2:
|
|
297
|
+
case "end":
|
|
298
|
+
return _context4.stop();
|
|
299
|
+
}
|
|
300
|
+
}, _callee4);
|
|
301
|
+
})));
|
|
302
|
+
it('check whether call midcall event is serviced or not', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee5() {
|
|
303
|
+
var call, logSpy, corelationId;
|
|
304
|
+
return _regenerator.default.wrap(function (_context5) {
|
|
305
|
+
while (1) switch (_context5.prev = _context5.next) {
|
|
306
|
+
case 0:
|
|
307
|
+
call = callManager.createCall(_types3.CallDirection.OUTBOUND, deviceId, mockLineId, dest);
|
|
308
|
+
dummyMidCallEvent.eventType = 'callState';
|
|
309
|
+
logSpy = jest.spyOn(_Logger.default, 'log');
|
|
310
|
+
call.handleMidCallEvent(dummyMidCallEvent);
|
|
311
|
+
_context5.next = 1;
|
|
312
|
+
return (0, _Utils.waitForMsecs)(50);
|
|
313
|
+
case 1:
|
|
314
|
+
corelationId = call.getCorrelationId();
|
|
315
|
+
expect(logSpy).toHaveBeenLastCalledWith("Received Midcall call event for correlationId : ".concat(corelationId), {
|
|
316
|
+
file: 'call',
|
|
317
|
+
method: 'handleMidCallEvent'
|
|
318
|
+
});
|
|
319
|
+
case 2:
|
|
320
|
+
case "end":
|
|
321
|
+
return _context5.stop();
|
|
322
|
+
}
|
|
323
|
+
}, _callee5);
|
|
324
|
+
})));
|
|
325
|
+
it('check call stats for active call', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee6() {
|
|
326
|
+
var call, callRtpStats, _t;
|
|
327
|
+
return _regenerator.default.wrap(function (_context6) {
|
|
328
|
+
while (1) switch (_context6.prev = _context6.next) {
|
|
329
|
+
case 0:
|
|
330
|
+
call = callManager.createCall(_types3.CallDirection.OUTBOUND, deviceId, mockLineId, dest);
|
|
331
|
+
_context6.prev = 1;
|
|
332
|
+
_context6.next = 2;
|
|
333
|
+
return call.getCallRtpStats();
|
|
334
|
+
case 2:
|
|
335
|
+
callRtpStats = _context6.sent;
|
|
336
|
+
_context6.next = 4;
|
|
337
|
+
break;
|
|
338
|
+
case 3:
|
|
339
|
+
_context6.prev = 3;
|
|
340
|
+
_t = _context6["catch"](1);
|
|
341
|
+
console.error(_t);
|
|
342
|
+
case 4:
|
|
343
|
+
expect(callRtpStats).toStrictEqual(disconnectStats);
|
|
344
|
+
case 5:
|
|
345
|
+
case "end":
|
|
346
|
+
return _context6.stop();
|
|
347
|
+
}
|
|
348
|
+
}, _callee6, null, [[1, 3]]);
|
|
349
|
+
})));
|
|
350
|
+
it('dial functionality tests for coverage', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee7() {
|
|
351
|
+
var mockStream, localAudioStream, warnSpy, call, bnrMetricSpy, logSpy;
|
|
352
|
+
return _regenerator.default.wrap(function (_context7) {
|
|
353
|
+
while (1) switch (_context7.prev = _context7.next) {
|
|
354
|
+
case 0:
|
|
355
|
+
mockStream = {
|
|
356
|
+
outputStream: {
|
|
357
|
+
getAudioTracks: jest.fn().mockReturnValue([mockTrack])
|
|
358
|
+
},
|
|
359
|
+
on: jest.fn(),
|
|
360
|
+
getEffectByKind: jest.fn().mockImplementation(function () {
|
|
361
|
+
return mockEffect;
|
|
362
|
+
})
|
|
363
|
+
};
|
|
364
|
+
localAudioStream = mockStream;
|
|
365
|
+
warnSpy = jest.spyOn(_Logger.default, 'warn');
|
|
366
|
+
call = (0, _call.createCall)(activeUrl, webex, _types3.CallDirection.OUTBOUND, deviceId, mockLineId, deleteCallFromCollection, defaultServiceIndicator, dest);
|
|
367
|
+
bnrMetricSpy = jest.spyOn(call['metricManager'], 'submitBNRMetric');
|
|
368
|
+
logSpy = jest.spyOn(_Logger.default, 'log');
|
|
369
|
+
call.dial(localAudioStream);
|
|
370
|
+
expect(mockTrack.enabled).toEqual(true);
|
|
371
|
+
expect(mockInternalMediaCoreModule.RoapMediaConnection).toBeCalledOnceWith(roapMediaConnectionConfig, roapMediaConnectionOptions, expect.any(String), expect.any(Function), expect.any(Function), expect.any(Function));
|
|
372
|
+
expect(call['mediaStateMachine'].state.value).toBe('S_SEND_ROAP_OFFER');
|
|
373
|
+
expect(logSpy).toHaveBeenCalled();
|
|
374
|
+
expect(bnrMetricSpy).toBeCalledOnceWith(_types4.METRIC_EVENT.BNR_ENABLED, _types4.METRIC_TYPE.BEHAVIORAL, call.getCallId(), call.getCorrelationId());
|
|
375
|
+
|
|
376
|
+
/* Now change the state and recall to check for error */
|
|
377
|
+
call['mediaStateMachine'].state.value = 'S_SEND_ROAP_OFFER';
|
|
378
|
+
call.dial(localAudioStream);
|
|
379
|
+
expect(call['mediaStateMachine'].state.value).toBe('S_SEND_ROAP_OFFER');
|
|
380
|
+
expect(warnSpy).toBeCalledOnceWith("Call cannot be dialed because the state is already : S_SEND_ROAP_OFFER", {
|
|
381
|
+
file: 'call',
|
|
382
|
+
method: 'dial'
|
|
383
|
+
});
|
|
384
|
+
case 1:
|
|
385
|
+
case "end":
|
|
386
|
+
return _context7.stop();
|
|
387
|
+
}
|
|
388
|
+
}, _callee7);
|
|
389
|
+
})));
|
|
390
|
+
it('answer functionality tests for coverage', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee8() {
|
|
391
|
+
var mockStream, localAudioStream, warnSpy, call, bnrMetricSpy, infoSpy;
|
|
392
|
+
return _regenerator.default.wrap(function (_context8) {
|
|
393
|
+
while (1) switch (_context8.prev = _context8.next) {
|
|
394
|
+
case 0:
|
|
395
|
+
mockStream = {
|
|
396
|
+
outputStream: {
|
|
397
|
+
getAudioTracks: jest.fn().mockReturnValue([mockTrack])
|
|
398
|
+
},
|
|
399
|
+
on: jest.fn(),
|
|
400
|
+
getEffectByKind: jest.fn().mockImplementation(function () {
|
|
401
|
+
return mockEffect;
|
|
402
|
+
})
|
|
403
|
+
};
|
|
404
|
+
localAudioStream = mockStream;
|
|
405
|
+
warnSpy = jest.spyOn(_Logger.default, 'warn');
|
|
406
|
+
call = (0, _call.createCall)(activeUrl, webex, _types3.CallDirection.OUTBOUND, deviceId, mockLineId, deleteCallFromCollection, defaultServiceIndicator, dest);
|
|
407
|
+
/** Cannot answer in idle state */
|
|
408
|
+
bnrMetricSpy = jest.spyOn(call['metricManager'], 'submitBNRMetric');
|
|
409
|
+
infoSpy = jest.spyOn(_Logger.default, 'info');
|
|
410
|
+
call.answer(localAudioStream);
|
|
411
|
+
expect(mockTrack.enabled).toEqual(true);
|
|
412
|
+
expect(mockInternalMediaCoreModule.RoapMediaConnection).toBeCalledOnceWith(roapMediaConnectionConfig, roapMediaConnectionOptions, expect.any(String), expect.any(Function), expect.any(Function), expect.any(Function));
|
|
413
|
+
expect(call['callStateMachine'].state.value).toBe('S_IDLE');
|
|
414
|
+
expect(warnSpy).toBeCalledOnceWith("Call cannot be answered because the state is : S_IDLE", {
|
|
415
|
+
file: 'call',
|
|
416
|
+
method: 'answer'
|
|
417
|
+
});
|
|
418
|
+
expect(infoSpy).toHaveBeenCalled();
|
|
419
|
+
|
|
420
|
+
/* Now change the state and recall to check for correct flow */
|
|
421
|
+
call['callStateMachine'].state.value = 'S_SEND_CALL_PROGRESS';
|
|
422
|
+
call.answer(localAudioStream);
|
|
423
|
+
expect(call['callStateMachine'].state.value).toBe('S_SEND_CALL_CONNECT');
|
|
424
|
+
expect(bnrMetricSpy).toBeCalledOnceWith(_types4.METRIC_EVENT.BNR_ENABLED, _types4.METRIC_TYPE.BEHAVIORAL, call.getCallId(), call.getCorrelationId());
|
|
425
|
+
case 1:
|
|
426
|
+
case "end":
|
|
427
|
+
return _context8.stop();
|
|
428
|
+
}
|
|
429
|
+
}, _callee8);
|
|
430
|
+
})));
|
|
431
|
+
it('testing enabling/disabling the BNR on an active call', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee9() {
|
|
432
|
+
var mockStream, localAudioStream, onStreamSpy, onEffectSpy, offStreamSpy, offEffectSpy, call, updateLocalTracksSpy, bnrMetricSpy;
|
|
433
|
+
return _regenerator.default.wrap(function (_context9) {
|
|
434
|
+
while (1) switch (_context9.prev = _context9.next) {
|
|
435
|
+
case 0:
|
|
436
|
+
mockStream = {
|
|
437
|
+
outputStream: {
|
|
438
|
+
getAudioTracks: jest.fn().mockReturnValue([mockTrack])
|
|
439
|
+
},
|
|
440
|
+
on: jest.fn(),
|
|
441
|
+
off: jest.fn(),
|
|
442
|
+
getEffectByKind: jest.fn()
|
|
443
|
+
};
|
|
444
|
+
localAudioStream = mockStream;
|
|
445
|
+
onStreamSpy = jest.spyOn(localAudioStream, 'on');
|
|
446
|
+
onEffectSpy = jest.spyOn(mockEffect, 'on');
|
|
447
|
+
offStreamSpy = jest.spyOn(localAudioStream, 'off');
|
|
448
|
+
offEffectSpy = jest.spyOn(mockEffect, 'off');
|
|
449
|
+
call = (0, _call.createCall)(activeUrl, webex, _types3.CallDirection.OUTBOUND, deviceId, mockLineId, deleteCallFromCollection, defaultServiceIndicator, dest);
|
|
450
|
+
call.dial(localAudioStream);
|
|
451
|
+
expect(mockTrack.enabled).toEqual(true);
|
|
452
|
+
expect(mockInternalMediaCoreModule.RoapMediaConnection).toBeCalledOnceWith(roapMediaConnectionConfig, roapMediaConnectionOptions, expect.any(String), expect.any(Function), expect.any(Function), expect.any(Function));
|
|
453
|
+
expect(call['mediaStateMachine'].state.value).toBe('S_SEND_ROAP_OFFER');
|
|
454
|
+
updateLocalTracksSpy = jest.spyOn(call['mediaConnection'], 'updateLocalTracks');
|
|
455
|
+
bnrMetricSpy = jest.spyOn(call['metricManager'], 'submitBNRMetric');
|
|
456
|
+
/* Update the stream with the effect */
|
|
457
|
+
jest.spyOn(localAudioStream, 'getEffectByKind').mockReturnValue(mockEffect);
|
|
458
|
+
|
|
459
|
+
/* Checking if listeners on the localAudioStream have been registered */
|
|
460
|
+
expect(onStreamSpy).toBeCalledTimes(2);
|
|
461
|
+
expect(onStreamSpy).toBeCalledWith(InternalMediaCoreModule.LocalStreamEventNames.OutputTrackChange, expect.any(Function));
|
|
462
|
+
expect(onStreamSpy).toBeCalledWith(InternalMediaCoreModule.LocalStreamEventNames.EffectAdded, expect.any(Function));
|
|
463
|
+
bnrMetricSpy.mockClear();
|
|
464
|
+
/* Invoking the callback function to trigger EffectAdded event to simulate adding effect to the stream */
|
|
465
|
+
onStreamSpy.mock.calls[1][1](mockEffect);
|
|
466
|
+
expect(onEffectSpy).toBeCalledWith(_mediaHelpers.EffectEvent.Enabled, expect.any(Function));
|
|
467
|
+
expect(onEffectSpy).toBeCalledWith(_mediaHelpers.EffectEvent.Disabled, expect.any(Function));
|
|
468
|
+
|
|
469
|
+
/* Send Enabled event on the effect, update track and send metrics for BNR disabled */
|
|
470
|
+
onStreamSpy.mock.calls[0][1](mockTrack);
|
|
471
|
+
onEffectSpy.mock.calls[0][1]();
|
|
472
|
+
expect(updateLocalTracksSpy).toBeCalledOnceWith({
|
|
473
|
+
audio: mockTrack
|
|
474
|
+
});
|
|
475
|
+
expect(bnrMetricSpy).toBeCalledOnceWith(_types4.METRIC_EVENT.BNR_ENABLED, _types4.METRIC_TYPE.BEHAVIORAL, call.getCallId(), call.getCorrelationId());
|
|
476
|
+
|
|
477
|
+
/* Clear the mocks */
|
|
478
|
+
updateLocalTracksSpy.mockClear();
|
|
479
|
+
bnrMetricSpy.mockClear();
|
|
480
|
+
|
|
481
|
+
/* Send Disabled event on the effect, update track and send metrics for BNR disabled */
|
|
482
|
+
mockEffect.isEnabled = false;
|
|
483
|
+
onStreamSpy.mock.calls[0][1](mockTrack);
|
|
484
|
+
onEffectSpy.mock.calls[1][1]();
|
|
485
|
+
expect(updateLocalTracksSpy).toBeCalledOnceWith({
|
|
486
|
+
audio: mockTrack
|
|
487
|
+
});
|
|
488
|
+
expect(bnrMetricSpy).toBeCalledOnceWith(_types4.METRIC_EVENT.BNR_DISABLED, _types4.METRIC_TYPE.BEHAVIORAL, call.getCallId(), call.getCorrelationId());
|
|
489
|
+
call.end();
|
|
490
|
+
_context9.next = 1;
|
|
491
|
+
return (0, _Utils.waitForMsecs)(50);
|
|
492
|
+
case 1:
|
|
493
|
+
/* Checks for switching off the listeners on call disconnect */
|
|
494
|
+
expect(offStreamSpy).toBeCalledTimes(2);
|
|
495
|
+
expect(offStreamSpy).toBeCalledWith(InternalMediaCoreModule.LocalStreamEventNames.OutputTrackChange, expect.any(Function));
|
|
496
|
+
expect(offStreamSpy).toBeCalledWith(InternalMediaCoreModule.LocalStreamEventNames.EffectAdded, expect.any(Function));
|
|
497
|
+
expect(offEffectSpy).toBeCalledWith(_mediaHelpers.EffectEvent.Enabled, expect.any(Function));
|
|
498
|
+
expect(offEffectSpy).toBeCalledWith(_mediaHelpers.EffectEvent.Disabled, expect.any(Function));
|
|
499
|
+
case 2:
|
|
500
|
+
case "end":
|
|
501
|
+
return _context9.stop();
|
|
502
|
+
}
|
|
503
|
+
}, _callee9);
|
|
504
|
+
})));
|
|
505
|
+
it('answer fails if localAudioTrack is empty', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee0() {
|
|
506
|
+
var mockStream, localAudioStream, warnSpy, call;
|
|
507
|
+
return _regenerator.default.wrap(function (_context0) {
|
|
508
|
+
while (1) switch (_context0.prev = _context0.next) {
|
|
509
|
+
case 0:
|
|
510
|
+
mockStream = {
|
|
511
|
+
outputStream: {
|
|
512
|
+
getAudioTracks: jest.fn().mockReturnValue([])
|
|
513
|
+
},
|
|
514
|
+
on: jest.fn(),
|
|
515
|
+
off: jest.fn(),
|
|
516
|
+
getEffectByKind: jest.fn()
|
|
517
|
+
};
|
|
518
|
+
localAudioStream = mockStream;
|
|
519
|
+
webex.request.mockReturnValue({
|
|
520
|
+
statusCode: 200,
|
|
521
|
+
body: {
|
|
522
|
+
device: {
|
|
523
|
+
deviceId: '8a67806f-fc4d-446b-a131-31e71ea5b010',
|
|
524
|
+
correlationId: '8a67806f-fc4d-446b-a131-31e71ea5b011'
|
|
525
|
+
},
|
|
526
|
+
callId: '8a67806f-fc4d-446b-a131-31e71ea5b020'
|
|
527
|
+
}
|
|
528
|
+
});
|
|
529
|
+
warnSpy = jest.spyOn(_Logger.default, 'warn');
|
|
530
|
+
call = (0, _call.createCall)(activeUrl, webex, _types3.CallDirection.OUTBOUND, deviceId, mockLineId, deleteCallFromCollection, defaultServiceIndicator, dest);
|
|
531
|
+
call.answer(localAudioStream);
|
|
532
|
+
_context0.next = 1;
|
|
533
|
+
return (0, _Utils.waitForMsecs)(50);
|
|
534
|
+
case 1:
|
|
535
|
+
expect(warnSpy).toBeCalledTimes(2);
|
|
536
|
+
expect(warnSpy).toBeCalledWith("Did not find a local track while answering the call ".concat(call.getCorrelationId()), {
|
|
537
|
+
file: 'call',
|
|
538
|
+
method: 'answer'
|
|
539
|
+
});
|
|
540
|
+
expect(call['callStateMachine'].state.value).toBe('S_CALL_CLEARED');
|
|
541
|
+
expect(call['mediaStateMachine'].state.value).toBe('S_ROAP_IDLE');
|
|
542
|
+
expect(call.getDisconnectReason().code).toBe(_types5.DisconnectCode.MEDIA_INACTIVITY);
|
|
543
|
+
expect(call.getDisconnectReason().cause).toBe(_types5.DisconnectCause.MEDIA_INACTIVITY);
|
|
544
|
+
expect(webex.request.mock.calls[0][0].body.metrics).toStrictEqual(disconnectStats);
|
|
545
|
+
case 2:
|
|
546
|
+
case "end":
|
|
547
|
+
return _context0.stop();
|
|
548
|
+
}
|
|
549
|
+
}, _callee0);
|
|
550
|
+
})));
|
|
551
|
+
it('dial fails if localAudioTrack is empty', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee1() {
|
|
552
|
+
var mockStream, localAudioStream, warnSpy, call;
|
|
553
|
+
return _regenerator.default.wrap(function (_context1) {
|
|
554
|
+
while (1) switch (_context1.prev = _context1.next) {
|
|
555
|
+
case 0:
|
|
556
|
+
mockStream = {
|
|
557
|
+
outputStream: {
|
|
558
|
+
getAudioTracks: jest.fn().mockReturnValue([])
|
|
559
|
+
},
|
|
560
|
+
on: jest.fn()
|
|
561
|
+
};
|
|
562
|
+
localAudioStream = mockStream;
|
|
563
|
+
warnSpy = jest.spyOn(_Logger.default, 'warn');
|
|
564
|
+
call = (0, _call.createCall)(activeUrl, webex, _types3.CallDirection.OUTBOUND, deviceId, mockLineId, deleteCallFromCollection, defaultServiceIndicator, dest);
|
|
565
|
+
call.dial(localAudioStream);
|
|
566
|
+
_context1.next = 1;
|
|
567
|
+
return (0, _Utils.waitForMsecs)(50);
|
|
568
|
+
case 1:
|
|
569
|
+
expect(warnSpy).toBeCalledTimes(1);
|
|
570
|
+
expect(warnSpy).toBeCalledWith("Did not find a local track while dialing the call ".concat(call.getCorrelationId()), {
|
|
571
|
+
file: 'call',
|
|
572
|
+
method: 'dial'
|
|
573
|
+
});
|
|
574
|
+
expect(call['callStateMachine'].state.value).toBe('S_IDLE');
|
|
575
|
+
expect(call['mediaStateMachine'].state.value).toBe('S_ROAP_IDLE');
|
|
576
|
+
expect(webex.request).not.toBeCalledOnceWith();
|
|
577
|
+
case 2:
|
|
578
|
+
case "end":
|
|
579
|
+
return _context1.stop();
|
|
580
|
+
}
|
|
581
|
+
}, _callee1);
|
|
582
|
+
})));
|
|
583
|
+
it('update media after call creation with valid stream', function () {
|
|
584
|
+
var callManager = (0, _callManager.getCallManager)(webex, defaultServiceIndicator);
|
|
585
|
+
var mockStream = {
|
|
586
|
+
outputStream: {
|
|
587
|
+
getAudioTracks: jest.fn().mockReturnValue([mockTrack])
|
|
588
|
+
},
|
|
589
|
+
on: jest.fn(),
|
|
590
|
+
off: jest.fn(),
|
|
591
|
+
getEffectByKind: jest.fn()
|
|
592
|
+
};
|
|
593
|
+
var localAudioStream = mockStream;
|
|
594
|
+
var onStream1Spy = jest.spyOn(localAudioStream, 'on');
|
|
595
|
+
var offStream1Spy = jest.spyOn(localAudioStream, 'off');
|
|
596
|
+
var call = callManager.createCall(_types3.CallDirection.OUTBOUND, deviceId, mockLineId, dest);
|
|
597
|
+
call.dial(localAudioStream);
|
|
598
|
+
expect(mockTrack.enabled).toEqual(true);
|
|
599
|
+
expect(onStream1Spy).toBeCalledTimes(2);
|
|
600
|
+
expect(onStream1Spy).toBeCalledWith(InternalMediaCoreModule.LocalStreamEventNames.OutputTrackChange, expect.any(Function));
|
|
601
|
+
expect(onStream1Spy).toBeCalledWith(InternalMediaCoreModule.LocalStreamEventNames.EffectAdded, expect.any(Function));
|
|
602
|
+
var mockTrack2 = {
|
|
603
|
+
enabled: true
|
|
604
|
+
};
|
|
605
|
+
var mockStream2 = {
|
|
606
|
+
outputStream: {
|
|
607
|
+
getAudioTracks: jest.fn().mockReturnValue([mockTrack2])
|
|
608
|
+
},
|
|
609
|
+
on: jest.fn(),
|
|
610
|
+
getEffectByKind: jest.fn()
|
|
611
|
+
};
|
|
612
|
+
var localAudioStream2 = mockStream2;
|
|
613
|
+
var onStream2Spy = jest.spyOn(localAudioStream2, 'on');
|
|
614
|
+
call.updateMedia(localAudioStream2);
|
|
615
|
+
expect(call['mediaConnection'].updateLocalTracks).toBeCalledOnceWith({
|
|
616
|
+
audio: mockTrack2
|
|
617
|
+
});
|
|
618
|
+
expect(call['localAudioStream']).toEqual(localAudioStream2);
|
|
619
|
+
expect(offStream1Spy).toBeCalledWith(InternalMediaCoreModule.LocalStreamEventNames.EffectAdded, expect.any(Function));
|
|
620
|
+
expect(onStream2Spy).toBeCalledWith(InternalMediaCoreModule.LocalStreamEventNames.OutputTrackChange, expect.any(Function));
|
|
621
|
+
expect(onStream2Spy).toBeCalledWith(InternalMediaCoreModule.LocalStreamEventNames.EffectAdded, expect.any(Function));
|
|
622
|
+
});
|
|
623
|
+
it('update media with invalid stream', function () {
|
|
624
|
+
var callManager = (0, _callManager.getCallManager)(webex, defaultServiceIndicator);
|
|
625
|
+
var warnSpy = jest.spyOn(_Logger.default, 'warn');
|
|
626
|
+
var infoSpy = jest.spyOn(_Logger.default, 'info');
|
|
627
|
+
var mockStream = {
|
|
628
|
+
outputStream: {
|
|
629
|
+
getAudioTracks: jest.fn().mockReturnValue([mockTrack])
|
|
630
|
+
},
|
|
631
|
+
on: jest.fn(),
|
|
632
|
+
getEffectByKind: jest.fn()
|
|
633
|
+
};
|
|
634
|
+
var localAudioStream = mockStream;
|
|
635
|
+
var call = callManager.createCall(_types3.CallDirection.OUTBOUND, deviceId, mockLineId, dest);
|
|
636
|
+
call.dial(localAudioStream);
|
|
637
|
+
expect(mockTrack.enabled).toEqual(true);
|
|
638
|
+
var errorStream = {
|
|
639
|
+
outputStream: {
|
|
640
|
+
getAudioTracks: jest.fn().mockReturnValue([])
|
|
641
|
+
}
|
|
642
|
+
};
|
|
643
|
+
var localAudioStream2 = errorStream;
|
|
644
|
+
call.updateMedia(localAudioStream2);
|
|
645
|
+
expect(call['mediaConnection'].updateLocalTracks).not.toBeCalled();
|
|
646
|
+
expect(warnSpy).toHaveBeenCalledWith("Did not find a local track while updating media for call ".concat(call.getCorrelationId(), ". Will not update media"), {
|
|
647
|
+
file: 'call',
|
|
648
|
+
method: 'updateMedia'
|
|
649
|
+
});
|
|
650
|
+
expect(infoSpy).toHaveBeenCalledWith('invoking with stream', {
|
|
651
|
+
file: 'call',
|
|
652
|
+
method: 'dial'
|
|
653
|
+
});
|
|
654
|
+
});
|
|
655
|
+
it('test system mute and user mute different scnearios', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee10() {
|
|
656
|
+
var logSpy, callManager, mockStream, localAudioStream, call;
|
|
657
|
+
return _regenerator.default.wrap(function (_context10) {
|
|
658
|
+
while (1) switch (_context10.prev = _context10.next) {
|
|
659
|
+
case 0:
|
|
660
|
+
logSpy = jest.spyOn(_Logger.default, 'info');
|
|
661
|
+
webex.request.mockReturnValue({
|
|
662
|
+
statusCode: 200,
|
|
663
|
+
body: {
|
|
664
|
+
device: {
|
|
665
|
+
deviceId: '8a67806f-fc4d-446b-a131-31e71ea5b010',
|
|
666
|
+
correlationId: '8a67806f-fc4d-446b-a131-31e71ea5b011'
|
|
667
|
+
},
|
|
668
|
+
callId: '8a67806f-fc4d-446b-a131-31e71ea5b020'
|
|
669
|
+
}
|
|
670
|
+
});
|
|
671
|
+
callManager = (0, _callManager.getCallManager)(webex, defaultServiceIndicator);
|
|
672
|
+
mockStream = {
|
|
673
|
+
on: jest.fn(),
|
|
674
|
+
setUserMuted: jest.fn(),
|
|
675
|
+
systemMuted: false,
|
|
676
|
+
userMuted: false
|
|
677
|
+
};
|
|
678
|
+
localAudioStream = mockStream;
|
|
679
|
+
call = callManager.createCall(dest, _types3.CallDirection.OUTBOUND, deviceId, mockLineId);
|
|
680
|
+
expect(call).toBeTruthy();
|
|
681
|
+
/* System mute is being triggered, mute state within call object should update to true */
|
|
682
|
+
mockStream.systemMuted = true;
|
|
683
|
+
call.mute(localAudioStream, _types5.MUTE_TYPE.SYSTEM);
|
|
684
|
+
expect(call.isMuted()).toEqual(true);
|
|
685
|
+
|
|
686
|
+
/* User mute is triggered, but no change will happen to the call object mute state since it is system muted */
|
|
687
|
+
logSpy.mockClear();
|
|
688
|
+
call.mute(localAudioStream, _types5.MUTE_TYPE.USER);
|
|
689
|
+
expect(call.isMuted()).toEqual(true);
|
|
690
|
+
expect(mockStream.setUserMuted).not.toBeCalledOnceWith(true);
|
|
691
|
+
expect(logSpy).toHaveBeenCalledWith("Call is muted on the system - ".concat(call.getCorrelationId(), "."), {
|
|
692
|
+
file: 'call',
|
|
693
|
+
method: 'mute'
|
|
694
|
+
});
|
|
695
|
+
|
|
696
|
+
/* System mute is being triggered, mute state within call object should update to false */
|
|
697
|
+
mockStream.systemMuted = false;
|
|
698
|
+
call.mute(localAudioStream, _types5.MUTE_TYPE.SYSTEM);
|
|
699
|
+
expect(call.isMuted()).toEqual(false);
|
|
700
|
+
|
|
701
|
+
/* User mute can be triggered now updating call object mute state as well */
|
|
702
|
+
call.mute(localAudioStream, _types5.MUTE_TYPE.USER);
|
|
703
|
+
expect(call.isMuted()).toEqual(true);
|
|
704
|
+
expect(mockStream.setUserMuted).toBeCalledOnceWith(true);
|
|
705
|
+
mockStream.userMuted = true;
|
|
706
|
+
|
|
707
|
+
/* System mute being triggered now won't update the mute state within call object but will block the user unmute */
|
|
708
|
+
logSpy.mockClear();
|
|
709
|
+
mockStream.systemMuted = true;
|
|
710
|
+
call.mute(localAudioStream, _types5.MUTE_TYPE.SYSTEM);
|
|
711
|
+
expect(call.isMuted()).toEqual(true);
|
|
712
|
+
expect(logSpy).toHaveBeenCalledWith("Call is muted by the user already - ".concat(call.getCorrelationId(), "."), {
|
|
713
|
+
file: 'call',
|
|
714
|
+
method: 'mute'
|
|
715
|
+
});
|
|
716
|
+
|
|
717
|
+
/* User mute now won't be able to update mute state back to false as system mute is also set */
|
|
718
|
+
call.mute(localAudioStream, _types5.MUTE_TYPE.USER);
|
|
719
|
+
expect(call.isMuted()).toEqual(true);
|
|
720
|
+
expect(mockStream.setUserMuted).not.toBeCalledOnceWith(false);
|
|
721
|
+
|
|
722
|
+
/* Revert the system mute but call mute state remains same */
|
|
723
|
+
mockStream.systemMuted = false;
|
|
724
|
+
call.mute(localAudioStream, _types5.MUTE_TYPE.SYSTEM);
|
|
725
|
+
expect(call.isMuted()).toEqual(true);
|
|
726
|
+
|
|
727
|
+
/* User mute will be able update the mute state now */
|
|
728
|
+
mockStream.setUserMuted.mockClear();
|
|
729
|
+
call.mute(localAudioStream, _types5.MUTE_TYPE.USER);
|
|
730
|
+
expect(call.isMuted()).toEqual(false);
|
|
731
|
+
expect(mockStream.setUserMuted).toBeCalledOnceWith(false);
|
|
732
|
+
case 1:
|
|
733
|
+
case "end":
|
|
734
|
+
return _context10.stop();
|
|
735
|
+
}
|
|
736
|
+
}, _callee10);
|
|
737
|
+
})));
|
|
738
|
+
describe('Guest Calling Flow Tests', function () {
|
|
739
|
+
var dummyEvent = {
|
|
740
|
+
type: 'E_SEND_CALL_SETUP',
|
|
741
|
+
data: undefined
|
|
742
|
+
};
|
|
743
|
+
var call;
|
|
744
|
+
it('outgoing call without guest calling must have callee', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee11() {
|
|
745
|
+
var requestSpy, requestArgs;
|
|
746
|
+
return _regenerator.default.wrap(function (_context11) {
|
|
747
|
+
while (1) switch (_context11.prev = _context11.next) {
|
|
748
|
+
case 0:
|
|
749
|
+
call = new _call.Call(activeUrl, webex, _types3.CallDirection.OUTBOUND, deviceId, mockLineId, function () {
|
|
750
|
+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
751
|
+
var dummy = 10;
|
|
752
|
+
}, defaultServiceIndicator, dest);
|
|
753
|
+
call['callStateMachine'].state.value = 'S_IDLE';
|
|
754
|
+
requestSpy = jest.spyOn(webex, 'request');
|
|
755
|
+
call.sendCallStateMachineEvt(dummyEvent);
|
|
756
|
+
requestArgs = requestSpy.mock.calls[0][0];
|
|
757
|
+
expect('callee' in requestArgs.body).toBe(true);
|
|
758
|
+
case 1:
|
|
759
|
+
case "end":
|
|
760
|
+
return _context11.stop();
|
|
761
|
+
}
|
|
762
|
+
}, _callee11);
|
|
763
|
+
})));
|
|
764
|
+
it('outgoing call for guest calling must not have callee', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee12() {
|
|
765
|
+
var requestSpy, requestArgs;
|
|
766
|
+
return _regenerator.default.wrap(function (_context12) {
|
|
767
|
+
while (1) switch (_context12.prev = _context12.next) {
|
|
768
|
+
case 0:
|
|
769
|
+
call = new _call.Call(activeUrl, webex, _types3.CallDirection.OUTBOUND, deviceId, mockLineId, function () {
|
|
770
|
+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
771
|
+
var dummy = 10;
|
|
772
|
+
}, defaultServiceIndicator);
|
|
773
|
+
call['callStateMachine'].state.value = 'S_IDLE';
|
|
774
|
+
requestSpy = jest.spyOn(webex, 'request');
|
|
775
|
+
call.sendCallStateMachineEvt(dummyEvent);
|
|
776
|
+
requestArgs = requestSpy.mock.calls[0][0];
|
|
777
|
+
expect('callee' in requestArgs.body).toBe(false);
|
|
778
|
+
case 1:
|
|
779
|
+
case "end":
|
|
780
|
+
return _context12.stop();
|
|
781
|
+
}
|
|
782
|
+
}, _callee12);
|
|
783
|
+
})));
|
|
784
|
+
});
|
|
785
|
+
});
|
|
786
|
+
describe('State Machine handler tests', function () {
|
|
787
|
+
var deviceId = '55dfb53f-bed2-36da-8e85-cee7f02aa68e';
|
|
788
|
+
var dest = {
|
|
789
|
+
type: _types3.CallType.URI,
|
|
790
|
+
address: 'tel:5003'
|
|
791
|
+
};
|
|
792
|
+
var mockStatusBody = {
|
|
793
|
+
device: {
|
|
794
|
+
deviceId: '123e4567-e89b-12d3-a456-426614174000',
|
|
795
|
+
correlationId: 'b5476d4c-f48b-475e-b4e2-994e24d14ca2'
|
|
796
|
+
},
|
|
797
|
+
callId: 'fcf86aa5-5539-4c9f-8b72-667786ae9b6c'
|
|
798
|
+
};
|
|
799
|
+
var call;
|
|
800
|
+
var dtmfMock;
|
|
801
|
+
beforeEach(function () {
|
|
802
|
+
call = new _call.Call(activeUrl, webex, _types3.CallDirection.OUTBOUND, deviceId, mockLineId, function () {
|
|
803
|
+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
804
|
+
var dummy = 10;
|
|
805
|
+
}, defaultServiceIndicator, dest);
|
|
806
|
+
jest.clearAllTimers();
|
|
807
|
+
jest.useFakeTimers();
|
|
808
|
+
call['callStateMachine'].state.value = 'S_IDLE';
|
|
809
|
+
call.mediaConnection = _testUtil.mediaConnection;
|
|
810
|
+
dtmfMock = jest.spyOn(call['mediaConnection'], 'insertDTMF');
|
|
811
|
+
});
|
|
812
|
+
|
|
813
|
+
// afterEach(() => call.removeAllListeners());
|
|
814
|
+
|
|
815
|
+
it('successful session refresh', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee13() {
|
|
816
|
+
var statusPayload, dummyEvent, funcSpy, logSpy;
|
|
817
|
+
return _regenerator.default.wrap(function (_context13) {
|
|
818
|
+
while (1) switch (_context13.prev = _context13.next) {
|
|
819
|
+
case 0:
|
|
820
|
+
statusPayload = {
|
|
821
|
+
statusCode: 200,
|
|
822
|
+
body: mockStatusBody
|
|
823
|
+
};
|
|
824
|
+
dummyEvent = {
|
|
825
|
+
type: 'E_CALL_ESTABLISHED'
|
|
826
|
+
};
|
|
827
|
+
call['callStateMachine'].state.value = 'S_SEND_CALL_CONNECT';
|
|
828
|
+
webex.request.mockReturnValue(statusPayload);
|
|
829
|
+
jest.spyOn(global, 'setInterval');
|
|
830
|
+
funcSpy = jest.spyOn(call, 'postStatus').mockResolvedValue(statusPayload);
|
|
831
|
+
logSpy = jest.spyOn(_Logger.default, 'info');
|
|
832
|
+
call.sendCallStateMachineEvt(dummyEvent);
|
|
833
|
+
jest.advanceTimersByTime(_constants.DEFAULT_SESSION_TIMER);
|
|
834
|
+
|
|
835
|
+
/* This is to flush all the promises from the Promise queue so that
|
|
836
|
+
* Jest.fakeTimers can advance time and also clear the promise Queue
|
|
837
|
+
*/
|
|
838
|
+
_context13.next = 1;
|
|
839
|
+
return (0, _testUtil.flushPromises)(3);
|
|
840
|
+
case 1:
|
|
841
|
+
expect(setInterval).toHaveBeenCalledTimes(1);
|
|
842
|
+
expect(setInterval).toHaveBeenCalledWith(expect.any(Function), _constants.DEFAULT_SESSION_TIMER);
|
|
843
|
+
expect(funcSpy).toBeCalledTimes(1);
|
|
844
|
+
expect(logSpy).toBeCalledWith('Session refresh successful', {
|
|
845
|
+
file: 'call',
|
|
846
|
+
method: 'scheduleCallKeepaliveInterval'
|
|
847
|
+
});
|
|
848
|
+
expect(logSpy).toHaveBeenCalledWith("".concat(_constants2.METHOD_START_MESSAGE, " with: ").concat(call.getCorrelationId()), {
|
|
849
|
+
file: 'call',
|
|
850
|
+
method: 'handleCallEstablished'
|
|
851
|
+
});
|
|
852
|
+
case 2:
|
|
853
|
+
case "end":
|
|
854
|
+
return _context13.stop();
|
|
855
|
+
}
|
|
856
|
+
}, _callee13);
|
|
857
|
+
})));
|
|
858
|
+
it('session refresh 401 emits token error and ends the call', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee14() {
|
|
859
|
+
var statusPayload, emitSpy, funcSpy;
|
|
860
|
+
return _regenerator.default.wrap(function (_context14) {
|
|
861
|
+
while (1) switch (_context14.prev = _context14.next) {
|
|
862
|
+
case 0:
|
|
863
|
+
expect.assertions(4);
|
|
864
|
+
statusPayload = {
|
|
865
|
+
statusCode: 401
|
|
866
|
+
};
|
|
867
|
+
webex.request.mockReturnValue(statusPayload);
|
|
868
|
+
jest.spyOn(global, 'clearInterval');
|
|
869
|
+
emitSpy = jest.spyOn(call, 'emit');
|
|
870
|
+
call.on(_types2.CALL_EVENT_KEYS.CALL_ERROR, function (errObj) {
|
|
871
|
+
expect(errObj.type).toStrictEqual(_types.ERROR_TYPE.TOKEN_ERROR);
|
|
872
|
+
});
|
|
873
|
+
funcSpy = jest.spyOn(call, 'postStatus').mockRejectedValue(statusPayload);
|
|
874
|
+
call['handleCallEstablished']({});
|
|
875
|
+
jest.advanceTimersByTime(_constants.DEFAULT_SESSION_TIMER);
|
|
876
|
+
_context14.next = 1;
|
|
877
|
+
return _promise.default.resolve();
|
|
878
|
+
case 1:
|
|
879
|
+
_context14.next = 2;
|
|
880
|
+
return _promise.default.resolve();
|
|
881
|
+
case 2:
|
|
882
|
+
// clearInterval is called twice: once in scheduleCallKeepaliveInterval (at start)
|
|
883
|
+
// and once in handleCallKeepaliveError when clearing timer due to error
|
|
884
|
+
expect(clearInterval).toHaveBeenCalledTimes(2);
|
|
885
|
+
expect(funcSpy).toBeCalledTimes(1);
|
|
886
|
+
expect(emitSpy).toHaveBeenCalledWith(_types2.CALL_EVENT_KEYS.DISCONNECT, call.getCorrelationId());
|
|
887
|
+
case 3:
|
|
888
|
+
case "end":
|
|
889
|
+
return _context14.stop();
|
|
890
|
+
}
|
|
891
|
+
}, _callee14);
|
|
892
|
+
})));
|
|
893
|
+
it('session refresh failure', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee15() {
|
|
894
|
+
var statusPayload, funcSpy;
|
|
895
|
+
return _regenerator.default.wrap(function (_context15) {
|
|
896
|
+
while (1) switch (_context15.prev = _context15.next) {
|
|
897
|
+
case 0:
|
|
898
|
+
expect.assertions(4);
|
|
899
|
+
statusPayload = {
|
|
900
|
+
statusCode: 403
|
|
901
|
+
};
|
|
902
|
+
webex.request.mockReturnValue(statusPayload);
|
|
903
|
+
jest.spyOn(global, 'clearInterval');
|
|
904
|
+
call.on(_types2.CALL_EVENT_KEYS.CALL_ERROR, function (errObj) {
|
|
905
|
+
expect(errObj.type).toStrictEqual(_types.ERROR_TYPE.FORBIDDEN_ERROR);
|
|
906
|
+
expect(errObj.message).toStrictEqual('An unauthorized action has been received. This action has been blocked. Please contact the administrator if this persists.');
|
|
907
|
+
});
|
|
908
|
+
funcSpy = jest.spyOn(call, 'postStatus').mockRejectedValue(statusPayload);
|
|
909
|
+
call['handleCallEstablished']({});
|
|
910
|
+
jest.advanceTimersByTime(_constants.DEFAULT_SESSION_TIMER);
|
|
911
|
+
|
|
912
|
+
/* This is to flush all the promises from the Promise queue so that
|
|
913
|
+
* Jest.fakeTimers can advance time and also clear the promise Queue
|
|
914
|
+
*/
|
|
915
|
+
_context15.next = 1;
|
|
916
|
+
return (0, _testUtil.flushPromises)(2);
|
|
917
|
+
case 1:
|
|
918
|
+
// clearInterval is called twice: once in scheduleCallKeepaliveInterval (at start)
|
|
919
|
+
// and once in handleCallKeepaliveError when clearing timer due to error
|
|
920
|
+
expect(clearInterval).toHaveBeenCalledTimes(2);
|
|
921
|
+
expect(funcSpy).toBeCalledTimes(1);
|
|
922
|
+
case 2:
|
|
923
|
+
case "end":
|
|
924
|
+
return _context15.stop();
|
|
925
|
+
}
|
|
926
|
+
}, _callee15);
|
|
927
|
+
})));
|
|
928
|
+
it('session refresh 500 schedules retry via retry-after or default interval', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee16() {
|
|
929
|
+
var errorPayload, okPayload, scheduleKeepaliveSpy, postStatusSpy;
|
|
930
|
+
return _regenerator.default.wrap(function (_context16) {
|
|
931
|
+
while (1) switch (_context16.prev = _context16.next) {
|
|
932
|
+
case 0:
|
|
933
|
+
errorPayload = {
|
|
934
|
+
statusCode: 500,
|
|
935
|
+
headers: {
|
|
936
|
+
'retry-after': 1
|
|
937
|
+
}
|
|
938
|
+
};
|
|
939
|
+
okPayload = {
|
|
940
|
+
statusCode: 200,
|
|
941
|
+
body: {}
|
|
942
|
+
};
|
|
943
|
+
scheduleKeepaliveSpy = jest.spyOn(call, 'scheduleCallKeepaliveInterval');
|
|
944
|
+
postStatusSpy = jest.spyOn(call, 'postStatus').mockRejectedValueOnce(errorPayload).mockResolvedValueOnce(okPayload);
|
|
945
|
+
if (call['sessionTimer'] === undefined) {
|
|
946
|
+
call['handleCallEstablished']({});
|
|
947
|
+
}
|
|
948
|
+
jest.advanceTimersByTime(_constants.DEFAULT_SESSION_TIMER);
|
|
949
|
+
_context16.next = 1;
|
|
950
|
+
return (0, _testUtil.flushPromises)(2);
|
|
951
|
+
case 1:
|
|
952
|
+
expect(postStatusSpy).toHaveBeenCalledTimes(1);
|
|
953
|
+
|
|
954
|
+
// Now advance by 1 second for the retry-after interval
|
|
955
|
+
jest.advanceTimersByTime(1000);
|
|
956
|
+
_context16.next = 2;
|
|
957
|
+
return (0, _testUtil.flushPromises)(2);
|
|
958
|
+
case 2:
|
|
959
|
+
expect(postStatusSpy).toHaveBeenCalledTimes(2);
|
|
960
|
+
expect(scheduleKeepaliveSpy).toHaveBeenCalledTimes(2);
|
|
961
|
+
case 3:
|
|
962
|
+
case "end":
|
|
963
|
+
return _context16.stop();
|
|
964
|
+
}
|
|
965
|
+
}, _callee16);
|
|
966
|
+
})));
|
|
967
|
+
it('scheduleCallKeepaliveInterval clears existing interval before creating new one', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee17() {
|
|
968
|
+
var firstTimer, secondTimer;
|
|
969
|
+
return _regenerator.default.wrap(function (_context17) {
|
|
970
|
+
while (1) switch (_context17.prev = _context17.next) {
|
|
971
|
+
case 0:
|
|
972
|
+
jest.spyOn(global, 'setInterval');
|
|
973
|
+
jest.spyOn(global, 'clearInterval');
|
|
974
|
+
|
|
975
|
+
// Setup the first keepalive interval
|
|
976
|
+
call['handleCallEstablished']({});
|
|
977
|
+
|
|
978
|
+
// At this point, scheduleCallKeepaliveInterval was called once
|
|
979
|
+
// It should have called clearInterval once (at the start) and setInterval once
|
|
980
|
+
expect(clearInterval).toHaveBeenCalledTimes(1);
|
|
981
|
+
expect(setInterval).toHaveBeenCalledTimes(1);
|
|
982
|
+
firstTimer = call['sessionTimer'];
|
|
983
|
+
expect(firstTimer).toBeDefined();
|
|
984
|
+
|
|
985
|
+
// Manually call scheduleCallKeepaliveInterval again to simulate a retry scenario
|
|
986
|
+
call['scheduleCallKeepaliveInterval']();
|
|
987
|
+
|
|
988
|
+
// clearInterval should have been called again to clear the previous timer
|
|
989
|
+
expect(clearInterval).toHaveBeenCalledTimes(2);
|
|
990
|
+
// A new interval should have been created
|
|
991
|
+
expect(setInterval).toHaveBeenCalledTimes(2);
|
|
992
|
+
|
|
993
|
+
// The sessionTimer should be different (new timer)
|
|
994
|
+
secondTimer = call['sessionTimer'];
|
|
995
|
+
expect(secondTimer).toBeDefined();
|
|
996
|
+
case 1:
|
|
997
|
+
case "end":
|
|
998
|
+
return _context17.stop();
|
|
999
|
+
}
|
|
1000
|
+
}, _callee17);
|
|
1001
|
+
})));
|
|
1002
|
+
it('keepalive ends after reaching max retry count', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee18() {
|
|
1003
|
+
var errorPayload, warnSpy, postStatusSpy;
|
|
1004
|
+
return _regenerator.default.wrap(function (_context18) {
|
|
1005
|
+
while (1) switch (_context18.prev = _context18.next) {
|
|
1006
|
+
case 0:
|
|
1007
|
+
errorPayload = {
|
|
1008
|
+
statusCode: 500,
|
|
1009
|
+
headers: {
|
|
1010
|
+
'retry-after': 1
|
|
1011
|
+
}
|
|
1012
|
+
};
|
|
1013
|
+
jest.spyOn(global, 'clearInterval');
|
|
1014
|
+
warnSpy = jest.spyOn(_Logger.default, 'warn');
|
|
1015
|
+
postStatusSpy = jest.spyOn(call, 'postStatus').mockRejectedValue(errorPayload); // Put the call in the S_CALL_ESTABLISHED state and set it as connected
|
|
1016
|
+
call['callStateMachine'].state.value = 'S_CALL_ESTABLISHED';
|
|
1017
|
+
call['connected'] = true;
|
|
1018
|
+
|
|
1019
|
+
// Call handleCallEstablished which will setup interval
|
|
1020
|
+
call['handleCallEstablished']({});
|
|
1021
|
+
|
|
1022
|
+
// Advance timer to trigger the first failure (uses DEFAULT_SESSION_TIMER)
|
|
1023
|
+
jest.advanceTimersByTime(_constants.DEFAULT_SESSION_TIMER);
|
|
1024
|
+
_context18.next = 1;
|
|
1025
|
+
return (0, _testUtil.flushPromises)(2);
|
|
1026
|
+
case 1:
|
|
1027
|
+
// Now advance by 1 second for each of the 4 retry attempts (retry-after: 1 second each)
|
|
1028
|
+
// Need to do this separately to allow state machine to process and create new intervals
|
|
1029
|
+
jest.advanceTimersByTime(1000);
|
|
1030
|
+
_context18.next = 2;
|
|
1031
|
+
return (0, _testUtil.flushPromises)(2);
|
|
1032
|
+
case 2:
|
|
1033
|
+
jest.advanceTimersByTime(1000);
|
|
1034
|
+
_context18.next = 3;
|
|
1035
|
+
return (0, _testUtil.flushPromises)(2);
|
|
1036
|
+
case 3:
|
|
1037
|
+
jest.advanceTimersByTime(1000);
|
|
1038
|
+
_context18.next = 4;
|
|
1039
|
+
return (0, _testUtil.flushPromises)(2);
|
|
1040
|
+
case 4:
|
|
1041
|
+
jest.advanceTimersByTime(1000);
|
|
1042
|
+
_context18.next = 5;
|
|
1043
|
+
return (0, _testUtil.flushPromises)(2);
|
|
1044
|
+
case 5:
|
|
1045
|
+
// The error handler should detect we're at max retry count and stop
|
|
1046
|
+
expect(warnSpy).toHaveBeenCalledWith("Max keepalive retry attempts reached. Aborting call keepalive for callId: ".concat(call.getCallId()), {
|
|
1047
|
+
file: 'call',
|
|
1048
|
+
method: 'keepaliveRetryCallback'
|
|
1049
|
+
});
|
|
1050
|
+
expect(postStatusSpy).toHaveBeenCalledTimes(5);
|
|
1051
|
+
expect(call['callKeepaliveRetryCount']).toBe(4);
|
|
1052
|
+
case 6:
|
|
1053
|
+
case "end":
|
|
1054
|
+
return _context18.stop();
|
|
1055
|
+
}
|
|
1056
|
+
}, _callee18);
|
|
1057
|
+
})));
|
|
1058
|
+
it('state changes during successful incoming call', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee19() {
|
|
1059
|
+
var statusPayload, dummyEvent, postMediaSpy, deleteSpy, dummyOkEvent;
|
|
1060
|
+
return _regenerator.default.wrap(function (_context19) {
|
|
1061
|
+
while (1) switch (_context19.prev = _context19.next) {
|
|
1062
|
+
case 0:
|
|
1063
|
+
statusPayload = {
|
|
1064
|
+
statusCode: 200,
|
|
1065
|
+
body: mockStatusBody
|
|
1066
|
+
};
|
|
1067
|
+
dummyEvent = {
|
|
1068
|
+
type: 'E_RECV_CALL_SETUP',
|
|
1069
|
+
data: {
|
|
1070
|
+
seq: 1,
|
|
1071
|
+
messageType: 'OFFER'
|
|
1072
|
+
}
|
|
1073
|
+
};
|
|
1074
|
+
postMediaSpy = jest.spyOn(call, 'postMedia');
|
|
1075
|
+
deleteSpy = jest.spyOn(call, 'delete');
|
|
1076
|
+
webex.request.mockReturnValue(statusPayload);
|
|
1077
|
+
call['direction'] = _types3.CallDirection.INBOUND;
|
|
1078
|
+
call.sendCallStateMachineEvt(dummyEvent);
|
|
1079
|
+
expect(call['callStateMachine'].state.value).toBe('S_SEND_CALL_PROGRESS');
|
|
1080
|
+
dummyEvent.type = 'E_SEND_CALL_CONNECT';
|
|
1081
|
+
call.sendCallStateMachineEvt(dummyEvent);
|
|
1082
|
+
expect(call['callStateMachine'].state.value).toBe('S_SEND_CALL_CONNECT');
|
|
1083
|
+
|
|
1084
|
+
/* we should expect to forward the roap offer message to mediaSdk for further processing */
|
|
1085
|
+
dummyEvent.type = 'E_RECV_ROAP_OFFER';
|
|
1086
|
+
call.sendMediaStateMachineEvt(dummyEvent);
|
|
1087
|
+
expect(_testUtil.mediaConnection.roapMessageReceived).toHaveBeenLastCalledWith(dummyEvent.data);
|
|
1088
|
+
|
|
1089
|
+
/* expect sending roap answer to mobius */
|
|
1090
|
+
dummyEvent.type = 'E_SEND_ROAP_ANSWER';
|
|
1091
|
+
call.sendMediaStateMachineEvt(dummyEvent);
|
|
1092
|
+
expect(postMediaSpy).toBeCalledOnceWith(dummyEvent.data);
|
|
1093
|
+
dummyOkEvent = {
|
|
1094
|
+
type: 'E_ROAP_OK',
|
|
1095
|
+
data: {
|
|
1096
|
+
received: true,
|
|
1097
|
+
message: {
|
|
1098
|
+
seq: 1,
|
|
1099
|
+
messageType: 'OFFER'
|
|
1100
|
+
}
|
|
1101
|
+
}
|
|
1102
|
+
};
|
|
1103
|
+
/* we receive roap Ok from mobius and expect mediaSdk to process it */
|
|
1104
|
+
call.sendMediaStateMachineEvt(dummyOkEvent);
|
|
1105
|
+
expect(_testUtil.mediaConnection.roapMessageReceived).toHaveBeenLastCalledWith(dummyOkEvent.data.message);
|
|
1106
|
+
expect(call['callStateMachine'].state.value).toBe('S_CALL_ESTABLISHED');
|
|
1107
|
+
expect(call.isConnected()).toBe(true);
|
|
1108
|
+
call.sendDigit('1');
|
|
1109
|
+
expect(dtmfMock).toBeCalledOnceWith('1');
|
|
1110
|
+
call.sendCallStateMachineEvt({
|
|
1111
|
+
type: 'E_RECV_CALL_DISCONNECT'
|
|
1112
|
+
});
|
|
1113
|
+
expect(deleteSpy).toBeCalledOnceWith();
|
|
1114
|
+
expect(call['callStateMachine'].state.value).toBe('S_RECV_CALL_DISCONNECT');
|
|
1115
|
+
case 1:
|
|
1116
|
+
case "end":
|
|
1117
|
+
return _context19.stop();
|
|
1118
|
+
}
|
|
1119
|
+
}, _callee19);
|
|
1120
|
+
})));
|
|
1121
|
+
it('state changes during unsuccessful incoming call due to no offer', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee20() {
|
|
1122
|
+
var statusPayload, dummyEvent;
|
|
1123
|
+
return _regenerator.default.wrap(function (_context20) {
|
|
1124
|
+
while (1) switch (_context20.prev = _context20.next) {
|
|
1125
|
+
case 0:
|
|
1126
|
+
call['direction'] = _types3.CallDirection.INBOUND;
|
|
1127
|
+
statusPayload = {
|
|
1128
|
+
statusCode: 200,
|
|
1129
|
+
body: mockStatusBody
|
|
1130
|
+
};
|
|
1131
|
+
dummyEvent = {
|
|
1132
|
+
type: 'E_RECV_CALL_SETUP',
|
|
1133
|
+
data: {
|
|
1134
|
+
seq: 1,
|
|
1135
|
+
message: {},
|
|
1136
|
+
type: 'OFFER'
|
|
1137
|
+
}
|
|
1138
|
+
};
|
|
1139
|
+
webex.request.mockReturnValue(statusPayload);
|
|
1140
|
+
call.sendCallStateMachineEvt(dummyEvent);
|
|
1141
|
+
expect(call['callStateMachine'].state.value).toBe('S_SEND_CALL_PROGRESS');
|
|
1142
|
+
_context20.next = 1;
|
|
1143
|
+
return call['handleOutgoingCallConnect']({
|
|
1144
|
+
type: 'E_SEND_CALL_CONNECT'
|
|
1145
|
+
});
|
|
1146
|
+
case 1:
|
|
1147
|
+
/* state should not change since there is no offer received. */
|
|
1148
|
+
expect(call['callStateMachine'].state.value).toBe('S_SEND_CALL_PROGRESS');
|
|
1149
|
+
expect(call.isConnected()).toBeFalsy();
|
|
1150
|
+
call.sendCallStateMachineEvt({
|
|
1151
|
+
type: 'E_RECV_CALL_DISCONNECT'
|
|
1152
|
+
});
|
|
1153
|
+
expect(call['callStateMachine'].state.value).toBe('S_RECV_CALL_DISCONNECT');
|
|
1154
|
+
case 2:
|
|
1155
|
+
case "end":
|
|
1156
|
+
return _context20.stop();
|
|
1157
|
+
}
|
|
1158
|
+
}, _callee20);
|
|
1159
|
+
})));
|
|
1160
|
+
it('processes callerId on received progress event in established state without emitting PROGRESS', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee21() {
|
|
1161
|
+
var callManager, statusPayload, call, dummyEvent, dummyOkEvent, emitSpy, startCallerIdSpy, mobiusProgressEvent;
|
|
1162
|
+
return _regenerator.default.wrap(function (_context21) {
|
|
1163
|
+
while (1) switch (_context21.prev = _context21.next) {
|
|
1164
|
+
case 0:
|
|
1165
|
+
callManager = (0, _callManager.getCallManager)(webex, defaultServiceIndicator);
|
|
1166
|
+
statusPayload = {
|
|
1167
|
+
statusCode: 200,
|
|
1168
|
+
body: mockStatusBody
|
|
1169
|
+
};
|
|
1170
|
+
call = callManager.createCall(_types3.CallDirection.OUTBOUND, deviceId, mockLineId, dest);
|
|
1171
|
+
webex.request.mockReturnValue(statusPayload);
|
|
1172
|
+
|
|
1173
|
+
// Move to S_SEND_CALL_SETUP
|
|
1174
|
+
dummyEvent = {
|
|
1175
|
+
type: 'E_SEND_CALL_SETUP',
|
|
1176
|
+
data: {
|
|
1177
|
+
seq: 1,
|
|
1178
|
+
message: {},
|
|
1179
|
+
type: 'OFFER'
|
|
1180
|
+
}
|
|
1181
|
+
};
|
|
1182
|
+
call.sendCallStateMachineEvt(dummyEvent);
|
|
1183
|
+
expect(call['callStateMachine'].state.value).toBe('S_SEND_CALL_SETUP');
|
|
1184
|
+
|
|
1185
|
+
// Complete media negotiation to allow connect -> established
|
|
1186
|
+
// Ask media SDK to initiate offer
|
|
1187
|
+
dummyEvent.type = 'E_SEND_ROAP_OFFER';
|
|
1188
|
+
dummyEvent.data = {
|
|
1189
|
+
seq: 1,
|
|
1190
|
+
messageType: 'OFFER',
|
|
1191
|
+
sdp: 'sdp'
|
|
1192
|
+
};
|
|
1193
|
+
call.sendMediaStateMachineEvt(dummyEvent);
|
|
1194
|
+
expect(call['mediaStateMachine'].state.value).toBe('S_SEND_ROAP_OFFER');
|
|
1195
|
+
dummyEvent.type = 'E_RECV_ROAP_ANSWER';
|
|
1196
|
+
dummyEvent.data = {
|
|
1197
|
+
seq: 1,
|
|
1198
|
+
messageType: 'ANSWER',
|
|
1199
|
+
sdp: 'sdp'
|
|
1200
|
+
};
|
|
1201
|
+
call.sendMediaStateMachineEvt(dummyEvent);
|
|
1202
|
+
expect(call['mediaStateMachine'].state.value).toBe('S_RECV_ROAP_ANSWER');
|
|
1203
|
+
// Send OK
|
|
1204
|
+
dummyOkEvent = {
|
|
1205
|
+
type: 'E_ROAP_OK',
|
|
1206
|
+
data: {
|
|
1207
|
+
received: false,
|
|
1208
|
+
message: {
|
|
1209
|
+
seq: 1,
|
|
1210
|
+
messageType: 'OK'
|
|
1211
|
+
}
|
|
1212
|
+
}
|
|
1213
|
+
};
|
|
1214
|
+
call.sendMediaStateMachineEvt(dummyOkEvent);
|
|
1215
|
+
expect(call['mediaStateMachine'].state.value).toBe('S_ROAP_OK');
|
|
1216
|
+
expect(call['mediaNegotiationCompleted']).toBe(true);
|
|
1217
|
+
|
|
1218
|
+
// Move call to established
|
|
1219
|
+
dummyEvent.type = 'E_RECV_CALL_CONNECT';
|
|
1220
|
+
dummyEvent.data = undefined;
|
|
1221
|
+
call.sendCallStateMachineEvt(dummyEvent);
|
|
1222
|
+
expect(call['callStateMachine'].state.value).toBe('S_CALL_ESTABLISHED');
|
|
1223
|
+
emitSpy = jest.spyOn(call, 'emit');
|
|
1224
|
+
startCallerIdSpy = jest.spyOn(call, 'startCallerIdResolution'); // Now send progress with callerId while established via CallManager (Mobius event)
|
|
1225
|
+
mobiusProgressEvent = {
|
|
1226
|
+
id: 'evt1',
|
|
1227
|
+
timestamp: (0, _now.default)(),
|
|
1228
|
+
trackingId: 'track-1',
|
|
1229
|
+
data: {
|
|
1230
|
+
eventType: _types5.MobiusEventType.CALL_PROGRESS,
|
|
1231
|
+
callerId: {
|
|
1232
|
+
from: '"Bob Marley" <sip:5010@207.182.171.130;user=phone>;tag=888068389-1654853820619-'
|
|
1233
|
+
},
|
|
1234
|
+
callProgressData: {
|
|
1235
|
+
inbandMedia: true,
|
|
1236
|
+
alerting: false
|
|
1237
|
+
},
|
|
1238
|
+
callId: call.getCallId(),
|
|
1239
|
+
callUrl: 'https://mobius.example/call',
|
|
1240
|
+
deviceId: deviceId,
|
|
1241
|
+
correlationId: call.getCorrelationId()
|
|
1242
|
+
}
|
|
1243
|
+
};
|
|
1244
|
+
callManager['dequeueWsEvents'](mobiusProgressEvent);
|
|
1245
|
+
|
|
1246
|
+
// CallerId resolution should be triggered exactly once (handled by CallManager)
|
|
1247
|
+
expect(startCallerIdSpy).toBeCalledOnceWith(mobiusProgressEvent.data.callerId);
|
|
1248
|
+
// Since it returns early in established state, PROGRESS event should not be emitted here
|
|
1249
|
+
expect(emitSpy.mock.calls.find(function (args) {
|
|
1250
|
+
return args && args[0] === _types2.CALL_EVENT_KEYS.PROGRESS;
|
|
1251
|
+
})).toBeUndefined();
|
|
1252
|
+
expect(call['callStateMachine'].state.value).not.toBe('S_RECV_CALL_PROGRESS');
|
|
1253
|
+
// Early media flag should not be set due to early return
|
|
1254
|
+
expect(call.earlyMedia).not.toBe(true);
|
|
1255
|
+
case 1:
|
|
1256
|
+
case "end":
|
|
1257
|
+
return _context21.stop();
|
|
1258
|
+
}
|
|
1259
|
+
}, _callee21);
|
|
1260
|
+
})));
|
|
1261
|
+
it('state changes during unsuccessful incoming call due error in call connect', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee22() {
|
|
1262
|
+
var warnSpy, errorSpy, stateMachineSpy, statusPayload, roapMessage;
|
|
1263
|
+
return _regenerator.default.wrap(function (_context22) {
|
|
1264
|
+
while (1) switch (_context22.prev = _context22.next) {
|
|
1265
|
+
case 0:
|
|
1266
|
+
warnSpy = jest.spyOn(_Logger.default, 'warn');
|
|
1267
|
+
errorSpy = jest.spyOn(_Logger.default, 'error');
|
|
1268
|
+
stateMachineSpy = jest.spyOn(call, 'sendCallStateMachineEvt');
|
|
1269
|
+
statusPayload = {
|
|
1270
|
+
statusCode: 200,
|
|
1271
|
+
body: mockStatusBody
|
|
1272
|
+
};
|
|
1273
|
+
roapMessage = {
|
|
1274
|
+
seq: 1,
|
|
1275
|
+
message: {},
|
|
1276
|
+
type: 'OFFER'
|
|
1277
|
+
};
|
|
1278
|
+
call['direction'] = _types3.CallDirection.INBOUND;
|
|
1279
|
+
webex.request.mockReturnValue(statusPayload);
|
|
1280
|
+
call.sendCallStateMachineEvt({
|
|
1281
|
+
type: 'E_RECV_CALL_SETUP'
|
|
1282
|
+
});
|
|
1283
|
+
expect(call['callStateMachine'].state.value).toBe('S_SEND_CALL_PROGRESS');
|
|
1284
|
+
call.sendMediaStateMachineEvt({
|
|
1285
|
+
type: 'E_RECV_ROAP_OFFER',
|
|
1286
|
+
data: roapMessage
|
|
1287
|
+
});
|
|
1288
|
+
webex.request.mockRejectedValueOnce({
|
|
1289
|
+
statusCode: 403
|
|
1290
|
+
}).mockResolvedValue(statusPayload);
|
|
1291
|
+
_context22.next = 1;
|
|
1292
|
+
return call['handleOutgoingCallConnect']({
|
|
1293
|
+
type: 'E_SEND_CALL_CONNECT'
|
|
1294
|
+
});
|
|
1295
|
+
case 1:
|
|
1296
|
+
expect(call['callStateMachine'].state.value).toBe('S_UNKNOWN');
|
|
1297
|
+
expect(stateMachineSpy).toBeCalledTimes(3);
|
|
1298
|
+
expect(warnSpy).toBeCalledTimes(3);
|
|
1299
|
+
expect(errorSpy).toBeCalledTimes(1);
|
|
1300
|
+
case 2:
|
|
1301
|
+
case "end":
|
|
1302
|
+
return _context22.stop();
|
|
1303
|
+
}
|
|
1304
|
+
}, _callee22);
|
|
1305
|
+
})));
|
|
1306
|
+
it('state changes during successful outgoing call', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee23() {
|
|
1307
|
+
var statusPayload, dummyEvent, postMediaSpy, logSpy, dummyOkEvent;
|
|
1308
|
+
return _regenerator.default.wrap(function (_context23) {
|
|
1309
|
+
while (1) switch (_context23.prev = _context23.next) {
|
|
1310
|
+
case 0:
|
|
1311
|
+
statusPayload = {
|
|
1312
|
+
statusCode: 200,
|
|
1313
|
+
body: mockStatusBody
|
|
1314
|
+
};
|
|
1315
|
+
dummyEvent = {
|
|
1316
|
+
type: 'E_SEND_CALL_SETUP',
|
|
1317
|
+
data: undefined
|
|
1318
|
+
};
|
|
1319
|
+
postMediaSpy = jest.spyOn(call, 'postMedia');
|
|
1320
|
+
logSpy = jest.spyOn(_Logger.default, 'log');
|
|
1321
|
+
webex.request.mockReturnValue(statusPayload);
|
|
1322
|
+
call.sendCallStateMachineEvt(dummyEvent);
|
|
1323
|
+
expect(call['callStateMachine'].state.value).toBe('S_SEND_CALL_SETUP');
|
|
1324
|
+
expect(logSpy).toHaveBeenCalled();
|
|
1325
|
+
dummyEvent.type = 'E_RECV_CALL_PROGRESS';
|
|
1326
|
+
call.sendCallStateMachineEvt(dummyEvent);
|
|
1327
|
+
expect(call['callStateMachine'].state.value).toBe('S_RECV_CALL_PROGRESS');
|
|
1328
|
+
dummyEvent.type = 'E_SEND_ROAP_OFFER';
|
|
1329
|
+
call.sendMediaStateMachineEvt(dummyEvent);
|
|
1330
|
+
|
|
1331
|
+
/**
|
|
1332
|
+
* Since the event doesn't have any data above, we should request media sdk for an offer here.
|
|
1333
|
+
* The below event is expected to be called again my mediaSdk.
|
|
1334
|
+
*/
|
|
1335
|
+
dummyEvent.data = {
|
|
1336
|
+
seq: 1,
|
|
1337
|
+
messageType: 'OFFER',
|
|
1338
|
+
sdp: 'sdp'
|
|
1339
|
+
};
|
|
1340
|
+
call.sendMediaStateMachineEvt(dummyEvent);
|
|
1341
|
+
expect(_testUtil.mediaConnection.initiateOffer).toHaveBeenCalledTimes(1);
|
|
1342
|
+
expect(postMediaSpy).toHaveBeenLastCalledWith(dummyEvent.data);
|
|
1343
|
+
dummyEvent.type = 'E_RECV_ROAP_ANSWER';
|
|
1344
|
+
call.sendMediaStateMachineEvt(dummyEvent);
|
|
1345
|
+
expect(_testUtil.mediaConnection.roapMessageReceived).toHaveBeenLastCalledWith(dummyEvent.data);
|
|
1346
|
+
dummyOkEvent = {
|
|
1347
|
+
type: 'E_ROAP_OK',
|
|
1348
|
+
data: {
|
|
1349
|
+
received: false,
|
|
1350
|
+
message: {
|
|
1351
|
+
seq: 1,
|
|
1352
|
+
messageType: 'OK'
|
|
1353
|
+
}
|
|
1354
|
+
}
|
|
1355
|
+
};
|
|
1356
|
+
call.sendMediaStateMachineEvt(dummyOkEvent);
|
|
1357
|
+
expect(postMediaSpy).toHaveBeenLastCalledWith(dummyOkEvent.data.message);
|
|
1358
|
+
dummyEvent.type = 'E_RECV_ROAP_OFFER_REQUEST';
|
|
1359
|
+
call.sendMediaStateMachineEvt(dummyEvent);
|
|
1360
|
+
expect(_testUtil.mediaConnection.roapMessageReceived).toHaveBeenLastCalledWith(dummyEvent.data);
|
|
1361
|
+
dummyEvent.type = 'E_SEND_ROAP_OFFER';
|
|
1362
|
+
call.sendMediaStateMachineEvt(dummyEvent);
|
|
1363
|
+
expect(postMediaSpy).toHaveBeenLastCalledWith(dummyEvent.data);
|
|
1364
|
+
dummyEvent.type = 'E_RECV_ROAP_ANSWER';
|
|
1365
|
+
call.sendMediaStateMachineEvt(dummyEvent);
|
|
1366
|
+
expect(_testUtil.mediaConnection.roapMessageReceived).toHaveBeenLastCalledWith(dummyEvent.data);
|
|
1367
|
+
call.sendMediaStateMachineEvt(dummyOkEvent);
|
|
1368
|
+
expect(postMediaSpy).toHaveBeenLastCalledWith(dummyOkEvent.data.message);
|
|
1369
|
+
|
|
1370
|
+
/* With the two roap offer/answer transactions that we simulated earlier
|
|
1371
|
+
we get a total 4 outgoing and 3 incoming roap messages.
|
|
1372
|
+
*/
|
|
1373
|
+
expect(postMediaSpy).toBeCalledTimes(4);
|
|
1374
|
+
expect(_testUtil.mediaConnection.roapMessageReceived).toBeCalledTimes(3);
|
|
1375
|
+
dummyEvent.type = 'E_RECV_CALL_CONNECT';
|
|
1376
|
+
call.sendCallStateMachineEvt(dummyEvent);
|
|
1377
|
+
expect(call['callStateMachine'].state.value).toBe('S_CALL_ESTABLISHED');
|
|
1378
|
+
expect(call.isConnected()).toBe(true);
|
|
1379
|
+
call.sendCallStateMachineEvt({
|
|
1380
|
+
type: 'E_SEND_CALL_DISCONNECT'
|
|
1381
|
+
});
|
|
1382
|
+
expect(call['callStateMachine'].state.value).toBe('S_SEND_CALL_DISCONNECT');
|
|
1383
|
+
case 1:
|
|
1384
|
+
case "end":
|
|
1385
|
+
return _context23.stop();
|
|
1386
|
+
}
|
|
1387
|
+
}, _callee23);
|
|
1388
|
+
})));
|
|
1389
|
+
it('outgoing call where we receive connect directly after setup. Media established before connect. test call and media state changes', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee24() {
|
|
1390
|
+
var statusPayload, dummySetupEvent, dummyConnectEvent, dummyOfferEvent, dummyAnswerEvent, dummyOkEvent, postMediaSpy;
|
|
1391
|
+
return _regenerator.default.wrap(function (_context24) {
|
|
1392
|
+
while (1) switch (_context24.prev = _context24.next) {
|
|
1393
|
+
case 0:
|
|
1394
|
+
statusPayload = {
|
|
1395
|
+
statusCode: 200,
|
|
1396
|
+
body: mockStatusBody
|
|
1397
|
+
};
|
|
1398
|
+
dummySetupEvent = {
|
|
1399
|
+
type: 'E_SEND_CALL_SETUP',
|
|
1400
|
+
data: undefined
|
|
1401
|
+
};
|
|
1402
|
+
dummyConnectEvent = {
|
|
1403
|
+
type: 'E_RECV_CALL_CONNECT',
|
|
1404
|
+
data: undefined
|
|
1405
|
+
};
|
|
1406
|
+
dummyOfferEvent = {
|
|
1407
|
+
type: 'E_SEND_ROAP_OFFER',
|
|
1408
|
+
data: undefined
|
|
1409
|
+
};
|
|
1410
|
+
dummyAnswerEvent = {
|
|
1411
|
+
type: 'E_RECV_ROAP_ANSWER',
|
|
1412
|
+
data: {
|
|
1413
|
+
seq: 1,
|
|
1414
|
+
messageType: 'ANSWER',
|
|
1415
|
+
sdp: 'sdp'
|
|
1416
|
+
}
|
|
1417
|
+
};
|
|
1418
|
+
dummyOkEvent = {
|
|
1419
|
+
type: 'E_ROAP_OK',
|
|
1420
|
+
data: {
|
|
1421
|
+
received: false,
|
|
1422
|
+
message: {
|
|
1423
|
+
seq: 1,
|
|
1424
|
+
messageType: 'OK'
|
|
1425
|
+
}
|
|
1426
|
+
}
|
|
1427
|
+
};
|
|
1428
|
+
postMediaSpy = jest.spyOn(call, 'postMedia');
|
|
1429
|
+
webex.request.mockReturnValue(statusPayload);
|
|
1430
|
+
call.sendCallStateMachineEvt(dummySetupEvent);
|
|
1431
|
+
expect(call['callStateMachine'].state.value).toBe('S_SEND_CALL_SETUP');
|
|
1432
|
+
call.sendMediaStateMachineEvt(dummyOfferEvent);
|
|
1433
|
+
|
|
1434
|
+
/**
|
|
1435
|
+
* Since the event doesn't have any data above, we should request media sdk for an offer here.
|
|
1436
|
+
* The below event is expected to be called again my mediaSdk.
|
|
1437
|
+
*/
|
|
1438
|
+
dummyOfferEvent.data = {
|
|
1439
|
+
seq: 1,
|
|
1440
|
+
messageType: 'OFFER',
|
|
1441
|
+
sdp: 'sdp'
|
|
1442
|
+
};
|
|
1443
|
+
call.sendMediaStateMachineEvt(dummyOfferEvent);
|
|
1444
|
+
expect(_testUtil.mediaConnection.initiateOffer).toHaveBeenCalledTimes(1);
|
|
1445
|
+
expect(postMediaSpy).toHaveBeenLastCalledWith(dummyOfferEvent.data);
|
|
1446
|
+
call.sendMediaStateMachineEvt(dummyAnswerEvent);
|
|
1447
|
+
expect(_testUtil.mediaConnection.roapMessageReceived).toHaveBeenLastCalledWith(dummyAnswerEvent.data);
|
|
1448
|
+
call.sendMediaStateMachineEvt(dummyOkEvent);
|
|
1449
|
+
expect(postMediaSpy).toHaveBeenLastCalledWith(dummyOkEvent.data.message);
|
|
1450
|
+
expect(call['mediaStateMachine'].state.value).toBe('S_ROAP_OK');
|
|
1451
|
+
call.sendCallStateMachineEvt(dummyConnectEvent);
|
|
1452
|
+
expect(call['callStateMachine'].state.value).toBe('S_CALL_ESTABLISHED');
|
|
1453
|
+
expect(call.isConnected()).toBe(true);
|
|
1454
|
+
call.sendCallStateMachineEvt({
|
|
1455
|
+
type: 'E_SEND_CALL_DISCONNECT'
|
|
1456
|
+
});
|
|
1457
|
+
expect(call['callStateMachine'].state.value).toBe('S_SEND_CALL_DISCONNECT');
|
|
1458
|
+
case 1:
|
|
1459
|
+
case "end":
|
|
1460
|
+
return _context24.stop();
|
|
1461
|
+
}
|
|
1462
|
+
}, _callee24);
|
|
1463
|
+
})));
|
|
1464
|
+
it('state changes during successful outgoing call with early media', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee25() {
|
|
1465
|
+
var statusPayload, dummyEvent;
|
|
1466
|
+
return _regenerator.default.wrap(function (_context25) {
|
|
1467
|
+
while (1) switch (_context25.prev = _context25.next) {
|
|
1468
|
+
case 0:
|
|
1469
|
+
statusPayload = {
|
|
1470
|
+
statusCode: 200,
|
|
1471
|
+
body: mockStatusBody
|
|
1472
|
+
};
|
|
1473
|
+
dummyEvent = {
|
|
1474
|
+
type: 'E_SEND_CALL_SETUP',
|
|
1475
|
+
data: {
|
|
1476
|
+
seq: 1,
|
|
1477
|
+
message: {},
|
|
1478
|
+
type: 'OFFER'
|
|
1479
|
+
}
|
|
1480
|
+
};
|
|
1481
|
+
webex.request.mockReturnValue(statusPayload);
|
|
1482
|
+
call.sendCallStateMachineEvt(dummyEvent);
|
|
1483
|
+
expect(call['callStateMachine'].state.value).toBe('S_SEND_CALL_SETUP');
|
|
1484
|
+
dummyEvent.type = 'E_RECV_CALL_PROGRESS';
|
|
1485
|
+
call.sendCallStateMachineEvt(dummyEvent);
|
|
1486
|
+
expect(call['callStateMachine'].state.value).toBe('S_RECV_CALL_PROGRESS');
|
|
1487
|
+
|
|
1488
|
+
/* Send a second callProgress event with inband media and it should handle */
|
|
1489
|
+
dummyEvent.data['callProgressData'] = {
|
|
1490
|
+
inbandMedia: true
|
|
1491
|
+
};
|
|
1492
|
+
dummyEvent.data['callerId'] = {
|
|
1493
|
+
from: '"Bob Marley" <sip:5010@207.182.171.130;user=phone>;tag=888068389-1654853820619-'
|
|
1494
|
+
};
|
|
1495
|
+
call.sendCallStateMachineEvt(dummyEvent);
|
|
1496
|
+
expect(call['callStateMachine'].state.value).toBe('S_RECV_CALL_PROGRESS');
|
|
1497
|
+
dummyEvent.type = 'E_RECV_CALL_CONNECT';
|
|
1498
|
+
call.sendCallStateMachineEvt(dummyEvent);
|
|
1499
|
+
expect(call['callStateMachine'].state.value).toBe('S_CALL_ESTABLISHED');
|
|
1500
|
+
call.sendCallStateMachineEvt({
|
|
1501
|
+
type: 'E_RECV_CALL_DISCONNECT'
|
|
1502
|
+
});
|
|
1503
|
+
expect(call['callStateMachine'].state.value).toBe('S_RECV_CALL_DISCONNECT');
|
|
1504
|
+
case 1:
|
|
1505
|
+
case "end":
|
|
1506
|
+
return _context25.stop();
|
|
1507
|
+
}
|
|
1508
|
+
}, _callee25);
|
|
1509
|
+
})));
|
|
1510
|
+
it('state changes during unsuccessful outgoing call due to error in call setup', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee26() {
|
|
1511
|
+
var statusPayload, dummyEvent, errorSpy;
|
|
1512
|
+
return _regenerator.default.wrap(function (_context26) {
|
|
1513
|
+
while (1) switch (_context26.prev = _context26.next) {
|
|
1514
|
+
case 0:
|
|
1515
|
+
statusPayload = {
|
|
1516
|
+
statusCode: 403,
|
|
1517
|
+
body: mockStatusBody
|
|
1518
|
+
};
|
|
1519
|
+
dummyEvent = {
|
|
1520
|
+
type: 'E_SEND_CALL_SETUP',
|
|
1521
|
+
data: {
|
|
1522
|
+
seq: 1,
|
|
1523
|
+
message: {},
|
|
1524
|
+
type: 'OFFER'
|
|
1525
|
+
}
|
|
1526
|
+
};
|
|
1527
|
+
webex.request.mockRejectedValueOnce(statusPayload);
|
|
1528
|
+
errorSpy = jest.spyOn(_Logger.default, 'error');
|
|
1529
|
+
call.sendCallStateMachineEvt(dummyEvent);
|
|
1530
|
+
_context26.next = 1;
|
|
1531
|
+
return (0, _testUtil.flushPromises)(3);
|
|
1532
|
+
case 1:
|
|
1533
|
+
expect(call['callStateMachine'].state.value).toBe('S_UNKNOWN');
|
|
1534
|
+
expect(errorSpy).toHaveBeenCalled();
|
|
1535
|
+
expect(uploadLogsSpy).toHaveBeenCalledWith({
|
|
1536
|
+
correlationId: call.getCorrelationId(),
|
|
1537
|
+
callId: call.getCallId()
|
|
1538
|
+
});
|
|
1539
|
+
case 2:
|
|
1540
|
+
case "end":
|
|
1541
|
+
return _context26.stop();
|
|
1542
|
+
}
|
|
1543
|
+
}, _callee26);
|
|
1544
|
+
})));
|
|
1545
|
+
it('state changes during unsuccessful outgoing call due to error in media ok', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee27() {
|
|
1546
|
+
var statusPayload, dummyEvent, warnSpy;
|
|
1547
|
+
return _regenerator.default.wrap(function (_context27) {
|
|
1548
|
+
while (1) switch (_context27.prev = _context27.next) {
|
|
1549
|
+
case 0:
|
|
1550
|
+
statusPayload = {
|
|
1551
|
+
statusCode: 403,
|
|
1552
|
+
body: mockStatusBody
|
|
1553
|
+
};
|
|
1554
|
+
dummyEvent = {
|
|
1555
|
+
type: 'E_ROAP_OK',
|
|
1556
|
+
data: {
|
|
1557
|
+
received: false,
|
|
1558
|
+
message: {}
|
|
1559
|
+
}
|
|
1560
|
+
};
|
|
1561
|
+
call['callStateMachine'].state.value = 'S_RECV_CALL_PROGRESS';
|
|
1562
|
+
call['earlyMedia'] = true;
|
|
1563
|
+
call['mediaStateMachine'].state.value = 'S_RECV_ROAP_ANSWER';
|
|
1564
|
+
webex.request.mockRejectedValue(statusPayload);
|
|
1565
|
+
warnSpy = jest.spyOn(_Logger.default, 'warn');
|
|
1566
|
+
jest.spyOn(Utils, 'uploadLogs').mockResolvedValue(undefined);
|
|
1567
|
+
_context27.next = 1;
|
|
1568
|
+
return call['handleRoapEstablished']({}, dummyEvent);
|
|
1569
|
+
case 1:
|
|
1570
|
+
_context27.next = 2;
|
|
1571
|
+
return (0, _testUtil.flushPromises)(2);
|
|
1572
|
+
case 2:
|
|
1573
|
+
expect(call.isConnected()).toBe(false);
|
|
1574
|
+
expect(call['mediaStateMachine'].state.value).toBe('S_ROAP_TEARDOWN');
|
|
1575
|
+
expect(call['callStateMachine'].state.value).toBe('S_CALL_CLEARED');
|
|
1576
|
+
expect(warnSpy).toHaveBeenCalledWith('Failed to process MediaOk request', {
|
|
1577
|
+
file: 'call',
|
|
1578
|
+
method: 'handleRoapEstablished'
|
|
1579
|
+
});
|
|
1580
|
+
expect(warnSpy).toHaveBeenCalledWith('Call failed due to media issue', {
|
|
1581
|
+
file: 'call',
|
|
1582
|
+
method: 'handleRoapError'
|
|
1583
|
+
});
|
|
1584
|
+
case 3:
|
|
1585
|
+
case "end":
|
|
1586
|
+
return _context27.stop();
|
|
1587
|
+
}
|
|
1588
|
+
}, _callee27);
|
|
1589
|
+
})));
|
|
1590
|
+
it('state changes during unsuccessful outgoing call since no sdp in offer', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee28() {
|
|
1591
|
+
var statusPayload, dummyEvent;
|
|
1592
|
+
return _regenerator.default.wrap(function (_context28) {
|
|
1593
|
+
while (1) switch (_context28.prev = _context28.next) {
|
|
1594
|
+
case 0:
|
|
1595
|
+
statusPayload = {
|
|
1596
|
+
statusCode: 403,
|
|
1597
|
+
body: mockStatusBody
|
|
1598
|
+
};
|
|
1599
|
+
dummyEvent = {
|
|
1600
|
+
type: 'E_SEND_ROAP_OFFER',
|
|
1601
|
+
data: {
|
|
1602
|
+
seq: 1,
|
|
1603
|
+
type: 'OFFER'
|
|
1604
|
+
}
|
|
1605
|
+
};
|
|
1606
|
+
webex.request.mockReturnValue(statusPayload);
|
|
1607
|
+
call['callStateMachine'].state.value = 'S_RECV_CALL_PROGRESS';
|
|
1608
|
+
call['mediaStateMachine'].state.value = 'S_ROAP_IDLE';
|
|
1609
|
+
call.sendMediaStateMachineEvt(dummyEvent);
|
|
1610
|
+
/* since there is no sdp, mediaMachine should remain in same state
|
|
1611
|
+
and an initiateOffer request to be sent to SDK only once */
|
|
1612
|
+
expect(call['mediaStateMachine'].state.value).toBe('S_SEND_ROAP_OFFER');
|
|
1613
|
+
expect(_testUtil.mediaConnection.initiateOffer).toBeCalledOnceWith();
|
|
1614
|
+
case 1:
|
|
1615
|
+
case "end":
|
|
1616
|
+
return _context28.stop();
|
|
1617
|
+
}
|
|
1618
|
+
}, _callee28);
|
|
1619
|
+
})));
|
|
1620
|
+
it('Outgoing Roap offer retry-after error case during midcall', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee29() {
|
|
1621
|
+
var statusPayload, funcSpy, stateMachineSpy, dummyEvent;
|
|
1622
|
+
return _regenerator.default.wrap(function (_context29) {
|
|
1623
|
+
while (1) switch (_context29.prev = _context29.next) {
|
|
1624
|
+
case 0:
|
|
1625
|
+
statusPayload = {
|
|
1626
|
+
statusCode: 503,
|
|
1627
|
+
headers: {
|
|
1628
|
+
'retry-after': 1
|
|
1629
|
+
},
|
|
1630
|
+
body: mockStatusBody
|
|
1631
|
+
};
|
|
1632
|
+
jest.spyOn(global, 'setTimeout');
|
|
1633
|
+
funcSpy = jest.spyOn(call, 'postMedia').mockRejectedValueOnce(statusPayload);
|
|
1634
|
+
stateMachineSpy = jest.spyOn(call, 'sendMediaStateMachineEvt');
|
|
1635
|
+
dummyEvent = {
|
|
1636
|
+
type: 'E_SEND_ROAP_OFFER',
|
|
1637
|
+
data: {
|
|
1638
|
+
seq: 1,
|
|
1639
|
+
message: {},
|
|
1640
|
+
type: 'OFFER',
|
|
1641
|
+
sdp: 'sdp'
|
|
1642
|
+
}
|
|
1643
|
+
};
|
|
1644
|
+
call['connected'] = true;
|
|
1645
|
+
_context29.next = 1;
|
|
1646
|
+
return call['handleOutgoingRoapOffer']({}, dummyEvent);
|
|
1647
|
+
case 1:
|
|
1648
|
+
jest.advanceTimersByTime(1005);
|
|
1649
|
+
expect(setTimeout).toHaveBeenCalledTimes(1);
|
|
1650
|
+
expect(funcSpy).toHaveBeenCalledWith(dummyEvent.data);
|
|
1651
|
+
|
|
1652
|
+
/* first failure , and second because of retry-after */
|
|
1653
|
+
expect(funcSpy).toHaveBeenCalledTimes(2);
|
|
1654
|
+
expect(stateMachineSpy).toBeCalledOnceWith(dummyEvent);
|
|
1655
|
+
case 2:
|
|
1656
|
+
case "end":
|
|
1657
|
+
return _context29.stop();
|
|
1658
|
+
}
|
|
1659
|
+
}, _callee29);
|
|
1660
|
+
})));
|
|
1661
|
+
it('Outgoing Roap offer retry-after error case during call establishment', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee30() {
|
|
1662
|
+
var statusPayload, funcSpy, stateMachineSpy, dummyEvent;
|
|
1663
|
+
return _regenerator.default.wrap(function (_context30) {
|
|
1664
|
+
while (1) switch (_context30.prev = _context30.next) {
|
|
1665
|
+
case 0:
|
|
1666
|
+
statusPayload = {
|
|
1667
|
+
statusCode: 503,
|
|
1668
|
+
headers: {
|
|
1669
|
+
'retry-after': 1
|
|
1670
|
+
},
|
|
1671
|
+
body: mockStatusBody
|
|
1672
|
+
};
|
|
1673
|
+
jest.spyOn(global, 'setTimeout');
|
|
1674
|
+
funcSpy = jest.spyOn(call, 'postMedia').mockRejectedValueOnce(statusPayload);
|
|
1675
|
+
stateMachineSpy = jest.spyOn(call, 'sendMediaStateMachineEvt');
|
|
1676
|
+
dummyEvent = {
|
|
1677
|
+
type: 'E_SEND_ROAP_OFFER',
|
|
1678
|
+
data: {
|
|
1679
|
+
seq: 1,
|
|
1680
|
+
message: {},
|
|
1681
|
+
type: 'OFFER',
|
|
1682
|
+
sdp: 'sdp'
|
|
1683
|
+
}
|
|
1684
|
+
};
|
|
1685
|
+
call['connected'] = false;
|
|
1686
|
+
_context30.next = 1;
|
|
1687
|
+
return call['handleOutgoingRoapOffer']({}, dummyEvent);
|
|
1688
|
+
case 1:
|
|
1689
|
+
jest.advanceTimersByTime(1005);
|
|
1690
|
+
expect(funcSpy).toBeCalledOnceWith(dummyEvent.data);
|
|
1691
|
+
|
|
1692
|
+
/* These shouldn't be called as call is not connected yet */
|
|
1693
|
+
expect(setTimeout).not.toHaveBeenCalled();
|
|
1694
|
+
expect(stateMachineSpy).not.toBeCalled();
|
|
1695
|
+
case 2:
|
|
1696
|
+
case "end":
|
|
1697
|
+
return _context30.stop();
|
|
1698
|
+
}
|
|
1699
|
+
}, _callee30);
|
|
1700
|
+
})));
|
|
1701
|
+
it('Outgoing Roap Answer retry-after error case during midcall', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee31() {
|
|
1702
|
+
var statusPayload, funcSpy, stateMachineSpy, dummyEvent;
|
|
1703
|
+
return _regenerator.default.wrap(function (_context31) {
|
|
1704
|
+
while (1) switch (_context31.prev = _context31.next) {
|
|
1705
|
+
case 0:
|
|
1706
|
+
statusPayload = {
|
|
1707
|
+
statusCode: 503,
|
|
1708
|
+
headers: {
|
|
1709
|
+
'retry-after': 1
|
|
1710
|
+
},
|
|
1711
|
+
body: mockStatusBody
|
|
1712
|
+
};
|
|
1713
|
+
jest.spyOn(global, 'setTimeout');
|
|
1714
|
+
funcSpy = jest.spyOn(call, 'postMedia').mockRejectedValueOnce(statusPayload);
|
|
1715
|
+
stateMachineSpy = jest.spyOn(call, 'sendMediaStateMachineEvt');
|
|
1716
|
+
dummyEvent = {
|
|
1717
|
+
type: 'E_SEND_ROAP_ANSWER',
|
|
1718
|
+
data: {
|
|
1719
|
+
seq: 1,
|
|
1720
|
+
message: {},
|
|
1721
|
+
type: 'OFFER'
|
|
1722
|
+
}
|
|
1723
|
+
};
|
|
1724
|
+
call['connected'] = true;
|
|
1725
|
+
call['mediaStateMachine'].state.value = 'S_RECV_ROAP_OFFER';
|
|
1726
|
+
_context31.next = 1;
|
|
1727
|
+
return call['handleOutgoingRoapAnswer']({}, dummyEvent);
|
|
1728
|
+
case 1:
|
|
1729
|
+
jest.advanceTimersByTime(1005);
|
|
1730
|
+
expect(setTimeout).toHaveBeenCalledTimes(1);
|
|
1731
|
+
expect(funcSpy).toHaveBeenCalledWith(dummyEvent.data);
|
|
1732
|
+
|
|
1733
|
+
/* first failure , and second because of retry-after */
|
|
1734
|
+
expect(funcSpy).toHaveBeenCalledTimes(2);
|
|
1735
|
+
expect(stateMachineSpy).toBeCalledOnceWith(dummyEvent);
|
|
1736
|
+
case 2:
|
|
1737
|
+
case "end":
|
|
1738
|
+
return _context31.stop();
|
|
1739
|
+
}
|
|
1740
|
+
}, _callee31);
|
|
1741
|
+
})));
|
|
1742
|
+
it('Outgoing Roap answer retry-after error case during call establishment', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee32() {
|
|
1743
|
+
var statusPayload, funcSpy, stateMachineSpy, dummyEvent;
|
|
1744
|
+
return _regenerator.default.wrap(function (_context32) {
|
|
1745
|
+
while (1) switch (_context32.prev = _context32.next) {
|
|
1746
|
+
case 0:
|
|
1747
|
+
statusPayload = {
|
|
1748
|
+
statusCode: 503,
|
|
1749
|
+
headers: {
|
|
1750
|
+
'retry-after': 1
|
|
1751
|
+
},
|
|
1752
|
+
body: mockStatusBody
|
|
1753
|
+
};
|
|
1754
|
+
jest.spyOn(global, 'setTimeout');
|
|
1755
|
+
funcSpy = jest.spyOn(call, 'postMedia').mockRejectedValueOnce(statusPayload);
|
|
1756
|
+
stateMachineSpy = jest.spyOn(call, 'sendMediaStateMachineEvt');
|
|
1757
|
+
dummyEvent = {
|
|
1758
|
+
type: 'E_SEND_ROAP_ANSWER',
|
|
1759
|
+
data: {
|
|
1760
|
+
seq: 1,
|
|
1761
|
+
message: {},
|
|
1762
|
+
type: 'OFFER'
|
|
1763
|
+
}
|
|
1764
|
+
};
|
|
1765
|
+
call['connected'] = false;
|
|
1766
|
+
_context32.next = 1;
|
|
1767
|
+
return call['handleOutgoingRoapAnswer']({}, dummyEvent);
|
|
1768
|
+
case 1:
|
|
1769
|
+
jest.advanceTimersByTime(1005);
|
|
1770
|
+
expect(funcSpy).toBeCalledOnceWith(dummyEvent.data);
|
|
1771
|
+
expect(funcSpy).toHaveBeenCalledTimes(1);
|
|
1772
|
+
|
|
1773
|
+
/* These shouldn't be called as call is not connected yet */
|
|
1774
|
+
expect(setTimeout).not.toHaveBeenCalled();
|
|
1775
|
+
expect(stateMachineSpy).not.toBeCalled();
|
|
1776
|
+
case 2:
|
|
1777
|
+
case "end":
|
|
1778
|
+
return _context32.stop();
|
|
1779
|
+
}
|
|
1780
|
+
}, _callee32);
|
|
1781
|
+
})));
|
|
1782
|
+
it('ROAP error during mid call', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee33() {
|
|
1783
|
+
var statusPayload, warnSpy, stateMachineSpy, funcSpy, errorEvent;
|
|
1784
|
+
return _regenerator.default.wrap(function (_context33) {
|
|
1785
|
+
while (1) switch (_context33.prev = _context33.next) {
|
|
1786
|
+
case 0:
|
|
1787
|
+
statusPayload = {
|
|
1788
|
+
statusCode: 200,
|
|
1789
|
+
body: mockStatusBody
|
|
1790
|
+
};
|
|
1791
|
+
warnSpy = jest.spyOn(_Logger.default, 'warn');
|
|
1792
|
+
stateMachineSpy = jest.spyOn(call, 'sendCallStateMachineEvt');
|
|
1793
|
+
funcSpy = jest.spyOn(call, 'postMedia').mockResolvedValue(statusPayload);
|
|
1794
|
+
errorEvent = {
|
|
1795
|
+
type: 'E_ROAP_ERROR',
|
|
1796
|
+
data: {
|
|
1797
|
+
seq: 2,
|
|
1798
|
+
messageType: 'ERROR',
|
|
1799
|
+
errorType: 'OUT_OF_ORDER'
|
|
1800
|
+
}
|
|
1801
|
+
};
|
|
1802
|
+
call['mediaStateMachine'].state.value = 'S_SEND_ROAP_ANSWER';
|
|
1803
|
+
call['connected'] = true;
|
|
1804
|
+
call.sendMediaStateMachineEvt(errorEvent);
|
|
1805
|
+
expect(funcSpy).toBeCalledOnceWith(errorEvent.data);
|
|
1806
|
+
expect(warnSpy).not.toHaveBeenCalled();
|
|
1807
|
+
expect(stateMachineSpy).not.toHaveBeenCalled();
|
|
1808
|
+
case 1:
|
|
1809
|
+
case "end":
|
|
1810
|
+
return _context33.stop();
|
|
1811
|
+
}
|
|
1812
|
+
}, _callee33);
|
|
1813
|
+
})));
|
|
1814
|
+
it('ROAP ok retry-after during mid call', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee34() {
|
|
1815
|
+
var statusPayload, funcSpy, stateMachineSpy, dummyEvent;
|
|
1816
|
+
return _regenerator.default.wrap(function (_context34) {
|
|
1817
|
+
while (1) switch (_context34.prev = _context34.next) {
|
|
1818
|
+
case 0:
|
|
1819
|
+
statusPayload = {
|
|
1820
|
+
statusCode: 503,
|
|
1821
|
+
headers: {
|
|
1822
|
+
'retry-after': 1
|
|
1823
|
+
},
|
|
1824
|
+
body: mockStatusBody
|
|
1825
|
+
};
|
|
1826
|
+
jest.spyOn(global, 'setTimeout');
|
|
1827
|
+
funcSpy = jest.spyOn(call, 'postMedia').mockRejectedValueOnce(statusPayload);
|
|
1828
|
+
stateMachineSpy = jest.spyOn(call, 'sendMediaStateMachineEvt');
|
|
1829
|
+
dummyEvent = {
|
|
1830
|
+
type: 'E_ROAP_OK',
|
|
1831
|
+
data: {
|
|
1832
|
+
seq: 1,
|
|
1833
|
+
message: {},
|
|
1834
|
+
type: 'OK'
|
|
1835
|
+
}
|
|
1836
|
+
};
|
|
1837
|
+
call['connected'] = true;
|
|
1838
|
+
call['mediaStateMachine'].state.value = 'S_RECV_ROAP_ANSWER';
|
|
1839
|
+
_context34.next = 1;
|
|
1840
|
+
return call['handleRoapEstablished']({}, dummyEvent);
|
|
1841
|
+
case 1:
|
|
1842
|
+
jest.advanceTimersByTime(1005);
|
|
1843
|
+
expect(setTimeout).toHaveBeenCalledTimes(1);
|
|
1844
|
+
expect(funcSpy).toHaveBeenCalled();
|
|
1845
|
+
|
|
1846
|
+
/* first failure , and second because of retry-after */
|
|
1847
|
+
expect(funcSpy).toHaveBeenCalledTimes(2);
|
|
1848
|
+
expect(stateMachineSpy).toBeCalledOnceWith(dummyEvent);
|
|
1849
|
+
case 2:
|
|
1850
|
+
case "end":
|
|
1851
|
+
return _context34.stop();
|
|
1852
|
+
}
|
|
1853
|
+
}, _callee34);
|
|
1854
|
+
})));
|
|
1855
|
+
it('Unable to communicate roap error with mobius', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee35() {
|
|
1856
|
+
var statusPayload, stateMachineSpy, funcSpy, errorEvent;
|
|
1857
|
+
return _regenerator.default.wrap(function (_context35) {
|
|
1858
|
+
while (1) switch (_context35.prev = _context35.next) {
|
|
1859
|
+
case 0:
|
|
1860
|
+
statusPayload = {
|
|
1861
|
+
statusCode: 403,
|
|
1862
|
+
body: mockStatusBody
|
|
1863
|
+
};
|
|
1864
|
+
stateMachineSpy = jest.spyOn(call, 'sendCallStateMachineEvt');
|
|
1865
|
+
funcSpy = jest.spyOn(call, 'postMedia').mockRejectedValue(statusPayload);
|
|
1866
|
+
errorEvent = {
|
|
1867
|
+
type: 'E_ROAP_ERROR',
|
|
1868
|
+
data: {
|
|
1869
|
+
seq: 2,
|
|
1870
|
+
messageType: 'ERROR',
|
|
1871
|
+
errorType: 'OUT_OF_ORDER'
|
|
1872
|
+
}
|
|
1873
|
+
};
|
|
1874
|
+
call['mediaStateMachine'].state.value = 'S_SEND_ROAP_ANSWER';
|
|
1875
|
+
call['connected'] = true;
|
|
1876
|
+
call.sendMediaStateMachineEvt(errorEvent);
|
|
1877
|
+
expect(funcSpy).toBeCalledOnceWith(errorEvent.data);
|
|
1878
|
+
expect(stateMachineSpy).not.toHaveBeenCalled();
|
|
1879
|
+
case 1:
|
|
1880
|
+
case "end":
|
|
1881
|
+
return _context35.stop();
|
|
1882
|
+
}
|
|
1883
|
+
}, _callee35);
|
|
1884
|
+
})));
|
|
1885
|
+
it('ROAP error during call establishment', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee36() {
|
|
1886
|
+
var statusPayload, warnSpy, stateMachineSpy, funcSpy, errorEvent;
|
|
1887
|
+
return _regenerator.default.wrap(function (_context36) {
|
|
1888
|
+
while (1) switch (_context36.prev = _context36.next) {
|
|
1889
|
+
case 0:
|
|
1890
|
+
statusPayload = {
|
|
1891
|
+
statusCode: 200,
|
|
1892
|
+
body: mockStatusBody
|
|
1893
|
+
};
|
|
1894
|
+
warnSpy = jest.spyOn(_Logger.default, 'warn');
|
|
1895
|
+
stateMachineSpy = jest.spyOn(call, 'sendCallStateMachineEvt');
|
|
1896
|
+
funcSpy = jest.spyOn(call, 'postMedia').mockResolvedValue(statusPayload);
|
|
1897
|
+
errorEvent = {
|
|
1898
|
+
type: 'E_ROAP_ERROR',
|
|
1899
|
+
data: {
|
|
1900
|
+
seq: 2,
|
|
1901
|
+
messageType: 'ERROR',
|
|
1902
|
+
errorType: 'OUT_OF_ORDER'
|
|
1903
|
+
}
|
|
1904
|
+
};
|
|
1905
|
+
call['connected'] = false;
|
|
1906
|
+
_context36.next = 1;
|
|
1907
|
+
return call['handleRoapError']({}, errorEvent);
|
|
1908
|
+
case 1:
|
|
1909
|
+
expect(funcSpy).toBeCalledOnceWith(errorEvent.data);
|
|
1910
|
+
expect(warnSpy).toBeCalledOnceWith('Call failed due to media issue', {
|
|
1911
|
+
file: 'call',
|
|
1912
|
+
method: 'handleRoapError'
|
|
1913
|
+
});
|
|
1914
|
+
expect(stateMachineSpy).toBeCalledOnceWith({
|
|
1915
|
+
data: {
|
|
1916
|
+
media: true
|
|
1917
|
+
},
|
|
1918
|
+
type: 'E_UNKNOWN'
|
|
1919
|
+
});
|
|
1920
|
+
case 2:
|
|
1921
|
+
case "end":
|
|
1922
|
+
return _context36.stop();
|
|
1923
|
+
}
|
|
1924
|
+
}, _callee36);
|
|
1925
|
+
})));
|
|
1926
|
+
it('incoming call: failing ROAP_ANSWER posts error path and tears down', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee37() {
|
|
1927
|
+
var statusPayload, warnSpy, postMediaSpy, setupEvent, connectEvent, offerEvent, answerEvent;
|
|
1928
|
+
return _regenerator.default.wrap(function (_context37) {
|
|
1929
|
+
while (1) switch (_context37.prev = _context37.next) {
|
|
1930
|
+
case 0:
|
|
1931
|
+
statusPayload = {
|
|
1932
|
+
statusCode: 403,
|
|
1933
|
+
body: mockStatusBody
|
|
1934
|
+
};
|
|
1935
|
+
warnSpy = jest.spyOn(_Logger.default, 'warn');
|
|
1936
|
+
postMediaSpy = jest.spyOn(call, 'postMedia').mockRejectedValueOnce(statusPayload); // Simulate inbound call flow
|
|
1937
|
+
call['direction'] = _types3.CallDirection.INBOUND;
|
|
1938
|
+
setupEvent = {
|
|
1939
|
+
type: 'E_RECV_CALL_SETUP',
|
|
1940
|
+
data: {
|
|
1941
|
+
seq: 1,
|
|
1942
|
+
messageType: 'OFFER'
|
|
1943
|
+
}
|
|
1944
|
+
};
|
|
1945
|
+
call.sendCallStateMachineEvt(setupEvent);
|
|
1946
|
+
expect(call['callStateMachine'].state.value).toBe('S_SEND_CALL_PROGRESS');
|
|
1947
|
+
connectEvent = {
|
|
1948
|
+
type: 'E_SEND_CALL_CONNECT'
|
|
1949
|
+
};
|
|
1950
|
+
call.sendCallStateMachineEvt(connectEvent);
|
|
1951
|
+
expect(call['callStateMachine'].state.value).toBe('S_SEND_CALL_CONNECT');
|
|
1952
|
+
offerEvent = {
|
|
1953
|
+
type: 'E_RECV_ROAP_OFFER',
|
|
1954
|
+
data: {
|
|
1955
|
+
seq: 1,
|
|
1956
|
+
messageType: 'OFFER'
|
|
1957
|
+
}
|
|
1958
|
+
};
|
|
1959
|
+
call.sendMediaStateMachineEvt(offerEvent);
|
|
1960
|
+
answerEvent = {
|
|
1961
|
+
type: 'E_SEND_ROAP_ANSWER',
|
|
1962
|
+
data: {
|
|
1963
|
+
seq: 1,
|
|
1964
|
+
messageType: 'ANSWER'
|
|
1965
|
+
}
|
|
1966
|
+
};
|
|
1967
|
+
_context37.next = 1;
|
|
1968
|
+
return call.sendMediaStateMachineEvt(answerEvent);
|
|
1969
|
+
case 1:
|
|
1970
|
+
_context37.next = 2;
|
|
1971
|
+
return (0, _testUtil.flushPromises)(2);
|
|
1972
|
+
case 2:
|
|
1973
|
+
expect(postMediaSpy).toBeCalledOnceWith(answerEvent.data);
|
|
1974
|
+
expect(warnSpy).toHaveBeenCalledWith('Failed to send MediaAnswer request', {
|
|
1975
|
+
file: 'call',
|
|
1976
|
+
method: 'handleOutgoingRoapAnswer'
|
|
1977
|
+
});
|
|
1978
|
+
expect(warnSpy).toHaveBeenCalledWith('Call failed due to media issue', {
|
|
1979
|
+
file: 'call',
|
|
1980
|
+
method: 'handleRoapError'
|
|
1981
|
+
});
|
|
1982
|
+
|
|
1983
|
+
// Final state should be torn down and cleared for unconnected call
|
|
1984
|
+
expect(call['mediaStateMachine'].state.value).toBe('S_ROAP_TEARDOWN');
|
|
1985
|
+
expect(call['callStateMachine'].state.value).toBe('S_CALL_CLEARED');
|
|
1986
|
+
case 3:
|
|
1987
|
+
case "end":
|
|
1988
|
+
return _context37.stop();
|
|
1989
|
+
}
|
|
1990
|
+
}, _callee37);
|
|
1991
|
+
})));
|
|
1992
|
+
it('state changes during successful incoming call with out of order events', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee38() {
|
|
1993
|
+
var statusPayload, dummyEvent, postMediaSpy, dummyOkEvent, dummyOfferEvent;
|
|
1994
|
+
return _regenerator.default.wrap(function (_context38) {
|
|
1995
|
+
while (1) switch (_context38.prev = _context38.next) {
|
|
1996
|
+
case 0:
|
|
1997
|
+
statusPayload = {
|
|
1998
|
+
statusCode: 200,
|
|
1999
|
+
body: mockStatusBody
|
|
2000
|
+
};
|
|
2001
|
+
dummyEvent = {
|
|
2002
|
+
type: 'E_RECV_CALL_SETUP',
|
|
2003
|
+
data: {
|
|
2004
|
+
seq: 1,
|
|
2005
|
+
messageType: 'OFFER'
|
|
2006
|
+
}
|
|
2007
|
+
};
|
|
2008
|
+
postMediaSpy = jest.spyOn(call, 'postMedia');
|
|
2009
|
+
webex.request.mockReturnValue(statusPayload);
|
|
2010
|
+
call['direction'] = _types3.CallDirection.INBOUND;
|
|
2011
|
+
call.sendCallStateMachineEvt(dummyEvent);
|
|
2012
|
+
expect(call['callStateMachine'].state.value).toBe('S_SEND_CALL_PROGRESS');
|
|
2013
|
+
dummyEvent.type = 'E_SEND_CALL_CONNECT';
|
|
2014
|
+
call.sendCallStateMachineEvt(dummyEvent);
|
|
2015
|
+
expect(call['callStateMachine'].state.value).toBe('S_SEND_CALL_CONNECT');
|
|
2016
|
+
|
|
2017
|
+
/* we should expect to forward the roap offer message to mediaSdk for further processing */
|
|
2018
|
+
dummyEvent.type = 'E_RECV_ROAP_OFFER';
|
|
2019
|
+
call.sendMediaStateMachineEvt(dummyEvent);
|
|
2020
|
+
expect(_testUtil.mediaConnection.roapMessageReceived).toHaveBeenLastCalledWith(dummyEvent.data);
|
|
2021
|
+
|
|
2022
|
+
/* expect sending roap answer to mobius */
|
|
2023
|
+
dummyEvent.type = 'E_SEND_ROAP_ANSWER';
|
|
2024
|
+
dummyEvent.data = {
|
|
2025
|
+
seq: 1,
|
|
2026
|
+
messageType: 'ANSWER'
|
|
2027
|
+
};
|
|
2028
|
+
_context38.next = 1;
|
|
2029
|
+
return call.sendMediaStateMachineEvt(dummyEvent);
|
|
2030
|
+
case 1:
|
|
2031
|
+
expect(postMediaSpy).toBeCalledOnceWith(dummyEvent.data);
|
|
2032
|
+
|
|
2033
|
+
/* we receive roap Offer Request followed by roap Ok from mobius and handle
|
|
2034
|
+
out of order events by buffering and processing them in sequence */
|
|
2035
|
+
dummyOkEvent = {
|
|
2036
|
+
type: 'E_ROAP_OK',
|
|
2037
|
+
data: {
|
|
2038
|
+
received: true,
|
|
2039
|
+
message: {
|
|
2040
|
+
seq: 1,
|
|
2041
|
+
messageType: 'OK'
|
|
2042
|
+
}
|
|
2043
|
+
}
|
|
2044
|
+
};
|
|
2045
|
+
dummyEvent.type = 'E_RECV_ROAP_OFFER_REQUEST';
|
|
2046
|
+
dummyEvent.data = {
|
|
2047
|
+
seq: 2,
|
|
2048
|
+
messageType: 'OFFER_REQUEST'
|
|
2049
|
+
};
|
|
2050
|
+
_context38.next = 2;
|
|
2051
|
+
return call.sendMediaStateMachineEvt(dummyEvent);
|
|
2052
|
+
case 2:
|
|
2053
|
+
expect(call['receivedRoapOKSeq']).toBe(0);
|
|
2054
|
+
expect(_testUtil.mediaConnection.roapMessageReceived).not.toHaveBeenLastCalledWith(dummyEvent.data);
|
|
2055
|
+
call.sendMediaStateMachineEvt(dummyOkEvent);
|
|
2056
|
+
expect(_testUtil.mediaConnection.roapMessageReceived).toHaveBeenNthCalledWith(2, dummyOkEvent.data.message);
|
|
2057
|
+
expect(_testUtil.mediaConnection.roapMessageReceived).toHaveBeenLastCalledWith(dummyEvent.data);
|
|
2058
|
+
dummyOfferEvent = {
|
|
2059
|
+
type: 'E_SEND_ROAP_OFFER',
|
|
2060
|
+
data: {
|
|
2061
|
+
seq: 2,
|
|
2062
|
+
messageType: 'OFFER',
|
|
2063
|
+
sdp: 'sdp'
|
|
2064
|
+
}
|
|
2065
|
+
};
|
|
2066
|
+
expect(call['mediaStateMachine'].state.value).toBe('S_RECV_ROAP_OFFER_REQUEST');
|
|
2067
|
+
call.sendMediaStateMachineEvt(dummyOfferEvent);
|
|
2068
|
+
expect(postMediaSpy).toHaveBeenLastCalledWith(dummyOfferEvent.data);
|
|
2069
|
+
dummyEvent.type = 'E_RECV_ROAP_ANSWER';
|
|
2070
|
+
dummyEvent.data = {
|
|
2071
|
+
seq: 2,
|
|
2072
|
+
messageType: 'ANSWER'
|
|
2073
|
+
};
|
|
2074
|
+
call.sendMediaStateMachineEvt(dummyEvent);
|
|
2075
|
+
expect(_testUtil.mediaConnection.roapMessageReceived).toHaveBeenLastCalledWith(dummyEvent.data);
|
|
2076
|
+
dummyOkEvent.data = {
|
|
2077
|
+
received: false,
|
|
2078
|
+
message: {
|
|
2079
|
+
seq: 2,
|
|
2080
|
+
messageType: 'OK'
|
|
2081
|
+
}
|
|
2082
|
+
};
|
|
2083
|
+
call.sendMediaStateMachineEvt(dummyOkEvent);
|
|
2084
|
+
expect(postMediaSpy).toHaveBeenLastCalledWith(dummyOkEvent.data.message);
|
|
2085
|
+
|
|
2086
|
+
/* With the two roap offer/answer transactions that we simulated earlier
|
|
2087
|
+
we get a total 4 outgoing and 3 incoming roap messages.
|
|
2088
|
+
*/
|
|
2089
|
+
expect(postMediaSpy).toBeCalledTimes(3);
|
|
2090
|
+
expect(_testUtil.mediaConnection.roapMessageReceived).toBeCalledTimes(4);
|
|
2091
|
+
expect(call['callStateMachine'].state.value).toBe('S_CALL_ESTABLISHED');
|
|
2092
|
+
expect(call.isConnected()).toBe(true);
|
|
2093
|
+
dummyEvent.type = 'E_CALL_HOLD';
|
|
2094
|
+
dummyEvent.data = {
|
|
2095
|
+
seq: 3,
|
|
2096
|
+
messageType: 'OFFER'
|
|
2097
|
+
};
|
|
2098
|
+
call.sendCallStateMachineEvt(dummyEvent);
|
|
2099
|
+
dummyEvent.type = 'E_RECV_ROAP_OFFER';
|
|
2100
|
+
_context38.next = 3;
|
|
2101
|
+
return call.sendMediaStateMachineEvt(dummyEvent);
|
|
2102
|
+
case 3:
|
|
2103
|
+
expect(_testUtil.mediaConnection.roapMessageReceived).toHaveBeenLastCalledWith(dummyEvent.data);
|
|
2104
|
+
dummyEvent.type = 'E_SEND_ROAP_ANSWER';
|
|
2105
|
+
dummyEvent.data = {
|
|
2106
|
+
seq: 3,
|
|
2107
|
+
messageType: 'ANSWER'
|
|
2108
|
+
};
|
|
2109
|
+
_context38.next = 4;
|
|
2110
|
+
return call.sendMediaStateMachineEvt(dummyEvent);
|
|
2111
|
+
case 4:
|
|
2112
|
+
expect(postMediaSpy).toHaveBeenLastCalledWith(dummyEvent.data);
|
|
2113
|
+
dummyEvent.type = 'E_RECV_ROAP_OFFER';
|
|
2114
|
+
dummyEvent.data = {
|
|
2115
|
+
seq: 4,
|
|
2116
|
+
messageType: 'OFFER'
|
|
2117
|
+
};
|
|
2118
|
+
call.sendMediaStateMachineEvt(dummyEvent);
|
|
2119
|
+
expect(call['receivedRoapOKSeq']).toBe(2);
|
|
2120
|
+
expect(_testUtil.mediaConnection.roapMessageReceived).not.toHaveBeenLastCalledWith(dummyEvent.data);
|
|
2121
|
+
dummyOkEvent.data = {
|
|
2122
|
+
received: true,
|
|
2123
|
+
message: {
|
|
2124
|
+
seq: 3,
|
|
2125
|
+
messageType: 'OK'
|
|
2126
|
+
}
|
|
2127
|
+
};
|
|
2128
|
+
_context38.next = 5;
|
|
2129
|
+
return call.sendMediaStateMachineEvt(dummyOkEvent);
|
|
2130
|
+
case 5:
|
|
2131
|
+
expect(_testUtil.mediaConnection.roapMessageReceived).toHaveBeenNthCalledWith(6, dummyOkEvent.data.message);
|
|
2132
|
+
expect(_testUtil.mediaConnection.roapMessageReceived).toHaveBeenLastCalledWith(dummyEvent.data);
|
|
2133
|
+
dummyEvent.type = 'E_SEND_ROAP_ANSWER';
|
|
2134
|
+
dummyEvent.data = {
|
|
2135
|
+
seq: 4,
|
|
2136
|
+
messageType: 'ANSWER'
|
|
2137
|
+
};
|
|
2138
|
+
expect(call['mediaStateMachine'].state.value).toBe('S_RECV_ROAP_OFFER');
|
|
2139
|
+
call.sendMediaStateMachineEvt(dummyEvent);
|
|
2140
|
+
expect(postMediaSpy).toHaveBeenLastCalledWith(dummyEvent.data);
|
|
2141
|
+
case 6:
|
|
2142
|
+
case "end":
|
|
2143
|
+
return _context38.stop();
|
|
2144
|
+
}
|
|
2145
|
+
}, _callee38);
|
|
2146
|
+
})));
|
|
2147
|
+
it('successfully handles out of order events when ROAP OK is received while executing outgoingRoapAnswer', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee39() {
|
|
2148
|
+
var mockStatusBody, statusPayload, dummyEvent, postMediaSpy, dummyOkEvent, dummyOfferEvent;
|
|
2149
|
+
return _regenerator.default.wrap(function (_context39) {
|
|
2150
|
+
while (1) switch (_context39.prev = _context39.next) {
|
|
2151
|
+
case 0:
|
|
2152
|
+
mockStatusBody = {
|
|
2153
|
+
device: {
|
|
2154
|
+
deviceId: '123e4567-e89b-12d3-a456-426614174000',
|
|
2155
|
+
correlationId: 'b5476d4c-f48b-475e-b4e2-994e24d14ca2'
|
|
2156
|
+
},
|
|
2157
|
+
callId: 'fcf86aa5-5539-4c9f-8b72-667786ae9b6c'
|
|
2158
|
+
};
|
|
2159
|
+
statusPayload = {
|
|
2160
|
+
statusCode: 200,
|
|
2161
|
+
body: mockStatusBody
|
|
2162
|
+
};
|
|
2163
|
+
dummyEvent = {
|
|
2164
|
+
type: 'E_RECV_CALL_SETUP',
|
|
2165
|
+
data: {
|
|
2166
|
+
seq: 1,
|
|
2167
|
+
messageType: 'OFFER'
|
|
2168
|
+
}
|
|
2169
|
+
};
|
|
2170
|
+
postMediaSpy = jest.spyOn(call, 'postMedia');
|
|
2171
|
+
webex.request.mockReturnValue(statusPayload);
|
|
2172
|
+
call['direction'] = _types3.CallDirection.INBOUND;
|
|
2173
|
+
call.sendCallStateMachineEvt(dummyEvent);
|
|
2174
|
+
expect(call['callStateMachine'].state.value).toBe('S_SEND_CALL_PROGRESS');
|
|
2175
|
+
dummyEvent.type = 'E_SEND_CALL_CONNECT';
|
|
2176
|
+
call.sendCallStateMachineEvt(dummyEvent);
|
|
2177
|
+
expect(call['callStateMachine'].state.value).toBe('S_SEND_CALL_CONNECT');
|
|
2178
|
+
|
|
2179
|
+
/* we should expect to forward the roap offer message to mediaSdk for further processing */
|
|
2180
|
+
dummyEvent.type = 'E_RECV_ROAP_OFFER';
|
|
2181
|
+
call.sendMediaStateMachineEvt(dummyEvent);
|
|
2182
|
+
expect(_testUtil.mediaConnection.roapMessageReceived).toHaveBeenLastCalledWith(dummyEvent.data);
|
|
2183
|
+
|
|
2184
|
+
/* expect sending roap answer to mobius */
|
|
2185
|
+
dummyEvent.type = 'E_SEND_ROAP_ANSWER';
|
|
2186
|
+
dummyEvent.data = {
|
|
2187
|
+
seq: 1,
|
|
2188
|
+
messageType: 'ANSWER'
|
|
2189
|
+
};
|
|
2190
|
+
_context39.next = 1;
|
|
2191
|
+
return call.sendMediaStateMachineEvt(dummyEvent);
|
|
2192
|
+
case 1:
|
|
2193
|
+
expect(postMediaSpy).toBeCalledOnceWith(dummyEvent.data);
|
|
2194
|
+
|
|
2195
|
+
/* we receive roap Offer Request followed by roap Ok from mobius and handle
|
|
2196
|
+
out of order events by buffering and processing them in sequence */
|
|
2197
|
+
dummyOkEvent = {
|
|
2198
|
+
type: 'E_ROAP_OK',
|
|
2199
|
+
data: {
|
|
2200
|
+
received: true,
|
|
2201
|
+
message: {
|
|
2202
|
+
seq: 1,
|
|
2203
|
+
messageType: 'OK'
|
|
2204
|
+
}
|
|
2205
|
+
}
|
|
2206
|
+
};
|
|
2207
|
+
dummyEvent.type = 'E_RECV_ROAP_OFFER_REQUEST';
|
|
2208
|
+
dummyEvent.data = {
|
|
2209
|
+
seq: 2,
|
|
2210
|
+
messageType: 'OFFER_REQUEST'
|
|
2211
|
+
};
|
|
2212
|
+
_context39.next = 2;
|
|
2213
|
+
return call.sendMediaStateMachineEvt(dummyEvent);
|
|
2214
|
+
case 2:
|
|
2215
|
+
expect(call['receivedRoapOKSeq']).toBe(0);
|
|
2216
|
+
expect(_testUtil.mediaConnection.roapMessageReceived).not.toHaveBeenLastCalledWith(dummyEvent.data);
|
|
2217
|
+
call.sendMediaStateMachineEvt(dummyOkEvent);
|
|
2218
|
+
expect(_testUtil.mediaConnection.roapMessageReceived).toHaveBeenNthCalledWith(2, dummyOkEvent.data.message);
|
|
2219
|
+
expect(_testUtil.mediaConnection.roapMessageReceived).toHaveBeenLastCalledWith(dummyEvent.data);
|
|
2220
|
+
dummyOfferEvent = {
|
|
2221
|
+
type: 'E_SEND_ROAP_OFFER',
|
|
2222
|
+
data: {
|
|
2223
|
+
seq: 2,
|
|
2224
|
+
messageType: 'OFFER',
|
|
2225
|
+
sdp: 'sdp'
|
|
2226
|
+
}
|
|
2227
|
+
};
|
|
2228
|
+
expect(call['mediaStateMachine'].state.value).toBe('S_RECV_ROAP_OFFER_REQUEST');
|
|
2229
|
+
call.sendMediaStateMachineEvt(dummyOfferEvent);
|
|
2230
|
+
expect(postMediaSpy).toHaveBeenLastCalledWith(dummyOfferEvent.data);
|
|
2231
|
+
dummyEvent.type = 'E_RECV_ROAP_ANSWER';
|
|
2232
|
+
dummyEvent.data = {
|
|
2233
|
+
seq: 2,
|
|
2234
|
+
messageType: 'ANSWER'
|
|
2235
|
+
};
|
|
2236
|
+
call.sendMediaStateMachineEvt(dummyEvent);
|
|
2237
|
+
expect(_testUtil.mediaConnection.roapMessageReceived).toHaveBeenLastCalledWith(dummyEvent.data);
|
|
2238
|
+
dummyOkEvent.data = {
|
|
2239
|
+
received: false,
|
|
2240
|
+
message: {
|
|
2241
|
+
seq: 2,
|
|
2242
|
+
messageType: 'OK'
|
|
2243
|
+
}
|
|
2244
|
+
};
|
|
2245
|
+
call.sendMediaStateMachineEvt(dummyOkEvent);
|
|
2246
|
+
expect(postMediaSpy).toHaveBeenLastCalledWith(dummyOkEvent.data.message);
|
|
2247
|
+
|
|
2248
|
+
/* With the two roap offer/answer transactions that we simulated earlier
|
|
2249
|
+
we get a total 4 outgoing and 3 incoming roap messages.
|
|
2250
|
+
*/
|
|
2251
|
+
expect(postMediaSpy).toBeCalledTimes(3);
|
|
2252
|
+
expect(_testUtil.mediaConnection.roapMessageReceived).toBeCalledTimes(4);
|
|
2253
|
+
expect(call['callStateMachine'].state.value).toBe('S_CALL_ESTABLISHED');
|
|
2254
|
+
expect(call.isConnected()).toBe(true);
|
|
2255
|
+
dummyEvent.type = 'E_CALL_HOLD';
|
|
2256
|
+
dummyEvent.data = {
|
|
2257
|
+
seq: 3,
|
|
2258
|
+
messageType: 'OFFER'
|
|
2259
|
+
};
|
|
2260
|
+
call.sendCallStateMachineEvt(dummyEvent);
|
|
2261
|
+
dummyEvent.type = 'E_RECV_ROAP_OFFER';
|
|
2262
|
+
_context39.next = 3;
|
|
2263
|
+
return call.sendMediaStateMachineEvt(dummyEvent);
|
|
2264
|
+
case 3:
|
|
2265
|
+
expect(_testUtil.mediaConnection.roapMessageReceived).toHaveBeenLastCalledWith(dummyEvent.data);
|
|
2266
|
+
dummyEvent.type = 'E_SEND_ROAP_ANSWER';
|
|
2267
|
+
dummyEvent.data = {
|
|
2268
|
+
seq: 3,
|
|
2269
|
+
messageType: 'ANSWER'
|
|
2270
|
+
};
|
|
2271
|
+
_context39.next = 4;
|
|
2272
|
+
return call.sendMediaStateMachineEvt(dummyEvent);
|
|
2273
|
+
case 4:
|
|
2274
|
+
expect(postMediaSpy).toHaveBeenLastCalledWith(dummyEvent.data);
|
|
2275
|
+
dummyEvent.type = 'E_RECV_ROAP_OFFER';
|
|
2276
|
+
dummyEvent.data = {
|
|
2277
|
+
seq: 4,
|
|
2278
|
+
messageType: 'OFFER'
|
|
2279
|
+
};
|
|
2280
|
+
dummyOkEvent.data = {
|
|
2281
|
+
received: true,
|
|
2282
|
+
message: {
|
|
2283
|
+
seq: 3,
|
|
2284
|
+
messageType: 'OK'
|
|
2285
|
+
}
|
|
2286
|
+
};
|
|
2287
|
+
call.sendMediaStateMachineEvt(dummyEvent);
|
|
2288
|
+
_context39.next = 5;
|
|
2289
|
+
return call.sendMediaStateMachineEvt(dummyOkEvent);
|
|
2290
|
+
case 5:
|
|
2291
|
+
expect(call['receivedRoapOKSeq']).toBe(3);
|
|
2292
|
+
expect(_testUtil.mediaConnection.roapMessageReceived).toHaveBeenNthCalledWith(6, dummyOkEvent.data.message);
|
|
2293
|
+
expect(_testUtil.mediaConnection.roapMessageReceived).toHaveBeenLastCalledWith(dummyEvent.data);
|
|
2294
|
+
dummyEvent.type = 'E_SEND_ROAP_ANSWER';
|
|
2295
|
+
dummyEvent.data = {
|
|
2296
|
+
seq: 4,
|
|
2297
|
+
messageType: 'ANSWER'
|
|
2298
|
+
};
|
|
2299
|
+
expect(call['mediaStateMachine'].state.value).toBe('S_RECV_ROAP_OFFER');
|
|
2300
|
+
call.sendMediaStateMachineEvt(dummyEvent);
|
|
2301
|
+
expect(postMediaSpy).toHaveBeenLastCalledWith(dummyEvent.data);
|
|
2302
|
+
case 6:
|
|
2303
|
+
case "end":
|
|
2304
|
+
return _context39.stop();
|
|
2305
|
+
}
|
|
2306
|
+
}, _callee39);
|
|
2307
|
+
})));
|
|
2308
|
+
it('handle hold event successfully when media received after progress but before connect', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee40() {
|
|
2309
|
+
var statusPayload, dummyEvent, postMediaSpy, infoSpy, dummyOkEvent;
|
|
2310
|
+
return _regenerator.default.wrap(function (_context40) {
|
|
2311
|
+
while (1) switch (_context40.prev = _context40.next) {
|
|
2312
|
+
case 0:
|
|
2313
|
+
statusPayload = {
|
|
2314
|
+
statusCode: 200,
|
|
2315
|
+
body: mockStatusBody
|
|
2316
|
+
};
|
|
2317
|
+
dummyEvent = {
|
|
2318
|
+
type: 'E_SEND_CALL_SETUP',
|
|
2319
|
+
data: {
|
|
2320
|
+
seq: 1,
|
|
2321
|
+
messageType: 'OFFER',
|
|
2322
|
+
sdp: 'sdp'
|
|
2323
|
+
}
|
|
2324
|
+
};
|
|
2325
|
+
postMediaSpy = jest.spyOn(call, 'postMedia');
|
|
2326
|
+
infoSpy = jest.spyOn(_Logger.default, 'info');
|
|
2327
|
+
webex.request.mockReturnValue(statusPayload);
|
|
2328
|
+
call.sendCallStateMachineEvt(dummyEvent);
|
|
2329
|
+
expect(call['callStateMachine'].state.value).toBe('S_SEND_CALL_SETUP');
|
|
2330
|
+
dummyEvent.type = 'E_SEND_ROAP_OFFER';
|
|
2331
|
+
call.sendMediaStateMachineEvt(dummyEvent);
|
|
2332
|
+
expect(postMediaSpy).toHaveBeenLastCalledWith(dummyEvent.data);
|
|
2333
|
+
dummyEvent.type = 'E_RECV_CALL_PROGRESS';
|
|
2334
|
+
dummyEvent.data = undefined;
|
|
2335
|
+
call.sendCallStateMachineEvt(dummyEvent);
|
|
2336
|
+
expect(call['callStateMachine'].state.value).toBe('S_RECV_CALL_PROGRESS');
|
|
2337
|
+
dummyEvent.type = 'E_RECV_ROAP_ANSWER';
|
|
2338
|
+
dummyEvent.data = {
|
|
2339
|
+
seq: 1,
|
|
2340
|
+
messageType: 'ANSWER',
|
|
2341
|
+
sdp: 'sdp'
|
|
2342
|
+
};
|
|
2343
|
+
call.sendMediaStateMachineEvt(dummyEvent);
|
|
2344
|
+
expect(_testUtil.mediaConnection.roapMessageReceived).toHaveBeenLastCalledWith(dummyEvent.data);
|
|
2345
|
+
expect(call['mediaNegotiationCompleted']).toBe(false);
|
|
2346
|
+
dummyOkEvent = {
|
|
2347
|
+
type: 'E_ROAP_OK',
|
|
2348
|
+
data: {
|
|
2349
|
+
received: false,
|
|
2350
|
+
message: {
|
|
2351
|
+
seq: 1,
|
|
2352
|
+
messageType: 'OK'
|
|
2353
|
+
}
|
|
2354
|
+
}
|
|
2355
|
+
};
|
|
2356
|
+
call.sendMediaStateMachineEvt(dummyOkEvent);
|
|
2357
|
+
expect(call['mediaNegotiationCompleted']).toBe(true);
|
|
2358
|
+
expect(postMediaSpy).toHaveBeenLastCalledWith(dummyOkEvent.data.message);
|
|
2359
|
+
dummyEvent.type = 'E_RECV_CALL_CONNECT';
|
|
2360
|
+
dummyEvent.data = undefined;
|
|
2361
|
+
call.sendCallStateMachineEvt(dummyEvent);
|
|
2362
|
+
|
|
2363
|
+
/* Call will move to connect state then immediately move to established state as
|
|
2364
|
+
media negotiation is already completed before connect was received
|
|
2365
|
+
*/
|
|
2366
|
+
expect(call['callStateMachine'].state.value).toBe('S_CALL_ESTABLISHED');
|
|
2367
|
+
expect(call.isConnected()).toBe(true);
|
|
2368
|
+
dummyEvent.type = 'E_CALL_HOLD';
|
|
2369
|
+
dummyEvent.data = undefined;
|
|
2370
|
+
call.sendCallStateMachineEvt(dummyEvent);
|
|
2371
|
+
expect(call['callStateMachine'].state.value).toBe('S_CALL_HOLD');
|
|
2372
|
+
expect(infoSpy).toHaveBeenLastCalledWith("".concat(_constants2.METHOD_START_MESSAGE, " with: ").concat(call.getCorrelationId()), {
|
|
2373
|
+
file: 'call',
|
|
2374
|
+
method: 'handleCallHold'
|
|
2375
|
+
});
|
|
2376
|
+
case 1:
|
|
2377
|
+
case "end":
|
|
2378
|
+
return _context40.stop();
|
|
2379
|
+
}
|
|
2380
|
+
}, _callee40);
|
|
2381
|
+
})));
|
|
2382
|
+
it('emits DISCONNECT before mobius delete request is invoked', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee41() {
|
|
2383
|
+
var emitSpy, deleteSpy;
|
|
2384
|
+
return _regenerator.default.wrap(function (_context41) {
|
|
2385
|
+
while (1) switch (_context41.prev = _context41.next) {
|
|
2386
|
+
case 0:
|
|
2387
|
+
emitSpy = jest.spyOn(call, 'emit');
|
|
2388
|
+
deleteSpy = jest.spyOn(call, 'delete').mockResolvedValue({
|
|
2389
|
+
statusCode: 200
|
|
2390
|
+
});
|
|
2391
|
+
call.sendCallStateMachineEvt({
|
|
2392
|
+
type: 'E_RECV_CALL_DISCONNECT'
|
|
2393
|
+
});
|
|
2394
|
+
_context41.next = 1;
|
|
2395
|
+
return (0, _testUtil.flushPromises)(1);
|
|
2396
|
+
case 1:
|
|
2397
|
+
expect(emitSpy).toHaveBeenCalledWith(_types2.CALL_EVENT_KEYS.DISCONNECT, call.getCorrelationId());
|
|
2398
|
+
expect(deleteSpy).toHaveBeenCalled();
|
|
2399
|
+
expect(emitSpy.mock.invocationCallOrder[0]).toBeLessThan(deleteSpy.mock.invocationCallOrder[0]);
|
|
2400
|
+
case 2:
|
|
2401
|
+
case "end":
|
|
2402
|
+
return _context41.stop();
|
|
2403
|
+
}
|
|
2404
|
+
}, _callee41);
|
|
2405
|
+
})));
|
|
2406
|
+
describe('Call event timers tests', function () {
|
|
2407
|
+
var callManager;
|
|
2408
|
+
beforeEach(function () {
|
|
2409
|
+
jest.useFakeTimers();
|
|
2410
|
+
callManager = (0, _callManager.getCallManager)(webex, defaultServiceIndicator);
|
|
2411
|
+
});
|
|
2412
|
+
afterEach(function () {
|
|
2413
|
+
jest.clearAllTimers();
|
|
2414
|
+
});
|
|
2415
|
+
it('times out if the next event is not received - 60 seconds timeout', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee42() {
|
|
2416
|
+
var statusPayload, dummyEvent, logSpy, emitSpy, deleteSpy, dummyOkEvent;
|
|
2417
|
+
return _regenerator.default.wrap(function (_context42) {
|
|
2418
|
+
while (1) switch (_context42.prev = _context42.next) {
|
|
2419
|
+
case 0:
|
|
2420
|
+
statusPayload = {
|
|
2421
|
+
statusCode: 200,
|
|
2422
|
+
body: mockStatusBody
|
|
2423
|
+
};
|
|
2424
|
+
dummyEvent = {
|
|
2425
|
+
type: 'E_SEND_CALL_SETUP',
|
|
2426
|
+
data: undefined
|
|
2427
|
+
};
|
|
2428
|
+
logSpy = jest.spyOn(_Logger.default, 'warn');
|
|
2429
|
+
emitSpy = jest.spyOn(call, 'emit');
|
|
2430
|
+
deleteSpy = jest.spyOn(call, 'delete');
|
|
2431
|
+
callManager.callCollection = {};
|
|
2432
|
+
webex.request.mockReturnValue(statusPayload);
|
|
2433
|
+
|
|
2434
|
+
// handleOutgoingCallSetup is asynchronous
|
|
2435
|
+
_context42.next = 1;
|
|
2436
|
+
return call.sendCallStateMachineEvt(dummyEvent);
|
|
2437
|
+
case 1:
|
|
2438
|
+
expect(call['callStateMachine'].state.value).toBe('S_SEND_CALL_SETUP');
|
|
2439
|
+
dummyEvent.type = 'E_RECV_CALL_PROGRESS';
|
|
2440
|
+
call.sendCallStateMachineEvt(dummyEvent);
|
|
2441
|
+
expect(call['callStateMachine'].state.value).toBe('S_RECV_CALL_PROGRESS');
|
|
2442
|
+
|
|
2443
|
+
// Media setup for the call
|
|
2444
|
+
dummyEvent.type = 'E_SEND_ROAP_OFFER';
|
|
2445
|
+
call.sendMediaStateMachineEvt(dummyEvent);
|
|
2446
|
+
dummyEvent.data = {
|
|
2447
|
+
seq: 1,
|
|
2448
|
+
messageType: 'OFFER',
|
|
2449
|
+
sdp: 'sdp'
|
|
2450
|
+
};
|
|
2451
|
+
call.sendMediaStateMachineEvt(dummyEvent);
|
|
2452
|
+
dummyEvent.type = 'E_RECV_ROAP_ANSWER';
|
|
2453
|
+
call.sendMediaStateMachineEvt(dummyEvent);
|
|
2454
|
+
dummyOkEvent = {
|
|
2455
|
+
type: 'E_ROAP_OK',
|
|
2456
|
+
data: {
|
|
2457
|
+
received: false,
|
|
2458
|
+
message: {
|
|
2459
|
+
seq: 1,
|
|
2460
|
+
messageType: 'OK'
|
|
2461
|
+
}
|
|
2462
|
+
}
|
|
2463
|
+
};
|
|
2464
|
+
call.sendMediaStateMachineEvt(dummyOkEvent);
|
|
2465
|
+
dummyEvent.type = 'E_RECV_ROAP_OFFER_REQUEST';
|
|
2466
|
+
call.sendMediaStateMachineEvt(dummyEvent);
|
|
2467
|
+
dummyEvent.type = 'E_SEND_ROAP_OFFER';
|
|
2468
|
+
call.sendMediaStateMachineEvt(dummyEvent);
|
|
2469
|
+
dummyEvent.type = 'E_RECV_ROAP_ANSWER';
|
|
2470
|
+
logSpy.mockClear();
|
|
2471
|
+
jest.advanceTimersByTime(60000);
|
|
2472
|
+
expect(logSpy.mock.calls[0][0]).toBe('Call timed out');
|
|
2473
|
+
expect(emitSpy).toHaveBeenCalledWith(_types2.CALL_EVENT_KEYS.DISCONNECT, call.getCorrelationId());
|
|
2474
|
+
expect(deleteSpy).toHaveBeenCalledTimes(1);
|
|
2475
|
+
expect(callManager.callCollection).toStrictEqual({});
|
|
2476
|
+
case 2:
|
|
2477
|
+
case "end":
|
|
2478
|
+
return _context42.stop();
|
|
2479
|
+
}
|
|
2480
|
+
}, _callee42);
|
|
2481
|
+
})));
|
|
2482
|
+
it('times out if the next event is not received - 10 seconds timeout', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee43() {
|
|
2483
|
+
var statusPayload, dummyEvent, call, emitSpy, deleteSpy, logSpy;
|
|
2484
|
+
return _regenerator.default.wrap(function (_context43) {
|
|
2485
|
+
while (1) switch (_context43.prev = _context43.next) {
|
|
2486
|
+
case 0:
|
|
2487
|
+
statusPayload = {
|
|
2488
|
+
statusCode: 200,
|
|
2489
|
+
body: mockStatusBody
|
|
2490
|
+
};
|
|
2491
|
+
dummyEvent = {
|
|
2492
|
+
type: 'E_SEND_CALL_SETUP',
|
|
2493
|
+
data: undefined
|
|
2494
|
+
};
|
|
2495
|
+
callManager.callCollection = {};
|
|
2496
|
+
call = callManager.createCall(dest, _types3.CallDirection.OUTBOUND, deviceId, mockLineId);
|
|
2497
|
+
emitSpy = jest.spyOn(call, 'emit');
|
|
2498
|
+
deleteSpy = jest.spyOn(call, 'delete');
|
|
2499
|
+
logSpy = jest.spyOn(_Logger.default, 'warn');
|
|
2500
|
+
webex.request.mockReturnValue(statusPayload);
|
|
2501
|
+
expect((0, _keys.default)(callManager.callCollection)[0]).toBe(call.getCorrelationId());
|
|
2502
|
+
|
|
2503
|
+
// handleOutgoingCallSetup is asynchronous
|
|
2504
|
+
_context43.next = 1;
|
|
2505
|
+
return call.sendCallStateMachineEvt(dummyEvent);
|
|
2506
|
+
case 1:
|
|
2507
|
+
expect(call['callStateMachine'].state.value).toBe('S_SEND_CALL_SETUP');
|
|
2508
|
+
logSpy.mockClear();
|
|
2509
|
+
jest.advanceTimersByTime(10000);
|
|
2510
|
+
expect(logSpy.mock.calls[0][0]).toBe('Call timed out');
|
|
2511
|
+
expect(emitSpy).toHaveBeenCalledWith(_types2.CALL_EVENT_KEYS.DISCONNECT, call.getCorrelationId());
|
|
2512
|
+
expect(deleteSpy).toHaveBeenCalledTimes(1);
|
|
2513
|
+
expect(callManager.callCollection).toStrictEqual({});
|
|
2514
|
+
case 2:
|
|
2515
|
+
case "end":
|
|
2516
|
+
return _context43.stop();
|
|
2517
|
+
}
|
|
2518
|
+
}, _callee43);
|
|
2519
|
+
})));
|
|
2520
|
+
});
|
|
2521
|
+
});
|
|
2522
|
+
describe('Supplementary Services tests', function () {
|
|
2523
|
+
var deviceId = '55dfb53f-bed2-36da-8e85-cee7f02aa68e';
|
|
2524
|
+
var dest = {
|
|
2525
|
+
type: _types3.CallType.URI,
|
|
2526
|
+
address: 'tel:5003'
|
|
2527
|
+
};
|
|
2528
|
+
var mockResponseBody = {
|
|
2529
|
+
device: {
|
|
2530
|
+
deviceId: '123e4567-e89b-12d3-a456-426614174000',
|
|
2531
|
+
correlationId: 'b5476d4c-f48b-475e-b4e2-994e24d14ca2'
|
|
2532
|
+
},
|
|
2533
|
+
callId: 'fcf86aa5-5539-4c9f-8b72-667786ae9b6c'
|
|
2534
|
+
};
|
|
2535
|
+
var call;
|
|
2536
|
+
beforeEach(function () {
|
|
2537
|
+
/* Since we are not actually testing from the start of a call , so it's good to set the below
|
|
2538
|
+
* parameters manually
|
|
2539
|
+
*/
|
|
2540
|
+
|
|
2541
|
+
call = new _call.Call(activeUrl, webex, _types3.CallDirection.OUTBOUND, deviceId, mockLineId, function () {
|
|
2542
|
+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
2543
|
+
var dummy = 10;
|
|
2544
|
+
}, defaultServiceIndicator, dest);
|
|
2545
|
+
call['connected'] = true;
|
|
2546
|
+
call['earlyMedia'] = false;
|
|
2547
|
+
|
|
2548
|
+
/* Also supplementary services will start always from Call_Established state */
|
|
2549
|
+
call['callStateMachine'].state.value = 'S_CALL_ESTABLISHED';
|
|
2550
|
+
call.removeAllListeners(_types2.CALL_EVENT_KEYS.CALL_ERROR);
|
|
2551
|
+
jest.clearAllTimers();
|
|
2552
|
+
jest.useFakeTimers();
|
|
2553
|
+
});
|
|
2554
|
+
describe('Call hold-resume tests', function () {
|
|
2555
|
+
var mockHeldEvent = {
|
|
2556
|
+
eventType: 'callState',
|
|
2557
|
+
eventData: {
|
|
2558
|
+
callState: 'HELD'
|
|
2559
|
+
}
|
|
2560
|
+
};
|
|
2561
|
+
var mockResumeEvent = {
|
|
2562
|
+
eventType: 'callState',
|
|
2563
|
+
eventData: {
|
|
2564
|
+
callState: 'CONNECTED'
|
|
2565
|
+
}
|
|
2566
|
+
};
|
|
2567
|
+
var dummyEvent = {
|
|
2568
|
+
type: 'E_SEND_ROAP_OFFER',
|
|
2569
|
+
data: {
|
|
2570
|
+
seq: 1,
|
|
2571
|
+
message: {},
|
|
2572
|
+
type: 'OFFER'
|
|
2573
|
+
}
|
|
2574
|
+
};
|
|
2575
|
+
beforeEach(function () {
|
|
2576
|
+
call.removeAllListeners();
|
|
2577
|
+
});
|
|
2578
|
+
it('Handle successful Call hold case without delayed http response', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee45() {
|
|
2579
|
+
var responsePayload, warnSpy, roapEvent;
|
|
2580
|
+
return _regenerator.default.wrap(function (_context45) {
|
|
2581
|
+
while (1) switch (_context45.prev = _context45.next) {
|
|
2582
|
+
case 0:
|
|
2583
|
+
expect.assertions(7);
|
|
2584
|
+
responsePayload = {
|
|
2585
|
+
statusCode: 200,
|
|
2586
|
+
body: mockResponseBody
|
|
2587
|
+
};
|
|
2588
|
+
jest.spyOn(webex, 'request').mockResolvedValue(responsePayload);
|
|
2589
|
+
jest.spyOn(global, 'setTimeout');
|
|
2590
|
+
jest.spyOn(global, 'clearTimeout');
|
|
2591
|
+
warnSpy = jest.spyOn(_Logger.default, 'warn');
|
|
2592
|
+
call['held'] = false;
|
|
2593
|
+
call.on(_types2.CALL_EVENT_KEYS.HELD, /*#__PURE__*/function () {
|
|
2594
|
+
var _ref45 = (0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee44(correlationId) {
|
|
2595
|
+
return _regenerator.default.wrap(function (_context44) {
|
|
2596
|
+
while (1) switch (_context44.prev = _context44.next) {
|
|
2597
|
+
case 0:
|
|
2598
|
+
expect(correlationId).toStrictEqual(call.getCorrelationId());
|
|
2599
|
+
case 1:
|
|
2600
|
+
case "end":
|
|
2601
|
+
return _context44.stop();
|
|
2602
|
+
}
|
|
2603
|
+
}, _callee44);
|
|
2604
|
+
}));
|
|
2605
|
+
return function (_x) {
|
|
2606
|
+
return _ref45.apply(this, arguments);
|
|
2607
|
+
};
|
|
2608
|
+
}());
|
|
2609
|
+
_context45.next = 1;
|
|
2610
|
+
return call.doHoldResume();
|
|
2611
|
+
case 1:
|
|
2612
|
+
_context45.next = 2;
|
|
2613
|
+
return (0, _testUtil.flushPromises)(2);
|
|
2614
|
+
case 2:
|
|
2615
|
+
expect(setTimeout).toHaveBeenCalledTimes(1);
|
|
2616
|
+
call.handleMidCallEvent(mockHeldEvent);
|
|
2617
|
+
|
|
2618
|
+
/* At this point, the Call State should be S_CALL_HOLD
|
|
2619
|
+
*/
|
|
2620
|
+
expect(call['callStateMachine'].state.value).toStrictEqual('S_CALL_HOLD');
|
|
2621
|
+
|
|
2622
|
+
/* We simulate the media Connection events manually here as we are not really testing with actual
|
|
2623
|
+
* media.
|
|
2624
|
+
*/
|
|
2625
|
+
call['handleIncomingRoapOffer']({}, dummyEvent);
|
|
2626
|
+
roapEvent = JSON.parse((0, _stringify.default)(dummyEvent));
|
|
2627
|
+
roapEvent.data.type = 'ANSWER';
|
|
2628
|
+
_context45.next = 3;
|
|
2629
|
+
return call['handleOutgoingRoapAnswer']({}, dummyEvent);
|
|
2630
|
+
case 3:
|
|
2631
|
+
roapEvent.data.type = 'OK';
|
|
2632
|
+
_context45.next = 4;
|
|
2633
|
+
return call['handleRoapEstablished']({}, dummyEvent);
|
|
2634
|
+
case 4:
|
|
2635
|
+
expect(clearTimeout).toHaveBeenCalledTimes(1);
|
|
2636
|
+
/* isHeld flag should be set and an Hold event should be emitted */
|
|
2637
|
+
expect(call.isHeld()).toStrictEqual(true);
|
|
2638
|
+
/* We should return back to call established state */
|
|
2639
|
+
expect(call['callStateMachine'].state.value).toStrictEqual('S_CALL_ESTABLISHED');
|
|
2640
|
+
jest.advanceTimersByTime(12000);
|
|
2641
|
+
expect(warnSpy).not.toHaveBeenCalledWith('Hold response timed out', {
|
|
2642
|
+
file: 'call',
|
|
2643
|
+
method: 'handleCallHold'
|
|
2644
|
+
});
|
|
2645
|
+
case 5:
|
|
2646
|
+
case "end":
|
|
2647
|
+
return _context45.stop();
|
|
2648
|
+
}
|
|
2649
|
+
}, _callee45);
|
|
2650
|
+
})));
|
|
2651
|
+
it('Handle successful Call hold case with delayed http response', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee47() {
|
|
2652
|
+
var responsePayload, warnSpy, roapEvent;
|
|
2653
|
+
return _regenerator.default.wrap(function (_context47) {
|
|
2654
|
+
while (1) switch (_context47.prev = _context47.next) {
|
|
2655
|
+
case 0:
|
|
2656
|
+
expect.assertions(8);
|
|
2657
|
+
responsePayload = {
|
|
2658
|
+
statusCode: 200,
|
|
2659
|
+
body: mockResponseBody
|
|
2660
|
+
};
|
|
2661
|
+
jest.spyOn(webex, 'request').mockResolvedValue(responsePayload);
|
|
2662
|
+
jest.spyOn(global, 'setTimeout');
|
|
2663
|
+
jest.spyOn(global, 'clearTimeout');
|
|
2664
|
+
warnSpy = jest.spyOn(_Logger.default, 'warn');
|
|
2665
|
+
call['held'] = false;
|
|
2666
|
+
call.on(_types2.CALL_EVENT_KEYS.HELD, /*#__PURE__*/function () {
|
|
2667
|
+
var _ref47 = (0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee46(correlationId) {
|
|
2668
|
+
return _regenerator.default.wrap(function (_context46) {
|
|
2669
|
+
while (1) switch (_context46.prev = _context46.next) {
|
|
2670
|
+
case 0:
|
|
2671
|
+
expect(correlationId).toStrictEqual(call.getCorrelationId());
|
|
2672
|
+
case 1:
|
|
2673
|
+
case "end":
|
|
2674
|
+
return _context46.stop();
|
|
2675
|
+
}
|
|
2676
|
+
}, _callee46);
|
|
2677
|
+
}));
|
|
2678
|
+
return function (_x2) {
|
|
2679
|
+
return _ref47.apply(this, arguments);
|
|
2680
|
+
};
|
|
2681
|
+
}());
|
|
2682
|
+
call.doHoldResume();
|
|
2683
|
+
_context47.next = 1;
|
|
2684
|
+
return _promise.default.resolve();
|
|
2685
|
+
case 1:
|
|
2686
|
+
_context47.next = 2;
|
|
2687
|
+
return _promise.default.resolve();
|
|
2688
|
+
case 2:
|
|
2689
|
+
expect(setTimeout).not.toHaveBeenCalled();
|
|
2690
|
+
call.handleMidCallEvent(mockHeldEvent);
|
|
2691
|
+
|
|
2692
|
+
/* At this point, the Call State should be S_CALL_HOLD
|
|
2693
|
+
*/
|
|
2694
|
+
expect(call['callStateMachine'].state.value).toStrictEqual('S_CALL_HOLD');
|
|
2695
|
+
expect(call.isHeld()).toBe(true);
|
|
2696
|
+
/* We simulate the media Connection events manually here as we are not really testing with actual
|
|
2697
|
+
* media.
|
|
2698
|
+
*/
|
|
2699
|
+
call['handleIncomingRoapOffer']({}, dummyEvent);
|
|
2700
|
+
roapEvent = JSON.parse((0, _stringify.default)(dummyEvent));
|
|
2701
|
+
roapEvent.data.type = 'ANSWER';
|
|
2702
|
+
_context47.next = 3;
|
|
2703
|
+
return call['handleOutgoingRoapAnswer']({}, dummyEvent);
|
|
2704
|
+
case 3:
|
|
2705
|
+
roapEvent.data.type = 'OK';
|
|
2706
|
+
_context47.next = 4;
|
|
2707
|
+
return call['handleRoapEstablished']({}, dummyEvent);
|
|
2708
|
+
case 4:
|
|
2709
|
+
expect(clearTimeout).not.toHaveBeenCalled();
|
|
2710
|
+
/* isHeld flag should be set and an Hold event should be emitted */
|
|
2711
|
+
expect(call.isHeld()).toStrictEqual(true);
|
|
2712
|
+
/* We should return back to call established state */
|
|
2713
|
+
expect(call['callStateMachine'].state.value).toStrictEqual('S_CALL_ESTABLISHED');
|
|
2714
|
+
jest.advanceTimersByTime(12000);
|
|
2715
|
+
expect(warnSpy).not.toHaveBeenCalledWith('Hold response timed out', {
|
|
2716
|
+
file: 'call',
|
|
2717
|
+
method: 'handleCallHold'
|
|
2718
|
+
});
|
|
2719
|
+
case 5:
|
|
2720
|
+
case "end":
|
|
2721
|
+
return _context47.stop();
|
|
2722
|
+
}
|
|
2723
|
+
}, _callee47);
|
|
2724
|
+
})));
|
|
2725
|
+
it('Handle failure Call Hold case during signalling', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee49() {
|
|
2726
|
+
var responsePayload;
|
|
2727
|
+
return _regenerator.default.wrap(function (_context49) {
|
|
2728
|
+
while (1) switch (_context49.prev = _context49.next) {
|
|
2729
|
+
case 0:
|
|
2730
|
+
expect.assertions(4);
|
|
2731
|
+
responsePayload = {
|
|
2732
|
+
statusCode: 503,
|
|
2733
|
+
body: mockResponseBody
|
|
2734
|
+
};
|
|
2735
|
+
jest.spyOn(webex, 'request').mockRejectedValue(responsePayload);
|
|
2736
|
+
call['held'] = false;
|
|
2737
|
+
call.on(_types2.CALL_EVENT_KEYS.HOLD_ERROR, /*#__PURE__*/function () {
|
|
2738
|
+
var _ref49 = (0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee48(errObj) {
|
|
2739
|
+
return _regenerator.default.wrap(function (_context48) {
|
|
2740
|
+
while (1) switch (_context48.prev = _context48.next) {
|
|
2741
|
+
case 0:
|
|
2742
|
+
expect(errObj.type).toStrictEqual(_types.ERROR_TYPE.SERVICE_UNAVAILABLE);
|
|
2743
|
+
expect(errObj.message).toStrictEqual('An unknown error occurred. Wait a moment and try again.');
|
|
2744
|
+
case 1:
|
|
2745
|
+
case "end":
|
|
2746
|
+
return _context48.stop();
|
|
2747
|
+
}
|
|
2748
|
+
}, _callee48);
|
|
2749
|
+
}));
|
|
2750
|
+
return function (_x3) {
|
|
2751
|
+
return _ref49.apply(this, arguments);
|
|
2752
|
+
};
|
|
2753
|
+
}());
|
|
2754
|
+
_context49.next = 1;
|
|
2755
|
+
return call.doHoldResume();
|
|
2756
|
+
case 1:
|
|
2757
|
+
_context49.next = 2;
|
|
2758
|
+
return (0, _testUtil.flushPromises)(2);
|
|
2759
|
+
case 2:
|
|
2760
|
+
expect(call.isHeld()).toStrictEqual(false);
|
|
2761
|
+
|
|
2762
|
+
/* At this point , the Call State should transition to S_CALL_ESTABLISHED
|
|
2763
|
+
*/
|
|
2764
|
+
expect(call['callStateMachine'].state.value).toStrictEqual('S_CALL_ESTABLISHED');
|
|
2765
|
+
case 3:
|
|
2766
|
+
case "end":
|
|
2767
|
+
return _context49.stop();
|
|
2768
|
+
}
|
|
2769
|
+
}, _callee49);
|
|
2770
|
+
})));
|
|
2771
|
+
it('Handle failure Call Hold case during offer/answer exchange', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee51() {
|
|
2772
|
+
var responsePayload, rejectPayload, roapEvent;
|
|
2773
|
+
return _regenerator.default.wrap(function (_context51) {
|
|
2774
|
+
while (1) switch (_context51.prev = _context51.next) {
|
|
2775
|
+
case 0:
|
|
2776
|
+
expect.assertions(5);
|
|
2777
|
+
responsePayload = {
|
|
2778
|
+
statusCode: 200,
|
|
2779
|
+
body: mockResponseBody
|
|
2780
|
+
};
|
|
2781
|
+
rejectPayload = {
|
|
2782
|
+
statusCode: 503,
|
|
2783
|
+
body: mockResponseBody
|
|
2784
|
+
};
|
|
2785
|
+
jest.spyOn(webex, 'request').mockResolvedValueOnce(responsePayload).mockRejectedValueOnce(rejectPayload);
|
|
2786
|
+
call['held'] = false;
|
|
2787
|
+
call.on(_types2.CALL_EVENT_KEYS.HOLD_ERROR, /*#__PURE__*/function () {
|
|
2788
|
+
var _ref51 = (0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee50(errObj) {
|
|
2789
|
+
return _regenerator.default.wrap(function (_context50) {
|
|
2790
|
+
while (1) switch (_context50.prev = _context50.next) {
|
|
2791
|
+
case 0:
|
|
2792
|
+
expect(errObj.type).toStrictEqual(_types.ERROR_TYPE.SERVICE_UNAVAILABLE);
|
|
2793
|
+
expect(errObj.message).toStrictEqual('An unknown error occurred. Wait a moment and try again.');
|
|
2794
|
+
case 1:
|
|
2795
|
+
case "end":
|
|
2796
|
+
return _context50.stop();
|
|
2797
|
+
}
|
|
2798
|
+
}, _callee50);
|
|
2799
|
+
}));
|
|
2800
|
+
return function (_x4) {
|
|
2801
|
+
return _ref51.apply(this, arguments);
|
|
2802
|
+
};
|
|
2803
|
+
}());
|
|
2804
|
+
call.doHoldResume();
|
|
2805
|
+
_context51.next = 1;
|
|
2806
|
+
return (0, _testUtil.flushPromises)(2);
|
|
2807
|
+
case 1:
|
|
2808
|
+
/* the Call State should transition to S_CALL_ESTABLISHED
|
|
2809
|
+
*/
|
|
2810
|
+
expect(call['callStateMachine'].state.value).toStrictEqual('S_CALL_HOLD');
|
|
2811
|
+
call['handleIncomingRoapOffer']({}, dummyEvent);
|
|
2812
|
+
roapEvent = JSON.parse((0, _stringify.default)(dummyEvent));
|
|
2813
|
+
/* We are intentionally failing the ROAP ANSWER */
|
|
2814
|
+
roapEvent.data.type = 'ANSWER';
|
|
2815
|
+
_context51.next = 2;
|
|
2816
|
+
return call['handleOutgoingRoapAnswer']({}, dummyEvent);
|
|
2817
|
+
case 2:
|
|
2818
|
+
expect(call.isHeld()).toStrictEqual(false);
|
|
2819
|
+
/* We should return back to call established state */
|
|
2820
|
+
expect(call['callStateMachine'].state.value).toStrictEqual('S_CALL_ESTABLISHED');
|
|
2821
|
+
case 3:
|
|
2822
|
+
case "end":
|
|
2823
|
+
return _context51.stop();
|
|
2824
|
+
}
|
|
2825
|
+
}, _callee51);
|
|
2826
|
+
})));
|
|
2827
|
+
it('Handle failure Call Hold case during roap ok out', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee52() {
|
|
2828
|
+
var responsePayload, warnSpy, roapEvent;
|
|
2829
|
+
return _regenerator.default.wrap(function (_context52) {
|
|
2830
|
+
while (1) switch (_context52.prev = _context52.next) {
|
|
2831
|
+
case 0:
|
|
2832
|
+
responsePayload = {
|
|
2833
|
+
statusCode: 200,
|
|
2834
|
+
body: mockResponseBody
|
|
2835
|
+
};
|
|
2836
|
+
jest.spyOn(webex, 'request').mockResolvedValue(responsePayload);
|
|
2837
|
+
jest.spyOn(global, 'setTimeout');
|
|
2838
|
+
jest.spyOn(global, 'clearTimeout');
|
|
2839
|
+
warnSpy = jest.spyOn(_Logger.default, 'warn');
|
|
2840
|
+
call['held'] = false;
|
|
2841
|
+
_context52.next = 1;
|
|
2842
|
+
return call.doHoldResume();
|
|
2843
|
+
case 1:
|
|
2844
|
+
_context52.next = 2;
|
|
2845
|
+
return (0, _testUtil.flushPromises)(2);
|
|
2846
|
+
case 2:
|
|
2847
|
+
expect(setTimeout).toHaveBeenCalledTimes(1);
|
|
2848
|
+
|
|
2849
|
+
/* At this point, the Call State should be S_CALL_HOLD
|
|
2850
|
+
*/
|
|
2851
|
+
expect(call['callStateMachine'].state.value).toStrictEqual('S_CALL_HOLD');
|
|
2852
|
+
|
|
2853
|
+
/* We simulate the media Connection events manually here as we are not really testing with actual
|
|
2854
|
+
* media.
|
|
2855
|
+
*/
|
|
2856
|
+
call['handleIncomingRoapOffer']({}, dummyEvent);
|
|
2857
|
+
roapEvent = JSON.parse((0, _stringify.default)(dummyEvent));
|
|
2858
|
+
roapEvent.data.type = 'ANSWER';
|
|
2859
|
+
_context52.next = 3;
|
|
2860
|
+
return call['handleOutgoingRoapAnswer']({}, dummyEvent);
|
|
2861
|
+
case 3:
|
|
2862
|
+
jest.spyOn(webex, 'request').mockRejectedValue({
|
|
2863
|
+
statusCode: 403
|
|
2864
|
+
});
|
|
2865
|
+
roapEvent.data.type = 'OK';
|
|
2866
|
+
_context52.next = 4;
|
|
2867
|
+
return call['handleRoapEstablished']({}, dummyEvent);
|
|
2868
|
+
case 4:
|
|
2869
|
+
/* this is for coverage */
|
|
2870
|
+
call['callStateMachine'].state.value = 'S_CALL_HOLD';
|
|
2871
|
+
_context52.next = 5;
|
|
2872
|
+
return call['handleRoapEstablished']({}, dummyEvent);
|
|
2873
|
+
case 5:
|
|
2874
|
+
expect(call.isHeld()).toStrictEqual(false);
|
|
2875
|
+
/* We should return back to call established state */
|
|
2876
|
+
expect(call['callStateMachine'].state.value).toStrictEqual('S_CALL_ESTABLISHED');
|
|
2877
|
+
expect(warnSpy).toHaveBeenCalledWith('Failed to process MediaOk request', {
|
|
2878
|
+
file: 'call',
|
|
2879
|
+
method: 'handleRoapEstablished'
|
|
2880
|
+
});
|
|
2881
|
+
case 6:
|
|
2882
|
+
case "end":
|
|
2883
|
+
return _context52.stop();
|
|
2884
|
+
}
|
|
2885
|
+
}, _callee52);
|
|
2886
|
+
})));
|
|
2887
|
+
it('Handle failure Call resume case during roap ok out', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee53() {
|
|
2888
|
+
var responsePayload, warnSpy, roapEvent;
|
|
2889
|
+
return _regenerator.default.wrap(function (_context53) {
|
|
2890
|
+
while (1) switch (_context53.prev = _context53.next) {
|
|
2891
|
+
case 0:
|
|
2892
|
+
responsePayload = {
|
|
2893
|
+
statusCode: 200,
|
|
2894
|
+
body: mockResponseBody
|
|
2895
|
+
};
|
|
2896
|
+
jest.spyOn(webex, 'request').mockResolvedValue(responsePayload);
|
|
2897
|
+
jest.spyOn(global, 'setTimeout');
|
|
2898
|
+
jest.spyOn(global, 'clearTimeout');
|
|
2899
|
+
warnSpy = jest.spyOn(_Logger.default, 'warn');
|
|
2900
|
+
call['held'] = true;
|
|
2901
|
+
_context53.next = 1;
|
|
2902
|
+
return call.doHoldResume();
|
|
2903
|
+
case 1:
|
|
2904
|
+
_context53.next = 2;
|
|
2905
|
+
return (0, _testUtil.flushPromises)(2);
|
|
2906
|
+
case 2:
|
|
2907
|
+
expect(setTimeout).toHaveBeenCalledTimes(1);
|
|
2908
|
+
|
|
2909
|
+
/* At this point, the Call State should be S_CALL_RESUME
|
|
2910
|
+
*/
|
|
2911
|
+
expect(call['callStateMachine'].state.value).toStrictEqual('S_CALL_RESUME');
|
|
2912
|
+
|
|
2913
|
+
/* We simulate the media Connection events manually here as we are not really testing with actual
|
|
2914
|
+
* media.
|
|
2915
|
+
*/
|
|
2916
|
+
call['handleIncomingRoapOffer']({}, dummyEvent);
|
|
2917
|
+
roapEvent = JSON.parse((0, _stringify.default)(dummyEvent));
|
|
2918
|
+
roapEvent.data.type = 'ANSWER';
|
|
2919
|
+
_context53.next = 3;
|
|
2920
|
+
return call['handleOutgoingRoapAnswer']({}, dummyEvent);
|
|
2921
|
+
case 3:
|
|
2922
|
+
jest.spyOn(webex, 'request').mockRejectedValue({
|
|
2923
|
+
statusCode: 403
|
|
2924
|
+
});
|
|
2925
|
+
roapEvent.data.type = 'OK';
|
|
2926
|
+
_context53.next = 4;
|
|
2927
|
+
return call['handleRoapEstablished']({}, dummyEvent);
|
|
2928
|
+
case 4:
|
|
2929
|
+
expect(call.isHeld()).toStrictEqual(true);
|
|
2930
|
+
/* We should return back to call established state */
|
|
2931
|
+
expect(call['callStateMachine'].state.value).toStrictEqual('S_CALL_ESTABLISHED');
|
|
2932
|
+
expect(warnSpy).toHaveBeenCalledWith('Failed to process MediaOk request', {
|
|
2933
|
+
file: 'call',
|
|
2934
|
+
method: 'handleRoapEstablished'
|
|
2935
|
+
});
|
|
2936
|
+
case 5:
|
|
2937
|
+
case "end":
|
|
2938
|
+
return _context53.stop();
|
|
2939
|
+
}
|
|
2940
|
+
}, _callee53);
|
|
2941
|
+
})));
|
|
2942
|
+
it('Handle Call hold case where successful Held response does not come', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee55() {
|
|
2943
|
+
var responsePayload, roapEvent;
|
|
2944
|
+
return _regenerator.default.wrap(function (_context55) {
|
|
2945
|
+
while (1) switch (_context55.prev = _context55.next) {
|
|
2946
|
+
case 0:
|
|
2947
|
+
expect.assertions(5);
|
|
2948
|
+
responsePayload = {
|
|
2949
|
+
statusCode: 200,
|
|
2950
|
+
body: mockResponseBody
|
|
2951
|
+
};
|
|
2952
|
+
jest.spyOn(webex, 'request').mockResolvedValue(responsePayload);
|
|
2953
|
+
call['held'] = false;
|
|
2954
|
+
call.on(_types2.CALL_EVENT_KEYS.HOLD_ERROR, /*#__PURE__*/function () {
|
|
2955
|
+
var _ref55 = (0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee54(errObj) {
|
|
2956
|
+
return _regenerator.default.wrap(function (_context54) {
|
|
2957
|
+
while (1) switch (_context54.prev = _context54.next) {
|
|
2958
|
+
case 0:
|
|
2959
|
+
expect(errObj.type).toStrictEqual(_types.ERROR_TYPE.TIMEOUT);
|
|
2960
|
+
expect(errObj.message).toStrictEqual('An error occurred while placing the call on hold. Wait a moment and try again.');
|
|
2961
|
+
case 1:
|
|
2962
|
+
case "end":
|
|
2963
|
+
return _context54.stop();
|
|
2964
|
+
}
|
|
2965
|
+
}, _callee54);
|
|
2966
|
+
}));
|
|
2967
|
+
return function (_x5) {
|
|
2968
|
+
return _ref55.apply(this, arguments);
|
|
2969
|
+
};
|
|
2970
|
+
}());
|
|
2971
|
+
jest.runAllTimers();
|
|
2972
|
+
call.doHoldResume();
|
|
2973
|
+
_context55.next = 1;
|
|
2974
|
+
return (0, _testUtil.flushPromises)(2);
|
|
2975
|
+
case 1:
|
|
2976
|
+
/* At this point, the Call State should be S_CALL_HOLD
|
|
2977
|
+
*/
|
|
2978
|
+
expect(call['callStateMachine'].state.value).toStrictEqual('S_CALL_HOLD');
|
|
2979
|
+
|
|
2980
|
+
/* We simulate the media Connection events manually here as we are not really testing with actual
|
|
2981
|
+
* media.
|
|
2982
|
+
*/
|
|
2983
|
+
call['handleIncomingRoapOffer']({}, dummyEvent);
|
|
2984
|
+
roapEvent = JSON.parse((0, _stringify.default)(dummyEvent));
|
|
2985
|
+
roapEvent.data.type = 'ANSWER';
|
|
2986
|
+
_context55.next = 2;
|
|
2987
|
+
return call['handleOutgoingRoapAnswer']({}, dummyEvent);
|
|
2988
|
+
case 2:
|
|
2989
|
+
roapEvent.data.type = 'OK';
|
|
2990
|
+
_context55.next = 3;
|
|
2991
|
+
return call['handleRoapEstablished']({}, dummyEvent);
|
|
2992
|
+
case 3:
|
|
2993
|
+
/* Advancing timer by 12 seconds so that it gets timed out */
|
|
2994
|
+
jest.advanceTimersByTime(12000);
|
|
2995
|
+
|
|
2996
|
+
/* isHeld flag should be unset */
|
|
2997
|
+
expect(call.isHeld()).toStrictEqual(false);
|
|
2998
|
+
/* We should return back to call established state */
|
|
2999
|
+
expect(call['callStateMachine'].state.value).toStrictEqual('S_CALL_ESTABLISHED');
|
|
3000
|
+
case 4:
|
|
3001
|
+
case "end":
|
|
3002
|
+
return _context55.stop();
|
|
3003
|
+
}
|
|
3004
|
+
}, _callee55);
|
|
3005
|
+
})));
|
|
3006
|
+
it('Handle successful Call Resume case without delayed http response', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee57() {
|
|
3007
|
+
var responsePayload, warnSpy, roapEvent;
|
|
3008
|
+
return _regenerator.default.wrap(function (_context57) {
|
|
3009
|
+
while (1) switch (_context57.prev = _context57.next) {
|
|
3010
|
+
case 0:
|
|
3011
|
+
expect.assertions(7);
|
|
3012
|
+
responsePayload = {
|
|
3013
|
+
statusCode: 200,
|
|
3014
|
+
body: mockResponseBody
|
|
3015
|
+
};
|
|
3016
|
+
jest.spyOn(webex, 'request').mockResolvedValue(responsePayload);
|
|
3017
|
+
jest.spyOn(global, 'setTimeout');
|
|
3018
|
+
jest.spyOn(global, 'clearTimeout');
|
|
3019
|
+
warnSpy = jest.spyOn(_Logger.default, 'warn');
|
|
3020
|
+
call['held'] = true;
|
|
3021
|
+
call.on(_types2.CALL_EVENT_KEYS.RESUMED, /*#__PURE__*/function () {
|
|
3022
|
+
var _ref57 = (0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee56(correlationId) {
|
|
3023
|
+
return _regenerator.default.wrap(function (_context56) {
|
|
3024
|
+
while (1) switch (_context56.prev = _context56.next) {
|
|
3025
|
+
case 0:
|
|
3026
|
+
expect(correlationId).toStrictEqual(call.getCorrelationId());
|
|
3027
|
+
case 1:
|
|
3028
|
+
case "end":
|
|
3029
|
+
return _context56.stop();
|
|
3030
|
+
}
|
|
3031
|
+
}, _callee56);
|
|
3032
|
+
}));
|
|
3033
|
+
return function (_x6) {
|
|
3034
|
+
return _ref57.apply(this, arguments);
|
|
3035
|
+
};
|
|
3036
|
+
}());
|
|
3037
|
+
_context57.next = 1;
|
|
3038
|
+
return call.doHoldResume();
|
|
3039
|
+
case 1:
|
|
3040
|
+
_context57.next = 2;
|
|
3041
|
+
return (0, _testUtil.flushPromises)(2);
|
|
3042
|
+
case 2:
|
|
3043
|
+
expect(setTimeout).toHaveBeenCalledTimes(1);
|
|
3044
|
+
call.handleMidCallEvent(mockResumeEvent);
|
|
3045
|
+
|
|
3046
|
+
/* At this point ,the Call State should be S_CALL_RESUME
|
|
3047
|
+
*/
|
|
3048
|
+
expect(call['callStateMachine'].state.value).toStrictEqual('S_CALL_RESUME');
|
|
3049
|
+
|
|
3050
|
+
/* We simulate the media Connection events manually here as we are not really testing with actual
|
|
3051
|
+
* media.
|
|
3052
|
+
*/
|
|
3053
|
+
call['handleIncomingRoapOffer']({}, dummyEvent);
|
|
3054
|
+
roapEvent = JSON.parse((0, _stringify.default)(dummyEvent));
|
|
3055
|
+
roapEvent.data.type = 'ANSWER';
|
|
3056
|
+
_context57.next = 3;
|
|
3057
|
+
return call['handleOutgoingRoapAnswer']({}, dummyEvent);
|
|
3058
|
+
case 3:
|
|
3059
|
+
roapEvent.data.type = 'OK';
|
|
3060
|
+
_context57.next = 4;
|
|
3061
|
+
return call['handleRoapEstablished']({}, dummyEvent);
|
|
3062
|
+
case 4:
|
|
3063
|
+
expect(clearTimeout).toHaveBeenCalledTimes(1);
|
|
3064
|
+
/* isHeld flag should not be set and an Resume event should be emitted */
|
|
3065
|
+
expect(call.isHeld()).toStrictEqual(false);
|
|
3066
|
+
/* We should return back to call established state */
|
|
3067
|
+
expect(call['callStateMachine'].state.value).toStrictEqual('S_CALL_ESTABLISHED');
|
|
3068
|
+
jest.advanceTimersByTime(12000);
|
|
3069
|
+
expect(warnSpy).not.toHaveBeenCalledWith('Resume response timed out', {
|
|
3070
|
+
file: 'call',
|
|
3071
|
+
method: 'handleCallResume'
|
|
3072
|
+
});
|
|
3073
|
+
case 5:
|
|
3074
|
+
case "end":
|
|
3075
|
+
return _context57.stop();
|
|
3076
|
+
}
|
|
3077
|
+
}, _callee57);
|
|
3078
|
+
})));
|
|
3079
|
+
it('Handle successful Call Resume case with delayed http response', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee59() {
|
|
3080
|
+
var responsePayload, warnSpy, roapEvent;
|
|
3081
|
+
return _regenerator.default.wrap(function (_context59) {
|
|
3082
|
+
while (1) switch (_context59.prev = _context59.next) {
|
|
3083
|
+
case 0:
|
|
3084
|
+
expect.assertions(7);
|
|
3085
|
+
responsePayload = {
|
|
3086
|
+
statusCode: 200,
|
|
3087
|
+
body: mockResponseBody
|
|
3088
|
+
};
|
|
3089
|
+
jest.spyOn(webex, 'request').mockResolvedValue(responsePayload);
|
|
3090
|
+
jest.spyOn(global, 'setTimeout');
|
|
3091
|
+
jest.spyOn(global, 'clearTimeout');
|
|
3092
|
+
warnSpy = jest.spyOn(_Logger.default, 'warn');
|
|
3093
|
+
call['held'] = true;
|
|
3094
|
+
call.on(_types2.CALL_EVENT_KEYS.RESUMED, /*#__PURE__*/function () {
|
|
3095
|
+
var _ref59 = (0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee58(correlationId) {
|
|
3096
|
+
return _regenerator.default.wrap(function (_context58) {
|
|
3097
|
+
while (1) switch (_context58.prev = _context58.next) {
|
|
3098
|
+
case 0:
|
|
3099
|
+
expect(correlationId).toStrictEqual(call.getCorrelationId());
|
|
3100
|
+
case 1:
|
|
3101
|
+
case "end":
|
|
3102
|
+
return _context58.stop();
|
|
3103
|
+
}
|
|
3104
|
+
}, _callee58);
|
|
3105
|
+
}));
|
|
3106
|
+
return function (_x7) {
|
|
3107
|
+
return _ref59.apply(this, arguments);
|
|
3108
|
+
};
|
|
3109
|
+
}());
|
|
3110
|
+
call.doHoldResume();
|
|
3111
|
+
_context59.next = 1;
|
|
3112
|
+
return _promise.default.resolve();
|
|
3113
|
+
case 1:
|
|
3114
|
+
_context59.next = 2;
|
|
3115
|
+
return _promise.default.resolve();
|
|
3116
|
+
case 2:
|
|
3117
|
+
expect(setTimeout).not.toHaveBeenCalled();
|
|
3118
|
+
call.handleMidCallEvent(mockResumeEvent);
|
|
3119
|
+
|
|
3120
|
+
/* At this point ,the Call State should be S_CALL_RESUME
|
|
3121
|
+
*/
|
|
3122
|
+
expect(call['callStateMachine'].state.value).toStrictEqual('S_CALL_RESUME');
|
|
3123
|
+
|
|
3124
|
+
/* We simulate the media Connection events manually here as we are not really testing with actual
|
|
3125
|
+
* media.
|
|
3126
|
+
*/
|
|
3127
|
+
call['handleIncomingRoapOffer']({}, dummyEvent);
|
|
3128
|
+
roapEvent = JSON.parse((0, _stringify.default)(dummyEvent));
|
|
3129
|
+
roapEvent.data.type = 'ANSWER';
|
|
3130
|
+
_context59.next = 3;
|
|
3131
|
+
return call['handleOutgoingRoapAnswer']({}, dummyEvent);
|
|
3132
|
+
case 3:
|
|
3133
|
+
roapEvent.data.type = 'OK';
|
|
3134
|
+
_context59.next = 4;
|
|
3135
|
+
return call['handleRoapEstablished']({}, dummyEvent);
|
|
3136
|
+
case 4:
|
|
3137
|
+
expect(clearTimeout).not.toHaveBeenCalled();
|
|
3138
|
+
/* isHeld flag should not be set and an Resume event should be emitted */
|
|
3139
|
+
expect(call.isHeld()).toStrictEqual(false);
|
|
3140
|
+
/* We should return back to call established state */
|
|
3141
|
+
expect(call['callStateMachine'].state.value).toStrictEqual('S_CALL_ESTABLISHED');
|
|
3142
|
+
jest.advanceTimersByTime(12000);
|
|
3143
|
+
expect(warnSpy).not.toHaveBeenCalledWith('Resume response timed out', {
|
|
3144
|
+
file: 'call',
|
|
3145
|
+
method: 'handleCallResume'
|
|
3146
|
+
});
|
|
3147
|
+
case 5:
|
|
3148
|
+
case "end":
|
|
3149
|
+
return _context59.stop();
|
|
3150
|
+
}
|
|
3151
|
+
}, _callee59);
|
|
3152
|
+
})));
|
|
3153
|
+
it('Handle failure Call Resume case during signalling', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee61() {
|
|
3154
|
+
var responsePayload;
|
|
3155
|
+
return _regenerator.default.wrap(function (_context61) {
|
|
3156
|
+
while (1) switch (_context61.prev = _context61.next) {
|
|
3157
|
+
case 0:
|
|
3158
|
+
expect.assertions(4);
|
|
3159
|
+
responsePayload = {
|
|
3160
|
+
statusCode: 503,
|
|
3161
|
+
body: mockResponseBody
|
|
3162
|
+
};
|
|
3163
|
+
jest.spyOn(webex, 'request').mockRejectedValue(responsePayload);
|
|
3164
|
+
call['held'] = true;
|
|
3165
|
+
call.on(_types2.CALL_EVENT_KEYS.RESUME_ERROR, /*#__PURE__*/function () {
|
|
3166
|
+
var _ref61 = (0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee60(errObj) {
|
|
3167
|
+
return _regenerator.default.wrap(function (_context60) {
|
|
3168
|
+
while (1) switch (_context60.prev = _context60.next) {
|
|
3169
|
+
case 0:
|
|
3170
|
+
expect(errObj.type).toStrictEqual(_types.ERROR_TYPE.SERVICE_UNAVAILABLE);
|
|
3171
|
+
expect(errObj.message).toStrictEqual('An unknown error occurred. Wait a moment and try again.');
|
|
3172
|
+
case 1:
|
|
3173
|
+
case "end":
|
|
3174
|
+
return _context60.stop();
|
|
3175
|
+
}
|
|
3176
|
+
}, _callee60);
|
|
3177
|
+
}));
|
|
3178
|
+
return function (_x8) {
|
|
3179
|
+
return _ref61.apply(this, arguments);
|
|
3180
|
+
};
|
|
3181
|
+
}());
|
|
3182
|
+
_context61.next = 1;
|
|
3183
|
+
return call.doHoldResume();
|
|
3184
|
+
case 1:
|
|
3185
|
+
_context61.next = 2;
|
|
3186
|
+
return (0, _testUtil.flushPromises)(2);
|
|
3187
|
+
case 2:
|
|
3188
|
+
expect(call.isHeld()).toStrictEqual(true);
|
|
3189
|
+
|
|
3190
|
+
/* At this point , the Call State should transition to S_CALL_ESTABLISHED
|
|
3191
|
+
*/
|
|
3192
|
+
|
|
3193
|
+
expect(call['callStateMachine'].state.value).toStrictEqual('S_CALL_ESTABLISHED');
|
|
3194
|
+
case 3:
|
|
3195
|
+
case "end":
|
|
3196
|
+
return _context61.stop();
|
|
3197
|
+
}
|
|
3198
|
+
}, _callee61);
|
|
3199
|
+
})));
|
|
3200
|
+
it('Handle failure Call Resume case during offer/answer exchange', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee63() {
|
|
3201
|
+
var responsePayload, rejectPayload, roapEvent;
|
|
3202
|
+
return _regenerator.default.wrap(function (_context63) {
|
|
3203
|
+
while (1) switch (_context63.prev = _context63.next) {
|
|
3204
|
+
case 0:
|
|
3205
|
+
expect.assertions(5);
|
|
3206
|
+
responsePayload = {
|
|
3207
|
+
statusCode: 200,
|
|
3208
|
+
body: mockResponseBody
|
|
3209
|
+
};
|
|
3210
|
+
rejectPayload = {
|
|
3211
|
+
statusCode: 503,
|
|
3212
|
+
body: mockResponseBody
|
|
3213
|
+
};
|
|
3214
|
+
jest.spyOn(webex, 'request').mockResolvedValueOnce(responsePayload).mockRejectedValueOnce(rejectPayload);
|
|
3215
|
+
call['held'] = true;
|
|
3216
|
+
call.on(_types2.CALL_EVENT_KEYS.RESUME_ERROR, /*#__PURE__*/function () {
|
|
3217
|
+
var _ref63 = (0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee62(errObj) {
|
|
3218
|
+
return _regenerator.default.wrap(function (_context62) {
|
|
3219
|
+
while (1) switch (_context62.prev = _context62.next) {
|
|
3220
|
+
case 0:
|
|
3221
|
+
expect(errObj.type).toStrictEqual(_types.ERROR_TYPE.SERVICE_UNAVAILABLE);
|
|
3222
|
+
expect(errObj.message).toStrictEqual('An unknown error occurred. Wait a moment and try again.');
|
|
3223
|
+
case 1:
|
|
3224
|
+
case "end":
|
|
3225
|
+
return _context62.stop();
|
|
3226
|
+
}
|
|
3227
|
+
}, _callee62);
|
|
3228
|
+
}));
|
|
3229
|
+
return function (_x9) {
|
|
3230
|
+
return _ref63.apply(this, arguments);
|
|
3231
|
+
};
|
|
3232
|
+
}());
|
|
3233
|
+
call.doHoldResume();
|
|
3234
|
+
_context63.next = 1;
|
|
3235
|
+
return (0, _testUtil.flushPromises)(2);
|
|
3236
|
+
case 1:
|
|
3237
|
+
/* At this point , the Call State should transition to S_CALL_ESTABLISHED
|
|
3238
|
+
*/
|
|
3239
|
+
|
|
3240
|
+
expect(call['callStateMachine'].state.value).toStrictEqual('S_CALL_RESUME');
|
|
3241
|
+
call['handleIncomingRoapOffer']({}, dummyEvent);
|
|
3242
|
+
roapEvent = JSON.parse((0, _stringify.default)(dummyEvent));
|
|
3243
|
+
/* We are intentionally failing the ROAP ANSWER */
|
|
3244
|
+
roapEvent.data.type = 'ANSWER';
|
|
3245
|
+
_context63.next = 2;
|
|
3246
|
+
return call['handleOutgoingRoapAnswer']({}, dummyEvent);
|
|
3247
|
+
case 2:
|
|
3248
|
+
expect(call.isHeld()).toStrictEqual(true);
|
|
3249
|
+
/* We should return back to call established state */
|
|
3250
|
+
expect(call['callStateMachine'].state.value).toStrictEqual('S_CALL_ESTABLISHED');
|
|
3251
|
+
case 3:
|
|
3252
|
+
case "end":
|
|
3253
|
+
return _context63.stop();
|
|
3254
|
+
}
|
|
3255
|
+
}, _callee63);
|
|
3256
|
+
})));
|
|
3257
|
+
it('Handle Call resume case where successful response does not come', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee65() {
|
|
3258
|
+
var responsePayload, roapEvent;
|
|
3259
|
+
return _regenerator.default.wrap(function (_context65) {
|
|
3260
|
+
while (1) switch (_context65.prev = _context65.next) {
|
|
3261
|
+
case 0:
|
|
3262
|
+
expect.assertions(5);
|
|
3263
|
+
responsePayload = {
|
|
3264
|
+
statusCode: 200,
|
|
3265
|
+
body: mockResponseBody
|
|
3266
|
+
};
|
|
3267
|
+
jest.spyOn(webex, 'request').mockResolvedValue(responsePayload);
|
|
3268
|
+
call['held'] = true;
|
|
3269
|
+
call.on(_types2.CALL_EVENT_KEYS.RESUME_ERROR, /*#__PURE__*/function () {
|
|
3270
|
+
var _ref65 = (0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee64(errObj) {
|
|
3271
|
+
return _regenerator.default.wrap(function (_context64) {
|
|
3272
|
+
while (1) switch (_context64.prev = _context64.next) {
|
|
3273
|
+
case 0:
|
|
3274
|
+
expect(errObj.type).toStrictEqual(_types.ERROR_TYPE.TIMEOUT);
|
|
3275
|
+
expect(errObj.message).toStrictEqual('An error occurred while resuming the call. Wait a moment and try again.');
|
|
3276
|
+
case 1:
|
|
3277
|
+
case "end":
|
|
3278
|
+
return _context64.stop();
|
|
3279
|
+
}
|
|
3280
|
+
}, _callee64);
|
|
3281
|
+
}));
|
|
3282
|
+
return function (_x0) {
|
|
3283
|
+
return _ref65.apply(this, arguments);
|
|
3284
|
+
};
|
|
3285
|
+
}());
|
|
3286
|
+
call.doHoldResume();
|
|
3287
|
+
_context65.next = 1;
|
|
3288
|
+
return (0, _testUtil.flushPromises)(2);
|
|
3289
|
+
case 1:
|
|
3290
|
+
/* At this point ,the Call State should be S_CALL_RESUME
|
|
3291
|
+
*/
|
|
3292
|
+
expect(call['callStateMachine'].state.value).toStrictEqual('S_CALL_RESUME');
|
|
3293
|
+
|
|
3294
|
+
/* We simulate the media Connection events manually here as we are not really testing with actual
|
|
3295
|
+
* media.
|
|
3296
|
+
*/
|
|
3297
|
+
call['handleIncomingRoapOffer']({}, dummyEvent);
|
|
3298
|
+
roapEvent = JSON.parse((0, _stringify.default)(dummyEvent));
|
|
3299
|
+
roapEvent.data.type = 'ANSWER';
|
|
3300
|
+
_context65.next = 2;
|
|
3301
|
+
return call['handleOutgoingRoapAnswer']({}, dummyEvent);
|
|
3302
|
+
case 2:
|
|
3303
|
+
roapEvent.data.type = 'OK';
|
|
3304
|
+
_context65.next = 3;
|
|
3305
|
+
return call['handleRoapEstablished']({}, dummyEvent);
|
|
3306
|
+
case 3:
|
|
3307
|
+
/* Advancing timer by 12 seconds so that it gets timed out */
|
|
3308
|
+
jest.advanceTimersByTime(12000);
|
|
3309
|
+
expect(call.isHeld()).toStrictEqual(true);
|
|
3310
|
+
/* We should return back to call established state */
|
|
3311
|
+
expect(call['callStateMachine'].state.value).toStrictEqual('S_CALL_ESTABLISHED');
|
|
3312
|
+
case 4:
|
|
3313
|
+
case "end":
|
|
3314
|
+
return _context65.stop();
|
|
3315
|
+
}
|
|
3316
|
+
}, _callee65);
|
|
3317
|
+
})));
|
|
3318
|
+
});
|
|
3319
|
+
describe('Call transfer tests', function () {
|
|
3320
|
+
var transfereeNumber = '5004';
|
|
3321
|
+
var transfereeDest = {
|
|
3322
|
+
type: _types3.CallType.URI,
|
|
3323
|
+
address: "tel:".concat(transfereeNumber)
|
|
3324
|
+
};
|
|
3325
|
+
var secondCall;
|
|
3326
|
+
|
|
3327
|
+
/* A spy on handleCallErrors to check whether it is being invoked or not depending on tests */
|
|
3328
|
+
var handleErrorSpy = jest.spyOn(Utils, 'handleCallErrors');
|
|
3329
|
+
var uploadLogsSpy = jest.spyOn(Utils, 'uploadLogs');
|
|
3330
|
+
var transferLoggingContext = {
|
|
3331
|
+
file: 'call',
|
|
3332
|
+
method: 'completeTransfer'
|
|
3333
|
+
};
|
|
3334
|
+
beforeEach(function () {
|
|
3335
|
+
/* We will put the first call on hold to perform the transfer */
|
|
3336
|
+
call['held'] = true;
|
|
3337
|
+
|
|
3338
|
+
/* We will create a second call just like the first call, but call is active, i.e, not held */
|
|
3339
|
+
secondCall = new _call.Call(activeUrl, webex, _types3.CallDirection.OUTBOUND, deviceId, mockLineId, function () {
|
|
3340
|
+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
3341
|
+
var dummy = 10;
|
|
3342
|
+
}, defaultServiceIndicator, transfereeDest);
|
|
3343
|
+
secondCall['connected'] = true;
|
|
3344
|
+
secondCall['earlyMedia'] = false;
|
|
3345
|
+
secondCall['callStateMachine'].state.value = 'S_CALL_ESTABLISHED';
|
|
3346
|
+
secondCall.removeAllListeners(_types2.CALL_EVENT_KEYS.CALL_ERROR);
|
|
3347
|
+
secondCall['held'] = false;
|
|
3348
|
+
});
|
|
3349
|
+
it('Handle successful consult transfer case ', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee68() {
|
|
3350
|
+
var responsePayload, requestSpy, warnSpy, infoSpy, metricSpy;
|
|
3351
|
+
return _regenerator.default.wrap(function (_context68) {
|
|
3352
|
+
while (1) switch (_context68.prev = _context68.next) {
|
|
3353
|
+
case 0:
|
|
3354
|
+
expect.assertions(12); // Updated to match actual assertion count
|
|
3355
|
+
responsePayload = {
|
|
3356
|
+
statusCode: 200,
|
|
3357
|
+
body: mockResponseBody
|
|
3358
|
+
};
|
|
3359
|
+
requestSpy = jest.spyOn(webex, 'request').mockResolvedValue(responsePayload);
|
|
3360
|
+
warnSpy = jest.spyOn(_Logger.default, 'warn');
|
|
3361
|
+
infoSpy = jest.spyOn(_Logger.default, 'info');
|
|
3362
|
+
metricSpy = jest.spyOn(call['metricManager'], 'submitCallMetric');
|
|
3363
|
+
call.on(_types2.CALL_EVENT_KEYS.DISCONNECT, /*#__PURE__*/function () {
|
|
3364
|
+
var _ref67 = (0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee66(correlationId) {
|
|
3365
|
+
return _regenerator.default.wrap(function (_context66) {
|
|
3366
|
+
while (1) switch (_context66.prev = _context66.next) {
|
|
3367
|
+
case 0:
|
|
3368
|
+
expect(correlationId).toStrictEqual(call.getCorrelationId());
|
|
3369
|
+
case 1:
|
|
3370
|
+
case "end":
|
|
3371
|
+
return _context66.stop();
|
|
3372
|
+
}
|
|
3373
|
+
}, _callee66);
|
|
3374
|
+
}));
|
|
3375
|
+
return function (_x1) {
|
|
3376
|
+
return _ref67.apply(this, arguments);
|
|
3377
|
+
};
|
|
3378
|
+
}());
|
|
3379
|
+
secondCall.on(_types2.CALL_EVENT_KEYS.DISCONNECT, /*#__PURE__*/function () {
|
|
3380
|
+
var _ref68 = (0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee67(correlationId) {
|
|
3381
|
+
return _regenerator.default.wrap(function (_context67) {
|
|
3382
|
+
while (1) switch (_context67.prev = _context67.next) {
|
|
3383
|
+
case 0:
|
|
3384
|
+
expect(correlationId).toStrictEqual(secondCall.getCorrelationId());
|
|
3385
|
+
case 1:
|
|
3386
|
+
case "end":
|
|
3387
|
+
return _context67.stop();
|
|
3388
|
+
}
|
|
3389
|
+
}, _callee67);
|
|
3390
|
+
}));
|
|
3391
|
+
return function (_x10) {
|
|
3392
|
+
return _ref68.apply(this, arguments);
|
|
3393
|
+
};
|
|
3394
|
+
}());
|
|
3395
|
+
_context68.next = 1;
|
|
3396
|
+
return call.completeTransfer(_types5.TransferType.CONSULT, secondCall.getCallId(), undefined);
|
|
3397
|
+
case 1:
|
|
3398
|
+
_context68.next = 2;
|
|
3399
|
+
return (0, _testUtil.flushPromises)(2);
|
|
3400
|
+
case 2:
|
|
3401
|
+
expect(requestSpy).toBeCalled();
|
|
3402
|
+
expect(metricSpy).toHaveBeenCalledWith(_types4.METRIC_EVENT.CALL, _types4.TRANSFER_ACTION.CONSULT, _types4.METRIC_TYPE.BEHAVIORAL, call.getCallId(), call.getCorrelationId(), undefined);
|
|
3403
|
+
call.sendCallStateMachineEvt({
|
|
3404
|
+
type: 'E_RECV_CALL_DISCONNECT'
|
|
3405
|
+
});
|
|
3406
|
+
secondCall.sendCallStateMachineEvt({
|
|
3407
|
+
type: 'E_RECV_CALL_DISCONNECT'
|
|
3408
|
+
});
|
|
3409
|
+
|
|
3410
|
+
/* We should return back to S_RECV_CALL_DISCONNECT state for both the calls */
|
|
3411
|
+
expect(call['callStateMachine'].state.value).toStrictEqual('S_RECV_CALL_DISCONNECT');
|
|
3412
|
+
expect(secondCall['callStateMachine'].state.value).toStrictEqual('S_RECV_CALL_DISCONNECT');
|
|
3413
|
+
expect(handleErrorSpy).not.toBeCalled();
|
|
3414
|
+
expect(uploadLogsSpy).not.toBeCalled();
|
|
3415
|
+
expect(infoSpy).toHaveBeenCalledWith("Initiating Consult transfer between : ".concat(call.getCallId(), " and ").concat(secondCall.getCallId()), transferLoggingContext);
|
|
3416
|
+
expect(warnSpy).not.toHaveBeenCalledWith("Consult Transfer failed for correlationId ".concat(call.getCorrelationId()), transferLoggingContext);
|
|
3417
|
+
expect(infoSpy).toHaveBeenCalledWith("Initiating Consult transfer between : ".concat(call.getCallId(), " and ").concat(secondCall.getCallId()), transferLoggingContext);
|
|
3418
|
+
expect(warnSpy).not.toHaveBeenCalledWith("Consult Transfer failed for correlationId ".concat(call.getCorrelationId()), transferLoggingContext);
|
|
3419
|
+
case 3:
|
|
3420
|
+
case "end":
|
|
3421
|
+
return _context68.stop();
|
|
3422
|
+
}
|
|
3423
|
+
}, _callee68);
|
|
3424
|
+
})));
|
|
3425
|
+
it('Handle successful blind transfer case ', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee70() {
|
|
3426
|
+
var responsePayload, requestSpy, warnSpy, infoSpy, metricSpy;
|
|
3427
|
+
return _regenerator.default.wrap(function (_context70) {
|
|
3428
|
+
while (1) switch (_context70.prev = _context70.next) {
|
|
3429
|
+
case 0:
|
|
3430
|
+
expect.assertions(10); // Updated to match actual assertion count
|
|
3431
|
+
responsePayload = {
|
|
3432
|
+
statusCode: 200,
|
|
3433
|
+
body: mockResponseBody
|
|
3434
|
+
};
|
|
3435
|
+
requestSpy = jest.spyOn(webex, 'request').mockResolvedValue(responsePayload);
|
|
3436
|
+
warnSpy = jest.spyOn(_Logger.default, 'warn');
|
|
3437
|
+
infoSpy = jest.spyOn(_Logger.default, 'info');
|
|
3438
|
+
metricSpy = jest.spyOn(call['metricManager'], 'submitCallMetric');
|
|
3439
|
+
call.on(_types2.CALL_EVENT_KEYS.DISCONNECT, /*#__PURE__*/function () {
|
|
3440
|
+
var _ref70 = (0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee69(correlationId) {
|
|
3441
|
+
return _regenerator.default.wrap(function (_context69) {
|
|
3442
|
+
while (1) switch (_context69.prev = _context69.next) {
|
|
3443
|
+
case 0:
|
|
3444
|
+
expect(correlationId).toStrictEqual(call.getCorrelationId());
|
|
3445
|
+
case 1:
|
|
3446
|
+
case "end":
|
|
3447
|
+
return _context69.stop();
|
|
3448
|
+
}
|
|
3449
|
+
}, _callee69);
|
|
3450
|
+
}));
|
|
3451
|
+
return function (_x11) {
|
|
3452
|
+
return _ref70.apply(this, arguments);
|
|
3453
|
+
};
|
|
3454
|
+
}());
|
|
3455
|
+
_context70.next = 1;
|
|
3456
|
+
return call.completeTransfer(_types5.TransferType.BLIND, undefined, transfereeNumber);
|
|
3457
|
+
case 1:
|
|
3458
|
+
_context70.next = 2;
|
|
3459
|
+
return (0, _testUtil.flushPromises)(2);
|
|
3460
|
+
case 2:
|
|
3461
|
+
expect(requestSpy).toBeCalled();
|
|
3462
|
+
expect(metricSpy).toHaveBeenCalledWith(_types4.METRIC_EVENT.CALL, _types4.TRANSFER_ACTION.BLIND, _types4.METRIC_TYPE.BEHAVIORAL, call.getCallId(), call.getCorrelationId(), undefined);
|
|
3463
|
+
call.sendCallStateMachineEvt({
|
|
3464
|
+
type: 'E_RECV_CALL_DISCONNECT'
|
|
3465
|
+
});
|
|
3466
|
+
|
|
3467
|
+
/* We should return back to S_RECV_CALL_DISCONNECT state */
|
|
3468
|
+
expect(call['callStateMachine'].state.value).toStrictEqual('S_RECV_CALL_DISCONNECT');
|
|
3469
|
+
expect(handleErrorSpy).not.toBeCalled();
|
|
3470
|
+
expect(uploadLogsSpy).not.toBeCalled();
|
|
3471
|
+
expect(infoSpy).toHaveBeenCalledWith("Initiating Blind transfer with : ".concat(transfereeNumber), transferLoggingContext);
|
|
3472
|
+
expect(warnSpy).not.toHaveBeenCalledWith("Blind Transfer failed for correlationId ".concat(call.getCorrelationId()), transferLoggingContext);
|
|
3473
|
+
expect(infoSpy).toHaveBeenCalledWith("Initiating Blind transfer with : ".concat(transfereeNumber), transferLoggingContext);
|
|
3474
|
+
expect(warnSpy).not.toHaveBeenCalledWith("Blind Transfer failed for correlationId ".concat(call.getCorrelationId()), transferLoggingContext);
|
|
3475
|
+
case 3:
|
|
3476
|
+
case "end":
|
|
3477
|
+
return _context70.stop();
|
|
3478
|
+
}
|
|
3479
|
+
}, _callee70);
|
|
3480
|
+
})));
|
|
3481
|
+
it('Handle unsuccessful blind transfer case', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee71() {
|
|
3482
|
+
var responsePayload, emitSpy, requestSpy, warnSpy, metricSpy;
|
|
3483
|
+
return _regenerator.default.wrap(function (_context71) {
|
|
3484
|
+
while (1) switch (_context71.prev = _context71.next) {
|
|
3485
|
+
case 0:
|
|
3486
|
+
responsePayload = {
|
|
3487
|
+
statusCode: 403,
|
|
3488
|
+
body: mockResponseBody
|
|
3489
|
+
};
|
|
3490
|
+
emitSpy = jest.spyOn(call, 'emit');
|
|
3491
|
+
requestSpy = jest.spyOn(webex, 'request').mockRejectedValue(responsePayload);
|
|
3492
|
+
warnSpy = jest.spyOn(_Logger.default, 'warn');
|
|
3493
|
+
metricSpy = jest.spyOn(call['metricManager'], 'submitCallMetric');
|
|
3494
|
+
call['broadworksCorrelationInfo'] = 'dummy-broadworks-correlation-info';
|
|
3495
|
+
_context71.next = 1;
|
|
3496
|
+
return call.completeTransfer(_types5.TransferType.BLIND, undefined, transfereeNumber);
|
|
3497
|
+
case 1:
|
|
3498
|
+
_context71.next = 2;
|
|
3499
|
+
return (0, _testUtil.flushPromises)(1);
|
|
3500
|
+
case 2:
|
|
3501
|
+
expect(requestSpy).toBeCalled();
|
|
3502
|
+
/* We should be in CALL_ESTABLISHED state */
|
|
3503
|
+
expect(call['callStateMachine'].state.value).toStrictEqual('S_CALL_ESTABLISHED');
|
|
3504
|
+
expect(call['held']).toBe(true);
|
|
3505
|
+
expect(handleErrorSpy).toHaveBeenCalledWith(expect.anything(), _types.ERROR_LAYER.CALL_CONTROL,
|
|
3506
|
+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
3507
|
+
expect.anything(), call.getCorrelationId(), responsePayload, 'completeTransfer', 'call');
|
|
3508
|
+
expect(uploadLogsSpy).toHaveBeenCalledWith({
|
|
3509
|
+
correlationId: call.getCorrelationId(),
|
|
3510
|
+
callId: call.getCallId(),
|
|
3511
|
+
broadworksCorrelationInfo: 'dummy-broadworks-correlation-info'
|
|
3512
|
+
});
|
|
3513
|
+
/* check whether error event is being emitted by sdk */
|
|
3514
|
+
expect(emitSpy).toBeCalledOnceWith(_types2.CALL_EVENT_KEYS.TRANSFER_ERROR, expect.any(_Errors.CallError));
|
|
3515
|
+
expect(warnSpy).toHaveBeenCalledWith("Blind Transfer failed for correlationId ".concat(call.getCorrelationId()), transferLoggingContext);
|
|
3516
|
+
expect(metricSpy).toHaveBeenCalledWith(_types4.METRIC_EVENT.CALL_ERROR, _types4.TRANSFER_ACTION.BLIND, _types4.METRIC_TYPE.BEHAVIORAL, call.getCallId(), call.getCorrelationId(), expect.any(_Errors.CallError));
|
|
3517
|
+
case 3:
|
|
3518
|
+
case "end":
|
|
3519
|
+
return _context71.stop();
|
|
3520
|
+
}
|
|
3521
|
+
}, _callee71);
|
|
3522
|
+
})));
|
|
3523
|
+
it('Handle unsuccessful consult transfer case', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee72() {
|
|
3524
|
+
var responsePayload, emitSpy, requestSpy, warnSpy, metricSpy;
|
|
3525
|
+
return _regenerator.default.wrap(function (_context72) {
|
|
3526
|
+
while (1) switch (_context72.prev = _context72.next) {
|
|
3527
|
+
case 0:
|
|
3528
|
+
responsePayload = {
|
|
3529
|
+
statusCode: 403,
|
|
3530
|
+
body: mockResponseBody
|
|
3531
|
+
};
|
|
3532
|
+
emitSpy = jest.spyOn(call, 'emit');
|
|
3533
|
+
requestSpy = jest.spyOn(webex, 'request').mockRejectedValue(responsePayload);
|
|
3534
|
+
warnSpy = jest.spyOn(_Logger.default, 'warn');
|
|
3535
|
+
metricSpy = jest.spyOn(call['metricManager'], 'submitCallMetric');
|
|
3536
|
+
_context72.next = 1;
|
|
3537
|
+
return call.completeTransfer(_types5.TransferType.CONSULT, secondCall.getCallId(), undefined);
|
|
3538
|
+
case 1:
|
|
3539
|
+
_context72.next = 2;
|
|
3540
|
+
return (0, _testUtil.flushPromises)(2);
|
|
3541
|
+
case 2:
|
|
3542
|
+
expect(requestSpy).toBeCalled();
|
|
3543
|
+
/* We should be in CALL_ESTABLISHED state */
|
|
3544
|
+
expect(call['callStateMachine'].state.value).toStrictEqual('S_CALL_ESTABLISHED');
|
|
3545
|
+
expect(call['held']).toBe(true);
|
|
3546
|
+
expect(secondCall['callStateMachine'].state.value).toStrictEqual('S_CALL_ESTABLISHED');
|
|
3547
|
+
expect(handleErrorSpy).toHaveBeenCalledWith(expect.anything(), _types.ERROR_LAYER.CALL_CONTROL,
|
|
3548
|
+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
3549
|
+
expect.anything(), call.getCorrelationId(), responsePayload, 'completeTransfer', 'call');
|
|
3550
|
+
expect(uploadLogsSpy).toHaveBeenCalledWith({
|
|
3551
|
+
correlationId: call.getCorrelationId(),
|
|
3552
|
+
callId: call.getCallId()
|
|
3553
|
+
});
|
|
3554
|
+
/* check whether error event is being emitted by sdk */
|
|
3555
|
+
expect(emitSpy).toHaveBeenCalledWith(_types2.CALL_EVENT_KEYS.TRANSFER_ERROR, expect.any(_Errors.CallError));
|
|
3556
|
+
expect(warnSpy).toHaveBeenCalledWith("Consult Transfer failed for correlationId ".concat(call.getCorrelationId()), transferLoggingContext);
|
|
3557
|
+
expect(metricSpy).toHaveBeenCalledWith(_types4.METRIC_EVENT.CALL_ERROR, _types4.TRANSFER_ACTION.CONSULT, _types4.METRIC_TYPE.BEHAVIORAL, call.getCallId(), call.getCorrelationId(), expect.any(_Errors.CallError));
|
|
3558
|
+
case 3:
|
|
3559
|
+
case "end":
|
|
3560
|
+
return _context72.stop();
|
|
3561
|
+
}
|
|
3562
|
+
}, _callee72);
|
|
3563
|
+
})));
|
|
3564
|
+
it('Handle blind transfer with undefined transferTarget', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee73() {
|
|
3565
|
+
var requestSpy, warnSpy;
|
|
3566
|
+
return _regenerator.default.wrap(function (_context73) {
|
|
3567
|
+
while (1) switch (_context73.prev = _context73.next) {
|
|
3568
|
+
case 0:
|
|
3569
|
+
requestSpy = jest.spyOn(webex, 'request');
|
|
3570
|
+
warnSpy = jest.spyOn(_Logger.default, 'warn');
|
|
3571
|
+
_context73.next = 1;
|
|
3572
|
+
return call.completeTransfer(_types5.TransferType.BLIND, undefined, undefined);
|
|
3573
|
+
case 1:
|
|
3574
|
+
/* We should be in CALL_ESTABLISHED state */
|
|
3575
|
+
expect(call['callStateMachine'].state.value).toStrictEqual('S_CALL_ESTABLISHED');
|
|
3576
|
+
expect(secondCall['callStateMachine'].state.value).toStrictEqual('S_CALL_ESTABLISHED');
|
|
3577
|
+
expect(handleErrorSpy).not.toBeCalled();
|
|
3578
|
+
expect(uploadLogsSpy).not.toBeCalled();
|
|
3579
|
+
expect(requestSpy).not.toBeCalled();
|
|
3580
|
+
expect(warnSpy).toBeCalledOnceWith("Invalid information received, transfer failed for correlationId: ".concat(call.getCorrelationId()), transferLoggingContext);
|
|
3581
|
+
case 2:
|
|
3582
|
+
case "end":
|
|
3583
|
+
return _context73.stop();
|
|
3584
|
+
}
|
|
3585
|
+
}, _callee73);
|
|
3586
|
+
})));
|
|
3587
|
+
it('Handle consult transfer with undefined transferCallId', /*#__PURE__*/(0, _asyncToGenerator2.default)(/*#__PURE__*/_regenerator.default.mark(function _callee74() {
|
|
3588
|
+
var requestSpy, warnSpy;
|
|
3589
|
+
return _regenerator.default.wrap(function (_context74) {
|
|
3590
|
+
while (1) switch (_context74.prev = _context74.next) {
|
|
3591
|
+
case 0:
|
|
3592
|
+
requestSpy = jest.spyOn(webex, 'request');
|
|
3593
|
+
warnSpy = jest.spyOn(_Logger.default, 'warn');
|
|
3594
|
+
_context74.next = 1;
|
|
3595
|
+
return call.completeTransfer(_types5.TransferType.CONSULT, undefined, undefined);
|
|
3596
|
+
case 1:
|
|
3597
|
+
/* We should be in CALL_ESTABLISHED state */
|
|
3598
|
+
expect(call['callStateMachine'].state.value).toStrictEqual('S_CALL_ESTABLISHED');
|
|
3599
|
+
expect(secondCall['callStateMachine'].state.value).toStrictEqual('S_CALL_ESTABLISHED');
|
|
3600
|
+
expect(handleErrorSpy).not.toBeCalled();
|
|
3601
|
+
expect(uploadLogsSpy).not.toBeCalled();
|
|
3602
|
+
expect(requestSpy).not.toBeCalled();
|
|
3603
|
+
expect(warnSpy).toBeCalledOnceWith("Invalid information received, transfer failed for correlationId: ".concat(call.getCorrelationId()), transferLoggingContext);
|
|
3604
|
+
case 2:
|
|
3605
|
+
case "end":
|
|
3606
|
+
return _context74.stop();
|
|
3607
|
+
}
|
|
3608
|
+
}, _callee74);
|
|
3609
|
+
})));
|
|
3610
|
+
});
|
|
3611
|
+
});
|
|
3612
|
+
//# sourceMappingURL=call.test.js.map
|