@rongcloud/plugin-call 5.2.3 → 5.2.4
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/index.d.ts +785 -797
- package/dist/index.esm.js +602 -558
- package/dist/index.umd.js +5 -6
- package/package.json +2 -2
package/dist/index.esm.js
CHANGED
|
@@ -1,9 +1,9 @@
|
|
|
1
|
-
var
|
|
2
|
-
var
|
|
3
|
-
var f = (s, e, t) => (
|
|
4
|
-
import { EventEmitter as
|
|
5
|
-
import { RCResolution as
|
|
6
|
-
const
|
|
1
|
+
var _e = Object.defineProperty;
|
|
2
|
+
var he = (s, e, t) => e in s ? _e(s, e, { enumerable: !0, configurable: !0, writable: !0, value: t }) : s[e] = t;
|
|
3
|
+
var f = (s, e, t) => (he(s, typeof e != "symbol" ? e + "" : e, t), t);
|
|
4
|
+
import { EventEmitter as ie, ConversationType as D, ErrorCode as W, isUndefined as ge, isBoolean as de, isString as Y, RTCJoinType as ne, LogL as K, VersionManage as q } from "@rongcloud/engine";
|
|
5
|
+
import { RCResolution as ue, RCRTCCode as p, RCKickReason as Q } from "@rongcloud/plugin-rtc";
|
|
6
|
+
const fe = (s) => {
|
|
7
7
|
const e = "0123456789abcdefghigklmnopqrstuvwxyzABCDEFGHIGKLMNOPQRSTUVWXYZ+/".split(""), t = e.length + 1;
|
|
8
8
|
let i = +s;
|
|
9
9
|
const n = [];
|
|
@@ -12,23 +12,23 @@ const ue = (s) => {
|
|
|
12
12
|
i = (i - o) / t, n.unshift(e[o]);
|
|
13
13
|
} while (i);
|
|
14
14
|
return n.join("");
|
|
15
|
-
},
|
|
15
|
+
}, Ie = () => "xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx".replace(/[xy]/g, (s) => {
|
|
16
16
|
const e = Math.random() * 16 | 0;
|
|
17
17
|
return (s === "x" ? e : e & 3 | 8).toString(16);
|
|
18
|
-
}),
|
|
19
|
-
let s =
|
|
20
|
-
return s = `${s.replace(/-/g, "")}0`, s = parseInt(s, 16), s =
|
|
21
|
-
},
|
|
18
|
+
}), oe = () => {
|
|
19
|
+
let s = Ie();
|
|
20
|
+
return s = `${s.replace(/-/g, "")}0`, s = parseInt(s, 16), s = fe(s), s.length > 22 && (s = s.slice(0, 22)), s;
|
|
21
|
+
}, ee = () => {
|
|
22
22
|
const s = Math.floor(Math.random() * 1e3);
|
|
23
|
-
let e =
|
|
23
|
+
let e = oe();
|
|
24
24
|
return e = e.replace(/\//g, "0"), [e, Date.now(), s].join("_");
|
|
25
|
-
},
|
|
25
|
+
}, Re = (s, e) => setTimeout(s, e), re = new ie(), Z = (s) => {
|
|
26
26
|
const e = "RCCallDeviceId";
|
|
27
27
|
let t = s.sessionStorage.getItem(e);
|
|
28
|
-
return t || (t =
|
|
28
|
+
return t || (t = oe(), s.sessionStorage.setItem(e, t)), t;
|
|
29
29
|
};
|
|
30
|
-
var b = /* @__PURE__ */ ((s) => (s[s.OUTGOING = 1] = "OUTGOING", s[s.INCOMING = 2] = "INCOMING", s[s.RINGING = 3] = "RINGING", s[s.CONNECTED = 4] = "CONNECTED", s[s.IDLE = 5] = "IDLE", s[s.ACCEPTED = 6] = "ACCEPTED", s))(b || {}),
|
|
31
|
-
const
|
|
30
|
+
var b = /* @__PURE__ */ ((s) => (s[s.OUTGOING = 1] = "OUTGOING", s[s.INCOMING = 2] = "INCOMING", s[s.RINGING = 3] = "RINGING", s[s.CONNECTED = 4] = "CONNECTED", s[s.IDLE = 5] = "IDLE", s[s.ACCEPTED = 6] = "ACCEPTED", s))(b || {}), l = /* @__PURE__ */ ((s) => (s[s.SUCCESS = 1e4] = "SUCCESS", s[s.STATE_MACHINE_EXIT = 53200] = "STATE_MACHINE_EXIT", s[s.SEND_MSG_ERROR = 53201] = "SEND_MSG_ERROR", s[s.REJECTED_BY_BLACKLIST = 53202] = "REJECTED_BY_BLACKLIST", s[s.NOT_IN_GROUP = 53203] = "NOT_IN_GROUP", s[s.GET_LOCAL_AUDIO_TRACK_ERROR = 53301] = "GET_LOCAL_AUDIO_TRACK_ERROR", s[s.GET_LOCAL_VIDEO_TRACK_ERROR = 53302] = "GET_LOCAL_VIDEO_TRACK_ERROR", s[s.GET_LOCAL_AUDIO_AND_VIDEO_TRACK_ERROR = 53303] = "GET_LOCAL_AUDIO_AND_VIDEO_TRACK_ERROR", s[s.JOIN_ROOM_ERROR = 53304] = "JOIN_ROOM_ERROR", s[s.AUDIO_PUBLISH_ERROR = 53305] = "AUDIO_PUBLISH_ERROR", s[s.VIDEO_PUBLISH_ERROR = 53306] = "VIDEO_PUBLISH_ERROR", s[s.AUDIO_AND_VIDEO_PUBLISH_ERROR = 53307] = "AUDIO_AND_VIDEO_PUBLISH_ERROR", s[s.QUERY_JOINED_USER_INFO_ERROR = 53308] = "QUERY_JOINED_USER_INFO_ERROR", s[s.MISSING_VIDEO_TRACK_ERROR = 53309] = "MISSING_VIDEO_TRACK_ERROR", s[s.UNPUBLISH_VIDEO_ERROR = 53310] = "UNPUBLISH_VIDEO_ERROR", s[s.CONVERSATION_NOT_GROUP_ERROR = 53311] = "CONVERSATION_NOT_GROUP_ERROR", s[s.NOT_IN_ROOM_ERROR = 53312] = "NOT_IN_ROOM_ERROR", s[s.PARAM_ERROR = 53313] = "PARAM_ERROR", s))(l || {}), g = /* @__PURE__ */ ((s) => (s[s.CANCEL = 1] = "CANCEL", s[s.REJECT = 2] = "REJECT", s[s.HANGUP = 3] = "HANGUP", s[s.BUSY_LINE = 4] = "BUSY_LINE", s[s.NO_RESPONSE = 5] = "NO_RESPONSE", s[s.ENGINE_UNSUPPORTED = 6] = "ENGINE_UNSUPPORTED", s[s.NETWORK_ERROR = 7] = "NETWORK_ERROR", s[s.GET_MEDIA_RESOURCES_ERROR = 8] = "GET_MEDIA_RESOURCES_ERROR", s[s.PUBLISH_ERROR = 9] = "PUBLISH_ERROR", s[s.SUBSCRIBE_ERROR = 10] = "SUBSCRIBE_ERROR", s[s.REMOTE_CANCEL = 11] = "REMOTE_CANCEL", s[s.REMOTE_REJECT = 12] = "REMOTE_REJECT", s[s.REMOTE_HANGUP = 13] = "REMOTE_HANGUP", s[s.REMOTE_BUSY_LINE = 14] = "REMOTE_BUSY_LINE", s[s.REMOTE_NO_RESPONSE = 15] = "REMOTE_NO_RESPONSE", s[s.REMOTE_ENGINE_UNSUPPORTED = 16] = "REMOTE_ENGINE_UNSUPPORTED", s[s.REMOTE_NETWORK_ERROR = 17] = "REMOTE_NETWORK_ERROR", s[s.REMOTE_GET_MEDIA_RESOURCE_ERROR = 18] = "REMOTE_GET_MEDIA_RESOURCE_ERROR", s[s.REMOTE_PUBLISH_ERROR = 19] = "REMOTE_PUBLISH_ERROR", s[s.REMOTE_SUBSCRIBE_ERROR = 20] = "REMOTE_SUBSCRIBE_ERROR", s[s.OTHER_CLIENT_JOINED_CALL = 21] = "OTHER_CLIENT_JOINED_CALL", s[s.OTHER_CLIENT_IN_CALL = 22] = "OTHER_CLIENT_IN_CALL", s[s.KICKED_BY_SERVER = 23] = "KICKED_BY_SERVER", s[s.ACCEPT_SYSTEM_CALL = 24] = "ACCEPT_SYSTEM_CALL", s[s.REMOTE_OTHER_CLIENT_JOINED_CALL = 31] = "REMOTE_OTHER_CLIENT_JOINED_CALL", s[s.REMOTE_OTHER_CLIENT_IN_CALL = 32] = "REMOTE_OTHER_CLIENT_IN_CALL", s[s.REMOTE_KICKED_BY_SERVER = 33] = "REMOTE_KICKED_BY_SERVER", s[s.REMOTE_ACCEPT_SYSTEM_CALL = 34] = "REMOTE_ACCEPT_SYSTEM_CALL", s[s.ACCEPT_BY_OTHER_CLIENT = 101] = "ACCEPT_BY_OTHER_CLIENT", s[s.HANGUP_BY_OTHER_CLIENT = 102] = "HANGUP_BY_OTHER_CLIENT", s[s.ADDED_TO_BLACKLIST = 103] = "ADDED_TO_BLACKLIST", s[s.SERVICE_NOT_OPENED = 104] = "SERVICE_NOT_OPENED", s))(g || {});
|
|
31
|
+
const G = {
|
|
32
32
|
[
|
|
33
33
|
1
|
|
34
34
|
/* CANCEL */
|
|
@@ -91,12 +91,12 @@ const H = {
|
|
|
91
91
|
]: 34
|
|
92
92
|
/* REMOTE_ACCEPT_SYSTEM_CALL */
|
|
93
93
|
};
|
|
94
|
-
var R = /* @__PURE__ */ ((s) => (s.VCInvite = "RC:VCInvite", s.VCRinging = "RC:VCRinging", s.VCAccept = "RC:VCAccept", s.VCHangup = "RC:VCHangup", s.VCModifyMem = "RC:VCModifyMem", s.VCModifyMedia = "RC:VCModifyMedia", s))(R || {}),
|
|
94
|
+
var R = /* @__PURE__ */ ((s) => (s.VCInvite = "RC:VCInvite", s.VCRinging = "RC:VCRinging", s.VCAccept = "RC:VCAccept", s.VCHangup = "RC:VCHangup", s.VCModifyMem = "RC:VCModifyMem", s.VCModifyMedia = "RC:VCModifyMedia", s))(R || {}), T = /* @__PURE__ */ ((s) => (s[s.WAITING = 0] = "WAITING", s[s.KEEPING = 1] = "KEEPING", s[s.END = 2] = "END", s))(T || {}), m = /* @__PURE__ */ ((s) => (s[s.NONE = 0] = "NONE", s[s.WAITING = 1] = "WAITING", s[s.KEEPING = 2] = "KEEPING", s))(m || {}), $ = /* @__PURE__ */ ((s) => (s[s.RCCallRoomTypeNormalCall = 0] = "RCCallRoomTypeNormalCall", s[s.RCCallRoomTypeAcrossCall = 7] = "RCCallRoomTypeAcrossCall", s))($ || {});
|
|
95
95
|
let V = class {
|
|
96
96
|
constructor(e, t) {
|
|
97
97
|
f(this, "_timerId", 0);
|
|
98
98
|
f(this, "_startTime", 0);
|
|
99
|
-
e && (this._timerId =
|
|
99
|
+
e && (this._timerId = Re(() => {
|
|
100
100
|
e();
|
|
101
101
|
}, t)), this._startTime = Date.now();
|
|
102
102
|
}
|
|
@@ -114,8 +114,8 @@ let V = class {
|
|
|
114
114
|
this._startTime = 0;
|
|
115
115
|
}
|
|
116
116
|
};
|
|
117
|
-
class
|
|
118
|
-
constructor(e, t, i, n, o, r, a,
|
|
117
|
+
class F {
|
|
118
|
+
constructor(e, t, i, n, o, r, a, h, c) {
|
|
119
119
|
/**
|
|
120
120
|
* 房间状态
|
|
121
121
|
*/
|
|
@@ -164,9 +164,8 @@ class Y {
|
|
|
164
164
|
* 是否是跨 appkey
|
|
165
165
|
*/
|
|
166
166
|
f(this, "_isCrossAppkey", !1);
|
|
167
|
-
f(this, "
|
|
168
|
-
|
|
169
|
-
this._context = e, this._runtime = t, this._logger = i, this._callMsgHandler = n, this._channelId = o, this._conversationType = r, this._targetId = a, this._mediaType = c, this._callId = l, this._callMsgHandler.registerStateMachineEvent(this._callId, "onRinging", this._onRinging.bind(this)), this._callMsgHandler.registerStateMachineEvent(this._callId, "onAccept", this._onAccept.bind(this)), this._callMsgHandler.registerStateMachineEvent(this._callId, "onMediaModify", this._onMediaModify.bind(this)), this._callMsgHandler.registerStateMachineEvent(this._callId, "onHungup", this._onHungup.bind(this));
|
|
167
|
+
f(this, "_hungupPushConfig");
|
|
168
|
+
this._context = e, this._runtime = t, this._logger = i, this._callMsgHandler = n, this._channelId = o, this._conversationType = r, this._targetId = a, this._mediaType = h, this._callId = c, this._callMsgHandler.registerStateMachineEvent(this._callId, "onRinging", this._onRinging.bind(this)), this._callMsgHandler.registerStateMachineEvent(this._callId, "onAccept", this._onAccept.bind(this)), this._callMsgHandler.registerStateMachineEvent(this._callId, "onMediaModify", this._onMediaModify.bind(this)), this._callMsgHandler.registerStateMachineEvent(this._callId, "onHungup", this._onHungup.bind(this));
|
|
170
169
|
}
|
|
171
170
|
/**
|
|
172
171
|
* 获取校正后超时时间
|
|
@@ -188,7 +187,7 @@ class Y {
|
|
|
188
187
|
this._logger.warn("_", `[RCCallStateMachine] notifyStateChange -> info: ${JSON.stringify({
|
|
189
188
|
state: e,
|
|
190
189
|
reason: t
|
|
191
|
-
})}`), this._endReason = t || null, this._sessionState !== e && (this._sessionState = e, (i = this._watchers) == null || i.onStateChange({ state: e, reason: t })), e ===
|
|
190
|
+
})}`), this._endReason = t || null, this._sessionState !== e && (this._sessionState = e, (i = this._watchers) == null || i.onStateChange({ state: e, reason: t })), e === T.END && (re.emit("onStateMachineClose", this._callId), this._callMsgHandler.unregisterStateMachineEvent(this._callId));
|
|
192
191
|
}
|
|
193
192
|
/**
|
|
194
193
|
* 通知 call 层人员状态变更及原因
|
|
@@ -204,20 +203,20 @@ class Y {
|
|
|
204
203
|
const { senderUserId: t, content: { user: i, reason: n }, messageType: o } = e;
|
|
205
204
|
this._userInfo[t] = {
|
|
206
205
|
userId: t,
|
|
207
|
-
state:
|
|
206
|
+
state: m.NONE,
|
|
208
207
|
isCaller: !1,
|
|
209
208
|
isRemote: !1
|
|
210
209
|
};
|
|
211
210
|
for (const a in this._userTimers)
|
|
212
211
|
this._clearTimerById(a);
|
|
213
212
|
let r = g.ACCEPT_BY_OTHER_CLIENT;
|
|
214
|
-
o === R.VCHangup && (n === g.BUSY_LINE ? r = g.OTHER_CLIENT_IN_CALL : n === g.NO_RESPONSE ? r = g.NO_RESPONSE : r = g.HANGUP_BY_OTHER_CLIENT), Object.assign(this._userInfo[t], i), this._notifyUserStateChange(this._userInfo[t], r), this._notifyStateChange(
|
|
213
|
+
o === R.VCHangup && (n === g.BUSY_LINE ? r = g.OTHER_CLIENT_IN_CALL : n === g.NO_RESPONSE ? r = g.NO_RESPONSE : r = g.HANGUP_BY_OTHER_CLIENT), Object.assign(this._userInfo[t], i), this._notifyUserStateChange(this._userInfo[t], r), this._notifyStateChange(T.END, r);
|
|
215
214
|
}
|
|
216
215
|
/**
|
|
217
216
|
* 正在通话中,且不是当前已接通用户设备(deviceId)发来的消息
|
|
218
217
|
*/
|
|
219
218
|
_isRemoteInvalidMsg(e, t) {
|
|
220
|
-
return !this._userInfo[e] || !this._userInfo[e].deviceId || !t ? !1 : this._userInfo[e].state ===
|
|
219
|
+
return !this._userInfo[e] || !this._userInfo[e].deviceId || !t ? !1 : this._userInfo[e].state === m.KEEPING && this._userInfo[e].deviceId !== t;
|
|
221
220
|
}
|
|
222
221
|
_onRinging(e) {
|
|
223
222
|
const { senderUserId: t, content: { user: i, deviceId: n } } = e, o = this._isCrossAppkey ? t.split("_")[1] : t;
|
|
@@ -237,16 +236,16 @@ class Y {
|
|
|
237
236
|
this._otherClientHandle(e);
|
|
238
237
|
return;
|
|
239
238
|
}
|
|
240
|
-
this._userTimers[r] && this._clearTimerById(r), (this._conversationType === D.PRIVATE ? [a, r] : [r]).forEach((
|
|
241
|
-
const _ =
|
|
242
|
-
this._userInfo[
|
|
243
|
-
userId:
|
|
244
|
-
state:
|
|
239
|
+
this._userTimers[r] && this._clearTimerById(r), (this._conversationType === D.PRIVATE ? [a, r] : [r]).forEach((c) => {
|
|
240
|
+
const _ = c === a;
|
|
241
|
+
this._userInfo[c] = {
|
|
242
|
+
userId: c,
|
|
243
|
+
state: m.KEEPING,
|
|
245
244
|
isCaller: _,
|
|
246
245
|
isRemote: _,
|
|
247
|
-
deviceId: _ ?
|
|
248
|
-
}, _ || (this._beginTimestamp = Date.now(), Object.assign(this._userInfo[r], i)), this._notifyUserStateChange(this._userInfo[
|
|
249
|
-
}), this.getCallerId() === a && this._notifyStateChange(
|
|
246
|
+
deviceId: _ ? Z(this._runtime) : n
|
|
247
|
+
}, _ || (this._beginTimestamp = Date.now(), Object.assign(this._userInfo[r], i)), this._notifyUserStateChange(this._userInfo[c]);
|
|
248
|
+
}), this.getCallerId() === a && this._notifyStateChange(T.KEEPING), this._watchers.onAccept({ userId: r });
|
|
250
249
|
}
|
|
251
250
|
_onMediaModify(e) {
|
|
252
251
|
const { senderUserId: t, content: { mediaType: i, user: n, deviceId: o } } = e;
|
|
@@ -260,22 +259,22 @@ class Y {
|
|
|
260
259
|
}));
|
|
261
260
|
}
|
|
262
261
|
_onHungup(e) {
|
|
263
|
-
const { senderUserId: t, content: i } = e, n = this._isCrossAppkey ? t.split("_")[1] : t, { reason: o, user: r, deviceId: a } = i,
|
|
262
|
+
const { senderUserId: t, content: i } = e, n = this._isCrossAppkey ? t.split("_")[1] : t, { reason: o, user: r, deviceId: a } = i, h = this._context.getCurrentId();
|
|
264
263
|
if (this._isRemoteInvalidMsg(n, a)) {
|
|
265
264
|
this._logger.debug("_", "[RCCallStateMachine] _onHungup -> not the remote device that is currently talking");
|
|
266
265
|
return;
|
|
267
266
|
}
|
|
268
|
-
if (
|
|
267
|
+
if (h === n) {
|
|
269
268
|
this._otherClientHandle(e);
|
|
270
269
|
return;
|
|
271
270
|
}
|
|
272
|
-
if (this._sessionState ===
|
|
271
|
+
if (this._sessionState === T.END) {
|
|
273
272
|
this._logger.info("_", `[RCCallStateMachine] Invalid hang up message, current room status has ended -> sessionState: ${this._sessionState}`);
|
|
274
273
|
return;
|
|
275
274
|
}
|
|
276
|
-
this._userInfo[n] && (this._userInfo[n].state =
|
|
277
|
-
const
|
|
278
|
-
(
|
|
275
|
+
this._userInfo[n] && (this._userInfo[n].state = m.NONE, this._endTimestamp = Date.now(), Object.assign(this._userInfo[n], r), this._notifyUserStateChange(this._userInfo[n], G[o]), delete this._userInfo[n]), G[o] === g.REMOTE_CANCEL ? this.getRemoteUserIds().length < 1 && this._clearTimerById(h) : this.getInviterId() === h ? this._clearTimerById(n) : this.getRemoteUserIds().length < 1 && this._clearTimerById(h);
|
|
276
|
+
const c = Object.keys(this._userInfo).length < 2, _ = this._inviterId === n, u = Object.values(this._userInfo).every((S) => S.state !== m.KEEPING);
|
|
277
|
+
(c || _ && u) && this._notifyStateChange(T.END, G[o]), this._watchers.onHungup({ userId: n, ...r }, G[o]);
|
|
279
278
|
}
|
|
280
279
|
/**
|
|
281
280
|
* 注册事件监听
|
|
@@ -293,38 +292,38 @@ class Y {
|
|
|
293
292
|
inviteUserIds: o,
|
|
294
293
|
user: r,
|
|
295
294
|
deviceId: a,
|
|
296
|
-
roomType:
|
|
295
|
+
roomType: h
|
|
297
296
|
} = i;
|
|
298
|
-
let
|
|
299
|
-
|
|
297
|
+
let c;
|
|
298
|
+
h === $.RCCallRoomTypeAcrossCall ? ([, c] = t.split("_"), this._watchers.crossAppkey(!0), this._isCrossAppkey = !0) : c = t;
|
|
300
299
|
const _ = this._context.getCurrentId();
|
|
301
|
-
if (this._isRemoteInvalidMsg(
|
|
300
|
+
if (this._isRemoteInvalidMsg(c, a)) {
|
|
302
301
|
this._logger.debug("_", "[RCCallStateMachine] __onInvite -> not the remote device that is currently talking");
|
|
303
302
|
return;
|
|
304
303
|
}
|
|
305
|
-
if (_ ===
|
|
304
|
+
if (_ === c)
|
|
306
305
|
return;
|
|
307
|
-
this._callerId = this._inviterId =
|
|
308
|
-
const
|
|
306
|
+
this._callerId = this._inviterId = c;
|
|
307
|
+
const u = [t, ...o];
|
|
309
308
|
this._callMsgHandler.sendRinging({
|
|
310
309
|
conversationType: this._conversationType,
|
|
311
310
|
targetId: this._targetId,
|
|
312
311
|
channelId: this._channelId,
|
|
313
312
|
callId: this._callId,
|
|
314
|
-
userIds:
|
|
313
|
+
userIds: u.filter((d) => this._isCrossAppkey ? d.split("_")[1] !== _ : d !== _)
|
|
315
314
|
});
|
|
316
|
-
const
|
|
317
|
-
[
|
|
318
|
-
this._userInfo[
|
|
319
|
-
userId:
|
|
320
|
-
state:
|
|
321
|
-
isCaller:
|
|
322
|
-
isRemote:
|
|
323
|
-
},
|
|
324
|
-
const
|
|
325
|
-
|
|
315
|
+
const S = this._isCrossAppkey ? [o[0].split("_")[1]] : o;
|
|
316
|
+
[c, ...S].forEach((d) => {
|
|
317
|
+
this._userInfo[d] = {
|
|
318
|
+
userId: d,
|
|
319
|
+
state: m.WAITING,
|
|
320
|
+
isCaller: d === c,
|
|
321
|
+
isRemote: d !== _
|
|
322
|
+
}, d === c && Object.assign(this._userInfo[d], r, { deviceId: a }), this._notifyUserStateChange(this._userInfo[d]), d !== c && (this._userTimers[d] = new V(() => {
|
|
323
|
+
const y = d === _ ? g.NO_RESPONSE : g.REMOTE_NO_RESPONSE;
|
|
324
|
+
d === _ ? this._hungupHandle(y) : (this._userInfo[d] && (this._userInfo[d].state = m.NONE), this._notifyUserStateChange(this._userInfo[d]), this._watchers.onHungup(this._userInfo[d], y), delete this._userInfo[d]);
|
|
326
325
|
}, this._getTimeout(n)));
|
|
327
|
-
}), this._notifyStateChange(
|
|
326
|
+
}), this._notifyStateChange(T.WAITING);
|
|
328
327
|
}
|
|
329
328
|
/**
|
|
330
329
|
* 收到 memberModify 消息时状态机更新(CallEngine 内部调用)
|
|
@@ -335,54 +334,54 @@ class Y {
|
|
|
335
334
|
user: o,
|
|
336
335
|
existedUserPofiles: r,
|
|
337
336
|
caller: a,
|
|
338
|
-
deviceId:
|
|
339
|
-
inviteUserIds:
|
|
337
|
+
deviceId: h,
|
|
338
|
+
inviteUserIds: c,
|
|
340
339
|
mediaType: _
|
|
341
|
-
} = i,
|
|
342
|
-
if (this._isRemoteInvalidMsg(t,
|
|
340
|
+
} = i, u = this._context.getCurrentId();
|
|
341
|
+
if (this._isRemoteInvalidMsg(t, h)) {
|
|
343
342
|
this._logger.debug("_", "[RCCallStateMachine] __onMemberModify -> not the remote device that is currently talking");
|
|
344
343
|
return;
|
|
345
344
|
}
|
|
346
|
-
if (
|
|
345
|
+
if (u === t)
|
|
347
346
|
return;
|
|
348
|
-
if (this._callerId = a, this._inviterId = t,
|
|
349
|
-
r.push({ userId:
|
|
350
|
-
}),
|
|
351
|
-
const
|
|
352
|
-
r.forEach((
|
|
353
|
-
|
|
347
|
+
if (this._callerId = a, this._inviterId = t, c.forEach((C) => {
|
|
348
|
+
r.push({ userId: C, mediaType: _, callStatus: b.INCOMING });
|
|
349
|
+
}), c.includes(u)) {
|
|
350
|
+
const C = [];
|
|
351
|
+
r.forEach((d) => {
|
|
352
|
+
d.userId !== u && C.push(d.userId);
|
|
354
353
|
}), this._callMsgHandler.sendRinging({
|
|
355
354
|
conversationType: this._conversationType,
|
|
356
355
|
targetId: this._targetId,
|
|
357
356
|
channelId: this._channelId,
|
|
358
357
|
callId: this._callId,
|
|
359
|
-
userIds:
|
|
360
|
-
}), this._notifyStateChange(
|
|
358
|
+
userIds: C
|
|
359
|
+
}), this._notifyStateChange(T.WAITING);
|
|
361
360
|
} else
|
|
362
361
|
this._watchers.onMemberModify({
|
|
363
362
|
sender: { userId: t, ...o },
|
|
364
|
-
invitedUsers:
|
|
363
|
+
invitedUsers: c.map((C) => ({ userId: C }))
|
|
365
364
|
});
|
|
366
|
-
r.forEach((
|
|
367
|
-
const { userId:
|
|
368
|
-
if (
|
|
369
|
-
userId:
|
|
370
|
-
state:
|
|
371
|
-
isCaller: t ===
|
|
372
|
-
isRemote:
|
|
373
|
-
},
|
|
374
|
-
if (!
|
|
365
|
+
r.forEach((C) => {
|
|
366
|
+
const { userId: d, callStatus: y } = C;
|
|
367
|
+
if (y !== b.IDLE && (this._userInfo[d] = {
|
|
368
|
+
userId: d,
|
|
369
|
+
state: y !== b.CONNECTED ? m.WAITING : m.KEEPING,
|
|
370
|
+
isCaller: t === d,
|
|
371
|
+
isRemote: u !== d
|
|
372
|
+
}, d === t && Object.assign(this._userInfo[d], o, { deviceId: h }), this._notifyUserStateChange(this._userInfo[d]), y !== b.CONNECTED && !this._userTimers[d])) {
|
|
373
|
+
if (!c.includes(d))
|
|
375
374
|
return;
|
|
376
|
-
this._userTimers[
|
|
377
|
-
this._userInfo[
|
|
378
|
-
const
|
|
379
|
-
this._notifyUserStateChange(this._userInfo[
|
|
375
|
+
this._userTimers[d] = new V(() => {
|
|
376
|
+
this._userInfo[d] && (this._userInfo[d].state = m.NONE);
|
|
377
|
+
const U = d === u ? g.NO_RESPONSE : g.REMOTE_NO_RESPONSE;
|
|
378
|
+
this._notifyUserStateChange(this._userInfo[d], U);
|
|
380
379
|
try {
|
|
381
|
-
this._watchers.onHungup(this._userInfo[
|
|
382
|
-
} catch (
|
|
383
|
-
this._logger.error("_", `[RCCallStateMachine] call onhungup error -> ${
|
|
380
|
+
this._watchers.onHungup(this._userInfo[d], U);
|
|
381
|
+
} catch (E) {
|
|
382
|
+
this._logger.error("_", `[RCCallStateMachine] call onhungup error -> ${E == null ? void 0 : E.stack}`);
|
|
384
383
|
}
|
|
385
|
-
delete this._userInfo[
|
|
384
|
+
delete this._userInfo[d], (Object.keys(this._userInfo).length < 2 || d === u) && this._notifyStateChange(T.END, U);
|
|
386
385
|
}, this._getTimeout(n));
|
|
387
386
|
}
|
|
388
387
|
});
|
|
@@ -393,8 +392,8 @@ class Y {
|
|
|
393
392
|
__handleInviteInSession() {
|
|
394
393
|
this._logger.info("_", "StateMachine -> __handleInviteInSession");
|
|
395
394
|
for (const e in this._userInfo)
|
|
396
|
-
this._userInfo[e].state && (this._userInfo[e].state =
|
|
397
|
-
this._notifyStateChange(
|
|
395
|
+
this._userInfo[e].state && (this._userInfo[e].state = m.NONE), this._notifyUserStateChange(this._userInfo[e]), this._clearTimerById(e);
|
|
396
|
+
this._notifyStateChange(T.END, g.BUSY_LINE), this._callMsgHandler.sendHungup({
|
|
398
397
|
channelId: this._channelId,
|
|
399
398
|
conversationType: this._conversationType,
|
|
400
399
|
targetId: this._targetId,
|
|
@@ -407,41 +406,39 @@ class Y {
|
|
|
407
406
|
* 主动呼叫 (CallEngine 内部调用)
|
|
408
407
|
* @param userIds 被邀请用户 ID 列表
|
|
409
408
|
* @param extra 消息的扩展信息
|
|
410
|
-
* @param
|
|
411
|
-
* @param pushContent 通知的内容
|
|
409
|
+
* @param pushConfig 移动端推送配置
|
|
412
410
|
*/
|
|
413
|
-
async __call(e, t = "", i
|
|
411
|
+
async __call(e, t = "", i, n = !1) {
|
|
414
412
|
this._logger.debug("_", `[RCCallStateMachine] invite -> userIds: ${JSON.stringify(e)}`);
|
|
415
|
-
const
|
|
416
|
-
roomType:
|
|
413
|
+
const o = this._callerId = this._inviterId = this._context.getCurrentId(), { code: r, message: a } = await this._callMsgHandler.sendInvite({
|
|
414
|
+
roomType: n ? $.RCCallRoomTypeAcrossCall : $.RCCallRoomTypeNormalCall,
|
|
417
415
|
channelId: this._channelId,
|
|
418
416
|
conversationType: this._conversationType,
|
|
419
417
|
targetId: this._targetId,
|
|
420
418
|
callId: this._callId,
|
|
421
419
|
extra: t,
|
|
422
|
-
|
|
423
|
-
pushContent: n,
|
|
420
|
+
pushConfig: i,
|
|
424
421
|
mediaType: this._mediaType,
|
|
425
|
-
inviteUserIds: e.filter((
|
|
422
|
+
inviteUserIds: e.filter((h) => h !== o)
|
|
426
423
|
});
|
|
427
|
-
if (this._isCrossAppkey =
|
|
428
|
-
const { sentTime:
|
|
429
|
-
(
|
|
430
|
-
const
|
|
431
|
-
this._userInfo[
|
|
432
|
-
userId:
|
|
433
|
-
state:
|
|
434
|
-
isCaller:
|
|
435
|
-
isRemote: !
|
|
436
|
-
}, this._notifyUserStateChange(this._userInfo[
|
|
437
|
-
this._userInfo[
|
|
438
|
-
}, this._getTimeout(
|
|
439
|
-
}), this._notifyStateChange(
|
|
424
|
+
if (this._isCrossAppkey = n, r === l.SUCCESS) {
|
|
425
|
+
const { sentTime: h } = a;
|
|
426
|
+
(n ? [o, e[0].split("_")[1]] : [o, ...e]).forEach((_) => {
|
|
427
|
+
const u = _ === o;
|
|
428
|
+
this._userInfo[_] = {
|
|
429
|
+
userId: _,
|
|
430
|
+
state: m.WAITING,
|
|
431
|
+
isCaller: u,
|
|
432
|
+
isRemote: !u
|
|
433
|
+
}, this._notifyUserStateChange(this._userInfo[_]), u || (this._userTimers[_] = new V(() => {
|
|
434
|
+
this._userInfo[_] && (this._userInfo[_].state = m.NONE), this._notifyUserStateChange(this._userInfo[_], g.REMOTE_NO_RESPONSE), this._watchers.onHungup(this._userInfo[_], g.REMOTE_NO_RESPONSE), delete this._userInfo[_], Object.keys(this._userInfo).length < 2 && this._notifyStateChange(T.END, g.REMOTE_NO_RESPONSE), this.getRemoteUserIds().length === 0 && this._userInfo[o].state !== m.KEEPING && this._hungupHandle(g.REMOTE_NO_RESPONSE);
|
|
435
|
+
}, this._getTimeout(h)));
|
|
436
|
+
}), this._notifyStateChange(T.WAITING);
|
|
440
437
|
} else {
|
|
441
|
-
const
|
|
442
|
-
this._notifyStateChange(
|
|
438
|
+
const h = r === l.REJECTED_BY_BLACKLIST ? g.ADDED_TO_BLACKLIST : g.NETWORK_ERROR;
|
|
439
|
+
this._notifyStateChange(T.END, h);
|
|
443
440
|
}
|
|
444
|
-
return { code:
|
|
441
|
+
return { code: r };
|
|
445
442
|
}
|
|
446
443
|
/**
|
|
447
444
|
* 接听
|
|
@@ -456,12 +453,12 @@ class Y {
|
|
|
456
453
|
mediaType: this._mediaType,
|
|
457
454
|
userIds: this.getRemoteUserIds()
|
|
458
455
|
});
|
|
459
|
-
if (this._clearTimerById(e), t ===
|
|
460
|
-
this._userInfo[e] && (this._userInfo[e].state =
|
|
456
|
+
if (this._clearTimerById(e), t === l.SUCCESS)
|
|
457
|
+
this._userInfo[e] && (this._userInfo[e].state = m.KEEPING), this._beginTimestamp = Date.now(), this._notifyUserStateChange(this._userInfo[e]), this._notifyStateChange(T.KEEPING);
|
|
461
458
|
else {
|
|
462
|
-
this._userInfo[e] && (this._userInfo[e].state =
|
|
463
|
-
const n = t ===
|
|
464
|
-
this._notifyStateChange(
|
|
459
|
+
this._userInfo[e] && (this._userInfo[e].state = m.NONE), this._notifyUserStateChange(this._userInfo[e]);
|
|
460
|
+
const n = t === l.REJECTED_BY_BLACKLIST ? g.ADDED_TO_BLACKLIST : g.NETWORK_ERROR;
|
|
461
|
+
this._notifyStateChange(T.END, n);
|
|
465
462
|
}
|
|
466
463
|
return { code: t };
|
|
467
464
|
}
|
|
@@ -471,59 +468,58 @@ class Y {
|
|
|
471
468
|
*/
|
|
472
469
|
async invite(e, t = {}) {
|
|
473
470
|
if (this._conversationType !== D.GROUP)
|
|
474
|
-
return { code:
|
|
471
|
+
return { code: l.CONVERSATION_NOT_GROUP_ERROR };
|
|
475
472
|
this._logger.debug("_", `[RCCallStateMachine] invite -> userIds: ${JSON.stringify(e)}`);
|
|
476
|
-
const i = this._context.getCurrentId(), n = Object.keys(this._userInfo), o = n.map((
|
|
477
|
-
let
|
|
478
|
-
return (e.includes(
|
|
479
|
-
userId:
|
|
473
|
+
const i = this._context.getCurrentId(), n = Object.keys(this._userInfo), o = n.map((_) => {
|
|
474
|
+
let u = b.CONNECTED;
|
|
475
|
+
return (e.includes(_) || this._userInfo[_].state === m.WAITING) && (u = b.RINGING), {
|
|
476
|
+
userId: _,
|
|
480
477
|
mediaType: this._mediaType,
|
|
481
|
-
callStatus:
|
|
482
|
-
mediaId:
|
|
478
|
+
callStatus: u,
|
|
479
|
+
mediaId: _
|
|
483
480
|
};
|
|
484
|
-
}), r =
|
|
481
|
+
}), { extra: r = "", pushConfig: a } = t, { code: h, message: c } = await this._callMsgHandler.sendMemeberModify({
|
|
485
482
|
channelId: this._channelId,
|
|
486
483
|
conversationType: this._conversationType,
|
|
487
484
|
targetId: this._targetId,
|
|
488
485
|
callId: this._callId,
|
|
489
486
|
extra: r,
|
|
490
|
-
|
|
491
|
-
pushContent: c,
|
|
487
|
+
pushConfig: a,
|
|
492
488
|
mediaType: this._mediaType,
|
|
493
489
|
// 除自己,其他所有人均需收到 modify 消息,以此更新状态
|
|
494
|
-
inviteUserIds: e.filter((
|
|
490
|
+
inviteUserIds: e.filter((_) => _ !== i),
|
|
495
491
|
callerId: this.getCallerId(),
|
|
496
492
|
existedUserPofiles: o,
|
|
497
|
-
directionalUserIdList: [...n, ...e].filter((
|
|
493
|
+
directionalUserIdList: [...n, ...e].filter((_) => _ !== i)
|
|
498
494
|
});
|
|
499
|
-
if (
|
|
500
|
-
const { sentTime:
|
|
501
|
-
e.forEach((
|
|
502
|
-
this._userInfo[
|
|
503
|
-
userId:
|
|
504
|
-
state:
|
|
495
|
+
if (h === l.SUCCESS) {
|
|
496
|
+
const { sentTime: _ } = c;
|
|
497
|
+
e.forEach((u) => {
|
|
498
|
+
this._userInfo[u] = {
|
|
499
|
+
userId: u,
|
|
500
|
+
state: m.WAITING,
|
|
505
501
|
isCaller: !1,
|
|
506
502
|
isRemote: !0
|
|
507
|
-
}, this._notifyUserStateChange(this._userInfo[
|
|
508
|
-
this._userInfo[
|
|
509
|
-
}, this._getTimeout(
|
|
503
|
+
}, this._notifyUserStateChange(this._userInfo[u]), this._userTimers[u] = new V(() => {
|
|
504
|
+
this._userInfo[u] && (this._userInfo[u].state = m.NONE), this._notifyUserStateChange(this._userInfo[u], g.REMOTE_NO_RESPONSE), this._watchers.onHungup(this._userInfo[u], g.REMOTE_NO_RESPONSE), delete this._userInfo[u], Object.keys(this._userInfo).length < 2 && this._notifyStateChange(T.END, g.REMOTE_NO_RESPONSE);
|
|
505
|
+
}, this._getTimeout(_));
|
|
510
506
|
});
|
|
511
507
|
} else
|
|
512
|
-
e.forEach((
|
|
513
|
-
this._userInfo[
|
|
514
|
-
userId:
|
|
515
|
-
state:
|
|
508
|
+
e.forEach((_) => {
|
|
509
|
+
this._userInfo[_] = {
|
|
510
|
+
userId: _,
|
|
511
|
+
state: m.NONE,
|
|
516
512
|
isCaller: !1,
|
|
517
513
|
isRemote: !0
|
|
518
514
|
};
|
|
519
|
-
const
|
|
520
|
-
this._notifyUserStateChange(this._userInfo[
|
|
515
|
+
const u = h === l.REJECTED_BY_BLACKLIST ? g.ADDED_TO_BLACKLIST : g.NETWORK_ERROR;
|
|
516
|
+
this._notifyUserStateChange(this._userInfo[_], u);
|
|
521
517
|
});
|
|
522
|
-
return { code:
|
|
518
|
+
return { code: h };
|
|
523
519
|
}
|
|
524
520
|
async _hungupHandle(e) {
|
|
525
521
|
const t = this._context.getCurrentId();
|
|
526
|
-
let i =
|
|
522
|
+
let i = l.SUCCESS;
|
|
527
523
|
const n = {
|
|
528
524
|
channelId: this._channelId,
|
|
529
525
|
conversationType: this._conversationType,
|
|
@@ -531,8 +527,7 @@ class Y {
|
|
|
531
527
|
callId: this._callId,
|
|
532
528
|
reason: e,
|
|
533
529
|
userIds: this.getRemoteUserIds(),
|
|
534
|
-
|
|
535
|
-
pushContent: this._hungupPushContent
|
|
530
|
+
pushConfig: this._hungupPushConfig
|
|
536
531
|
};
|
|
537
532
|
if (e === g.OTHER_CLIENT_JOINED_CALL)
|
|
538
533
|
this._callMsgHandler.sendHungup(n);
|
|
@@ -542,20 +537,20 @@ class Y {
|
|
|
542
537
|
}
|
|
543
538
|
this._endTimestamp = Date.now();
|
|
544
539
|
for (const o in this._userInfo)
|
|
545
|
-
this._userInfo[o].state =
|
|
546
|
-
return Object.keys(this._userInfo).length < 2 && this._notifyStateChange(
|
|
540
|
+
this._userInfo[o].state = m.NONE, o === t ? this._notifyUserStateChange(this._userInfo[o], e) : this._notifyUserStateChange(this._userInfo[o]), delete this._userInfo[o];
|
|
541
|
+
return Object.keys(this._userInfo).length < 2 && this._notifyStateChange(T.END, e), { code: i };
|
|
547
542
|
}
|
|
548
543
|
/**
|
|
549
544
|
* 挂断
|
|
550
545
|
*/
|
|
551
|
-
async hungup(
|
|
546
|
+
async hungup() {
|
|
552
547
|
this._logger.debug("_", "[RCCallStateMachine] hungup");
|
|
553
|
-
const
|
|
554
|
-
let
|
|
555
|
-
Object.keys(this._userTimers).length > 0 && (this._userInfo[
|
|
556
|
-
for (const
|
|
557
|
-
this._clearTimerById(
|
|
558
|
-
return this._hungupHandle(
|
|
548
|
+
const e = this._context.getCurrentId();
|
|
549
|
+
let t = g.HANGUP;
|
|
550
|
+
Object.keys(this._userTimers).length > 0 && (this._userInfo[e].isCaller ? t = g.CANCEL : this._userInfo[e].state === m.WAITING && (t = g.REJECT));
|
|
551
|
+
for (const i in this._userTimers)
|
|
552
|
+
this._clearTimerById(i);
|
|
553
|
+
return this._hungupHandle(t);
|
|
559
554
|
}
|
|
560
555
|
/**
|
|
561
556
|
* 修改通话媒体类型
|
|
@@ -571,7 +566,7 @@ class Y {
|
|
|
571
566
|
mediaType: e,
|
|
572
567
|
userIds: this.getRemoteUserIds()
|
|
573
568
|
});
|
|
574
|
-
return t ===
|
|
569
|
+
return t === l.SUCCESS && (this._mediaType = e), { code: t };
|
|
575
570
|
}
|
|
576
571
|
/**
|
|
577
572
|
* 用户加入通话补偿机制(rtc userJoin 事件触发)
|
|
@@ -581,7 +576,7 @@ class Y {
|
|
|
581
576
|
this._logger.debug("_", `[RCCallStateMachine] userJoin -> userIds: ${JSON.stringify(e)}`), setTimeout(() => {
|
|
582
577
|
e.forEach((t) => {
|
|
583
578
|
const i = this._userInfo[t];
|
|
584
|
-
i && i.state !==
|
|
579
|
+
i && i.state !== m.KEEPING && (i.state = m.KEEPING, this._notifyUserStateChange(i)), this._sessionState !== T.KEEPING && this._notifyStateChange(T.KEEPING), this._clearTimerById(t);
|
|
585
580
|
});
|
|
586
581
|
}, 300);
|
|
587
582
|
}
|
|
@@ -593,10 +588,10 @@ class Y {
|
|
|
593
588
|
this._logger.debug("_", `[RCCallStateMachine] userLeave -> userIds: ${JSON.stringify(e)}`), setTimeout(() => {
|
|
594
589
|
e.forEach((t) => {
|
|
595
590
|
const i = this._userInfo[t];
|
|
596
|
-
i && i.state !==
|
|
591
|
+
i && i.state !== m.NONE && (i.state = m.NONE, this._notifyUserStateChange(i, g.REMOTE_HANGUP), this._watchers.onHungup(i, g.REMOTE_HANGUP), delete this._userInfo[t]), new V(() => {
|
|
597
592
|
const n = this.getRemoteUsers();
|
|
598
593
|
n.length === 1 && n[0].state === 1 && this._hungupHandle(g.REMOTE_NETWORK_ERROR);
|
|
599
|
-
}, 6e4), Object.keys(this._userInfo).length < 2 && this._sessionState !==
|
|
594
|
+
}, 6e4), Object.keys(this._userInfo).length < 2 && this._sessionState !== T.END && (this._endTimestamp = Date.now(), this._notifyStateChange(T.END, g.REMOTE_HANGUP));
|
|
600
595
|
});
|
|
601
596
|
}, 300);
|
|
602
597
|
}
|
|
@@ -607,8 +602,8 @@ class Y {
|
|
|
607
602
|
close(e) {
|
|
608
603
|
this._hungupHandle(e);
|
|
609
604
|
}
|
|
610
|
-
setHungupPushConfig(e
|
|
611
|
-
this.
|
|
605
|
+
setHungupPushConfig(e) {
|
|
606
|
+
this._hungupPushConfig = e;
|
|
612
607
|
}
|
|
613
608
|
/**
|
|
614
609
|
* 通话唯一标识
|
|
@@ -655,7 +650,7 @@ class Y {
|
|
|
655
650
|
* 获取房间状态
|
|
656
651
|
*/
|
|
657
652
|
getState() {
|
|
658
|
-
return this._sessionState === null ?
|
|
653
|
+
return this._sessionState === null ? T.END : this._sessionState;
|
|
659
654
|
}
|
|
660
655
|
/**
|
|
661
656
|
* 获取人员状态
|
|
@@ -702,8 +697,8 @@ class Y {
|
|
|
702
697
|
return this._logger.debug("_", `[RCCallStateMachine] getSummary -> summary: ${JSON.stringify(n)}`), n;
|
|
703
698
|
}
|
|
704
699
|
}
|
|
705
|
-
var
|
|
706
|
-
class
|
|
700
|
+
var O = /* @__PURE__ */ ((s) => (s[s.AUDIO = 1] = "AUDIO", s[s.AUDIO_VIDEO = 2] = "AUDIO_VIDEO", s))(O || {}), ae = /* @__PURE__ */ ((s) => (s[s.ADD = 1] = "ADD", s[s.REMOVE = 2] = "REMOVE", s))(ae || {}), w = /* @__PURE__ */ ((s) => (s.WEB = "Web", s.IOS = "iOS", s.ANDROID = "Android", s))(w || {});
|
|
701
|
+
class Ce {
|
|
707
702
|
constructor(e, t, i) {
|
|
708
703
|
f(this, "_messages", []);
|
|
709
704
|
f(this, "_channelId");
|
|
@@ -729,8 +724,8 @@ class fe {
|
|
|
729
724
|
targetId: o,
|
|
730
725
|
senderUserId: r,
|
|
731
726
|
content: a
|
|
732
|
-
} = e, { callId:
|
|
733
|
-
this._channelId = i, this._conversationType = n, this._targetId = o, this._callId =
|
|
727
|
+
} = e, { callId: h, mediaType: c } = a;
|
|
728
|
+
this._channelId = i, this._conversationType = n, this._targetId = o, this._callId = h, this._mediaType = c;
|
|
734
729
|
const _ = r === t;
|
|
735
730
|
this._inviterId = r, this._endReason = g.REMOTE_NO_RESPONSE, this._canGenRecord();
|
|
736
731
|
}
|
|
@@ -760,7 +755,7 @@ class fe {
|
|
|
760
755
|
*/
|
|
761
756
|
_doHungup(e) {
|
|
762
757
|
const { content: t, sentTime: i, senderUserId: n } = e, { reason: o } = t, r = this._context.getCurrentId(), a = n === r;
|
|
763
|
-
this._endReason = a ? o :
|
|
758
|
+
this._endReason = a ? o : G[o], this._endTimestamp = i, this._canGenRecord();
|
|
764
759
|
}
|
|
765
760
|
/**
|
|
766
761
|
* 只修改通话类型
|
|
@@ -825,11 +820,11 @@ class fe {
|
|
|
825
820
|
} while (this._messages.length > 0);
|
|
826
821
|
}
|
|
827
822
|
}
|
|
828
|
-
const
|
|
829
|
-
[
|
|
830
|
-
[
|
|
823
|
+
const Ee = ["RC:VCAccept", "RC:VCRinging", "RC:VCSummary", "RC:VCHangup", "RC:VCInvite", "RC:VCModifyMedia", "RC:VCModifyMem"], me = {
|
|
824
|
+
[W.REJECTED_BY_BLACKLIST]: l.REJECTED_BY_BLACKLIST,
|
|
825
|
+
[W.NOT_IN_GROUP]: l.NOT_IN_GROUP
|
|
831
826
|
};
|
|
832
|
-
class
|
|
827
|
+
class Te extends ie {
|
|
833
828
|
constructor(t, i, n, o = 60 * 1e3, r) {
|
|
834
829
|
super();
|
|
835
830
|
f(this, "_watchers", {});
|
|
@@ -838,18 +833,18 @@ class Re extends se {
|
|
|
838
833
|
f(this, "_hadHandleMsgTimer", !1);
|
|
839
834
|
f(this, "_offlineRecorder");
|
|
840
835
|
f(this, "_deviceId", "");
|
|
841
|
-
this._context = t, this._runtime = i, this._logger = n, this._offlineMsgItv = o, this._getStateMachine = r, this._deviceId =
|
|
836
|
+
this._context = t, this._runtime = i, this._logger = n, this._offlineMsgItv = o, this._getStateMachine = r, this._deviceId = Z(i), this._context.onmessage = this._onMessage.bind(this), this._offlineRecorder = new Ce(this._context, this._logger, (a) => {
|
|
842
837
|
this._logger.info("_", `[CallMessageHandler] offlineRecorder -> ${JSON.stringify(a)}`), this._watchers.onOfflineRecord && this._watchers.onOfflineRecord(a);
|
|
843
838
|
});
|
|
844
839
|
}
|
|
845
840
|
_onMessage(t) {
|
|
846
|
-
if (
|
|
841
|
+
if (Ee.includes(t.messageType)) {
|
|
847
842
|
this._logger.debug("_", `[CallMessageHandler] _onMessage -> call message: ${JSON.stringify(t)}`);
|
|
848
843
|
try {
|
|
849
844
|
const n = Date.now(), { sentTime: o } = t;
|
|
850
845
|
let r = 0;
|
|
851
|
-
this._msgBufferList.forEach(({ msg: { sentTime: a } },
|
|
852
|
-
o >= a && (r =
|
|
846
|
+
this._msgBufferList.forEach(({ msg: { sentTime: a } }, h) => {
|
|
847
|
+
o >= a && (r = h + 1);
|
|
853
848
|
}), this._msgBufferList.splice(r, 0, {
|
|
854
849
|
markTime: n,
|
|
855
850
|
msg: t
|
|
@@ -915,44 +910,44 @@ class Re extends se {
|
|
|
915
910
|
conversationType: o,
|
|
916
911
|
messageType: r,
|
|
917
912
|
sentTime: a,
|
|
918
|
-
senderUserId:
|
|
919
|
-
content: { callId:
|
|
920
|
-
} = n[0].msg,
|
|
921
|
-
if (
|
|
922
|
-
const
|
|
923
|
-
for (let
|
|
924
|
-
const
|
|
925
|
-
if (
|
|
926
|
-
|
|
913
|
+
senderUserId: h,
|
|
914
|
+
content: { callId: c, inviteUserIds: _ }
|
|
915
|
+
} = n[0].msg, u = [R.VCInvite, R.VCModifyMem].includes(r), S = this._context.getServerTime() - a, C = S < this._offlineMsgItv;
|
|
916
|
+
if (C || this._logger.warn("_", `offline msg delayTime: ${S}ms`), u) {
|
|
917
|
+
const d = [];
|
|
918
|
+
for (let E = 0; E < n.length; E++) {
|
|
919
|
+
const M = n[E].msg, { content: { callId: N } } = M;
|
|
920
|
+
if (c === N)
|
|
921
|
+
d.push(M);
|
|
927
922
|
else
|
|
928
923
|
break;
|
|
929
924
|
}
|
|
930
|
-
if (this._logger.warn("_", `taskMsgList length: ${
|
|
931
|
-
const
|
|
932
|
-
this._msgBufferList = this._msgBufferList.slice(
|
|
933
|
-
const
|
|
934
|
-
n = n.slice(
|
|
925
|
+
if (this._logger.warn("_", `taskMsgList length: ${d.length}`), d.length > 0) {
|
|
926
|
+
const E = this._msgBufferList.findIndex((N) => N.msg.messageUId === d[d.length - 1].messageUId);
|
|
927
|
+
this._msgBufferList = this._msgBufferList.slice(E + 1);
|
|
928
|
+
const M = n.findIndex((N) => N.msg.messageUId === d[d.length - 1].messageUId);
|
|
929
|
+
n = n.slice(M + 1);
|
|
935
930
|
}
|
|
936
|
-
const
|
|
931
|
+
const y = (() => {
|
|
937
932
|
if (o !== D.PRIVATE)
|
|
938
933
|
return !1;
|
|
939
|
-
const
|
|
940
|
-
return
|
|
941
|
-
})(),
|
|
934
|
+
const E = d.length === 1, M = d.every((N) => [R.VCInvite, R.VCModifyMedia, R.VCRinging].includes(N.messageType));
|
|
935
|
+
return E || M;
|
|
936
|
+
})(), U = (() => {
|
|
942
937
|
if (o !== D.GROUP)
|
|
943
938
|
return !1;
|
|
944
|
-
let
|
|
945
|
-
for (let
|
|
946
|
-
const { senderUserId:
|
|
947
|
-
if (
|
|
939
|
+
let E = !1, M = !0, N = [h, ..._], A = !1;
|
|
940
|
+
for (let v = 0; v < d.length; v++) {
|
|
941
|
+
const { senderUserId: I, messageType: L } = d[v];
|
|
942
|
+
if (L === R.VCHangup && I === this._context.getCurrentId())
|
|
948
943
|
break;
|
|
949
|
-
|
|
944
|
+
L === R.VCHangup && (A = h === I, N = N.filter((k) => I !== k)), L === R.VCAccept && (M = !1);
|
|
950
945
|
}
|
|
951
|
-
return !(
|
|
946
|
+
return !(M && A) && N.length > 1 && (E = !0), E;
|
|
952
947
|
})();
|
|
953
|
-
|
|
948
|
+
C && (y || U) && d.forEach(this._onRecvOnlineCallMsg, this), this._offlineRecorder.onRecvOfflineMsgs(d);
|
|
954
949
|
} else
|
|
955
|
-
|
|
950
|
+
C && this._getStateMachine(c) ? this._onRecvOnlineCallMsg(n[0].msg) : this._logger.debug("_", `[CallMessageHandler] unexcepted offline msg -> ${JSON.stringify(n[0].msg)}`), n.shift(), this._msgBufferList.shift();
|
|
956
951
|
} while (n.length > 0);
|
|
957
952
|
} else {
|
|
958
953
|
i.forEach(({ msg: o }) => {
|
|
@@ -991,38 +986,27 @@ class Re extends se {
|
|
|
991
986
|
targetId: o,
|
|
992
987
|
content: r,
|
|
993
988
|
messageType: a,
|
|
994
|
-
directionalUserIdList:
|
|
995
|
-
|
|
996
|
-
|
|
997
|
-
pushContent: d
|
|
998
|
-
} = t, I = {
|
|
989
|
+
directionalUserIdList: h,
|
|
990
|
+
pushConfig: c
|
|
991
|
+
} = t, _ = {
|
|
999
992
|
channelId: i,
|
|
1000
993
|
messageType: a,
|
|
1001
994
|
content: r,
|
|
1002
|
-
directionalUserIdList:
|
|
995
|
+
directionalUserIdList: h
|
|
1003
996
|
};
|
|
1004
997
|
if ([R.VCInvite, R.VCModifyMem, R.VCHangup].includes(a)) {
|
|
1005
|
-
const
|
|
1006
|
-
|
|
1007
|
-
|
|
1008
|
-
|
|
1009
|
-
|
|
1010
|
-
|
|
1011
|
-
|
|
1012
|
-
},
|
|
1013
|
-
iOSConfig: {
|
|
1014
|
-
apnsCollapseId: r.callId
|
|
1015
|
-
},
|
|
1016
|
-
disablePushTitle: !1,
|
|
1017
|
-
forceShowDetailContent: !1,
|
|
1018
|
-
templateId: ""
|
|
1019
|
-
};
|
|
1020
|
-
I.pushConfig = N;
|
|
998
|
+
const C = c || { pushTitle: "", pushContent: "", pushData: "" };
|
|
999
|
+
C.androidConfig = Object.assign((c == null ? void 0 : c.androidConfig) || {}, {
|
|
1000
|
+
categoryHW: "VOIP",
|
|
1001
|
+
categoryVivo: "IM"
|
|
1002
|
+
}), C.iOSConfig = Object.assign(C.iOSConfig || {}, {
|
|
1003
|
+
apnsCollapseId: r.callId
|
|
1004
|
+
}), C.disablePushTitle = !1, C.forceShowDetailContent = !1, _.pushConfig = C;
|
|
1021
1005
|
}
|
|
1022
|
-
const { code:
|
|
1023
|
-
return
|
|
1024
|
-
code:
|
|
1025
|
-
}) : { code:
|
|
1006
|
+
const { code: u, data: S } = await this._context.sendMessage(n, o, _);
|
|
1007
|
+
return u !== W.SUCCESS ? (this._logger.error("_", `CallMessageHandler] sendCallMesage error -> code: ${u}`), {
|
|
1008
|
+
code: me[u] || l.SEND_MSG_ERROR
|
|
1009
|
+
}) : { code: l.SUCCESS, message: S };
|
|
1026
1010
|
}
|
|
1027
1011
|
/**
|
|
1028
1012
|
* 发送邀请消息
|
|
@@ -1034,14 +1018,13 @@ class Re extends se {
|
|
|
1034
1018
|
conversationType: o,
|
|
1035
1019
|
targetId: r,
|
|
1036
1020
|
callId: a,
|
|
1037
|
-
mediaType:
|
|
1038
|
-
inviteUserIds:
|
|
1021
|
+
mediaType: h,
|
|
1022
|
+
inviteUserIds: c,
|
|
1039
1023
|
extra: _,
|
|
1040
|
-
|
|
1041
|
-
pushContent: I
|
|
1024
|
+
pushConfig: u
|
|
1042
1025
|
} = t;
|
|
1043
1026
|
this._logger.warn("_", "CallMessageHandler] sendCallMesage sendInvite", JSON.stringify(t)), this._watchers.sendAccept && this._watchers.sendAccept({ callId: a });
|
|
1044
|
-
const
|
|
1027
|
+
const S = {
|
|
1045
1028
|
platform: w.WEB,
|
|
1046
1029
|
deviceId: this._deviceId,
|
|
1047
1030
|
callId: a,
|
|
@@ -1049,25 +1032,23 @@ class Re extends se {
|
|
|
1049
1032
|
extra: _,
|
|
1050
1033
|
engineType: 4,
|
|
1051
1034
|
channelInfo: { Id: a, Key: "" },
|
|
1052
|
-
mediaType:
|
|
1053
|
-
inviteUserIds:
|
|
1035
|
+
mediaType: h,
|
|
1036
|
+
inviteUserIds: c,
|
|
1054
1037
|
observerUserIds: [],
|
|
1055
1038
|
user: this._userInfo
|
|
1056
1039
|
};
|
|
1057
|
-
return
|
|
1040
|
+
return u && (u.pushData = JSON.stringify({
|
|
1041
|
+
mediaType: h,
|
|
1042
|
+
userIdList: c,
|
|
1043
|
+
callId: a
|
|
1044
|
+
})), this._sendCallMessage({
|
|
1058
1045
|
channelId: n,
|
|
1059
1046
|
conversationType: o,
|
|
1060
1047
|
targetId: r,
|
|
1061
|
-
content:
|
|
1048
|
+
content: S,
|
|
1062
1049
|
messageType: R.VCInvite,
|
|
1063
|
-
directionalUserIdList: o === D.GROUP ?
|
|
1064
|
-
|
|
1065
|
-
pushContent: I,
|
|
1066
|
-
pushData: JSON.stringify({
|
|
1067
|
-
mediaType: c,
|
|
1068
|
-
userIdList: l,
|
|
1069
|
-
callId: a
|
|
1070
|
-
})
|
|
1050
|
+
directionalUserIdList: o === D.GROUP ? c : [r],
|
|
1051
|
+
pushConfig: u
|
|
1071
1052
|
});
|
|
1072
1053
|
}
|
|
1073
1054
|
/**
|
|
@@ -1080,43 +1061,40 @@ class Re extends se {
|
|
|
1080
1061
|
targetId: o,
|
|
1081
1062
|
callId: r,
|
|
1082
1063
|
mediaType: a,
|
|
1083
|
-
inviteUserIds:
|
|
1084
|
-
callerId:
|
|
1064
|
+
inviteUserIds: h,
|
|
1065
|
+
callerId: c,
|
|
1085
1066
|
existedUserPofiles: _,
|
|
1086
|
-
directionalUserIdList:
|
|
1087
|
-
extra:
|
|
1088
|
-
|
|
1089
|
-
|
|
1090
|
-
} = t, N = {
|
|
1067
|
+
directionalUserIdList: u,
|
|
1068
|
+
extra: S,
|
|
1069
|
+
pushConfig: C
|
|
1070
|
+
} = t, d = {
|
|
1091
1071
|
platform: w.WEB,
|
|
1092
1072
|
// TODO 与 IM 一致
|
|
1093
1073
|
deviceId: this._deviceId,
|
|
1094
1074
|
callId: r,
|
|
1095
|
-
extra:
|
|
1075
|
+
extra: S,
|
|
1096
1076
|
engineType: 4,
|
|
1097
1077
|
channelInfo: { Id: r, Key: "" },
|
|
1098
1078
|
mediaType: a,
|
|
1099
|
-
inviteUserIds:
|
|
1079
|
+
inviteUserIds: h,
|
|
1100
1080
|
observerUserIds: [],
|
|
1101
1081
|
user: this._userInfo,
|
|
1102
|
-
caller:
|
|
1103
|
-
modifyMemType:
|
|
1082
|
+
caller: c,
|
|
1083
|
+
modifyMemType: ae.ADD,
|
|
1104
1084
|
existedUserPofiles: _
|
|
1105
1085
|
};
|
|
1106
|
-
return
|
|
1086
|
+
return C && (C.pushData = JSON.stringify({
|
|
1087
|
+
mediaType: a,
|
|
1088
|
+
userIdList: h,
|
|
1089
|
+
callId: r
|
|
1090
|
+
})), this._sendCallMessage({
|
|
1107
1091
|
channelId: i,
|
|
1108
1092
|
conversationType: n,
|
|
1109
1093
|
targetId: o,
|
|
1110
|
-
content:
|
|
1094
|
+
content: d,
|
|
1111
1095
|
messageType: R.VCModifyMem,
|
|
1112
|
-
directionalUserIdList:
|
|
1113
|
-
|
|
1114
|
-
pushContent: u,
|
|
1115
|
-
pushData: JSON.stringify({
|
|
1116
|
-
mediaType: a,
|
|
1117
|
-
userIdList: c,
|
|
1118
|
-
callId: r
|
|
1119
|
-
})
|
|
1096
|
+
directionalUserIdList: u,
|
|
1097
|
+
pushConfig: C
|
|
1120
1098
|
});
|
|
1121
1099
|
}
|
|
1122
1100
|
/**
|
|
@@ -1129,7 +1107,7 @@ class Re extends se {
|
|
|
1129
1107
|
targetId: o,
|
|
1130
1108
|
callId: r,
|
|
1131
1109
|
userIds: a
|
|
1132
|
-
} = t,
|
|
1110
|
+
} = t, h = {
|
|
1133
1111
|
platform: w.WEB,
|
|
1134
1112
|
deviceId: this._deviceId,
|
|
1135
1113
|
callId: r,
|
|
@@ -1139,7 +1117,7 @@ class Re extends se {
|
|
|
1139
1117
|
channelId: i,
|
|
1140
1118
|
conversationType: n,
|
|
1141
1119
|
targetId: o,
|
|
1142
|
-
content:
|
|
1120
|
+
content: h,
|
|
1143
1121
|
messageType: R.VCRinging,
|
|
1144
1122
|
directionalUserIdList: a
|
|
1145
1123
|
});
|
|
@@ -1154,8 +1132,8 @@ class Re extends se {
|
|
|
1154
1132
|
targetId: o,
|
|
1155
1133
|
callId: r,
|
|
1156
1134
|
mediaType: a,
|
|
1157
|
-
userIds:
|
|
1158
|
-
} = t,
|
|
1135
|
+
userIds: h
|
|
1136
|
+
} = t, c = {
|
|
1159
1137
|
platform: w.WEB,
|
|
1160
1138
|
deviceId: this._deviceId,
|
|
1161
1139
|
callId: r,
|
|
@@ -1166,9 +1144,9 @@ class Re extends se {
|
|
|
1166
1144
|
channelId: i,
|
|
1167
1145
|
conversationType: n,
|
|
1168
1146
|
targetId: o,
|
|
1169
|
-
content:
|
|
1147
|
+
content: c,
|
|
1170
1148
|
messageType: R.VCAccept,
|
|
1171
|
-
directionalUserIdList:
|
|
1149
|
+
directionalUserIdList: h
|
|
1172
1150
|
});
|
|
1173
1151
|
}
|
|
1174
1152
|
/**
|
|
@@ -1181,29 +1159,26 @@ class Re extends se {
|
|
|
1181
1159
|
targetId: o,
|
|
1182
1160
|
callId: r,
|
|
1183
1161
|
reason: a,
|
|
1184
|
-
userIds:
|
|
1185
|
-
|
|
1186
|
-
|
|
1187
|
-
} = t, d = {
|
|
1162
|
+
userIds: h,
|
|
1163
|
+
pushConfig: c
|
|
1164
|
+
} = t, _ = {
|
|
1188
1165
|
platform: w.WEB,
|
|
1189
1166
|
deviceId: this._deviceId,
|
|
1190
1167
|
callId: r,
|
|
1191
1168
|
reason: a,
|
|
1192
1169
|
user: this._userInfo
|
|
1193
1170
|
};
|
|
1194
|
-
return
|
|
1171
|
+
return c && (c.pushData = JSON.stringify({
|
|
1172
|
+
callId: r,
|
|
1173
|
+
reason: a
|
|
1174
|
+
})), this._sendCallMessage({
|
|
1195
1175
|
channelId: i,
|
|
1196
1176
|
conversationType: n,
|
|
1197
1177
|
targetId: o,
|
|
1198
|
-
content:
|
|
1178
|
+
content: _,
|
|
1199
1179
|
messageType: R.VCHangup,
|
|
1200
|
-
|
|
1201
|
-
|
|
1202
|
-
pushData: JSON.stringify({
|
|
1203
|
-
callId: r,
|
|
1204
|
-
reason: a
|
|
1205
|
-
}),
|
|
1206
|
-
directionalUserIdList: c
|
|
1180
|
+
pushConfig: c,
|
|
1181
|
+
directionalUserIdList: h
|
|
1207
1182
|
});
|
|
1208
1183
|
}
|
|
1209
1184
|
/**
|
|
@@ -1216,8 +1191,8 @@ class Re extends se {
|
|
|
1216
1191
|
targetId: o,
|
|
1217
1192
|
callId: r,
|
|
1218
1193
|
mediaType: a,
|
|
1219
|
-
userIds:
|
|
1220
|
-
} = t,
|
|
1194
|
+
userIds: h
|
|
1195
|
+
} = t, c = {
|
|
1221
1196
|
platform: w.WEB,
|
|
1222
1197
|
deviceId: this._deviceId,
|
|
1223
1198
|
callId: r,
|
|
@@ -1228,47 +1203,47 @@ class Re extends se {
|
|
|
1228
1203
|
channelId: i,
|
|
1229
1204
|
conversationType: n,
|
|
1230
1205
|
targetId: o,
|
|
1231
|
-
content:
|
|
1206
|
+
content: c,
|
|
1232
1207
|
messageType: R.VCModifyMedia,
|
|
1233
|
-
directionalUserIdList:
|
|
1208
|
+
directionalUserIdList: h
|
|
1234
1209
|
});
|
|
1235
1210
|
}
|
|
1236
1211
|
}
|
|
1237
1212
|
var P = /* @__PURE__ */ ((s) => (s.ZH = "zh", s.EN = "en", s))(P || {});
|
|
1238
|
-
const
|
|
1213
|
+
const Se = {
|
|
1239
1214
|
PushTitle: {
|
|
1240
1215
|
AUDIO: "You have a voice call",
|
|
1241
1216
|
VIDEO: "You have a video call"
|
|
1242
1217
|
}
|
|
1243
|
-
},
|
|
1218
|
+
}, pe = {
|
|
1244
1219
|
PushTitle: {
|
|
1245
1220
|
AUDIO: "您有一条音频通话",
|
|
1246
1221
|
VIDEO: "您有一条视频通话"
|
|
1247
1222
|
}
|
|
1248
1223
|
};
|
|
1249
|
-
class
|
|
1224
|
+
class ce {
|
|
1250
1225
|
static set(e) {
|
|
1251
1226
|
this._lang = e;
|
|
1252
1227
|
}
|
|
1253
1228
|
static get() {
|
|
1254
|
-
return this._lang === P.EN ?
|
|
1229
|
+
return this._lang === P.EN ? Se : pe;
|
|
1255
1230
|
}
|
|
1256
1231
|
}
|
|
1257
|
-
f(
|
|
1258
|
-
class
|
|
1232
|
+
f(ce, "_lang", P.ZH);
|
|
1233
|
+
class Oe {
|
|
1259
1234
|
/**
|
|
1260
1235
|
* 初始化
|
|
1261
1236
|
*/
|
|
1262
1237
|
constructor(e, t, i, n, o) {
|
|
1263
1238
|
f(this, "_stateMachine", {});
|
|
1264
1239
|
f(this, "_callMsgHandler");
|
|
1265
|
-
this._context = e, this._runtime = t, this._logger = i, this._watchers = n, this._options = o, this._logger.warn("_", "RCCallEngine Version: 5.2.
|
|
1240
|
+
this._context = e, this._runtime = t, this._logger = i, this._watchers = n, this._options = o, this._logger.warn("_", "RCCallEngine Version: 5.2.4 CommitId: 0a7b424ef05373eeea5ff3c35af1286656a750fd"), Z(t), this._callMsgHandler = new Te(this._context, this._runtime, this._logger, this._options.offlineMsgItv, this._getStateMachine.bind(this)), this._callMsgHandler.registerEventListener({
|
|
1266
1241
|
onInvite: this._onInvite.bind(this),
|
|
1267
1242
|
sendAccept: this._handleSendAccept.bind(this),
|
|
1268
1243
|
onOfflineRecord: this._watchers.onOfflineRecord
|
|
1269
|
-
}),
|
|
1244
|
+
}), re.on("onStateMachineClose", (r) => {
|
|
1270
1245
|
delete this._stateMachine[r];
|
|
1271
|
-
}),
|
|
1246
|
+
}), ce.set(o.lang);
|
|
1272
1247
|
}
|
|
1273
1248
|
_onInvite(e) {
|
|
1274
1249
|
const {
|
|
@@ -1278,21 +1253,21 @@ class me {
|
|
|
1278
1253
|
content: o,
|
|
1279
1254
|
messageType: r,
|
|
1280
1255
|
senderUserId: a,
|
|
1281
|
-
pushConfig:
|
|
1256
|
+
pushConfig: h
|
|
1282
1257
|
} = e;
|
|
1283
1258
|
this._logger.warn("_", `RCCallEngine _onInvite:targetId ${n} senderUserId: ${a}`);
|
|
1284
1259
|
const {
|
|
1285
|
-
mediaType:
|
|
1260
|
+
mediaType: c,
|
|
1286
1261
|
callId: _,
|
|
1287
|
-
extra:
|
|
1288
|
-
roomType:
|
|
1262
|
+
extra: u,
|
|
1263
|
+
roomType: S
|
|
1289
1264
|
} = o;
|
|
1290
|
-
let
|
|
1291
|
-
|
|
1292
|
-
const
|
|
1293
|
-
if (this._context.getCurrentId() ===
|
|
1265
|
+
let C;
|
|
1266
|
+
S !== $.RCCallRoomTypeAcrossCall ? C = a : [, C] = a.split("_");
|
|
1267
|
+
const d = this._context.getCurrentId();
|
|
1268
|
+
if (this._context.getCurrentId() === C || r === R.VCModifyMem && o.existedUserPofiles.some((E) => E.userId === d) && !this._stateMachine[_])
|
|
1294
1269
|
return;
|
|
1295
|
-
this._stateMachine[_] ? r === R.VCModifyMem && this._stateMachine[_].__onMemberModify(e) : (this._stateMachine[_] = new
|
|
1270
|
+
this._stateMachine[_] ? r === R.VCModifyMem && this._stateMachine[_].__onMemberModify(e) : (this._stateMachine[_] = new F(
|
|
1296
1271
|
this._context,
|
|
1297
1272
|
this._runtime,
|
|
1298
1273
|
this._logger,
|
|
@@ -1300,9 +1275,9 @@ class me {
|
|
|
1300
1275
|
t,
|
|
1301
1276
|
i,
|
|
1302
1277
|
n,
|
|
1303
|
-
|
|
1278
|
+
c,
|
|
1304
1279
|
_
|
|
1305
|
-
), this._logger.info("_", `[RCCallEngine] RCCallStateMachine successfully created -> callId: ${_}`), r === R.VCInvite ? this._stateMachine[_].__onInvite(e) : r === R.VCModifyMem && this._stateMachine[_].__onMemberModify(e), this._watchers.onInvite(this._stateMachine[_],
|
|
1280
|
+
), this._logger.info("_", `[RCCallEngine] RCCallStateMachine successfully created -> callId: ${_}`), r === R.VCInvite ? this._stateMachine[_].__onInvite(e) : r === R.VCModifyMem && this._stateMachine[_].__onMemberModify(e), this._watchers.onInvite(this._stateMachine[_], u), Object.keys(this._stateMachine).filter((E) => _ !== E).length > 0 && !this._options.isAllowAcceptNewCall && this._stateMachine[_].__handleInviteInSession());
|
|
1306
1281
|
}
|
|
1307
1282
|
/**
|
|
1308
1283
|
* 允许接听新的通话时,接听完新的通话后,挂断其他通话
|
|
@@ -1332,22 +1307,20 @@ class me {
|
|
|
1332
1307
|
* @param targetId 对方 ID
|
|
1333
1308
|
* @param mediaType 媒体类型
|
|
1334
1309
|
* @param extra 消息扩展信息
|
|
1335
|
-
* @param
|
|
1336
|
-
* @param pushContent 通知的内容
|
|
1310
|
+
* @param pushConfig 推送配置
|
|
1337
1311
|
*/
|
|
1338
|
-
async call(e, t, i, n = "", o
|
|
1312
|
+
async call(e, t, i, n = "", o, r = !1) {
|
|
1339
1313
|
this._logger.debug("_", `[RCCallEngine] call -> args: ${JSON.stringify({
|
|
1340
1314
|
channelId: e,
|
|
1341
1315
|
targetId: t,
|
|
1342
1316
|
mediaType: i,
|
|
1343
1317
|
extra: n,
|
|
1344
|
-
|
|
1345
|
-
pushContent: r
|
|
1318
|
+
pushConfig: o
|
|
1346
1319
|
})}`);
|
|
1347
|
-
const
|
|
1320
|
+
const a = ee();
|
|
1348
1321
|
if (Object.keys(this._stateMachine).length > 0)
|
|
1349
|
-
return { code:
|
|
1350
|
-
this._stateMachine[
|
|
1322
|
+
return { code: l.STATE_MACHINE_EXIT };
|
|
1323
|
+
this._stateMachine[a] = new F(
|
|
1351
1324
|
this._context,
|
|
1352
1325
|
this._runtime,
|
|
1353
1326
|
this._logger,
|
|
@@ -1356,13 +1329,13 @@ class me {
|
|
|
1356
1329
|
D.PRIVATE,
|
|
1357
1330
|
t,
|
|
1358
1331
|
i,
|
|
1359
|
-
|
|
1332
|
+
a
|
|
1360
1333
|
);
|
|
1361
|
-
const { code:
|
|
1362
|
-
return
|
|
1363
|
-
code:
|
|
1364
|
-
stateMachine: this._stateMachine[
|
|
1365
|
-
} : { code:
|
|
1334
|
+
const { code: c } = await this._stateMachine[a].__call([t], n, o, r);
|
|
1335
|
+
return c === l.SUCCESS ? {
|
|
1336
|
+
code: l.SUCCESS,
|
|
1337
|
+
stateMachine: this._stateMachine[a]
|
|
1338
|
+
} : { code: c };
|
|
1366
1339
|
}
|
|
1367
1340
|
/**
|
|
1368
1341
|
* 群呼
|
|
@@ -1371,15 +1344,14 @@ class me {
|
|
|
1371
1344
|
* @param mediaType 媒体类型
|
|
1372
1345
|
* @param userIds 被邀请人员列表
|
|
1373
1346
|
* @param extra 消息扩展信息
|
|
1374
|
-
* @param
|
|
1375
|
-
* @param pushContent 通知的内容
|
|
1347
|
+
* @param pushConfig 推送配置
|
|
1376
1348
|
*/
|
|
1377
|
-
async callInGroup(e, t, i, n, o = "", r
|
|
1349
|
+
async callInGroup(e, t, i, n, o = "", r) {
|
|
1378
1350
|
this._logger.debug("_", `[RCCallEngine] callInGroup -> args: ${JSON.stringify({ channelId: e, targetId: t, mediaType: i })}`);
|
|
1379
|
-
const
|
|
1351
|
+
const a = ee();
|
|
1380
1352
|
if (Object.keys(this._stateMachine).length > 0)
|
|
1381
|
-
return { code:
|
|
1382
|
-
this._stateMachine[
|
|
1353
|
+
return { code: l.STATE_MACHINE_EXIT };
|
|
1354
|
+
this._stateMachine[a] = new F(
|
|
1383
1355
|
this._context,
|
|
1384
1356
|
this._runtime,
|
|
1385
1357
|
this._logger,
|
|
@@ -1388,13 +1360,13 @@ class me {
|
|
|
1388
1360
|
D.GROUP,
|
|
1389
1361
|
t,
|
|
1390
1362
|
i,
|
|
1391
|
-
|
|
1363
|
+
a
|
|
1392
1364
|
);
|
|
1393
|
-
const { code:
|
|
1394
|
-
return
|
|
1395
|
-
code:
|
|
1396
|
-
stateMachine: this._stateMachine[
|
|
1397
|
-
} : { code:
|
|
1365
|
+
const { code: c } = await this._stateMachine[a].__call(n, o, r);
|
|
1366
|
+
return c === l.SUCCESS ? {
|
|
1367
|
+
code: l.SUCCESS,
|
|
1368
|
+
stateMachine: this._stateMachine[a]
|
|
1369
|
+
} : { code: c };
|
|
1398
1370
|
}
|
|
1399
1371
|
/**
|
|
1400
1372
|
* 销毁当前的状态机
|
|
@@ -1404,7 +1376,7 @@ class me {
|
|
|
1404
1376
|
}
|
|
1405
1377
|
}
|
|
1406
1378
|
var x = /* @__PURE__ */ ((s) => (s[s.CALLER = 1] = "CALLER", s[s.CALLEE = 2] = "CALLEE", s))(x || {});
|
|
1407
|
-
class
|
|
1379
|
+
class ye {
|
|
1408
1380
|
constructor() {
|
|
1409
1381
|
f(this, "list", {});
|
|
1410
1382
|
}
|
|
@@ -1441,17 +1413,17 @@ class pe {
|
|
|
1441
1413
|
});
|
|
1442
1414
|
}
|
|
1443
1415
|
}
|
|
1444
|
-
const
|
|
1445
|
-
function
|
|
1416
|
+
const B = new ye();
|
|
1417
|
+
function Me(s) {
|
|
1446
1418
|
return Object.values(P).includes(s);
|
|
1447
1419
|
}
|
|
1448
|
-
function
|
|
1449
|
-
return Object.values(
|
|
1420
|
+
function Ne(s) {
|
|
1421
|
+
return Object.values(ne).includes(s);
|
|
1450
1422
|
}
|
|
1451
|
-
function
|
|
1452
|
-
return [
|
|
1423
|
+
function Ae(s) {
|
|
1424
|
+
return [K.DEBUG, K.INFO, K.WARN, K.ERROR].includes(s);
|
|
1453
1425
|
}
|
|
1454
|
-
const
|
|
1426
|
+
const Ue = (s) => {
|
|
1455
1427
|
if (!s)
|
|
1456
1428
|
return { result: !1, msg: "Initialization missing parameter -> options" };
|
|
1457
1429
|
if (typeof s != "object")
|
|
@@ -1459,8 +1431,8 @@ const Me = (s) => {
|
|
|
1459
1431
|
const e = ["rtcClient", "onSession", "onSessionClose"], t = Object.keys(s), i = [];
|
|
1460
1432
|
return e.forEach((n) => {
|
|
1461
1433
|
t.includes(n) || i.push(n);
|
|
1462
|
-
}), i.length ? { result: !1, msg: `Initialization missing parameter -> "${i.join(",")}"` } : typeof s.rtcClient != "object" ? { result: !1, msg: "Initialization 'rtcClient' parameter must be of type 'object'" } : typeof s.onSession != "function" ? { result: !1, msg: "Initialization 'onSession' parameter must be of type 'function'" } : typeof s.onSessionClose != "function" ? { result: !1, msg: "Initialization 'onSessionClose' parameter must be of type 'function'" } : typeof s.isAllowSubscribeRetry < "u" && typeof s.isAllowSubscribeRetry != "boolean" ? { result: !1, msg: "Initialization 'isAllowSubscribeRetry' parameter must be of type 'boolean'" } : typeof s.isAllowPublishRetry < "u" && typeof s.isAllowPublishRetry != "boolean" ? { result: !1, msg: "Initialization 'isAllowPublishRetry' parameter must be of type 'boolean'" } : typeof s.isOffCameraWhenVideoDisable < "u" && typeof s.isOffCameraWhenVideoDisable != "boolean" ? { result: !1, msg: "Initialization 'isOffCameraWhenVideoDisable' parameter must be of type 'boolean'" } : typeof s.joinType < "u" && !
|
|
1463
|
-
},
|
|
1434
|
+
}), i.length ? { result: !1, msg: `Initialization missing parameter -> "${i.join(",")}"` } : typeof s.rtcClient != "object" ? { result: !1, msg: "Initialization 'rtcClient' parameter must be of type 'object'" } : typeof s.onSession != "function" ? { result: !1, msg: "Initialization 'onSession' parameter must be of type 'function'" } : typeof s.onSessionClose != "function" ? { result: !1, msg: "Initialization 'onSessionClose' parameter must be of type 'function'" } : typeof s.isAllowSubscribeRetry < "u" && typeof s.isAllowSubscribeRetry != "boolean" ? { result: !1, msg: "Initialization 'isAllowSubscribeRetry' parameter must be of type 'boolean'" } : typeof s.isAllowPublishRetry < "u" && typeof s.isAllowPublishRetry != "boolean" ? { result: !1, msg: "Initialization 'isAllowPublishRetry' parameter must be of type 'boolean'" } : typeof s.isOffCameraWhenVideoDisable < "u" && typeof s.isOffCameraWhenVideoDisable != "boolean" ? { result: !1, msg: "Initialization 'isOffCameraWhenVideoDisable' parameter must be of type 'boolean'" } : typeof s.joinType < "u" && !Ne(s.joinType) ? { result: !1, msg: "Initialization 'joinType' parameter must be of type correct type" } : typeof s.isAllowDemotionGetStream < "u" && typeof s.isAllowDemotionGetStream != "boolean" ? { result: !1, msg: "Initialization 'isAllowDemotionGetStream' parameter must be of type 'boolean'" } : typeof s.lang < "u" && !Me(s.lang) ? { result: !1, msg: "Initialization 'lang' parameter must be of type correct type" } : typeof s.logOutputLevel < "u" && !Ae(s.logOutputLevel) ? { result: !1, msg: "Initialization 'logOutputLevel' parameter must be of type correct type" } : { result: !0 };
|
|
1435
|
+
}, j = (s) => {
|
|
1464
1436
|
if (!s)
|
|
1465
1437
|
return { result: !1, msg: "missing parameter -> listener" };
|
|
1466
1438
|
if (typeof s != "object")
|
|
@@ -1469,19 +1441,73 @@ const Me = (s) => {
|
|
|
1469
1441
|
return e.forEach((n) => {
|
|
1470
1442
|
t.includes(n) || i.push(n);
|
|
1471
1443
|
}), i.length ? { result: !1, msg: `missing parameter -> "${i.join(",")}"` } : typeof s.onRinging != "function" ? { result: !1, msg: "'onRinging' parameter must be of type 'function'" } : typeof s.onAccept != "function" ? { result: !1, msg: "'onAccept' parameter must be of type 'function'" } : typeof s.onHungup != "function" ? { result: !1, msg: "'onHungup' parameter must be of type 'function'" } : typeof s.onTrackReady != "function" ? { result: !1, msg: "'onTrackReady' parameter must be of type 'function'" } : { result: !0 };
|
|
1472
|
-
},
|
|
1473
|
-
const
|
|
1474
|
-
|
|
1444
|
+
}, te = (s) => s && typeof s == "string" ? { result: !0 } : { result: !1, msg: "'targetId' parameter is required, must be of type 'string'" }, se = (s) => s === O.AUDIO || s === O.AUDIO_VIDEO ? { result: !0 } : { result: !1, msg: "'mediaType' parameter is required, must be of type 'RCCallMediaType'" }, X = (s) => typeof s == "string" ? { result: !0 } : { result: !1, msg: "'extra' parameter must be of type 'string'" }, J = (s) => {
|
|
1445
|
+
const e = ["pushTitle", "pushContent", "pushData", "iOSConfig", "androidConfig", "disablePushTitle", "templateId"];
|
|
1446
|
+
if (!Object.keys(s).every((a) => e.includes(a)))
|
|
1447
|
+
return {
|
|
1448
|
+
code: l.PARAM_ERROR,
|
|
1449
|
+
errorMsg: `right key: ${e.join(",")}`
|
|
1450
|
+
};
|
|
1451
|
+
for (let a = 0; a < e.length; a++) {
|
|
1452
|
+
const h = e[a];
|
|
1453
|
+
if (h === "disablePushTitle") {
|
|
1454
|
+
if (!ge(s[h]) && !de(s[h]))
|
|
1455
|
+
return {
|
|
1456
|
+
code: l.PARAM_ERROR,
|
|
1457
|
+
errorMsg: `${h} type is boolean`
|
|
1458
|
+
};
|
|
1459
|
+
} else if ((h === "pushTitle" || h === "templateId") && s[h] && !Y(s[h]))
|
|
1460
|
+
return {
|
|
1461
|
+
code: l.PARAM_ERROR,
|
|
1462
|
+
errorMsg: `${h} type is string`
|
|
1463
|
+
};
|
|
1464
|
+
}
|
|
1465
|
+
const { iOSConfig: i, androidConfig: n } = s, o = ve(i || {});
|
|
1466
|
+
if (o.code !== l.SUCCESS)
|
|
1467
|
+
return o;
|
|
1468
|
+
const r = Le(n || {});
|
|
1469
|
+
return r.code !== l.SUCCESS ? r : { code: l.SUCCESS };
|
|
1470
|
+
}, ve = (s) => {
|
|
1471
|
+
const { threadId: e, richMediaUri: t } = s || {};
|
|
1472
|
+
return e && !Y(e) ? {
|
|
1473
|
+
code: l.PARAM_ERROR,
|
|
1474
|
+
errorMsg: "iOSConfig.threadId type is string"
|
|
1475
|
+
} : t && !Y(t) ? {
|
|
1476
|
+
code: l.PARAM_ERROR,
|
|
1477
|
+
errorMsg: "iOSConfig.richMediaUri type is string"
|
|
1478
|
+
} : { code: l.SUCCESS };
|
|
1479
|
+
}, Le = (s) => {
|
|
1480
|
+
const e = ["notificationId", "channelIdMi", "miLargeIconUrl", "channelIdHW", "categoryHW", "importanceHW", "imageUrlHW", "channelIdOPPO", "typeVivo", "categoryVivo", "fcmChannelId", "fcmCollapseKey", "fcmImageUrl", "importanceHonor", "imageUrlHonor"];
|
|
1481
|
+
if (!Object.keys(s).every((i) => e.includes(i)))
|
|
1482
|
+
return {
|
|
1483
|
+
code: l.PARAM_ERROR,
|
|
1484
|
+
errorMsg: `right key: ${e.join(",")}`
|
|
1485
|
+
};
|
|
1486
|
+
for (let i = 0; i < e.length; i++) {
|
|
1487
|
+
const n = e[i];
|
|
1488
|
+
if (n === "importanceHW" || n === "importanceHonor") {
|
|
1489
|
+
if (s[n] && s[n] !== "LOW" && s[n] !== "NORMAL")
|
|
1490
|
+
return {
|
|
1491
|
+
code: l.PARAM_ERROR,
|
|
1492
|
+
errorMsg: `androidConfig.${n} value is NORMAL|LOW`
|
|
1493
|
+
};
|
|
1494
|
+
} else if (s[n] && !Y(s[n]))
|
|
1495
|
+
return {
|
|
1496
|
+
code: l.PARAM_ERROR,
|
|
1497
|
+
errorMsg: `androidConfig.${n} type is string`
|
|
1498
|
+
};
|
|
1499
|
+
}
|
|
1500
|
+
return { code: l.SUCCESS };
|
|
1475
1501
|
}, le = (s) => Array.isArray(s) ? s.length ? s.every((e) => typeof e == "string" && e.length > 0) ? { result: !0 } : { result: !1, msg: "'userIds' parameter is required" } : { result: !1, msg: "'userIds' parameter is required, must be of type 'string[]'" } : { result: !1, msg: "'userIds' parameter is required, must be of type 'string[]'" };
|
|
1476
|
-
function
|
|
1502
|
+
function De(s) {
|
|
1477
1503
|
return ["FPS_10", "FPS_15", "FPS_24", "FPS_30"].includes(s);
|
|
1478
1504
|
}
|
|
1479
|
-
const
|
|
1480
|
-
class
|
|
1505
|
+
const ke = (s) => !!ue[s], be = (s) => s && s.audio && typeof s.audio.micphoneId < "u" && typeof s.audio.micphoneId != "string" ? { result: !1, msg: "'constraints.audio.micphoneId' must be of type 'string'" } : s && s.audio && typeof s.audio.sampleRate < "u" && typeof s.audio.sampleRate != "number" ? { result: !1, msg: "'constraints.audio.sampleRate' must be of type 'number'" } : s && s.video && typeof s.video.cameraId < "u" && typeof s.video.cameraId != "string" ? { result: !1, msg: "'constraints.video.cameraId' must be of type 'string'" } : s && s.video && typeof s.video.frameRate < "u" && typeof s.video.frameRate != "string" ? { result: !1, msg: "'constraints.video.frameRate' must be of type 'string'" } : s && s.video && typeof s.video.frameRate < "u" && !De(s.video.frameRate) ? { result: !1, msg: "'frameRate' value is out of range" } : s && s.video && typeof s.video.resolution < "u" && typeof s.video.resolution != "string" ? { result: !1, msg: "'constraints.video.frameRate' must be of type 'string'" } : s && s.video && typeof s.video.resolution < "u" && !ke(s.video.resolution) ? { result: !1, msg: "'resolution' value is out of range" } : s && s.video && (!s.video.frameRate || !s.video.resolution) ? { result: !1, msg: "'resolution' and 'resolution' is required" } : { result: !0 }, we = (s, e) => setTimeout(s, e);
|
|
1506
|
+
class Pe {
|
|
1481
1507
|
constructor(e, t) {
|
|
1482
1508
|
f(this, "_timerId", 0);
|
|
1483
1509
|
f(this, "_startTime", 0);
|
|
1484
|
-
e && (this._timerId =
|
|
1510
|
+
e && (this._timerId = we(() => {
|
|
1485
1511
|
e();
|
|
1486
1512
|
}, t)), this._startTime = Date.now();
|
|
1487
1513
|
}
|
|
@@ -1499,7 +1525,7 @@ class De {
|
|
|
1499
1525
|
this._startTime = 0;
|
|
1500
1526
|
}
|
|
1501
1527
|
}
|
|
1502
|
-
class
|
|
1528
|
+
class z {
|
|
1503
1529
|
constructor(e, t, i, n = {}) {
|
|
1504
1530
|
/**
|
|
1505
1531
|
* RTC房间实例
|
|
@@ -1522,28 +1548,28 @@ class F {
|
|
|
1522
1548
|
* 用户状态变更
|
|
1523
1549
|
* @param info
|
|
1524
1550
|
*/
|
|
1525
|
-
onUserStateChange: ({ user:
|
|
1526
|
-
this._logger.info("_", `[RCCallSession onUserStateChange] userId->${
|
|
1551
|
+
onUserStateChange: ({ user: o, reason: r }) => {
|
|
1552
|
+
this._logger.info("_", `[RCCallSession onUserStateChange] userId->${o == null ? void 0 : o.userId} state->${o == null ? void 0 : o.state} reason->${r}`);
|
|
1527
1553
|
},
|
|
1528
1554
|
/**
|
|
1529
1555
|
* 房间状态变更
|
|
1530
1556
|
* @param
|
|
1531
1557
|
*/
|
|
1532
|
-
onStateChange: async (
|
|
1533
|
-
const { state:
|
|
1534
|
-
if (this._logger.info("_", `[RCCallSession onStateChange] : state->${
|
|
1535
|
-
const
|
|
1536
|
-
this._logger.info("_", `[RCCallSession onStateChange] roomId: ${
|
|
1558
|
+
onStateChange: async (o) => {
|
|
1559
|
+
const { state: r, reason: a } = o;
|
|
1560
|
+
if (this._logger.info("_", `[RCCallSession onStateChange] : state->${r} reason->${a}`), r === T.KEEPING) {
|
|
1561
|
+
const h = this._stateMachine.getCallId();
|
|
1562
|
+
this._logger.info("_", `[RCCallSession onStateChange] roomId: ${h}`);
|
|
1537
1563
|
try {
|
|
1538
|
-
await this._joinRoom(
|
|
1539
|
-
} catch (
|
|
1540
|
-
this._exceptionClose(g.NETWORK_ERROR), this._logger.error("_", `[RCCallSession onStateChange] joinRoom throw exception roomId -> ${
|
|
1564
|
+
await this._joinRoom(h);
|
|
1565
|
+
} catch (c) {
|
|
1566
|
+
this._exceptionClose(g.NETWORK_ERROR), this._logger.error("_", `[RCCallSession onStateChange] joinRoom throw exception roomId -> ${h}`), console.error(c);
|
|
1541
1567
|
}
|
|
1542
|
-
} else if (
|
|
1568
|
+
} else if (r === T.END) {
|
|
1543
1569
|
if (!this._room) {
|
|
1544
1570
|
this._options.localTracks && this._destroyTracks(this._options.localTracks);
|
|
1545
|
-
const
|
|
1546
|
-
|
|
1571
|
+
const h = this._stateMachine.getSummary();
|
|
1572
|
+
B.emit("sessionClose", { session: this, summaryInfo: h });
|
|
1547
1573
|
return;
|
|
1548
1574
|
}
|
|
1549
1575
|
this._options.localTracks && this._destroyTracks(this._options.localTracks), this._logger.info("_", "[RCCallSession onStateChange] localTracks destroyed"), this._leaveRoom(), this._room = null;
|
|
@@ -1553,70 +1579,68 @@ class F {
|
|
|
1553
1579
|
* 收到响铃
|
|
1554
1580
|
* @param sender 发起用户信息
|
|
1555
1581
|
*/
|
|
1556
|
-
onRinging: (
|
|
1557
|
-
this._logger.info("_", `[RCCallSession onRinging]sender: sender.userId -> ${
|
|
1582
|
+
onRinging: (o) => {
|
|
1583
|
+
this._logger.info("_", `[RCCallSession onRinging]sender: sender.userId -> ${o.userId}`);
|
|
1558
1584
|
try {
|
|
1559
|
-
this._listener.onRinging(
|
|
1560
|
-
} catch (
|
|
1561
|
-
this._logger.error("_", "[RCCallSession onRinging] method exception -> onRinging"), console.error(
|
|
1585
|
+
this._listener.onRinging(o, this);
|
|
1586
|
+
} catch (r) {
|
|
1587
|
+
this._logger.error("_", "[RCCallSession onRinging] method exception -> onRinging"), console.error(r);
|
|
1562
1588
|
}
|
|
1563
1589
|
},
|
|
1564
1590
|
/**
|
|
1565
1591
|
* 当远端用户同意接听
|
|
1566
1592
|
*/
|
|
1567
|
-
onAccept: (
|
|
1568
|
-
this._logger.info("_", `[RCCallSession onAccept]sender: sender.userId -> ${
|
|
1593
|
+
onAccept: (o) => {
|
|
1594
|
+
this._logger.info("_", `[RCCallSession onAccept]sender: sender.userId -> ${o.userId}`);
|
|
1569
1595
|
try {
|
|
1570
|
-
this._listener.onAccept(
|
|
1571
|
-
} catch (
|
|
1572
|
-
this._logger.error("_", "[RCCallSession onAccept] method exception -> onAccept"), console.error(
|
|
1596
|
+
this._listener.onAccept(o, this);
|
|
1597
|
+
} catch (r) {
|
|
1598
|
+
this._logger.error("_", "[RCCallSession onAccept] method exception -> onAccept"), console.error(r);
|
|
1573
1599
|
}
|
|
1574
1600
|
},
|
|
1575
1601
|
/**
|
|
1576
1602
|
* 当有远端用户挂断
|
|
1577
1603
|
*/
|
|
1578
|
-
onHungup: (
|
|
1579
|
-
this._logger.info("_", `[RCCallSession onHungup]sender: sender.userId -> ${
|
|
1604
|
+
onHungup: (o, r) => {
|
|
1605
|
+
this._logger.info("_", `[RCCallSession onHungup]sender: sender.userId -> ${o.userId} reason->${r}`);
|
|
1580
1606
|
try {
|
|
1581
|
-
this._listener.onHungup(
|
|
1582
|
-
} catch (
|
|
1583
|
-
this._logger.error("_", "[RCCallSession onHungup] method exception -> onHungup"), console.error(
|
|
1607
|
+
this._listener.onHungup(o, r, this);
|
|
1608
|
+
} catch (a) {
|
|
1609
|
+
this._logger.error("_", "[RCCallSession onHungup] method exception -> onHungup"), console.error(a);
|
|
1584
1610
|
}
|
|
1585
1611
|
},
|
|
1586
1612
|
/**
|
|
1587
1613
|
* 收到人员变更
|
|
1588
1614
|
* @param sender 发起用户信息
|
|
1589
1615
|
*/
|
|
1590
|
-
onMemberModify: ({ sender:
|
|
1591
|
-
this._logger.info("_", `[RCCallSession onMemberModify] sender.userId -> ${
|
|
1616
|
+
onMemberModify: ({ sender: o, invitedUsers: r }) => {
|
|
1617
|
+
this._logger.info("_", `[RCCallSession onMemberModify] sender.userId -> ${o.userId}`);
|
|
1592
1618
|
try {
|
|
1593
|
-
this._listener.onMemberModify(
|
|
1594
|
-
} catch (
|
|
1595
|
-
this._logger.error("_", "[RCCallSession onMemberModify] method exception -> onMemberModify"), console.error(
|
|
1619
|
+
this._listener.onMemberModify(o, r, this);
|
|
1620
|
+
} catch (a) {
|
|
1621
|
+
this._logger.error("_", "[RCCallSession onMemberModify] method exception -> onMemberModify"), console.error(a);
|
|
1596
1622
|
}
|
|
1597
1623
|
},
|
|
1598
1624
|
/**
|
|
1599
1625
|
* 收到通话类型变更 (通话降级)
|
|
1600
1626
|
* @param sender 发起用户信息
|
|
1601
1627
|
*/
|
|
1602
|
-
onMediaModify: ({ sender:
|
|
1603
|
-
this._logger.info("_", `[RCCallSession onMediaModify]sender: sender.userId -> ${
|
|
1628
|
+
onMediaModify: ({ sender: o, mediaType: r }) => {
|
|
1629
|
+
this._logger.info("_", `[RCCallSession onMediaModify]sender: sender.userId -> ${o.userId} mediaType: ${r}`), r === O.AUDIO && this._setMediaTypeToAudio();
|
|
1604
1630
|
try {
|
|
1605
|
-
this._listener.onMediaModify(
|
|
1606
|
-
} catch (
|
|
1607
|
-
this._logger.error("_", "[RCCallSession onMediaModify] method exception -> onMediaModify"), console.error(
|
|
1631
|
+
this._listener.onMediaModify(o, r, this);
|
|
1632
|
+
} catch (a) {
|
|
1633
|
+
this._logger.error("_", "[RCCallSession onMediaModify] method exception -> onMediaModify"), console.error(a);
|
|
1608
1634
|
}
|
|
1609
1635
|
},
|
|
1610
1636
|
/**
|
|
1611
1637
|
* 是否跨appkey
|
|
1612
1638
|
* @param sender 发起用户信息
|
|
1613
1639
|
*/
|
|
1614
|
-
crossAppkey: (
|
|
1615
|
-
this._logger.info("_", `[RCCallSession crossAppkey] 是否跨 appkey: ${
|
|
1640
|
+
crossAppkey: (o) => {
|
|
1641
|
+
this._logger.info("_", `[RCCallSession crossAppkey] 是否跨 appkey: ${o}`), this._options.isCrossAppkey = o;
|
|
1616
1642
|
}
|
|
1617
|
-
});
|
|
1618
|
-
const { pushTitle: o, pushContent: r } = this._options.hungupPushConfig;
|
|
1619
|
-
this._stateMachine.setHungupPushConfig(o, r);
|
|
1643
|
+
}), this._stateMachine.setHungupPushConfig(this._options.hungupPushConfig || { pushTitle: "", pushContent: "" });
|
|
1620
1644
|
}
|
|
1621
1645
|
/**
|
|
1622
1646
|
* 加入房间
|
|
@@ -1626,28 +1650,28 @@ class F {
|
|
|
1626
1650
|
try {
|
|
1627
1651
|
this._options.isCrossAppkey ? t = await this._rtcClient.joinCrossRTCRoom(e, this._options.joinType) : t = await this._rtcClient.joinRTCRoom(e, this._options.joinType);
|
|
1628
1652
|
const { code: i, userIds: n, room: o } = t;
|
|
1629
|
-
if (i !==
|
|
1630
|
-
return i ===
|
|
1631
|
-
if (this._stateMachine.getConversationType() === D.GROUP && this._stateMachine.userJoin([this._rtcClient.getCurrentId()]), this._stateMachine.getState() ===
|
|
1632
|
-
return await this._rtcClient.leaveRoom(o), this._room = null, { code:
|
|
1633
|
-
n.length < 1 && (this.joinRoomTimer = new
|
|
1653
|
+
if (i !== p.SUCCESS)
|
|
1654
|
+
return i === p.NOT_OPEN_VIDEO_AUDIO_SERVER && this._exceptionClose(g.SERVICE_NOT_OPENED), i === p.SIGNAL_JOIN_RTC_ROOM_REFUSED ? this._exceptionClose(g.OTHER_CLIENT_IN_CALL) : this._exceptionClose(g.NETWORK_ERROR), this._logger.info("_", `[RCCallClient _joinRoom] join room failed: roomId -> ${e} RCRTCCode -> ${i}`), { code: l.JOIN_ROOM_ERROR };
|
|
1655
|
+
if (this._stateMachine.getConversationType() === D.GROUP && this._stateMachine.userJoin([this._rtcClient.getCurrentId()]), this._stateMachine.getState() === T.END)
|
|
1656
|
+
return await this._rtcClient.leaveRoom(o), this._room = null, { code: l.SUCCESS };
|
|
1657
|
+
n.length < 1 && (this.joinRoomTimer = new Pe(() => {
|
|
1634
1658
|
this._exceptionClose(g.REMOTE_NETWORK_ERROR);
|
|
1635
1659
|
}, 6e4)), this._room = o;
|
|
1636
1660
|
} catch (i) {
|
|
1637
|
-
return this._exceptionClose(g.NETWORK_ERROR), this._logger.error("_", `[RCCallSession _joinRoom] _rtcClient.joinRTCRoom throw exception roomId -> ${e}`), console.error(i), { code:
|
|
1661
|
+
return this._exceptionClose(g.NETWORK_ERROR), this._logger.error("_", `[RCCallSession _joinRoom] _rtcClient.joinRTCRoom throw exception roomId -> ${e}`), console.error(i), { code: l.JOIN_ROOM_ERROR };
|
|
1638
1662
|
}
|
|
1639
1663
|
this._registerRoomEventListener(), this._registerReportListener();
|
|
1640
1664
|
try {
|
|
1641
1665
|
await this._subscribeInRoomRemoteTrack();
|
|
1642
1666
|
} catch (i) {
|
|
1643
|
-
return this._exceptionClose(g.SUBSCRIBE_ERROR), this._logger.error("_", `[RCCallSession _joinRoom] _subscribeInRoomRemoteTrack Exception roomId -> ${e}`), console.error(i), { code:
|
|
1667
|
+
return this._exceptionClose(g.SUBSCRIBE_ERROR), this._logger.error("_", `[RCCallSession _joinRoom] _subscribeInRoomRemoteTrack Exception roomId -> ${e}`), console.error(i), { code: l.JOIN_ROOM_ERROR };
|
|
1644
1668
|
}
|
|
1645
1669
|
try {
|
|
1646
1670
|
await this._publish();
|
|
1647
1671
|
} catch (i) {
|
|
1648
|
-
return this._exceptionClose(g.PUBLISH_ERROR), this._logger.error("_", `[RCCallSession _joinRoom] _publish Exception roomId -> ${e}`), console.error(i), { code:
|
|
1672
|
+
return this._exceptionClose(g.PUBLISH_ERROR), this._logger.error("_", `[RCCallSession _joinRoom] _publish Exception roomId -> ${e}`), console.error(i), { code: l.JOIN_ROOM_ERROR };
|
|
1649
1673
|
}
|
|
1650
|
-
return { code:
|
|
1674
|
+
return { code: l.SUCCESS };
|
|
1651
1675
|
}
|
|
1652
1676
|
/**
|
|
1653
1677
|
* (初始化房间的时候) 订阅远程的流,把远程的流抛给用户
|
|
@@ -1656,7 +1680,7 @@ class F {
|
|
|
1656
1680
|
const e = this._room.getRemoteTracks();
|
|
1657
1681
|
if (e.length) {
|
|
1658
1682
|
const { code: t } = await this._subscribeRetry(e, this._options.isAllowSubscribeRetry, this._RETRYCOUNT);
|
|
1659
|
-
t !==
|
|
1683
|
+
t !== p.SUCCESS && (this._exceptionClose(g.SUBSCRIBE_ERROR), this._logger.error("_", `[RCCallSession _subscribeInRoomRemoteTrack] Resource subscription failed roomId -> ${this._stateMachine.getCallId()} RTC code -> ${t}`));
|
|
1660
1684
|
}
|
|
1661
1685
|
}
|
|
1662
1686
|
/**
|
|
@@ -1667,7 +1691,7 @@ class F {
|
|
|
1667
1691
|
*/
|
|
1668
1692
|
async _subscribeRetry(e, t = !1, i = 0) {
|
|
1669
1693
|
const { code: n } = await this._room.subscribe(e);
|
|
1670
|
-
if (n !==
|
|
1694
|
+
if (n !== p.SUCCESS) {
|
|
1671
1695
|
try {
|
|
1672
1696
|
this._listener.onTrackSubscribeFail && this._listener.onTrackSubscribeFail(n, this);
|
|
1673
1697
|
} catch (o) {
|
|
@@ -1686,7 +1710,7 @@ class F {
|
|
|
1686
1710
|
*/
|
|
1687
1711
|
async _publish() {
|
|
1688
1712
|
const e = this._options.localTracks, { code: t } = await this._publishRetry(e, this._options.isAllowPublishRetry, this._RETRYCOUNT);
|
|
1689
|
-
if (t !==
|
|
1713
|
+
if (t !== p.SUCCESS) {
|
|
1690
1714
|
this._exceptionClose(g.PUBLISH_ERROR), this._logger.info("_", `[RCCallSession _publist] Resource publishing failed: roomId -> ${this._stateMachine.getCallId()} RCRTCCode -> ${t}`);
|
|
1691
1715
|
return;
|
|
1692
1716
|
}
|
|
@@ -1700,7 +1724,7 @@ class F {
|
|
|
1700
1724
|
*/
|
|
1701
1725
|
async _publishRetry(e, t = !1, i = 0) {
|
|
1702
1726
|
const { code: n } = await this._room.publish(e);
|
|
1703
|
-
if (n !==
|
|
1727
|
+
if (n !== p.SUCCESS) {
|
|
1704
1728
|
try {
|
|
1705
1729
|
this._listener.onTrackPublishFail && this._listener.onTrackPublishFail(n, this);
|
|
1706
1730
|
} catch (o) {
|
|
@@ -1724,7 +1748,7 @@ class F {
|
|
|
1724
1748
|
this._logger.error("_", "[RCCallSession _leaveRoom] leaveRoom throw exception"), console.error(e);
|
|
1725
1749
|
} finally {
|
|
1726
1750
|
const e = this._stateMachine.getSummary();
|
|
1727
|
-
|
|
1751
|
+
B.emit("sessionClose", { session: this, summaryInfo: e });
|
|
1728
1752
|
}
|
|
1729
1753
|
}
|
|
1730
1754
|
/**
|
|
@@ -1738,7 +1762,7 @@ class F {
|
|
|
1738
1762
|
* 用户调用的,注册session上的监听
|
|
1739
1763
|
*/
|
|
1740
1764
|
registerSessionListener(e) {
|
|
1741
|
-
const t =
|
|
1765
|
+
const t = j(e);
|
|
1742
1766
|
if (!t.result)
|
|
1743
1767
|
throw new Error(`[RCCallSession registerSessionListener] ${t.msg}`);
|
|
1744
1768
|
this._listener = { ...e };
|
|
@@ -1747,40 +1771,40 @@ class F {
|
|
|
1747
1771
|
* 调RTC API 获得本地流
|
|
1748
1772
|
*/
|
|
1749
1773
|
async _getLocalTrackCore(e, t) {
|
|
1750
|
-
if (e ===
|
|
1774
|
+
if (e === O.AUDIO) {
|
|
1751
1775
|
const { code: o, track: r } = await this._rtcClient.createMicrophoneAudioTrack("RongCloudRTC", t && t.audio && { ...t.audio });
|
|
1752
|
-
return o !==
|
|
1776
|
+
return o !== p.SUCCESS ? (this._logger.error("_", `[RCCallSession _getLocalTrackCore] get Audio local tracks failed RCT code -> ${o}`), { code: l.GET_LOCAL_AUDIO_TRACK_ERROR }) : (this._logger.info("_", "[RCCallSession _getLocalTrackCore] successfully get Audio local tracks"), { code: l.SUCCESS, tracks: [r] });
|
|
1753
1777
|
}
|
|
1754
1778
|
const { code: i, tracks: n } = await this._rtcClient.createMicrophoneAndCameraTracks("RongCloudRTC", t && { ...t });
|
|
1755
|
-
return i !==
|
|
1779
|
+
return i !== p.SUCCESS ? (this._logger.error("_", `[RCCallSession _getLocalTrackCore] get Audio and Video local tracks failed RCT code -> ${i}`), { code: l.GET_LOCAL_AUDIO_AND_VIDEO_TRACK_ERROR }) : (this._logger.info("_", "[RCCallSession _getLocalTrackCore] successfully get audio and video local tracks"), { code: l.SUCCESS, tracks: n });
|
|
1756
1780
|
}
|
|
1757
1781
|
async _getLocalTrack(e, t) {
|
|
1758
|
-
if (this._options.isAllowDemotionGetStream && e ===
|
|
1759
|
-
const { code: o, tracks: r } = await this._getLocalTrackCore(
|
|
1760
|
-
if (o !==
|
|
1761
|
-
const { code: a, tracks:
|
|
1762
|
-
return a !==
|
|
1782
|
+
if (this._options.isAllowDemotionGetStream && e === O.AUDIO_VIDEO) {
|
|
1783
|
+
const { code: o, tracks: r } = await this._getLocalTrackCore(O.AUDIO_VIDEO, t);
|
|
1784
|
+
if (o !== l.SUCCESS) {
|
|
1785
|
+
const { code: a, tracks: h } = await this._getLocalTrackCore(O.AUDIO, t);
|
|
1786
|
+
return a !== l.SUCCESS ? (this._exceptionClose(g.GET_MEDIA_RESOURCES_ERROR), { code: a }) : { code: a, tracks: h };
|
|
1763
1787
|
}
|
|
1764
1788
|
return { code: o, tracks: r };
|
|
1765
1789
|
}
|
|
1766
1790
|
const { code: i, tracks: n } = await this._getLocalTrackCore(e, t);
|
|
1767
|
-
return i !==
|
|
1791
|
+
return i !== l.SUCCESS ? (this._exceptionClose(g.GET_MEDIA_RESOURCES_ERROR), { code: i }) : { code: i, tracks: n };
|
|
1768
1792
|
}
|
|
1769
1793
|
/**
|
|
1770
1794
|
* 通话中更换音频设备
|
|
1771
1795
|
*/
|
|
1772
1796
|
async changeAudioDevice(e) {
|
|
1773
1797
|
const t = [], i = [], { code: n, track: o } = await this._rtcClient.createMicrophoneAudioTrack("RongCloudRTC", e);
|
|
1774
|
-
if (n !==
|
|
1775
|
-
return this._logger.error("_", `[RCCallSession changeDevice] get local Audio tracks failed RCTLib code -> ${n}`), { code:
|
|
1798
|
+
if (n !== p.SUCCESS)
|
|
1799
|
+
return this._logger.error("_", `[RCCallSession changeDevice] get local Audio tracks failed RCTLib code -> ${n}`), { code: l.GET_LOCAL_AUDIO_TRACK_ERROR };
|
|
1776
1800
|
if (this._options.localTracks && this._options.localTracks.forEach((r) => {
|
|
1777
1801
|
r.isAudioTrack() || i.push(r);
|
|
1778
1802
|
}), t.push(o), i.push(o), this._options.localTracks = i, this._notifyTrackReady(t), this._room) {
|
|
1779
1803
|
const { code: r } = await this._room.publish(t);
|
|
1780
|
-
if (r !==
|
|
1781
|
-
return { code:
|
|
1804
|
+
if (r !== p.SUCCESS)
|
|
1805
|
+
return { code: l.AUDIO_PUBLISH_ERROR };
|
|
1782
1806
|
}
|
|
1783
|
-
return { code:
|
|
1807
|
+
return { code: l.SUCCESS };
|
|
1784
1808
|
}
|
|
1785
1809
|
/**
|
|
1786
1810
|
* 群呼叫中继续邀请
|
|
@@ -1790,28 +1814,36 @@ class F {
|
|
|
1790
1814
|
* @deprecated 5.1.2 废弃 options.pushContent 通知内容
|
|
1791
1815
|
*/
|
|
1792
1816
|
async invite(e, t = {}) {
|
|
1793
|
-
|
|
1794
|
-
|
|
1795
|
-
|
|
1796
|
-
|
|
1817
|
+
const { extra: i = "" } = t, n = this._options.callPushConfig ? this._options.callPushConfig : t.pushConfig, o = [le(e), X(i)];
|
|
1818
|
+
if (n) {
|
|
1819
|
+
const { code: c, errorMsg: _ } = J(n);
|
|
1820
|
+
if (c !== l.SUCCESS)
|
|
1821
|
+
return this._logger.error("_", `[RCCallSession invite] param error, errorMsg: ${_}`), { code: c };
|
|
1822
|
+
}
|
|
1823
|
+
const r = [];
|
|
1824
|
+
if (!o.every((c) => (!c.result && r.push(c.msg), c.result)))
|
|
1825
|
+
throw new Error(`[RCCallClient invite] ${r.join(`
|
|
1797
1826
|
`)}`);
|
|
1798
|
-
const { code:
|
|
1799
|
-
|
|
1827
|
+
const { code: h } = await this._stateMachine.invite(e, {
|
|
1828
|
+
extra: i,
|
|
1829
|
+
pushConfig: n
|
|
1830
|
+
});
|
|
1831
|
+
return { code: h };
|
|
1800
1832
|
}
|
|
1801
1833
|
/**
|
|
1802
1834
|
* 同意接听
|
|
1803
1835
|
*/
|
|
1804
1836
|
async accept(e) {
|
|
1805
|
-
const t =
|
|
1837
|
+
const t = be(e);
|
|
1806
1838
|
if (!t.result)
|
|
1807
1839
|
throw new Error(`[RCCallSession accept] ${t.msg}`);
|
|
1808
|
-
|
|
1840
|
+
B.emit("hungupOtherSession", { session: this });
|
|
1809
1841
|
const i = this._stateMachine.getMediaType(), { code: n, tracks: o } = await this._getLocalTrack(i, e);
|
|
1810
|
-
if (n !==
|
|
1842
|
+
if (n !== l.SUCCESS)
|
|
1811
1843
|
return { code: n };
|
|
1812
1844
|
this._options.localTracks = o;
|
|
1813
1845
|
const { code: r } = await this._stateMachine.accept();
|
|
1814
|
-
return r !==
|
|
1846
|
+
return r !== l.SUCCESS ? (this._logger.error("_", `[RCCallSession accept]Send accept message failed -> code: ${r}`), { code: r }) : { code: r };
|
|
1815
1847
|
}
|
|
1816
1848
|
/**
|
|
1817
1849
|
* 挂断
|
|
@@ -1825,7 +1857,7 @@ class F {
|
|
|
1825
1857
|
*/
|
|
1826
1858
|
async _changeMediaType(e) {
|
|
1827
1859
|
const { code: t } = await this._stateMachine.changeMediaType(e);
|
|
1828
|
-
return t !==
|
|
1860
|
+
return t !== l.SUCCESS && this._logger.error("_", `[RCCallSession _changeMediaType] change media type fail code-> ${t}`), { code: t };
|
|
1829
1861
|
}
|
|
1830
1862
|
/**
|
|
1831
1863
|
* 获得本地视频
|
|
@@ -1846,14 +1878,14 @@ class F {
|
|
|
1846
1878
|
*/
|
|
1847
1879
|
async _setMediaTypeToAudioAndVideo() {
|
|
1848
1880
|
const { code: e, track: t } = await this._rtcClient.createCameraVideoTrack();
|
|
1849
|
-
if (e !==
|
|
1850
|
-
return { code:
|
|
1881
|
+
if (e !== p.SUCCESS)
|
|
1882
|
+
return { code: l.GET_LOCAL_AUDIO_AND_VIDEO_TRACK_ERROR };
|
|
1851
1883
|
const { code: i } = await this._room.publish([t]);
|
|
1852
|
-
if (i !==
|
|
1884
|
+
if (i !== p.SUCCESS) {
|
|
1853
1885
|
this._logger.error("_", `[RCCallSession _enableVideo] Resource publishing failed: RCRTCCode -> ${e}`);
|
|
1854
1886
|
return;
|
|
1855
1887
|
}
|
|
1856
|
-
this._notifyTrackReady([t]), this._changeMediaType(
|
|
1888
|
+
this._notifyTrackReady([t]), this._changeMediaType(O.AUDIO_VIDEO);
|
|
1857
1889
|
}
|
|
1858
1890
|
/**
|
|
1859
1891
|
* 把通话的MediaType降级到音频
|
|
@@ -1866,7 +1898,7 @@ class F {
|
|
|
1866
1898
|
i.mute();
|
|
1867
1899
|
});
|
|
1868
1900
|
const { code: t } = await this._room.unpublish(e);
|
|
1869
|
-
t !==
|
|
1901
|
+
t !== p.SUCCESS && this._logger.error("_", `[RCCallSession disableVideo] unpublish failed -> ${t}`), this._destroyTracks(e);
|
|
1870
1902
|
}
|
|
1871
1903
|
}
|
|
1872
1904
|
/**
|
|
@@ -1874,55 +1906,55 @@ class F {
|
|
|
1874
1906
|
*
|
|
1875
1907
|
*/
|
|
1876
1908
|
async descendAbility() {
|
|
1877
|
-
const { code: e } = await this._changeMediaType(
|
|
1878
|
-
return e ===
|
|
1909
|
+
const { code: e } = await this._changeMediaType(O.AUDIO);
|
|
1910
|
+
return e === l.SUCCESS && this._setMediaTypeToAudio(), { code: e };
|
|
1879
1911
|
}
|
|
1880
1912
|
/**
|
|
1881
1913
|
* 禁用视频track
|
|
1882
1914
|
*/
|
|
1883
1915
|
async disableVideoTrack() {
|
|
1884
1916
|
if (!this._room)
|
|
1885
|
-
return this._logger.error("_", `[RCCallSession disableAudioTrack] Room missing audio track -> ${
|
|
1917
|
+
return this._logger.error("_", `[RCCallSession disableAudioTrack] Room missing audio track -> ${l.NOT_IN_ROOM_ERROR}`), { code: l.NOT_IN_ROOM_ERROR };
|
|
1886
1918
|
const e = this._getLocalVideoTracks();
|
|
1887
1919
|
if (!e.length)
|
|
1888
|
-
return this._logger.error("_", `[RCCallSession disableVideoTrack] Room missing video track -> ${
|
|
1920
|
+
return this._logger.error("_", `[RCCallSession disableVideoTrack] Room missing video track -> ${l.MISSING_VIDEO_TRACK_ERROR}`), { code: l.MISSING_VIDEO_TRACK_ERROR };
|
|
1889
1921
|
if (e.forEach((i) => {
|
|
1890
1922
|
i.mute();
|
|
1891
1923
|
}), !this._options.isOffCameraWhenVideoDisable)
|
|
1892
|
-
return { code:
|
|
1924
|
+
return { code: l.SUCCESS };
|
|
1893
1925
|
const { code: t } = await this._room.unpublish(e);
|
|
1894
|
-
return t !==
|
|
1926
|
+
return t !== p.SUCCESS ? (this._logger.error("_", `[RCCallSession disableVideo] unpublish failed -> ${t}`), { code: l.UNPUBLISH_VIDEO_ERROR }) : (e.forEach((i) => {
|
|
1895
1927
|
i.destroy();
|
|
1896
|
-
}), { code:
|
|
1928
|
+
}), { code: l.SUCCESS });
|
|
1897
1929
|
}
|
|
1898
1930
|
/**
|
|
1899
1931
|
* 启用视频track
|
|
1900
1932
|
*/
|
|
1901
1933
|
async enableVideoTrack() {
|
|
1902
1934
|
if (!this._room)
|
|
1903
|
-
return this._logger.error("_", `[RCCallSession disableAudioTrack] Room missing audio track -> ${
|
|
1935
|
+
return this._logger.error("_", `[RCCallSession disableAudioTrack] Room missing audio track -> ${l.NOT_IN_ROOM_ERROR}`), { code: l.NOT_IN_ROOM_ERROR };
|
|
1904
1936
|
if (!this._options.isOffCameraWhenVideoDisable) {
|
|
1905
1937
|
const o = this._getLocalVideoTracks();
|
|
1906
1938
|
return o.length ? (o.forEach((r) => {
|
|
1907
1939
|
r.unmute();
|
|
1908
|
-
}), { code:
|
|
1940
|
+
}), { code: l.SUCCESS }) : (this._logger.error("_", `[RCCallSession EnableVideoTrack] Room missing video track -> ${l.MISSING_VIDEO_TRACK_ERROR}`), { code: l.MISSING_VIDEO_TRACK_ERROR });
|
|
1909
1941
|
}
|
|
1910
1942
|
const { code: e, track: t } = await this._rtcClient.createCameraVideoTrack();
|
|
1911
|
-
if (e !==
|
|
1912
|
-
return this._logger.error("_", `[RCCallSession EnableVideoTrack] Get Resource failed: RCRTCCode -> ${e}`), { code:
|
|
1943
|
+
if (e !== p.SUCCESS)
|
|
1944
|
+
return this._logger.error("_", `[RCCallSession EnableVideoTrack] Get Resource failed: RCRTCCode -> ${e}`), { code: l.GET_LOCAL_VIDEO_TRACK_ERROR };
|
|
1913
1945
|
const i = [];
|
|
1914
1946
|
this._options.localTracks && this._options.localTracks.forEach((o) => {
|
|
1915
1947
|
o.isVideoTrack() ? o.destroy() : i.push(o);
|
|
1916
1948
|
}), i.push(t), this._options.localTracks = i, t.mute();
|
|
1917
1949
|
const { code: n } = await this._room.publish([t]);
|
|
1918
|
-
return n !==
|
|
1950
|
+
return n !== p.SUCCESS ? (this._logger.error("_", `[RCCallSession EnableVideoTrack] Resource publishing failed: RCRTCCode -> ${e}`), { code: l.VIDEO_PUBLISH_ERROR }) : (t.unmute(), this._notifyTrackReady([t]), { code: l.SUCCESS });
|
|
1919
1951
|
}
|
|
1920
1952
|
/**
|
|
1921
1953
|
* 禁用音频track
|
|
1922
1954
|
*/
|
|
1923
1955
|
async disableAudioTrack() {
|
|
1924
1956
|
if (!this._room)
|
|
1925
|
-
return this._logger.error("_", `[RCCallSession disableAudioTrack] Room missing audio track -> ${
|
|
1957
|
+
return this._logger.error("_", `[RCCallSession disableAudioTrack] Room missing audio track -> ${l.NOT_IN_ROOM_ERROR}`), { code: l.NOT_IN_ROOM_ERROR };
|
|
1926
1958
|
this._getLocalAudioTracks().forEach((t) => {
|
|
1927
1959
|
t.mute();
|
|
1928
1960
|
});
|
|
@@ -1932,10 +1964,10 @@ class F {
|
|
|
1932
1964
|
*/
|
|
1933
1965
|
async enableAudioTrack() {
|
|
1934
1966
|
if (!this._room)
|
|
1935
|
-
return this._logger.error("_", `[RCCallSession disableAudioTrack] Room missing audio track -> ${
|
|
1967
|
+
return this._logger.error("_", `[RCCallSession disableAudioTrack] Room missing audio track -> ${l.NOT_IN_ROOM_ERROR}`), { code: l.NOT_IN_ROOM_ERROR };
|
|
1936
1968
|
const e = this._getLocalAudioTracks();
|
|
1937
1969
|
if (!e.length)
|
|
1938
|
-
return this._logger.error("_", `[RCCallSession disableAudioTrack] Room missing audio track -> ${
|
|
1970
|
+
return this._logger.error("_", `[RCCallSession disableAudioTrack] Room missing audio track -> ${l.MISSING_VIDEO_TRACK_ERROR}`), { code: l.MISSING_VIDEO_TRACK_ERROR };
|
|
1939
1971
|
e.forEach((t) => {
|
|
1940
1972
|
t.unmute();
|
|
1941
1973
|
});
|
|
@@ -1976,7 +2008,7 @@ class F {
|
|
|
1976
2008
|
*/
|
|
1977
2009
|
onKickOff: (e, t) => {
|
|
1978
2010
|
const i = this._rtcClient.getCurrentId();
|
|
1979
|
-
this._stateMachine.userLeave([i]), e ? (t ===
|
|
2011
|
+
this._stateMachine.userLeave([i]), e ? (t === Q.SERVER_KICK && this._exceptionClose(g.KICKED_BY_SERVER), t === Q.OTHER_KICK && this._exceptionClose(g.OTHER_CLIENT_JOINED_CALL)) : this._exceptionClose(g.NETWORK_ERROR);
|
|
1980
2012
|
},
|
|
1981
2013
|
/**
|
|
1982
2014
|
* 接收到房间信令时回调,用户可通过房间实例的 `sendMessage(name, content)` 接口发送信令
|
|
@@ -2038,7 +2070,7 @@ class F {
|
|
|
2038
2070
|
onTrackPublish: async (e) => {
|
|
2039
2071
|
if (this._room) {
|
|
2040
2072
|
const { code: t } = await this._room.subscribe(e);
|
|
2041
|
-
t !==
|
|
2073
|
+
t !== p.SUCCESS && this._logger.error("_", `[RCCallSession onTrackPublish] subscribe failed RTCCode ->${t}`);
|
|
2042
2074
|
}
|
|
2043
2075
|
},
|
|
2044
2076
|
/**
|
|
@@ -2053,7 +2085,7 @@ class F {
|
|
|
2053
2085
|
* @param track RCRemoteTrack 类实例
|
|
2054
2086
|
*/
|
|
2055
2087
|
onTrackReady: (e) => {
|
|
2056
|
-
this._stateMachine.getMediaType() ===
|
|
2088
|
+
this._stateMachine.getMediaType() === O.AUDIO && e.isVideoTrack() || this._notifyTrackReady([e]);
|
|
2057
2089
|
},
|
|
2058
2090
|
/**
|
|
2059
2091
|
* 人员加入
|
|
@@ -2181,7 +2213,7 @@ class F {
|
|
|
2181
2213
|
return this._stateMachine.getMediaType();
|
|
2182
2214
|
}
|
|
2183
2215
|
}
|
|
2184
|
-
class
|
|
2216
|
+
class He {
|
|
2185
2217
|
constructor(e, t, i, n) {
|
|
2186
2218
|
/**
|
|
2187
2219
|
* rtc实例
|
|
@@ -2199,8 +2231,14 @@ class ke {
|
|
|
2199
2231
|
* session列表
|
|
2200
2232
|
*/
|
|
2201
2233
|
f(this, "_sessionList", []);
|
|
2202
|
-
|
|
2203
|
-
|
|
2234
|
+
/**
|
|
2235
|
+
* 移动端呼叫推送配置
|
|
2236
|
+
*/
|
|
2237
|
+
f(this, "_callPushConfig");
|
|
2238
|
+
/**
|
|
2239
|
+
* 移动端挂断推送配置
|
|
2240
|
+
*/
|
|
2241
|
+
f(this, "_hungupPushConfig");
|
|
2204
2242
|
this._context = e, this._runtime = t, this._logger = i, this._rtcClient = n.rtcClient, this._options = {
|
|
2205
2243
|
/**
|
|
2206
2244
|
* 是否允许发布重试, 默认不允许
|
|
@@ -2220,7 +2258,7 @@ class ke {
|
|
|
2220
2258
|
* RTCJoinType.REFUSE = 1,当前加入拒绝
|
|
2221
2259
|
* RTCJoinType.COEXIST = 2 两个设备共存
|
|
2222
2260
|
*/
|
|
2223
|
-
joinType:
|
|
2261
|
+
joinType: ne.COEXIST,
|
|
2224
2262
|
/**
|
|
2225
2263
|
* 允许降级获得流,获得音视频不成功 ,降级获得音频, 默认不允许
|
|
2226
2264
|
*/
|
|
@@ -2230,7 +2268,7 @@ class ke {
|
|
|
2230
2268
|
*/
|
|
2231
2269
|
lang: P.ZH,
|
|
2232
2270
|
...n
|
|
2233
|
-
}, this._callEngine = new
|
|
2271
|
+
}, this._callEngine = new Oe(this._context, t, this._logger, {
|
|
2234
2272
|
/**
|
|
2235
2273
|
* 监听收到invite
|
|
2236
2274
|
*/
|
|
@@ -2244,16 +2282,16 @@ class ke {
|
|
|
2244
2282
|
* 语言设置 (推送), 不传默认为中文
|
|
2245
2283
|
*/
|
|
2246
2284
|
lang: this._options.lang || P.ZH
|
|
2247
|
-
}),
|
|
2285
|
+
}), B.on("sessionClose", ({ session: o, summaryInfo: r }) => {
|
|
2248
2286
|
this._removeSession(o);
|
|
2249
2287
|
try {
|
|
2250
2288
|
this._options.onSessionClose(o, r);
|
|
2251
2289
|
} catch (a) {
|
|
2252
2290
|
this._logger.error("_", "[RCCCallClient] options.onSessionClose exception"), console.log(a);
|
|
2253
2291
|
}
|
|
2254
|
-
}),
|
|
2292
|
+
}), B.on("hungupOtherSession", ({ session: o }) => {
|
|
2255
2293
|
const r = o.getSessionId();
|
|
2256
|
-
this._logger.info("_", `[RCCallClient hungupOtherSession] sessionId ready to accept -> ${r}`), this._logger.info("_", `[RCCallClient hungupOtherSession] sessionList ->${this._sessionList.map((
|
|
2294
|
+
this._logger.info("_", `[RCCallClient hungupOtherSession] sessionId ready to accept -> ${r}`), this._logger.info("_", `[RCCallClient hungupOtherSession] sessionList ->${this._sessionList.map((h) => h.getSessionId()).join(",")}`);
|
|
2257
2295
|
let a = 0;
|
|
2258
2296
|
for (; this._sessionList.length > 1; )
|
|
2259
2297
|
this._sessionList[a].getSessionId() !== r ? (this._sessionList[a].hungup(), this._sessionList.splice(a, 1)) : a++;
|
|
@@ -2265,7 +2303,7 @@ class ke {
|
|
|
2265
2303
|
*/
|
|
2266
2304
|
_onInvite(e, t) {
|
|
2267
2305
|
this._logger.info("_", "[RCCallClient _onInvite] Received invite message");
|
|
2268
|
-
const i = new
|
|
2306
|
+
const i = new z(e, this._rtcClient, this._logger, {
|
|
2269
2307
|
// 是否允许订阅重试
|
|
2270
2308
|
isAllowSubscribeRetry: this._options.isAllowSubscribeRetry,
|
|
2271
2309
|
// 是否允许发布重试
|
|
@@ -2292,7 +2330,7 @@ class ke {
|
|
|
2292
2330
|
this._logger.error("_", "[RCCallClient _options.onSession] onSession exception"), console.log(n);
|
|
2293
2331
|
}
|
|
2294
2332
|
if (i._listener) {
|
|
2295
|
-
const n =
|
|
2333
|
+
const n = j(i._listener);
|
|
2296
2334
|
if (!n.result)
|
|
2297
2335
|
throw new Error(n.msg);
|
|
2298
2336
|
} else
|
|
@@ -2333,14 +2371,14 @@ class ke {
|
|
|
2333
2371
|
*/
|
|
2334
2372
|
async startCrossCall({
|
|
2335
2373
|
targetId: e,
|
|
2336
|
-
mediaType: t =
|
|
2374
|
+
mediaType: t = O.AUDIO,
|
|
2337
2375
|
listener: i,
|
|
2338
2376
|
constraints: n,
|
|
2339
2377
|
channelId: o = "",
|
|
2340
2378
|
extra: r = "",
|
|
2341
2379
|
pushTitle: a = "",
|
|
2342
|
-
pushContent:
|
|
2343
|
-
bitrate:
|
|
2380
|
+
pushContent: h = "",
|
|
2381
|
+
bitrate: c
|
|
2344
2382
|
}) {
|
|
2345
2383
|
return this.__call({
|
|
2346
2384
|
targetId: e,
|
|
@@ -2350,8 +2388,8 @@ class ke {
|
|
|
2350
2388
|
channelId: o,
|
|
2351
2389
|
extra: r,
|
|
2352
2390
|
pushTitle: a,
|
|
2353
|
-
pushContent:
|
|
2354
|
-
bitrate:
|
|
2391
|
+
pushContent: h,
|
|
2392
|
+
bitrate: c,
|
|
2355
2393
|
isCrossAppkey: !0
|
|
2356
2394
|
});
|
|
2357
2395
|
}
|
|
@@ -2370,14 +2408,14 @@ class ke {
|
|
|
2370
2408
|
*/
|
|
2371
2409
|
async call({
|
|
2372
2410
|
targetId: e,
|
|
2373
|
-
mediaType: t =
|
|
2411
|
+
mediaType: t = O.AUDIO,
|
|
2374
2412
|
listener: i,
|
|
2375
2413
|
constraints: n,
|
|
2376
2414
|
channelId: o = "",
|
|
2377
2415
|
extra: r = "",
|
|
2378
2416
|
pushTitle: a = "",
|
|
2379
|
-
pushContent:
|
|
2380
|
-
bitrate:
|
|
2417
|
+
pushContent: h = "",
|
|
2418
|
+
bitrate: c
|
|
2381
2419
|
}) {
|
|
2382
2420
|
return this.__call({
|
|
2383
2421
|
targetId: e,
|
|
@@ -2387,41 +2425,44 @@ class ke {
|
|
|
2387
2425
|
channelId: o,
|
|
2388
2426
|
extra: r,
|
|
2389
2427
|
pushTitle: a,
|
|
2390
|
-
pushContent:
|
|
2391
|
-
bitrate:
|
|
2428
|
+
pushContent: h,
|
|
2429
|
+
bitrate: c
|
|
2392
2430
|
});
|
|
2393
2431
|
}
|
|
2394
2432
|
async __call({
|
|
2395
2433
|
targetId: e,
|
|
2396
|
-
mediaType: t =
|
|
2434
|
+
mediaType: t = O.AUDIO,
|
|
2397
2435
|
listener: i,
|
|
2398
2436
|
constraints: n,
|
|
2399
2437
|
channelId: o = "",
|
|
2400
2438
|
extra: r = "",
|
|
2401
2439
|
pushTitle: a = "",
|
|
2402
|
-
pushContent:
|
|
2403
|
-
bitrate:
|
|
2440
|
+
pushContent: h = "",
|
|
2441
|
+
bitrate: c,
|
|
2404
2442
|
isCrossAppkey: _ = !1
|
|
2405
2443
|
}) {
|
|
2406
|
-
const
|
|
2407
|
-
this._logger.info("_", `[RCCallClient call] extra->${r}
|
|
2408
|
-
const
|
|
2409
|
-
if (!
|
|
2410
|
-
throw new Error(`[RCCallClient call] ${
|
|
2444
|
+
const u = this._callPushConfig ? this._callPushConfig : { pushTitle: a, pushContent: h };
|
|
2445
|
+
this._logger.info("_", `[RCCallClient call] extra->${r} pushConfig->${JSON.stringify(u)}`);
|
|
2446
|
+
const S = [te(e), se(t), j(i), X(r)], C = [];
|
|
2447
|
+
if (!S.every((I) => (!I.result && C.push(I.msg), I.result)))
|
|
2448
|
+
throw new Error(`[RCCallClient call] ${C.join(`
|
|
2411
2449
|
`)}`);
|
|
2412
|
-
|
|
2413
|
-
|
|
2414
|
-
|
|
2415
|
-
|
|
2416
|
-
M =
|
|
2417
|
-
|
|
2418
|
-
|
|
2450
|
+
const { code: y, errorMsg: U } = J(u);
|
|
2451
|
+
if (y !== l.SUCCESS)
|
|
2452
|
+
return this._logger.error("_", `[RCCallClient call] param error, errorMsg: ${U}`), { code: y };
|
|
2453
|
+
let E = [];
|
|
2454
|
+
const { code: M, tracks: N } = await this._getLocalTrack(t, n);
|
|
2455
|
+
if (M !== l.SUCCESS)
|
|
2456
|
+
return { code: M };
|
|
2457
|
+
E = N, E.forEach((I) => {
|
|
2458
|
+
var L, k, H;
|
|
2459
|
+
I.isAudioTrack() && (c != null && c.audio) && I.setBitrate(c == null ? void 0 : c.audio), I.isVideoTrack() && (c != null && c.video) && I.setBitrate((L = c == null ? void 0 : c.video) == null ? void 0 : L.max, (k = c == null ? void 0 : c.video) == null ? void 0 : k.min, (H = c == null ? void 0 : c.video) == null ? void 0 : H.start), i.onTrackReady(I);
|
|
2419
2460
|
});
|
|
2420
|
-
const { code:
|
|
2421
|
-
if (
|
|
2461
|
+
const { code: A, stateMachine: v } = await this._callEngine.call(o, e, t, r, u, _);
|
|
2462
|
+
if (A === l.SUCCESS && v) {
|
|
2422
2463
|
this._logger.info("_", "[RCCallClient call] successfully created state machine");
|
|
2423
|
-
const
|
|
2424
|
-
localTracks:
|
|
2464
|
+
const I = new z(v, this._rtcClient, this._logger, {
|
|
2465
|
+
localTracks: E,
|
|
2425
2466
|
// 是否允许订阅重试
|
|
2426
2467
|
isAllowSubscribeRetry: this._options.isAllowSubscribeRetry,
|
|
2427
2468
|
// 是否允许订阅重试
|
|
@@ -2442,11 +2483,11 @@ class ke {
|
|
|
2442
2483
|
callPushConfig: this._callPushConfig,
|
|
2443
2484
|
hungupPushConfig: this._hungupPushConfig
|
|
2444
2485
|
});
|
|
2445
|
-
return
|
|
2486
|
+
return I.registerSessionListener(i), this._sessionList.push(I), this._logger.info("_", `[RCCallClient call] successfully created session object, sessionId: ${I.getSessionId()}`), { code: A, session: I };
|
|
2446
2487
|
}
|
|
2447
|
-
return this._logger.error("_", `[RCCallClient call] call failed code ->: ${
|
|
2448
|
-
|
|
2449
|
-
}), { code:
|
|
2488
|
+
return this._logger.error("_", `[RCCallClient call] call failed code ->: ${A}`), E.forEach((I) => {
|
|
2489
|
+
I.mute(), I.destroy();
|
|
2490
|
+
}), { code: A };
|
|
2450
2491
|
}
|
|
2451
2492
|
/**
|
|
2452
2493
|
* 发起群组呼叫
|
|
@@ -2464,32 +2505,35 @@ class ke {
|
|
|
2464
2505
|
async callInGroup({
|
|
2465
2506
|
targetId: e,
|
|
2466
2507
|
userIds: t,
|
|
2467
|
-
mediaType: i =
|
|
2508
|
+
mediaType: i = O.AUDIO,
|
|
2468
2509
|
listener: n,
|
|
2469
2510
|
constraints: o,
|
|
2470
2511
|
channelId: r = "",
|
|
2471
2512
|
extra: a = "",
|
|
2472
|
-
pushTitle:
|
|
2473
|
-
pushContent:
|
|
2513
|
+
pushTitle: h = "",
|
|
2514
|
+
pushContent: c = "",
|
|
2474
2515
|
bitrate: _
|
|
2475
2516
|
}) {
|
|
2476
|
-
const
|
|
2477
|
-
if (!
|
|
2478
|
-
throw new Error(`[RCCallClient callInGroup] ${
|
|
2517
|
+
const u = this._callPushConfig ? this._callPushConfig : { pushTitle: h, pushContent: c }, S = [te(e), le(t), se(i), j(n), X(a)], C = [];
|
|
2518
|
+
if (!S.every((I) => (!I.result && C.push(I.msg), I.result)))
|
|
2519
|
+
throw new Error(`[RCCallClient callInGroup] ${C.join(`
|
|
2479
2520
|
`)}`);
|
|
2480
|
-
|
|
2481
|
-
|
|
2482
|
-
|
|
2483
|
-
|
|
2484
|
-
M =
|
|
2485
|
-
|
|
2486
|
-
|
|
2521
|
+
const { code: y, errorMsg: U } = J(u);
|
|
2522
|
+
if (y !== l.SUCCESS)
|
|
2523
|
+
return this._logger.error("_", `[RCCallClient call] param error, errorMsg: ${U}`), { code: y };
|
|
2524
|
+
let E = [];
|
|
2525
|
+
const { code: M, tracks: N } = await this._getLocalTrack(i, o);
|
|
2526
|
+
if (M !== l.SUCCESS)
|
|
2527
|
+
return { code: M };
|
|
2528
|
+
E = N, E.forEach((I) => {
|
|
2529
|
+
var L, k, H;
|
|
2530
|
+
I.isAudioTrack() && (_ != null && _.audio) && I.setBitrate(_ == null ? void 0 : _.audio), I.isVideoTrack() && (_ != null && _.video) && I.setBitrate((L = _ == null ? void 0 : _.video) == null ? void 0 : L.max, (k = _ == null ? void 0 : _.video) == null ? void 0 : k.min, (H = _ == null ? void 0 : _.video) == null ? void 0 : H.start), n.onTrackReady(I);
|
|
2487
2531
|
});
|
|
2488
|
-
const { code:
|
|
2489
|
-
if (
|
|
2532
|
+
const { code: A, stateMachine: v } = await this._callEngine.callInGroup(r, e, i, t, a, u);
|
|
2533
|
+
if (A === l.SUCCESS && v) {
|
|
2490
2534
|
this._logger.info("_", "[RCCallClient callInGroup] successfully created state machine");
|
|
2491
|
-
const
|
|
2492
|
-
localTracks:
|
|
2535
|
+
const I = new z(v, this._rtcClient, this._logger, {
|
|
2536
|
+
localTracks: E,
|
|
2493
2537
|
// 是否允许订阅重试
|
|
2494
2538
|
isAllowSubscribeRetry: this._options.isAllowSubscribeRetry,
|
|
2495
2539
|
// 是否允许发布重试
|
|
@@ -2509,34 +2553,34 @@ class ke {
|
|
|
2509
2553
|
callPushConfig: this._callPushConfig,
|
|
2510
2554
|
hungupPushConfig: this._hungupPushConfig
|
|
2511
2555
|
});
|
|
2512
|
-
return
|
|
2556
|
+
return I.registerSessionListener(n), this._sessionList.push(I), this._logger.info("_", `[RCCallClient callInGroup] successfully created session object, sessionId: ${I.getSessionId()}`), { code: A, session: I };
|
|
2513
2557
|
}
|
|
2514
|
-
return this._logger.info("_", `[RCCallClient callInGroup] callInGroup failed code -> ${
|
|
2515
|
-
|
|
2516
|
-
}), { code:
|
|
2558
|
+
return this._logger.info("_", `[RCCallClient callInGroup] callInGroup failed code -> ${A}`), E.forEach((I) => {
|
|
2559
|
+
I.mute(), I.destroy();
|
|
2560
|
+
}), { code: A };
|
|
2517
2561
|
}
|
|
2518
2562
|
/**
|
|
2519
2563
|
* 调RTC API 获得本地流
|
|
2520
2564
|
*/
|
|
2521
2565
|
async _getLocalTrackCore(e, t) {
|
|
2522
|
-
if (e ===
|
|
2566
|
+
if (e === O.AUDIO) {
|
|
2523
2567
|
const { code: o, track: r } = await this._rtcClient.createMicrophoneAudioTrack("RongCloudRTC", t && t.audio && { ...t.audio });
|
|
2524
|
-
return o !==
|
|
2568
|
+
return o !== p.SUCCESS ? (this._logger.error("_", `[RCCallClient _getTrack] get Audio local tracks failed RCT code -> ${o}`), { code: l.GET_LOCAL_AUDIO_TRACK_ERROR }) : (this._logger.info("_", "[RCCallClient _getTrack] successfully get Audio local tracks"), { code: l.SUCCESS, tracks: [r] });
|
|
2525
2569
|
}
|
|
2526
2570
|
const { code: i, tracks: n } = await this._rtcClient.createMicrophoneAndCameraTracks("RongCloudRTC", t && { ...t });
|
|
2527
|
-
return i !==
|
|
2571
|
+
return i !== p.SUCCESS ? (this._logger.error("_", `[RCCallClient _getTrack] get Audio and Video local tracks failed RCT code -> ${i}`), { code: l.GET_LOCAL_AUDIO_AND_VIDEO_TRACK_ERROR }) : (this._logger.info("_", "[RCCallClient _getTrack] successfully get audio and video local tracks"), { code: l.SUCCESS, tracks: n });
|
|
2528
2572
|
}
|
|
2529
2573
|
async _getLocalTrack(e, t) {
|
|
2530
|
-
if (this._options.isAllowDemotionGetStream && e ===
|
|
2531
|
-
const { code: o, tracks: r } = await this._getLocalTrackCore(
|
|
2532
|
-
if (o !==
|
|
2533
|
-
const { code: a, tracks:
|
|
2534
|
-
return a !==
|
|
2574
|
+
if (this._options.isAllowDemotionGetStream && e === O.AUDIO_VIDEO) {
|
|
2575
|
+
const { code: o, tracks: r } = await this._getLocalTrackCore(O.AUDIO_VIDEO, t);
|
|
2576
|
+
if (o !== l.SUCCESS) {
|
|
2577
|
+
const { code: a, tracks: h } = await this._getLocalTrackCore(O.AUDIO, t);
|
|
2578
|
+
return a !== l.SUCCESS ? { code: a } : { code: a, tracks: h };
|
|
2535
2579
|
}
|
|
2536
2580
|
return { code: o, tracks: r };
|
|
2537
2581
|
}
|
|
2538
2582
|
const { code: i, tracks: n } = await this._getLocalTrackCore(e, t);
|
|
2539
|
-
return i !==
|
|
2583
|
+
return i !== l.SUCCESS ? { code: i } : { code: i, tracks: n };
|
|
2540
2584
|
}
|
|
2541
2585
|
/**
|
|
2542
2586
|
* 从sessionList删除某个session
|
|
@@ -2550,7 +2594,7 @@ class ke {
|
|
|
2550
2594
|
*/
|
|
2551
2595
|
async getJoinedRoomInfo() {
|
|
2552
2596
|
const { code: e, data: t } = await this._context.getRTCJoinedUserInfo(this._context.getCurrentId());
|
|
2553
|
-
return e !==
|
|
2597
|
+
return e !== W.SUCCESS ? (this._logger.error("_", `getJoinedUserInfo error code: ${e}`), { code: l.QUERY_JOINED_USER_INFO_ERROR }) : { code: l.SUCCESS, data: t };
|
|
2554
2598
|
}
|
|
2555
2599
|
/**
|
|
2556
2600
|
* 设置呼叫、挂断推送数据
|
|
@@ -2561,39 +2605,39 @@ class ke {
|
|
|
2561
2605
|
* @param hungupPushConfig.pushTitle 挂断推送标题
|
|
2562
2606
|
* @param hungupPushConfig.pushContent 挂断推送内容
|
|
2563
2607
|
*/
|
|
2564
|
-
setPushConfig(e
|
|
2565
|
-
const i = [e, t].map((
|
|
2566
|
-
|
|
2567
|
-
|
|
2568
|
-
`
|
|
2608
|
+
setPushConfig(e, t) {
|
|
2609
|
+
const i = [e, t].map((n) => J(n));
|
|
2610
|
+
for (const { code: n, errorMsg: o } of i)
|
|
2611
|
+
if (n !== l.SUCCESS)
|
|
2612
|
+
return this._logger.error("_", `[RCCallClient setPushConfig] param error, errorMsg: ${o}`), { code: n };
|
|
2569
2613
|
this._callPushConfig = e, this._hungupPushConfig = t;
|
|
2570
2614
|
}
|
|
2571
2615
|
}
|
|
2572
|
-
|
|
2573
|
-
const
|
|
2616
|
+
q.add("plugin-call", "5.2.4");
|
|
2617
|
+
const Be = {
|
|
2574
2618
|
tag: "RCCall",
|
|
2575
2619
|
verify(s) {
|
|
2576
2620
|
return s.tag === "browser";
|
|
2577
2621
|
},
|
|
2578
2622
|
setup(s, e, t) {
|
|
2579
|
-
const i =
|
|
2623
|
+
const i = Ue(t);
|
|
2580
2624
|
if (!i.result)
|
|
2581
2625
|
throw new Error(`[RCCallLib installer steup]${i.msg}`);
|
|
2582
|
-
if (!
|
|
2583
|
-
throw new Error(`The current engine version '${
|
|
2626
|
+
if (!q.validEngine("5.9.0"))
|
|
2627
|
+
throw new Error(`The current engine version '${q.getInfo().engine}' error, plugin-call required engine version at least '5.9.0'.`);
|
|
2584
2628
|
const n = s.createLogger("RCCall", "RTC");
|
|
2585
|
-
return t.logOutputLevel && n.setOutputLevel(t.logOutputLevel), typeof t.logLevel < "u" && n.warn("_", "The 'logLevel' parameter is deprecated, please use 'logOutputLevel' instead."), n.warn("_", "RCCall Version: 5.2.
|
|
2629
|
+
return t.logOutputLevel && n.setOutputLevel(t.logOutputLevel), typeof t.logLevel < "u" && n.warn("_", "The 'logLevel' parameter is deprecated, please use 'logOutputLevel' instead."), n.warn("_", "RCCall Version: 5.2.4, Commit: 0a7b424ef05373eeea5ff3c35af1286656a750fd"), new He(s, e, n, t);
|
|
2586
2630
|
}
|
|
2587
2631
|
};
|
|
2588
2632
|
export {
|
|
2589
|
-
|
|
2633
|
+
He as RCCallClient,
|
|
2590
2634
|
g as RCCallEndReason,
|
|
2591
|
-
|
|
2635
|
+
l as RCCallErrorCode,
|
|
2592
2636
|
P as RCCallLanguage,
|
|
2593
|
-
|
|
2594
|
-
|
|
2595
|
-
|
|
2596
|
-
|
|
2597
|
-
|
|
2637
|
+
O as RCCallMediaType,
|
|
2638
|
+
z as RCCallSession,
|
|
2639
|
+
T as RCCallSessionState,
|
|
2640
|
+
m as RCCallUserState,
|
|
2641
|
+
Be as installer
|
|
2598
2642
|
};
|
|
2599
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"index.esm.js","sources":["../../../engine/helper.ts","../../../engine/core/enums/MsgCallStatus.ts","../../../engine/core/enums/RCCallCode.ts","../../../engine/core/enums/RCCallEndReason.ts","../../../engine/core/enums/RCCallMessageType.ts","../../../engine/core/enums/RCCallSessionState.ts","../../../engine/core/enums/RCCallUserState.ts","../../../engine/core/enums/RCCrossCallType.ts","../../../engine/core/Timer.ts","../../../engine/core/StateMachine.ts","../../../engine/core/enums/RCCallMediaType.ts","../../../engine/core/enums/MemberModifyType.ts","../../../engine/core/enums/Platform.ts","../../../engine/core/OfflineRecorder.ts","../../../engine/core/MessageHandler.ts","../../../engine/core/enums/RCCallLang.ts","../../../engine/core/locale/en.ts","../../../engine/core/locale/zh.ts","../../../engine/core/locale/index.ts","../../../engine/index.ts","../../../src/enums.ts","../../../src/utils.ts","../../../src/eventEmitter.ts","../../../src/validation.ts","../../../src/helper.ts","../../../src/timer.ts","../../../src/RCCallSession.ts","../../../src/RCCallClient.ts","../../../src/index.ts"],"sourcesContent":["import { EventEmitter, IRuntime } from '@rongcloud/engine';\n\nconst string10to64 = (number: number) => {\n  const chars = '0123456789abcdefghigklmnopqrstuvwxyzABCDEFGHIGKLMNOPQRSTUVWXYZ+/'.split('');\n  const radix = chars.length + 1;\n  let qutient = +number;\n  const arr = [];\n  do {\n    const mod = qutient % radix;\n    qutient = (qutient - mod) / radix;\n    arr.unshift(chars[mod]);\n  } while (qutient);\n  return arr.join('');\n};\n\nconst getUUID = () => 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, (c) => {\n  const r = Math.random() * 16 | 0;\n  const v = c === 'x' ? r : (r & 0x3 | 0x8);\n  return v.toString(16);\n});\n\n/* 获取 22 位的 UUID */\nexport const getUUID22 = () => {\n  let uuid: string | number = getUUID();\n  uuid = `${uuid.replace(/-/g, '')}0`;\n  uuid = parseInt(uuid, 16);\n  uuid = string10to64(uuid);\n  if (uuid.length > 22) {\n    uuid = uuid.slice(0, 22);\n  }\n  return uuid;\n};\n\n/**\n * 生成随机 id 字符串\n */\nexport const generateRandomId = (): string => {\n  const random = Math.floor(Math.random() * 1000);\n  let uuid = getUUID22();\n  uuid = uuid.replace(/\\//g, '0');\n  const info = [uuid, Date.now(), random];\n  return info.join('_');\n};\n\nexport const timerSetTimeout = (func: Function, timeout: number): number => setTimeout(func, timeout) as number;\n\nexport const timerInterval = (fun: Function, itv: number): number => setInterval(fun, itv) as number;\n\nexport const eventEmitter = new EventEmitter();\n\n/**\n * call deviceId 存储在 sessionStorage 中\n */\nexport const getCallDeviceId = (runtime: IRuntime) => {\n  const key = 'RCCallDeviceId';\n  let uuid = runtime.sessionStorage.getItem(key);\n  if (!uuid) {\n    uuid = getUUID22();\n    runtime.sessionStorage.setItem(key, uuid);\n  }\n  return uuid;\n};\n","/**\n * 注释\n * TODO\n */\nexport enum MsgCallStatus {\n  OUTGOING = 1, // waiting\n  INCOMING, // waiting\n  RINGING, // waiting\n  CONNECTED, // keeping\n  IDLE, // end\n  ACCEPTED // waiting\n}\n","/**\n * 错误码，与移动端对齐\n * @description\n * 1. `51000 - 51999` 为 Android 专用段\n * 2. `52000 - 52999` 为 iOS 专用段\n * 3. `53000 - 53199` 为 Web RTC 专用段\n * 4. `53200 - 53499` 为 Web CallLib 专用段\n * *  `53200 - 53299` 为 Web CallEngine 专用端\n * *  `53300 - 53499` 为 Web Call 专用端\n * 5. `53500 - 53999` 为 Web 保留段\n */\nexport enum RCCallErrorCode {\n  /**\n   * 成功\n   */\n  SUCCESS = 10000,\n  /**\n   * 存在未结束的状态机\n   */\n  STATE_MACHINE_EXIT = 53200,\n  /**\n   * 发送 IM 消息失败\n   */\n  SEND_MSG_ERROR = 53201,\n  /**\n   * 被对方加入黑名单\n   */\n  REJECTED_BY_BLACKLIST = 53202,\n  /**\n   * 当前用户不再群组中\n   */\n  NOT_IN_GROUP = 53203,\n\n  /**\n   * Call 相关\n   */\n  /**\n   * 获得本地音频流失败\n   */\n  GET_LOCAL_AUDIO_TRACK_ERROR = 53301,\n\n  /**\n   * 获得本地视频流失败\n   */\n  GET_LOCAL_VIDEO_TRACK_ERROR = 53302,\n\n  /**\n   * 获得本地音视频流失败\n   */\n  GET_LOCAL_AUDIO_AND_VIDEO_TRACK_ERROR = 53303,\n\n  /**\n   * 加入房间失败\n   */\n  JOIN_ROOM_ERROR = 53304,\n\n  /**\n   * 发布音频失败\n   */\n  AUDIO_PUBLISH_ERROR = 53305,\n\n  /**\n   * 发布视频失败\n   */\n  VIDEO_PUBLISH_ERROR = 53306,\n\n  /**\n   * 发布音视频失败\n   */\n  AUDIO_AND_VIDEO_PUBLISH_ERROR = 53307,\n\n  /**\n   * 查询房间用户信息失败\n   */\n  QUERY_JOINED_USER_INFO_ERROR = 53308,\n\n  /**\n   * 禁用启用视频时，房间内缺少视频流\n   */\n  MISSING_VIDEO_TRACK_ERROR = 53309,\n\n  /**\n   * 取消发布视频失败\n   */\n  UNPUBLISH_VIDEO_ERROR = 53310,\n\n  /**\n   * 会话不是群组\n   */\n  CONVERSATION_NOT_GROUP_ERROR = 53311,\n\n  /**\n  * 不在room中禁用音视频\n  */\n  NOT_IN_ROOM_ERROR = 53312\n\n}\n","/**\n * 挂断原因\n * @description\n * 根据原有 HangupReason 设计，己方原因与对方原因有相差 10 的差距.\n * 现有本地原因取值范围: 1 ~ 10, 远端原因取值范围: 11 ~ 20.\n * 为便于 HangupReason 扩展，保留 100 以内的取值.\n * 需要再次扩展时，己方原因使用: 21 ~ 30, 对应对方原因使用: 31 ~ 40,\n * 以此类推，\n * 己方原因使用: 41 ~ 50, 对方原因使用: 51 ~ 60,\n * 己方原因使用: 61 ~ 70, 对方原因使用: 71 ~ 80,\n * 己方原因使用: 71 ~ 90, 对方原因使用: 91 ~ 100.\n *\n * 各平台独有字段范围\n * Android 201 ~ 299\n * iOS     301 ~ 399\n * Web     401 ~ 499\n * 详细文档：https://gitbook.rongcloud.net/rtc-docs/#/rtc-client/ios/analysis/calllib/HangupReason\n */\nexport enum RCCallEndReason {\n  /**\n   * 己方取消已发出的通话请求\n   */\n  CANCEL = 1,\n  /**\n   * 己方拒绝收到的通话请求\n   */\n  REJECT = 2,\n  /**\n   * 己方挂断\n   */\n  HANGUP = 3,\n  /**\n   * 己方忙碌\n   */\n  BUSY_LINE = 4,\n  /**\n   * 己方未接听\n   */\n  NO_RESPONSE = 5,\n  /**\n   * 己方不支持当前音视频引擎\n   */\n  ENGINE_UNSUPPORTED = 6,\n  /**\n   * 己方网络错误\n   */\n  NETWORK_ERROR = 7,\n  /**\n   * 己方摄像头资源获取失败，可能是权限原因\n   */\n  GET_MEDIA_RESOURCES_ERROR = 8,\n  /**\n   * 己方资源发布失败\n   */\n  PUBLISH_ERROR = 9,\n  /**\n   * 己方订阅资源失败\n   */\n  SUBSCRIBE_ERROR = 10,\n  /**\n   * 对方取消发出的通话请求\n   */\n  REMOTE_CANCEL = 11,\n  /**\n   * 对方拒绝收到的通话请求\n   */\n  REMOTE_REJECT = 12,\n  /**\n   * 通话过程中对方挂断\n   */\n  REMOTE_HANGUP = 13,\n  /**\n   * 对方忙碌\n   */\n  REMOTE_BUSY_LINE = 14,\n  /**\n   * 对方未接听\n   */\n  REMOTE_NO_RESPONSE = 15,\n  /**\n   * 对方引擎不支持\n   */\n  REMOTE_ENGINE_UNSUPPORTED = 16,\n  /**\n   * 对方网络错误\n   */\n  REMOTE_NETWORK_ERROR = 17,\n  /**\n   * 对方摄像头资源获取失败，可能是权限原因\n   */\n  REMOTE_GET_MEDIA_RESOURCE_ERROR = 18,\n  /**\n   * 远端资源发布失败\n   */\n  REMOTE_PUBLISH_ERROR = 19,\n  /**\n   * 远端订阅资源失败\n   */\n  REMOTE_SUBSCRIBE_ERROR = 20,\n  /**\n   * 己方其他端已加入新通话\n   */\n  OTHER_CLIENT_JOINED_CALL = 21,\n  /**\n   * 己方其他端已在通话中\n   */\n  OTHER_CLIENT_IN_CALL = 22,\n  /**\n   * 己方被禁止通话\n   */\n  KICKED_BY_SERVER = 23,\n  /**\n   * 己方接听系统通话（移动端特有）\n  */\n  ACCEPT_SYSTEM_CALL = 24,\n  /**\n   * 远端其他端已加入新通话\n   */\n  REMOTE_OTHER_CLIENT_JOINED_CALL = 31,\n  /**\n   * 远端其他端已在通话中\n   */\n  REMOTE_OTHER_CLIENT_IN_CALL = 32,\n  /**\n   * 远端被禁止通话\n   */\n  REMOTE_KICKED_BY_SERVER = 33,\n  /**\n   * 远端接听系统通话（移动端特有）\n  */\n  REMOTE_ACCEPT_SYSTEM_CALL = 34,\n  /**\n   * 其他端接听\n   */\n  ACCEPT_BY_OTHER_CLIENT = 101,\n  /**\n   * 其他端挂断\n   */\n  HANGUP_BY_OTHER_CLIENT = 102,\n  /**\n   * 己方被对方加入黑名单\n   */\n  ADDED_TO_BLACKLIST = 103,\n  /**\n   * 音视频服务未开通\n   */\n  SERVICE_NOT_OPENED = 104\n}\n\n/**\n * 己方原因转为对方原因\n */\nexport const CallRemoteEndReason: { [key: number]: RCCallEndReason } = {\n  [RCCallEndReason.CANCEL]: RCCallEndReason.REMOTE_CANCEL,\n  [RCCallEndReason.REJECT]: RCCallEndReason.REMOTE_REJECT,\n  [RCCallEndReason.HANGUP]: RCCallEndReason.REMOTE_HANGUP,\n  [RCCallEndReason.BUSY_LINE]: RCCallEndReason.REMOTE_BUSY_LINE,\n  [RCCallEndReason.NO_RESPONSE]: RCCallEndReason.REMOTE_NO_RESPONSE,\n  [RCCallEndReason.ENGINE_UNSUPPORTED]: RCCallEndReason.REMOTE_ENGINE_UNSUPPORTED,\n  [RCCallEndReason.NETWORK_ERROR]: RCCallEndReason.REMOTE_NETWORK_ERROR,\n  [RCCallEndReason.GET_MEDIA_RESOURCES_ERROR]: RCCallEndReason.REMOTE_GET_MEDIA_RESOURCE_ERROR,\n  [RCCallEndReason.PUBLISH_ERROR]: RCCallEndReason.REMOTE_PUBLISH_ERROR,\n  [RCCallEndReason.SUBSCRIBE_ERROR]: RCCallEndReason.REMOTE_SUBSCRIBE_ERROR,\n  [RCCallEndReason.OTHER_CLIENT_JOINED_CALL]: RCCallEndReason.REMOTE_OTHER_CLIENT_JOINED_CALL,\n  [RCCallEndReason.OTHER_CLIENT_IN_CALL]: RCCallEndReason.REMOTE_OTHER_CLIENT_IN_CALL,\n  [RCCallEndReason.KICKED_BY_SERVER]: RCCallEndReason.REMOTE_KICKED_BY_SERVER,\n  [RCCallEndReason.REMOTE_NO_RESPONSE]: RCCallEndReason.NO_RESPONSE,\n  [RCCallEndReason.ACCEPT_SYSTEM_CALL]: RCCallEndReason.REMOTE_ACCEPT_SYSTEM_CALL,\n};\n","export enum RCCallMessageType {\n  /**\n   * 邀请消息\n   */\n  VCInvite = 'RC:VCInvite',\n  /**\n   * 响铃消息\n   */\n  VCRinging = 'RC:VCRinging',\n  /**\n   * 接听消息\n   */\n  VCAccept = 'RC:VCAccept',\n  /**\n   * 挂断消息\n   */\n  VCHangup = 'RC:VCHangup',\n  /**\n   * 群呼中 人员变更消息\n   */\n  VCModifyMem = 'RC:VCModifyMem',\n  /**\n   * 媒体类型修改消息\n   */\n  VCModifyMedia = 'RC:VCModifyMedia',\n}\n","export enum RCCallSessionState {\n  /**\n   * 等待建立连接\n   */\n  WAITING,\n  /**\n   * 会话维持中\n   */\n  KEEPING,\n  /**\n   * 会话已结束\n   */\n  END\n}\n","export enum RCCallUserState {\n  /**\n   * 用户不存在于通话中\n   */\n  NONE = 0,\n  /**\n   * 等待接听\n   */\n  WAITING,\n  /**\n   * 通话中\n   */\n  KEEPING,\n}\n","export enum RCCrossCallType {\n  /*!\n  同App通话\n  */\n  RCCallRoomTypeNormalCall = 0,\n  /*!\n  跨App通话\n  */\n  RCCallRoomTypeAcrossCall = 7\n}\n","import { timerSetTimeout } from '../helper';\n\nexport class Timer {\n  private _timerId: number = 0\n\n  private _startTime: number = 0\n\n  constructor(callback: Function, timeout: number) {\n    if (callback) {\n      this._timerId = timerSetTimeout(() => {\n        callback();\n      }, timeout);\n    }\n    this._startTime = Date.now();\n  }\n\n  stop(): {\n    startTime: number,\n    endTime: number,\n    duration: number\n    } {\n    clearTimeout(this._timerId);\n\n    const endTime = Date.now();\n    let duration = endTime - this._startTime;\n    if (this._startTime === 0) {\n      duration = 0;\n    }\n\n    return {\n      startTime: this._startTime,\n      endTime,\n      duration,\n    };\n  }\n\n  reset() {\n    this._startTime = 0;\n  }\n}\n","import {\n  ConversationType, ILogger, IReceivedMessage, IRuntime, RTCPluginContext,\n} from '@rongcloud/engine';\nimport { eventEmitter, getCallDeviceId } from '../helper';\nimport { MsgCallStatus } from './enums/MsgCallStatus';\nimport { RCCallErrorCode } from './enums/RCCallCode';\nimport { CallRemoteEndReason, RCCallEndReason } from './enums/RCCallEndReason';\nimport { RCCallMediaType } from './enums/RCCallMediaType';\nimport { RCCallMessageType } from './enums/RCCallMessageType';\nimport { RCCallSessionState } from './enums/RCCallSessionState';\nimport { RCCallUserState } from './enums/RCCallUserState';\nimport {\n  IExistedUserPofiles, IHungupMsgContent, IInviteMsgContent, IMemberModifyMsgContent, IRingingMsgContent,\n} from './interfaces/IMessageHandler';\nimport {\n  ICallStateMachineWatchers, IEndSummary, IInvitedUsers, IUserData, IUserProfile, IInviteOptions,\n} from './interfaces/IStateMachine';\nimport { CallMessageHandler } from './MessageHandler';\nimport { RCCrossCallType } from './enums/RCCrossCallType';\nimport { Timer } from './Timer';\n\nexport class RCCallStateMachine {\n  /**\n   * 房间状态\n   */\n  private _sessionState: RCCallSessionState | null = null\n\n  /**\n   * 用户状态及信息\n   */\n  private _userInfo: { [userId: string]: IUserData } = {}\n\n  /**\n   * 用户计时器映射\n   */\n  private _userTimers: { [userId: string]: Timer } = {}\n\n  /**\n   * 监听器\n   */\n  private _watchers!: ICallStateMachineWatchers\n\n  /**\n   * 呼叫超时时间 (单位：毫秒)\n   */\n  private _callTimeout: number = 60 * 1000\n\n  /**\n   * 通话建立开始时间\n   */\n  private _beginTimestamp: number = 0\n\n  /**\n   * 通话结束时间\n   */\n  private _endTimestamp: number = 0\n\n  /**\n   * 通话结束原因\n   */\n  private _endReason: RCCallEndReason | null = null\n\n  /**\n   * 主叫 ID\n   * 发起邀请为当前用户 ID\n   * 收到邀请为 senderUserId\n   * 收到人员变更邀请为消息体中 callerId\n   */\n  private _callerId: string | null = null\n\n  /**\n   * 当次通话邀请者 ID\n   * 发起邀请为当前用户 ID、收到邀请为 senderUserId、收到人员变更邀请为消息体中 senderUserId\n   */\n  private _inviterId: string | null = null\n\n  /**\n   * 是否是跨 appkey\n   */\n  private _isCrossAppkey: boolean = false\n\n  private _hungupPushTitle: string = ''\n\n  private _hungupPushContent: string = ''\n\n  constructor(\n    private readonly _context: RTCPluginContext,\n    private readonly _runtime: IRuntime,\n    private readonly _logger: ILogger,\n    private readonly _callMsgHandler: CallMessageHandler,\n    private readonly _channelId: string,\n    private readonly _conversationType: ConversationType,\n    private readonly _targetId: string,\n    private _mediaType: RCCallMediaType,\n    private readonly _callId: string,\n  ) {\n    this._callMsgHandler.registerStateMachineEvent(this._callId, 'onRinging', this._onRinging.bind(this));\n    this._callMsgHandler.registerStateMachineEvent(this._callId, 'onAccept', this._onAccept.bind(this));\n    this._callMsgHandler.registerStateMachineEvent(this._callId, 'onMediaModify', this._onMediaModify.bind(this));\n    this._callMsgHandler.registerStateMachineEvent(this._callId, 'onHungup', this._onHungup.bind(this));\n  }\n\n  /**\n   * 获取校正后超时时间\n   */\n  private _getTimeout(sentTime: number): number {\n    let delayTime = this._context.getServerTime() - sentTime;\n    if (delayTime < 0) {\n      delayTime = 500; // 假设延迟时间为 500 ms\n    }\n    const timeout = this._callTimeout - delayTime;\n    this._logger.warn('_', `_getTimeout -> timeout: ${timeout}`);\n    return timeout;\n  }\n\n  private _clearTimerById(userId: string) {\n    this._logger.debug('_', `[RCCallStateMachine] before _clearTimerById  -> userId: ${userId} userTimers: ${JSON.stringify(this._userTimers)}`);\n    if (this._userTimers[userId]) {\n      this._userTimers[userId].stop();\n      delete this._userTimers[userId];\n    }\n    this._logger.debug('_', `[RCCallStateMachine] after _clearTimerById -> userTimers: ${JSON.stringify(this._userTimers)}`);\n  }\n\n  /**\n   * 通知 call 层房间状态变更及原因\n   */\n  private _notifyStateChange(state: RCCallSessionState, reason?: RCCallEndReason) {\n    this._logger.warn('_', `[RCCallStateMachine] notifyStateChange -> info: ${JSON.stringify({\n      state, reason,\n    })}`);\n\n    this._endReason = reason || null;\n    if (this._sessionState !== state) {\n      this._sessionState = state;\n      this._watchers?.onStateChange({ state, reason });\n    }\n    if (state === RCCallSessionState.END) {\n      // 当状态机结束时，需在 CallEngine 清除\n      eventEmitter.emit('onStateMachineClose', this._callId);\n      this._callMsgHandler.unregisterStateMachineEvent(this._callId);\n    }\n  }\n\n  /**\n   * 通知 call 层人员状态变更及原因\n   */\n  private _notifyUserStateChange(user: IUserData, reason?: RCCallEndReason) {\n    this._logger.warn('_', `[RCCallStateMachine] notifyUserStateChange -> info: ${JSON.stringify({\n      user, reason,\n    })}`);\n    this._watchers?.onUserStateChange({ user, reason });\n  }\n\n  private _otherClientHandle(message: IReceivedMessage) {\n    const { senderUserId, content: { user: userProfile, reason: hungupReason }, messageType } = message;\n    this._userInfo[senderUserId] = {\n      userId: senderUserId,\n      state: RCCallUserState.NONE,\n      isCaller: false,\n      isRemote: false,\n    };\n    // 多端接听、多端拒绝，清除收到邀请后起的计时器\n    for (const userId in this._userTimers) {\n      this._clearTimerById(userId);\n    }\n    // 其他端接听\n    let endReason = RCCallEndReason.ACCEPT_BY_OTHER_CLIENT;\n    if (messageType === RCCallMessageType.VCHangup) {\n      if (hungupReason === RCCallEndReason.BUSY_LINE) {\n        endReason = RCCallEndReason.OTHER_CLIENT_IN_CALL;\n      } else if (hungupReason === RCCallEndReason.NO_RESPONSE) {\n        endReason = RCCallEndReason.NO_RESPONSE;\n      } else {\n        endReason = RCCallEndReason.HANGUP_BY_OTHER_CLIENT;\n      }\n    }\n    // 添加用户简要信息\n    Object.assign(this._userInfo[senderUserId], <IUserProfile>userProfile);\n    this._notifyUserStateChange(this._userInfo[senderUserId], endReason);\n    this._notifyStateChange(RCCallSessionState.END, endReason);\n  }\n\n  /**\n   * 正在通话中，且不是当前已接通用户设备（deviceId）发来的消息\n  */\n  private _isRemoteInvalidMsg(remoteUserId: string, msgDeviceId: string): boolean {\n    if (!this._userInfo[remoteUserId]) {\n      return false;\n    }\n\n    if (!this._userInfo[remoteUserId].deviceId || !msgDeviceId) {\n      return false;\n    }\n\n    if ((this._userInfo[remoteUserId].state === RCCallUserState.KEEPING && this._userInfo[remoteUserId].deviceId !== msgDeviceId)) {\n      return true;\n    }\n\n    return false;\n  }\n\n  private _onRinging(message: IReceivedMessage) {\n    const { senderUserId: suid, content: { user: userProfile, deviceId } } = message;\n    const senderUserId = this._isCrossAppkey ? suid.split('_')[1] : suid;\n    // 正在通话中，远端多端发的消息直接过滤\n    if (this._isRemoteInvalidMsg(senderUserId, deviceId)) {\n      this._logger.debug('_', '[RCCallStateMachine] onRinging -> not the remote device that is currently talking');\n      return;\n    }\n\n    if (this._context.getCurrentId() === senderUserId) {\n      return; // 多端处理\n    }\n    this._watchers.onRinging({ userId: senderUserId, ...<IUserProfile>userProfile });\n  }\n\n  private _onAccept(message: IReceivedMessage) {\n    const { senderUserId: suid, content: { user: userProfile, deviceId: senderDeviceId }, sentTime } = message;\n    const senderUserId = this._isCrossAppkey ? suid.split('_')[1] : suid;\n    const currentUserId = this._context.getCurrentId();\n    // 正在通话中，远端多端发的消息直接过滤\n    if (this._isRemoteInvalidMsg(senderUserId, senderDeviceId)) {\n      this._logger.debug('_', '[RCCallStateMachine] _onAccept -> not the remote device that is currently talking');\n      return;\n    }\n\n    if (currentUserId === senderUserId) {\n      // 多端处理\n      this._otherClientHandle(message);\n      return;\n    }\n\n    // 群组通话时： A、B 通话 A邀请C, C同意接听，这时B没有C的userId对应的计时器，所以这里判断一下\n    if (this._userTimers[senderUserId]) {\n      // 清除呼叫超时计时器\n      this._clearTimerById(senderUserId);\n    }\n\n    // 修改并通知房间人员状态\n    const ids = (this._conversationType === ConversationType.PRIVATE) ? [currentUserId, senderUserId] : [senderUserId];\n    ids.forEach((userId) => {\n      const isCurrentUserId = userId === currentUserId;\n      this._userInfo[userId] = {\n        userId,\n        state: RCCallUserState.KEEPING,\n        isCaller: isCurrentUserId,\n        isRemote: isCurrentUserId,\n        deviceId: isCurrentUserId ? getCallDeviceId(this._runtime) : senderDeviceId,\n      };\n      if (!isCurrentUserId) {\n        this._beginTimestamp = Date.now();\n        // 添加用户简要信息\n        Object.assign(this._userInfo[senderUserId], <IUserProfile>userProfile);\n      }\n      this._notifyUserStateChange(this._userInfo[userId]);\n    });\n    if (this.getCallerId() === currentUserId) {\n      this._notifyStateChange(RCCallSessionState.KEEPING);\n    }\n    // 抛出onAccept时状态已经就绪\n    this._watchers.onAccept({ userId: senderUserId });\n  }\n\n  private _onMediaModify(message: IReceivedMessage) {\n    const { senderUserId, content: { mediaType, user: userProfile, deviceId } } = message;\n    // 正在通话中，远端多端发的消息直接过滤\n    if (this._isRemoteInvalidMsg(senderUserId, deviceId)) {\n      this._logger.debug('_', '[RCCallStateMachine] _onMediaModify -> not the remote device that is currently talking');\n      return;\n    }\n\n    if (this._context.getCurrentId() === senderUserId) { // 多端处理\n      return;\n    }\n\n    // 更新 mediaType\n    this._mediaType = mediaType;\n    this._watchers.onMediaModify({\n      sender: { userId: senderUserId, ...<IUserProfile>userProfile },\n      mediaType,\n    });\n  }\n\n  private _onHungup(message: IReceivedMessage) {\n    const { senderUserId: suid, content } = message;\n    const senderUserId = this._isCrossAppkey ? suid.split('_')[1] : suid;\n    const { reason, user: userProfile, deviceId } = content as IHungupMsgContent;\n    const currentUserId = this._context.getCurrentId();\n    // 正在通话中，远端多端发的消息直接过滤\n    if (this._isRemoteInvalidMsg(senderUserId, deviceId)) {\n      this._logger.debug('_', '[RCCallStateMachine] _onHungup -> not the remote device that is currently talking');\n      return;\n    }\n\n    if (currentUserId === senderUserId) { // 多端处理 抛出多端已处理 reason\n      this._otherClientHandle(message);\n      return;\n    }\n\n    if (this._sessionState === RCCallSessionState.END) {\n      // 如果己方房间状态已结束，再收到 hungup 消息不再处理\n      this._logger.info('_', `[RCCallStateMachine] Invalid hang up message, current room status has ended -> sessionState: ${this._sessionState}`);\n      return;\n    }\n\n    if (this._userInfo[senderUserId]) {\n      // 修改内存态数据并通知\n      this._userInfo[senderUserId].state = RCCallUserState.NONE;\n      this._endTimestamp = Date.now();\n      // 添加用户简要信息\n      Object.assign(this._userInfo[senderUserId], <IUserProfile>userProfile);\n      this._notifyUserStateChange(this._userInfo[senderUserId], CallRemoteEndReason[reason]);\n      delete this._userInfo[senderUserId];\n    }\n\n    // timer 清除\n    if (CallRemoteEndReason[reason] === RCCallEndReason.REMOTE_CANCEL) {\n      // 远端取消通话，没有远端用户，清除己端的接听超时计时器\n      this.getRemoteUserIds().length < 1 && this._clearTimerById(currentUserId);\n    } else if (this.getInviterId() === currentUserId) {\n      // 远端拒绝、忙碌、未接听等，通过自己是否是主叫来判断要清除呼叫超时计时器 或 接听超时计时器\n      this._clearTimerById(senderUserId);\n    } else {\n      this.getRemoteUserIds().length < 1 && this._clearTimerById(currentUserId);\n    }\n\n    // 房间人员信息少于两个，通知房间状态结束\n    const isLessThanTwo = Object.keys(this._userInfo).length < 2;\n    // 群呼中邀请者挂断（非呼叫发起者）且房间中被邀请者都未接听，通知房间状态结束\n    const isInviteUser = this._inviterId === senderUserId;\n    const isNoOneAnswered = Object.values(this._userInfo).every((user) => user.state !== RCCallUserState.KEEPING);\n    if (isLessThanTwo || (isInviteUser && isNoOneAnswered)) {\n      this._notifyStateChange(RCCallSessionState.END, CallRemoteEndReason[reason]);\n    }\n\n    // 抛出 onHungup 时状态已经就绪\n    this._watchers.onHungup({ userId: senderUserId, ...<IUserProfile>userProfile }, CallRemoteEndReason[reason]);\n  }\n\n  /**\n   * 注册事件监听\n   * @params watchers\n   */\n  registerEventListener(watchers: ICallStateMachineWatchers) {\n    this._watchers = watchers;\n  }\n\n  /**\n   * 收到 invite 消息时状态机更新（CallEngine 内部调用）\n   * @param message 接收消息\n   */\n  __onInvite(message: IReceivedMessage) {\n    const { senderUserId: suid, content, sentTime } = message;\n    const {\n      inviteUserIds: inids, user: userProfile, deviceId, roomType,\n    } = content as IInviteMsgContent;\n    // 处理跨 appkey 邀请 senderUserId 带有 appkey 与本地 currentUserId 匹配问题\n    // 同时记录本地是否为跨 appkey 状态\n    let senderUserId: string;\n    if (roomType === RCCrossCallType.RCCallRoomTypeAcrossCall) {\n      [, senderUserId] = suid.split('_');\n      this._watchers.crossAppkey(true);\n      this._isCrossAppkey = true;\n    } else {\n      senderUserId = suid;\n    }\n    const currentUserId = this._context.getCurrentId();\n    // 正在通话中，远端多端发的消息直接过滤\n    if (this._isRemoteInvalidMsg(senderUserId, deviceId)) {\n      this._logger.debug('_', '[RCCallStateMachine] __onInvite -> not the remote device that is currently talking');\n      return;\n    }\n\n    if (currentUserId === senderUserId) {\n      // 多端处理\n      return;\n    }\n\n    this._callerId = this._inviterId = senderUserId;\n    const userIdList = [suid, ...inids];\n    // 收到邀请后，内部回应响铃消息, userIds 传除自己的所有人\n    this._callMsgHandler.sendRinging({\n      conversationType: this._conversationType,\n      targetId: this._targetId,\n      channelId: this._channelId,\n      callId: this._callId,\n      userIds: userIdList.filter((id) => {\n        if (this._isCrossAppkey) {\n          return id.split('_')[1] !== currentUserId;\n        }\n        return id !== currentUserId;\n      }),\n    });\n\n    const inviteUserIds = this._isCrossAppkey ? [inids[0].split('_')[1]] : inids;\n\n    // 同步本地用户列表需要使用拆分后的 UserId 保证本地其他方法匹配 userid\n    const allUserIds = [senderUserId, ...inviteUserIds];\n\n    // 修改并通知房间人员状态\n    allUserIds.forEach((userId) => {\n      this._userInfo[userId] = {\n        userId,\n        state: RCCallUserState.WAITING,\n        isCaller: userId === senderUserId,\n        isRemote: userId !== currentUserId,\n      };\n      if (userId === senderUserId) {\n        // 给 senderUser 添加用户简要信息及 deviceId\n        Object.assign(this._userInfo[userId], <IUserProfile>userProfile, { deviceId });\n      }\n      this._notifyUserStateChange(this._userInfo[userId]);\n      // 给所有被邀请人启动接听计时器\n      if (userId !== senderUserId) {\n        this._userTimers[userId] = new Timer(() => {\n          const reason = userId === currentUserId ? RCCallEndReason.NO_RESPONSE : RCCallEndReason.REMOTE_NO_RESPONSE;\n          if (userId === currentUserId) { // 群聊中己方超时需发送挂断\n            this._hungupHandle(reason);\n          } else { // 其他人员超时只通知人员状态变更\n            this._userInfo[userId] && (this._userInfo[userId].state = RCCallUserState.NONE);\n            this._notifyUserStateChange(this._userInfo[userId]);\n            this._watchers.onHungup(this._userInfo[userId], reason);\n            delete this._userInfo[userId];\n          }\n        }, this._getTimeout(sentTime));\n      }\n    });\n    // 房间状态通知\n    this._notifyStateChange(RCCallSessionState.WAITING);\n  }\n\n  /**\n   * 收到 memberModify 消息时状态机更新（CallEngine 内部调用）\n   * @param message 接收消息\n   */\n  __onMemberModify(message: IReceivedMessage) {\n    const { senderUserId, content, sentTime } = message;\n    const {\n      user: userProfile, existedUserPofiles, caller, deviceId, inviteUserIds, mediaType,\n    } = content as IMemberModifyMsgContent;\n    const currentUserId = this._context.getCurrentId();\n    // 正在通话中，远端多端发的消息直接过滤\n    // this._userInfo[senderUserId] && !this._userInfo[senderUserId].deviceId && !deviceId &&\n    if (this._isRemoteInvalidMsg(senderUserId, deviceId)) {\n      this._logger.debug('_', '[RCCallStateMachine] __onMemberModify -> not the remote device that is currently talking');\n      return;\n    }\n    if (currentUserId === senderUserId) {\n      // 多端处理\n      return;\n    }\n    this._callerId = caller;\n    this._inviterId = senderUserId;\n    inviteUserIds.forEach((id) => {\n      existedUserPofiles.push({ userId: id, mediaType, callStatus: MsgCallStatus.INCOMING });\n    });\n    // 己方状态为 NONE (未存在己方用户信息) 说明自己为被邀请者, 需发送 ringing\n    const isNewInvitedUser = inviteUserIds.includes(currentUserId);\n    if (isNewInvitedUser) {\n      const needRingingUserIds: string[] = [];\n      existedUserPofiles.forEach((userInfo) => {\n        if (userInfo.userId !== currentUserId) {\n          needRingingUserIds.push(userInfo.userId);\n        }\n      });\n      this._callMsgHandler.sendRinging({\n        conversationType: this._conversationType,\n        targetId: this._targetId,\n        channelId: this._channelId,\n        callId: this._callId,\n        userIds: needRingingUserIds,\n      });\n\n      // 被邀请者通知 call 层房间状态变更\n      this._notifyStateChange(RCCallSessionState.WAITING);\n    } else {\n      // 已在通话流程中用户，抛出人员变更监听\n      this._watchers.onMemberModify({\n        sender: { userId: senderUserId, ...userProfile },\n        invitedUsers: inviteUserIds.map((id) => ({ userId: id })),\n      });\n    }\n\n    // 更新全量用户状态\n    existedUserPofiles.forEach((userInfo) => {\n      const { userId, callStatus } = userInfo;\n      // 人员为结束状态时，不再更新\n      if (callStatus === MsgCallStatus.IDLE) return;\n      this._userInfo[userId] = {\n        userId,\n        state: callStatus !== MsgCallStatus.CONNECTED ? RCCallUserState.WAITING : RCCallUserState.KEEPING,\n        isCaller: senderUserId === userId,\n        isRemote: currentUserId !== userId,\n      };\n      // 给 senderUser 添加用户简要信息 及 deviceId\n      if (userId === senderUserId) {\n        Object.assign(this._userInfo[userId], <IUserProfile>userProfile, { deviceId });\n      }\n      // 通知人员变更\n      this._notifyUserStateChange(this._userInfo[userId]);\n      // 除了 callMsgStatus.connected ,其他人都起计时器\n\n      if (callStatus !== MsgCallStatus.CONNECTED && !this._userTimers[userId]) {\n        /**\n         * 仅给被邀请成员启动定时器\n         */\n        if (!inviteUserIds.includes(userId)) {\n          return;\n        }\n        // 启动计时器\n        this._userTimers[userId] = new Timer(() => {\n          // 呼叫超时处理状态、并通知 call 层\n          this._userInfo[userId] && (this._userInfo[userId].state = RCCallUserState.NONE);\n          const reason = userId === currentUserId ? RCCallEndReason.NO_RESPONSE : RCCallEndReason.REMOTE_NO_RESPONSE;\n          // 通知 call 层人员状态变更\n          this._notifyUserStateChange(this._userInfo[userId], reason);\n          try {\n            // 呼叫超时未接抛出onHungup\n            this._watchers.onHungup(this._userInfo[userId], reason);\n          } catch (error: any) {\n            this._logger.error('_', `[RCCallStateMachine] call onhungup error -> ${error?.stack}`);\n          }\n\n          delete this._userInfo[userId];\n          // 房间人员信息少于两个 或 未接听的是自己，通知房间状态结束\n          if (Object.keys(this._userInfo).length < 2 || userId === currentUserId) {\n            this._notifyStateChange(RCCallSessionState.END, reason);\n          }\n        }, this._getTimeout(sentTime));\n      }\n    });\n  }\n\n  /**\n   * 处理已有 session ，不允许再接听新会话情况\n   */\n  __handleInviteInSession() {\n    this._logger.info('_', 'StateMachine -> __handleInviteInSession');\n    // 修改所有用户状态并抛出，清空计时器\n    for (const userId in this._userInfo) {\n      this._userInfo[userId].state && (this._userInfo[userId].state = RCCallUserState.NONE);\n      this._notifyUserStateChange(this._userInfo[userId]);\n      // 将已存在的计时器都清掉\n      this._clearTimerById(userId);\n    }\n    // 直接终止当前 session 状态\n    this._notifyStateChange(RCCallSessionState.END, RCCallEndReason.BUSY_LINE);\n    // 发送挂断消息，并携带 己方忙碌 原因\n    this._callMsgHandler.sendHungup({\n      channelId: this._channelId,\n      conversationType: this._conversationType,\n      targetId: this._targetId,\n      callId: this._callId,\n      reason: RCCallEndReason.BUSY_LINE,\n      userIds: this.getRemoteUserIds(),\n    });\n  }\n\n  /**\n   * 主动呼叫 (CallEngine 内部调用)\n   * @param userIds 被邀请用户 ID 列表\n   * @param extra 消息的扩展信息\n   * @param pushTitle 通知的标题\n   * @param pushContent 通知的内容\n   */\n  async __call(userIds: string[], extra: string = '', pushTitle: string = '', pushContent: string = '', isCrossAppkey = false) {\n    this._logger.debug('_', `[RCCallStateMachine] invite -> userIds: ${JSON.stringify(userIds)}`);\n    // 发送邀请消息\n    const currentUserId = this._callerId = this._inviterId = this._context.getCurrentId();\n    const { code, message } = await this._callMsgHandler.sendInvite({\n      roomType: isCrossAppkey ? RCCrossCallType.RCCallRoomTypeAcrossCall : RCCrossCallType.RCCallRoomTypeNormalCall,\n      channelId: this._channelId,\n      conversationType: this._conversationType,\n      targetId: this._targetId,\n      callId: this._callId,\n      extra,\n      pushTitle,\n      pushContent,\n      mediaType: this._mediaType,\n      inviteUserIds: userIds.filter((id) => id !== currentUserId),\n    });\n\n    this._isCrossAppkey = isCrossAppkey;\n\n    if (code === RCCallErrorCode.SUCCESS) {\n      const { sentTime } = message!;\n\n      /**\n       * 跨 appkey 仅存在单聊，对端 userId 需取下划线后面的部分\n       */\n      const ids = isCrossAppkey ? [currentUserId, ...[userIds[0].split('_')[1]]] : [currentUserId, ...userIds];\n\n      // 遍历更新房间所有人员状态、并给被叫启动计时器\n      ids.forEach((userId) => {\n        const isCaller = userId === currentUserId;\n        this._userInfo[userId] = {\n          userId,\n          state: RCCallUserState.WAITING,\n          isCaller,\n          isRemote: !isCaller,\n        };\n        // 通知 call 层人员状态变更\n        this._notifyUserStateChange(this._userInfo[userId]);\n\n        // 启动呼叫超时计时器\n        if (!isCaller) {\n          this._userTimers[userId] = new Timer(() => {\n            // 呼叫超时处理状态、并通知 call 层\n            this._userInfo[userId] && (this._userInfo[userId].state = RCCallUserState.NONE);\n            // 通知 call 层人员状态变更\n            this._notifyUserStateChange(this._userInfo[userId], RCCallEndReason.REMOTE_NO_RESPONSE);\n\n            // 呼叫超时未接抛出onHungup\n            this._watchers.onHungup(this._userInfo[userId], RCCallEndReason.REMOTE_NO_RESPONSE);\n\n            delete this._userInfo[userId];\n            // 房间人员信息少于两个，通知房间状态\n            if (Object.keys(this._userInfo).length < 2) {\n              this._notifyStateChange(RCCallSessionState.END, RCCallEndReason.REMOTE_NO_RESPONSE);\n            }\n            // 远端人员大于 1 且己方不为正在通话时，发送 hungup, 告诉远端自己退出通话\n            const isNeedSendHungup = this.getRemoteUserIds().length === 0 && (this._userInfo[currentUserId].state !== RCCallUserState.KEEPING);\n            if (isNeedSendHungup) {\n              this._hungupHandle(RCCallEndReason.REMOTE_NO_RESPONSE);\n            }\n          }, this._getTimeout(sentTime));\n        }\n      });\n\n      // 通知 call 层房间状态变更\n      this._notifyStateChange(RCCallSessionState.WAITING);\n    } else {\n      const endCode = code === RCCallErrorCode.REJECTED_BY_BLACKLIST ? RCCallEndReason.ADDED_TO_BLACKLIST : RCCallEndReason.NETWORK_ERROR;\n      this._notifyStateChange(RCCallSessionState.END, endCode);\n    }\n\n    return { code };\n  }\n\n  /**\n   * 接听\n   */\n  async accept(): Promise<{ code: RCCallErrorCode }> {\n    this._logger.debug('_', '[RCCallStateMachine] accept');\n\n    // 发送接听消息\n    const currentUserId = this._context.getCurrentId();\n    const { code, message } = await this._callMsgHandler.sendAccept({\n      channelId: this._channelId,\n      conversationType: this._conversationType,\n      targetId: this._targetId,\n      callId: this._callId,\n      mediaType: this._mediaType,\n      userIds: this.getRemoteUserIds(),\n    });\n\n    // 清除接听超时计时器\n    this._clearTimerById(currentUserId);\n    if (code === RCCallErrorCode.SUCCESS) {\n      const { sentTime } = message!;\n      this._userInfo[currentUserId] && (this._userInfo[currentUserId].state = RCCallUserState.KEEPING);\n      this._beginTimestamp = Date.now();\n      // 通知 call 层人员状态变更\n      this._notifyUserStateChange(this._userInfo[currentUserId]);\n      // 修改并通知 call 层房间状态变更\n      this._notifyStateChange(RCCallSessionState.KEEPING);\n    } else {\n      this._userInfo[currentUserId] && (this._userInfo[currentUserId].state = RCCallUserState.NONE);\n      // 通知 call 层人员状态变更\n      this._notifyUserStateChange(this._userInfo[currentUserId]);\n      // 修改并通知 call 层房间状态变更\n      const endReason = code === RCCallErrorCode.REJECTED_BY_BLACKLIST ? RCCallEndReason.ADDED_TO_BLACKLIST : RCCallEndReason.NETWORK_ERROR;\n      this._notifyStateChange(RCCallSessionState.END, endReason);\n    }\n\n    return { code };\n  }\n\n  /**\n   * 群呼叫中继续邀请\n   * @param userIds 被邀请用户 ID 列表\n   */\n  async invite(userIds: string[], options: IInviteOptions = {}): Promise<{ code: RCCallErrorCode }> {\n    if (this._conversationType !== ConversationType.GROUP) {\n      return { code: RCCallErrorCode.CONVERSATION_NOT_GROUP_ERROR };\n    }\n\n    this._logger.debug('_', `[RCCallStateMachine] invite -> userIds: ${JSON.stringify(userIds)}`);\n    const currentUserId = this._context.getCurrentId();\n    const existedUserIds = Object.keys(this._userInfo);\n    const existedUserPofiles: IExistedUserPofiles[] = existedUserIds.map((userId) => {\n      let callStatus = MsgCallStatus.CONNECTED;\n      // 包含被邀请者，或者用户状态为 wating 时，状态为 响铃\n      if (userIds.includes(userId) || this._userInfo[userId].state === RCCallUserState.WAITING) {\n        callStatus = MsgCallStatus.RINGING;\n      }\n      return {\n        userId,\n        mediaType: this._mediaType,\n        callStatus,\n        mediaId: userId,\n      };\n    });\n\n    const extra = options.extra || '';\n    const pushTitle = options.pushTitle || '';\n    const pushContent = options.pushContent || '';\n\n    // 发送人员变更消息\n    const { code, message } = await this._callMsgHandler.sendMemeberModify({\n      channelId: this._channelId,\n      conversationType: this._conversationType,\n      targetId: this._targetId,\n      callId: this._callId,\n      extra,\n      pushTitle,\n      pushContent,\n      mediaType: this._mediaType,\n      // 除自己，其他所有人均需收到 modify 消息，以此更新状态\n      inviteUserIds: userIds.filter((id) => id !== currentUserId),\n      callerId: this.getCallerId(),\n      existedUserPofiles,\n      directionalUserIdList: [...existedUserIds, ...userIds].filter((id) => id !== currentUserId),\n    });\n\n    if (code === RCCallErrorCode.SUCCESS) {\n      const { sentTime } = message!;\n      // 修改被邀请人状态，并通知\n      userIds.forEach((userId) => {\n        this._userInfo[userId] = {\n          userId,\n          state: RCCallUserState.WAITING,\n          isCaller: false,\n          isRemote: true,\n        };\n        // 通知 call 层人员状态变更\n        this._notifyUserStateChange(this._userInfo[userId]);\n        // 启动呼叫计时器\n        this._userTimers[userId] = new Timer(() => {\n          // 呼叫超时处理状态、并通知 call 层\n          this._userInfo[userId] && (this._userInfo[userId].state = RCCallUserState.NONE);\n          // 通知 call 层人员状态变更\n          this._notifyUserStateChange(this._userInfo[userId], RCCallEndReason.REMOTE_NO_RESPONSE);\n\n          // 呼叫超时未接抛出onHungup\n          this._watchers.onHungup(this._userInfo[userId], RCCallEndReason.REMOTE_NO_RESPONSE);\n\n          delete this._userInfo[userId];\n          // 房间人员信息少于两个，通知房间状态\n          if (Object.keys(this._userInfo).length < 2) {\n            this._notifyStateChange(RCCallSessionState.END, RCCallEndReason.REMOTE_NO_RESPONSE);\n          }\n        }, this._getTimeout(sentTime));\n      });\n    } else {\n      // 通知人员变更\n      userIds.forEach((userId) => {\n        this._userInfo[userId] = {\n          userId,\n          state: RCCallUserState.NONE,\n          isCaller: false,\n          isRemote: true,\n        };\n        // 通知 call 层人员状态变更\n\n        const endReason = code === RCCallErrorCode.REJECTED_BY_BLACKLIST ? RCCallEndReason.ADDED_TO_BLACKLIST : RCCallEndReason.NETWORK_ERROR;\n        this._notifyUserStateChange(this._userInfo[userId], endReason);\n      });\n    }\n\n    return { code };\n  }\n\n  private async _hungupHandle(reason: RCCallEndReason): Promise<{ code: RCCallErrorCode }> {\n    const currentUserId = this._context.getCurrentId();\n    // 发送挂断类型消息\n    let code = RCCallErrorCode.SUCCESS;\n\n    const hangupParams = {\n      channelId: this._channelId,\n      conversationType: this._conversationType,\n      targetId: this._targetId,\n      callId: this._callId,\n      reason,\n      userIds: this.getRemoteUserIds(),\n      pushTitle: this._hungupPushTitle,\n      pushContent: this._hungupPushContent,\n    };\n\n    if (reason === RCCallEndReason.OTHER_CLIENT_JOINED_CALL) {\n      // 己端被多端加入 RTC 房间挤掉后，不再等发挂断消息的结果，防止走 userLeave 导致挂断原因不准确\n      this._callMsgHandler.sendHungup(hangupParams);\n    } else {\n      const { code: msgCode } = await this._callMsgHandler.sendHungup(hangupParams);\n      code = msgCode;\n    }\n\n    this._endTimestamp = Date.now();\n    // 更新通话人员状态、通话结束时间 并通知\n    for (const userId in this._userInfo) {\n      this._userInfo[userId].state = RCCallUserState.NONE;\n      if (userId === currentUserId) {\n        this._notifyUserStateChange(this._userInfo[userId], reason);\n      } else {\n        this._notifyUserStateChange(this._userInfo[userId]);\n      }\n      delete this._userInfo[userId];\n    }\n\n    if (Object.keys(this._userInfo).length < 2) {\n      // 清空内存态数据，并通知\n      this._notifyStateChange(RCCallSessionState.END, reason);\n    }\n\n    return { code };\n  }\n\n  /**\n   * 挂断\n   */\n  async hungup(pushTitle: string = '', pushContent: string = ''): Promise<{ code: RCCallErrorCode }> {\n    this._logger.debug('_', '[RCCallStateMachine] hungup');\n    const currentUserId = this._context.getCurrentId();\n    // 默认挂断 reason 为通话过程中，己方正常取消通话\n    let reason = RCCallEndReason.HANGUP;\n    /**\n     * 若超时计时器存在, 己方为 caller 原因为己方取消通话,\n     * 己方为 callee 且未进入通话中，原因为己方拒绝通话\n     */\n    if (Object.keys(this._userTimers).length > 0) {\n      if (this._userInfo[currentUserId].isCaller) {\n        reason = RCCallEndReason.CANCEL;\n      } else if (this._userInfo[currentUserId].state === RCCallUserState.WAITING) {\n        reason = RCCallEndReason.REJECT;\n      }\n    }\n\n    // 清除所有超时计时器\n    for (const userId in this._userTimers) {\n      this._clearTimerById(userId);\n    }\n    return this._hungupHandle(reason);\n  }\n\n  /**\n   * 修改通话媒体类型\n   * @param mediaType RCCallMediaType.AUDIO 改为音频通话 | RCCallMediaType.AUDIO_VIDEO 改为音视频通话\n   */\n  async changeMediaType(mediaType: RCCallMediaType): Promise<{ code: RCCallErrorCode }> {\n    this._logger.debug('_', `[RCCallStateMachine] changeMediaType -> mediaType: ${mediaType}`);\n    const { code } = await this._callMsgHandler.sendMediaModify({\n      channelId: this._channelId,\n      conversationType: this._conversationType,\n      targetId: this._targetId,\n      callId: this._callId,\n      mediaType,\n      userIds: this.getRemoteUserIds(),\n    });\n    if (code === RCCallErrorCode.SUCCESS) {\n      this._mediaType = mediaType;\n    }\n    return { code };\n  }\n\n  /**\n   * 用户加入通话补偿机制（rtc userJoin 事件触发）\n   * 主叫呼叫后，未收到被叫 accept 消息，但收到了 userJoin 同样补偿更新用户、房间状态、呼叫计时器\n   */\n  userJoin(userIds: string[]) {\n    this._logger.debug('_', `[RCCallStateMachine] userJoin -> userIds: ${JSON.stringify(userIds)}`);\n    // 延迟 300ms 防止 userJion 和 accept 消息都有的情况下， userJoin 比 accept 先到\n    setTimeout(() => {\n      userIds.forEach((userId) => {\n        const userInfo = this._userInfo[userId];\n        // 更新人员状态 (// 群组通话时： A向B、C发起通话 B先接听，C后接听，这时B没有C的userId对应的_userInfo，所以这里判断一下)\n        if (userInfo && userInfo.state !== RCCallUserState.KEEPING) {\n          userInfo.state = RCCallUserState.KEEPING;\n          this._notifyUserStateChange(userInfo);\n        }\n        // 更新房间状态\n        if (this._sessionState !== RCCallSessionState.KEEPING) {\n          this._notifyStateChange(RCCallSessionState.KEEPING);\n        }\n        // 停止并清除呼叫计时器\n        this._clearTimerById(userId);\n      });\n    }, 300);\n  }\n\n  /**\n   * 用户离开通话补偿机制（rtc userLeave、kickOff 事件触发）\n   * 通话中远端用户挂断，挂断消息未到，但是监听到 rtc userLeave 同样补偿更新用户、房间状态\n   */\n  userLeave(userIds: string[]) {\n    this._logger.debug('_', `[RCCallStateMachine] userLeave -> userIds: ${JSON.stringify(userIds)}`);\n    // 延迟 300ms 防止 userLeave 和 hungup 消息都有的情况下，userLeave 比 hungup 先到\n    setTimeout(() => {\n      userIds.forEach((userId) => {\n        const userInfo = this._userInfo[userId];\n        // 更新人员状态\n        if (userInfo && userInfo.state !== RCCallUserState.NONE) {\n          userInfo.state = RCCallUserState.NONE;\n          this._notifyUserStateChange(userInfo, RCCallEndReason.REMOTE_HANGUP);\n          this._watchers.onHungup(userInfo, RCCallEndReason.REMOTE_HANGUP);\n          delete this._userInfo[userId];\n        }\n        // 主叫方群聊邀请被叫方后离线，主叫方此时为等待状态后不再上线，被叫方接听后挂断至剩余一人时等待一分钟后挂断\n        const remoteUsersTimer = new Timer(() => {\n          const remoteUsers = this.getRemoteUsers();\n          // 远端只有一个用户且是等待状态，等待一分钟后挂断\n          if (remoteUsers.length === 1 && remoteUsers[0].state === 1) {\n            this._hungupHandle(RCCallEndReason.REMOTE_NETWORK_ERROR);\n          }\n        }, 60000);\n        // 更新房间状态\n        if (Object.keys(this._userInfo).length < 2 && this._sessionState !== RCCallSessionState.END) {\n          this._endTimestamp = Date.now();\n          this._notifyStateChange(RCCallSessionState.END, RCCallEndReason.REMOTE_HANGUP);\n        }\n      });\n    }, 300);\n  }\n\n  /**\n   * Call 层己方异常失败后调用的方法\n   * 触发时机：音视频服务异常、获取资源失败、加入 RTC 房间失败、发布|订阅失败\n   */\n  close(reason: RCCallEndReason) {\n    this._hungupHandle(reason);\n  }\n\n  setHungupPushConfig(pushConfig: string = '', pushContent: string = '') {\n    this._hungupPushTitle = pushConfig;\n    this._hungupPushContent = pushContent;\n  }\n\n  /**\n   * 通话唯一标识\n   */\n  getCallId(): string {\n    return this._callId;\n  }\n\n  /**\n   * 多组织 ID\n   */\n  getChannelId(): string {\n    return this._channelId;\n  }\n\n  /**\n   * 目标 ID，单呼对方人员 Id, 群呼群组 Id\n   */\n  getTargetId(): string {\n    return this._targetId;\n  }\n\n  /**\n   * 获取会话类型\n   */\n  getConversationType(): ConversationType {\n    return this._conversationType;\n  }\n\n  /**\n   * 获取远端成员 ID 列表\n   */\n  getRemoteUserIds(): string[] {\n    const allUserIds = Object.keys(this._userInfo);\n    const remoteUserIds = allUserIds.filter((id) => this._context.getCurrentId() !== id);\n    return remoteUserIds;\n  }\n\n  /**\n   * 获取远端成员信息列表\n   */\n  getRemoteUsers(): IUserData[] {\n    const remoteUser: IUserData[] = [];\n    const currentUserId = this._context.getCurrentId();\n    for (const uid in this._userInfo) {\n      const { userId } = this._userInfo[uid];\n      if (userId !== currentUserId) {\n        remoteUser.push(this._userInfo[uid]);\n      }\n    }\n    return remoteUser;\n  }\n\n  /**\n   * 获取房间状态\n   */\n  getState(): RCCallSessionState {\n    return this._sessionState === null ? RCCallSessionState.END : this._sessionState;\n  }\n\n  /**\n   * 获取人员状态\n   */\n  getUserState(userId: string): RCCallUserState {\n    return this._userInfo[userId]?.state;\n  }\n\n  /**\n   * 获取会话发起者 Id\n   */\n  getCallerId(): string {\n    return this._callerId!;\n  }\n\n  /**\n   * 获取当次会话邀请者 Id\n   */\n  getInviterId(): string {\n    return this._inviterId!;\n  }\n\n  /**\n   * 获取当前通话媒体类型\n   */\n  getMediaType(): RCCallMediaType {\n    return this._mediaType;\n  }\n\n  /**\n   * 通话挂断后可调用\n   */\n  getSummary(): IEndSummary {\n    // 通话时间计算\n    const beginTimestamp = this._beginTimestamp;\n    const endTimestamp = this._endTimestamp;\n    let duration = 0;\n    if (endTimestamp > beginTimestamp && beginTimestamp !== 0) {\n      duration = endTimestamp - beginTimestamp;\n    }\n\n    const summary = {\n      conversationType: this._conversationType,\n      channelId: this._channelId,\n      targetId: this._targetId,\n      mediaType: this._mediaType,\n      beginTimestamp,\n      endTimestamp,\n      duration,\n      endReason: this._endReason!,\n    };\n\n    this._logger.debug('_', `[RCCallStateMachine] getSummary -> summary: ${JSON.stringify(summary)}`);\n    return summary;\n  }\n}\n","/**\n * 通话媒体类型\n */\nexport enum RCCallMediaType {\n  /**\n   * 音频通话\n   */\n  AUDIO = 1,\n  /**\n   * 视频通话\n   */\n  AUDIO_VIDEO\n}\n","export enum MemberModifyType {\n  ADD = 1,\n  REMOVE\n}\n","/**\n * 平台\n */\nexport enum Platform {\n  WEB = 'Web',\n  IOS = 'iOS',\n  ANDROID = 'Android'\n}\n","import {\n  ConversationType, ILogger, IReceivedMessage, RTCPluginContext,\n} from '@rongcloud/engine';\nimport { CallRemoteEndReason, RCCallEndReason } from './enums/RCCallEndReason';\nimport { RCCallMediaType } from './enums/RCCallMediaType';\nimport { RCCallMessageType } from './enums/RCCallMessageType';\nimport { IHungupMsgContent, IInviteMsgContent, IMediaModifyMsgContent } from './interfaces/IMessageHandler';\n\nexport interface IOfflineRecord {\n  channelId: string,\n  conversationType: ConversationType,\n  targetId: string,\n  mediaType: RCCallMediaType,\n  callId: string,\n  inviterId: string,\n  endReason: RCCallEndReason,\n  beginTimestamp: number,\n  endTimestamp: number,\n  duration: number\n}\n\n/**\n * 离线通话记录器\n */\nexport class OfflineRecorder {\n  private _messages: IReceivedMessage[] | [] = []\n\n  private _channelId!: string\n\n  private _conversationType!: ConversationType\n\n  private _targetId!: string\n\n  private _mediaType!: RCCallMediaType\n\n  private _callId!: string\n\n  private _callerId!: string\n\n  private _inviterId!: string\n\n  private _endReason!: RCCallEndReason\n\n  private _beginTimestamp: number = 0\n\n  private _endTimestamp: number = 0\n\n  constructor(\n    private readonly _context: RTCPluginContext,\n    private readonly _logger: ILogger,\n    private readonly _onRecord: (record: IOfflineRecord) => void,\n  ) {\n\n  }\n\n  /**\n   * 到 invite | memberModify 结束\n   * 原因根据己方是否为主叫，主叫为远端未接听 被叫被己端未接听\n   */\n  private _doInvite(message: IReceivedMessage) {\n    const currentUserId = this._context.getCurrentId();\n    const {\n      channelId, conversationType, targetId, senderUserId, content,\n    } = message;\n    const { callId, mediaType } = content as IInviteMsgContent;\n    this._channelId = channelId!;\n    this._conversationType = conversationType;\n    this._targetId = targetId;\n    this._callId = callId;\n    this._mediaType = mediaType;\n    const isCaller = senderUserId === currentUserId;\n    this._inviterId = senderUserId;\n    this._endReason = isCaller ? RCCallEndReason.REMOTE_NO_RESPONSE : RCCallEndReason.REMOTE_NO_RESPONSE;\n    this._canGenRecord();\n  }\n\n  /**\n   * 到 invite | memberModify 结束\n   * 原因根据己方是否为主叫，主叫为远端未接听 被叫被己端未接听\n   */\n  private _doMemberModify(message: IReceivedMessage) {\n    this._doInvite(message);\n  }\n\n  /**\n   * 用 invite | memberModify 计算的离线记录\n   */\n  private _doRinging(message: IReceivedMessage) {\n    this._canGenRecord();\n  }\n\n  /**\n   * 到 accept 说明通话已建立\n   * 原因默认己方正常挂断\n   */\n  private _doAccept(message: IReceivedMessage) {\n    this._endReason = RCCallEndReason.HANGUP;\n    this._beginTimestamp = message.sentTime;\n    this._canGenRecord();\n  }\n\n  /**\n   * 到 hungup 说明为正常挂断\n   * 原因取消息体里挂断原因\n   */\n  private _doHungup(message: IReceivedMessage) {\n    const { content, sentTime, senderUserId } = message;\n    const { reason } = content as IHungupMsgContent;\n    const currentUserId = this._context.getCurrentId();\n    const isSelfSend = senderUserId === currentUserId;\n    this._endReason = isSelfSend ? reason : CallRemoteEndReason[reason];\n    this._endTimestamp = sentTime;\n\n    this._canGenRecord();\n  }\n\n  /**\n   * 只修改通话类型\n   */\n  private _doMediaModify(message: IReceivedMessage) {\n    const { content } = message;\n    const { mediaType } = content as IMediaModifyMsgContent;\n    this._mediaType = mediaType;\n    this._canGenRecord();\n  }\n\n  private _canGenRecord() {\n    if (this._messages.length === 0) {\n      let duration = 0;\n      const isNeedDurationReason = [\n        RCCallEndReason.HANGUP,\n        RCCallEndReason.REMOTE_HANGUP,\n        RCCallEndReason.OTHER_CLIENT_JOINED_CALL,\n        RCCallEndReason.REMOTE_OTHER_CLIENT_JOINED_CALL,\n        RCCallEndReason.KICKED_BY_SERVER,\n        RCCallEndReason.REMOTE_KICKED_BY_SERVER,\n        RCCallEndReason.ACCEPT_SYSTEM_CALL,\n        RCCallEndReason.REMOTE_ACCEPT_SYSTEM_CALL,\n      ].includes(this._endReason);\n      if (isNeedDurationReason) {\n        duration = this._endTimestamp - this._beginTimestamp;\n      }\n      this._onRecord({\n        channelId: this._channelId,\n        conversationType: this._conversationType,\n        targetId: this._targetId,\n        callId: this._callId,\n        inviterId: this._inviterId,\n        mediaType: this._mediaType,\n        endReason: this._endReason,\n        beginTimestamp: this._beginTimestamp,\n        endTimestamp: this._endTimestamp,\n        duration,\n      });\n    }\n  }\n\n  onRecvOfflineMsgs(messages: IReceivedMessage[]) {\n    this._messages = messages;\n    do {\n      const msg = this._messages.shift()!;\n      const { messageType, content: { callId } } = msg;\n      switch (messageType) {\n        case RCCallMessageType.VCInvite:\n          this._doInvite(msg);\n          break;\n        case RCCallMessageType.VCRinging:\n          this._doRinging(msg);\n          break;\n        case RCCallMessageType.VCAccept:\n          this._doAccept(msg);\n          break;\n        case RCCallMessageType.VCModifyMem:\n          this._doMemberModify(msg);\n          break;\n        case RCCallMessageType.VCModifyMedia:\n          this._doMediaModify(msg);\n          break;\n        case RCCallMessageType.VCHangup:\n          this._doHungup(msg);\n          break;\n        default:\n          this._logger.debug('_', `[OfflineRecorder] onRecvOfflineMsgs -> unexpected message: ${JSON.stringify(msg)}`);\n          break;\n      }\n    } while (this._messages.length > 0);\n  }\n}\n","import {\n  ConversationType, ErrorCode, EventEmitter, ILogger, IPushConfig, IReceivedMessage, IRuntime, ISendMsgOptions, RTCPluginContext,\n} from '@rongcloud/engine';\nimport { MemberModifyType } from './enums/MemberModifyType';\nimport { RCCallErrorCode } from './enums/RCCallCode';\nimport { RCCallMessageType } from './enums/RCCallMessageType';\nimport { IUserProfile } from './interfaces/IStateMachine';\nimport {\n  IAcceptMsgContent, IAcceptMsgOptions, ICallMsgOption, IHungupMsgContent, IHungupMsgOptions,\n  IInviteMsgContent, IInviteMsgOptions, IMediaModifyMsgContent, IMediaModifyMsgOptions, IMemberModifyMsgContent,\n  IMemberModifyMsgOptions, IMsgListener, IRingingMsgContent, IRingingMsgOptions,\n} from './interfaces/IMessageHandler';\nimport { Platform } from './enums/Platform';\nimport { getCallDeviceId } from '../helper';\nimport { IOfflineRecord, OfflineRecorder } from './OfflineRecorder';\nimport { RCCallStateMachine } from '..';\n\nconst callMsgTypes = ['RC:VCAccept', 'RC:VCRinging', 'RC:VCSummary', 'RC:VCHangup', 'RC:VCInvite', 'RC:VCModifyMedia', 'RC:VCModifyMem'];\n\ntype MsgListenerKeys = keyof IMsgListener\n\ninterface IMsgBufferItem {\n  markTime: number,\n  msg: IReceivedMessage\n}\n\nconst EngineErrorCodeToCallErrorCode: { [key: number]: RCCallErrorCode } = {\n  [ErrorCode.REJECTED_BY_BLACKLIST]: RCCallErrorCode.REJECTED_BY_BLACKLIST,\n  [ErrorCode.NOT_IN_GROUP]: RCCallErrorCode.NOT_IN_GROUP,\n};\n\n/**\n * 消息接收处理: 在线消息、离线消息\n * 发送消息处理: 发送不同类型消息封装\n */\nexport class CallMessageHandler extends EventEmitter {\n  private _watchers: IMsgListener = {}\n\n  private _userInfo: IUserProfile = {}\n\n  private _msgBufferList: IMsgBufferItem[] = []\n\n  private _hadHandleMsgTimer: boolean = false\n\n  private _offlineRecorder!: OfflineRecorder\n\n  private _deviceId: string = ''\n\n  constructor(\n    private readonly _context: RTCPluginContext,\n    private readonly _runtime: IRuntime,\n    private readonly _logger: ILogger,\n    /**\n     * 离线消息处理时间间隔\n     */\n    private readonly _offlineMsgItv: number = 60 * 1000,\n    private readonly _getStateMachine: (callId: string) => RCCallStateMachine | null,\n  ) {\n    super();\n    // deviceId\n    this._deviceId = getCallDeviceId(_runtime);\n    // 处理消息收发\n    this._context.onmessage = this._onMessage.bind(this);\n    // 处理离线消息记录\n    this._offlineRecorder = new OfflineRecorder(this._context, this._logger, (record: IOfflineRecord) => {\n      this._logger.info('_', `[CallMessageHandler] offlineRecorder -> ${JSON.stringify(record)}`);\n      this._watchers.onOfflineRecord && this._watchers.onOfflineRecord(record);\n    });\n  }\n\n  private _onMessage(message: IReceivedMessage) {\n    const isCallMsg = callMsgTypes.includes(message.messageType);\n    if (isCallMsg) {\n      this._logger.debug('_', `[CallMessageHandler] _onMessage -> call message: ${JSON.stringify(message)}`);\n      // 通过遍历 bufferList 查找要插入的节点\n      try {\n        const markTime = Date.now();\n        const { sentTime: currentMsgSentTime } = message;\n        let insertIndex = 0;\n        // 需遍历数组全部项，故使用 forEach, findIndex 找到第一个符合条件的就会终止\n        this._msgBufferList.forEach(({ msg: { sentTime } }, index) => {\n          if (currentMsgSentTime >= sentTime) {\n            // insertIndex 需为当前符合条件元素的后一位，故 +1\n            insertIndex = index + 1;\n          }\n        });\n        this._msgBufferList.splice(insertIndex, 0, {\n          markTime,\n          msg: message,\n        });\n        this._logger.warn('_', `onMessage -> msgBufferList: ${this._msgBufferList.length}`);\n      } catch (error) {\n        this._logger.error('_', `[CallMessageHandler] splice buffer msg error -> ${(error as Error).message}`);\n      }\n      // 启动消息处理\n      this._handleBufferMsgs();\n      return true;\n    }\n    return false;\n  }\n\n  /**\n   * 在线消息抛给状态机处理\n   */\n  private _onRecvOnlineCallMsg(message: IReceivedMessage) {\n    this._logger.info('_', `onMessage -> _onRecvOnlineCallMsg: ${message.messageType}`);\n    const { content: { callId } } = message;\n    // 在线消息直接抛出\n    switch (message.messageType) {\n      case RCCallMessageType.VCInvite:\n        this._watchers.onInvite && this._watchers.onInvite(message);\n        break;\n      case RCCallMessageType.VCRinging:\n        super.emit(`${callId}onRinging`, message);\n        break;\n      case RCCallMessageType.VCAccept:\n        super.emit(`${callId}onAccept`, message);\n        break;\n      case RCCallMessageType.VCModifyMem:\n        // 收到人员变更抛出 onInvite 生成状态机实例\n        this._watchers.onInvite && this._watchers.onInvite(message);\n        break;\n      case RCCallMessageType.VCModifyMedia:\n        super.emit(`${callId}onMediaModify`, message);\n        break;\n      case RCCallMessageType.VCHangup:\n        super.emit(`${callId}onHungup`, message);\n        break;\n      default:\n        this._logger.warn('_', `[CallMessageHandler] onRecvOnlineCallMsg -> unexpected message: ${JSON.stringify(message)}`);\n        break;\n    }\n  }\n\n  /**\n   * 消息 buffer 列表处理逻辑\n   * 1、每 20ms 检查一次 buffer list\n   * 2、取出已经延迟 200 的消息列表进行消费 | 无延迟 200ms 内消息直接递归\n   * 3、消费分为 离线消息消费逻辑、在线消息消费逻辑，消费后递归\n  */\n  private _handleBufferMsgs() {\n    // 消息 buffer 列表长度为 0 时加锁\n    if (this._msgBufferList.length === 0 || this._hadHandleMsgTimer) {\n      this._logger.warn('_', '_handleBufferMsgs return');\n      return;\n    }\n    // 需要加锁，收到多次消息后，可能会起多个 timer\n    this._hadHandleMsgTimer = true;\n    setTimeout(() => {\n      // 取出大于 200 ms 消息列表\n      const currentTime = Date.now();\n      const buffers = this._msgBufferList.filter((item) => currentTime - item.markTime >= 200);\n      this._logger.debug('_', `[CallMessageHandler] handleBufferMsgs -> lists over 200ms : ${JSON.stringify(buffers.map(({ msg: { messageUId, isOffLineMessage, content: { callId } } }) => ({ messageUId, isOffLineMessage, callId })))}`);\n      if (buffers.length === 0) {\n        // 没有延迟 200ms 的消息，消费逻辑解锁并递归执行\n        this._hadHandleMsgTimer = false;\n        this._handleBufferMsgs();\n        return;\n      }\n\n      if (buffers[0].msg.isOffLineMessage) {\n        // 当第一条消息为离线消息时，直接从 buffer 中取出所有离线消息，进行消息\n        let offlineBuffers = this._msgBufferList.filter((item) => item.msg.isOffLineMessage);\n\n        // 离线消息处理逻辑\n        do {\n          const {\n            conversationType, messageType, sentTime, senderUserId, content: { callId: inviteCallId, inviteUserIds },\n          } = offlineBuffers[0].msg;\n\n          const isInviteMsgType = [RCCallMessageType.VCInvite, RCCallMessageType.VCModifyMem].includes(messageType as RCCallMessageType);\n\n          // 计算当前时间与消息发送时间差\n          const delayTime = this._context.getServerTime() - sentTime;\n          const isLessThanOfflineMsgItv = delayTime < this._offlineMsgItv;\n\n          if (!isLessThanOfflineMsgItv) {\n            this._logger.warn('_', `offline msg delayTime: ${delayTime}ms`);\n          }\n\n          if (isInviteMsgType) { // 取出的第一条消息为 invite | memModify\n            // 取出相同 CallId 消息列表\n            const taskMsgList: IReceivedMessage[] = [];\n            for (let i = 0; i < offlineBuffers.length; i++) {\n              const item = offlineBuffers[i].msg;\n              const { content: { callId: otherCallId } } = item;\n              if (inviteCallId === otherCallId) {\n                taskMsgList.push(item);\n              } else {\n                break;\n              }\n            }\n            this._logger.warn('_', `taskMsgList length: ${taskMsgList.length}`);\n\n            if (taskMsgList.length > 0) { // 防止 taskMsgList 为 0\n              // 找出 msgBufferList 中已消费的消息最大 index\n              const delIndex = this._msgBufferList.findIndex((item) => item.msg.messageUId === taskMsgList[taskMsgList.length - 1].messageUId);\n              // 删除消费过 msgBufferList 的消息\n              this._msgBufferList = this._msgBufferList.slice(delIndex + 1);\n\n              // 找出 offlineMsgs 中已消费的消息最大 index\n              const delOfflineIndex = offlineBuffers.findIndex((item) => item.msg.messageUId === taskMsgList[taskMsgList.length - 1].messageUId);\n              // 删除消费过 offlineBuffers 的消息\n              offlineBuffers = offlineBuffers.slice(delOfflineIndex + 1);\n            }\n            /**\n             * 单聊未结束通话判断\n             * 如果消息在 60s 内，判断是否未成对（只有 invite 或只有 invite 或 ringing 成对抛给 离线记录器，未成对抛给状态机\n             */\n            const isUnfinishedPrivateCall: boolean = (() => {\n              if (conversationType !== ConversationType.PRIVATE) return false;\n              const isOnlyInvite = taskMsgList.length === 1;\n              const hasInviteAndRinging = taskMsgList.every((item) => [RCCallMessageType.VCInvite, RCCallMessageType.VCModifyMedia, RCCallMessageType.VCRinging].includes(item.messageType as RCCallMessageType));\n              return isOnlyInvite || hasInviteAndRinging;\n            })();\n            /**\n             * 群聊未结束通话判断\n             * 通过 list 中 invite 和 memberModify 消息中取出所有参与通话的 userId\n             * 遍历 list 找出所有 hangup 的 sendUserId\n             *     总人数 - 挂断人数 > 1  且剩余人数包含自己，认为通话未挂断进状态机\n             *     总人数 - 挂断人数 <= 1 通话挂断，进离线消息处理器\n             *     主叫直接挂断且其他人未接听直接进离线处理器\n             */\n            const isUnfinishedGrpCall: boolean = (() => {\n              if (conversationType !== ConversationType.GROUP) return false;\n              let isUnfinished = false;\n              let noOneAccept = true;\n              let allUserIds = [senderUserId, ...inviteUserIds];\n              let isCallerHungup = false;\n              for (let i = 0; i < taskMsgList.length; i++) {\n                const { senderUserId: taskMsgSenderUserId, messageType } = taskMsgList[i];\n                // 只要自己挂断直接跳出循环，进离线消息\n                if (messageType === RCCallMessageType.VCHangup && taskMsgSenderUserId === this._context.getCurrentId()) {\n                  break;\n                }\n                // 找出所有挂断的人员，以及通话发起者是否挂断\n                if (messageType === RCCallMessageType.VCHangup) {\n                  isCallerHungup = senderUserId === taskMsgSenderUserId;\n                  allUserIds = allUserIds.filter((id) => taskMsgSenderUserId !== id);\n                }\n                // 判断是否无人接听\n                if (messageType === RCCallMessageType.VCAccept) {\n                  noOneAccept = false;\n                }\n              }\n              // 不是主叫挂断且无人接听，且剩余人数大于 1\n              if (!(noOneAccept && isCallerHungup) && allUserIds.length > 1) {\n                isUnfinished = true;\n              }\n              return isUnfinished;\n            })();\n\n            if (isLessThanOfflineMsgItv && (isUnfinishedPrivateCall || isUnfinishedGrpCall)) {\n              taskMsgList.forEach(this._onRecvOnlineCallMsg, this);\n            }\n            this._offlineRecorder.onRecvOfflineMsgs(taskMsgList);\n          } else {\n            if (isLessThanOfflineMsgItv && this._getStateMachine(inviteCallId)) {\n              this._onRecvOnlineCallMsg(offlineBuffers[0].msg);\n            } else {\n              this._logger.debug('_', `[CallMessageHandler] unexcepted offline msg -> ${JSON.stringify(offlineBuffers[0].msg)}`);\n            }\n            offlineBuffers.shift();\n            this._msgBufferList.shift();\n          }\n        } while (offlineBuffers.length > 0);\n      } else {\n        // 在线消息处理逻辑\n        buffers.forEach(({ msg }) => {\n          this._onRecvOnlineCallMsg(msg!);\n        });\n        // 找出 msgBufferList 中已消费的消息最大 index\n        const delCount = buffers.length;\n        // 删除消费过的消息\n        this._msgBufferList.splice(0, delCount);\n        this._logger.debug('_', `timer online msg handle -> delCount: ${delCount} msgBufferList: ${this._msgBufferList.length}`);\n      }\n      this._hadHandleMsgTimer = false;\n      this._handleBufferMsgs();\n    }, 20);\n  }\n\n  registerEventListener(listener: IMsgListener) {\n    Object.assign(this._watchers, listener);\n  }\n\n  registerStateMachineEvent(callId: string, funcName: MsgListenerKeys, event: (...args: any[]) => void) {\n    const eventType = callId + funcName;\n    super.on(eventType, event);\n  }\n\n  unregisterStateMachineEvent(callId: string) {\n    ['onRinging', 'onAccept', 'onHungup', 'onMediaModify'].forEach((funcName) => {\n      const eventType = callId + funcName;\n      super.removeAll(eventType);\n    });\n  }\n\n  registerUserInfo(userInfo: IUserProfile) {\n    this._userInfo = userInfo;\n  }\n\n  /**\n   * 发送 IM 消息\n   */\n  private async _sendCallMessage(options: ICallMsgOption): Promise<{ code: RCCallErrorCode, message?: IReceivedMessage }> {\n    this._logger.debug('_', `CallMessageHandler] sendCallMesage -> message: ${JSON.stringify(options)}`);\n    const {\n      channelId, conversationType, targetId, content, messageType, directionalUserIdList, pushTitle, pushData, pushContent,\n    } = options;\n    const sendOpts: ISendMsgOptions = {\n      channelId,\n      messageType,\n      content,\n      directionalUserIdList,\n    };\n\n    if ([RCCallMessageType.VCInvite, RCCallMessageType.VCModifyMem, RCCallMessageType.VCHangup].includes(messageType)) {\n      const pushConifg: IPushConfig = {\n        pushTitle: pushTitle || '',\n        pushContent: pushContent || '',\n        pushData: pushData || '',\n        androidConfig: {\n          categoryHW: 'VOIP',\n          categoryVivo: 'IM',\n        },\n        iOSConfig: {\n          apnsCollapseId: content.callId,\n        },\n        disablePushTitle: false,\n        forceShowDetailContent: false,\n        templateId: '',\n      };\n      sendOpts.pushConfig = pushConifg;\n    }\n    const { code, data: message } = await this._context.sendMessage(conversationType, targetId, sendOpts);\n    if (code !== ErrorCode.SUCCESS) {\n      this._logger.error('_', `CallMessageHandler] sendCallMesage error -> code: ${code}`);\n      return {\n        code: EngineErrorCodeToCallErrorCode[code] || RCCallErrorCode.SEND_MSG_ERROR,\n      };\n    }\n    return { code: RCCallErrorCode.SUCCESS, message };\n  }\n\n  /**\n   * 发送邀请消息\n   */\n  async sendInvite(options: IInviteMsgOptions) {\n    const {\n      roomType, channelId, conversationType, targetId, callId, mediaType, inviteUserIds, extra, pushTitle, pushContent,\n    } = options;\n    this._logger.warn('_', 'CallMessageHandler] sendCallMesage sendInvite', JSON.stringify(options));\n    this._watchers.sendAccept && this._watchers.sendAccept({ callId });\n    const content: IInviteMsgContent = {\n      platform: Platform.WEB,\n      deviceId: this._deviceId,\n      callId,\n      roomType,\n      extra,\n      engineType: 4,\n      channelInfo: { Id: callId, Key: '' },\n      mediaType,\n      inviteUserIds,\n      observerUserIds: [],\n      user: this._userInfo,\n    };\n    return this._sendCallMessage({\n      channelId,\n      conversationType,\n      targetId,\n      content,\n      messageType: RCCallMessageType.VCInvite,\n      directionalUserIdList: conversationType === ConversationType.GROUP ? inviteUserIds : [targetId],\n      pushTitle,\n      pushContent,\n      pushData: JSON.stringify({\n        mediaType,\n        userIdList: inviteUserIds,\n        callId,\n      }),\n    });\n  }\n\n  /**\n   * 发送人员变更消息\n   */\n  async sendMemeberModify(options: IMemberModifyMsgOptions) {\n    const {\n      channelId, conversationType, targetId, callId, mediaType, inviteUserIds, callerId,\n      existedUserPofiles, directionalUserIdList, extra, pushTitle, pushContent,\n    } = options;\n    const content: IMemberModifyMsgContent = {\n      platform: Platform.WEB, // TODO 与 IM 一致\n      deviceId: this._deviceId,\n      callId,\n      extra,\n      engineType: 4,\n      channelInfo: { Id: callId, Key: '' },\n      mediaType,\n      inviteUserIds,\n      observerUserIds: [],\n      user: this._userInfo,\n      caller: callerId,\n      modifyMemType: MemberModifyType.ADD,\n      existedUserPofiles,\n    };\n    return this._sendCallMessage({\n      channelId,\n      conversationType,\n      targetId,\n      content,\n      messageType: RCCallMessageType.VCModifyMem,\n      directionalUserIdList,\n      pushTitle,\n      pushContent,\n      pushData: JSON.stringify({\n        mediaType,\n        userIdList: inviteUserIds,\n        callId,\n      }),\n    });\n  }\n\n  /**\n   * 发送响铃消息\n   */\n  sendRinging(options: IRingingMsgOptions) {\n    const {\n      channelId, conversationType, targetId, callId, userIds,\n    } = options;\n    const content: IRingingMsgContent = {\n      platform: Platform.WEB,\n      deviceId: this._deviceId,\n      callId,\n      user: this._userInfo,\n    };\n    return this._sendCallMessage({\n      channelId,\n      conversationType,\n      targetId,\n      content,\n      messageType: RCCallMessageType.VCRinging,\n      directionalUserIdList: userIds,\n    });\n  }\n\n  /**\n   * 发送同意接听消息\n   */\n  sendAccept(options: IAcceptMsgOptions) {\n    const {\n      channelId, conversationType, targetId, callId, mediaType, userIds,\n    } = options;\n    const content: IAcceptMsgContent = {\n      platform: Platform.WEB,\n      deviceId: this._deviceId,\n      callId,\n      mediaType,\n      user: this._userInfo,\n    };\n    return this._sendCallMessage({\n      channelId,\n      conversationType,\n      targetId,\n      content,\n      messageType: RCCallMessageType.VCAccept,\n      directionalUserIdList: userIds,\n    });\n  }\n\n  /**\n   * 发送挂断消息\n   */\n  sendHungup(options: IHungupMsgOptions) {\n    const {\n      channelId, conversationType, targetId, callId, reason, userIds, pushTitle, pushContent,\n    } = options;\n    const content: IHungupMsgContent = {\n      platform: Platform.WEB,\n      deviceId: this._deviceId,\n      callId,\n      reason,\n      user: this._userInfo,\n    };\n    return this._sendCallMessage({\n      channelId,\n      conversationType,\n      targetId,\n      content,\n      messageType: RCCallMessageType.VCHangup,\n      pushTitle,\n      pushContent,\n      pushData: JSON.stringify({\n        callId, reason,\n      }),\n      directionalUserIdList: userIds,\n    });\n  }\n\n  /**\n   * 发送媒体变更消息\n   */\n  sendMediaModify(options: IMediaModifyMsgOptions) {\n    const {\n      channelId, conversationType, targetId, callId, mediaType, userIds,\n    } = options;\n    const content: IMediaModifyMsgContent = {\n      platform: Platform.WEB,\n      deviceId: this._deviceId,\n      callId,\n      mediaType,\n      user: this._userInfo,\n    };\n    return this._sendCallMessage({\n      channelId,\n      conversationType,\n      targetId,\n      content,\n      messageType: RCCallMessageType.VCModifyMedia,\n      directionalUserIdList: userIds,\n    });\n  }\n}\n","export enum RCCallLanguage {\n  ZH = 'zh',\n  EN = 'en'\n}\n","export const EN = {\n  PushTitle: {\n    AUDIO: 'You have a voice call',\n    VIDEO: 'You have a video call',\n  },\n};\n","export const ZH = {\n  PushTitle: {\n    AUDIO: '您有一条音频通话',\n    VIDEO: '您有一条视频通话',\n  },\n};\n","import { RCCallLanguage } from '../enums/RCCallLang';\nimport { EN } from './en';\nimport { ZH } from './zh';\n\n/**\n * CallEngine 全局语言设置，当前仅支持中、英文\n */\nexport class Local {\n  static _lang: RCCallLanguage = RCCallLanguage.ZH\n\n  static set(lang: RCCallLanguage) {\n    this._lang = lang;\n  }\n\n  static get() {\n    if (this._lang === RCCallLanguage.EN) {\n      return EN;\n    }\n    return ZH;\n  }\n}\n","import {\n  BasicLogger, ConversationType, IReceivedMessage, IRuntime, RTCPluginContext,\n} from '@rongcloud/engine';\nimport { RCCallStateMachine } from './core/StateMachine';\nimport {\n  ICallStateMachineWatchers, IEndSummary, IInvitedUsers, IMediaModifyInfo, IMemberModifyInfo, ISenderInfo, IStateChangeInfo, IUserProfile, IUserStateChangeInfo, IUserData, IInviteOptions,\n} from './core/interfaces/IStateMachine';\nimport { RCCallErrorCode } from './core/enums/RCCallCode';\nimport { RCCallUserState } from './core/enums/RCCallUserState';\nimport { RCCallSessionState } from './core/enums/RCCallSessionState';\nimport { RCCallEndReason } from './core/enums/RCCallEndReason';\nimport { RCCallMediaType } from './core/enums/RCCallMediaType';\nimport { CallMessageHandler } from './core/MessageHandler';\nimport { eventEmitter, generateRandomId, getCallDeviceId } from './helper';\nimport { RCCallLanguage } from './core/enums/RCCallLang';\nimport { ICallEngineOptions, ICallEngineWatchers } from './core/interfaces/ICallEngine';\nimport { RCCallMessageType } from './core/enums/RCCallMessageType';\nimport { IInviteMsgContent } from './core/interfaces/IMessageHandler';\nimport { Local } from './core/locale';\nimport { IOfflineRecord } from './core/OfflineRecorder';\nimport { RCCrossCallType } from './core/enums/RCCrossCallType';\n\nclass RCCallEngine {\n  private _stateMachine: { [callId: string]: RCCallStateMachine } = {}\n\n  private _callMsgHandler: CallMessageHandler\n\n  /**\n   * 初始化\n   */\n  constructor(\n    /**\n     * engine PlguinContext 实例\n     */\n    private readonly _context: RTCPluginContext,\n    /**\n     * 运行时相关\n     */\n    private readonly _runtime: IRuntime,\n    /**\n     * engine 日志模块实例，由 CallLib 层初始化\n     */\n    private readonly _logger: BasicLogger,\n    /**\n     * 监听方法\n     */\n    private readonly _watchers: ICallEngineWatchers,\n    /**\n     * 其他配置项\n     */\n    private readonly _options: ICallEngineOptions,\n  ) {\n    this._logger.warn('_', `RCCallEngine Version: ${__VERSION__} CommitId: ${__COMMIT_ID__}`);\n    // 设置 DeviceId\n    getCallDeviceId(_runtime);\n    // 监听 IM 消息\n    this._callMsgHandler = new CallMessageHandler(this._context, this._runtime, this._logger, this._options.offlineMsgItv, this._getStateMachine.bind(this));\n    // 注册消息模块监听\n    this._callMsgHandler.registerEventListener({\n      onInvite: this._onInvite.bind(this),\n      sendAccept: this._handleSendAccept.bind(this),\n      onOfflineRecord: this._watchers.onOfflineRecord,\n    });\n    // 监听状态机关闭\n    eventEmitter.on('onStateMachineClose', (callId: string) => {\n      delete this._stateMachine[callId];\n    });\n    // 设置语言\n    Local.set(_options.lang);\n  }\n\n  private _onInvite(msg: IReceivedMessage) {\n    const {\n      channelId, conversationType, targetId, content, messageType, senderUserId: suid, pushConfig,\n    } = msg;\n    this._logger.warn('_', `RCCallEngine _onInvite:targetId ${targetId} senderUserId: ${suid}`);\n    const {\n      mediaType, callId, extra, roomType,\n    } = content as IInviteMsgContent;\n    let senderUserId: string;\n    if (roomType !== RCCrossCallType.RCCallRoomTypeAcrossCall) {\n      senderUserId = suid;\n    } else {\n      [, senderUserId] = suid.split('_');\n    }\n    const crtUserId = this._context.getCurrentId();\n    if (this._context.getCurrentId() === senderUserId) { // 多端处理\n      return;\n    }\n\n    /**\n     * 群聊通话有其他端处理时，不再处理此通话中的邀请消息\n     * 条件1: 此人已被邀请通话\n     * 条件2: 本端无此通话的状态机\n     */\n    if (messageType === RCCallMessageType.VCModifyMem) {\n      const isKeeping = content.existedUserPofiles.some((item: { userId: string }) => (item.userId === crtUserId));\n      if (isKeeping && !this._stateMachine[callId]) {\n        return;\n      }\n    }\n\n    const stateMachine = this._stateMachine[callId];\n    if (!stateMachine) {\n      this._stateMachine[callId] = new RCCallStateMachine(\n        this._context,\n        this._runtime,\n        this._logger,\n        this._callMsgHandler,\n        channelId!,\n        conversationType,\n        targetId,\n        mediaType,\n        callId,\n      );\n      this._logger.info('_', `[RCCallEngine] RCCallStateMachine successfully created -> callId: ${callId}`);\n\n      if (messageType === RCCallMessageType.VCInvite) {\n        // 状态机内部处理 invite 消息\n        this._stateMachine[callId].__onInvite(msg);\n      } else if (messageType === RCCallMessageType.VCModifyMem) {\n        this._stateMachine[callId].__onMemberModify(msg);\n      }\n      this._watchers.onInvite(this._stateMachine[callId], extra);\n\n      const hasOtherStateMachine = Object.keys(this._stateMachine).filter((otherCallId) => callId !== otherCallId).length > 0;\n      if (hasOtherStateMachine && !(this._options.isAllowAcceptNewCall || false)) {\n        this._stateMachine[callId].__handleInviteInSession();\n      }\n    } else if (messageType === RCCallMessageType.VCModifyMem) {\n      this._stateMachine[callId].__onMemberModify(msg);\n    }\n  }\n\n  /**\n   * 允许接听新的通话时，接听完新的通话后，挂断其他通话\n   */\n  private _handleSendAccept(info: { callId: string }) {\n    if (this._options.isAllowAcceptNewCall) {\n      const { callId } = info;\n      for (const id in this._stateMachine) {\n        if (callId !== id) {\n          this._stateMachine[id].hungup();\n          delete this._stateMachine[id];\n        }\n      }\n    }\n  }\n\n  /**\n   * 根据 callId 获取状态机\n  */\n  private _getStateMachine(callId: string): RCCallStateMachine | null {\n    return this._stateMachine[callId];\n  }\n\n  /**\n   * 注册用户信息, 发送 call 消息时用户信息携带\n   */\n  registerUserInfo(userInfo: IUserProfile) {\n    this._logger.debug('_', `[RCCallEngine] registerUserInfo -> userInfo: ${JSON.stringify(userInfo)}`);\n\n    this._callMsgHandler.registerUserInfo(userInfo);\n  }\n\n  /**\n   * 单呼\n   * @param channelId 组织 ID\n   * @param targetId  对方 ID\n   * @param mediaType 媒体类型\n   * @param extra 消息扩展信息\n   * @param pushTitle 通知的标题\n   * @param pushContent 通知的内容\n   */\n  async call(\n    channelId: string,\n    targetId: string,\n    mediaType: number,\n    extra: string = '',\n    pushTitle: string = '',\n    pushContent: string = '',\n    isCrossAppkey = false,\n  ): Promise<{ code: RCCallErrorCode, stateMachine?: RCCallStateMachine }> {\n    this._logger.debug('_', `[RCCallEngine] call -> args: ${JSON.stringify({\n      channelId, targetId, mediaType, extra, pushTitle, pushContent,\n    })}`);\n    const callId = generateRandomId();\n    const hasStateMachine = Object.keys(this._stateMachine).length > 0;\n    if (hasStateMachine) {\n      return { code: RCCallErrorCode.STATE_MACHINE_EXIT };\n    }\n    this._stateMachine[callId] = new RCCallStateMachine(\n      this._context,\n      this._runtime,\n      this._logger,\n      this._callMsgHandler,\n      channelId,\n      ConversationType.PRIVATE,\n      targetId,\n      mediaType,\n      callId,\n    );\n    const { code } = await this._stateMachine[callId].__call([targetId], extra, pushTitle, pushContent, isCrossAppkey);\n    if (code === RCCallErrorCode.SUCCESS) {\n      return {\n        code: RCCallErrorCode.SUCCESS,\n        stateMachine: this._stateMachine[callId],\n      };\n    }\n    return { code };\n  }\n\n  /**\n   * 群呼\n   * @param channelId 组织 ID\n   * @param targetId  群组 ID\n   * @param mediaType 媒体类型\n   * @param userIds 被邀请人员列表\n   * @param extra 消息扩展信息\n   * @param pushTitle 通知的标题\n   * @param pushContent 通知的内容\n   */\n  async callInGroup(\n    channelId: string,\n    targetId: string,\n    mediaType: number,\n    userIds: string[],\n    extra: string = '',\n    pushTitle: string = '',\n    pushContent: string = '',\n  ): Promise<{ code: RCCallErrorCode, stateMachine?: RCCallStateMachine }> {\n    this._logger.debug('_', `[RCCallEngine] callInGroup -> args: ${JSON.stringify({ channelId, targetId, mediaType })}`);\n    const callId = generateRandomId();\n    const hasStateMachine = Object.keys(this._stateMachine).length > 0;\n    if (hasStateMachine) {\n      return { code: RCCallErrorCode.STATE_MACHINE_EXIT };\n    }\n    this._stateMachine[callId] = new RCCallStateMachine(\n      this._context,\n      this._runtime,\n      this._logger,\n      this._callMsgHandler,\n      channelId,\n      ConversationType.GROUP,\n      targetId,\n      mediaType,\n      callId,\n    );\n    const { code } = await this._stateMachine[callId].__call(userIds, extra, pushTitle, pushContent);\n    if (code === RCCallErrorCode.SUCCESS) {\n      return {\n        code: RCCallErrorCode.SUCCESS,\n        stateMachine: this._stateMachine[callId],\n      };\n    }\n    return { code };\n  }\n\n  /**\n   * 销毁当前的状态机\n   */\n  destroy() {\n    this._logger.debug('_', '[RCCallEngine] destroy');\n    this._stateMachine = {};\n  }\n}\n\nexport {\n  RCCallEngine,\n  RCCallStateMachine, RCCallErrorCode,\n  RCCallUserState,\n  RCCallSessionState,\n  RCCallEndReason,\n  RCCallMediaType,\n  RCCallLanguage,\n};\n\nexport type {\n  ICallStateMachineWatchers, IEndSummary,\n  ICallEngineWatchers,\n  IUserStateChangeInfo,\n  IStateChangeInfo,\n  ISenderInfo,\n  IMemberModifyInfo,\n  IMediaModifyInfo,\n  IInvitedUsers,\n  IUserData,\n  IUserProfile,\n  IOfflineRecord,\n  IInviteOptions,\n};\n","/**\n * 产生session的场景\n */\nexport enum ProduceTypes {\n  /**\n   * 主叫\n   */\n  CALLER = 1,\n\n  /**\n   * 被叫\n   */\n  CALLEE = 2\n}\n","class EventEmitter {\n  private list: {[propName: string]: any[]} = {}\n\n  public on(event: string, fun:(data?:any)=>void) {\n    (this.list[event] || (this.list[event] = [])).push(fun);\n    return this;\n  }\n\n  public once(event: string, fun:(data?:any)=>void) {\n    const on = (data?:any) => {\n      this.off(event, on);\n      fun.call(this, data);\n    };\n    on.fun = fun;\n    this.on(event, on);\n  }\n\n  public off(event: string, fun?:(data?:any)=>void) {\n    const funs = this.list[event];\n    if (!funs) {\n      return false;\n    }\n    if (!fun) {\n      // 如果没有传 fn 的话，就会将 event 值对应缓存列表中的 fun 都清空\n      funs && (funs.length = 0);\n    } else {\n      // 若有 fun，遍历缓存列表，看看传入的 fn 与哪个函数相同，如果相同就直接从缓存列表中删掉即可\n      let cb;\n      for (let i = 0, { length } = funs; i < length; i++) {\n        cb = funs[i];\n        if (cb === fun || cb.fun === fun) {\n          funs.splice(i, 1);\n          break;\n        }\n      }\n    }\n  }\n\n  public emit(event: string, data?:any) {\n    // 第一个参数是对应的 event 值，直接用数组的 shift 方法取出\n    const funs = [...this.list[event]];\n\n    // 如果缓存列表里没有 fun 就返回 false\n    if (!funs || funs.length === 0) {\n      return false;\n    }\n    // 遍历 event 值对应的缓存列表，依次执行 fn\n    funs.forEach((fun) => {\n      fun.call(this, data);\n    });\n  }\n}\nexport {\n  EventEmitter,\n};\n","import { EventEmitter } from './utils';\n\nexport default new EventEmitter();\n","import { RTCJoinType, LogL } from '@rongcloud/engine';\nimport { RCCallLanguage, RCCallMediaType } from '@rc-embed/call-engine';\nimport { RCResolution } from '@rongcloud/plugin-rtc';\n\nimport {\n  ISessionListener, IRCCallInitOptions, IValidationResult, IMediaStreamConstraints, IPushConfig,\n} from './interface';\n\nfunction isLanguage(val: string) {\n  const values: string[] = Object.values(RCCallLanguage);\n  return values.includes(val);\n}\n\nfunction isJoinType(val: number) {\n  const values = Object.values(RTCJoinType) as number[];\n  return values.includes(val);\n}\n\nfunction isLogLevel(val: number) {\n  return [LogL.DEBUG, LogL.INFO, LogL.WARN, LogL.ERROR].includes(val);\n}\n\nexport const validateCallInitOptions = (options: IRCCallInitOptions): IValidationResult => {\n  if (!options) {\n    return { result: false, msg: 'Initialization missing parameter -> options' };\n  }\n  if (typeof options !== 'object') {\n    return { result: false, msg: 'Initialization options must be an object' };\n  }\n  const keyNames: string[] = ['rtcClient', 'onSession', 'onSessionClose'];\n  const keys: string[] = Object.keys(options);\n  const missingKeys: string[] = [];\n\n  // 校验填项是否都包含，如果哪个不包含就收集起来\n  keyNames.forEach((key: string) => {\n    if (!keys.includes(key)) {\n      missingKeys.push(key);\n    }\n  });\n\n  // 如果缺少必填的监听函数或对象\n  if (missingKeys.length) {\n    return { result: false, msg: `Initialization missing parameter -> \"${missingKeys.join(',')}\"` };\n  }\n  if (typeof options.rtcClient !== 'object') {\n    return { result: false, msg: 'Initialization \\'rtcClient\\' parameter must be of type \\'object\\'' };\n  }\n  if (typeof options.onSession !== 'function') {\n    return { result: false, msg: 'Initialization \\'onSession\\' parameter must be of type \\'function\\'' };\n  }\n  if (typeof options.onSessionClose !== 'function') {\n    return { result: false, msg: 'Initialization \\'onSessionClose\\' parameter must be of type \\'function\\'' };\n  }\n\n  // 如果传了isAllowSubscribeRetry 但不是boolean类型\n  if (typeof options.isAllowSubscribeRetry !== 'undefined' && typeof options.isAllowSubscribeRetry !== 'boolean') {\n    return { result: false, msg: 'Initialization \\'isAllowSubscribeRetry\\' parameter must be of type \\'boolean\\'' };\n  }\n\n  // 如果传了，但不是boolean类型\n  if (typeof options.isAllowPublishRetry !== 'undefined' && typeof options.isAllowPublishRetry !== 'boolean') {\n    return { result: false, msg: 'Initialization \\'isAllowPublishRetry\\' parameter must be of type \\'boolean\\'' };\n  }\n\n  // 如果传了，但不是boolean类型\n  if (typeof options.isOffCameraWhenVideoDisable !== 'undefined' && typeof options.isOffCameraWhenVideoDisable !== 'boolean') {\n    return { result: false, msg: 'Initialization \\'isOffCameraWhenVideoDisable\\' parameter must be of type \\'boolean\\'' };\n  }\n\n  // 如果传了，但不是RTCJoinType的枚举\n  if (typeof options.joinType !== 'undefined' && !isJoinType(options.joinType!)) {\n    return { result: false, msg: 'Initialization \\'joinType\\' parameter must be of type correct type' };\n  }\n\n  // 如果传了，但不是boolean类型\n  if (typeof options.isAllowDemotionGetStream !== 'undefined' && typeof options.isAllowDemotionGetStream !== 'boolean') {\n    return { result: false, msg: 'Initialization \\'isAllowDemotionGetStream\\' parameter must be of type \\'boolean\\'' };\n  }\n\n  // 如果传了，但不是RCCallLanguage的枚举\n  if (typeof options.lang !== 'undefined' && !isLanguage(options.lang!)) {\n    return { result: false, msg: 'Initialization \\'lang\\' parameter must be of type correct type' };\n  }\n\n  if (typeof options.logOutputLevel !== 'undefined' && !isLogLevel(options.logOutputLevel!)) {\n    return { result: false, msg: 'Initialization \\'logOutputLevel\\' parameter must be of type correct type' };\n  }\n  return { result: true };\n};\n\n/**\n * 校验registerSessionListener参数\n */\nexport const validateListener = (listener: ISessionListener): IValidationResult => {\n  if (!listener) {\n    return { result: false, msg: 'missing parameter -> listener' };\n  }\n  if (typeof listener !== 'object') {\n    return { result: false, msg: 'listener must be an object' };\n  }\n  const keyNames: string[] = ['onRinging', 'onAccept', 'onHungup', 'onTrackReady'];\n  const keys: string[] = Object.keys(listener);\n  const missingKeys: string[] = [];\n  keyNames.forEach((key: string) => {\n    if (!keys.includes(key)) {\n      missingKeys.push(key);\n    }\n  });\n  if (missingKeys.length) {\n    return { result: false, msg: `missing parameter -> \"${missingKeys.join(',')}\"` };\n  }\n  if (typeof listener.onRinging !== 'function') {\n    return { result: false, msg: '\\'onRinging\\' parameter must be of type \\'function\\'' };\n  }\n  if (typeof listener.onAccept !== 'function') {\n    return { result: false, msg: '\\'onAccept\\' parameter must be of type \\'function\\'' };\n  }\n  if (typeof listener.onHungup !== 'function') {\n    return { result: false, msg: '\\'onHungup\\' parameter must be of type \\'function\\'' };\n  }\n  if (typeof listener.onTrackReady !== 'function') {\n    return { result: false, msg: '\\'onTrackReady\\' parameter must be of type \\'function\\'' };\n  }\n  return { result: true };\n};\n\nexport const validateTargetId = (targetId: string): IValidationResult => {\n  if (targetId && typeof targetId === 'string') {\n    return { result: true };\n  }\n  return { result: false, msg: '\\'targetId\\' parameter is required, must be of type \\'string\\'' };\n};\n\nexport const validateMediaType = (mediaType: number): IValidationResult => {\n  if (mediaType === RCCallMediaType.AUDIO || mediaType === RCCallMediaType.AUDIO_VIDEO) {\n    return { result: true };\n  }\n  return { result: false, msg: '\\'mediaType\\' parameter is required, must be of type \\'RCCallMediaType\\'' };\n};\nexport const validateExtra = (extra: string): IValidationResult => {\n  if (typeof extra === 'string') {\n    return { result: true };\n  }\n  return { result: false, msg: '\\'extra\\' parameter must be of type \\'string\\'' };\n};\nexport const validatePushTitle = (pushTitle: string): IValidationResult => {\n  if (typeof pushTitle === 'string') {\n    return { result: true };\n  }\n  return { result: false, msg: '\\'pushTitle\\' parameter must be of type \\'string\\'' };\n};\nexport const validatePushContent = (pushContent: string): IValidationResult => {\n  if (typeof pushContent === 'string') {\n    return { result: true };\n  }\n  return { result: false, msg: '\\'pushContent\\' parameter must be of type \\'string\\'' };\n};\n\nexport const validatePushConfig = (pushConfig: IPushConfig) => {\n  const { pushTitle = '', pushContent = '' } = pushConfig;\n  const conclusion: IValidationResult[] = [];\n  conclusion.push(validatePushTitle(pushTitle));\n  conclusion.push(validatePushContent(pushContent));\n  return conclusion;\n};\n\nexport const validateUserIds = (userIds: string[]): IValidationResult => {\n  if (!Array.isArray(userIds)) {\n    return { result: false, msg: '\\'userIds\\' parameter is required, must be of type \\'string[]\\'' };\n  }\n  if (!userIds.length) {\n    return { result: false, msg: '\\'userIds\\' parameter is required, must be of type \\'string[]\\'' };\n  }\n  if (!userIds.every((str) => typeof str === 'string' && str.length > 0)) {\n    return { result: false, msg: '\\'userIds\\' parameter is required' };\n  }\n  return { result: true };\n};\n\nfunction isRCFrameRate(val: string): boolean {\n  const arrs = ['FPS_10', 'FPS_15', 'FPS_24', 'FPS_30'];\n  return arrs.includes(val);\n}\n\nconst isValidResolution = (resolution?: RCResolution): boolean => !!RCResolution[resolution!];\n\nexport const validateMediaStreamConstraints = (constraints: IMediaStreamConstraints): IValidationResult => {\n  if (constraints && constraints.audio && typeof constraints.audio.micphoneId !== 'undefined' && typeof constraints.audio.micphoneId !== 'string') {\n    return { result: false, msg: '\\'constraints.audio.micphoneId\\' must be of type \\'string\\'' };\n  }\n  if (constraints && constraints.audio && typeof constraints.audio.sampleRate !== 'undefined' && typeof constraints.audio.sampleRate !== 'number') {\n    return { result: false, msg: '\\'constraints.audio.sampleRate\\' must be of type \\'number\\'' };\n  }\n  if (constraints && constraints.video && typeof constraints.video.cameraId !== 'undefined' && typeof constraints.video.cameraId !== 'string') {\n    return { result: false, msg: '\\'constraints.video.cameraId\\' must be of type \\'string\\'' };\n  }\n  // if (constraints && constraints.video && typeof constraints.video.faceMode !== 'undefined' && constraints.video.cameraId !== 'user' && constraints.video.faceMode !== 'environment') {\n  //   return { result: false, msg: '\\'constraints.video.cameraId\\' must be  \\'user\\' or \\'environment\\'' }\n  // }\n\n  if (constraints && constraints.video && typeof constraints.video.frameRate !== 'undefined' && typeof constraints.video.frameRate !== 'string') {\n    return { result: false, msg: '\\'constraints.video.frameRate\\' must be of type \\'string\\'' };\n  }\n\n  if (constraints && constraints.video && typeof constraints.video.frameRate !== 'undefined' && !isRCFrameRate(constraints.video.frameRate)) {\n    return { result: false, msg: '\\'frameRate\\' value is out of range' };\n  }\n\n  if (constraints && constraints.video && typeof constraints.video.resolution !== 'undefined' && typeof constraints.video.resolution !== 'string') {\n    return { result: false, msg: '\\'constraints.video.frameRate\\' must be of type \\'string\\'' };\n  }\n\n  if (constraints && constraints.video && typeof constraints.video.resolution !== 'undefined' && !isValidResolution(constraints.video.resolution)) {\n    return { result: false, msg: '\\'resolution\\' value is out of range' };\n  }\n  if (constraints && constraints.video && (!constraints.video.frameRate || !constraints.video.resolution)) {\n    return { result: false, msg: '\\'resolution\\' and \\'resolution\\' is required' };\n  }\n\n  return { result: true };\n};\n","export const timerSetTimeout = (func: Function, timeout: number): number => setTimeout(func, timeout) as number;\n","import { timerSetTimeout } from './helper';\n\nexport class Timer {\n  private _timerId: number = 0\n\n  private _startTime: number = 0\n\n  constructor(callback: Function, timeout: number) {\n    if (callback) {\n      this._timerId = timerSetTimeout(() => {\n        callback();\n      }, timeout);\n    }\n    this._startTime = Date.now();\n  }\n\n  stop(): {\n    startTime: number,\n    endTime: number,\n    duration: number\n    } {\n    clearTimeout(this._timerId);\n\n    const endTime = Date.now();\n    let duration = endTime - this._startTime;\n    if (this._startTime === 0) {\n      duration = 0;\n    }\n\n    return {\n      startTime: this._startTime,\n      endTime,\n      duration,\n    };\n  }\n\n  reset() {\n    this._startTime = 0;\n  }\n}\n","import { ConversationType, ILogger } from '@rongcloud/engine';\nimport {\n  IMediaModifyInfo,\n  IMemberModifyInfo,\n  ISenderInfo,\n  IStateChangeInfo,\n  IUserData,\n  IUserStateChangeInfo,\n  IInviteOptions,\n  RCCallEndReason,\n  RCCallErrorCode,\n  RCCallMediaType,\n  RCCallSessionState,\n  RCCallStateMachine,\n  RCCallUserState,\n} from '@rc-embed/call-engine';\nimport {\n  IRCRTCStateReport,\n  RCKickReason,\n  RCLocalTrack,\n  RCRemoteAudioTrack,\n  RCRemoteTrack,\n  RCRemoteVideoTrack,\n  RCRTCClient,\n  RCRTCCode,\n  RCRTCPingResult,\n  RCRTCRoom,\n  IMicphoneAudioProfile,\n} from '@rongcloud/plugin-rtc';\n\nimport { ProduceTypes } from './enums';\n\nimport {\n  IDeviceChangeParams,\n  IMediaStreamConstraints,\n  IMuteUser,\n  IRCCallSessionOptions,\n  ISessionListener,\n  IValidationResult,\n} from './interface';\nimport eventEmitter from './eventEmitter';\nimport {\n  validateListener, validateMediaStreamConstraints, validateUserIds, validateExtra, validatePushTitle, validatePushContent,\n} from './validation';\nimport { Timer } from './timer';\n\nexport class RCCallSession {\n  /**\n   * RTC房间实例\n   */\n  private _room!: RCRTCRoom\n\n  /**\n   * 用户传进来的 对session的监听 (要在RCCallClient的_onInvite里判断，要求执行完onSession必须注册session的监听，所以这里是public)\n   */\n  public _listener: ISessionListener | null = null\n\n  /**\n   * RTC订阅、发布重试的次数\n   */\n  private readonly _RETRYCOUNT: number = 2\n\n  /**\n   * 加入房间定时器\n   */\n\n  private joinRoomTimer: any = null\n\n  constructor(\n\n    /**\n     * 状态机实例\n     */\n    private _stateMachine: RCCallStateMachine,\n\n    /**\n     * rtc实例\n     */\n    private readonly _rtcClient: RCRTCClient,\n\n    private readonly _logger: ILogger,\n\n    /**\n     * session的其它选项\n     */\n    private _options: IRCCallSessionOptions = {},\n\n  ) {\n    // 监听状态机\n    this._stateMachine.registerEventListener({\n      /**\n       * 用户状态变更\n       * @param info\n       */\n      onUserStateChange: ({ user, reason }: IUserStateChangeInfo) => {\n        this._logger.info('_', `[RCCallSession onUserStateChange] userId->${user?.userId} state->${user?.state} reason->${reason}`);\n      },\n\n      /**\n       * 房间状态变更\n       * @param\n       */\n      onStateChange: async (info: IStateChangeInfo) => {\n        const { state, reason } = info;\n        this._logger.info('_', `[RCCallSession onStateChange] : state->${state} reason->${reason}`);\n\n        // 如果在通话中，就加房间\n        if (state === RCCallSessionState.KEEPING) {\n          const roomId: string = this._stateMachine.getCallId();\n          this._logger.info('_', `[RCCallSession onStateChange] roomId: ${roomId}`);\n          try {\n            // 加房间\n            await this._joinRoom(roomId);\n          } catch (error) {\n            this._exceptionClose(RCCallEndReason.NETWORK_ERROR);\n            this._logger.error('_', `[RCCallSession onStateChange] joinRoom throw exception roomId -> ${roomId}`);\n            console.error(error);\n          }\n\n          /**\n           *  以下三条只要满足一条，状态会变成RCCallSessionState.END\n           *  1、本端用户自己主动挂断\n           *  2、服务端把本端用户踢出RTC房间\n           *  3、房间里小于2个人\n           */\n        } else if (state === RCCallSessionState.END) {\n          // 还未加入房间就挂断\n          if (!this._room) {\n            // 销毁本地流，关闭摄像头\n            this._options.localTracks && this._destroyTracks(this._options.localTracks);\n            const summaryInfo = this._stateMachine.getSummary();\n            eventEmitter.emit('sessionClose', { session: this, summaryInfo });\n            return;\n          }\n\n          this._options.localTracks && this._destroyTracks(this._options.localTracks);\n          this._logger.info('_', '[RCCallSession onStateChange] localTracks destroyed');\n          this._leaveRoom();\n          this._room = null as unknown as RCRTCRoom;\n        }\n      },\n\n      /**\n       * 收到响铃\n       * @param sender 发起用户信息\n       */\n      onRinging: (sender: ISenderInfo) => {\n        this._logger.info('_', `[RCCallSession onRinging]sender: sender.userId -> ${sender.userId}`);\n\n        try {\n          // 通知用户响铃\n          this._listener!.onRinging(sender, this);\n        } catch (error) {\n          this._logger.error('_', '[RCCallSession onRinging] method exception -> onRinging');\n          console.error(error);\n        }\n      },\n\n      /**\n         * 当远端用户同意接听\n         */\n      onAccept: (sender: ISenderInfo) => {\n        this._logger.info('_', `[RCCallSession onAccept]sender: sender.userId -> ${sender.userId}`);\n        try {\n          // 通知本端，远端用户已接听\n          this._listener!.onAccept(sender, this);\n        } catch (error) {\n          this._logger.error('_', '[RCCallSession onAccept] method exception -> onAccept');\n          console.error(error);\n        }\n      },\n\n      /**\n         * 当有远端用户挂断\n         */\n      onHungup: (sender: ISenderInfo, reason: RCCallEndReason) => {\n        this._logger.info('_', `[RCCallSession onHungup]sender: sender.userId -> ${sender.userId} reason->${reason}`);\n        try {\n          // 通知本端，远端用户已挂断\n          this._listener!.onHungup(sender, reason, this);\n        } catch (error) {\n          this._logger.error('_', '[RCCallSession onHungup] method exception -> onHungup');\n          console.error(error);\n        }\n      },\n\n      /**\n       * 收到人员变更\n       * @param sender 发起用户信息\n       */\n      onMemberModify: ({ sender, invitedUsers }: IMemberModifyInfo) => {\n        this._logger.info('_', `[RCCallSession onMemberModify] sender.userId -> ${sender.userId}`);\n        try {\n          // 通知用户人员变更\n          this._listener!.onMemberModify(sender, invitedUsers, this);\n        } catch (error) {\n          this._logger.error('_', '[RCCallSession onMemberModify] method exception -> onMemberModify');\n          console.error(error);\n        }\n      },\n\n      /**\n       * 收到通话类型变更 (通话降级)\n       * @param sender 发起用户信息\n       */\n      onMediaModify: ({ sender, mediaType }: IMediaModifyInfo) => {\n        this._logger.info('_', `[RCCallSession onMediaModify]sender: sender.userId -> ${sender.userId} mediaType: ${mediaType}`);\n        if (mediaType === RCCallMediaType.AUDIO) {\n          // 远端收到通话降级通知后，远端执行降级通话(不发消息)\n          this._setMediaTypeToAudio();\n        }\n        try {\n          this._listener!.onMediaModify(sender, mediaType, this);\n        } catch (error) {\n          this._logger.error('_', '[RCCallSession onMediaModify] method exception -> onMediaModify');\n          console.error(error);\n        }\n      },\n      /**\n       * 是否跨appkey\n       * @param sender 发起用户信息\n       */\n      crossAppkey: (isCrossAppkey: boolean) => {\n        this._logger.info('_', `[RCCallSession crossAppkey] 是否跨 appkey: ${isCrossAppkey}`);\n        this._options.isCrossAppkey = isCrossAppkey;\n      },\n    });\n\n    /**\n     * 设置挂断的推送信息\n     */\n    const { pushTitle, pushContent } = this._options.hungupPushConfig!;\n    this._stateMachine.setHungupPushConfig(pushTitle, pushContent);\n  }\n\n  /**\n   *  加入房间\n   */\n  private async _joinRoom(roomId: string): Promise<{ code: RCCallErrorCode }> {\n    let callBack;\n    try {\n      // 加房间\n      if (this._options.isCrossAppkey) {\n        callBack = await this._rtcClient.joinCrossRTCRoom(roomId, this._options.joinType);\n      } else {\n        callBack = await this._rtcClient.joinRTCRoom(roomId, this._options.joinType);\n      }\n\n      const { code, userIds, room } = callBack;\n\n      if (code !== RCRTCCode.SUCCESS) {\n        // 如果音视频服务未开通\n        if (code === RCRTCCode.NOT_OPEN_VIDEO_AUDIO_SERVER) {\n          this._exceptionClose(RCCallEndReason.SERVICE_NOT_OPENED);\n          // 己方其他端已在通话中\n        } if (code === RCRTCCode.SIGNAL_JOIN_RTC_ROOM_REFUSED) {\n          this._exceptionClose(RCCallEndReason.OTHER_CLIENT_IN_CALL);\n        } else {\n          this._exceptionClose(RCCallEndReason.NETWORK_ERROR);\n        }\n\n        this._logger.info('_', `[RCCallClient _joinRoom] join room failed: roomId -> ${roomId} RCRTCCode -> ${code}`);\n        return { code: RCCallErrorCode.JOIN_ROOM_ERROR };\n      }\n\n      /**\n       * 群聊本端加入房间后，更新人员状态为通话中\n       */\n      const conversationType = this._stateMachine.getConversationType();\n      conversationType === ConversationType.GROUP && this._stateMachine.userJoin([this._rtcClient.getCurrentId()]);\n\n      /**\n       * 针对私有云 @rongcloud/plugin-rtc@5.1.10-enterprise.7 增加补丁，\n       * 私有云升完 RTC sdk 版本后删掉此补丁\n       * 加完房间后，对方挂断了，需退出房间\n       */\n      if (this._stateMachine.getState() === RCCallSessionState.END) {\n        await this._rtcClient.leaveRoom(room!);\n        this._room = null as unknown as RCRTCRoom;\n        return { code: RCCallErrorCode.SUCCESS };\n      }\n\n      // 被叫方加入房间成功，但主叫方在加入房间前离线\n      if (userIds!.length < 1) {\n        this.joinRoomTimer = new Timer(() => {\n          this._exceptionClose(RCCallEndReason.REMOTE_NETWORK_ERROR);\n        }, 60000);\n      }\n      this._room = room as RCRTCRoom;\n    } catch (error) {\n      this._exceptionClose(RCCallEndReason.NETWORK_ERROR);\n      this._logger.error('_', `[RCCallSession _joinRoom] _rtcClient.joinRTCRoom throw exception roomId -> ${roomId}`);\n      console.error(error);\n      return { code: RCCallErrorCode.JOIN_ROOM_ERROR };\n    }\n\n    // 房间上注册监听事件\n    this._registerRoomEventListener();\n\n    // 注册房间质量数据监听器\n    this._registerReportListener();\n\n    try {\n      // 订阅远程的流，把远程的流抛给用户\n      await this._subscribeInRoomRemoteTrack();\n    } catch (error) {\n      // 结束通话session\n      this._exceptionClose(RCCallEndReason.SUBSCRIBE_ERROR);\n      this._logger.error('_', `[RCCallSession _joinRoom] _subscribeInRoomRemoteTrack Exception roomId -> ${roomId}`);\n      console.error(error);\n      return { code: RCCallErrorCode.JOIN_ROOM_ERROR };\n    }\n\n    try {\n      // 往房间里发布本地资源\n      await this._publish();\n    } catch (error) {\n      // 结束通话session\n      this._exceptionClose(RCCallEndReason.PUBLISH_ERROR);\n      this._logger.error('_', `[RCCallSession _joinRoom] _publish Exception roomId -> ${roomId}`);\n      console.error(error);\n      return { code: RCCallErrorCode.JOIN_ROOM_ERROR };\n    }\n    return { code: RCCallErrorCode.SUCCESS };\n  }\n\n  /**\n   * (初始化房间的时候) 订阅远程的流，把远程的流抛给用户\n   */\n  private async _subscribeInRoomRemoteTrack() {\n    // 获取所有远程已发布的音视频资源列表\n    const tracks: RCRemoteTrack[] = this._room.getRemoteTracks();\n    if (tracks.length) {\n      const { code } = await this._subscribeRetry(tracks, this._options.isAllowSubscribeRetry, this._RETRYCOUNT);\n      if (code !== RCRTCCode.SUCCESS) {\n        this._exceptionClose(RCCallEndReason.SUBSCRIBE_ERROR);\n        this._logger.error('_', `[RCCallSession _subscribeInRoomRemoteTrack] Resource subscription failed roomId -> ${this._stateMachine.getCallId()} RTC code -> ${code}`);\n      }\n    }\n  }\n\n  /**\n   * 可以重试的订阅\n   * @param params.tracks tracks\n   * @param params.isAllowSubscribeRetry 是否允许重试\n   * @param params.count 允许重试的次数\n   */\n  private async _subscribeRetry(tracks: RCRemoteTrack[], isAllowSubscribeRetry: boolean = false, count: number = 0): Promise<{ code: RCRTCCode }> {\n    const { code } = await this._room.subscribe(tracks);\n    if (code !== RCRTCCode.SUCCESS) {\n      try {\n        this._listener!.onTrackSubscribeFail && this._listener!.onTrackSubscribeFail(code, this);\n      } catch (error) {\n        this._logger.error('_', '[RCCallSession] _listener.onTrackSubscribeFail exception');\n        console.error(error);\n      }\n\n      // 如果不允许重试，直接返回\n      if (!isAllowSubscribeRetry) {\n        return { code };\n      }\n      if (count > 0) {\n        count--;\n        return this._subscribeRetry(tracks, isAllowSubscribeRetry, count);\n      }\n    }\n    return { code };\n  }\n\n  /**\n   * 发布本地资源的逻辑\n   *\n   */\n  private async _publish() {\n    const tracks = this._options.localTracks!;\n    const { code } = await this._publishRetry(tracks, this._options.isAllowPublishRetry, this._RETRYCOUNT);\n\n    // 若资源发布失败\n    if (code !== RCRTCCode.SUCCESS) {\n      this._exceptionClose(RCCallEndReason.PUBLISH_ERROR);\n      this._logger.info('_', `[RCCallSession _publist] Resource publishing failed: roomId -> ${this._stateMachine.getCallId()} RCRTCCode -> ${code}`);\n      return;\n    }\n\n    // 如果是主动发起的呼叫，已提前抛出了资源, 被动呼叫，这里才需要抛出\n    if (this._options.produceType === ProduceTypes.CALLEE) {\n      // 向外抛出本地流, 通知业务层trackReady\n      this._notifyTrackReady(tracks);\n    }\n  }\n\n  /**\n   * 可以重试的发布\n   * @param params.tracks tracks\n   * @param params.isAllowPublishRetry 是否允许重试\n   * @param params.count 允许重试的次数\n   */\n  private async _publishRetry(tracks: RCLocalTrack[], isAllowPublishRetry: boolean = false, count: number = 0): Promise<{ code: RCRTCCode }> {\n    const { code } = await this._room.publish(tracks);\n    if (code !== RCRTCCode.SUCCESS) {\n      try {\n        this._listener!.onTrackPublishFail && this._listener!.onTrackPublishFail(code, this);\n      } catch (error) {\n        this._logger.error('_', '[RCCallSession] _listener.onTrackPublishFail exception');\n        console.error(error);\n      }\n\n      // 如果不允许重试，直接返回\n      if (!isAllowPublishRetry) {\n        return { code };\n      }\n      if (count > 0) {\n        count--;\n        return this._publishRetry(tracks, isAllowPublishRetry, count);\n      }\n    }\n    return { code };\n  }\n\n  /**\n   * 退出房间\n   */\n  private async _leaveRoom() {\n    try {\n      // 退出房间\n      const callBack = await this._rtcClient.leaveRoom(this._room);\n      // 成功退出房间，触发RCCallClient实例上的onSessionClose监听，抛给用户信息\n      this._logger.info('_', `[RCCallSession _leaveRoom] Successfully exited the room code: ${callBack.code}`);\n    } catch (error) {\n      this._logger.error('_', '[RCCallSession _leaveRoom] leaveRoom throw exception');\n      console.error(error);\n    } finally {\n      const summaryInfo = this._stateMachine.getSummary();\n      eventEmitter.emit('sessionClose', { session: this, summaryInfo });\n    }\n  }\n\n  /**\n   * 出现异常后要处理的逻辑,\n   * @param endReason 原因\n   */\n  private _exceptionClose(endReason: RCCallEndReason) {\n    // 销毁本地流\n    this._options.localTracks && this._destroyTracks(this._options.localTracks);\n\n    // 结束状态机\n    this._stateMachine.close(endReason);\n  }\n\n  /**\n   * 用户调用的，注册session上的监听\n   */\n  public registerSessionListener(listener: ISessionListener): void {\n    // 先校验listener, 如果不通过，会trow error\n    const conclusion: IValidationResult = validateListener(listener);\n    if (!conclusion.result) {\n      throw new Error(`[RCCallSession registerSessionListener] ${conclusion.msg}`);\n    }\n    this._listener = { ...listener };\n  }\n\n  /**\n   * 调RTC API 获得本地流\n   */\n  private async _getLocalTrackCore(mediaType: RCCallMediaType, constraints?: IMediaStreamConstraints): Promise<{ code: RCCallErrorCode, tracks?: RCLocalTrack[] }> {\n    // 检测是否能够获得本地流\n    if (mediaType === RCCallMediaType.AUDIO) {\n      const { code, track } = await this._rtcClient.createMicrophoneAudioTrack('RongCloudRTC', constraints && constraints.audio && { ...constraints.audio });\n      if (code !== RCRTCCode.SUCCESS) {\n        this._logger.error('_', `[RCCallSession _getLocalTrackCore] get Audio local tracks failed RCT code -> ${code}`);\n        return { code: RCCallErrorCode.GET_LOCAL_AUDIO_TRACK_ERROR };\n      }\n      this._logger.info('_', '[RCCallSession _getLocalTrackCore] successfully get Audio local tracks');\n      return { code: RCCallErrorCode.SUCCESS, tracks: [track!] };\n    }\n    const { code, tracks } = await this._rtcClient.createMicrophoneAndCameraTracks('RongCloudRTC', constraints && { ...constraints });\n    if (code !== RCRTCCode.SUCCESS) {\n      this._logger.error('_', `[RCCallSession _getLocalTrackCore] get Audio and Video local tracks failed RCT code -> ${code}`);\n      return { code: RCCallErrorCode.GET_LOCAL_AUDIO_AND_VIDEO_TRACK_ERROR };\n    }\n    this._logger.info('_', '[RCCallSession _getLocalTrackCore] successfully get audio and video local tracks');\n    return { code: RCCallErrorCode.SUCCESS, tracks };\n  }\n\n  private async _getLocalTrack(mediaType: RCCallMediaType, constraints?: IMediaStreamConstraints): Promise<{ code: RCCallErrorCode, tracks?: RCLocalTrack[] }> {\n    // 并且是获得音视频, 并且 （如果获得音视频不成功，允许降级获得音频）\n    if (this._options.isAllowDemotionGetStream && mediaType === RCCallMediaType.AUDIO_VIDEO) {\n      const { code, tracks } = await this._getLocalTrackCore(RCCallMediaType.AUDIO_VIDEO, constraints);\n\n      // 如果音视频不能获得，就降级获得音频\n      if (code !== RCCallErrorCode.SUCCESS) {\n        const { code, tracks } = await this._getLocalTrackCore(RCCallMediaType.AUDIO, constraints);\n        if (code !== RCCallErrorCode.SUCCESS) {\n          // 获取资源失败，需要调状态机state 为 end\n          this._exceptionClose(RCCallEndReason.GET_MEDIA_RESOURCES_ERROR);\n          return { code };\n        }\n        return { code, tracks };\n      }\n      return { code, tracks };\n    }\n    const { code: _code, tracks } = await this._getLocalTrackCore(mediaType, constraints);\n    if (_code !== RCCallErrorCode.SUCCESS) {\n      // 获取资源失败，需要调状态机state 为 end\n      this._exceptionClose(RCCallEndReason.GET_MEDIA_RESOURCES_ERROR);\n\n      return { code: _code };\n    }\n    return { code: _code, tracks };\n  }\n\n  /**\n   * 通话中更换音频设备\n   */\n  public async changeAudioDevice(audioConstraints?: IMicphoneAudioProfile): Promise<{ code: RCCallErrorCode }> {\n    // 新设备的track\n    const recentTracks: RCLocalTrack[] = [];\n\n    // 整理后的本地track\n    const localTracks: RCLocalTrack[] = [];\n    const { code, track } = await this._rtcClient.createMicrophoneAudioTrack('RongCloudRTC', audioConstraints);\n    if (code !== RCRTCCode.SUCCESS) {\n      this._logger.error('_', `[RCCallSession changeDevice] get local Audio tracks failed RCTLib code -> ${code}`);\n      return { code: RCCallErrorCode.GET_LOCAL_AUDIO_TRACK_ERROR };\n    }\n\n    this._options.localTracks && this._options.localTracks.forEach((track: RCLocalTrack) => {\n      if (track.isAudioTrack()) {\n        // 之前的音频都销毁，（RTCLib SDK内部会在 addLocalTrack 清理同类型轨道数据，所以这里注释掉）\n        // track.destroy()\n      } else {\n        // 只把之前的视频留下\n        localTracks.push(track);\n      }\n    });\n\n    recentTracks.push(track!);\n\n    // 加上本地新产生的音频\n    localTracks.push(track!);\n    this._options.localTracks = localTracks;\n    // 通知业务层trackReady\n    this._notifyTrackReady(recentTracks);\n\n    // 如果当前已加入房间，发布新流\n    if (this._room) {\n      // 发布新流\n      const { code } = await this._room.publish(recentTracks);\n      if (code !== RCRTCCode.SUCCESS) {\n        return { code: RCCallErrorCode.AUDIO_PUBLISH_ERROR };\n      }\n    }\n    return { code: RCCallErrorCode.SUCCESS };\n  }\n\n  /**\n   * 群呼叫中继续邀请\n   * @param userIds 被邀请用户 ID 列表\n   * @param options.extra 消息的扩展信息\n   * @deprecated 5.1.2 废弃 options.pushTitle 通知的标题\n   * @deprecated 5.1.2 废弃 options.pushContent 通知内容\n   */\n  public async invite(userIds: string[], options: IInviteOptions = {}): Promise<{ code: RCCallErrorCode }> {\n    const { extra = '' } = options;\n    const { pushTitle = '', pushContent = '' } = (this._options.callPushConfig?.pushTitle || this._options.callPushConfig?.pushContent) ? this._options.callPushConfig : options;\n    const conclusion: IValidationResult[] = [validateUserIds(userIds), validateExtra(extra), validatePushTitle(pushTitle), validatePushContent(pushContent)];\n    const messages: string[] = [];\n    const result = conclusion.every((obj: IValidationResult) => {\n      !obj.result && messages.push(obj.msg!);\n      return obj.result;\n    });\n\n    if (!result) {\n      throw new Error(`[RCCallClient invite] ${messages.join('\\n')}`);\n    }\n\n    const { code } = await this._stateMachine.invite(userIds, { extra, pushTitle, pushContent });\n    return { code };\n  }\n\n  /**\n   * 同意接听\n   */\n  public async accept(constraints?: IMediaStreamConstraints): Promise<{ code: RCCallErrorCode }> {\n    const conclusion: IValidationResult = validateMediaStreamConstraints(constraints!);\n    if (!conclusion.result) {\n      throw new Error(`[RCCallSession accept] ${conclusion.msg}`);\n    }\n\n    // 接听之前，先挂断当前之外的session，现阶段不允许用户先择接听session，事先会在状态机内部挂断，这里抛出去，会清理其它的seesion\n    eventEmitter.emit('hungupOtherSession', { session: this });\n    const mediaType = this._stateMachine.getMediaType();\n    const { code: _code, tracks } = await this._getLocalTrack(mediaType, constraints);\n    if (_code !== RCCallErrorCode.SUCCESS) {\n      return { code: _code };\n    }\n    this._options.localTracks = tracks;\n\n    // 发送接听的消息\n    const { code } = await this._stateMachine.accept();\n    if (code !== RCCallErrorCode.SUCCESS) {\n      this._logger.error('_', `[RCCallSession accept]Send accept message failed -> code: ${code}`);\n      return { code };\n    }\n    return { code };\n  }\n\n  /**\n   * 挂断\n   */\n  public async hungup(): Promise<{ code: RCCallErrorCode; }> {\n    // const summaryInfo = this._stateMachine.getSummary()\n    // eventEmitter.emit('sessionClose', { session: this, summaryInfo })\n    return this._stateMachine.hungup();\n  }\n\n  /**\n   * 通话媒体变更\n   *  @param mediaType RCCallMediaType.AUDIO 改为音频通话 | RCCallMediaType.AUDIO_VIDEO 改为音视频通话\n   */\n  public async _changeMediaType(mediaType: RCCallMediaType): Promise<{ code: RCCallErrorCode }> {\n    const { code } = await this._stateMachine.changeMediaType(mediaType);\n    if (code !== RCCallErrorCode.SUCCESS) {\n      this._logger.error('_', `[RCCallSession _changeMediaType] change media type fail code-> ${code}`);\n    }\n    return { code };\n  }\n\n  /**\n   * 获得本地视频\n   */\n  private _getLocalVideoTracks(): RCLocalTrack[] {\n    let localVideoTracks: RCLocalTrack[] = [];\n    if (!this._room) {\n      return localVideoTracks;\n    }\n    if (this._options.localTracks) {\n      localVideoTracks = this._options.localTracks.filter((track) => track.isVideoTrack());\n    }\n    return localVideoTracks;\n  }\n\n  /**\n   * 获得本地音频\n   */\n  private _getLocalAudioTracks(): RCLocalTrack[] {\n    let localAudiotracks: RCLocalTrack[] = [];\n    if (!this._room) {\n      return localAudiotracks;\n    }\n    if (this._options.localTracks) {\n      localAudiotracks = this._options.localTracks.filter((track) => track.isAudioTrack());\n    }\n    return localAudiotracks;\n  }\n\n  /**\n   * 把通话的MediaType升级到音视频\n   */\n  private async _setMediaTypeToAudioAndVideo() {\n    // 获得本端视频资源\n    const { code, track } = await this._rtcClient.createCameraVideoTrack();\n    if (code !== RCRTCCode.SUCCESS) {\n      return { code: RCCallErrorCode.GET_LOCAL_AUDIO_AND_VIDEO_TRACK_ERROR };\n    }\n\n    // 发布本端视频资源\n    const { code: _code } = await this._room.publish([track!]);\n\n    // 若资源发布失败\n    if (_code !== RCRTCCode.SUCCESS) {\n      this._logger.error('_', `[RCCallSession _enableVideo] Resource publishing failed: RCRTCCode -> ${code}`);\n      return;\n    }\n\n    // 通知业务层trackReady\n    this._notifyTrackReady([track!]);\n\n    // 发消息\n    this._changeMediaType(RCCallMediaType.AUDIO_VIDEO);\n  }\n\n  /**\n   * 把通话的MediaType降级到音频\n   * @param isSendMesssage 是否需要发消息, 默认发消息\n   */\n  private async _setMediaTypeToAudio() {\n    const tracks: RCLocalTrack[] = this._getLocalVideoTracks();\n    if (tracks.length) {\n      // 禁用视频\n      tracks.forEach((track: RCLocalTrack) => {\n        track.mute();\n      });\n\n      // 取消发布视频\n      const { code } = await this._room.unpublish(tracks);\n      if (code !== RCRTCCode.SUCCESS) {\n        this._logger.error('_', `[RCCallSession disableVideo] unpublish failed -> ${code}`);\n      }\n\n      // 关闭摄像头\n      this._destroyTracks(tracks);\n    }\n  }\n\n  /**\n   * 通话降级，目前需求只做通话降级，音视频可以降级为音频，音频不能升到音视频, 发消息成功才算降级成功\n   *\n   */\n  public async descendAbility(): Promise<{ code: RCCallErrorCode }> {\n    const { code } = await this._changeMediaType(RCCallMediaType.AUDIO);\n    if (code === RCCallErrorCode.SUCCESS) {\n      this._setMediaTypeToAudio();\n    }\n    return { code };\n  }\n\n  /**\n   * 禁用视频track\n   */\n  public async disableVideoTrack(): Promise<{ code: RCCallErrorCode }> {\n    if (!this._room) {\n      this._logger.error('_', `[RCCallSession disableAudioTrack] Room missing audio track -> ${RCCallErrorCode.NOT_IN_ROOM_ERROR}`);\n      return { code: RCCallErrorCode.NOT_IN_ROOM_ERROR };\n    }\n\n    const tracks: RCLocalTrack[] = this._getLocalVideoTracks();\n    if (!tracks.length) {\n      this._logger.error('_', `[RCCallSession disableVideoTrack] Room missing video track -> ${RCCallErrorCode.MISSING_VIDEO_TRACK_ERROR}`);\n      return { code: RCCallErrorCode.MISSING_VIDEO_TRACK_ERROR };\n    }\n\n    // 禁用视频\n    tracks.forEach((track: RCLocalTrack) => {\n      track.mute();\n    });\n\n    // 如果不需关闭摄像头\n    if (!this._options.isOffCameraWhenVideoDisable) {\n      return { code: RCCallErrorCode.SUCCESS };\n    }\n\n    // 取消发布视频\n    const { code } = await this._room.unpublish(tracks);\n    if (code !== RCRTCCode.SUCCESS) {\n      this._logger.error('_', `[RCCallSession disableVideo] unpublish failed -> ${code}`);\n\n      return { code: RCCallErrorCode.UNPUBLISH_VIDEO_ERROR };\n    }\n\n    tracks.forEach((track: RCLocalTrack) => {\n      // 关闭摄像头\n      track.destroy();\n    });\n\n    return { code: RCCallErrorCode.SUCCESS };\n  }\n\n  /**\n   * 启用视频track\n   */\n  public async enableVideoTrack(): Promise<{ code: RCCallErrorCode }> {\n    if (!this._room) {\n      this._logger.error('_', `[RCCallSession disableAudioTrack] Room missing audio track -> ${RCCallErrorCode.NOT_IN_ROOM_ERROR}`);\n      return { code: RCCallErrorCode.NOT_IN_ROOM_ERROR };\n    }\n\n    // 如果不需关闭摄像头\n    if (!this._options.isOffCameraWhenVideoDisable) {\n      const tracks: RCLocalTrack[] = this._getLocalVideoTracks();\n      if (!tracks.length) {\n        this._logger.error('_', `[RCCallSession EnableVideoTrack] Room missing video track -> ${RCCallErrorCode.MISSING_VIDEO_TRACK_ERROR}`);\n        return { code: RCCallErrorCode.MISSING_VIDEO_TRACK_ERROR };\n      }\n\n      // 启用视频\n      tracks.forEach((track: RCLocalTrack) => {\n        track.unmute();\n      });\n      return { code: RCCallErrorCode.SUCCESS };\n    }\n\n    // 获得本端视频资源\n    const { code, track } = await this._rtcClient.createCameraVideoTrack();\n    if (code !== RCRTCCode.SUCCESS) {\n      this._logger.error('_', `[RCCallSession EnableVideoTrack] Get Resource failed: RCRTCCode -> ${code}`);\n      return { code: RCCallErrorCode.GET_LOCAL_VIDEO_TRACK_ERROR };\n    }\n    const localTracks: RCLocalTrack[] = [];\n    this._options.localTracks && this._options.localTracks.forEach((track: RCLocalTrack) => {\n      if (track.isVideoTrack()) {\n        // 之前的视频都销毁\n        track.destroy();\n      } else {\n        // 只留下之前的音频\n        localTracks.push(track);\n      }\n    });\n\n    // 加上本地新产生的视频\n    localTracks.push(track!);\n    this._options.localTracks = localTracks;\n\n    // 为了触发对方的onVideoMuteChange 先禁用\n    track!.mute();\n\n    // 发布本端视频资源\n    const { code: _code } = await this._room.publish([track!]);\n\n    // 若资源发布失败\n    if (_code !== RCRTCCode.SUCCESS) {\n      this._logger.error('_', `[RCCallSession EnableVideoTrack] Resource publishing failed: RCRTCCode -> ${code}`);\n      return { code: RCCallErrorCode.VIDEO_PUBLISH_ERROR };\n    }\n\n    // 启用\n    track!.unmute();\n\n    // 通知业务层trackReady\n    this._notifyTrackReady([track!]);\n    return { code: RCCallErrorCode.SUCCESS };\n  }\n\n  /**\n   * 禁用音频track\n   */\n  public async disableAudioTrack() {\n    if (!this._room) {\n      this._logger.error('_', `[RCCallSession disableAudioTrack] Room missing audio track -> ${RCCallErrorCode.NOT_IN_ROOM_ERROR}`);\n      return { code: RCCallErrorCode.NOT_IN_ROOM_ERROR };\n    }\n\n    const tracks: RCLocalTrack[] = this._getLocalAudioTracks();\n\n    // 禁用音频\n    tracks.forEach((track: RCLocalTrack) => {\n      track.mute();\n    });\n  }\n\n  /**\n   * 启用音频track\n   */\n  public async enableAudioTrack() {\n    if (!this._room) {\n      this._logger.error('_', `[RCCallSession disableAudioTrack] Room missing audio track -> ${RCCallErrorCode.NOT_IN_ROOM_ERROR}`);\n      return { code: RCCallErrorCode.NOT_IN_ROOM_ERROR };\n    }\n\n    const tracks: RCLocalTrack[] = this._getLocalAudioTracks();\n\n    if (!tracks.length) {\n      this._logger.error('_', `[RCCallSession disableAudioTrack] Room missing audio track -> ${RCCallErrorCode.MISSING_VIDEO_TRACK_ERROR}`);\n      return { code: RCCallErrorCode.MISSING_VIDEO_TRACK_ERROR };\n    }\n\n    // 启用音频\n    tracks.forEach((track: RCLocalTrack) => {\n      track.unmute();\n    });\n  }\n\n  /**\n   * 销毁本地流\n   */\n  private _destroyTracks(tracks: RCLocalTrack[]) {\n    tracks.forEach((track: RCLocalTrack) => {\n      track.destroy();\n    });\n  }\n\n  /**\n   * 向外抛出本地流\n   */\n  private _notifyTrackReady(tracks: RCLocalTrack[] | RCRemoteTrack[]) {\n    tracks.forEach((track: RCLocalTrack | RCRemoteTrack) => {\n      try {\n        this._listener!.onTrackReady(track, this);\n      } catch (error) {\n        this._logger.error('_', '[RCCallSession _notifyTrackReady] _listener onTrackReady exception');\n        console.error(error);\n      }\n    });\n  }\n\n  /**\n   * 房间上注册事件\n   */\n  private _registerRoomEventListener() {\n    this._room.registerRoomEventListener(\n      {\n        /**\n         * 本端被踢出房间时触发\n         * @description 被踢出房间可能是由于服务端超出一定时间未能收到 rtcPing 消息，所以认为己方离线。\n         * 另一种可能是己方 rtcPing 失败次数超出上限，故而主动断线\n         * @param byServer\n         * 当值为 false 时，说明本端 rtcPing 超时\n         * 当值为 true 时，说明本端收到被踢出房间通知\n         */\n        onKickOff: (byServer: boolean, state?: RCKickReason | undefined) => {\n          const currentUserId: string = this._rtcClient.getCurrentId();\n          this._stateMachine.userLeave([currentUserId]);\n\n          if (!byServer) {\n            this._exceptionClose(RCCallEndReason.NETWORK_ERROR);\n          } else {\n            if (state === RCKickReason.SERVER_KICK) {\n              this._exceptionClose(RCCallEndReason.KICKED_BY_SERVER);\n            }\n            if (state === RCKickReason.OTHER_KICK) {\n              this._exceptionClose(RCCallEndReason.OTHER_CLIENT_JOINED_CALL);\n            }\n          }\n        },\n        /**\n         * 接收到房间信令时回调，用户可通过房间实例的 `sendMessage(name, content)` 接口发送信令\n         * @param name 信令名\n         * @param content 信令内容\n         * @param senderUserId 发送者 Id\n         * @param messageUId 消息唯一标识\n         */\n        onMessageReceive(name: string, content: any, senderUserId: string, messageUId: string) {\n        },\n        /**\n         * 监听房间属性变更通知\n         * @param name\n         * @param content\n         */\n        onRoomAttributeChange(name: string, content: string) {\n        },\n        /**\n         * 发布者禁用/启用音频\n         * @param audioTrack RCRemoteAudioTrack 类实例\n         */\n        onAudioMuteChange: (audioTrack: RCRemoteAudioTrack) => {\n          this._logger.info('_', `[RCCallSession onAudioMuteChange] userId->${audioTrack.getUserId()} muted -> ${audioTrack.isOwnerMuted()}`);\n          const muteUser: IMuteUser = {\n            userId: audioTrack.getUserId(),\n            muted: audioTrack.isOwnerMuted(),\n            kind: 'audio',\n            trackId: audioTrack.getTrackId(),\n          };\n          try {\n            // 通知给业务\n            this._listener!.onAudioMuteChange(muteUser, this);\n          } catch (error) {\n            this._logger.error('_', '[RCCallSession onAudioMuteChange] Missing listening method -> onTrackMuteChange');\n            console.error(error);\n          }\n        },\n        /**\n         * 发布者禁用/启用视频\n         * @param videoTrack RCRemoteVideoTrack 类实例对象\n         */\n        onVideoMuteChange: (videoTrack: RCRemoteVideoTrack) => {\n          this._logger.info('_', `[RCCallSession onVideoMuteChange]userId->${videoTrack.getUserId()} muted -> ${videoTrack.isOwnerMuted()}`);\n          const muteUser: IMuteUser = {\n            userId: videoTrack.getUserId(),\n            muted: videoTrack.isOwnerMuted(),\n            kind: 'video',\n            trackId: videoTrack.getTrackId(),\n          };\n\n          try {\n            // 通知给业务\n            this._listener!.onVideoMuteChange(muteUser, this);\n          } catch (error) {\n            this._logger.error('_', '[RCCallSession onVideoMuteChange] Missing listening method -> onVideoMuteChange');\n            console.error(error);\n          }\n        },\n        /**\n         * 房间内其他用户新发布资源时触发\n         * 如需获取加入房间之前房间内某个用户发布的资源列表，可使用 room.getRemoteTracksByUserId('userId') 获取\n         * @param tracks 新发布的音轨与视轨数据列表，包含新发布的 RCRemoteAudioTrack 与 RCRemoteVideoTrack 实例\n         */\n        onTrackPublish: async (tracks: RCRemoteTrack[]) => {\n          // 退出房间后，还会走到这？？，所以判断一下，没有room不执行订阅\n          if (this._room) {\n            // 按业务需求选择需要订阅资源，通过 room.subscribe 接口进行订阅\n            const { code } = await this._room.subscribe(tracks);\n            if (code !== RCRTCCode.SUCCESS) {\n              this._logger.error('_', `[RCCallSession onTrackPublish] subscribe failed RTCCode ->${code}`);\n            }\n          }\n        },\n        /**\n         * 房间用户取消发布资源\n         * @param tracks 被取消发布的音轨与视轨数据列表\n         * @description 当资源被取消发布时，SDK 内部会取消对相关资源的订阅，业务层仅需处理 UI 业务\n         */\n        onTrackUnpublish: (tracks: RCRemoteTrack[]) => {\n\n        },\n        /**\n         * 订阅的音视频流通道已建立, track 已可以进行播放\n         * @param track RCRemoteTrack 类实例\n         */\n        onTrackReady: (track: RCRemoteTrack) => {\n          const mediaType = this._stateMachine.getMediaType();\n\n          // 有时对方没有降级成功，扔抛过来视频，这时的视频不对外抛出\n          if (mediaType === RCCallMediaType.AUDIO && track.isVideoTrack()) {\n            return;\n          }\n\n          // 执行用户的onTrackReady监听\n          this._notifyTrackReady([track]);\n        },\n        /**\n         * 人员加入\n         * @param userIds 加入的人员 id 列表\n         */\n        onUserJoin: (userIds: string[]) => {\n          // 有人加入清除定时器\n          if (this.joinRoomTimer) {\n            this.joinRoomTimer.stop();\n          }\n          this._stateMachine.userJoin(userIds);\n        },\n        /**\n         * 人员退出\n         * @param userIds\n         */\n        onUserLeave: (userIds: string[]) => {\n          this._logger.info('_', `[RCCallSession onUserLeave] listening onUserLeave userIds -> ${userIds?.join(',')}`);\n          this._stateMachine.userLeave(userIds);\n        },\n        /**\n         * RTC 每次 Ping 结果\n         */\n        onPing: (result: RCRTCPingResult) => {\n          this._logger.info('_', `[RCCallSession onPing]${result}`);\n          try {\n            // 通知给业务\n            this._listener!.onPing && this._listener!.onPing(result, this);\n          } catch (error) {\n            this._logger.error('_', '[RCCallSession onPing] listening onPing exception');\n            console.error(error);\n          }\n        },\n      },\n    );\n  }\n\n  /**\n   * 注册房间质量数据监听器\n   */\n  private _registerReportListener() {\n    // 注册房间质量数据监听器\n    this._room.registerReportListener({\n      /**\n       * 用于接收状态数据报告\n       * @param report\n       */\n      onStateReport: (report: IRCRTCStateReport) => {\n        try {\n          this._listener!.onRTCStateReport && this._listener!.onRTCStateReport(report, this);\n        } catch (error) {\n          this._logger.error('_', '[RCCallSession onStateReport] listener onStateReport exception');\n          console.error(error);\n        }\n      },\n\n      /**\n       * ~ICE 连接状态变更通知~\n       * @since version 5.1.5\n       */\n      onICEConnectionStateChange: (state: RTCIceConnectionState) => {\n        try {\n          this._listener!.onICEConnectionStateChange && this._listener!.onICEConnectionStateChange(state, this);\n        } catch (error) {\n          this._logger.error('_', '[RCCallSession onICEConnectionStateChange] onICEConnectionStateChange exception');\n          console.error(error);\n        }\n      },\n\n    });\n  }\n\n  /**\n   *  通话唯一标识\n   */\n  public getSessionId(): string {\n    return this._stateMachine.getCallId();\n  }\n\n  /**\n   *  获取房间当前会话 Id，当房间内已无成员时房间会回收，重新加入时 sessionId 将更新，(用户录制资源用的)\n   */\n  public getRTCSessionId(): string | null {\n    return this._room ? this._room.getSessionId() : null;\n  }\n\n  /**\n   *  目标 ID，单呼对方人员 Id, 群呼群组 Id\n   */\n  public getTargetId(): string {\n    return this._stateMachine.getTargetId();\n  }\n\n  /**\n   *  获取会话类型\n   */\n  public getConversationType(): ConversationType {\n    return this._stateMachine.getConversationType();\n  }\n\n  /**\n   *  组织 ID\n   */\n  public getChannelId(): string {\n    return this._stateMachine.getChannelId();\n  }\n\n  /**\n   * 房间人员列表，不包含本端信息\n   */\n  public getRemoteUsers(): IUserData[] {\n    return this._stateMachine.getRemoteUsers();\n  }\n\n  /**\n * 房间人员列表，不包含本端信息\n */\n  public getUsers(): IUserData[] {\n    return this._stateMachine.getRemoteUsers();\n  }\n\n  /**\n   * 获取人员状态\n   */\n  public getUserState(userId: string): RCCallUserState {\n    if (!userId || typeof userId !== 'string') {\n      throw new Error('userId is required, must be of type \\'string\\'');\n    }\n    return this._stateMachine.getUserState(userId);\n  }\n\n  /**\n   * 获取session状态\n   */\n  public getState(): RCCallSessionState {\n    return this._stateMachine.getState();\n  }\n\n  /**\n   * 获得会话发起者id\n   */\n  public getCallerId(): string {\n    return this._stateMachine.getCallerId();\n  }\n\n  /**\n   * 获得mediaType\n   */\n  public getMediaType(): RCCallMediaType {\n    return this._stateMachine.getMediaType();\n  }\n}\n","import {\n  RTCPluginContext, IRuntime, IRTCJoinedInfo, ErrorCode, RTCJoinType, ILogger, BasicLogger,\n} from '@rongcloud/engine';\nimport {\n  RCCallEngine, RCCallStateMachine, RCCallErrorCode, RCCallLanguage, RCCallMediaType, IOfflineRecord,\n} from '@rc-embed/call-engine';\nimport { RCRTCClient, RCRTCCode, RCLocalTrack } from '@rongcloud/plugin-rtc';\nimport {\n  IRCCallInGroupParams, IRCCallInitOptions, IRCCallParams, IMediaStreamConstraints, IValidationResult, IPushConfig,\n} from './interface';\nimport { ProduceTypes } from './enums';\nimport eventEmitter from './eventEmitter';\nimport { RCCallSession } from './RCCallSession';\nimport {\n  validateListener, validateTargetId, validateMediaType, validateUserIds, validateExtra, validatePushTitle, validatePushContent, validatePushConfig,\n} from './validation';\n\nexport default class RCCallClient {\n  /**\n   * rtc实例\n   */\n  private readonly _rtcClient: RCRTCClient\n\n  /**\n   * callEngine层实例\n   */\n  private readonly _callEngine: RCCallEngine\n\n  /**\n   * 其它参数\n   */\n  private _options: IRCCallInitOptions\n\n  /**\n   * session列表\n   */\n  private _sessionList: RCCallSession[] = []\n\n  private _callPushConfig: IPushConfig = {}\n\n  private _hungupPushConfig: IPushConfig ={}\n\n  constructor(\n    private _context: RTCPluginContext,\n    private readonly _runtime: IRuntime,\n    private readonly _logger: BasicLogger,\n    _options: IRCCallInitOptions,\n  ) {\n    this._rtcClient = _options.rtcClient;\n\n    this._options = { /**\n       * 是否允许发布重试， 默认不允许\n       */\n      isAllowPublishRetry: false,\n\n      /**\n       * 是否允许订阅重试，默认不允许\n       */\n      isAllowSubscribeRetry: false,\n      /**\n       * 禁用视频时关摄像头, 默认关闭\n       */\n      isOffCameraWhenVideoDisable: true,\n      /**\n       * RTC 房间加入类型，默认   RTCJoinType.COEXIST = 2 两个设备共存\n       *     RTCJoinType.KICK = 0,踢前一个设备\n       *     RTCJoinType.REFUSE = 1,当前加入拒绝\n       *     RTCJoinType.COEXIST = 2 两个设备共存\n       */\n      joinType: RTCJoinType.COEXIST,\n\n      /**\n       * 允许降级获得流，获得音视频不成功 ，降级获得音频, 默认不允许\n       */\n      isAllowDemotionGetStream: false,\n\n      /**\n       * 语言设置 (推送), 不传默认为中文\n       */\n      lang: RCCallLanguage.ZH,\n      ..._options,\n    };\n\n    // 初始化callEngine, 并监听onInvite\n    this._callEngine = new RCCallEngine(this._context, _runtime, this._logger, {\n\n      /**\n       * 监听收到invite\n       */\n      onInvite: this._onInvite.bind(this),\n\n      /**\n       * 监听离线消息报告\n       */\n      onOfflineRecord: this._onOfflineRecord.bind(this),\n    }, {\n\n      /**\n       * 语言设置 (推送), 不传默认为中文\n       */\n      lang: this._options.lang || RCCallLanguage.ZH,\n\n    });\n\n    eventEmitter.on('sessionClose', ({ session, summaryInfo }) => {\n      // 从sessionList去掉这个关闭的session\n      this._removeSession(session);\n\n      try {\n        this._options.onSessionClose(session, summaryInfo);\n      } catch (error) {\n        this._logger.error('_', '[RCCCallClient] options.onSessionClose exception');\n        console.log(error);\n      }\n    });\n\n    // 接听之前挂断其它的session\n    eventEmitter.on('hungupOtherSession', ({ session }) => {\n      const id = session.getSessionId();\n      this._logger.info('_', `[RCCallClient hungupOtherSession] sessionId ready to accept -> ${id}`);\n      this._logger.info('_', `[RCCallClient hungupOtherSession] sessionList ->${this._sessionList.map((ses) => ses.getSessionId()).join(',')}`);\n      let i = 0;\n      while (this._sessionList.length > 1) {\n        // 如果与要接听的session不一致\n        if (this._sessionList[i].getSessionId() !== id) {\n          // 挂断\n          this._sessionList[i].hungup();\n\n          // 挂断后删除\n          this._sessionList.splice(i, 1);\n        } else {\n          // 如果是要接听的session，跳过这个索引，所以加1\n          i++;\n        }\n      }\n      this._logger.info('_', `[RCCallClient hungupOtherSession] current sessionList length ->${this._sessionList.length}`);\n    });\n  }\n\n  /**\n   * 监听onInvite\n   */\n  private _onInvite(stateMachine: RCCallStateMachine, extra?: string) {\n    this._logger.info('_', '[RCCallClient _onInvite] Received invite message');\n    const session = new RCCallSession(stateMachine, this._rtcClient, this._logger, {\n\n      // 是否允许订阅重试\n      isAllowSubscribeRetry: this._options.isAllowSubscribeRetry,\n\n      // 是否允许发布重试\n      isAllowPublishRetry: this._options.isAllowPublishRetry,\n\n      /**\n       * 禁用视频时关摄像头\n       */\n      isOffCameraWhenVideoDisable: this._options.isOffCameraWhenVideoDisable,\n\n      /**\n       * RTC 房间加入类型\n       */\n      joinType: this._options.joinType,\n\n      // 允许降级获得流，获得音视频不成功 ，降级获得音频, 默认不允许\n      isAllowDemotionGetStream: this._options.isAllowDemotionGetStream,\n\n      // 标明是被叫产生的session\n      produceType: ProduceTypes.CALLEE,\n\n      callPushConfig: this._callPushConfig,\n\n      hungupPushConfig: this._hungupPushConfig,\n    });\n    this._logger.info('_', '[RCCallClient _onInvite] Received invite message, successfully created session');\n\n    /**\n     * 如果通话的时候不允许接听新的通话，直接挂断， 这些工作在callEngine里完成\n     */\n    this._sessionList.push(session);\n\n    try {\n      // 执行用户API的监听\n      this._options.onSession(session, extra);\n    } catch (error) {\n      this._logger.error('_', '[RCCallClient _options.onSession] onSession exception');\n      console.log(error);\n    }\n\n    // 必须在onSession里注册session监听事件，这里检测一下有没有注册\n    if (session._listener) {\n      const conclusion: IValidationResult = validateListener(session._listener);\n      if (!conclusion.result) {\n        throw new Error(conclusion.msg);\n      }\n    } else {\n      this._logger.error('_', '[RCCallClient _options.onSession] session Must Have Listener');\n      throw new Error('[RCCallSession  _options.onSession] session Must Have Listener');\n    }\n  }\n\n  /**\n   * 监听离线消息报告\n   * @param record\n   */\n  public _onOfflineRecord(record: IOfflineRecord) {\n    try {\n      // 执行用户API的监听\n      this._options.onOfflineRecord && this._options.onOfflineRecord(record);\n    } catch (error) {\n      this._logger.error('_', '[RCCallClient _options.onOfflineRecord] onOfflineRecord exception');\n      console.log(error);\n    }\n  }\n\n  /**\n   * 注册用户信息。注册后，在发起邀请或挂断等操作时，会将该信息一并发送给对端\n   * @param info.name        用户名称\n   * @param info.portraitUri 用户头像信息\n   * @param info.extra       预留拓展字段\n   */\n  public registerUserInfo(info: { name?: string; portraitUri?: string; extra?: string; } = {}) {\n    this._callEngine.registerUserInfo(info);\n    this._logger.info('_', '[RCCallClient registerUserInfo] successfully register user info data');\n  }\n\n  /**\n  * 跨App单呼，\b发送invite消息，回调回来接收stateMachine, 建session\n  * @param params.targetId 被呼叫一方的用户 id 必填\n  * @param params.mediaType 音频呼叫 or 音视频呼叫  必填\n  * @param params.listener (session上的监听) 必填\n  * @param params.constraints 获取音频或音视频资源时的参数 可选\n  * @param params.channelId 组织 Id 可选\n  * @param params.extra 消息扩展信息\n  * @deprecated 5.1.2 版本废弃 params.pushTitle 通知的标题\n  * @deprecated 5.1.2 版本废弃 params.pushContent 通知的内容\n  * @param params.bitrate 需要设置的码率参数\n  *\n  */\n  public async startCrossCall({\n    targetId, mediaType = RCCallMediaType.AUDIO, listener, constraints, channelId = '', extra = '', pushTitle = '', pushContent = '', bitrate,\n  }: IRCCallParams): Promise<{ code: RCCallErrorCode, session?: RCCallSession }> {\n    return this.__call({\n      targetId, mediaType, listener, constraints, channelId, extra, pushTitle, pushContent, bitrate, isCrossAppkey: true,\n    });\n  }\n\n  /**\n   * 单呼，\b发送invite消息，回调回来接收stateMachine, 建session\n   * @param params.targetId 被呼叫一方的用户 id 必填\n   * @param params.mediaType 音频呼叫 or 音视频呼叫  必填\n   * @param params.listener (session上的监听) 必填\n   * @param params.constraints 获取音频或音视频资源时的参数 可选\n   * @param params.channelId 组织 Id 可选\n   * @param params.extra 消息扩展信息\n   * @deprecated 5.1.2 版本废弃 params.pushTitle 通知的标题\n   * @deprecated 5.1.2 版本废弃 params.pushContent 通知的内容\n   * @param params.bitrate 需要设置的码率参数\n   *\n   */\n  public async call({\n    targetId, mediaType = RCCallMediaType.AUDIO, listener, constraints, channelId = '', extra = '', pushTitle = '', pushContent = '', bitrate,\n  }: IRCCallParams): Promise<{ code: RCCallErrorCode, session?: RCCallSession }> {\n    return this.__call({\n      targetId, mediaType, listener, constraints, channelId, extra, pushTitle, pushContent, bitrate,\n    });\n  }\n\n  private async __call({\n    targetId, mediaType = RCCallMediaType.AUDIO, listener, constraints, channelId = '', extra = '', pushTitle = '', pushContent = '', bitrate, isCrossAppkey = false,\n  }: IRCCallParams): Promise<{ code: RCCallErrorCode, session?: RCCallSession }> {\n    const { pushTitle: pushTitleValue = '', pushContent: pushContentValue = '' } = (this._callPushConfig.pushTitle || this._callPushConfig.pushContent) ? this._callPushConfig : { pushTitle, pushContent };\n    this._logger.info('_', `[RCCallClient call] extra->${extra} pushTitle->${pushTitleValue} pushContent->${pushContentValue}`);\n    const conclusion: IValidationResult[] = [validateTargetId(targetId), validateMediaType(mediaType), validateListener(listener), validateExtra(extra), validatePushTitle(pushTitleValue), validatePushContent(pushContentValue)];\n    const messages: string[] = [];\n    const result = conclusion.every((obj: IValidationResult) => {\n      !obj.result && messages.push(obj.msg!);\n      return obj.result;\n    });\n    if (!result) {\n      throw new Error(`[RCCallClient call] ${messages.join('\\n')}`);\n    }\n\n    let localTracks: RCLocalTrack[] = [];\n\n    const { code: _code, tracks } = await this._getLocalTrack(mediaType, constraints);\n    if (_code !== RCCallErrorCode.SUCCESS) {\n      return { code: _code };\n    }\n    localTracks = tracks!;\n\n    localTracks.forEach((track) => {\n      // 设置码率\n      if (track.isAudioTrack() && bitrate?.audio) {\n        track.setBitrate(bitrate?.audio);\n      }\n      if (track.isVideoTrack() && bitrate?.video) {\n        track.setBitrate(bitrate?.video?.max, bitrate?.video?.min, bitrate?.video?.start);\n      }\n      // 向外抛出本地流\n      listener.onTrackReady(track);\n    });\n\n    // 调用callEngine的call返回一个状态机的实例\n    const { code, stateMachine } = await this._callEngine.call(channelId, targetId, mediaType, extra, pushTitleValue, pushContentValue, isCrossAppkey);\n    if (code === RCCallErrorCode.SUCCESS && stateMachine) {\n      this._logger.info('_', '[RCCallClient call] successfully created state machine');\n      const session = new RCCallSession(stateMachine, this._rtcClient, this._logger, {\n        localTracks,\n\n        // 是否允许订阅重试\n        isAllowSubscribeRetry: this._options.isAllowSubscribeRetry,\n\n        // 是否允许订阅重试\n        isAllowPublishRetry: this._options.isAllowPublishRetry,\n\n        /**\n         * 禁用视频时关摄像头\n         */\n        isOffCameraWhenVideoDisable: this._options.isOffCameraWhenVideoDisable,\n\n        /**\n         * RTC 房间加入类型\n         */\n        joinType: this._options.joinType,\n\n        // 允许降级获得流，获得音视频不成功 ，降级获得音频, 默认不允许\n        isAllowDemotionGetStream: this._options.isAllowDemotionGetStream,\n\n        // 标明是主叫产生的session\n        produceType: ProduceTypes.CALLER,\n\n        isCrossAppkey,\n\n        callPushConfig: this._callPushConfig,\n\n        hungupPushConfig: this._hungupPushConfig,\n      });\n\n      // session上注册监听事件\n      session.registerSessionListener(listener);\n\n      this._sessionList.push(session);\n      this._logger.info('_', `[RCCallClient call] successfully created session object, sessionId: ${session.getSessionId()}`);\n      return { code, session };\n    }\n    this._logger.error('_', `[RCCallClient call] call failed code ->: ${code}`);\n    localTracks.forEach((track) => {\n      // 禁用视频\n      track.mute();\n\n      // 关闭摄像头\n      track.destroy();\n    });\n\n    return { code };\n  }\n\n  /**\n   * 发起群组呼叫\n   * @param params.targetId 群组 Id 必填\n   * @param params.userIds 被呼叫的群内成员 Id 必填\n   * @param params.mediaType 音频呼叫 or 音视频呼叫 必填\n   * @param params.listener (session上的监听) 必填\n   * @param params.constraints 获取音频或音视频资源时的参数 可选\n   * @param params.channelId 组织 Id 可选\n   * @param params.extra 消息扩展信息 可选\n   * @deprecated 5.1.2 版本废弃 params.pushTitle 通知的标题\n   * @deprecated 5.1.2 版本废弃 params.pushContent 通知的内容\n   * @param params.bitrate 需要设置的码率参数\n   */\n  public async callInGroup({\n    targetId, userIds, mediaType = RCCallMediaType.AUDIO, listener, constraints, channelId = '', extra = '', pushTitle = '', pushContent = '', bitrate,\n  }: IRCCallInGroupParams): Promise<{ code: RCCallErrorCode, session?: RCCallSession }> {\n    const { pushTitle: pushTitleValue = '', pushContent: pushContentValue = '' } = (this._callPushConfig.pushTitle || this._callPushConfig.pushContent) ? this._callPushConfig : { pushTitle, pushContent };\n    const conclusion: IValidationResult[] = [validateTargetId(targetId), validateUserIds(userIds), validateMediaType(mediaType), validateListener(listener), validateExtra(extra), validatePushTitle(pushTitleValue), validatePushContent(pushContentValue)];\n    const messages: string[] = [];\n    const result = conclusion.every((obj: IValidationResult) => {\n      !obj.result && messages.push(obj.msg!);\n      return obj.result;\n    });\n\n    if (!result) {\n      throw new Error(`[RCCallClient callInGroup] ${messages.join('\\n')}`);\n    }\n\n    let localTracks: RCLocalTrack[] = [];\n\n    const { code: _code, tracks } = await this._getLocalTrack(mediaType, constraints);\n    if (_code !== RCCallErrorCode.SUCCESS) {\n      return { code: _code };\n    }\n    localTracks = tracks!;\n\n    localTracks.forEach((track) => {\n      // 设置码率\n      if (track.isAudioTrack() && bitrate?.audio) {\n        track.setBitrate(bitrate?.audio);\n      }\n      if (track.isVideoTrack() && bitrate?.video) {\n        track.setBitrate(bitrate?.video?.max, bitrate?.video?.min, bitrate?.video?.start);\n      }\n      // 向外抛出本地流\n      listener.onTrackReady(track);\n    });\n\n    // 往组里发消息\n    const { code, stateMachine } = await this._callEngine.callInGroup(channelId, targetId, mediaType, userIds, extra, pushTitleValue, pushContentValue);\n    if (code === RCCallErrorCode.SUCCESS && stateMachine) {\n      this._logger.info('_', '[RCCallClient callInGroup] successfully created state machine');\n      const session = new RCCallSession(stateMachine, this._rtcClient, this._logger, {\n        localTracks,\n\n        // 是否允许订阅重试\n        isAllowSubscribeRetry: this._options.isAllowSubscribeRetry,\n\n        // 是否允许发布重试\n        isAllowPublishRetry: this._options.isAllowPublishRetry,\n\n        /**\n         * 禁用视频时关摄像头\n         */\n        isOffCameraWhenVideoDisable: this._options.isOffCameraWhenVideoDisable,\n\n        /**\n         * RTC 房间加入类型\n         */\n        joinType: this._options.joinType,\n\n        // 允许降级获得流，获得音视频不成功 ，降级获得音频, 默认不允许\n        isAllowDemotionGetStream: this._options.isAllowDemotionGetStream,\n\n        // 标明是主叫产生的session\n        produceType: ProduceTypes.CALLER,\n\n        callPushConfig: this._callPushConfig,\n\n        hungupPushConfig: this._hungupPushConfig,\n      });\n\n      // session上注册监听事件\n      session.registerSessionListener(listener);\n      this._sessionList.push(session);\n      this._logger.info('_', `[RCCallClient callInGroup] successfully created session object, sessionId: ${session.getSessionId()}`);\n      return { code, session };\n    }\n    this._logger.info('_', `[RCCallClient callInGroup] callInGroup failed code -> ${code}`);\n    localTracks.forEach((track) => {\n      // 禁用视频\n      track.mute();\n\n      // 关闭摄像头\n      track.destroy();\n    });\n\n    return { code };\n  }\n\n  /**\n   * 调RTC API 获得本地流\n   */\n  private async _getLocalTrackCore(mediaType: RCCallMediaType, constraints?: IMediaStreamConstraints): Promise<{ code: RCCallErrorCode, tracks?: RCLocalTrack[] }> {\n    // 检测是否能够获得本地流\n    if (mediaType === RCCallMediaType.AUDIO) {\n      const { code, track } = await this._rtcClient.createMicrophoneAudioTrack('RongCloudRTC', constraints && constraints.audio && { ...constraints.audio });\n      if (code !== RCRTCCode.SUCCESS) {\n        this._logger.error('_', `[RCCallClient _getTrack] get Audio local tracks failed RCT code -> ${code}`);\n        return { code: RCCallErrorCode.GET_LOCAL_AUDIO_TRACK_ERROR };\n      }\n      this._logger.info('_', '[RCCallClient _getTrack] successfully get Audio local tracks');\n      return { code: RCCallErrorCode.SUCCESS, tracks: [track!] };\n    }\n    const { code, tracks } = await this._rtcClient.createMicrophoneAndCameraTracks('RongCloudRTC', constraints && { ...constraints });\n    if (code !== RCRTCCode.SUCCESS) {\n      this._logger.error('_', `[RCCallClient _getTrack] get Audio and Video local tracks failed RCT code -> ${code}`);\n      return { code: RCCallErrorCode.GET_LOCAL_AUDIO_AND_VIDEO_TRACK_ERROR };\n    }\n    this._logger.info('_', '[RCCallClient _getTrack] successfully get audio and video local tracks');\n    return { code: RCCallErrorCode.SUCCESS, tracks };\n  }\n\n  private async _getLocalTrack(mediaType: RCCallMediaType, constraints?: IMediaStreamConstraints): Promise<{ code: RCCallErrorCode, tracks?: RCLocalTrack[] }> {\n    // 如果是允许降级获得流，并且是获得音视频\n    if (this._options.isAllowDemotionGetStream && mediaType === RCCallMediaType.AUDIO_VIDEO) {\n      const { code, tracks } = await this._getLocalTrackCore(RCCallMediaType.AUDIO_VIDEO, constraints);\n\n      // 如果音视频不能获得，就降组获得音频\n      if (code !== RCCallErrorCode.SUCCESS) {\n        const { code, tracks } = await this._getLocalTrackCore(RCCallMediaType.AUDIO, constraints);\n        if (code !== RCCallErrorCode.SUCCESS) {\n          return { code };\n        }\n        return { code, tracks };\n      }\n      return { code, tracks };\n    }\n    const { code: _code, tracks } = await this._getLocalTrackCore(mediaType, constraints);\n    if (_code !== RCCallErrorCode.SUCCESS) {\n      return { code: _code };\n    }\n    return { code: _code, tracks };\n  }\n\n  /**\n   * 从sessionList删除某个session\n   */\n  private _removeSession(session: RCCallSession) {\n    const id = session.getSessionId();\n    this._sessionList = this._sessionList.filter((session) => session.getSessionId() !== id);\n  }\n\n  /**\n   * 获取己方其他端加入通话（已加入 RTC 房间）的用户信息\n   */\n  public async getJoinedRoomInfo(): Promise<{ code: RCCallErrorCode, data?: IRTCJoinedInfo[] }> {\n    const { code, data } = await this._context.getRTCJoinedUserInfo(this._context.getCurrentId());\n    if (code !== ErrorCode.SUCCESS) {\n      this._logger.error('_', `getJoinedUserInfo error code: ${code}`);\n      return { code: RCCallErrorCode.QUERY_JOINED_USER_INFO_ERROR };\n    }\n\n    return { code: RCCallErrorCode.SUCCESS, data };\n  }\n\n  /**\n   * 设置呼叫、挂断推送数据\n   * @param callPushConfig 呼叫推送配置\n   * @param callPushConfig.pushTitle 呼叫推送标题\n   * @param callPushConfig.pushContent 呼叫推送内容\n   * @param hungupPushConfig 挂断推送配置\n   * @param hungupPushConfig.pushTitle 挂断推送标题\n   * @param hungupPushConfig.pushContent 挂断推送内容\n   */\n  public setPushConfig(callPushConfig: IPushConfig = {}, hungupPushConfig: IPushConfig = {}) {\n    const conclusion = [callPushConfig, hungupPushConfig].map((pushConfig: IPushConfig) => validatePushConfig(pushConfig))[0];\n\n    const messages: string[] = [];\n    const result = conclusion.every((obj: IValidationResult) => {\n      !obj.result && messages.push(obj.msg!);\n      return obj.result;\n    });\n\n    if (!result) {\n      throw new Error(`[RCCallClient callInGroup callPushConfig or hungupPushConfig] ${messages.join('\\n')}`);\n    }\n\n    this._callPushConfig = callPushConfig;\n    this._hungupPushConfig = hungupPushConfig;\n  }\n}\n","import {\n  IPluginGenerator, IRuntime, RTCPluginContext, VersionManage,\n} from '@rongcloud/engine';\nimport {\n  RCCallErrorCode, RCCallLanguage, RCCallEndReason, RCCallMediaType, RCCallUserState, RCCallSessionState, IEndSummary, IInvitedUsers, ISenderInfo, IOfflineRecord,\n} from '@rc-embed/call-engine';\nimport { IRCRTCStateReport } from '@rongcloud/plugin-rtc';\nimport {\n  IRCCallInitOptions, IRCCallParams, IRCCallInGroupParams, ISessionListener, IMuteUser, IMediaStreamConstraints, IDeviceChangeParams, IValidationResult, IPushConfig,\n} from './interface';\nimport RCCallClient from './RCCallClient';\nimport { RCCallSession } from './RCCallSession';\nimport { validateCallInitOptions } from './validation';\n\n// plugin-call 版本上报\nVersionManage.add('plugin-call', __VERSION__);\n\nconst installer: IPluginGenerator<RCCallClient, IRCCallInitOptions> = {\n  tag: 'RCCall',\n  verify(runtime: IRuntime) {\n    return runtime.tag === 'browser';\n  },\n  setup(context: RTCPluginContext, runtime: IRuntime, options: IRCCallInitOptions): RCCallClient {\n    // 先校验参数\n    const conclusion: IValidationResult = validateCallInitOptions(options);\n    if (!conclusion.result) {\n      throw new Error(`[RCCallLib installer steup]${conclusion.msg}`);\n    }\n\n    // 校验当前安装的 engine 版本是否可用\n    if (!VersionManage.validEngine(__REQUIRED_ENGINE_VERSION__)) {\n      throw new Error(`The current engine version '${VersionManage.getInfo().engine}' error, plugin-call required engine version at least '${__REQUIRED_ENGINE_VERSION__}'.`);\n    }\n\n    const logger = context.createLogger('RCCall', 'RTC');\n    options.logOutputLevel && logger.setOutputLevel(options.logOutputLevel);\n\n    if (typeof options.logLevel !== 'undefined') {\n      logger.warn('_', 'The \\'logLevel\\' parameter is deprecated, please use \\'logOutputLevel\\' instead.');\n    }\n\n    logger.warn('_', `RCCall Version: ${__VERSION__}, Commit: ${__COMMIT_ID__}`);\n\n    return new RCCallClient(context, runtime, logger, options);\n  },\n};\n\nexport {\n  installer,\n  RCCallClient,\n  RCCallSession,\n  RCCallLanguage,\n  RCCallErrorCode,\n  RCCallEndReason,\n  RCCallMediaType,\n  RCCallUserState,\n  RCCallSessionState,\n};\n\nexport type {\n  IEndSummary,\n  IRCCallInitOptions,\n  IRCCallParams,\n  IRCCallInGroupParams,\n  ISessionListener,\n  ISenderInfo,\n  IMuteUser,\n  IInvitedUsers,\n  IMediaStreamConstraints,\n  IDeviceChangeParams,\n  IOfflineRecord,\n  IRCRTCStateReport,\n  IPushConfig,\n};\n"],"names":["string10to64","number","chars","radix","qutient","arr","mod","getUUID","c","r","getUUID22","uuid","generateRandomId","random","timerSetTimeout","func","timeout","eventEmitter","EventEmitter","getCallDeviceId","runtime","key","MsgCallStatus","MsgCallStatus2","RCCallErrorCode","RCCallEndReason","CallRemoteEndReason","RCCallMessageType","RCCallSessionState","RCCallSessionState2","RCCallUserState","RCCallUserState2","RCCrossCallType","Timer$1","callback","__publicField","endTime","duration","RCCallStateMachine","_context","_runtime","_logger","_callMsgHandler","_channelId","_conversationType","_targetId","_mediaType","_callId","sentTime","delayTime","userId","state","reason","_a","user","message","senderUserId","userProfile","hungupReason","messageType","endReason","remoteUserId","msgDeviceId","suid","deviceId","senderDeviceId","currentUserId","ConversationType","isCurrentUserId","mediaType","content","isLessThanTwo","isInviteUser","isNoOneAnswered","watchers","inids","roomType","userIdList","id","inviteUserIds","Timer","existedUserPofiles","caller","needRingingUserIds","userInfo","callStatus","error","userIds","extra","pushTitle","pushContent","isCrossAppkey","code","isCaller","endCode","options","existedUserIds","hangupParams","msgCode","remoteUsers","pushConfig","remoteUser","uid","beginTimestamp","endTimestamp","summary","RCCallMediaType","RCCallMediaType2","MemberModifyType","MemberModifyType2","Platform","OfflineRecorder","_onRecord","channelId","conversationType","targetId","callId","isSelfSend","messages","msg","callMsgTypes","EngineErrorCodeToCallErrorCode","ErrorCode","CallMessageHandler","_offlineMsgItv","_getStateMachine","record","markTime","currentMsgSentTime","insertIndex","index","currentTime","buffers","item","messageUId","isOffLineMessage","offlineBuffers","inviteCallId","isInviteMsgType","isLessThanOfflineMsgItv","taskMsgList","i","otherCallId","delIndex","delOfflineIndex","isUnfinishedPrivateCall","isOnlyInvite","hasInviteAndRinging","isUnfinishedGrpCall","isUnfinished","noOneAccept","allUserIds","isCallerHungup","taskMsgSenderUserId","delCount","listener","funcName","event","eventType","directionalUserIdList","pushData","sendOpts","pushConifg","callerId","RCCallLanguage","EN","ZH","Local","lang","RCCallEngine","_watchers","_options","crtUserId","info","ProduceTypes","fun","on","data","funs","cb","length","isLanguage","val","isJoinType","RTCJoinType","isLogLevel","LogL","validateCallInitOptions","keyNames","keys","missingKeys","validateListener","validateTargetId","validateMediaType","validateExtra","validatePushTitle","validatePushContent","validatePushConfig","conclusion","validateUserIds","str","isRCFrameRate","isValidResolution","resolution","RCResolution","validateMediaStreamConstraints","constraints","RCCallSession","_stateMachine","_rtcClient","roomId","summaryInfo","sender","invitedUsers","callBack","room","RCRTCCode","tracks","isAllowSubscribeRetry","count","isAllowPublishRetry","track","_code","audioConstraints","recentTracks","localTracks","_b","obj","localVideoTracks","localAudiotracks","byServer","RCKickReason","name","audioTrack","muteUser","videoTrack","result","report","RCCallClient","session","ses","stateMachine","bitrate","pushTitleValue","pushContentValue","_c","callPushConfig","hungupPushConfig","VersionManage","installer","context","logger"],"mappings":"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"}
|
|
2643
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"index.esm.js","sources":["../../../engine/helper.ts","../../../engine/core/enums/MsgCallStatus.ts","../../../engine/core/enums/RCCallCode.ts","../../../engine/core/enums/RCCallEndReason.ts","../../../engine/core/enums/RCCallMessageType.ts","../../../engine/core/enums/RCCallSessionState.ts","../../../engine/core/enums/RCCallUserState.ts","../../../engine/core/enums/RCCrossCallType.ts","../../../engine/core/Timer.ts","../../../engine/core/StateMachine.ts","../../../engine/core/enums/RCCallMediaType.ts","../../../engine/core/enums/MemberModifyType.ts","../../../engine/core/enums/Platform.ts","../../../engine/core/OfflineRecorder.ts","../../../engine/core/MessageHandler.ts","../../../engine/core/enums/RCCallLang.ts","../../../engine/core/locale/en.ts","../../../engine/core/locale/zh.ts","../../../engine/core/locale/index.ts","../../../engine/index.ts","../../../src/enums.ts","../../../src/utils.ts","../../../src/eventEmitter.ts","../../../src/validation.ts","../../../src/helper.ts","../../../src/timer.ts","../../../src/RCCallSession.ts","../../../src/RCCallClient.ts","../../../src/index.ts"],"sourcesContent":["import { EventEmitter, IRuntime } from '@rongcloud/engine';\n\nconst string10to64 = (number: number) => {\n  const chars = '0123456789abcdefghigklmnopqrstuvwxyzABCDEFGHIGKLMNOPQRSTUVWXYZ+/'.split('');\n  const radix = chars.length + 1;\n  let qutient = +number;\n  const arr = [];\n  do {\n    const mod = qutient % radix;\n    qutient = (qutient - mod) / radix;\n    arr.unshift(chars[mod]);\n  } while (qutient);\n  return arr.join('');\n};\n\nconst getUUID = () => 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, (c) => {\n  const r = Math.random() * 16 | 0;\n  const v = c === 'x' ? r : (r & 0x3 | 0x8);\n  return v.toString(16);\n});\n\n/* 获取 22 位的 UUID */\nexport const getUUID22 = () => {\n  let uuid: string | number = getUUID();\n  uuid = `${uuid.replace(/-/g, '')}0`;\n  uuid = parseInt(uuid, 16);\n  uuid = string10to64(uuid);\n  if (uuid.length > 22) {\n    uuid = uuid.slice(0, 22);\n  }\n  return uuid;\n};\n\n/**\n * 生成随机 id 字符串\n */\nexport const generateRandomId = (): string => {\n  const random = Math.floor(Math.random() * 1000);\n  let uuid = getUUID22();\n  uuid = uuid.replace(/\\//g, '0');\n  const info = [uuid, Date.now(), random];\n  return info.join('_');\n};\n\nexport const timerSetTimeout = (func: Function, timeout: number): number => setTimeout(func, timeout) as number;\n\nexport const timerInterval = (fun: Function, itv: number): number => setInterval(fun, itv) as number;\n\nexport const eventEmitter = new EventEmitter();\n\n/**\n * call deviceId 存储在 sessionStorage 中\n */\nexport const getCallDeviceId = (runtime: IRuntime) => {\n  const key = 'RCCallDeviceId';\n  let uuid = runtime.sessionStorage.getItem(key);\n  if (!uuid) {\n    uuid = getUUID22();\n    runtime.sessionStorage.setItem(key, uuid);\n  }\n  return uuid;\n};\n","/**\n * 注释\n * TODO\n */\nexport enum MsgCallStatus {\n  OUTGOING = 1, // waiting\n  INCOMING, // waiting\n  RINGING, // waiting\n  CONNECTED, // keeping\n  IDLE, // end\n  ACCEPTED // waiting\n}\n","/**\n * 错误码，与移动端对齐\n * @description\n * 1. `51000 - 51999` 为 Android 专用段\n * 2. `52000 - 52999` 为 iOS 专用段\n * 3. `53000 - 53199` 为 Web RTC 专用段\n * 4. `53200 - 53499` 为 Web CallLib 专用段\n * *  `53200 - 53299` 为 Web CallEngine 专用端\n * *  `53300 - 53499` 为 Web Call 专用端\n * 5. `53500 - 53999` 为 Web 保留段\n */\nexport enum RCCallErrorCode {\n  /**\n   * 成功\n   */\n  SUCCESS = 10000,\n  /**\n   * 存在未结束的状态机\n   */\n  STATE_MACHINE_EXIT = 53200,\n  /**\n   * 发送 IM 消息失败\n   */\n  SEND_MSG_ERROR = 53201,\n  /**\n   * 被对方加入黑名单\n   */\n  REJECTED_BY_BLACKLIST = 53202,\n  /**\n   * 当前用户不再群组中\n   */\n  NOT_IN_GROUP = 53203,\n\n  /**\n   * Call 相关\n   */\n  /**\n   * 获得本地音频流失败\n   */\n  GET_LOCAL_AUDIO_TRACK_ERROR = 53301,\n\n  /**\n   * 获得本地视频流失败\n   */\n  GET_LOCAL_VIDEO_TRACK_ERROR = 53302,\n\n  /**\n   * 获得本地音视频流失败\n   */\n  GET_LOCAL_AUDIO_AND_VIDEO_TRACK_ERROR = 53303,\n\n  /**\n   * 加入房间失败\n   */\n  JOIN_ROOM_ERROR = 53304,\n\n  /**\n   * 发布音频失败\n   */\n  AUDIO_PUBLISH_ERROR = 53305,\n\n  /**\n   * 发布视频失败\n   */\n  VIDEO_PUBLISH_ERROR = 53306,\n\n  /**\n   * 发布音视频失败\n   */\n  AUDIO_AND_VIDEO_PUBLISH_ERROR = 53307,\n\n  /**\n   * 查询房间用户信息失败\n   */\n  QUERY_JOINED_USER_INFO_ERROR = 53308,\n\n  /**\n   * 禁用启用视频时，房间内缺少视频流\n   */\n  MISSING_VIDEO_TRACK_ERROR = 53309,\n\n  /**\n   * 取消发布视频失败\n   */\n  UNPUBLISH_VIDEO_ERROR = 53310,\n\n  /**\n   * 会话不是群组\n   */\n  CONVERSATION_NOT_GROUP_ERROR = 53311,\n\n  /**\n  * 不在room中禁用音视频\n  */\n  NOT_IN_ROOM_ERROR = 53312,\n\n  /**\n   * 参数错误\n   */\n  PARAM_ERROR = 53313\n}\n","/**\n * 挂断原因\n * @description\n * 根据原有 HangupReason 设计，己方原因与对方原因有相差 10 的差距.\n * 现有本地原因取值范围: 1 ~ 10, 远端原因取值范围: 11 ~ 20.\n * 为便于 HangupReason 扩展，保留 100 以内的取值.\n * 需要再次扩展时，己方原因使用: 21 ~ 30, 对应对方原因使用: 31 ~ 40,\n * 以此类推，\n * 己方原因使用: 41 ~ 50, 对方原因使用: 51 ~ 60,\n * 己方原因使用: 61 ~ 70, 对方原因使用: 71 ~ 80,\n * 己方原因使用: 71 ~ 90, 对方原因使用: 91 ~ 100.\n *\n * 各平台独有字段范围\n * Android 201 ~ 299\n * iOS     301 ~ 399\n * Web     401 ~ 499\n * 详细文档：https://gitbook.rongcloud.net/rtc-docs/#/rtc-client/ios/analysis/calllib/HangupReason\n */\nexport enum RCCallEndReason {\n  /**\n   * 己方取消已发出的通话请求\n   */\n  CANCEL = 1,\n  /**\n   * 己方拒绝收到的通话请求\n   */\n  REJECT = 2,\n  /**\n   * 己方挂断\n   */\n  HANGUP = 3,\n  /**\n   * 己方忙碌\n   */\n  BUSY_LINE = 4,\n  /**\n   * 己方未接听\n   */\n  NO_RESPONSE = 5,\n  /**\n   * 己方不支持当前音视频引擎\n   */\n  ENGINE_UNSUPPORTED = 6,\n  /**\n   * 己方网络错误\n   */\n  NETWORK_ERROR = 7,\n  /**\n   * 己方摄像头资源获取失败，可能是权限原因\n   */\n  GET_MEDIA_RESOURCES_ERROR = 8,\n  /**\n   * 己方资源发布失败\n   */\n  PUBLISH_ERROR = 9,\n  /**\n   * 己方订阅资源失败\n   */\n  SUBSCRIBE_ERROR = 10,\n  /**\n   * 对方取消发出的通话请求\n   */\n  REMOTE_CANCEL = 11,\n  /**\n   * 对方拒绝收到的通话请求\n   */\n  REMOTE_REJECT = 12,\n  /**\n   * 通话过程中对方挂断\n   */\n  REMOTE_HANGUP = 13,\n  /**\n   * 对方忙碌\n   */\n  REMOTE_BUSY_LINE = 14,\n  /**\n   * 对方未接听\n   */\n  REMOTE_NO_RESPONSE = 15,\n  /**\n   * 对方引擎不支持\n   */\n  REMOTE_ENGINE_UNSUPPORTED = 16,\n  /**\n   * 对方网络错误\n   */\n  REMOTE_NETWORK_ERROR = 17,\n  /**\n   * 对方摄像头资源获取失败，可能是权限原因\n   */\n  REMOTE_GET_MEDIA_RESOURCE_ERROR = 18,\n  /**\n   * 远端资源发布失败\n   */\n  REMOTE_PUBLISH_ERROR = 19,\n  /**\n   * 远端订阅资源失败\n   */\n  REMOTE_SUBSCRIBE_ERROR = 20,\n  /**\n   * 己方其他端已加入新通话\n   */\n  OTHER_CLIENT_JOINED_CALL = 21,\n  /**\n   * 己方其他端已在通话中\n   */\n  OTHER_CLIENT_IN_CALL = 22,\n  /**\n   * 己方被禁止通话\n   */\n  KICKED_BY_SERVER = 23,\n  /**\n   * 己方接听系统通话（移动端特有）\n  */\n  ACCEPT_SYSTEM_CALL = 24,\n  /**\n   * 远端其他端已加入新通话\n   */\n  REMOTE_OTHER_CLIENT_JOINED_CALL = 31,\n  /**\n   * 远端其他端已在通话中\n   */\n  REMOTE_OTHER_CLIENT_IN_CALL = 32,\n  /**\n   * 远端被禁止通话\n   */\n  REMOTE_KICKED_BY_SERVER = 33,\n  /**\n   * 远端接听系统通话（移动端特有）\n  */\n  REMOTE_ACCEPT_SYSTEM_CALL = 34,\n  /**\n   * 其他端接听\n   */\n  ACCEPT_BY_OTHER_CLIENT = 101,\n  /**\n   * 其他端挂断\n   */\n  HANGUP_BY_OTHER_CLIENT = 102,\n  /**\n   * 己方被对方加入黑名单\n   */\n  ADDED_TO_BLACKLIST = 103,\n  /**\n   * 音视频服务未开通\n   */\n  SERVICE_NOT_OPENED = 104\n}\n\n/**\n * 己方原因转为对方原因\n */\nexport const CallRemoteEndReason: { [key: number]: RCCallEndReason } = {\n  [RCCallEndReason.CANCEL]: RCCallEndReason.REMOTE_CANCEL,\n  [RCCallEndReason.REJECT]: RCCallEndReason.REMOTE_REJECT,\n  [RCCallEndReason.HANGUP]: RCCallEndReason.REMOTE_HANGUP,\n  [RCCallEndReason.BUSY_LINE]: RCCallEndReason.REMOTE_BUSY_LINE,\n  [RCCallEndReason.NO_RESPONSE]: RCCallEndReason.REMOTE_NO_RESPONSE,\n  [RCCallEndReason.ENGINE_UNSUPPORTED]: RCCallEndReason.REMOTE_ENGINE_UNSUPPORTED,\n  [RCCallEndReason.NETWORK_ERROR]: RCCallEndReason.REMOTE_NETWORK_ERROR,\n  [RCCallEndReason.GET_MEDIA_RESOURCES_ERROR]: RCCallEndReason.REMOTE_GET_MEDIA_RESOURCE_ERROR,\n  [RCCallEndReason.PUBLISH_ERROR]: RCCallEndReason.REMOTE_PUBLISH_ERROR,\n  [RCCallEndReason.SUBSCRIBE_ERROR]: RCCallEndReason.REMOTE_SUBSCRIBE_ERROR,\n  [RCCallEndReason.OTHER_CLIENT_JOINED_CALL]: RCCallEndReason.REMOTE_OTHER_CLIENT_JOINED_CALL,\n  [RCCallEndReason.OTHER_CLIENT_IN_CALL]: RCCallEndReason.REMOTE_OTHER_CLIENT_IN_CALL,\n  [RCCallEndReason.KICKED_BY_SERVER]: RCCallEndReason.REMOTE_KICKED_BY_SERVER,\n  [RCCallEndReason.REMOTE_NO_RESPONSE]: RCCallEndReason.NO_RESPONSE,\n  [RCCallEndReason.ACCEPT_SYSTEM_CALL]: RCCallEndReason.REMOTE_ACCEPT_SYSTEM_CALL,\n};\n","export enum RCCallMessageType {\n  /**\n   * 邀请消息\n   */\n  VCInvite = 'RC:VCInvite',\n  /**\n   * 响铃消息\n   */\n  VCRinging = 'RC:VCRinging',\n  /**\n   * 接听消息\n   */\n  VCAccept = 'RC:VCAccept',\n  /**\n   * 挂断消息\n   */\n  VCHangup = 'RC:VCHangup',\n  /**\n   * 群呼中 人员变更消息\n   */\n  VCModifyMem = 'RC:VCModifyMem',\n  /**\n   * 媒体类型修改消息\n   */\n  VCModifyMedia = 'RC:VCModifyMedia',\n}\n","export enum RCCallSessionState {\n  /**\n   * 等待建立连接\n   */\n  WAITING,\n  /**\n   * 会话维持中\n   */\n  KEEPING,\n  /**\n   * 会话已结束\n   */\n  END\n}\n","export enum RCCallUserState {\n  /**\n   * 用户不存在于通话中\n   */\n  NONE = 0,\n  /**\n   * 等待接听\n   */\n  WAITING,\n  /**\n   * 通话中\n   */\n  KEEPING,\n}\n","export enum RCCrossCallType {\n  /*!\n  同App通话\n  */\n  RCCallRoomTypeNormalCall = 0,\n  /*!\n  跨App通话\n  */\n  RCCallRoomTypeAcrossCall = 7\n}\n","import { timerSetTimeout } from '../helper';\n\nexport class Timer {\n  private _timerId: number = 0\n\n  private _startTime: number = 0\n\n  constructor(callback: Function, timeout: number) {\n    if (callback) {\n      this._timerId = timerSetTimeout(() => {\n        callback();\n      }, timeout);\n    }\n    this._startTime = Date.now();\n  }\n\n  stop(): {\n    startTime: number,\n    endTime: number,\n    duration: number\n    } {\n    clearTimeout(this._timerId);\n\n    const endTime = Date.now();\n    let duration = endTime - this._startTime;\n    if (this._startTime === 0) {\n      duration = 0;\n    }\n\n    return {\n      startTime: this._startTime,\n      endTime,\n      duration,\n    };\n  }\n\n  reset() {\n    this._startTime = 0;\n  }\n}\n","import {\n  ConversationType, ILogger, IReceivedMessage, IRuntime, RTCPluginContext,\n} from '@rongcloud/engine';\nimport { eventEmitter, getCallDeviceId } from '../helper';\nimport { MsgCallStatus } from './enums/MsgCallStatus';\nimport { RCCallErrorCode } from './enums/RCCallCode';\nimport { CallRemoteEndReason, RCCallEndReason } from './enums/RCCallEndReason';\nimport { RCCallMediaType } from './enums/RCCallMediaType';\nimport { RCCallMessageType } from './enums/RCCallMessageType';\nimport { RCCallSessionState } from './enums/RCCallSessionState';\nimport { RCCallUserState } from './enums/RCCallUserState';\nimport {\n  IExistedUserPofiles, IHungupMsgContent, IInviteMsgContent, IMemberModifyMsgContent, IRingingMsgContent,\n} from './interfaces/IMessageHandler';\nimport {\n  ICallStateMachineWatchers, IEndSummary, IInvitedUsers, IUserData, IUserProfile, IInviteOptions,\n} from './interfaces/IStateMachine';\nimport { CallMessageHandler } from './MessageHandler';\nimport { RCCrossCallType } from './enums/RCCrossCallType';\nimport { Timer } from './Timer';\nimport { IPushConfig } from '../../src/interface';\n\nexport class RCCallStateMachine {\n  /**\n   * 房间状态\n   */\n  private _sessionState: RCCallSessionState | null = null\n\n  /**\n   * 用户状态及信息\n   */\n  private _userInfo: { [userId: string]: IUserData } = {}\n\n  /**\n   * 用户计时器映射\n   */\n  private _userTimers: { [userId: string]: Timer } = {}\n\n  /**\n   * 监听器\n   */\n  private _watchers!: ICallStateMachineWatchers\n\n  /**\n   * 呼叫超时时间 (单位：毫秒)\n   */\n  private _callTimeout: number = 60 * 1000\n\n  /**\n   * 通话建立开始时间\n   */\n  private _beginTimestamp: number = 0\n\n  /**\n   * 通话结束时间\n   */\n  private _endTimestamp: number = 0\n\n  /**\n   * 通话结束原因\n   */\n  private _endReason: RCCallEndReason | null = null\n\n  /**\n   * 主叫 ID\n   * 发起邀请为当前用户 ID\n   * 收到邀请为 senderUserId\n   * 收到人员变更邀请为消息体中 callerId\n   */\n  private _callerId: string | null = null\n\n  /**\n   * 当次通话邀请者 ID\n   * 发起邀请为当前用户 ID、收到邀请为 senderUserId、收到人员变更邀请为消息体中 senderUserId\n   */\n  private _inviterId: string | null = null\n\n  /**\n   * 是否是跨 appkey\n   */\n  private _isCrossAppkey: boolean = false\n\n  private _hungupPushConfig?: IPushConfig\n\n  constructor(\n    private readonly _context: RTCPluginContext,\n    private readonly _runtime: IRuntime,\n    private readonly _logger: ILogger,\n    private readonly _callMsgHandler: CallMessageHandler,\n    private readonly _channelId: string,\n    private readonly _conversationType: ConversationType,\n    private readonly _targetId: string,\n    private _mediaType: RCCallMediaType,\n    private readonly _callId: string,\n  ) {\n    this._callMsgHandler.registerStateMachineEvent(this._callId, 'onRinging', this._onRinging.bind(this));\n    this._callMsgHandler.registerStateMachineEvent(this._callId, 'onAccept', this._onAccept.bind(this));\n    this._callMsgHandler.registerStateMachineEvent(this._callId, 'onMediaModify', this._onMediaModify.bind(this));\n    this._callMsgHandler.registerStateMachineEvent(this._callId, 'onHungup', this._onHungup.bind(this));\n  }\n\n  /**\n   * 获取校正后超时时间\n   */\n  private _getTimeout(sentTime: number): number {\n    let delayTime = this._context.getServerTime() - sentTime;\n    if (delayTime < 0) {\n      delayTime = 500; // 假设延迟时间为 500 ms\n    }\n    const timeout = this._callTimeout - delayTime;\n    this._logger.warn('_', `_getTimeout -> timeout: ${timeout}`);\n    return timeout;\n  }\n\n  private _clearTimerById(userId: string) {\n    this._logger.debug('_', `[RCCallStateMachine] before _clearTimerById  -> userId: ${userId} userTimers: ${JSON.stringify(this._userTimers)}`);\n    if (this._userTimers[userId]) {\n      this._userTimers[userId].stop();\n      delete this._userTimers[userId];\n    }\n    this._logger.debug('_', `[RCCallStateMachine] after _clearTimerById -> userTimers: ${JSON.stringify(this._userTimers)}`);\n  }\n\n  /**\n   * 通知 call 层房间状态变更及原因\n   */\n  private _notifyStateChange(state: RCCallSessionState, reason?: RCCallEndReason) {\n    this._logger.warn('_', `[RCCallStateMachine] notifyStateChange -> info: ${JSON.stringify({\n      state, reason,\n    })}`);\n\n    this._endReason = reason || null;\n    if (this._sessionState !== state) {\n      this._sessionState = state;\n      this._watchers?.onStateChange({ state, reason });\n    }\n    if (state === RCCallSessionState.END) {\n      // 当状态机结束时，需在 CallEngine 清除\n      eventEmitter.emit('onStateMachineClose', this._callId);\n      this._callMsgHandler.unregisterStateMachineEvent(this._callId);\n    }\n  }\n\n  /**\n   * 通知 call 层人员状态变更及原因\n   */\n  private _notifyUserStateChange(user: IUserData, reason?: RCCallEndReason) {\n    this._logger.warn('_', `[RCCallStateMachine] notifyUserStateChange -> info: ${JSON.stringify({\n      user, reason,\n    })}`);\n    this._watchers?.onUserStateChange({ user, reason });\n  }\n\n  private _otherClientHandle(message: IReceivedMessage) {\n    const { senderUserId, content: { user: userProfile, reason: hungupReason }, messageType } = message;\n    this._userInfo[senderUserId] = {\n      userId: senderUserId,\n      state: RCCallUserState.NONE,\n      isCaller: false,\n      isRemote: false,\n    };\n    // 多端接听、多端拒绝，清除收到邀请后起的计时器\n    for (const userId in this._userTimers) {\n      this._clearTimerById(userId);\n    }\n    // 其他端接听\n    let endReason = RCCallEndReason.ACCEPT_BY_OTHER_CLIENT;\n    if (messageType === RCCallMessageType.VCHangup) {\n      if (hungupReason === RCCallEndReason.BUSY_LINE) {\n        endReason = RCCallEndReason.OTHER_CLIENT_IN_CALL;\n      } else if (hungupReason === RCCallEndReason.NO_RESPONSE) {\n        endReason = RCCallEndReason.NO_RESPONSE;\n      } else {\n        endReason = RCCallEndReason.HANGUP_BY_OTHER_CLIENT;\n      }\n    }\n    // 添加用户简要信息\n    Object.assign(this._userInfo[senderUserId], <IUserProfile>userProfile);\n    this._notifyUserStateChange(this._userInfo[senderUserId], endReason);\n    this._notifyStateChange(RCCallSessionState.END, endReason);\n  }\n\n  /**\n   * 正在通话中，且不是当前已接通用户设备（deviceId）发来的消息\n  */\n  private _isRemoteInvalidMsg(remoteUserId: string, msgDeviceId: string): boolean {\n    if (!this._userInfo[remoteUserId]) {\n      return false;\n    }\n\n    if (!this._userInfo[remoteUserId].deviceId || !msgDeviceId) {\n      return false;\n    }\n\n    if ((this._userInfo[remoteUserId].state === RCCallUserState.KEEPING && this._userInfo[remoteUserId].deviceId !== msgDeviceId)) {\n      return true;\n    }\n\n    return false;\n  }\n\n  private _onRinging(message: IReceivedMessage) {\n    const { senderUserId: suid, content: { user: userProfile, deviceId } } = message;\n    const senderUserId = this._isCrossAppkey ? suid.split('_')[1] : suid;\n    // 正在通话中，远端多端发的消息直接过滤\n    if (this._isRemoteInvalidMsg(senderUserId, deviceId)) {\n      this._logger.debug('_', '[RCCallStateMachine] onRinging -> not the remote device that is currently talking');\n      return;\n    }\n\n    if (this._context.getCurrentId() === senderUserId) {\n      return; // 多端处理\n    }\n    this._watchers.onRinging({ userId: senderUserId, ...<IUserProfile>userProfile });\n  }\n\n  private _onAccept(message: IReceivedMessage) {\n    const { senderUserId: suid, content: { user: userProfile, deviceId: senderDeviceId }, sentTime } = message;\n    const senderUserId = this._isCrossAppkey ? suid.split('_')[1] : suid;\n    const currentUserId = this._context.getCurrentId();\n    // 正在通话中，远端多端发的消息直接过滤\n    if (this._isRemoteInvalidMsg(senderUserId, senderDeviceId)) {\n      this._logger.debug('_', '[RCCallStateMachine] _onAccept -> not the remote device that is currently talking');\n      return;\n    }\n\n    if (currentUserId === senderUserId) {\n      // 多端处理\n      this._otherClientHandle(message);\n      return;\n    }\n\n    // 群组通话时： A、B 通话 A邀请C, C同意接听，这时B没有C的userId对应的计时器，所以这里判断一下\n    if (this._userTimers[senderUserId]) {\n      // 清除呼叫超时计时器\n      this._clearTimerById(senderUserId);\n    }\n\n    // 修改并通知房间人员状态\n    const ids = (this._conversationType === ConversationType.PRIVATE) ? [currentUserId, senderUserId] : [senderUserId];\n    ids.forEach((userId) => {\n      const isCurrentUserId = userId === currentUserId;\n      this._userInfo[userId] = {\n        userId,\n        state: RCCallUserState.KEEPING,\n        isCaller: isCurrentUserId,\n        isRemote: isCurrentUserId,\n        deviceId: isCurrentUserId ? getCallDeviceId(this._runtime) : senderDeviceId,\n      };\n      if (!isCurrentUserId) {\n        this._beginTimestamp = Date.now();\n        // 添加用户简要信息\n        Object.assign(this._userInfo[senderUserId], <IUserProfile>userProfile);\n      }\n      this._notifyUserStateChange(this._userInfo[userId]);\n    });\n    if (this.getCallerId() === currentUserId) {\n      this._notifyStateChange(RCCallSessionState.KEEPING);\n    }\n    // 抛出onAccept时状态已经就绪\n    this._watchers.onAccept({ userId: senderUserId });\n  }\n\n  private _onMediaModify(message: IReceivedMessage) {\n    const { senderUserId, content: { mediaType, user: userProfile, deviceId } } = message;\n    // 正在通话中，远端多端发的消息直接过滤\n    if (this._isRemoteInvalidMsg(senderUserId, deviceId)) {\n      this._logger.debug('_', '[RCCallStateMachine] _onMediaModify -> not the remote device that is currently talking');\n      return;\n    }\n\n    if (this._context.getCurrentId() === senderUserId) { // 多端处理\n      return;\n    }\n\n    // 更新 mediaType\n    this._mediaType = mediaType;\n    this._watchers.onMediaModify({\n      sender: { userId: senderUserId, ...<IUserProfile>userProfile },\n      mediaType,\n    });\n  }\n\n  private _onHungup(message: IReceivedMessage) {\n    const { senderUserId: suid, content } = message;\n    const senderUserId = this._isCrossAppkey ? suid.split('_')[1] : suid;\n    const { reason, user: userProfile, deviceId } = content as IHungupMsgContent;\n    const currentUserId = this._context.getCurrentId();\n    // 正在通话中，远端多端发的消息直接过滤\n    if (this._isRemoteInvalidMsg(senderUserId, deviceId)) {\n      this._logger.debug('_', '[RCCallStateMachine] _onHungup -> not the remote device that is currently talking');\n      return;\n    }\n\n    if (currentUserId === senderUserId) { // 多端处理 抛出多端已处理 reason\n      this._otherClientHandle(message);\n      return;\n    }\n\n    if (this._sessionState === RCCallSessionState.END) {\n      // 如果己方房间状态已结束，再收到 hungup 消息不再处理\n      this._logger.info('_', `[RCCallStateMachine] Invalid hang up message, current room status has ended -> sessionState: ${this._sessionState}`);\n      return;\n    }\n\n    if (this._userInfo[senderUserId]) {\n      // 修改内存态数据并通知\n      this._userInfo[senderUserId].state = RCCallUserState.NONE;\n      this._endTimestamp = Date.now();\n      // 添加用户简要信息\n      Object.assign(this._userInfo[senderUserId], <IUserProfile>userProfile);\n      this._notifyUserStateChange(this._userInfo[senderUserId], CallRemoteEndReason[reason]);\n      delete this._userInfo[senderUserId];\n    }\n\n    // timer 清除\n    if (CallRemoteEndReason[reason] === RCCallEndReason.REMOTE_CANCEL) {\n      // 远端取消通话，没有远端用户，清除己端的接听超时计时器\n      this.getRemoteUserIds().length < 1 && this._clearTimerById(currentUserId);\n    } else if (this.getInviterId() === currentUserId) {\n      // 远端拒绝、忙碌、未接听等，通过自己是否是主叫来判断要清除呼叫超时计时器 或 接听超时计时器\n      this._clearTimerById(senderUserId);\n    } else {\n      this.getRemoteUserIds().length < 1 && this._clearTimerById(currentUserId);\n    }\n\n    // 房间人员信息少于两个，通知房间状态结束\n    const isLessThanTwo = Object.keys(this._userInfo).length < 2;\n    // 群呼中邀请者挂断（非呼叫发起者）且房间中被邀请者都未接听，通知房间状态结束\n    const isInviteUser = this._inviterId === senderUserId;\n    const isNoOneAnswered = Object.values(this._userInfo).every((user) => user.state !== RCCallUserState.KEEPING);\n    if (isLessThanTwo || (isInviteUser && isNoOneAnswered)) {\n      this._notifyStateChange(RCCallSessionState.END, CallRemoteEndReason[reason]);\n    }\n\n    // 抛出 onHungup 时状态已经就绪\n    this._watchers.onHungup({ userId: senderUserId, ...<IUserProfile>userProfile }, CallRemoteEndReason[reason]);\n  }\n\n  /**\n   * 注册事件监听\n   * @params watchers\n   */\n  registerEventListener(watchers: ICallStateMachineWatchers) {\n    this._watchers = watchers;\n  }\n\n  /**\n   * 收到 invite 消息时状态机更新（CallEngine 内部调用）\n   * @param message 接收消息\n   */\n  __onInvite(message: IReceivedMessage) {\n    const { senderUserId: suid, content, sentTime } = message;\n    const {\n      inviteUserIds: inids, user: userProfile, deviceId, roomType,\n    } = content as IInviteMsgContent;\n    // 处理跨 appkey 邀请 senderUserId 带有 appkey 与本地 currentUserId 匹配问题\n    // 同时记录本地是否为跨 appkey 状态\n    let senderUserId: string;\n    if (roomType === RCCrossCallType.RCCallRoomTypeAcrossCall) {\n      [, senderUserId] = suid.split('_');\n      this._watchers.crossAppkey(true);\n      this._isCrossAppkey = true;\n    } else {\n      senderUserId = suid;\n    }\n    const currentUserId = this._context.getCurrentId();\n    // 正在通话中，远端多端发的消息直接过滤\n    if (this._isRemoteInvalidMsg(senderUserId, deviceId)) {\n      this._logger.debug('_', '[RCCallStateMachine] __onInvite -> not the remote device that is currently talking');\n      return;\n    }\n\n    if (currentUserId === senderUserId) {\n      // 多端处理\n      return;\n    }\n\n    this._callerId = this._inviterId = senderUserId;\n    const userIdList = [suid, ...inids];\n    // 收到邀请后，内部回应响铃消息, userIds 传除自己的所有人\n    this._callMsgHandler.sendRinging({\n      conversationType: this._conversationType,\n      targetId: this._targetId,\n      channelId: this._channelId,\n      callId: this._callId,\n      userIds: userIdList.filter((id) => {\n        if (this._isCrossAppkey) {\n          return id.split('_')[1] !== currentUserId;\n        }\n        return id !== currentUserId;\n      }),\n    });\n\n    const inviteUserIds = this._isCrossAppkey ? [inids[0].split('_')[1]] : inids;\n\n    // 同步本地用户列表需要使用拆分后的 UserId 保证本地其他方法匹配 userid\n    const allUserIds = [senderUserId, ...inviteUserIds];\n\n    // 修改并通知房间人员状态\n    allUserIds.forEach((userId) => {\n      this._userInfo[userId] = {\n        userId,\n        state: RCCallUserState.WAITING,\n        isCaller: userId === senderUserId,\n        isRemote: userId !== currentUserId,\n      };\n      if (userId === senderUserId) {\n        // 给 senderUser 添加用户简要信息及 deviceId\n        Object.assign(this._userInfo[userId], <IUserProfile>userProfile, { deviceId });\n      }\n      this._notifyUserStateChange(this._userInfo[userId]);\n      // 给所有被邀请人启动接听计时器\n      if (userId !== senderUserId) {\n        this._userTimers[userId] = new Timer(() => {\n          const reason = userId === currentUserId ? RCCallEndReason.NO_RESPONSE : RCCallEndReason.REMOTE_NO_RESPONSE;\n          if (userId === currentUserId) { // 群聊中己方超时需发送挂断\n            this._hungupHandle(reason);\n          } else { // 其他人员超时只通知人员状态变更\n            this._userInfo[userId] && (this._userInfo[userId].state = RCCallUserState.NONE);\n            this._notifyUserStateChange(this._userInfo[userId]);\n            this._watchers.onHungup(this._userInfo[userId], reason);\n            delete this._userInfo[userId];\n          }\n        }, this._getTimeout(sentTime));\n      }\n    });\n    // 房间状态通知\n    this._notifyStateChange(RCCallSessionState.WAITING);\n  }\n\n  /**\n   * 收到 memberModify 消息时状态机更新（CallEngine 内部调用）\n   * @param message 接收消息\n   */\n  __onMemberModify(message: IReceivedMessage) {\n    const { senderUserId, content, sentTime } = message;\n    const {\n      user: userProfile, existedUserPofiles, caller, deviceId, inviteUserIds, mediaType,\n    } = content as IMemberModifyMsgContent;\n    const currentUserId = this._context.getCurrentId();\n    // 正在通话中，远端多端发的消息直接过滤\n    // this._userInfo[senderUserId] && !this._userInfo[senderUserId].deviceId && !deviceId &&\n    if (this._isRemoteInvalidMsg(senderUserId, deviceId)) {\n      this._logger.debug('_', '[RCCallStateMachine] __onMemberModify -> not the remote device that is currently talking');\n      return;\n    }\n    if (currentUserId === senderUserId) {\n      // 多端处理\n      return;\n    }\n    this._callerId = caller;\n    this._inviterId = senderUserId;\n    inviteUserIds.forEach((id) => {\n      existedUserPofiles.push({ userId: id, mediaType, callStatus: MsgCallStatus.INCOMING });\n    });\n    // 己方状态为 NONE (未存在己方用户信息) 说明自己为被邀请者, 需发送 ringing\n    const isNewInvitedUser = inviteUserIds.includes(currentUserId);\n    if (isNewInvitedUser) {\n      const needRingingUserIds: string[] = [];\n      existedUserPofiles.forEach((userInfo) => {\n        if (userInfo.userId !== currentUserId) {\n          needRingingUserIds.push(userInfo.userId);\n        }\n      });\n      this._callMsgHandler.sendRinging({\n        conversationType: this._conversationType,\n        targetId: this._targetId,\n        channelId: this._channelId,\n        callId: this._callId,\n        userIds: needRingingUserIds,\n      });\n\n      // 被邀请者通知 call 层房间状态变更\n      this._notifyStateChange(RCCallSessionState.WAITING);\n    } else {\n      // 已在通话流程中用户，抛出人员变更监听\n      this._watchers.onMemberModify({\n        sender: { userId: senderUserId, ...userProfile },\n        invitedUsers: inviteUserIds.map((id) => ({ userId: id })),\n      });\n    }\n\n    // 更新全量用户状态\n    existedUserPofiles.forEach((userInfo) => {\n      const { userId, callStatus } = userInfo;\n      // 人员为结束状态时，不再更新\n      if (callStatus === MsgCallStatus.IDLE) return;\n      this._userInfo[userId] = {\n        userId,\n        state: callStatus !== MsgCallStatus.CONNECTED ? RCCallUserState.WAITING : RCCallUserState.KEEPING,\n        isCaller: senderUserId === userId,\n        isRemote: currentUserId !== userId,\n      };\n      // 给 senderUser 添加用户简要信息 及 deviceId\n      if (userId === senderUserId) {\n        Object.assign(this._userInfo[userId], <IUserProfile>userProfile, { deviceId });\n      }\n      // 通知人员变更\n      this._notifyUserStateChange(this._userInfo[userId]);\n      // 除了 callMsgStatus.connected ,其他人都起计时器\n\n      if (callStatus !== MsgCallStatus.CONNECTED && !this._userTimers[userId]) {\n        /**\n         * 仅给被邀请成员启动定时器\n         */\n        if (!inviteUserIds.includes(userId)) {\n          return;\n        }\n        // 启动计时器\n        this._userTimers[userId] = new Timer(() => {\n          // 呼叫超时处理状态、并通知 call 层\n          this._userInfo[userId] && (this._userInfo[userId].state = RCCallUserState.NONE);\n          const reason = userId === currentUserId ? RCCallEndReason.NO_RESPONSE : RCCallEndReason.REMOTE_NO_RESPONSE;\n          // 通知 call 层人员状态变更\n          this._notifyUserStateChange(this._userInfo[userId], reason);\n          try {\n            // 呼叫超时未接抛出onHungup\n            this._watchers.onHungup(this._userInfo[userId], reason);\n          } catch (error: any) {\n            this._logger.error('_', `[RCCallStateMachine] call onhungup error -> ${error?.stack}`);\n          }\n\n          delete this._userInfo[userId];\n          // 房间人员信息少于两个 或 未接听的是自己，通知房间状态结束\n          if (Object.keys(this._userInfo).length < 2 || userId === currentUserId) {\n            this._notifyStateChange(RCCallSessionState.END, reason);\n          }\n        }, this._getTimeout(sentTime));\n      }\n    });\n  }\n\n  /**\n   * 处理已有 session ，不允许再接听新会话情况\n   */\n  __handleInviteInSession() {\n    this._logger.info('_', 'StateMachine -> __handleInviteInSession');\n    // 修改所有用户状态并抛出，清空计时器\n    for (const userId in this._userInfo) {\n      this._userInfo[userId].state && (this._userInfo[userId].state = RCCallUserState.NONE);\n      this._notifyUserStateChange(this._userInfo[userId]);\n      // 将已存在的计时器都清掉\n      this._clearTimerById(userId);\n    }\n    // 直接终止当前 session 状态\n    this._notifyStateChange(RCCallSessionState.END, RCCallEndReason.BUSY_LINE);\n    // 发送挂断消息，并携带 己方忙碌 原因\n    this._callMsgHandler.sendHungup({\n      channelId: this._channelId,\n      conversationType: this._conversationType,\n      targetId: this._targetId,\n      callId: this._callId,\n      reason: RCCallEndReason.BUSY_LINE,\n      userIds: this.getRemoteUserIds(),\n    });\n  }\n\n  /**\n   * 主动呼叫 (CallEngine 内部调用)\n   * @param userIds 被邀请用户 ID 列表\n   * @param extra 消息的扩展信息\n   * @param pushConfig 移动端推送配置\n   */\n  async __call(userIds: string[], extra: string = '', pushConfig: IPushConfig, isCrossAppkey = false) {\n    this._logger.debug('_', `[RCCallStateMachine] invite -> userIds: ${JSON.stringify(userIds)}`);\n    // 发送邀请消息\n    const currentUserId = this._callerId = this._inviterId = this._context.getCurrentId();\n    const { code, message } = await this._callMsgHandler.sendInvite({\n      roomType: isCrossAppkey ? RCCrossCallType.RCCallRoomTypeAcrossCall : RCCrossCallType.RCCallRoomTypeNormalCall,\n      channelId: this._channelId,\n      conversationType: this._conversationType,\n      targetId: this._targetId,\n      callId: this._callId,\n      extra,\n      pushConfig,\n      mediaType: this._mediaType,\n      inviteUserIds: userIds.filter((id) => id !== currentUserId),\n    });\n\n    this._isCrossAppkey = isCrossAppkey;\n\n    if (code === RCCallErrorCode.SUCCESS) {\n      const { sentTime } = message!;\n\n      /**\n       * 跨 appkey 仅存在单聊，对端 userId 需取下划线后面的部分\n       */\n      const ids = isCrossAppkey ? [currentUserId, ...[userIds[0].split('_')[1]]] : [currentUserId, ...userIds];\n\n      // 遍历更新房间所有人员状态、并给被叫启动计时器\n      ids.forEach((userId) => {\n        const isCaller = userId === currentUserId;\n        this._userInfo[userId] = {\n          userId,\n          state: RCCallUserState.WAITING,\n          isCaller,\n          isRemote: !isCaller,\n        };\n        // 通知 call 层人员状态变更\n        this._notifyUserStateChange(this._userInfo[userId]);\n\n        // 启动呼叫超时计时器\n        if (!isCaller) {\n          this._userTimers[userId] = new Timer(() => {\n            // 呼叫超时处理状态、并通知 call 层\n            this._userInfo[userId] && (this._userInfo[userId].state = RCCallUserState.NONE);\n            // 通知 call 层人员状态变更\n            this._notifyUserStateChange(this._userInfo[userId], RCCallEndReason.REMOTE_NO_RESPONSE);\n\n            // 呼叫超时未接抛出onHungup\n            this._watchers.onHungup(this._userInfo[userId], RCCallEndReason.REMOTE_NO_RESPONSE);\n\n            delete this._userInfo[userId];\n            // 房间人员信息少于两个，通知房间状态\n            if (Object.keys(this._userInfo).length < 2) {\n              this._notifyStateChange(RCCallSessionState.END, RCCallEndReason.REMOTE_NO_RESPONSE);\n            }\n            // 远端人员大于 1 且己方不为正在通话时，发送 hungup, 告诉远端自己退出通话\n            const isNeedSendHungup = this.getRemoteUserIds().length === 0 && (this._userInfo[currentUserId].state !== RCCallUserState.KEEPING);\n            if (isNeedSendHungup) {\n              this._hungupHandle(RCCallEndReason.REMOTE_NO_RESPONSE);\n            }\n          }, this._getTimeout(sentTime));\n        }\n      });\n\n      // 通知 call 层房间状态变更\n      this._notifyStateChange(RCCallSessionState.WAITING);\n    } else {\n      const endCode = code === RCCallErrorCode.REJECTED_BY_BLACKLIST ? RCCallEndReason.ADDED_TO_BLACKLIST : RCCallEndReason.NETWORK_ERROR;\n      this._notifyStateChange(RCCallSessionState.END, endCode);\n    }\n\n    return { code };\n  }\n\n  /**\n   * 接听\n   */\n  async accept(): Promise<{ code: RCCallErrorCode }> {\n    this._logger.debug('_', '[RCCallStateMachine] accept');\n\n    // 发送接听消息\n    const currentUserId = this._context.getCurrentId();\n    const { code, message } = await this._callMsgHandler.sendAccept({\n      channelId: this._channelId,\n      conversationType: this._conversationType,\n      targetId: this._targetId,\n      callId: this._callId,\n      mediaType: this._mediaType,\n      userIds: this.getRemoteUserIds(),\n    });\n\n    // 清除接听超时计时器\n    this._clearTimerById(currentUserId);\n    if (code === RCCallErrorCode.SUCCESS) {\n      const { sentTime } = message!;\n      this._userInfo[currentUserId] && (this._userInfo[currentUserId].state = RCCallUserState.KEEPING);\n      this._beginTimestamp = Date.now();\n      // 通知 call 层人员状态变更\n      this._notifyUserStateChange(this._userInfo[currentUserId]);\n      // 修改并通知 call 层房间状态变更\n      this._notifyStateChange(RCCallSessionState.KEEPING);\n    } else {\n      this._userInfo[currentUserId] && (this._userInfo[currentUserId].state = RCCallUserState.NONE);\n      // 通知 call 层人员状态变更\n      this._notifyUserStateChange(this._userInfo[currentUserId]);\n      // 修改并通知 call 层房间状态变更\n      const endReason = code === RCCallErrorCode.REJECTED_BY_BLACKLIST ? RCCallEndReason.ADDED_TO_BLACKLIST : RCCallEndReason.NETWORK_ERROR;\n      this._notifyStateChange(RCCallSessionState.END, endReason);\n    }\n\n    return { code };\n  }\n\n  /**\n   * 群呼叫中继续邀请\n   * @param userIds 被邀请用户 ID 列表\n   */\n  async invite(userIds: string[], options: IInviteOptions = {}): Promise<{ code: RCCallErrorCode }> {\n    if (this._conversationType !== ConversationType.GROUP) {\n      return { code: RCCallErrorCode.CONVERSATION_NOT_GROUP_ERROR };\n    }\n\n    this._logger.debug('_', `[RCCallStateMachine] invite -> userIds: ${JSON.stringify(userIds)}`);\n    const currentUserId = this._context.getCurrentId();\n    const existedUserIds = Object.keys(this._userInfo);\n    const existedUserPofiles: IExistedUserPofiles[] = existedUserIds.map((userId) => {\n      let callStatus = MsgCallStatus.CONNECTED;\n      // 包含被邀请者，或者用户状态为 wating 时，状态为 响铃\n      if (userIds.includes(userId) || this._userInfo[userId].state === RCCallUserState.WAITING) {\n        callStatus = MsgCallStatus.RINGING;\n      }\n      return {\n        userId,\n        mediaType: this._mediaType,\n        callStatus,\n        mediaId: userId,\n      };\n    });\n\n    const { extra = '', pushConfig } = options;\n\n    // 发送人员变更消息\n    const { code, message } = await this._callMsgHandler.sendMemeberModify({\n      channelId: this._channelId,\n      conversationType: this._conversationType,\n      targetId: this._targetId,\n      callId: this._callId,\n      extra,\n      pushConfig,\n      mediaType: this._mediaType,\n      // 除自己，其他所有人均需收到 modify 消息，以此更新状态\n      inviteUserIds: userIds.filter((id) => id !== currentUserId),\n      callerId: this.getCallerId(),\n      existedUserPofiles,\n      directionalUserIdList: [...existedUserIds, ...userIds].filter((id) => id !== currentUserId),\n    });\n\n    if (code === RCCallErrorCode.SUCCESS) {\n      const { sentTime } = message!;\n      // 修改被邀请人状态，并通知\n      userIds.forEach((userId) => {\n        this._userInfo[userId] = {\n          userId,\n          state: RCCallUserState.WAITING,\n          isCaller: false,\n          isRemote: true,\n        };\n        // 通知 call 层人员状态变更\n        this._notifyUserStateChange(this._userInfo[userId]);\n        // 启动呼叫计时器\n        this._userTimers[userId] = new Timer(() => {\n          // 呼叫超时处理状态、并通知 call 层\n          this._userInfo[userId] && (this._userInfo[userId].state = RCCallUserState.NONE);\n          // 通知 call 层人员状态变更\n          this._notifyUserStateChange(this._userInfo[userId], RCCallEndReason.REMOTE_NO_RESPONSE);\n\n          // 呼叫超时未接抛出onHungup\n          this._watchers.onHungup(this._userInfo[userId], RCCallEndReason.REMOTE_NO_RESPONSE);\n\n          delete this._userInfo[userId];\n          // 房间人员信息少于两个，通知房间状态\n          if (Object.keys(this._userInfo).length < 2) {\n            this._notifyStateChange(RCCallSessionState.END, RCCallEndReason.REMOTE_NO_RESPONSE);\n          }\n        }, this._getTimeout(sentTime));\n      });\n    } else {\n      // 通知人员变更\n      userIds.forEach((userId) => {\n        this._userInfo[userId] = {\n          userId,\n          state: RCCallUserState.NONE,\n          isCaller: false,\n          isRemote: true,\n        };\n        // 通知 call 层人员状态变更\n\n        const endReason = code === RCCallErrorCode.REJECTED_BY_BLACKLIST ? RCCallEndReason.ADDED_TO_BLACKLIST : RCCallEndReason.NETWORK_ERROR;\n        this._notifyUserStateChange(this._userInfo[userId], endReason);\n      });\n    }\n\n    return { code };\n  }\n\n  private async _hungupHandle(reason: RCCallEndReason): Promise<{ code: RCCallErrorCode }> {\n    const currentUserId = this._context.getCurrentId();\n    // 发送挂断类型消息\n    let code = RCCallErrorCode.SUCCESS;\n\n    const hangupParams = {\n      channelId: this._channelId,\n      conversationType: this._conversationType,\n      targetId: this._targetId,\n      callId: this._callId,\n      reason,\n      userIds: this.getRemoteUserIds(),\n      pushConfig: this._hungupPushConfig,\n    };\n\n    if (reason === RCCallEndReason.OTHER_CLIENT_JOINED_CALL) {\n      // 己端被多端加入 RTC 房间挤掉后，不再等发挂断消息的结果，防止走 userLeave 导致挂断原因不准确\n      this._callMsgHandler.sendHungup(hangupParams);\n    } else {\n      const { code: msgCode } = await this._callMsgHandler.sendHungup(hangupParams);\n      code = msgCode;\n    }\n\n    this._endTimestamp = Date.now();\n    // 更新通话人员状态、通话结束时间 并通知\n    for (const userId in this._userInfo) {\n      this._userInfo[userId].state = RCCallUserState.NONE;\n      if (userId === currentUserId) {\n        this._notifyUserStateChange(this._userInfo[userId], reason);\n      } else {\n        this._notifyUserStateChange(this._userInfo[userId]);\n      }\n      delete this._userInfo[userId];\n    }\n\n    if (Object.keys(this._userInfo).length < 2) {\n      // 清空内存态数据，并通知\n      this._notifyStateChange(RCCallSessionState.END, reason);\n    }\n\n    return { code };\n  }\n\n  /**\n   * 挂断\n   */\n  async hungup(): Promise<{ code: RCCallErrorCode }> {\n    this._logger.debug('_', '[RCCallStateMachine] hungup');\n    const currentUserId = this._context.getCurrentId();\n    // 默认挂断 reason 为通话过程中，己方正常取消通话\n    let reason = RCCallEndReason.HANGUP;\n    /**\n     * 若超时计时器存在, 己方为 caller 原因为己方取消通话,\n     * 己方为 callee 且未进入通话中，原因为己方拒绝通话\n     */\n    if (Object.keys(this._userTimers).length > 0) {\n      if (this._userInfo[currentUserId].isCaller) {\n        reason = RCCallEndReason.CANCEL;\n      } else if (this._userInfo[currentUserId].state === RCCallUserState.WAITING) {\n        reason = RCCallEndReason.REJECT;\n      }\n    }\n\n    // 清除所有超时计时器\n    for (const userId in this._userTimers) {\n      this._clearTimerById(userId);\n    }\n    return this._hungupHandle(reason);\n  }\n\n  /**\n   * 修改通话媒体类型\n   * @param mediaType RCCallMediaType.AUDIO 改为音频通话 | RCCallMediaType.AUDIO_VIDEO 改为音视频通话\n   */\n  async changeMediaType(mediaType: RCCallMediaType): Promise<{ code: RCCallErrorCode }> {\n    this._logger.debug('_', `[RCCallStateMachine] changeMediaType -> mediaType: ${mediaType}`);\n    const { code } = await this._callMsgHandler.sendMediaModify({\n      channelId: this._channelId,\n      conversationType: this._conversationType,\n      targetId: this._targetId,\n      callId: this._callId,\n      mediaType,\n      userIds: this.getRemoteUserIds(),\n    });\n    if (code === RCCallErrorCode.SUCCESS) {\n      this._mediaType = mediaType;\n    }\n    return { code };\n  }\n\n  /**\n   * 用户加入通话补偿机制（rtc userJoin 事件触发）\n   * 主叫呼叫后，未收到被叫 accept 消息，但收到了 userJoin 同样补偿更新用户、房间状态、呼叫计时器\n   */\n  userJoin(userIds: string[]) {\n    this._logger.debug('_', `[RCCallStateMachine] userJoin -> userIds: ${JSON.stringify(userIds)}`);\n    // 延迟 300ms 防止 userJion 和 accept 消息都有的情况下， userJoin 比 accept 先到\n    setTimeout(() => {\n      userIds.forEach((userId) => {\n        const userInfo = this._userInfo[userId];\n        // 更新人员状态 (// 群组通话时： A向B、C发起通话 B先接听，C后接听，这时B没有C的userId对应的_userInfo，所以这里判断一下)\n        if (userInfo && userInfo.state !== RCCallUserState.KEEPING) {\n          userInfo.state = RCCallUserState.KEEPING;\n          this._notifyUserStateChange(userInfo);\n        }\n        // 更新房间状态\n        if (this._sessionState !== RCCallSessionState.KEEPING) {\n          this._notifyStateChange(RCCallSessionState.KEEPING);\n        }\n        // 停止并清除呼叫计时器\n        this._clearTimerById(userId);\n      });\n    }, 300);\n  }\n\n  /**\n   * 用户离开通话补偿机制（rtc userLeave、kickOff 事件触发）\n   * 通话中远端用户挂断，挂断消息未到，但是监听到 rtc userLeave 同样补偿更新用户、房间状态\n   */\n  userLeave(userIds: string[]) {\n    this._logger.debug('_', `[RCCallStateMachine] userLeave -> userIds: ${JSON.stringify(userIds)}`);\n    // 延迟 300ms 防止 userLeave 和 hungup 消息都有的情况下，userLeave 比 hungup 先到\n    setTimeout(() => {\n      userIds.forEach((userId) => {\n        const userInfo = this._userInfo[userId];\n        // 更新人员状态\n        if (userInfo && userInfo.state !== RCCallUserState.NONE) {\n          userInfo.state = RCCallUserState.NONE;\n          this._notifyUserStateChange(userInfo, RCCallEndReason.REMOTE_HANGUP);\n          this._watchers.onHungup(userInfo, RCCallEndReason.REMOTE_HANGUP);\n          delete this._userInfo[userId];\n        }\n        // 主叫方群聊邀请被叫方后离线，主叫方此时为等待状态后不再上线，被叫方接听后挂断至剩余一人时等待一分钟后挂断\n        const remoteUsersTimer = new Timer(() => {\n          const remoteUsers = this.getRemoteUsers();\n          // 远端只有一个用户且是等待状态，等待一分钟后挂断\n          if (remoteUsers.length === 1 && remoteUsers[0].state === 1) {\n            this._hungupHandle(RCCallEndReason.REMOTE_NETWORK_ERROR);\n          }\n        }, 60000);\n        // 更新房间状态\n        if (Object.keys(this._userInfo).length < 2 && this._sessionState !== RCCallSessionState.END) {\n          this._endTimestamp = Date.now();\n          this._notifyStateChange(RCCallSessionState.END, RCCallEndReason.REMOTE_HANGUP);\n        }\n      });\n    }, 300);\n  }\n\n  /**\n   * Call 层己方异常失败后调用的方法\n   * 触发时机：音视频服务异常、获取资源失败、加入 RTC 房间失败、发布|订阅失败\n   */\n  close(reason: RCCallEndReason) {\n    this._hungupHandle(reason);\n  }\n\n  setHungupPushConfig(pushConfig: IPushConfig) {\n    this._hungupPushConfig = pushConfig;\n  }\n\n  /**\n   * 通话唯一标识\n   */\n  getCallId(): string {\n    return this._callId;\n  }\n\n  /**\n   * 多组织 ID\n   */\n  getChannelId(): string {\n    return this._channelId;\n  }\n\n  /**\n   * 目标 ID，单呼对方人员 Id, 群呼群组 Id\n   */\n  getTargetId(): string {\n    return this._targetId;\n  }\n\n  /**\n   * 获取会话类型\n   */\n  getConversationType(): ConversationType {\n    return this._conversationType;\n  }\n\n  /**\n   * 获取远端成员 ID 列表\n   */\n  getRemoteUserIds(): string[] {\n    const allUserIds = Object.keys(this._userInfo);\n    const remoteUserIds = allUserIds.filter((id) => this._context.getCurrentId() !== id);\n    return remoteUserIds;\n  }\n\n  /**\n   * 获取远端成员信息列表\n   */\n  getRemoteUsers(): IUserData[] {\n    const remoteUser: IUserData[] = [];\n    const currentUserId = this._context.getCurrentId();\n    for (const uid in this._userInfo) {\n      const { userId } = this._userInfo[uid];\n      if (userId !== currentUserId) {\n        remoteUser.push(this._userInfo[uid]);\n      }\n    }\n    return remoteUser;\n  }\n\n  /**\n   * 获取房间状态\n   */\n  getState(): RCCallSessionState {\n    return this._sessionState === null ? RCCallSessionState.END : this._sessionState;\n  }\n\n  /**\n   * 获取人员状态\n   */\n  getUserState(userId: string): RCCallUserState {\n    return this._userInfo[userId]?.state;\n  }\n\n  /**\n   * 获取会话发起者 Id\n   */\n  getCallerId(): string {\n    return this._callerId!;\n  }\n\n  /**\n   * 获取当次会话邀请者 Id\n   */\n  getInviterId(): string {\n    return this._inviterId!;\n  }\n\n  /**\n   * 获取当前通话媒体类型\n   */\n  getMediaType(): RCCallMediaType {\n    return this._mediaType;\n  }\n\n  /**\n   * 通话挂断后可调用\n   */\n  getSummary(): IEndSummary {\n    // 通话时间计算\n    const beginTimestamp = this._beginTimestamp;\n    const endTimestamp = this._endTimestamp;\n    let duration = 0;\n    if (endTimestamp > beginTimestamp && beginTimestamp !== 0) {\n      duration = endTimestamp - beginTimestamp;\n    }\n\n    const summary = {\n      conversationType: this._conversationType,\n      channelId: this._channelId,\n      targetId: this._targetId,\n      mediaType: this._mediaType,\n      beginTimestamp,\n      endTimestamp,\n      duration,\n      endReason: this._endReason!,\n    };\n\n    this._logger.debug('_', `[RCCallStateMachine] getSummary -> summary: ${JSON.stringify(summary)}`);\n    return summary;\n  }\n}\n","/**\n * 通话媒体类型\n */\nexport enum RCCallMediaType {\n  /**\n   * 音频通话\n   */\n  AUDIO = 1,\n  /**\n   * 视频通话\n   */\n  AUDIO_VIDEO\n}\n","export enum MemberModifyType {\n  ADD = 1,\n  REMOVE\n}\n","/**\n * 平台\n */\nexport enum Platform {\n  WEB = 'Web',\n  IOS = 'iOS',\n  ANDROID = 'Android'\n}\n","import {\n  ConversationType, ILogger, IReceivedMessage, RTCPluginContext,\n} from '@rongcloud/engine';\nimport { CallRemoteEndReason, RCCallEndReason } from './enums/RCCallEndReason';\nimport { RCCallMediaType } from './enums/RCCallMediaType';\nimport { RCCallMessageType } from './enums/RCCallMessageType';\nimport { IHungupMsgContent, IInviteMsgContent, IMediaModifyMsgContent } from './interfaces/IMessageHandler';\n\nexport interface IOfflineRecord {\n  channelId: string,\n  conversationType: ConversationType,\n  targetId: string,\n  mediaType: RCCallMediaType,\n  callId: string,\n  inviterId: string,\n  endReason: RCCallEndReason,\n  beginTimestamp: number,\n  endTimestamp: number,\n  duration: number\n}\n\n/**\n * 离线通话记录器\n */\nexport class OfflineRecorder {\n  private _messages: IReceivedMessage[] | [] = []\n\n  private _channelId!: string\n\n  private _conversationType!: ConversationType\n\n  private _targetId!: string\n\n  private _mediaType!: RCCallMediaType\n\n  private _callId!: string\n\n  private _callerId!: string\n\n  private _inviterId!: string\n\n  private _endReason!: RCCallEndReason\n\n  private _beginTimestamp: number = 0\n\n  private _endTimestamp: number = 0\n\n  constructor(\n    private readonly _context: RTCPluginContext,\n    private readonly _logger: ILogger,\n    private readonly _onRecord: (record: IOfflineRecord) => void,\n  ) {\n\n  }\n\n  /**\n   * 到 invite | memberModify 结束\n   * 原因根据己方是否为主叫，主叫为远端未接听 被叫被己端未接听\n   */\n  private _doInvite(message: IReceivedMessage) {\n    const currentUserId = this._context.getCurrentId();\n    const {\n      channelId, conversationType, targetId, senderUserId, content,\n    } = message;\n    const { callId, mediaType } = content as IInviteMsgContent;\n    this._channelId = channelId!;\n    this._conversationType = conversationType;\n    this._targetId = targetId;\n    this._callId = callId;\n    this._mediaType = mediaType;\n    const isCaller = senderUserId === currentUserId;\n    this._inviterId = senderUserId;\n    this._endReason = isCaller ? RCCallEndReason.REMOTE_NO_RESPONSE : RCCallEndReason.REMOTE_NO_RESPONSE;\n    this._canGenRecord();\n  }\n\n  /**\n   * 到 invite | memberModify 结束\n   * 原因根据己方是否为主叫，主叫为远端未接听 被叫被己端未接听\n   */\n  private _doMemberModify(message: IReceivedMessage) {\n    this._doInvite(message);\n  }\n\n  /**\n   * 用 invite | memberModify 计算的离线记录\n   */\n  private _doRinging(message: IReceivedMessage) {\n    this._canGenRecord();\n  }\n\n  /**\n   * 到 accept 说明通话已建立\n   * 原因默认己方正常挂断\n   */\n  private _doAccept(message: IReceivedMessage) {\n    this._endReason = RCCallEndReason.HANGUP;\n    this._beginTimestamp = message.sentTime;\n    this._canGenRecord();\n  }\n\n  /**\n   * 到 hungup 说明为正常挂断\n   * 原因取消息体里挂断原因\n   */\n  private _doHungup(message: IReceivedMessage) {\n    const { content, sentTime, senderUserId } = message;\n    const { reason } = content as IHungupMsgContent;\n    const currentUserId = this._context.getCurrentId();\n    const isSelfSend = senderUserId === currentUserId;\n    this._endReason = isSelfSend ? reason : CallRemoteEndReason[reason];\n    this._endTimestamp = sentTime;\n\n    this._canGenRecord();\n  }\n\n  /**\n   * 只修改通话类型\n   */\n  private _doMediaModify(message: IReceivedMessage) {\n    const { content } = message;\n    const { mediaType } = content as IMediaModifyMsgContent;\n    this._mediaType = mediaType;\n    this._canGenRecord();\n  }\n\n  private _canGenRecord() {\n    if (this._messages.length === 0) {\n      let duration = 0;\n      const isNeedDurationReason = [\n        RCCallEndReason.HANGUP,\n        RCCallEndReason.REMOTE_HANGUP,\n        RCCallEndReason.OTHER_CLIENT_JOINED_CALL,\n        RCCallEndReason.REMOTE_OTHER_CLIENT_JOINED_CALL,\n        RCCallEndReason.KICKED_BY_SERVER,\n        RCCallEndReason.REMOTE_KICKED_BY_SERVER,\n        RCCallEndReason.ACCEPT_SYSTEM_CALL,\n        RCCallEndReason.REMOTE_ACCEPT_SYSTEM_CALL,\n      ].includes(this._endReason);\n      if (isNeedDurationReason) {\n        duration = this._endTimestamp - this._beginTimestamp;\n      }\n      this._onRecord({\n        channelId: this._channelId,\n        conversationType: this._conversationType,\n        targetId: this._targetId,\n        callId: this._callId,\n        inviterId: this._inviterId,\n        mediaType: this._mediaType,\n        endReason: this._endReason,\n        beginTimestamp: this._beginTimestamp,\n        endTimestamp: this._endTimestamp,\n        duration,\n      });\n    }\n  }\n\n  onRecvOfflineMsgs(messages: IReceivedMessage[]) {\n    this._messages = messages;\n    do {\n      const msg = this._messages.shift()!;\n      const { messageType, content: { callId } } = msg;\n      switch (messageType) {\n        case RCCallMessageType.VCInvite:\n          this._doInvite(msg);\n          break;\n        case RCCallMessageType.VCRinging:\n          this._doRinging(msg);\n          break;\n        case RCCallMessageType.VCAccept:\n          this._doAccept(msg);\n          break;\n        case RCCallMessageType.VCModifyMem:\n          this._doMemberModify(msg);\n          break;\n        case RCCallMessageType.VCModifyMedia:\n          this._doMediaModify(msg);\n          break;\n        case RCCallMessageType.VCHangup:\n          this._doHungup(msg);\n          break;\n        default:\n          this._logger.debug('_', `[OfflineRecorder] onRecvOfflineMsgs -> unexpected message: ${JSON.stringify(msg)}`);\n          break;\n      }\n    } while (this._messages.length > 0);\n  }\n}\n","import {\n  ConversationType, ErrorCode, EventEmitter, ILogger, IPushConfig as IEnginePushConfig, IReceivedMessage, IRuntime, ISendMsgOptions, RTCPluginContext,\n} from '@rongcloud/engine';\nimport { MemberModifyType } from './enums/MemberModifyType';\nimport { RCCallErrorCode } from './enums/RCCallCode';\nimport { RCCallMessageType } from './enums/RCCallMessageType';\nimport { IUserProfile } from './interfaces/IStateMachine';\nimport {\n  IAcceptMsgContent, IAcceptMsgOptions, ICallMsgOption, IHungupMsgContent, IHungupMsgOptions,\n  IInviteMsgContent, IInviteMsgOptions, IMediaModifyMsgContent, IMediaModifyMsgOptions, IMemberModifyMsgContent,\n  IMemberModifyMsgOptions, IMsgListener, IRingingMsgContent, IRingingMsgOptions,\n} from './interfaces/IMessageHandler';\nimport { Platform } from './enums/Platform';\nimport { getCallDeviceId } from '../helper';\nimport { IOfflineRecord, OfflineRecorder } from './OfflineRecorder';\nimport { RCCallStateMachine } from '..';\n\nconst callMsgTypes = ['RC:VCAccept', 'RC:VCRinging', 'RC:VCSummary', 'RC:VCHangup', 'RC:VCInvite', 'RC:VCModifyMedia', 'RC:VCModifyMem'];\n\ntype MsgListenerKeys = keyof IMsgListener\n\ninterface IMsgBufferItem {\n  markTime: number,\n  msg: IReceivedMessage\n}\n\nconst EngineErrorCodeToCallErrorCode: { [key: number]: RCCallErrorCode } = {\n  [ErrorCode.REJECTED_BY_BLACKLIST]: RCCallErrorCode.REJECTED_BY_BLACKLIST,\n  [ErrorCode.NOT_IN_GROUP]: RCCallErrorCode.NOT_IN_GROUP,\n};\n\n/**\n * 消息接收处理: 在线消息、离线消息\n * 发送消息处理: 发送不同类型消息封装\n */\nexport class CallMessageHandler extends EventEmitter {\n  private _watchers: IMsgListener = {}\n\n  private _userInfo: IUserProfile = {}\n\n  private _msgBufferList: IMsgBufferItem[] = []\n\n  private _hadHandleMsgTimer: boolean = false\n\n  private _offlineRecorder!: OfflineRecorder\n\n  private _deviceId: string = ''\n\n  constructor(\n    private readonly _context: RTCPluginContext,\n    private readonly _runtime: IRuntime,\n    private readonly _logger: ILogger,\n    /**\n     * 离线消息处理时间间隔\n     */\n    private readonly _offlineMsgItv: number = 60 * 1000,\n    private readonly _getStateMachine: (callId: string) => RCCallStateMachine | null,\n  ) {\n    super();\n    // deviceId\n    this._deviceId = getCallDeviceId(_runtime);\n    // 处理消息收发\n    this._context.onmessage = this._onMessage.bind(this);\n    // 处理离线消息记录\n    this._offlineRecorder = new OfflineRecorder(this._context, this._logger, (record: IOfflineRecord) => {\n      this._logger.info('_', `[CallMessageHandler] offlineRecorder -> ${JSON.stringify(record)}`);\n      this._watchers.onOfflineRecord && this._watchers.onOfflineRecord(record);\n    });\n  }\n\n  private _onMessage(message: IReceivedMessage) {\n    const isCallMsg = callMsgTypes.includes(message.messageType);\n    if (isCallMsg) {\n      this._logger.debug('_', `[CallMessageHandler] _onMessage -> call message: ${JSON.stringify(message)}`);\n      // 通过遍历 bufferList 查找要插入的节点\n      try {\n        const markTime = Date.now();\n        const { sentTime: currentMsgSentTime } = message;\n        let insertIndex = 0;\n        // 需遍历数组全部项，故使用 forEach, findIndex 找到第一个符合条件的就会终止\n        this._msgBufferList.forEach(({ msg: { sentTime } }, index) => {\n          if (currentMsgSentTime >= sentTime) {\n            // insertIndex 需为当前符合条件元素的后一位，故 +1\n            insertIndex = index + 1;\n          }\n        });\n        this._msgBufferList.splice(insertIndex, 0, {\n          markTime,\n          msg: message,\n        });\n        this._logger.warn('_', `onMessage -> msgBufferList: ${this._msgBufferList.length}`);\n      } catch (error) {\n        this._logger.error('_', `[CallMessageHandler] splice buffer msg error -> ${(error as Error).message}`);\n      }\n      // 启动消息处理\n      this._handleBufferMsgs();\n      return true;\n    }\n    return false;\n  }\n\n  /**\n   * 在线消息抛给状态机处理\n   */\n  private _onRecvOnlineCallMsg(message: IReceivedMessage) {\n    this._logger.info('_', `onMessage -> _onRecvOnlineCallMsg: ${message.messageType}`);\n    const { content: { callId } } = message;\n    // 在线消息直接抛出\n    switch (message.messageType) {\n      case RCCallMessageType.VCInvite:\n        this._watchers.onInvite && this._watchers.onInvite(message);\n        break;\n      case RCCallMessageType.VCRinging:\n        super.emit(`${callId}onRinging`, message);\n        break;\n      case RCCallMessageType.VCAccept:\n        super.emit(`${callId}onAccept`, message);\n        break;\n      case RCCallMessageType.VCModifyMem:\n        // 收到人员变更抛出 onInvite 生成状态机实例\n        this._watchers.onInvite && this._watchers.onInvite(message);\n        break;\n      case RCCallMessageType.VCModifyMedia:\n        super.emit(`${callId}onMediaModify`, message);\n        break;\n      case RCCallMessageType.VCHangup:\n        super.emit(`${callId}onHungup`, message);\n        break;\n      default:\n        this._logger.warn('_', `[CallMessageHandler] onRecvOnlineCallMsg -> unexpected message: ${JSON.stringify(message)}`);\n        break;\n    }\n  }\n\n  /**\n   * 消息 buffer 列表处理逻辑\n   * 1、每 20ms 检查一次 buffer list\n   * 2、取出已经延迟 200 的消息列表进行消费 | 无延迟 200ms 内消息直接递归\n   * 3、消费分为 离线消息消费逻辑、在线消息消费逻辑，消费后递归\n  */\n  private _handleBufferMsgs() {\n    // 消息 buffer 列表长度为 0 时加锁\n    if (this._msgBufferList.length === 0 || this._hadHandleMsgTimer) {\n      this._logger.warn('_', '_handleBufferMsgs return');\n      return;\n    }\n    // 需要加锁，收到多次消息后，可能会起多个 timer\n    this._hadHandleMsgTimer = true;\n    setTimeout(() => {\n      // 取出大于 200 ms 消息列表\n      const currentTime = Date.now();\n      const buffers = this._msgBufferList.filter((item) => currentTime - item.markTime >= 200);\n      this._logger.debug('_', `[CallMessageHandler] handleBufferMsgs -> lists over 200ms : ${JSON.stringify(buffers.map(({ msg: { messageUId, isOffLineMessage, content: { callId } } }) => ({ messageUId, isOffLineMessage, callId })))}`);\n      if (buffers.length === 0) {\n        // 没有延迟 200ms 的消息，消费逻辑解锁并递归执行\n        this._hadHandleMsgTimer = false;\n        this._handleBufferMsgs();\n        return;\n      }\n\n      if (buffers[0].msg.isOffLineMessage) {\n        // 当第一条消息为离线消息时，直接从 buffer 中取出所有离线消息，进行消息\n        let offlineBuffers = this._msgBufferList.filter((item) => item.msg.isOffLineMessage);\n\n        // 离线消息处理逻辑\n        do {\n          const {\n            conversationType, messageType, sentTime, senderUserId, content: { callId: inviteCallId, inviteUserIds },\n          } = offlineBuffers[0].msg;\n\n          const isInviteMsgType = [RCCallMessageType.VCInvite, RCCallMessageType.VCModifyMem].includes(messageType as RCCallMessageType);\n\n          // 计算当前时间与消息发送时间差\n          const delayTime = this._context.getServerTime() - sentTime;\n          const isLessThanOfflineMsgItv = delayTime < this._offlineMsgItv;\n\n          if (!isLessThanOfflineMsgItv) {\n            this._logger.warn('_', `offline msg delayTime: ${delayTime}ms`);\n          }\n\n          if (isInviteMsgType) { // 取出的第一条消息为 invite | memModify\n            // 取出相同 CallId 消息列表\n            const taskMsgList: IReceivedMessage[] = [];\n            for (let i = 0; i < offlineBuffers.length; i++) {\n              const item = offlineBuffers[i].msg;\n              const { content: { callId: otherCallId } } = item;\n              if (inviteCallId === otherCallId) {\n                taskMsgList.push(item);\n              } else {\n                break;\n              }\n            }\n            this._logger.warn('_', `taskMsgList length: ${taskMsgList.length}`);\n\n            if (taskMsgList.length > 0) { // 防止 taskMsgList 为 0\n              // 找出 msgBufferList 中已消费的消息最大 index\n              const delIndex = this._msgBufferList.findIndex((item) => item.msg.messageUId === taskMsgList[taskMsgList.length - 1].messageUId);\n              // 删除消费过 msgBufferList 的消息\n              this._msgBufferList = this._msgBufferList.slice(delIndex + 1);\n\n              // 找出 offlineMsgs 中已消费的消息最大 index\n              const delOfflineIndex = offlineBuffers.findIndex((item) => item.msg.messageUId === taskMsgList[taskMsgList.length - 1].messageUId);\n              // 删除消费过 offlineBuffers 的消息\n              offlineBuffers = offlineBuffers.slice(delOfflineIndex + 1);\n            }\n            /**\n             * 单聊未结束通话判断\n             * 如果消息在 60s 内，判断是否未成对（只有 invite 或只有 invite 或 ringing 成对抛给 离线记录器，未成对抛给状态机\n             */\n            const isUnfinishedPrivateCall: boolean = (() => {\n              if (conversationType !== ConversationType.PRIVATE) return false;\n              const isOnlyInvite = taskMsgList.length === 1;\n              const hasInviteAndRinging = taskMsgList.every((item) => [RCCallMessageType.VCInvite, RCCallMessageType.VCModifyMedia, RCCallMessageType.VCRinging].includes(item.messageType as RCCallMessageType));\n              return isOnlyInvite || hasInviteAndRinging;\n            })();\n            /**\n             * 群聊未结束通话判断\n             * 通过 list 中 invite 和 memberModify 消息中取出所有参与通话的 userId\n             * 遍历 list 找出所有 hangup 的 sendUserId\n             *     总人数 - 挂断人数 > 1  且剩余人数包含自己，认为通话未挂断进状态机\n             *     总人数 - 挂断人数 <= 1 通话挂断，进离线消息处理器\n             *     主叫直接挂断且其他人未接听直接进离线处理器\n             */\n            const isUnfinishedGrpCall: boolean = (() => {\n              if (conversationType !== ConversationType.GROUP) return false;\n              let isUnfinished = false;\n              let noOneAccept = true;\n              let allUserIds = [senderUserId, ...inviteUserIds];\n              let isCallerHungup = false;\n              for (let i = 0; i < taskMsgList.length; i++) {\n                const { senderUserId: taskMsgSenderUserId, messageType } = taskMsgList[i];\n                // 只要自己挂断直接跳出循环，进离线消息\n                if (messageType === RCCallMessageType.VCHangup && taskMsgSenderUserId === this._context.getCurrentId()) {\n                  break;\n                }\n                // 找出所有挂断的人员，以及通话发起者是否挂断\n                if (messageType === RCCallMessageType.VCHangup) {\n                  isCallerHungup = senderUserId === taskMsgSenderUserId;\n                  allUserIds = allUserIds.filter((id) => taskMsgSenderUserId !== id);\n                }\n                // 判断是否无人接听\n                if (messageType === RCCallMessageType.VCAccept) {\n                  noOneAccept = false;\n                }\n              }\n              // 不是主叫挂断且无人接听，且剩余人数大于 1\n              if (!(noOneAccept && isCallerHungup) && allUserIds.length > 1) {\n                isUnfinished = true;\n              }\n              return isUnfinished;\n            })();\n\n            if (isLessThanOfflineMsgItv && (isUnfinishedPrivateCall || isUnfinishedGrpCall)) {\n              taskMsgList.forEach(this._onRecvOnlineCallMsg, this);\n            }\n            this._offlineRecorder.onRecvOfflineMsgs(taskMsgList);\n          } else {\n            if (isLessThanOfflineMsgItv && this._getStateMachine(inviteCallId)) {\n              this._onRecvOnlineCallMsg(offlineBuffers[0].msg);\n            } else {\n              this._logger.debug('_', `[CallMessageHandler] unexcepted offline msg -> ${JSON.stringify(offlineBuffers[0].msg)}`);\n            }\n            offlineBuffers.shift();\n            this._msgBufferList.shift();\n          }\n        } while (offlineBuffers.length > 0);\n      } else {\n        // 在线消息处理逻辑\n        buffers.forEach(({ msg }) => {\n          this._onRecvOnlineCallMsg(msg!);\n        });\n        // 找出 msgBufferList 中已消费的消息最大 index\n        const delCount = buffers.length;\n        // 删除消费过的消息\n        this._msgBufferList.splice(0, delCount);\n        this._logger.debug('_', `timer online msg handle -> delCount: ${delCount} msgBufferList: ${this._msgBufferList.length}`);\n      }\n      this._hadHandleMsgTimer = false;\n      this._handleBufferMsgs();\n    }, 20);\n  }\n\n  registerEventListener(listener: IMsgListener) {\n    Object.assign(this._watchers, listener);\n  }\n\n  registerStateMachineEvent(callId: string, funcName: MsgListenerKeys, event: (...args: any[]) => void) {\n    const eventType = callId + funcName;\n    super.on(eventType, event);\n  }\n\n  unregisterStateMachineEvent(callId: string) {\n    ['onRinging', 'onAccept', 'onHungup', 'onMediaModify'].forEach((funcName) => {\n      const eventType = callId + funcName;\n      super.removeAll(eventType);\n    });\n  }\n\n  registerUserInfo(userInfo: IUserProfile) {\n    this._userInfo = userInfo;\n  }\n\n  /**\n   * 发送 IM 消息\n   */\n  private async _sendCallMessage(options: ICallMsgOption): Promise<{ code: RCCallErrorCode, message?: IReceivedMessage }> {\n    this._logger.debug('_', `CallMessageHandler] sendCallMesage -> message: ${JSON.stringify(options)}`);\n    const {\n      channelId, conversationType, targetId, content, messageType, directionalUserIdList, pushConfig,\n    } = options;\n    const sendOpts: ISendMsgOptions = {\n      channelId,\n      messageType,\n      content,\n      directionalUserIdList,\n    };\n\n    if ([RCCallMessageType.VCInvite, RCCallMessageType.VCModifyMem, RCCallMessageType.VCHangup].includes(messageType)) {\n      const newPushConfig = pushConfig || { pushTitle: '', pushContent: '', pushData: '' };\n      newPushConfig.androidConfig = Object.assign(pushConfig?.androidConfig || {}, {\n        categoryHW: 'VOIP',\n        categoryVivo: 'IM',\n      });\n      newPushConfig.iOSConfig = Object.assign(newPushConfig.iOSConfig || {}, {\n        apnsCollapseId: content.callId,\n      });\n      newPushConfig.disablePushTitle = false;\n      newPushConfig.forceShowDetailContent = false;\n      sendOpts.pushConfig = newPushConfig as IEnginePushConfig;\n    }\n    const { code, data: message } = await this._context.sendMessage(conversationType, targetId, sendOpts);\n    if (code !== ErrorCode.SUCCESS) {\n      this._logger.error('_', `CallMessageHandler] sendCallMesage error -> code: ${code}`);\n      return {\n        code: EngineErrorCodeToCallErrorCode[code] || RCCallErrorCode.SEND_MSG_ERROR,\n      };\n    }\n    return { code: RCCallErrorCode.SUCCESS, message };\n  }\n\n  /**\n   * 发送邀请消息\n   */\n  async sendInvite(options: IInviteMsgOptions) {\n    const {\n      roomType, channelId, conversationType, targetId, callId, mediaType, inviteUserIds, extra, pushConfig,\n    } = options;\n    this._logger.warn('_', 'CallMessageHandler] sendCallMesage sendInvite', JSON.stringify(options));\n    this._watchers.sendAccept && this._watchers.sendAccept({ callId });\n    const content: IInviteMsgContent = {\n      platform: Platform.WEB,\n      deviceId: this._deviceId,\n      callId,\n      roomType,\n      extra,\n      engineType: 4,\n      channelInfo: { Id: callId, Key: '' },\n      mediaType,\n      inviteUserIds,\n      observerUserIds: [],\n      user: this._userInfo,\n    };\n\n    if (pushConfig) {\n      pushConfig.pushData = JSON.stringify({\n        mediaType,\n        userIdList: inviteUserIds,\n        callId,\n      });\n    }\n\n    return this._sendCallMessage({\n      channelId,\n      conversationType,\n      targetId,\n      content,\n      messageType: RCCallMessageType.VCInvite,\n      directionalUserIdList: conversationType === ConversationType.GROUP ? inviteUserIds : [targetId],\n      pushConfig,\n    });\n  }\n\n  /**\n   * 发送人员变更消息\n   */\n  async sendMemeberModify(options: IMemberModifyMsgOptions) {\n    const {\n      channelId, conversationType, targetId, callId, mediaType, inviteUserIds, callerId,\n      existedUserPofiles, directionalUserIdList, extra, pushConfig,\n    } = options;\n    const content: IMemberModifyMsgContent = {\n      platform: Platform.WEB, // TODO 与 IM 一致\n      deviceId: this._deviceId,\n      callId,\n      extra,\n      engineType: 4,\n      channelInfo: { Id: callId, Key: '' },\n      mediaType,\n      inviteUserIds,\n      observerUserIds: [],\n      user: this._userInfo,\n      caller: callerId,\n      modifyMemType: MemberModifyType.ADD,\n      existedUserPofiles,\n    };\n\n    if (pushConfig) {\n      pushConfig.pushData = JSON.stringify({\n        mediaType,\n        userIdList: inviteUserIds,\n        callId,\n      });\n    }\n    return this._sendCallMessage({\n      channelId,\n      conversationType,\n      targetId,\n      content,\n      messageType: RCCallMessageType.VCModifyMem,\n      directionalUserIdList,\n      pushConfig,\n    });\n  }\n\n  /**\n   * 发送响铃消息\n   */\n  sendRinging(options: IRingingMsgOptions) {\n    const {\n      channelId, conversationType, targetId, callId, userIds,\n    } = options;\n    const content: IRingingMsgContent = {\n      platform: Platform.WEB,\n      deviceId: this._deviceId,\n      callId,\n      user: this._userInfo,\n    };\n    return this._sendCallMessage({\n      channelId,\n      conversationType,\n      targetId,\n      content,\n      messageType: RCCallMessageType.VCRinging,\n      directionalUserIdList: userIds,\n    });\n  }\n\n  /**\n   * 发送同意接听消息\n   */\n  sendAccept(options: IAcceptMsgOptions) {\n    const {\n      channelId, conversationType, targetId, callId, mediaType, userIds,\n    } = options;\n    const content: IAcceptMsgContent = {\n      platform: Platform.WEB,\n      deviceId: this._deviceId,\n      callId,\n      mediaType,\n      user: this._userInfo,\n    };\n    return this._sendCallMessage({\n      channelId,\n      conversationType,\n      targetId,\n      content,\n      messageType: RCCallMessageType.VCAccept,\n      directionalUserIdList: userIds,\n    });\n  }\n\n  /**\n   * 发送挂断消息\n   */\n  sendHungup(options: IHungupMsgOptions) {\n    const {\n      channelId, conversationType, targetId, callId, reason, userIds, pushConfig,\n    } = options;\n    const content: IHungupMsgContent = {\n      platform: Platform.WEB,\n      deviceId: this._deviceId,\n      callId,\n      reason,\n      user: this._userInfo,\n    };\n\n    if (pushConfig) {\n      pushConfig.pushData = JSON.stringify({\n        callId, reason,\n      });\n    }\n\n    return this._sendCallMessage({\n      channelId,\n      conversationType,\n      targetId,\n      content,\n      messageType: RCCallMessageType.VCHangup,\n      pushConfig,\n      directionalUserIdList: userIds,\n    });\n  }\n\n  /**\n   * 发送媒体变更消息\n   */\n  sendMediaModify(options: IMediaModifyMsgOptions) {\n    const {\n      channelId, conversationType, targetId, callId, mediaType, userIds,\n    } = options;\n    const content: IMediaModifyMsgContent = {\n      platform: Platform.WEB,\n      deviceId: this._deviceId,\n      callId,\n      mediaType,\n      user: this._userInfo,\n    };\n    return this._sendCallMessage({\n      channelId,\n      conversationType,\n      targetId,\n      content,\n      messageType: RCCallMessageType.VCModifyMedia,\n      directionalUserIdList: userIds,\n    });\n  }\n}\n","export enum RCCallLanguage {\n  ZH = 'zh',\n  EN = 'en'\n}\n","export const EN = {\n  PushTitle: {\n    AUDIO: 'You have a voice call',\n    VIDEO: 'You have a video call',\n  },\n};\n","export const ZH = {\n  PushTitle: {\n    AUDIO: '您有一条音频通话',\n    VIDEO: '您有一条视频通话',\n  },\n};\n","import { RCCallLanguage } from '../enums/RCCallLang';\nimport { EN } from './en';\nimport { ZH } from './zh';\n\n/**\n * CallEngine 全局语言设置，当前仅支持中、英文\n */\nexport class Local {\n  static _lang: RCCallLanguage = RCCallLanguage.ZH\n\n  static set(lang: RCCallLanguage) {\n    this._lang = lang;\n  }\n\n  static get() {\n    if (this._lang === RCCallLanguage.EN) {\n      return EN;\n    }\n    return ZH;\n  }\n}\n","import {\n  BasicLogger, ConversationType, IReceivedMessage, IRuntime, RTCPluginContext,\n} from '@rongcloud/engine';\nimport { RCCallStateMachine } from './core/StateMachine';\nimport {\n  ICallStateMachineWatchers, IEndSummary, IInvitedUsers, IMediaModifyInfo, IMemberModifyInfo, ISenderInfo, IStateChangeInfo, IUserProfile, IUserStateChangeInfo, IUserData, IInviteOptions,\n} from './core/interfaces/IStateMachine';\nimport { RCCallErrorCode } from './core/enums/RCCallCode';\nimport { RCCallUserState } from './core/enums/RCCallUserState';\nimport { RCCallSessionState } from './core/enums/RCCallSessionState';\nimport { RCCallEndReason } from './core/enums/RCCallEndReason';\nimport { RCCallMediaType } from './core/enums/RCCallMediaType';\nimport { CallMessageHandler } from './core/MessageHandler';\nimport { eventEmitter, generateRandomId, getCallDeviceId } from './helper';\nimport { RCCallLanguage } from './core/enums/RCCallLang';\nimport { ICallEngineOptions, ICallEngineWatchers } from './core/interfaces/ICallEngine';\nimport { RCCallMessageType } from './core/enums/RCCallMessageType';\nimport { IInviteMsgContent } from './core/interfaces/IMessageHandler';\nimport { Local } from './core/locale';\nimport { IOfflineRecord } from './core/OfflineRecorder';\nimport { RCCrossCallType } from './core/enums/RCCrossCallType';\nimport { IPushConfig } from '../src/interface';\n\nclass RCCallEngine {\n  private _stateMachine: { [callId: string]: RCCallStateMachine } = {}\n\n  private _callMsgHandler: CallMessageHandler\n\n  /**\n   * 初始化\n   */\n  constructor(\n    /**\n     * engine PlguinContext 实例\n     */\n    private readonly _context: RTCPluginContext,\n    /**\n     * 运行时相关\n     */\n    private readonly _runtime: IRuntime,\n    /**\n     * engine 日志模块实例，由 CallLib 层初始化\n     */\n    private readonly _logger: BasicLogger,\n    /**\n     * 监听方法\n     */\n    private readonly _watchers: ICallEngineWatchers,\n    /**\n     * 其他配置项\n     */\n    private readonly _options: ICallEngineOptions,\n  ) {\n    this._logger.warn('_', `RCCallEngine Version: ${__VERSION__} CommitId: ${__COMMIT_ID__}`);\n    // 设置 DeviceId\n    getCallDeviceId(_runtime);\n    // 监听 IM 消息\n    this._callMsgHandler = new CallMessageHandler(this._context, this._runtime, this._logger, this._options.offlineMsgItv, this._getStateMachine.bind(this));\n    // 注册消息模块监听\n    this._callMsgHandler.registerEventListener({\n      onInvite: this._onInvite.bind(this),\n      sendAccept: this._handleSendAccept.bind(this),\n      onOfflineRecord: this._watchers.onOfflineRecord,\n    });\n    // 监听状态机关闭\n    eventEmitter.on('onStateMachineClose', (callId: string) => {\n      delete this._stateMachine[callId];\n    });\n    // 设置语言\n    Local.set(_options.lang);\n  }\n\n  private _onInvite(msg: IReceivedMessage) {\n    const {\n      channelId, conversationType, targetId, content, messageType, senderUserId: suid, pushConfig,\n    } = msg;\n    this._logger.warn('_', `RCCallEngine _onInvite:targetId ${targetId} senderUserId: ${suid}`);\n    const {\n      mediaType, callId, extra, roomType,\n    } = content as IInviteMsgContent;\n    let senderUserId: string;\n    if (roomType !== RCCrossCallType.RCCallRoomTypeAcrossCall) {\n      senderUserId = suid;\n    } else {\n      [, senderUserId] = suid.split('_');\n    }\n    const crtUserId = this._context.getCurrentId();\n    if (this._context.getCurrentId() === senderUserId) { // 多端处理\n      return;\n    }\n\n    /**\n     * 群聊通话有其他端处理时，不再处理此通话中的邀请消息\n     * 条件1: 此人已被邀请通话\n     * 条件2: 本端无此通话的状态机\n     */\n    if (messageType === RCCallMessageType.VCModifyMem) {\n      const isKeeping = content.existedUserPofiles.some((item: { userId: string }) => (item.userId === crtUserId));\n      if (isKeeping && !this._stateMachine[callId]) {\n        return;\n      }\n    }\n\n    const stateMachine = this._stateMachine[callId];\n    if (!stateMachine) {\n      this._stateMachine[callId] = new RCCallStateMachine(\n        this._context,\n        this._runtime,\n        this._logger,\n        this._callMsgHandler,\n        channelId!,\n        conversationType,\n        targetId,\n        mediaType,\n        callId,\n      );\n      this._logger.info('_', `[RCCallEngine] RCCallStateMachine successfully created -> callId: ${callId}`);\n\n      if (messageType === RCCallMessageType.VCInvite) {\n        // 状态机内部处理 invite 消息\n        this._stateMachine[callId].__onInvite(msg);\n      } else if (messageType === RCCallMessageType.VCModifyMem) {\n        this._stateMachine[callId].__onMemberModify(msg);\n      }\n      this._watchers.onInvite(this._stateMachine[callId], extra);\n\n      const hasOtherStateMachine = Object.keys(this._stateMachine).filter((otherCallId) => callId !== otherCallId).length > 0;\n      if (hasOtherStateMachine && !(this._options.isAllowAcceptNewCall || false)) {\n        this._stateMachine[callId].__handleInviteInSession();\n      }\n    } else if (messageType === RCCallMessageType.VCModifyMem) {\n      this._stateMachine[callId].__onMemberModify(msg);\n    }\n  }\n\n  /**\n   * 允许接听新的通话时，接听完新的通话后，挂断其他通话\n   */\n  private _handleSendAccept(info: { callId: string }) {\n    if (this._options.isAllowAcceptNewCall) {\n      const { callId } = info;\n      for (const id in this._stateMachine) {\n        if (callId !== id) {\n          this._stateMachine[id].hungup();\n          delete this._stateMachine[id];\n        }\n      }\n    }\n  }\n\n  /**\n   * 根据 callId 获取状态机\n  */\n  private _getStateMachine(callId: string): RCCallStateMachine | null {\n    return this._stateMachine[callId];\n  }\n\n  /**\n   * 注册用户信息, 发送 call 消息时用户信息携带\n   */\n  registerUserInfo(userInfo: IUserProfile) {\n    this._logger.debug('_', `[RCCallEngine] registerUserInfo -> userInfo: ${JSON.stringify(userInfo)}`);\n\n    this._callMsgHandler.registerUserInfo(userInfo);\n  }\n\n  /**\n   * 单呼\n   * @param channelId 组织 ID\n   * @param targetId  对方 ID\n   * @param mediaType 媒体类型\n   * @param extra 消息扩展信息\n   * @param pushConfig 推送配置\n   */\n  async call(\n    channelId: string,\n    targetId: string,\n    mediaType: number,\n    extra: string = '',\n    pushConfig: IPushConfig,\n    isCrossAppkey = false,\n  ): Promise<{ code: RCCallErrorCode, stateMachine?: RCCallStateMachine }> {\n    this._logger.debug('_', `[RCCallEngine] call -> args: ${JSON.stringify({\n      channelId, targetId, mediaType, extra, pushConfig,\n    })}`);\n    const callId = generateRandomId();\n    const hasStateMachine = Object.keys(this._stateMachine).length > 0;\n    if (hasStateMachine) {\n      return { code: RCCallErrorCode.STATE_MACHINE_EXIT };\n    }\n    this._stateMachine[callId] = new RCCallStateMachine(\n      this._context,\n      this._runtime,\n      this._logger,\n      this._callMsgHandler,\n      channelId,\n      ConversationType.PRIVATE,\n      targetId,\n      mediaType,\n      callId,\n    );\n    const { code } = await this._stateMachine[callId].__call([targetId], extra, pushConfig, isCrossAppkey);\n    if (code === RCCallErrorCode.SUCCESS) {\n      return {\n        code: RCCallErrorCode.SUCCESS,\n        stateMachine: this._stateMachine[callId],\n      };\n    }\n    return { code };\n  }\n\n  /**\n   * 群呼\n   * @param channelId 组织 ID\n   * @param targetId  群组 ID\n   * @param mediaType 媒体类型\n   * @param userIds 被邀请人员列表\n   * @param extra 消息扩展信息\n   * @param pushConfig 推送配置\n   */\n  async callInGroup(\n    channelId: string,\n    targetId: string,\n    mediaType: number,\n    userIds: string[],\n    extra: string = '',\n    pushConfig: IPushConfig,\n  ): Promise<{ code: RCCallErrorCode, stateMachine?: RCCallStateMachine }> {\n    this._logger.debug('_', `[RCCallEngine] callInGroup -> args: ${JSON.stringify({ channelId, targetId, mediaType })}`);\n    const callId = generateRandomId();\n    const hasStateMachine = Object.keys(this._stateMachine).length > 0;\n    if (hasStateMachine) {\n      return { code: RCCallErrorCode.STATE_MACHINE_EXIT };\n    }\n    this._stateMachine[callId] = new RCCallStateMachine(\n      this._context,\n      this._runtime,\n      this._logger,\n      this._callMsgHandler,\n      channelId,\n      ConversationType.GROUP,\n      targetId,\n      mediaType,\n      callId,\n    );\n    const { code } = await this._stateMachine[callId].__call(userIds, extra, pushConfig);\n    if (code === RCCallErrorCode.SUCCESS) {\n      return {\n        code: RCCallErrorCode.SUCCESS,\n        stateMachine: this._stateMachine[callId],\n      };\n    }\n    return { code };\n  }\n\n  /**\n   * 销毁当前的状态机\n   */\n  destroy() {\n    this._logger.debug('_', '[RCCallEngine] destroy');\n    this._stateMachine = {};\n  }\n}\n\nexport {\n  RCCallEngine,\n  RCCallStateMachine, RCCallErrorCode,\n  RCCallUserState,\n  RCCallSessionState,\n  RCCallEndReason,\n  RCCallMediaType,\n  RCCallLanguage,\n};\n\nexport type {\n  ICallStateMachineWatchers, IEndSummary,\n  ICallEngineWatchers,\n  IUserStateChangeInfo,\n  IStateChangeInfo,\n  ISenderInfo,\n  IMemberModifyInfo,\n  IMediaModifyInfo,\n  IInvitedUsers,\n  IUserData,\n  IUserProfile,\n  IOfflineRecord,\n  IInviteOptions,\n};\n","/**\n * 产生session的场景\n */\nexport enum ProduceTypes {\n  /**\n   * 主叫\n   */\n  CALLER = 1,\n\n  /**\n   * 被叫\n   */\n  CALLEE = 2\n}\n","class EventEmitter {\n  private list: {[propName: string]: any[]} = {}\n\n  public on(event: string, fun:(data?:any)=>void) {\n    (this.list[event] || (this.list[event] = [])).push(fun);\n    return this;\n  }\n\n  public once(event: string, fun:(data?:any)=>void) {\n    const on = (data?:any) => {\n      this.off(event, on);\n      fun.call(this, data);\n    };\n    on.fun = fun;\n    this.on(event, on);\n  }\n\n  public off(event: string, fun?:(data?:any)=>void) {\n    const funs = this.list[event];\n    if (!funs) {\n      return false;\n    }\n    if (!fun) {\n      // 如果没有传 fn 的话，就会将 event 值对应缓存列表中的 fun 都清空\n      funs && (funs.length = 0);\n    } else {\n      // 若有 fun，遍历缓存列表，看看传入的 fn 与哪个函数相同，如果相同就直接从缓存列表中删掉即可\n      let cb;\n      for (let i = 0, { length } = funs; i < length; i++) {\n        cb = funs[i];\n        if (cb === fun || cb.fun === fun) {\n          funs.splice(i, 1);\n          break;\n        }\n      }\n    }\n  }\n\n  public emit(event: string, data?:any) {\n    // 第一个参数是对应的 event 值，直接用数组的 shift 方法取出\n    const funs = [...this.list[event]];\n\n    // 如果缓存列表里没有 fun 就返回 false\n    if (!funs || funs.length === 0) {\n      return false;\n    }\n    // 遍历 event 值对应的缓存列表，依次执行 fn\n    funs.forEach((fun) => {\n      fun.call(this, data);\n    });\n  }\n}\nexport {\n  EventEmitter,\n};\n","import { EventEmitter } from './utils';\n\nexport default new EventEmitter();\n","import {\n  RTCJoinType, LogL, isUndefined, isBoolean, isString, IiOSPushConfig,\n} from '@rongcloud/engine';\nimport { RCCallErrorCode, RCCallLanguage, RCCallMediaType } from '@rc-embed/call-engine';\nimport { RCResolution } from '@rongcloud/plugin-rtc';\n\nimport {\n  ISessionListener, IRCCallInitOptions, IValidationResult, IMediaStreamConstraints, IPushConfig,\n} from './interface';\n\nfunction isLanguage(val: string) {\n  const values: string[] = Object.values(RCCallLanguage);\n  return values.includes(val);\n}\n\nfunction isJoinType(val: number) {\n  const values = Object.values(RTCJoinType) as number[];\n  return values.includes(val);\n}\n\nfunction isLogLevel(val: number) {\n  return [LogL.DEBUG, LogL.INFO, LogL.WARN, LogL.ERROR].includes(val);\n}\n\nexport const validateCallInitOptions = (options: IRCCallInitOptions): IValidationResult => {\n  if (!options) {\n    return { result: false, msg: 'Initialization missing parameter -> options' };\n  }\n  if (typeof options !== 'object') {\n    return { result: false, msg: 'Initialization options must be an object' };\n  }\n  const keyNames: string[] = ['rtcClient', 'onSession', 'onSessionClose'];\n  const keys: string[] = Object.keys(options);\n  const missingKeys: string[] = [];\n\n  // 校验填项是否都包含，如果哪个不包含就收集起来\n  keyNames.forEach((key: string) => {\n    if (!keys.includes(key)) {\n      missingKeys.push(key);\n    }\n  });\n\n  // 如果缺少必填的监听函数或对象\n  if (missingKeys.length) {\n    return { result: false, msg: `Initialization missing parameter -> \"${missingKeys.join(',')}\"` };\n  }\n  if (typeof options.rtcClient !== 'object') {\n    return { result: false, msg: 'Initialization \\'rtcClient\\' parameter must be of type \\'object\\'' };\n  }\n  if (typeof options.onSession !== 'function') {\n    return { result: false, msg: 'Initialization \\'onSession\\' parameter must be of type \\'function\\'' };\n  }\n  if (typeof options.onSessionClose !== 'function') {\n    return { result: false, msg: 'Initialization \\'onSessionClose\\' parameter must be of type \\'function\\'' };\n  }\n\n  // 如果传了isAllowSubscribeRetry 但不是boolean类型\n  if (typeof options.isAllowSubscribeRetry !== 'undefined' && typeof options.isAllowSubscribeRetry !== 'boolean') {\n    return { result: false, msg: 'Initialization \\'isAllowSubscribeRetry\\' parameter must be of type \\'boolean\\'' };\n  }\n\n  // 如果传了，但不是boolean类型\n  if (typeof options.isAllowPublishRetry !== 'undefined' && typeof options.isAllowPublishRetry !== 'boolean') {\n    return { result: false, msg: 'Initialization \\'isAllowPublishRetry\\' parameter must be of type \\'boolean\\'' };\n  }\n\n  // 如果传了，但不是boolean类型\n  if (typeof options.isOffCameraWhenVideoDisable !== 'undefined' && typeof options.isOffCameraWhenVideoDisable !== 'boolean') {\n    return { result: false, msg: 'Initialization \\'isOffCameraWhenVideoDisable\\' parameter must be of type \\'boolean\\'' };\n  }\n\n  // 如果传了，但不是RTCJoinType的枚举\n  if (typeof options.joinType !== 'undefined' && !isJoinType(options.joinType!)) {\n    return { result: false, msg: 'Initialization \\'joinType\\' parameter must be of type correct type' };\n  }\n\n  // 如果传了，但不是boolean类型\n  if (typeof options.isAllowDemotionGetStream !== 'undefined' && typeof options.isAllowDemotionGetStream !== 'boolean') {\n    return { result: false, msg: 'Initialization \\'isAllowDemotionGetStream\\' parameter must be of type \\'boolean\\'' };\n  }\n\n  // 如果传了，但不是RCCallLanguage的枚举\n  if (typeof options.lang !== 'undefined' && !isLanguage(options.lang!)) {\n    return { result: false, msg: 'Initialization \\'lang\\' parameter must be of type correct type' };\n  }\n\n  if (typeof options.logOutputLevel !== 'undefined' && !isLogLevel(options.logOutputLevel!)) {\n    return { result: false, msg: 'Initialization \\'logOutputLevel\\' parameter must be of type correct type' };\n  }\n  return { result: true };\n};\n\n/**\n * 校验registerSessionListener参数\n */\nexport const validateListener = (listener: ISessionListener): IValidationResult => {\n  if (!listener) {\n    return { result: false, msg: 'missing parameter -> listener' };\n  }\n  if (typeof listener !== 'object') {\n    return { result: false, msg: 'listener must be an object' };\n  }\n  const keyNames: string[] = ['onRinging', 'onAccept', 'onHungup', 'onTrackReady'];\n  const keys: string[] = Object.keys(listener);\n  const missingKeys: string[] = [];\n  keyNames.forEach((key: string) => {\n    if (!keys.includes(key)) {\n      missingKeys.push(key);\n    }\n  });\n  if (missingKeys.length) {\n    return { result: false, msg: `missing parameter -> \"${missingKeys.join(',')}\"` };\n  }\n  if (typeof listener.onRinging !== 'function') {\n    return { result: false, msg: '\\'onRinging\\' parameter must be of type \\'function\\'' };\n  }\n  if (typeof listener.onAccept !== 'function') {\n    return { result: false, msg: '\\'onAccept\\' parameter must be of type \\'function\\'' };\n  }\n  if (typeof listener.onHungup !== 'function') {\n    return { result: false, msg: '\\'onHungup\\' parameter must be of type \\'function\\'' };\n  }\n  if (typeof listener.onTrackReady !== 'function') {\n    return { result: false, msg: '\\'onTrackReady\\' parameter must be of type \\'function\\'' };\n  }\n  return { result: true };\n};\n\nexport const validateTargetId = (targetId: string): IValidationResult => {\n  if (targetId && typeof targetId === 'string') {\n    return { result: true };\n  }\n  return { result: false, msg: '\\'targetId\\' parameter is required, must be of type \\'string\\'' };\n};\n\nexport const validateMediaType = (mediaType: number): IValidationResult => {\n  if (mediaType === RCCallMediaType.AUDIO || mediaType === RCCallMediaType.AUDIO_VIDEO) {\n    return { result: true };\n  }\n  return { result: false, msg: '\\'mediaType\\' parameter is required, must be of type \\'RCCallMediaType\\'' };\n};\nexport const validateExtra = (extra: string): IValidationResult => {\n  if (typeof extra === 'string') {\n    return { result: true };\n  }\n  return { result: false, msg: '\\'extra\\' parameter must be of type \\'string\\'' };\n};\nexport const validatePushTitle = (pushTitle: string): IValidationResult => {\n  if (typeof pushTitle === 'string') {\n    return { result: true };\n  }\n  return { result: false, msg: '\\'pushTitle\\' parameter must be of type \\'string\\'' };\n};\nexport const validatePushContent = (pushContent: string): IValidationResult => {\n  if (typeof pushContent === 'string') {\n    return { result: true };\n  }\n  return { result: false, msg: '\\'pushContent\\' parameter must be of type \\'string\\'' };\n};\n\n/**\n * 校验 pushConfig\n */\nexport const validatePushConfig = (pushConfig: IPushConfig) => {\n  const pushConfigKeys = ['pushTitle', 'pushContent', 'pushData', 'iOSConfig', 'androidConfig', 'disablePushTitle', 'templateId'];\n  const keyIsRight = Object.keys(pushConfig).every((key) => pushConfigKeys.includes(key));\n  if (!keyIsRight) {\n    return {\n      code: RCCallErrorCode.PARAM_ERROR,\n      errorMsg: `right key: ${pushConfigKeys.join(',')}`,\n    };\n  }\n\n  for (let index = 0; index < pushConfigKeys.length; index++) {\n    const key = pushConfigKeys[index];\n    /**\n     * 校验 boolean 类型\n     */\n    if (key === 'disablePushTitle') {\n      if (!isUndefined(pushConfig[key]) && !isBoolean(pushConfig[key])) {\n        return {\n          code: RCCallErrorCode.PARAM_ERROR,\n          errorMsg: `${key} type is boolean`,\n        };\n      }\n    } else if (key === 'pushTitle'\n      || key === 'templateId') {\n      /**\n       * 校验 string 类型\n       */\n      if (pushConfig[key] && !isString(pushConfig[key])) {\n        return {\n          code: RCCallErrorCode.PARAM_ERROR,\n          errorMsg: `${key} type is string`,\n        };\n      }\n    }\n  }\n  const { iOSConfig, androidConfig } = pushConfig;\n\n  /**\n   * 校验 iOSConfig\n   */\n  const validateIOSConfigRes = validateIOSConfig(iOSConfig || {});\n  if (validateIOSConfigRes.code !== RCCallErrorCode.SUCCESS) {\n    return validateIOSConfigRes;\n  }\n\n  /**\n   * 校验 androidConfig\n   */\n  const validateAndroidConfigRes = validateAndroidConfig(androidConfig || {});\n  if (validateAndroidConfigRes.code !== RCCallErrorCode.SUCCESS) {\n    return validateAndroidConfigRes;\n  }\n\n  return { code: RCCallErrorCode.SUCCESS };\n};\n\nconst validateIOSConfig = (iOSConfig: IiOSPushConfig) => {\n  const { threadId, richMediaUri } = iOSConfig || {};\n  if (threadId && !isString(threadId)) {\n    return {\n      code: RCCallErrorCode.PARAM_ERROR,\n      errorMsg: 'iOSConfig.threadId type is string',\n    };\n  }\n\n  if (richMediaUri && !isString(richMediaUri)) {\n    return {\n      code: RCCallErrorCode.PARAM_ERROR,\n      errorMsg: 'iOSConfig.richMediaUri type is string',\n    };\n  }\n  return { code: RCCallErrorCode.SUCCESS };\n};\n\nconst validateAndroidConfig = (androidConfig: any) => {\n  const androidConfigKeys = ['notificationId', 'channelIdMi', 'miLargeIconUrl', 'channelIdHW', 'categoryHW', 'importanceHW', 'imageUrlHW', 'channelIdOPPO', 'typeVivo', 'categoryVivo', 'fcmChannelId', 'fcmCollapseKey', 'fcmImageUrl', 'importanceHonor', 'imageUrlHonor'];\n  const keyIsRight = Object.keys(androidConfig).every((key) => androidConfigKeys.includes(key));\n  if (!keyIsRight) {\n    return {\n      code: RCCallErrorCode.PARAM_ERROR,\n      errorMsg: `right key: ${androidConfigKeys.join(',')}`,\n    };\n  }\n\n  for (let index = 0; index < androidConfigKeys.length; index++) {\n    const key = androidConfigKeys[index];\n    if (key === 'importanceHW' || key === 'importanceHonor') {\n      if (androidConfig[key] && androidConfig[key] !== 'LOW' && androidConfig[key] !== 'NORMAL') {\n        return {\n          code: RCCallErrorCode.PARAM_ERROR,\n          errorMsg: `androidConfig.${key} value is NORMAL|LOW`,\n        };\n      }\n    } else if (androidConfig[key] && !isString(androidConfig[key])) {\n      /**\n       * 校验 string 类型\n       */\n      return {\n        code: RCCallErrorCode.PARAM_ERROR,\n        errorMsg: `androidConfig.${key} type is string`,\n      };\n    }\n  }\n  return { code: RCCallErrorCode.SUCCESS };\n};\n\nexport const validateUserIds = (userIds: string[]): IValidationResult => {\n  if (!Array.isArray(userIds)) {\n    return { result: false, msg: '\\'userIds\\' parameter is required, must be of type \\'string[]\\'' };\n  }\n  if (!userIds.length) {\n    return { result: false, msg: '\\'userIds\\' parameter is required, must be of type \\'string[]\\'' };\n  }\n  if (!userIds.every((str) => typeof str === 'string' && str.length > 0)) {\n    return { result: false, msg: '\\'userIds\\' parameter is required' };\n  }\n  return { result: true };\n};\n\nfunction isRCFrameRate(val: string): boolean {\n  const arrs = ['FPS_10', 'FPS_15', 'FPS_24', 'FPS_30'];\n  return arrs.includes(val);\n}\n\nconst isValidResolution = (resolution?: RCResolution): boolean => !!RCResolution[resolution!];\n\nexport const validateMediaStreamConstraints = (constraints: IMediaStreamConstraints): IValidationResult => {\n  if (constraints && constraints.audio && typeof constraints.audio.micphoneId !== 'undefined' && typeof constraints.audio.micphoneId !== 'string') {\n    return { result: false, msg: '\\'constraints.audio.micphoneId\\' must be of type \\'string\\'' };\n  }\n  if (constraints && constraints.audio && typeof constraints.audio.sampleRate !== 'undefined' && typeof constraints.audio.sampleRate !== 'number') {\n    return { result: false, msg: '\\'constraints.audio.sampleRate\\' must be of type \\'number\\'' };\n  }\n  if (constraints && constraints.video && typeof constraints.video.cameraId !== 'undefined' && typeof constraints.video.cameraId !== 'string') {\n    return { result: false, msg: '\\'constraints.video.cameraId\\' must be of type \\'string\\'' };\n  }\n  // if (constraints && constraints.video && typeof constraints.video.faceMode !== 'undefined' && constraints.video.cameraId !== 'user' && constraints.video.faceMode !== 'environment') {\n  //   return { result: false, msg: '\\'constraints.video.cameraId\\' must be  \\'user\\' or \\'environment\\'' }\n  // }\n\n  if (constraints && constraints.video && typeof constraints.video.frameRate !== 'undefined' && typeof constraints.video.frameRate !== 'string') {\n    return { result: false, msg: '\\'constraints.video.frameRate\\' must be of type \\'string\\'' };\n  }\n\n  if (constraints && constraints.video && typeof constraints.video.frameRate !== 'undefined' && !isRCFrameRate(constraints.video.frameRate)) {\n    return { result: false, msg: '\\'frameRate\\' value is out of range' };\n  }\n\n  if (constraints && constraints.video && typeof constraints.video.resolution !== 'undefined' && typeof constraints.video.resolution !== 'string') {\n    return { result: false, msg: '\\'constraints.video.frameRate\\' must be of type \\'string\\'' };\n  }\n\n  if (constraints && constraints.video && typeof constraints.video.resolution !== 'undefined' && !isValidResolution(constraints.video.resolution)) {\n    return { result: false, msg: '\\'resolution\\' value is out of range' };\n  }\n  if (constraints && constraints.video && (!constraints.video.frameRate || !constraints.video.resolution)) {\n    return { result: false, msg: '\\'resolution\\' and \\'resolution\\' is required' };\n  }\n\n  return { result: true };\n};\n","export const timerSetTimeout = (func: Function, timeout: number): number => setTimeout(func, timeout) as number;\n","import { timerSetTimeout } from './helper';\n\nexport class Timer {\n  private _timerId: number = 0\n\n  private _startTime: number = 0\n\n  constructor(callback: Function, timeout: number) {\n    if (callback) {\n      this._timerId = timerSetTimeout(() => {\n        callback();\n      }, timeout);\n    }\n    this._startTime = Date.now();\n  }\n\n  stop(): {\n    startTime: number,\n    endTime: number,\n    duration: number\n    } {\n    clearTimeout(this._timerId);\n\n    const endTime = Date.now();\n    let duration = endTime - this._startTime;\n    if (this._startTime === 0) {\n      duration = 0;\n    }\n\n    return {\n      startTime: this._startTime,\n      endTime,\n      duration,\n    };\n  }\n\n  reset() {\n    this._startTime = 0;\n  }\n}\n","import { ConversationType, ILogger } from '@rongcloud/engine';\nimport {\n  IMediaModifyInfo,\n  IMemberModifyInfo,\n  ISenderInfo,\n  IStateChangeInfo,\n  IUserData,\n  IUserStateChangeInfo,\n  IInviteOptions,\n  RCCallEndReason,\n  RCCallErrorCode,\n  RCCallMediaType,\n  RCCallSessionState,\n  RCCallStateMachine,\n  RCCallUserState,\n} from '@rc-embed/call-engine';\nimport {\n  IRCRTCStateReport,\n  RCKickReason,\n  RCLocalTrack,\n  RCRemoteAudioTrack,\n  RCRemoteTrack,\n  RCRemoteVideoTrack,\n  RCRTCClient,\n  RCRTCCode,\n  RCRTCPingResult,\n  RCRTCRoom,\n  IMicphoneAudioProfile,\n} from '@rongcloud/plugin-rtc';\n\nimport { ProduceTypes } from './enums';\n\nimport {\n  IDeviceChangeParams,\n  IMediaStreamConstraints,\n  IMuteUser,\n  IRCCallSessionOptions,\n  ISessionListener,\n  IValidationResult,\n} from './interface';\nimport eventEmitter from './eventEmitter';\nimport {\n  validateListener, validateMediaStreamConstraints, validateUserIds, validateExtra, validatePushTitle, validatePushContent, validatePushConfig,\n} from './validation';\nimport { Timer } from './timer';\n\nexport class RCCallSession {\n  /**\n   * RTC房间实例\n   */\n  private _room!: RCRTCRoom\n\n  /**\n   * 用户传进来的 对session的监听 (要在RCCallClient的_onInvite里判断，要求执行完onSession必须注册session的监听，所以这里是public)\n   */\n  public _listener: ISessionListener | null = null\n\n  /**\n   * RTC订阅、发布重试的次数\n   */\n  private readonly _RETRYCOUNT: number = 2\n\n  /**\n   * 加入房间定时器\n   */\n\n  private joinRoomTimer: any = null\n\n  constructor(\n\n    /**\n     * 状态机实例\n     */\n    private _stateMachine: RCCallStateMachine,\n\n    /**\n     * rtc实例\n     */\n    private readonly _rtcClient: RCRTCClient,\n\n    private readonly _logger: ILogger,\n\n    /**\n     * session的其它选项\n     */\n    private _options: IRCCallSessionOptions = {},\n\n  ) {\n    // 监听状态机\n    this._stateMachine.registerEventListener({\n      /**\n       * 用户状态变更\n       * @param info\n       */\n      onUserStateChange: ({ user, reason }: IUserStateChangeInfo) => {\n        this._logger.info('_', `[RCCallSession onUserStateChange] userId->${user?.userId} state->${user?.state} reason->${reason}`);\n      },\n\n      /**\n       * 房间状态变更\n       * @param\n       */\n      onStateChange: async (info: IStateChangeInfo) => {\n        const { state, reason } = info;\n        this._logger.info('_', `[RCCallSession onStateChange] : state->${state} reason->${reason}`);\n\n        // 如果在通话中，就加房间\n        if (state === RCCallSessionState.KEEPING) {\n          const roomId: string = this._stateMachine.getCallId();\n          this._logger.info('_', `[RCCallSession onStateChange] roomId: ${roomId}`);\n          try {\n            // 加房间\n            await this._joinRoom(roomId);\n          } catch (error) {\n            this._exceptionClose(RCCallEndReason.NETWORK_ERROR);\n            this._logger.error('_', `[RCCallSession onStateChange] joinRoom throw exception roomId -> ${roomId}`);\n            console.error(error);\n          }\n\n          /**\n           *  以下三条只要满足一条，状态会变成RCCallSessionState.END\n           *  1、本端用户自己主动挂断\n           *  2、服务端把本端用户踢出RTC房间\n           *  3、房间里小于2个人\n           */\n        } else if (state === RCCallSessionState.END) {\n          // 还未加入房间就挂断\n          if (!this._room) {\n            // 销毁本地流，关闭摄像头\n            this._options.localTracks && this._destroyTracks(this._options.localTracks);\n            const summaryInfo = this._stateMachine.getSummary();\n            eventEmitter.emit('sessionClose', { session: this, summaryInfo });\n            return;\n          }\n\n          this._options.localTracks && this._destroyTracks(this._options.localTracks);\n          this._logger.info('_', '[RCCallSession onStateChange] localTracks destroyed');\n          this._leaveRoom();\n          this._room = null as unknown as RCRTCRoom;\n        }\n      },\n\n      /**\n       * 收到响铃\n       * @param sender 发起用户信息\n       */\n      onRinging: (sender: ISenderInfo) => {\n        this._logger.info('_', `[RCCallSession onRinging]sender: sender.userId -> ${sender.userId}`);\n\n        try {\n          // 通知用户响铃\n          this._listener!.onRinging(sender, this);\n        } catch (error) {\n          this._logger.error('_', '[RCCallSession onRinging] method exception -> onRinging');\n          console.error(error);\n        }\n      },\n\n      /**\n         * 当远端用户同意接听\n         */\n      onAccept: (sender: ISenderInfo) => {\n        this._logger.info('_', `[RCCallSession onAccept]sender: sender.userId -> ${sender.userId}`);\n        try {\n          // 通知本端，远端用户已接听\n          this._listener!.onAccept(sender, this);\n        } catch (error) {\n          this._logger.error('_', '[RCCallSession onAccept] method exception -> onAccept');\n          console.error(error);\n        }\n      },\n\n      /**\n         * 当有远端用户挂断\n         */\n      onHungup: (sender: ISenderInfo, reason: RCCallEndReason) => {\n        this._logger.info('_', `[RCCallSession onHungup]sender: sender.userId -> ${sender.userId} reason->${reason}`);\n        try {\n          // 通知本端，远端用户已挂断\n          this._listener!.onHungup(sender, reason, this);\n        } catch (error) {\n          this._logger.error('_', '[RCCallSession onHungup] method exception -> onHungup');\n          console.error(error);\n        }\n      },\n\n      /**\n       * 收到人员变更\n       * @param sender 发起用户信息\n       */\n      onMemberModify: ({ sender, invitedUsers }: IMemberModifyInfo) => {\n        this._logger.info('_', `[RCCallSession onMemberModify] sender.userId -> ${sender.userId}`);\n        try {\n          // 通知用户人员变更\n          this._listener!.onMemberModify(sender, invitedUsers, this);\n        } catch (error) {\n          this._logger.error('_', '[RCCallSession onMemberModify] method exception -> onMemberModify');\n          console.error(error);\n        }\n      },\n\n      /**\n       * 收到通话类型变更 (通话降级)\n       * @param sender 发起用户信息\n       */\n      onMediaModify: ({ sender, mediaType }: IMediaModifyInfo) => {\n        this._logger.info('_', `[RCCallSession onMediaModify]sender: sender.userId -> ${sender.userId} mediaType: ${mediaType}`);\n        if (mediaType === RCCallMediaType.AUDIO) {\n          // 远端收到通话降级通知后，远端执行降级通话(不发消息)\n          this._setMediaTypeToAudio();\n        }\n        try {\n          this._listener!.onMediaModify(sender, mediaType, this);\n        } catch (error) {\n          this._logger.error('_', '[RCCallSession onMediaModify] method exception -> onMediaModify');\n          console.error(error);\n        }\n      },\n      /**\n       * 是否跨appkey\n       * @param sender 发起用户信息\n       */\n      crossAppkey: (isCrossAppkey: boolean) => {\n        this._logger.info('_', `[RCCallSession crossAppkey] 是否跨 appkey: ${isCrossAppkey}`);\n        this._options.isCrossAppkey = isCrossAppkey;\n      },\n    });\n\n    /**\n     * 设置挂断的推送信息\n     */\n    this._stateMachine.setHungupPushConfig(this._options.hungupPushConfig || { pushTitle: '', pushContent: '' });\n  }\n\n  /**\n   *  加入房间\n   */\n  private async _joinRoom(roomId: string): Promise<{ code: RCCallErrorCode }> {\n    let callBack;\n    try {\n      // 加房间\n      if (this._options.isCrossAppkey) {\n        callBack = await this._rtcClient.joinCrossRTCRoom(roomId, this._options.joinType);\n      } else {\n        callBack = await this._rtcClient.joinRTCRoom(roomId, this._options.joinType);\n      }\n\n      const { code, userIds, room } = callBack;\n\n      if (code !== RCRTCCode.SUCCESS) {\n        // 如果音视频服务未开通\n        if (code === RCRTCCode.NOT_OPEN_VIDEO_AUDIO_SERVER) {\n          this._exceptionClose(RCCallEndReason.SERVICE_NOT_OPENED);\n          // 己方其他端已在通话中\n        } if (code === RCRTCCode.SIGNAL_JOIN_RTC_ROOM_REFUSED) {\n          this._exceptionClose(RCCallEndReason.OTHER_CLIENT_IN_CALL);\n        } else {\n          this._exceptionClose(RCCallEndReason.NETWORK_ERROR);\n        }\n\n        this._logger.info('_', `[RCCallClient _joinRoom] join room failed: roomId -> ${roomId} RCRTCCode -> ${code}`);\n        return { code: RCCallErrorCode.JOIN_ROOM_ERROR };\n      }\n\n      /**\n       * 群聊本端加入房间后，更新人员状态为通话中\n       */\n      const conversationType = this._stateMachine.getConversationType();\n      conversationType === ConversationType.GROUP && this._stateMachine.userJoin([this._rtcClient.getCurrentId()]);\n\n      /**\n       * 针对私有云 @rongcloud/plugin-rtc@5.1.10-enterprise.7 增加补丁，\n       * 私有云升完 RTC sdk 版本后删掉此补丁\n       * 加完房间后，对方挂断了，需退出房间\n       */\n      if (this._stateMachine.getState() === RCCallSessionState.END) {\n        await this._rtcClient.leaveRoom(room!);\n        this._room = null as unknown as RCRTCRoom;\n        return { code: RCCallErrorCode.SUCCESS };\n      }\n\n      // 被叫方加入房间成功，但主叫方在加入房间前离线\n      if (userIds!.length < 1) {\n        this.joinRoomTimer = new Timer(() => {\n          this._exceptionClose(RCCallEndReason.REMOTE_NETWORK_ERROR);\n        }, 60000);\n      }\n      this._room = room as RCRTCRoom;\n    } catch (error) {\n      this._exceptionClose(RCCallEndReason.NETWORK_ERROR);\n      this._logger.error('_', `[RCCallSession _joinRoom] _rtcClient.joinRTCRoom throw exception roomId -> ${roomId}`);\n      console.error(error);\n      return { code: RCCallErrorCode.JOIN_ROOM_ERROR };\n    }\n\n    // 房间上注册监听事件\n    this._registerRoomEventListener();\n\n    // 注册房间质量数据监听器\n    this._registerReportListener();\n\n    try {\n      // 订阅远程的流，把远程的流抛给用户\n      await this._subscribeInRoomRemoteTrack();\n    } catch (error) {\n      // 结束通话session\n      this._exceptionClose(RCCallEndReason.SUBSCRIBE_ERROR);\n      this._logger.error('_', `[RCCallSession _joinRoom] _subscribeInRoomRemoteTrack Exception roomId -> ${roomId}`);\n      console.error(error);\n      return { code: RCCallErrorCode.JOIN_ROOM_ERROR };\n    }\n\n    try {\n      // 往房间里发布本地资源\n      await this._publish();\n    } catch (error) {\n      // 结束通话session\n      this._exceptionClose(RCCallEndReason.PUBLISH_ERROR);\n      this._logger.error('_', `[RCCallSession _joinRoom] _publish Exception roomId -> ${roomId}`);\n      console.error(error);\n      return { code: RCCallErrorCode.JOIN_ROOM_ERROR };\n    }\n    return { code: RCCallErrorCode.SUCCESS };\n  }\n\n  /**\n   * (初始化房间的时候) 订阅远程的流，把远程的流抛给用户\n   */\n  private async _subscribeInRoomRemoteTrack() {\n    // 获取所有远程已发布的音视频资源列表\n    const tracks: RCRemoteTrack[] = this._room.getRemoteTracks();\n    if (tracks.length) {\n      const { code } = await this._subscribeRetry(tracks, this._options.isAllowSubscribeRetry, this._RETRYCOUNT);\n      if (code !== RCRTCCode.SUCCESS) {\n        this._exceptionClose(RCCallEndReason.SUBSCRIBE_ERROR);\n        this._logger.error('_', `[RCCallSession _subscribeInRoomRemoteTrack] Resource subscription failed roomId -> ${this._stateMachine.getCallId()} RTC code -> ${code}`);\n      }\n    }\n  }\n\n  /**\n   * 可以重试的订阅\n   * @param params.tracks tracks\n   * @param params.isAllowSubscribeRetry 是否允许重试\n   * @param params.count 允许重试的次数\n   */\n  private async _subscribeRetry(tracks: RCRemoteTrack[], isAllowSubscribeRetry: boolean = false, count: number = 0): Promise<{ code: RCRTCCode }> {\n    const { code } = await this._room.subscribe(tracks);\n    if (code !== RCRTCCode.SUCCESS) {\n      try {\n        this._listener!.onTrackSubscribeFail && this._listener!.onTrackSubscribeFail(code, this);\n      } catch (error) {\n        this._logger.error('_', '[RCCallSession] _listener.onTrackSubscribeFail exception');\n        console.error(error);\n      }\n\n      // 如果不允许重试，直接返回\n      if (!isAllowSubscribeRetry) {\n        return { code };\n      }\n      if (count > 0) {\n        count--;\n        return this._subscribeRetry(tracks, isAllowSubscribeRetry, count);\n      }\n    }\n    return { code };\n  }\n\n  /**\n   * 发布本地资源的逻辑\n   *\n   */\n  private async _publish() {\n    const tracks = this._options.localTracks!;\n    const { code } = await this._publishRetry(tracks, this._options.isAllowPublishRetry, this._RETRYCOUNT);\n\n    // 若资源发布失败\n    if (code !== RCRTCCode.SUCCESS) {\n      this._exceptionClose(RCCallEndReason.PUBLISH_ERROR);\n      this._logger.info('_', `[RCCallSession _publist] Resource publishing failed: roomId -> ${this._stateMachine.getCallId()} RCRTCCode -> ${code}`);\n      return;\n    }\n\n    // 如果是主动发起的呼叫，已提前抛出了资源, 被动呼叫，这里才需要抛出\n    if (this._options.produceType === ProduceTypes.CALLEE) {\n      // 向外抛出本地流, 通知业务层trackReady\n      this._notifyTrackReady(tracks);\n    }\n  }\n\n  /**\n   * 可以重试的发布\n   * @param params.tracks tracks\n   * @param params.isAllowPublishRetry 是否允许重试\n   * @param params.count 允许重试的次数\n   */\n  private async _publishRetry(tracks: RCLocalTrack[], isAllowPublishRetry: boolean = false, count: number = 0): Promise<{ code: RCRTCCode }> {\n    const { code } = await this._room.publish(tracks);\n    if (code !== RCRTCCode.SUCCESS) {\n      try {\n        this._listener!.onTrackPublishFail && this._listener!.onTrackPublishFail(code, this);\n      } catch (error) {\n        this._logger.error('_', '[RCCallSession] _listener.onTrackPublishFail exception');\n        console.error(error);\n      }\n\n      // 如果不允许重试，直接返回\n      if (!isAllowPublishRetry) {\n        return { code };\n      }\n      if (count > 0) {\n        count--;\n        return this._publishRetry(tracks, isAllowPublishRetry, count);\n      }\n    }\n    return { code };\n  }\n\n  /**\n   * 退出房间\n   */\n  private async _leaveRoom() {\n    try {\n      // 退出房间\n      const callBack = await this._rtcClient.leaveRoom(this._room);\n      // 成功退出房间，触发RCCallClient实例上的onSessionClose监听，抛给用户信息\n      this._logger.info('_', `[RCCallSession _leaveRoom] Successfully exited the room code: ${callBack.code}`);\n    } catch (error) {\n      this._logger.error('_', '[RCCallSession _leaveRoom] leaveRoom throw exception');\n      console.error(error);\n    } finally {\n      const summaryInfo = this._stateMachine.getSummary();\n      eventEmitter.emit('sessionClose', { session: this, summaryInfo });\n    }\n  }\n\n  /**\n   * 出现异常后要处理的逻辑,\n   * @param endReason 原因\n   */\n  private _exceptionClose(endReason: RCCallEndReason) {\n    // 销毁本地流\n    this._options.localTracks && this._destroyTracks(this._options.localTracks);\n\n    // 结束状态机\n    this._stateMachine.close(endReason);\n  }\n\n  /**\n   * 用户调用的，注册session上的监听\n   */\n  public registerSessionListener(listener: ISessionListener): void {\n    // 先校验listener, 如果不通过，会trow error\n    const conclusion: IValidationResult = validateListener(listener);\n    if (!conclusion.result) {\n      throw new Error(`[RCCallSession registerSessionListener] ${conclusion.msg}`);\n    }\n    this._listener = { ...listener };\n  }\n\n  /**\n   * 调RTC API 获得本地流\n   */\n  private async _getLocalTrackCore(mediaType: RCCallMediaType, constraints?: IMediaStreamConstraints): Promise<{ code: RCCallErrorCode, tracks?: RCLocalTrack[] }> {\n    // 检测是否能够获得本地流\n    if (mediaType === RCCallMediaType.AUDIO) {\n      const { code, track } = await this._rtcClient.createMicrophoneAudioTrack('RongCloudRTC', constraints && constraints.audio && { ...constraints.audio });\n      if (code !== RCRTCCode.SUCCESS) {\n        this._logger.error('_', `[RCCallSession _getLocalTrackCore] get Audio local tracks failed RCT code -> ${code}`);\n        return { code: RCCallErrorCode.GET_LOCAL_AUDIO_TRACK_ERROR };\n      }\n      this._logger.info('_', '[RCCallSession _getLocalTrackCore] successfully get Audio local tracks');\n      return { code: RCCallErrorCode.SUCCESS, tracks: [track!] };\n    }\n    const { code, tracks } = await this._rtcClient.createMicrophoneAndCameraTracks('RongCloudRTC', constraints && { ...constraints });\n    if (code !== RCRTCCode.SUCCESS) {\n      this._logger.error('_', `[RCCallSession _getLocalTrackCore] get Audio and Video local tracks failed RCT code -> ${code}`);\n      return { code: RCCallErrorCode.GET_LOCAL_AUDIO_AND_VIDEO_TRACK_ERROR };\n    }\n    this._logger.info('_', '[RCCallSession _getLocalTrackCore] successfully get audio and video local tracks');\n    return { code: RCCallErrorCode.SUCCESS, tracks };\n  }\n\n  private async _getLocalTrack(mediaType: RCCallMediaType, constraints?: IMediaStreamConstraints): Promise<{ code: RCCallErrorCode, tracks?: RCLocalTrack[] }> {\n    // 并且是获得音视频, 并且 （如果获得音视频不成功，允许降级获得音频）\n    if (this._options.isAllowDemotionGetStream && mediaType === RCCallMediaType.AUDIO_VIDEO) {\n      const { code, tracks } = await this._getLocalTrackCore(RCCallMediaType.AUDIO_VIDEO, constraints);\n\n      // 如果音视频不能获得，就降级获得音频\n      if (code !== RCCallErrorCode.SUCCESS) {\n        const { code, tracks } = await this._getLocalTrackCore(RCCallMediaType.AUDIO, constraints);\n        if (code !== RCCallErrorCode.SUCCESS) {\n          // 获取资源失败，需要调状态机state 为 end\n          this._exceptionClose(RCCallEndReason.GET_MEDIA_RESOURCES_ERROR);\n          return { code };\n        }\n        return { code, tracks };\n      }\n      return { code, tracks };\n    }\n    const { code: _code, tracks } = await this._getLocalTrackCore(mediaType, constraints);\n    if (_code !== RCCallErrorCode.SUCCESS) {\n      // 获取资源失败，需要调状态机state 为 end\n      this._exceptionClose(RCCallEndReason.GET_MEDIA_RESOURCES_ERROR);\n\n      return { code: _code };\n    }\n    return { code: _code, tracks };\n  }\n\n  /**\n   * 通话中更换音频设备\n   */\n  public async changeAudioDevice(audioConstraints?: IMicphoneAudioProfile): Promise<{ code: RCCallErrorCode }> {\n    // 新设备的track\n    const recentTracks: RCLocalTrack[] = [];\n\n    // 整理后的本地track\n    const localTracks: RCLocalTrack[] = [];\n    const { code, track } = await this._rtcClient.createMicrophoneAudioTrack('RongCloudRTC', audioConstraints);\n    if (code !== RCRTCCode.SUCCESS) {\n      this._logger.error('_', `[RCCallSession changeDevice] get local Audio tracks failed RCTLib code -> ${code}`);\n      return { code: RCCallErrorCode.GET_LOCAL_AUDIO_TRACK_ERROR };\n    }\n\n    this._options.localTracks && this._options.localTracks.forEach((track: RCLocalTrack) => {\n      if (track.isAudioTrack()) {\n        // 之前的音频都销毁，（RTCLib SDK内部会在 addLocalTrack 清理同类型轨道数据，所以这里注释掉）\n        // track.destroy()\n      } else {\n        // 只把之前的视频留下\n        localTracks.push(track);\n      }\n    });\n\n    recentTracks.push(track!);\n\n    // 加上本地新产生的音频\n    localTracks.push(track!);\n    this._options.localTracks = localTracks;\n    // 通知业务层trackReady\n    this._notifyTrackReady(recentTracks);\n\n    // 如果当前已加入房间，发布新流\n    if (this._room) {\n      // 发布新流\n      const { code } = await this._room.publish(recentTracks);\n      if (code !== RCRTCCode.SUCCESS) {\n        return { code: RCCallErrorCode.AUDIO_PUBLISH_ERROR };\n      }\n    }\n    return { code: RCCallErrorCode.SUCCESS };\n  }\n\n  /**\n   * 群呼叫中继续邀请\n   * @param userIds 被邀请用户 ID 列表\n   * @param options.extra 消息的扩展信息\n   * @deprecated 5.1.2 废弃 options.pushTitle 通知的标题\n   * @deprecated 5.1.2 废弃 options.pushContent 通知内容\n   */\n  public async invite(userIds: string[], options: IInviteOptions = {}): Promise<{ code: RCCallErrorCode }> {\n    const { extra = '' } = options;\n    const pushConfig = this._options.callPushConfig ? this._options.callPushConfig : options.pushConfig;\n    const conclusion: IValidationResult[] = [validateUserIds(userIds), validateExtra(extra)];\n    if (pushConfig) {\n      const { code, errorMsg } = validatePushConfig(pushConfig);\n      if (code !== RCCallErrorCode.SUCCESS) {\n        this._logger.error('_', `[RCCallSession invite] param error, errorMsg: ${errorMsg}`);\n        return { code };\n      }\n    }\n    const messages: string[] = [];\n    const result = conclusion.every((obj: IValidationResult) => {\n      !obj.result && messages.push(obj.msg!);\n      return obj.result;\n    });\n\n    if (!result) {\n      throw new Error(`[RCCallClient invite] ${messages.join('\\n')}`);\n    }\n\n    const { code } = await this._stateMachine.invite(userIds, {\n      extra, pushConfig,\n    });\n    return { code };\n  }\n\n  /**\n   * 同意接听\n   */\n  public async accept(constraints?: IMediaStreamConstraints): Promise<{ code: RCCallErrorCode }> {\n    const conclusion: IValidationResult = validateMediaStreamConstraints(constraints!);\n    if (!conclusion.result) {\n      throw new Error(`[RCCallSession accept] ${conclusion.msg}`);\n    }\n\n    // 接听之前，先挂断当前之外的session，现阶段不允许用户先择接听session，事先会在状态机内部挂断，这里抛出去，会清理其它的seesion\n    eventEmitter.emit('hungupOtherSession', { session: this });\n    const mediaType = this._stateMachine.getMediaType();\n    const { code: _code, tracks } = await this._getLocalTrack(mediaType, constraints);\n    if (_code !== RCCallErrorCode.SUCCESS) {\n      return { code: _code };\n    }\n    this._options.localTracks = tracks;\n\n    // 发送接听的消息\n    const { code } = await this._stateMachine.accept();\n    if (code !== RCCallErrorCode.SUCCESS) {\n      this._logger.error('_', `[RCCallSession accept]Send accept message failed -> code: ${code}`);\n      return { code };\n    }\n    return { code };\n  }\n\n  /**\n   * 挂断\n   */\n  public async hungup(): Promise<{ code: RCCallErrorCode; }> {\n    // const summaryInfo = this._stateMachine.getSummary()\n    // eventEmitter.emit('sessionClose', { session: this, summaryInfo })\n    return this._stateMachine.hungup();\n  }\n\n  /**\n   * 通话媒体变更\n   *  @param mediaType RCCallMediaType.AUDIO 改为音频通话 | RCCallMediaType.AUDIO_VIDEO 改为音视频通话\n   */\n  public async _changeMediaType(mediaType: RCCallMediaType): Promise<{ code: RCCallErrorCode }> {\n    const { code } = await this._stateMachine.changeMediaType(mediaType);\n    if (code !== RCCallErrorCode.SUCCESS) {\n      this._logger.error('_', `[RCCallSession _changeMediaType] change media type fail code-> ${code}`);\n    }\n    return { code };\n  }\n\n  /**\n   * 获得本地视频\n   */\n  private _getLocalVideoTracks(): RCLocalTrack[] {\n    let localVideoTracks: RCLocalTrack[] = [];\n    if (!this._room) {\n      return localVideoTracks;\n    }\n    if (this._options.localTracks) {\n      localVideoTracks = this._options.localTracks.filter((track) => track.isVideoTrack());\n    }\n    return localVideoTracks;\n  }\n\n  /**\n   * 获得本地音频\n   */\n  private _getLocalAudioTracks(): RCLocalTrack[] {\n    let localAudiotracks: RCLocalTrack[] = [];\n    if (!this._room) {\n      return localAudiotracks;\n    }\n    if (this._options.localTracks) {\n      localAudiotracks = this._options.localTracks.filter((track) => track.isAudioTrack());\n    }\n    return localAudiotracks;\n  }\n\n  /**\n   * 把通话的MediaType升级到音视频\n   */\n  private async _setMediaTypeToAudioAndVideo() {\n    // 获得本端视频资源\n    const { code, track } = await this._rtcClient.createCameraVideoTrack();\n    if (code !== RCRTCCode.SUCCESS) {\n      return { code: RCCallErrorCode.GET_LOCAL_AUDIO_AND_VIDEO_TRACK_ERROR };\n    }\n\n    // 发布本端视频资源\n    const { code: _code } = await this._room.publish([track!]);\n\n    // 若资源发布失败\n    if (_code !== RCRTCCode.SUCCESS) {\n      this._logger.error('_', `[RCCallSession _enableVideo] Resource publishing failed: RCRTCCode -> ${code}`);\n      return;\n    }\n\n    // 通知业务层trackReady\n    this._notifyTrackReady([track!]);\n\n    // 发消息\n    this._changeMediaType(RCCallMediaType.AUDIO_VIDEO);\n  }\n\n  /**\n   * 把通话的MediaType降级到音频\n   * @param isSendMesssage 是否需要发消息, 默认发消息\n   */\n  private async _setMediaTypeToAudio() {\n    const tracks: RCLocalTrack[] = this._getLocalVideoTracks();\n    if (tracks.length) {\n      // 禁用视频\n      tracks.forEach((track: RCLocalTrack) => {\n        track.mute();\n      });\n\n      // 取消发布视频\n      const { code } = await this._room.unpublish(tracks);\n      if (code !== RCRTCCode.SUCCESS) {\n        this._logger.error('_', `[RCCallSession disableVideo] unpublish failed -> ${code}`);\n      }\n\n      // 关闭摄像头\n      this._destroyTracks(tracks);\n    }\n  }\n\n  /**\n   * 通话降级，目前需求只做通话降级，音视频可以降级为音频，音频不能升到音视频, 发消息成功才算降级成功\n   *\n   */\n  public async descendAbility(): Promise<{ code: RCCallErrorCode }> {\n    const { code } = await this._changeMediaType(RCCallMediaType.AUDIO);\n    if (code === RCCallErrorCode.SUCCESS) {\n      this._setMediaTypeToAudio();\n    }\n    return { code };\n  }\n\n  /**\n   * 禁用视频track\n   */\n  public async disableVideoTrack(): Promise<{ code: RCCallErrorCode }> {\n    if (!this._room) {\n      this._logger.error('_', `[RCCallSession disableAudioTrack] Room missing audio track -> ${RCCallErrorCode.NOT_IN_ROOM_ERROR}`);\n      return { code: RCCallErrorCode.NOT_IN_ROOM_ERROR };\n    }\n\n    const tracks: RCLocalTrack[] = this._getLocalVideoTracks();\n    if (!tracks.length) {\n      this._logger.error('_', `[RCCallSession disableVideoTrack] Room missing video track -> ${RCCallErrorCode.MISSING_VIDEO_TRACK_ERROR}`);\n      return { code: RCCallErrorCode.MISSING_VIDEO_TRACK_ERROR };\n    }\n\n    // 禁用视频\n    tracks.forEach((track: RCLocalTrack) => {\n      track.mute();\n    });\n\n    // 如果不需关闭摄像头\n    if (!this._options.isOffCameraWhenVideoDisable) {\n      return { code: RCCallErrorCode.SUCCESS };\n    }\n\n    // 取消发布视频\n    const { code } = await this._room.unpublish(tracks);\n    if (code !== RCRTCCode.SUCCESS) {\n      this._logger.error('_', `[RCCallSession disableVideo] unpublish failed -> ${code}`);\n\n      return { code: RCCallErrorCode.UNPUBLISH_VIDEO_ERROR };\n    }\n\n    tracks.forEach((track: RCLocalTrack) => {\n      // 关闭摄像头\n      track.destroy();\n    });\n\n    return { code: RCCallErrorCode.SUCCESS };\n  }\n\n  /**\n   * 启用视频track\n   */\n  public async enableVideoTrack(): Promise<{ code: RCCallErrorCode }> {\n    if (!this._room) {\n      this._logger.error('_', `[RCCallSession disableAudioTrack] Room missing audio track -> ${RCCallErrorCode.NOT_IN_ROOM_ERROR}`);\n      return { code: RCCallErrorCode.NOT_IN_ROOM_ERROR };\n    }\n\n    // 如果不需关闭摄像头\n    if (!this._options.isOffCameraWhenVideoDisable) {\n      const tracks: RCLocalTrack[] = this._getLocalVideoTracks();\n      if (!tracks.length) {\n        this._logger.error('_', `[RCCallSession EnableVideoTrack] Room missing video track -> ${RCCallErrorCode.MISSING_VIDEO_TRACK_ERROR}`);\n        return { code: RCCallErrorCode.MISSING_VIDEO_TRACK_ERROR };\n      }\n\n      // 启用视频\n      tracks.forEach((track: RCLocalTrack) => {\n        track.unmute();\n      });\n      return { code: RCCallErrorCode.SUCCESS };\n    }\n\n    // 获得本端视频资源\n    const { code, track } = await this._rtcClient.createCameraVideoTrack();\n    if (code !== RCRTCCode.SUCCESS) {\n      this._logger.error('_', `[RCCallSession EnableVideoTrack] Get Resource failed: RCRTCCode -> ${code}`);\n      return { code: RCCallErrorCode.GET_LOCAL_VIDEO_TRACK_ERROR };\n    }\n    const localTracks: RCLocalTrack[] = [];\n    this._options.localTracks && this._options.localTracks.forEach((track: RCLocalTrack) => {\n      if (track.isVideoTrack()) {\n        // 之前的视频都销毁\n        track.destroy();\n      } else {\n        // 只留下之前的音频\n        localTracks.push(track);\n      }\n    });\n\n    // 加上本地新产生的视频\n    localTracks.push(track!);\n    this._options.localTracks = localTracks;\n\n    // 为了触发对方的onVideoMuteChange 先禁用\n    track!.mute();\n\n    // 发布本端视频资源\n    const { code: _code } = await this._room.publish([track!]);\n\n    // 若资源发布失败\n    if (_code !== RCRTCCode.SUCCESS) {\n      this._logger.error('_', `[RCCallSession EnableVideoTrack] Resource publishing failed: RCRTCCode -> ${code}`);\n      return { code: RCCallErrorCode.VIDEO_PUBLISH_ERROR };\n    }\n\n    // 启用\n    track!.unmute();\n\n    // 通知业务层trackReady\n    this._notifyTrackReady([track!]);\n    return { code: RCCallErrorCode.SUCCESS };\n  }\n\n  /**\n   * 禁用音频track\n   */\n  public async disableAudioTrack() {\n    if (!this._room) {\n      this._logger.error('_', `[RCCallSession disableAudioTrack] Room missing audio track -> ${RCCallErrorCode.NOT_IN_ROOM_ERROR}`);\n      return { code: RCCallErrorCode.NOT_IN_ROOM_ERROR };\n    }\n\n    const tracks: RCLocalTrack[] = this._getLocalAudioTracks();\n\n    // 禁用音频\n    tracks.forEach((track: RCLocalTrack) => {\n      track.mute();\n    });\n  }\n\n  /**\n   * 启用音频track\n   */\n  public async enableAudioTrack() {\n    if (!this._room) {\n      this._logger.error('_', `[RCCallSession disableAudioTrack] Room missing audio track -> ${RCCallErrorCode.NOT_IN_ROOM_ERROR}`);\n      return { code: RCCallErrorCode.NOT_IN_ROOM_ERROR };\n    }\n\n    const tracks: RCLocalTrack[] = this._getLocalAudioTracks();\n\n    if (!tracks.length) {\n      this._logger.error('_', `[RCCallSession disableAudioTrack] Room missing audio track -> ${RCCallErrorCode.MISSING_VIDEO_TRACK_ERROR}`);\n      return { code: RCCallErrorCode.MISSING_VIDEO_TRACK_ERROR };\n    }\n\n    // 启用音频\n    tracks.forEach((track: RCLocalTrack) => {\n      track.unmute();\n    });\n  }\n\n  /**\n   * 销毁本地流\n   */\n  private _destroyTracks(tracks: RCLocalTrack[]) {\n    tracks.forEach((track: RCLocalTrack) => {\n      track.destroy();\n    });\n  }\n\n  /**\n   * 向外抛出本地流\n   */\n  private _notifyTrackReady(tracks: RCLocalTrack[] | RCRemoteTrack[]) {\n    tracks.forEach((track: RCLocalTrack | RCRemoteTrack) => {\n      try {\n        this._listener!.onTrackReady(track, this);\n      } catch (error) {\n        this._logger.error('_', '[RCCallSession _notifyTrackReady] _listener onTrackReady exception');\n        console.error(error);\n      }\n    });\n  }\n\n  /**\n   * 房间上注册事件\n   */\n  private _registerRoomEventListener() {\n    this._room.registerRoomEventListener(\n      {\n        /**\n         * 本端被踢出房间时触发\n         * @description 被踢出房间可能是由于服务端超出一定时间未能收到 rtcPing 消息，所以认为己方离线。\n         * 另一种可能是己方 rtcPing 失败次数超出上限，故而主动断线\n         * @param byServer\n         * 当值为 false 时，说明本端 rtcPing 超时\n         * 当值为 true 时，说明本端收到被踢出房间通知\n         */\n        onKickOff: (byServer: boolean, state?: RCKickReason | undefined) => {\n          const currentUserId: string = this._rtcClient.getCurrentId();\n          this._stateMachine.userLeave([currentUserId]);\n\n          if (!byServer) {\n            this._exceptionClose(RCCallEndReason.NETWORK_ERROR);\n          } else {\n            if (state === RCKickReason.SERVER_KICK) {\n              this._exceptionClose(RCCallEndReason.KICKED_BY_SERVER);\n            }\n            if (state === RCKickReason.OTHER_KICK) {\n              this._exceptionClose(RCCallEndReason.OTHER_CLIENT_JOINED_CALL);\n            }\n          }\n        },\n        /**\n         * 接收到房间信令时回调，用户可通过房间实例的 `sendMessage(name, content)` 接口发送信令\n         * @param name 信令名\n         * @param content 信令内容\n         * @param senderUserId 发送者 Id\n         * @param messageUId 消息唯一标识\n         */\n        onMessageReceive(name: string, content: any, senderUserId: string, messageUId: string) {\n        },\n        /**\n         * 监听房间属性变更通知\n         * @param name\n         * @param content\n         */\n        onRoomAttributeChange(name: string, content: string) {\n        },\n        /**\n         * 发布者禁用/启用音频\n         * @param audioTrack RCRemoteAudioTrack 类实例\n         */\n        onAudioMuteChange: (audioTrack: RCRemoteAudioTrack) => {\n          this._logger.info('_', `[RCCallSession onAudioMuteChange] userId->${audioTrack.getUserId()} muted -> ${audioTrack.isOwnerMuted()}`);\n          const muteUser: IMuteUser = {\n            userId: audioTrack.getUserId(),\n            muted: audioTrack.isOwnerMuted(),\n            kind: 'audio',\n            trackId: audioTrack.getTrackId(),\n          };\n          try {\n            // 通知给业务\n            this._listener!.onAudioMuteChange(muteUser, this);\n          } catch (error) {\n            this._logger.error('_', '[RCCallSession onAudioMuteChange] Missing listening method -> onTrackMuteChange');\n            console.error(error);\n          }\n        },\n        /**\n         * 发布者禁用/启用视频\n         * @param videoTrack RCRemoteVideoTrack 类实例对象\n         */\n        onVideoMuteChange: (videoTrack: RCRemoteVideoTrack) => {\n          this._logger.info('_', `[RCCallSession onVideoMuteChange]userId->${videoTrack.getUserId()} muted -> ${videoTrack.isOwnerMuted()}`);\n          const muteUser: IMuteUser = {\n            userId: videoTrack.getUserId(),\n            muted: videoTrack.isOwnerMuted(),\n            kind: 'video',\n            trackId: videoTrack.getTrackId(),\n          };\n\n          try {\n            // 通知给业务\n            this._listener!.onVideoMuteChange(muteUser, this);\n          } catch (error) {\n            this._logger.error('_', '[RCCallSession onVideoMuteChange] Missing listening method -> onVideoMuteChange');\n            console.error(error);\n          }\n        },\n        /**\n         * 房间内其他用户新发布资源时触发\n         * 如需获取加入房间之前房间内某个用户发布的资源列表，可使用 room.getRemoteTracksByUserId('userId') 获取\n         * @param tracks 新发布的音轨与视轨数据列表，包含新发布的 RCRemoteAudioTrack 与 RCRemoteVideoTrack 实例\n         */\n        onTrackPublish: async (tracks: RCRemoteTrack[]) => {\n          // 退出房间后，还会走到这？？，所以判断一下，没有room不执行订阅\n          if (this._room) {\n            // 按业务需求选择需要订阅资源，通过 room.subscribe 接口进行订阅\n            const { code } = await this._room.subscribe(tracks);\n            if (code !== RCRTCCode.SUCCESS) {\n              this._logger.error('_', `[RCCallSession onTrackPublish] subscribe failed RTCCode ->${code}`);\n            }\n          }\n        },\n        /**\n         * 房间用户取消发布资源\n         * @param tracks 被取消发布的音轨与视轨数据列表\n         * @description 当资源被取消发布时，SDK 内部会取消对相关资源的订阅，业务层仅需处理 UI 业务\n         */\n        onTrackUnpublish: (tracks: RCRemoteTrack[]) => {\n\n        },\n        /**\n         * 订阅的音视频流通道已建立, track 已可以进行播放\n         * @param track RCRemoteTrack 类实例\n         */\n        onTrackReady: (track: RCRemoteTrack) => {\n          const mediaType = this._stateMachine.getMediaType();\n\n          // 有时对方没有降级成功，扔抛过来视频，这时的视频不对外抛出\n          if (mediaType === RCCallMediaType.AUDIO && track.isVideoTrack()) {\n            return;\n          }\n\n          // 执行用户的onTrackReady监听\n          this._notifyTrackReady([track]);\n        },\n        /**\n         * 人员加入\n         * @param userIds 加入的人员 id 列表\n         */\n        onUserJoin: (userIds: string[]) => {\n          // 有人加入清除定时器\n          if (this.joinRoomTimer) {\n            this.joinRoomTimer.stop();\n          }\n          this._stateMachine.userJoin(userIds);\n        },\n        /**\n         * 人员退出\n         * @param userIds\n         */\n        onUserLeave: (userIds: string[]) => {\n          this._logger.info('_', `[RCCallSession onUserLeave] listening onUserLeave userIds -> ${userIds?.join(',')}`);\n          this._stateMachine.userLeave(userIds);\n        },\n        /**\n         * RTC 每次 Ping 结果\n         */\n        onPing: (result: RCRTCPingResult) => {\n          this._logger.info('_', `[RCCallSession onPing]${result}`);\n          try {\n            // 通知给业务\n            this._listener!.onPing && this._listener!.onPing(result, this);\n          } catch (error) {\n            this._logger.error('_', '[RCCallSession onPing] listening onPing exception');\n            console.error(error);\n          }\n        },\n      },\n    );\n  }\n\n  /**\n   * 注册房间质量数据监听器\n   */\n  private _registerReportListener() {\n    // 注册房间质量数据监听器\n    this._room.registerReportListener({\n      /**\n       * 用于接收状态数据报告\n       * @param report\n       */\n      onStateReport: (report: IRCRTCStateReport) => {\n        try {\n          this._listener!.onRTCStateReport && this._listener!.onRTCStateReport(report, this);\n        } catch (error) {\n          this._logger.error('_', '[RCCallSession onStateReport] listener onStateReport exception');\n          console.error(error);\n        }\n      },\n\n      /**\n       * ~ICE 连接状态变更通知~\n       * @since version 5.1.5\n       */\n      onICEConnectionStateChange: (state: RTCIceConnectionState) => {\n        try {\n          this._listener!.onICEConnectionStateChange && this._listener!.onICEConnectionStateChange(state, this);\n        } catch (error) {\n          this._logger.error('_', '[RCCallSession onICEConnectionStateChange] onICEConnectionStateChange exception');\n          console.error(error);\n        }\n      },\n\n    });\n  }\n\n  /**\n   *  通话唯一标识\n   */\n  public getSessionId(): string {\n    return this._stateMachine.getCallId();\n  }\n\n  /**\n   *  获取房间当前会话 Id，当房间内已无成员时房间会回收，重新加入时 sessionId 将更新，(用户录制资源用的)\n   */\n  public getRTCSessionId(): string | null {\n    return this._room ? this._room.getSessionId() : null;\n  }\n\n  /**\n   *  目标 ID，单呼对方人员 Id, 群呼群组 Id\n   */\n  public getTargetId(): string {\n    return this._stateMachine.getTargetId();\n  }\n\n  /**\n   *  获取会话类型\n   */\n  public getConversationType(): ConversationType {\n    return this._stateMachine.getConversationType();\n  }\n\n  /**\n   *  组织 ID\n   */\n  public getChannelId(): string {\n    return this._stateMachine.getChannelId();\n  }\n\n  /**\n   * 房间人员列表，不包含本端信息\n   */\n  public getRemoteUsers(): IUserData[] {\n    return this._stateMachine.getRemoteUsers();\n  }\n\n  /**\n * 房间人员列表，不包含本端信息\n */\n  public getUsers(): IUserData[] {\n    return this._stateMachine.getRemoteUsers();\n  }\n\n  /**\n   * 获取人员状态\n   */\n  public getUserState(userId: string): RCCallUserState {\n    if (!userId || typeof userId !== 'string') {\n      throw new Error('userId is required, must be of type \\'string\\'');\n    }\n    return this._stateMachine.getUserState(userId);\n  }\n\n  /**\n   * 获取session状态\n   */\n  public getState(): RCCallSessionState {\n    return this._stateMachine.getState();\n  }\n\n  /**\n   * 获得会话发起者id\n   */\n  public getCallerId(): string {\n    return this._stateMachine.getCallerId();\n  }\n\n  /**\n   * 获得mediaType\n   */\n  public getMediaType(): RCCallMediaType {\n    return this._stateMachine.getMediaType();\n  }\n}\n","import {\n  RTCPluginContext, IRuntime, IRTCJoinedInfo, ErrorCode, RTCJoinType, ILogger, BasicLogger,\n} from '@rongcloud/engine';\nimport {\n  RCCallEngine, RCCallStateMachine, RCCallErrorCode, RCCallLanguage, RCCallMediaType, IOfflineRecord,\n} from '@rc-embed/call-engine';\nimport { RCRTCClient, RCRTCCode, RCLocalTrack } from '@rongcloud/plugin-rtc';\nimport {\n  IRCCallInGroupParams, IRCCallInitOptions, IRCCallParams, IMediaStreamConstraints, IValidationResult, IPushConfig,\n} from './interface';\nimport { ProduceTypes } from './enums';\nimport eventEmitter from './eventEmitter';\nimport { RCCallSession } from './RCCallSession';\nimport {\n  validateListener, validateTargetId, validateMediaType, validateUserIds, validateExtra, validatePushConfig,\n} from './validation';\n\nexport default class RCCallClient {\n  /**\n   * rtc实例\n   */\n  private readonly _rtcClient: RCRTCClient\n\n  /**\n   * callEngine层实例\n   */\n  private readonly _callEngine: RCCallEngine\n\n  /**\n   * 其它参数\n   */\n  private _options: IRCCallInitOptions\n\n  /**\n   * session列表\n   */\n  private _sessionList: RCCallSession[] = []\n\n  /**\n   * 移动端呼叫推送配置\n   */\n  private _callPushConfig?: IPushConfig\n\n  /**\n   * 移动端挂断推送配置\n   */\n  private _hungupPushConfig?: IPushConfig\n\n  constructor(\n    private _context: RTCPluginContext,\n    private readonly _runtime: IRuntime,\n    private readonly _logger: BasicLogger,\n    _options: IRCCallInitOptions,\n  ) {\n    this._rtcClient = _options.rtcClient;\n\n    this._options = { /**\n       * 是否允许发布重试， 默认不允许\n       */\n      isAllowPublishRetry: false,\n\n      /**\n       * 是否允许订阅重试，默认不允许\n       */\n      isAllowSubscribeRetry: false,\n      /**\n       * 禁用视频时关摄像头, 默认关闭\n       */\n      isOffCameraWhenVideoDisable: true,\n      /**\n       * RTC 房间加入类型，默认   RTCJoinType.COEXIST = 2 两个设备共存\n       *     RTCJoinType.KICK = 0,踢前一个设备\n       *     RTCJoinType.REFUSE = 1,当前加入拒绝\n       *     RTCJoinType.COEXIST = 2 两个设备共存\n       */\n      joinType: RTCJoinType.COEXIST,\n\n      /**\n       * 允许降级获得流，获得音视频不成功 ，降级获得音频, 默认不允许\n       */\n      isAllowDemotionGetStream: false,\n\n      /**\n       * 语言设置 (推送), 不传默认为中文\n       */\n      lang: RCCallLanguage.ZH,\n      ..._options,\n    };\n\n    // 初始化callEngine, 并监听onInvite\n    this._callEngine = new RCCallEngine(this._context, _runtime, this._logger, {\n\n      /**\n       * 监听收到invite\n       */\n      onInvite: this._onInvite.bind(this),\n\n      /**\n       * 监听离线消息报告\n       */\n      onOfflineRecord: this._onOfflineRecord.bind(this),\n    }, {\n\n      /**\n       * 语言设置 (推送), 不传默认为中文\n       */\n      lang: this._options.lang || RCCallLanguage.ZH,\n\n    });\n\n    eventEmitter.on('sessionClose', ({ session, summaryInfo }) => {\n      // 从sessionList去掉这个关闭的session\n      this._removeSession(session);\n\n      try {\n        this._options.onSessionClose(session, summaryInfo);\n      } catch (error) {\n        this._logger.error('_', '[RCCCallClient] options.onSessionClose exception');\n        console.log(error);\n      }\n    });\n\n    // 接听之前挂断其它的session\n    eventEmitter.on('hungupOtherSession', ({ session }) => {\n      const id = session.getSessionId();\n      this._logger.info('_', `[RCCallClient hungupOtherSession] sessionId ready to accept -> ${id}`);\n      this._logger.info('_', `[RCCallClient hungupOtherSession] sessionList ->${this._sessionList.map((ses) => ses.getSessionId()).join(',')}`);\n      let i = 0;\n      while (this._sessionList.length > 1) {\n        // 如果与要接听的session不一致\n        if (this._sessionList[i].getSessionId() !== id) {\n          // 挂断\n          this._sessionList[i].hungup();\n\n          // 挂断后删除\n          this._sessionList.splice(i, 1);\n        } else {\n          // 如果是要接听的session，跳过这个索引，所以加1\n          i++;\n        }\n      }\n      this._logger.info('_', `[RCCallClient hungupOtherSession] current sessionList length ->${this._sessionList.length}`);\n    });\n  }\n\n  /**\n   * 监听onInvite\n   */\n  private _onInvite(stateMachine: RCCallStateMachine, extra?: string) {\n    this._logger.info('_', '[RCCallClient _onInvite] Received invite message');\n    const session = new RCCallSession(stateMachine, this._rtcClient, this._logger, {\n\n      // 是否允许订阅重试\n      isAllowSubscribeRetry: this._options.isAllowSubscribeRetry,\n\n      // 是否允许发布重试\n      isAllowPublishRetry: this._options.isAllowPublishRetry,\n\n      /**\n       * 禁用视频时关摄像头\n       */\n      isOffCameraWhenVideoDisable: this._options.isOffCameraWhenVideoDisable,\n\n      /**\n       * RTC 房间加入类型\n       */\n      joinType: this._options.joinType,\n\n      // 允许降级获得流，获得音视频不成功 ，降级获得音频, 默认不允许\n      isAllowDemotionGetStream: this._options.isAllowDemotionGetStream,\n\n      // 标明是被叫产生的session\n      produceType: ProduceTypes.CALLEE,\n\n      callPushConfig: this._callPushConfig,\n\n      hungupPushConfig: this._hungupPushConfig,\n    });\n    this._logger.info('_', '[RCCallClient _onInvite] Received invite message, successfully created session');\n\n    /**\n     * 如果通话的时候不允许接听新的通话，直接挂断， 这些工作在callEngine里完成\n     */\n    this._sessionList.push(session);\n\n    try {\n      // 执行用户API的监听\n      this._options.onSession(session, extra);\n    } catch (error) {\n      this._logger.error('_', '[RCCallClient _options.onSession] onSession exception');\n      console.log(error);\n    }\n\n    // 必须在onSession里注册session监听事件，这里检测一下有没有注册\n    if (session._listener) {\n      const conclusion: IValidationResult = validateListener(session._listener);\n      if (!conclusion.result) {\n        throw new Error(conclusion.msg);\n      }\n    } else {\n      this._logger.error('_', '[RCCallClient _options.onSession] session Must Have Listener');\n      throw new Error('[RCCallSession  _options.onSession] session Must Have Listener');\n    }\n  }\n\n  /**\n   * 监听离线消息报告\n   * @param record\n   */\n  public _onOfflineRecord(record: IOfflineRecord) {\n    try {\n      // 执行用户API的监听\n      this._options.onOfflineRecord && this._options.onOfflineRecord(record);\n    } catch (error) {\n      this._logger.error('_', '[RCCallClient _options.onOfflineRecord] onOfflineRecord exception');\n      console.log(error);\n    }\n  }\n\n  /**\n   * 注册用户信息。注册后，在发起邀请或挂断等操作时，会将该信息一并发送给对端\n   * @param info.name        用户名称\n   * @param info.portraitUri 用户头像信息\n   * @param info.extra       预留拓展字段\n   */\n  public registerUserInfo(info: { name?: string; portraitUri?: string; extra?: string; } = {}) {\n    this._callEngine.registerUserInfo(info);\n    this._logger.info('_', '[RCCallClient registerUserInfo] successfully register user info data');\n  }\n\n  /**\n  * 跨App单呼，\b发送invite消息，回调回来接收stateMachine, 建session\n  * @param params.targetId 被呼叫一方的用户 id 必填\n  * @param params.mediaType 音频呼叫 or 音视频呼叫  必填\n  * @param params.listener (session上的监听) 必填\n  * @param params.constraints 获取音频或音视频资源时的参数 可选\n  * @param params.channelId 组织 Id 可选\n  * @param params.extra 消息扩展信息\n  * @deprecated 5.1.2 版本废弃 params.pushTitle 通知的标题\n  * @deprecated 5.1.2 版本废弃 params.pushContent 通知的内容\n  * @param params.bitrate 需要设置的码率参数\n  *\n  */\n  public async startCrossCall({\n    targetId, mediaType = RCCallMediaType.AUDIO, listener, constraints, channelId = '', extra = '', pushTitle = '', pushContent = '', bitrate,\n  }: IRCCallParams): Promise<{ code: RCCallErrorCode, session?: RCCallSession }> {\n    return this.__call({\n      targetId, mediaType, listener, constraints, channelId, extra, pushTitle, pushContent, bitrate, isCrossAppkey: true,\n    });\n  }\n\n  /**\n   * 单呼，\b发送invite消息，回调回来接收stateMachine, 建session\n   * @param params.targetId 被呼叫一方的用户 id 必填\n   * @param params.mediaType 音频呼叫 or 音视频呼叫  必填\n   * @param params.listener (session上的监听) 必填\n   * @param params.constraints 获取音频或音视频资源时的参数 可选\n   * @param params.channelId 组织 Id 可选\n   * @param params.extra 消息扩展信息\n   * @deprecated 5.1.2 版本废弃 params.pushTitle 通知的标题\n   * @deprecated 5.1.2 版本废弃 params.pushContent 通知的内容\n   * @param params.bitrate 需要设置的码率参数\n   *\n   */\n  public async call({\n    targetId, mediaType = RCCallMediaType.AUDIO, listener, constraints, channelId = '', extra = '', pushTitle = '', pushContent = '', bitrate,\n  }: IRCCallParams): Promise<{ code: RCCallErrorCode, session?: RCCallSession }> {\n    return this.__call({\n      targetId, mediaType, listener, constraints, channelId, extra, pushTitle, pushContent, bitrate,\n    });\n  }\n\n  private async __call({\n    targetId, mediaType = RCCallMediaType.AUDIO, listener, constraints, channelId = '', extra = '', pushTitle = '', pushContent = '', bitrate, isCrossAppkey = false,\n  }: IRCCallParams): Promise<{ code: RCCallErrorCode, session?: RCCallSession }> {\n    const pushConfig = this._callPushConfig ? this._callPushConfig : { pushTitle, pushContent };\n    this._logger.info('_', `[RCCallClient call] extra->${extra} pushConfig->${JSON.stringify(pushConfig)}`);\n    const conclusion: IValidationResult[] = [validateTargetId(targetId), validateMediaType(mediaType), validateListener(listener), validateExtra(extra)];\n    const messages: string[] = [];\n    const result = conclusion.every((obj: IValidationResult) => {\n      !obj.result && messages.push(obj.msg!);\n      return obj.result;\n    });\n    if (!result) {\n      throw new Error(`[RCCallClient call] ${messages.join('\\n')}`);\n    }\n\n    const { code: validatePushCode, errorMsg } = validatePushConfig(pushConfig);\n    if (validatePushCode !== RCCallErrorCode.SUCCESS) {\n      this._logger.error('_', `[RCCallClient call] param error, errorMsg: ${errorMsg}`);\n      return { code: validatePushCode };\n    }\n\n    let localTracks: RCLocalTrack[] = [];\n\n    const { code: _code, tracks } = await this._getLocalTrack(mediaType, constraints);\n    if (_code !== RCCallErrorCode.SUCCESS) {\n      return { code: _code };\n    }\n    localTracks = tracks!;\n\n    localTracks.forEach((track) => {\n      // 设置码率\n      if (track.isAudioTrack() && bitrate?.audio) {\n        track.setBitrate(bitrate?.audio);\n      }\n      if (track.isVideoTrack() && bitrate?.video) {\n        track.setBitrate(bitrate?.video?.max, bitrate?.video?.min, bitrate?.video?.start);\n      }\n      // 向外抛出本地流\n      listener.onTrackReady(track);\n    });\n\n    // 调用callEngine的call返回一个状态机的实例\n    const { code, stateMachine } = await this._callEngine.call(channelId, targetId, mediaType, extra, pushConfig, isCrossAppkey);\n    if (code === RCCallErrorCode.SUCCESS && stateMachine) {\n      this._logger.info('_', '[RCCallClient call] successfully created state machine');\n      const session = new RCCallSession(stateMachine, this._rtcClient, this._logger, {\n        localTracks,\n\n        // 是否允许订阅重试\n        isAllowSubscribeRetry: this._options.isAllowSubscribeRetry,\n\n        // 是否允许订阅重试\n        isAllowPublishRetry: this._options.isAllowPublishRetry,\n\n        /**\n         * 禁用视频时关摄像头\n         */\n        isOffCameraWhenVideoDisable: this._options.isOffCameraWhenVideoDisable,\n\n        /**\n         * RTC 房间加入类型\n         */\n        joinType: this._options.joinType,\n\n        // 允许降级获得流，获得音视频不成功 ，降级获得音频, 默认不允许\n        isAllowDemotionGetStream: this._options.isAllowDemotionGetStream,\n\n        // 标明是主叫产生的session\n        produceType: ProduceTypes.CALLER,\n\n        isCrossAppkey,\n\n        callPushConfig: this._callPushConfig,\n\n        hungupPushConfig: this._hungupPushConfig,\n      });\n\n      // session上注册监听事件\n      session.registerSessionListener(listener);\n\n      this._sessionList.push(session);\n      this._logger.info('_', `[RCCallClient call] successfully created session object, sessionId: ${session.getSessionId()}`);\n      return { code, session };\n    }\n    this._logger.error('_', `[RCCallClient call] call failed code ->: ${code}`);\n    localTracks.forEach((track) => {\n      // 禁用视频\n      track.mute();\n\n      // 关闭摄像头\n      track.destroy();\n    });\n\n    return { code };\n  }\n\n  /**\n   * 发起群组呼叫\n   * @param params.targetId 群组 Id 必填\n   * @param params.userIds 被呼叫的群内成员 Id 必填\n   * @param params.mediaType 音频呼叫 or 音视频呼叫 必填\n   * @param params.listener (session上的监听) 必填\n   * @param params.constraints 获取音频或音视频资源时的参数 可选\n   * @param params.channelId 组织 Id 可选\n   * @param params.extra 消息扩展信息 可选\n   * @deprecated 5.1.2 版本废弃 params.pushTitle 通知的标题\n   * @deprecated 5.1.2 版本废弃 params.pushContent 通知的内容\n   * @param params.bitrate 需要设置的码率参数\n   */\n  public async callInGroup({\n    targetId, userIds, mediaType = RCCallMediaType.AUDIO, listener, constraints, channelId = '', extra = '', pushTitle = '', pushContent = '', bitrate,\n  }: IRCCallInGroupParams): Promise<{ code: RCCallErrorCode, session?: RCCallSession }> {\n    const pushConfig = this._callPushConfig ? this._callPushConfig : { pushTitle, pushContent };\n    const conclusion: IValidationResult[] = [validateTargetId(targetId), validateUserIds(userIds), validateMediaType(mediaType), validateListener(listener), validateExtra(extra)];\n    const messages: string[] = [];\n    const result = conclusion.every((obj: IValidationResult) => {\n      !obj.result && messages.push(obj.msg!);\n      return obj.result;\n    });\n\n    if (!result) {\n      throw new Error(`[RCCallClient callInGroup] ${messages.join('\\n')}`);\n    }\n\n    const { code: validatePushCode, errorMsg } = validatePushConfig(pushConfig);\n    if (validatePushCode !== RCCallErrorCode.SUCCESS) {\n      this._logger.error('_', `[RCCallClient call] param error, errorMsg: ${errorMsg}`);\n      return { code: validatePushCode };\n    }\n\n    let localTracks: RCLocalTrack[] = [];\n\n    const { code: _code, tracks } = await this._getLocalTrack(mediaType, constraints);\n    if (_code !== RCCallErrorCode.SUCCESS) {\n      return { code: _code };\n    }\n    localTracks = tracks!;\n\n    localTracks.forEach((track) => {\n      // 设置码率\n      if (track.isAudioTrack() && bitrate?.audio) {\n        track.setBitrate(bitrate?.audio);\n      }\n      if (track.isVideoTrack() && bitrate?.video) {\n        track.setBitrate(bitrate?.video?.max, bitrate?.video?.min, bitrate?.video?.start);\n      }\n      // 向外抛出本地流\n      listener.onTrackReady(track);\n    });\n\n    // 往组里发消息\n    const { code, stateMachine } = await this._callEngine.callInGroup(channelId, targetId, mediaType, userIds, extra, pushConfig);\n    if (code === RCCallErrorCode.SUCCESS && stateMachine) {\n      this._logger.info('_', '[RCCallClient callInGroup] successfully created state machine');\n      const session = new RCCallSession(stateMachine, this._rtcClient, this._logger, {\n        localTracks,\n\n        // 是否允许订阅重试\n        isAllowSubscribeRetry: this._options.isAllowSubscribeRetry,\n\n        // 是否允许发布重试\n        isAllowPublishRetry: this._options.isAllowPublishRetry,\n\n        /**\n         * 禁用视频时关摄像头\n         */\n        isOffCameraWhenVideoDisable: this._options.isOffCameraWhenVideoDisable,\n\n        /**\n         * RTC 房间加入类型\n         */\n        joinType: this._options.joinType,\n\n        // 允许降级获得流，获得音视频不成功 ，降级获得音频, 默认不允许\n        isAllowDemotionGetStream: this._options.isAllowDemotionGetStream,\n\n        // 标明是主叫产生的session\n        produceType: ProduceTypes.CALLER,\n\n        callPushConfig: this._callPushConfig,\n\n        hungupPushConfig: this._hungupPushConfig,\n      });\n\n      // session上注册监听事件\n      session.registerSessionListener(listener);\n      this._sessionList.push(session);\n      this._logger.info('_', `[RCCallClient callInGroup] successfully created session object, sessionId: ${session.getSessionId()}`);\n      return { code, session };\n    }\n    this._logger.info('_', `[RCCallClient callInGroup] callInGroup failed code -> ${code}`);\n    localTracks.forEach((track) => {\n      // 禁用视频\n      track.mute();\n\n      // 关闭摄像头\n      track.destroy();\n    });\n\n    return { code };\n  }\n\n  /**\n   * 调RTC API 获得本地流\n   */\n  private async _getLocalTrackCore(mediaType: RCCallMediaType, constraints?: IMediaStreamConstraints): Promise<{ code: RCCallErrorCode, tracks?: RCLocalTrack[] }> {\n    // 检测是否能够获得本地流\n    if (mediaType === RCCallMediaType.AUDIO) {\n      const { code, track } = await this._rtcClient.createMicrophoneAudioTrack('RongCloudRTC', constraints && constraints.audio && { ...constraints.audio });\n      if (code !== RCRTCCode.SUCCESS) {\n        this._logger.error('_', `[RCCallClient _getTrack] get Audio local tracks failed RCT code -> ${code}`);\n        return { code: RCCallErrorCode.GET_LOCAL_AUDIO_TRACK_ERROR };\n      }\n      this._logger.info('_', '[RCCallClient _getTrack] successfully get Audio local tracks');\n      return { code: RCCallErrorCode.SUCCESS, tracks: [track!] };\n    }\n    const { code, tracks } = await this._rtcClient.createMicrophoneAndCameraTracks('RongCloudRTC', constraints && { ...constraints });\n    if (code !== RCRTCCode.SUCCESS) {\n      this._logger.error('_', `[RCCallClient _getTrack] get Audio and Video local tracks failed RCT code -> ${code}`);\n      return { code: RCCallErrorCode.GET_LOCAL_AUDIO_AND_VIDEO_TRACK_ERROR };\n    }\n    this._logger.info('_', '[RCCallClient _getTrack] successfully get audio and video local tracks');\n    return { code: RCCallErrorCode.SUCCESS, tracks };\n  }\n\n  private async _getLocalTrack(mediaType: RCCallMediaType, constraints?: IMediaStreamConstraints): Promise<{ code: RCCallErrorCode, tracks?: RCLocalTrack[] }> {\n    // 如果是允许降级获得流，并且是获得音视频\n    if (this._options.isAllowDemotionGetStream && mediaType === RCCallMediaType.AUDIO_VIDEO) {\n      const { code, tracks } = await this._getLocalTrackCore(RCCallMediaType.AUDIO_VIDEO, constraints);\n\n      // 如果音视频不能获得，就降组获得音频\n      if (code !== RCCallErrorCode.SUCCESS) {\n        const { code, tracks } = await this._getLocalTrackCore(RCCallMediaType.AUDIO, constraints);\n        if (code !== RCCallErrorCode.SUCCESS) {\n          return { code };\n        }\n        return { code, tracks };\n      }\n      return { code, tracks };\n    }\n    const { code: _code, tracks } = await this._getLocalTrackCore(mediaType, constraints);\n    if (_code !== RCCallErrorCode.SUCCESS) {\n      return { code: _code };\n    }\n    return { code: _code, tracks };\n  }\n\n  /**\n   * 从sessionList删除某个session\n   */\n  private _removeSession(session: RCCallSession) {\n    const id = session.getSessionId();\n    this._sessionList = this._sessionList.filter((session) => session.getSessionId() !== id);\n  }\n\n  /**\n   * 获取己方其他端加入通话（已加入 RTC 房间）的用户信息\n   */\n  public async getJoinedRoomInfo(): Promise<{ code: RCCallErrorCode, data?: IRTCJoinedInfo[] }> {\n    const { code, data } = await this._context.getRTCJoinedUserInfo(this._context.getCurrentId());\n    if (code !== ErrorCode.SUCCESS) {\n      this._logger.error('_', `getJoinedUserInfo error code: ${code}`);\n      return { code: RCCallErrorCode.QUERY_JOINED_USER_INFO_ERROR };\n    }\n\n    return { code: RCCallErrorCode.SUCCESS, data };\n  }\n\n  /**\n   * 设置呼叫、挂断推送数据\n   * @param callPushConfig 呼叫推送配置\n   * @param callPushConfig.pushTitle 呼叫推送标题\n   * @param callPushConfig.pushContent 呼叫推送内容\n   * @param hungupPushConfig 挂断推送配置\n   * @param hungupPushConfig.pushTitle 挂断推送标题\n   * @param hungupPushConfig.pushContent 挂断推送内容\n   */\n  public setPushConfig(callPushConfig: IPushConfig, hungupPushConfig: IPushConfig) {\n    const validateRes = [callPushConfig, hungupPushConfig].map((item) => validatePushConfig(item));\n\n    for (const { code, errorMsg } of validateRes) {\n      if (code !== RCCallErrorCode.SUCCESS) {\n        this._logger.error('_', `[RCCallClient setPushConfig] param error, errorMsg: ${errorMsg}`);\n        return { code };\n      }\n    }\n\n    this._callPushConfig = callPushConfig;\n    this._hungupPushConfig = hungupPushConfig;\n  }\n}\n","import {\n  IPluginGenerator, IRuntime, RTCPluginContext, VersionManage,\n} from '@rongcloud/engine';\nimport {\n  RCCallErrorCode, RCCallLanguage, RCCallEndReason, RCCallMediaType, RCCallUserState, RCCallSessionState, IEndSummary, IInvitedUsers, ISenderInfo, IOfflineRecord,\n} from '@rc-embed/call-engine';\nimport { IRCRTCStateReport } from '@rongcloud/plugin-rtc';\nimport {\n  IRCCallInitOptions, IRCCallParams, IRCCallInGroupParams, ISessionListener, IMuteUser, IMediaStreamConstraints, IDeviceChangeParams, IValidationResult, IPushConfig,\n} from './interface';\nimport RCCallClient from './RCCallClient';\nimport { RCCallSession } from './RCCallSession';\nimport { validateCallInitOptions } from './validation';\n\n// plugin-call 版本上报\nVersionManage.add('plugin-call', __VERSION__);\n\nconst installer: IPluginGenerator<RCCallClient, IRCCallInitOptions> = {\n  tag: 'RCCall',\n  verify(runtime: IRuntime) {\n    return runtime.tag === 'browser';\n  },\n  setup(context: RTCPluginContext, runtime: IRuntime, options: IRCCallInitOptions): RCCallClient {\n    // 先校验参数\n    const conclusion: IValidationResult = validateCallInitOptions(options);\n    if (!conclusion.result) {\n      throw new Error(`[RCCallLib installer steup]${conclusion.msg}`);\n    }\n\n    // 校验当前安装的 engine 版本是否可用\n    if (!VersionManage.validEngine(__REQUIRED_ENGINE_VERSION__)) {\n      throw new Error(`The current engine version '${VersionManage.getInfo().engine}' error, plugin-call required engine version at least '${__REQUIRED_ENGINE_VERSION__}'.`);\n    }\n\n    const logger = context.createLogger('RCCall', 'RTC');\n    options.logOutputLevel && logger.setOutputLevel(options.logOutputLevel);\n\n    if (typeof options.logLevel !== 'undefined') {\n      logger.warn('_', 'The \\'logLevel\\' parameter is deprecated, please use \\'logOutputLevel\\' instead.');\n    }\n\n    logger.warn('_', `RCCall Version: ${__VERSION__}, Commit: ${__COMMIT_ID__}`);\n\n    return new RCCallClient(context, runtime, logger, options);\n  },\n};\n\nexport {\n  installer,\n  RCCallClient,\n  RCCallSession,\n  RCCallLanguage,\n  RCCallErrorCode,\n  RCCallEndReason,\n  RCCallMediaType,\n  RCCallUserState,\n  RCCallSessionState,\n};\n\nexport type {\n  IEndSummary,\n  IRCCallInitOptions,\n  IRCCallParams,\n  IRCCallInGroupParams,\n  ISessionListener,\n  ISenderInfo,\n  IMuteUser,\n  IInvitedUsers,\n  IMediaStreamConstraints,\n  IDeviceChangeParams,\n  IOfflineRecord,\n  IRCRTCStateReport,\n  IPushConfig,\n};\n"],"names":["string10to64","number","chars","radix","qutient","arr","mod","getUUID","c","r","getUUID22","uuid","generateRandomId","random","timerSetTimeout","func","timeout","eventEmitter","EventEmitter","getCallDeviceId","runtime","key","MsgCallStatus","MsgCallStatus2","RCCallErrorCode","RCCallEndReason","CallRemoteEndReason","RCCallMessageType","RCCallSessionState","RCCallSessionState2","RCCallUserState","RCCallUserState2","RCCrossCallType","Timer$1","callback","__publicField","endTime","duration","RCCallStateMachine","_context","_runtime","_logger","_callMsgHandler","_channelId","_conversationType","_targetId","_mediaType","_callId","sentTime","delayTime","userId","state","reason","_a","user","message","senderUserId","userProfile","hungupReason","messageType","endReason","remoteUserId","msgDeviceId","suid","deviceId","senderDeviceId","currentUserId","ConversationType","isCurrentUserId","mediaType","content","isLessThanTwo","isInviteUser","isNoOneAnswered","watchers","inids","roomType","userIdList","id","inviteUserIds","Timer","existedUserPofiles","caller","needRingingUserIds","userInfo","callStatus","error","userIds","extra","pushConfig","isCrossAppkey","code","isCaller","endCode","options","existedUserIds","hangupParams","msgCode","remoteUsers","remoteUser","uid","beginTimestamp","endTimestamp","summary","RCCallMediaType","RCCallMediaType2","MemberModifyType","MemberModifyType2","Platform","OfflineRecorder","_onRecord","channelId","conversationType","targetId","callId","isSelfSend","messages","msg","callMsgTypes","EngineErrorCodeToCallErrorCode","ErrorCode","CallMessageHandler","_offlineMsgItv","_getStateMachine","record","markTime","currentMsgSentTime","insertIndex","index","currentTime","buffers","item","messageUId","isOffLineMessage","offlineBuffers","inviteCallId","isInviteMsgType","isLessThanOfflineMsgItv","taskMsgList","i","otherCallId","delIndex","delOfflineIndex","isUnfinishedPrivateCall","isOnlyInvite","hasInviteAndRinging","isUnfinishedGrpCall","isUnfinished","noOneAccept","allUserIds","isCallerHungup","taskMsgSenderUserId","delCount","listener","funcName","event","eventType","directionalUserIdList","sendOpts","newPushConfig","callerId","RCCallLanguage","EN","ZH","Local","lang","RCCallEngine","_watchers","_options","crtUserId","info","ProduceTypes","fun","on","data","funs","cb","length","isLanguage","val","isJoinType","RTCJoinType","isLogLevel","LogL","validateCallInitOptions","keyNames","keys","missingKeys","validateListener","validateTargetId","validateMediaType","validateExtra","validatePushConfig","pushConfigKeys","isUndefined","isBoolean","isString","iOSConfig","androidConfig","validateIOSConfigRes","validateIOSConfig","validateAndroidConfigRes","validateAndroidConfig","threadId","richMediaUri","androidConfigKeys","validateUserIds","str","isRCFrameRate","isValidResolution","resolution","RCResolution","validateMediaStreamConstraints","constraints","RCCallSession","_stateMachine","_rtcClient","roomId","summaryInfo","sender","invitedUsers","callBack","room","RCRTCCode","tracks","isAllowSubscribeRetry","count","isAllowPublishRetry","conclusion","track","_code","audioConstraints","recentTracks","localTracks","errorMsg","obj","localVideoTracks","localAudiotracks","byServer","RCKickReason","name","audioTrack","muteUser","videoTrack","result","report","RCCallClient","session","ses","stateMachine","pushTitle","pushContent","bitrate","validatePushCode","_b","_c","callPushConfig","hungupPushConfig","validateRes","VersionManage","installer","context","logger"],"mappings":"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"}
|