@webex/plugin-meetings 2.60.0 → 2.60.1-next.10

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.
Files changed (539) hide show
  1. package/README.md +58 -8
  2. package/dist/annotation/annotation.types.d.ts +42 -0
  3. package/dist/annotation/annotation.types.js +7 -0
  4. package/dist/annotation/annotation.types.js.map +1 -0
  5. package/dist/annotation/constants.d.ts +31 -0
  6. package/dist/annotation/constants.js +41 -0
  7. package/dist/annotation/constants.js.map +1 -0
  8. package/dist/annotation/index.d.ts +117 -0
  9. package/dist/annotation/index.js +357 -0
  10. package/dist/annotation/index.js.map +1 -0
  11. package/dist/breakouts/breakout.d.ts +8 -0
  12. package/dist/breakouts/breakout.js +215 -0
  13. package/dist/breakouts/breakout.js.map +1 -0
  14. package/dist/breakouts/collection.d.ts +5 -0
  15. package/dist/breakouts/collection.js +22 -0
  16. package/dist/breakouts/collection.js.map +1 -0
  17. package/dist/breakouts/edit-lock-error.d.ts +15 -0
  18. package/dist/breakouts/edit-lock-error.js +51 -0
  19. package/dist/breakouts/edit-lock-error.js.map +1 -0
  20. package/dist/breakouts/events.d.ts +8 -0
  21. package/dist/breakouts/events.js +44 -0
  22. package/dist/breakouts/events.js.map +1 -0
  23. package/dist/breakouts/index.d.ts +5 -0
  24. package/dist/breakouts/index.js +1047 -0
  25. package/dist/breakouts/index.js.map +1 -0
  26. package/dist/breakouts/request.d.ts +22 -0
  27. package/dist/breakouts/request.js +77 -0
  28. package/dist/breakouts/request.js.map +1 -0
  29. package/dist/breakouts/utils.d.ts +15 -0
  30. package/dist/breakouts/utils.js +64 -0
  31. package/dist/breakouts/utils.js.map +1 -0
  32. package/dist/common/browser-detection.js +2 -3
  33. package/dist/common/browser-detection.js.map +1 -1
  34. package/dist/common/collection.js +3 -4
  35. package/dist/common/collection.js.map +1 -1
  36. package/dist/common/config.js +1 -2
  37. package/dist/common/config.js.map +1 -1
  38. package/dist/common/errors/captcha-error.js +1 -2
  39. package/dist/common/errors/captcha-error.js.map +1 -1
  40. package/dist/common/errors/intent-to-join.js +1 -2
  41. package/dist/common/errors/intent-to-join.js.map +1 -1
  42. package/dist/common/errors/join-meeting.js +1 -2
  43. package/dist/common/errors/join-meeting.js.map +1 -1
  44. package/dist/common/errors/media.js +1 -2
  45. package/dist/common/errors/media.js.map +1 -1
  46. package/dist/common/errors/no-meeting-info.d.ts +14 -0
  47. package/dist/common/errors/no-meeting-info.js +50 -0
  48. package/dist/common/errors/no-meeting-info.js.map +1 -0
  49. package/dist/common/errors/parameter.js +3 -4
  50. package/dist/common/errors/parameter.js.map +1 -1
  51. package/dist/common/errors/password-error.js +1 -2
  52. package/dist/common/errors/password-error.js.map +1 -1
  53. package/dist/common/errors/permission.js +1 -2
  54. package/dist/common/errors/permission.js.map +1 -1
  55. package/dist/common/errors/{reclaim-host-role-error.js → reclaim-host-role-errors.js} +7 -11
  56. package/dist/common/errors/reclaim-host-role-errors.js.map +1 -0
  57. package/dist/common/errors/reconnection-in-progress.js +1 -2
  58. package/dist/common/errors/reconnection-in-progress.js.map +1 -1
  59. package/dist/common/errors/reconnection.js +1 -2
  60. package/dist/common/errors/reconnection.js.map +1 -1
  61. package/dist/common/errors/stats.js +1 -2
  62. package/dist/common/errors/stats.js.map +1 -1
  63. package/dist/common/errors/webex-errors.d.ts +20 -8
  64. package/dist/common/errors/webex-errors.js +48 -28
  65. package/dist/common/errors/webex-errors.js.map +1 -1
  66. package/dist/common/errors/webex-meetings-error.js +1 -2
  67. package/dist/common/errors/webex-meetings-error.js.map +1 -1
  68. package/dist/common/events/events-scope.js +1 -2
  69. package/dist/common/events/events-scope.js.map +1 -1
  70. package/dist/common/events/events.js +1 -2
  71. package/dist/common/events/events.js.map +1 -1
  72. package/dist/common/events/trigger-proxy.js +1 -2
  73. package/dist/common/events/trigger-proxy.js.map +1 -1
  74. package/dist/common/events/util.js +1 -2
  75. package/dist/common/events/util.js.map +1 -1
  76. package/dist/common/logs/logger-config.js +1 -2
  77. package/dist/common/logs/logger-config.js.map +1 -1
  78. package/dist/common/logs/logger-proxy.js +2 -3
  79. package/dist/common/logs/logger-proxy.js.map +1 -1
  80. package/dist/common/logs/request.d.ts +3 -1
  81. package/dist/common/logs/request.js +8 -5
  82. package/dist/common/logs/request.js.map +1 -1
  83. package/dist/common/queue.d.ts +9 -7
  84. package/dist/common/queue.js +22 -9
  85. package/dist/common/queue.js.map +1 -1
  86. package/dist/config.d.ts +6 -7
  87. package/dist/config.js +8 -10
  88. package/dist/config.js.map +1 -1
  89. package/dist/constants.d.ts +234 -100
  90. package/dist/constants.js +433 -444
  91. package/dist/constants.js.map +1 -1
  92. package/dist/controls-options-manager/constants.js +3 -6
  93. package/dist/controls-options-manager/constants.js.map +1 -1
  94. package/dist/controls-options-manager/enums.d.ts +11 -1
  95. package/dist/controls-options-manager/enums.js +15 -6
  96. package/dist/controls-options-manager/enums.js.map +1 -1
  97. package/dist/controls-options-manager/index.d.ts +17 -1
  98. package/dist/controls-options-manager/index.js +127 -38
  99. package/dist/controls-options-manager/index.js.map +1 -1
  100. package/dist/controls-options-manager/types.d.ts +43 -0
  101. package/dist/controls-options-manager/types.js +7 -0
  102. package/dist/controls-options-manager/types.js.map +1 -0
  103. package/dist/controls-options-manager/util.d.ts +1 -7
  104. package/dist/controls-options-manager/util.js +309 -19
  105. package/dist/controls-options-manager/util.js.map +1 -1
  106. package/dist/index.d.ts +6 -3
  107. package/dist/index.js +121 -5
  108. package/dist/index.js.map +1 -1
  109. package/dist/interceptors/index.d.ts +2 -0
  110. package/dist/interceptors/index.js +15 -0
  111. package/dist/interceptors/index.js.map +1 -0
  112. package/dist/interceptors/locusRetry.d.ts +27 -0
  113. package/dist/interceptors/locusRetry.js +94 -0
  114. package/dist/interceptors/locusRetry.js.map +1 -0
  115. package/dist/interpretation/collection.d.ts +5 -0
  116. package/dist/interpretation/collection.js +22 -0
  117. package/dist/interpretation/collection.js.map +1 -0
  118. package/dist/interpretation/index.d.ts +5 -0
  119. package/dist/interpretation/index.js +365 -0
  120. package/dist/interpretation/index.js.map +1 -0
  121. package/dist/interpretation/siLanguage.d.ts +5 -0
  122. package/dist/interpretation/siLanguage.js +24 -0
  123. package/dist/interpretation/siLanguage.js.map +1 -0
  124. package/dist/locus-info/controlsUtils.js +100 -11
  125. package/dist/locus-info/controlsUtils.js.map +1 -1
  126. package/dist/locus-info/embeddedAppsUtils.js +3 -4
  127. package/dist/locus-info/embeddedAppsUtils.js.map +1 -1
  128. package/dist/locus-info/fullState.js +1 -2
  129. package/dist/locus-info/fullState.js.map +1 -1
  130. package/dist/locus-info/hostUtils.js +1 -2
  131. package/dist/locus-info/hostUtils.js.map +1 -1
  132. package/dist/locus-info/index.d.ts +57 -4
  133. package/dist/locus-info/index.js +425 -84
  134. package/dist/locus-info/index.js.map +1 -1
  135. package/dist/locus-info/infoUtils.js +13 -5
  136. package/dist/locus-info/infoUtils.js.map +1 -1
  137. package/dist/locus-info/mediaSharesUtils.js +58 -3
  138. package/dist/locus-info/mediaSharesUtils.js.map +1 -1
  139. package/dist/locus-info/parser.d.ts +66 -6
  140. package/dist/locus-info/parser.js +253 -80
  141. package/dist/locus-info/parser.js.map +1 -1
  142. package/dist/locus-info/selfUtils.js +97 -13
  143. package/dist/locus-info/selfUtils.js.map +1 -1
  144. package/dist/media/index.d.ts +2 -0
  145. package/dist/media/index.js +107 -319
  146. package/dist/media/index.js.map +1 -1
  147. package/dist/media/properties.d.ts +38 -53
  148. package/dist/media/properties.js +96 -153
  149. package/dist/media/properties.js.map +1 -1
  150. package/dist/media/util.js +1 -22
  151. package/dist/media/util.js.map +1 -1
  152. package/dist/mediaQualityMetrics/config.d.ts +234 -230
  153. package/dist/mediaQualityMetrics/config.js +302 -498
  154. package/dist/mediaQualityMetrics/config.js.map +1 -1
  155. package/dist/meeting/in-meeting-actions.d.ts +88 -0
  156. package/dist/meeting/in-meeting-actions.js +94 -3
  157. package/dist/meeting/in-meeting-actions.js.map +1 -1
  158. package/dist/meeting/index.d.ts +705 -520
  159. package/dist/meeting/index.js +5047 -3089
  160. package/dist/meeting/index.js.map +1 -1
  161. package/dist/meeting/locusMediaRequest.d.ts +74 -0
  162. package/dist/meeting/locusMediaRequest.js +291 -0
  163. package/dist/meeting/locusMediaRequest.js.map +1 -0
  164. package/dist/meeting/muteState.d.ts +93 -25
  165. package/dist/meeting/muteState.js +224 -133
  166. package/dist/meeting/muteState.js.map +1 -1
  167. package/dist/meeting/request.d.ts +82 -47
  168. package/dist/meeting/request.js +304 -199
  169. package/dist/meeting/request.js.map +1 -1
  170. package/dist/meeting/request.type.d.ts +11 -0
  171. package/dist/meeting/request.type.js +7 -0
  172. package/dist/meeting/request.type.js.map +1 -0
  173. package/dist/meeting/state.js +1 -2
  174. package/dist/meeting/state.js.map +1 -1
  175. package/dist/meeting/util.d.ts +118 -1
  176. package/dist/meeting/util.js +676 -435
  177. package/dist/meeting/util.js.map +1 -1
  178. package/dist/meeting/voicea-meeting.d.ts +20 -0
  179. package/dist/meeting/voicea-meeting.js +201 -0
  180. package/dist/meeting/voicea-meeting.js.map +1 -0
  181. package/dist/meeting-info/collection.js +3 -4
  182. package/dist/meeting-info/collection.js.map +1 -1
  183. package/dist/meeting-info/index.d.ts +13 -1
  184. package/dist/meeting-info/index.js +74 -7
  185. package/dist/meeting-info/index.js.map +1 -1
  186. package/dist/meeting-info/meeting-info-v2.d.ts +31 -1
  187. package/dist/meeting-info/meeting-info-v2.js +200 -63
  188. package/dist/meeting-info/meeting-info-v2.js.map +1 -1
  189. package/dist/meeting-info/request.js +1 -2
  190. package/dist/meeting-info/request.js.map +1 -1
  191. package/dist/meeting-info/util.js +2 -3
  192. package/dist/meeting-info/util.js.map +1 -1
  193. package/dist/meeting-info/utilv2.js +39 -41
  194. package/dist/meeting-info/utilv2.js.map +1 -1
  195. package/dist/meetings/collection.d.ts +17 -0
  196. package/dist/meetings/collection.js +42 -4
  197. package/dist/meetings/collection.js.map +1 -1
  198. package/dist/meetings/index.d.ts +114 -20
  199. package/dist/meetings/index.js +540 -126
  200. package/dist/meetings/index.js.map +1 -1
  201. package/dist/meetings/meetings.types.d.ts +4 -0
  202. package/dist/meetings/meetings.types.js +7 -0
  203. package/dist/meetings/meetings.types.js.map +1 -0
  204. package/dist/meetings/request.js +4 -3
  205. package/dist/meetings/request.js.map +1 -1
  206. package/dist/meetings/util.js +107 -6
  207. package/dist/meetings/util.js.map +1 -1
  208. package/dist/member/index.d.ts +13 -1
  209. package/dist/member/index.js +45 -2
  210. package/dist/member/index.js.map +1 -1
  211. package/dist/member/member.types.js +3 -4
  212. package/dist/member/member.types.js.map +1 -1
  213. package/dist/member/types.d.ts +32 -0
  214. package/dist/member/types.js +23 -0
  215. package/dist/member/types.js.map +1 -0
  216. package/dist/member/util.js +120 -29
  217. package/dist/member/util.js.map +1 -1
  218. package/dist/members/collection.d.ts +5 -0
  219. package/dist/members/collection.js +11 -2
  220. package/dist/members/collection.js.map +1 -1
  221. package/dist/members/index.d.ts +56 -11
  222. package/dist/members/index.js +174 -47
  223. package/dist/members/index.js.map +1 -1
  224. package/dist/members/request.d.ts +67 -11
  225. package/dist/members/request.js +102 -54
  226. package/dist/members/request.js.map +1 -1
  227. package/dist/members/types.js +3 -4
  228. package/dist/members/types.js.map +1 -1
  229. package/dist/members/util.d.ts +214 -1
  230. package/dist/members/util.js +327 -284
  231. package/dist/members/util.js.map +1 -1
  232. package/dist/metrics/constants.d.ts +15 -6
  233. package/dist/metrics/constants.js +17 -9
  234. package/dist/metrics/constants.js.map +1 -1
  235. package/dist/metrics/index.d.ts +4 -111
  236. package/dist/metrics/index.js +4 -452
  237. package/dist/metrics/index.js.map +1 -1
  238. package/dist/multistream/mediaRequestManager.d.ts +118 -0
  239. package/dist/multistream/mediaRequestManager.js +344 -0
  240. package/dist/multistream/mediaRequestManager.js.map +1 -0
  241. package/dist/multistream/receiveSlot.d.ts +68 -0
  242. package/dist/multistream/receiveSlot.js +200 -0
  243. package/dist/multistream/receiveSlot.js.map +1 -0
  244. package/dist/multistream/receiveSlotManager.d.ts +56 -0
  245. package/dist/multistream/receiveSlotManager.js +174 -0
  246. package/dist/multistream/receiveSlotManager.js.map +1 -0
  247. package/dist/multistream/remoteMedia.d.ts +72 -0
  248. package/dist/multistream/remoteMedia.js +268 -0
  249. package/dist/multistream/remoteMedia.js.map +1 -0
  250. package/dist/multistream/remoteMediaGroup.d.ts +47 -0
  251. package/dist/multistream/remoteMediaGroup.js +267 -0
  252. package/dist/multistream/remoteMediaGroup.js.map +1 -0
  253. package/dist/multistream/remoteMediaManager.d.ts +285 -0
  254. package/dist/multistream/remoteMediaManager.js +1211 -0
  255. package/dist/multistream/remoteMediaManager.js.map +1 -0
  256. package/dist/multistream/sendSlotManager.d.ts +61 -0
  257. package/dist/multistream/sendSlotManager.js +236 -0
  258. package/dist/multistream/sendSlotManager.js.map +1 -0
  259. package/dist/networkQualityMonitor/index.js +5 -4
  260. package/dist/networkQualityMonitor/index.js.map +1 -1
  261. package/dist/personal-meeting-room/index.js +2 -3
  262. package/dist/personal-meeting-room/index.js.map +1 -1
  263. package/dist/personal-meeting-room/request.js +2 -3
  264. package/dist/personal-meeting-room/request.js.map +1 -1
  265. package/dist/personal-meeting-room/util.js +1 -2
  266. package/dist/personal-meeting-room/util.js.map +1 -1
  267. package/dist/reachability/clusterReachability.d.ts +109 -0
  268. package/dist/reachability/clusterReachability.js +357 -0
  269. package/dist/reachability/clusterReachability.js.map +1 -0
  270. package/dist/reachability/index.d.ts +61 -95
  271. package/dist/reachability/index.js +304 -392
  272. package/dist/reachability/index.js.map +1 -1
  273. package/dist/reachability/request.d.ts +7 -3
  274. package/dist/reachability/request.js +18 -10
  275. package/dist/reachability/request.js.map +1 -1
  276. package/dist/reachability/util.d.ts +8 -0
  277. package/dist/reachability/util.js +29 -0
  278. package/dist/reachability/util.js.map +1 -0
  279. package/dist/reactions/constants.d.ts +3 -0
  280. package/dist/reactions/constants.js +12 -0
  281. package/dist/reactions/constants.js.map +1 -0
  282. package/dist/reactions/reactions.d.ts +2 -2
  283. package/dist/reactions/reactions.js +4 -6
  284. package/dist/reactions/reactions.js.map +1 -1
  285. package/dist/reactions/reactions.type.d.ts +23 -3
  286. package/dist/reactions/reactions.type.js +21 -23
  287. package/dist/reactions/reactions.type.js.map +1 -1
  288. package/dist/reconnection-manager/index.d.ts +32 -8
  289. package/dist/reconnection-manager/index.js +285 -232
  290. package/dist/reconnection-manager/index.js.map +1 -1
  291. package/dist/recording-controller/enums.js +4 -5
  292. package/dist/recording-controller/enums.js.map +1 -1
  293. package/dist/recording-controller/index.d.ts +15 -1
  294. package/dist/recording-controller/index.js +57 -46
  295. package/dist/recording-controller/index.js.map +1 -1
  296. package/dist/recording-controller/util.d.ts +5 -4
  297. package/dist/recording-controller/util.js +10 -10
  298. package/dist/recording-controller/util.js.map +1 -1
  299. package/dist/roap/index.d.ts +9 -47
  300. package/dist/roap/index.js +100 -238
  301. package/dist/roap/index.js.map +1 -1
  302. package/dist/roap/request.d.ts +18 -12
  303. package/dist/roap/request.js +126 -180
  304. package/dist/roap/request.js.map +1 -1
  305. package/dist/roap/turnDiscovery.d.ts +27 -16
  306. package/dist/roap/turnDiscovery.js +115 -105
  307. package/dist/roap/turnDiscovery.js.map +1 -1
  308. package/dist/rtcMetrics/constants.d.ts +4 -0
  309. package/dist/rtcMetrics/constants.js +11 -0
  310. package/dist/rtcMetrics/constants.js.map +1 -0
  311. package/dist/rtcMetrics/index.d.ts +54 -0
  312. package/dist/rtcMetrics/index.js +140 -0
  313. package/dist/rtcMetrics/index.js.map +1 -0
  314. package/dist/statsAnalyzer/global.d.ts +1 -83
  315. package/dist/statsAnalyzer/global.js +2 -85
  316. package/dist/statsAnalyzer/global.js.map +1 -1
  317. package/dist/statsAnalyzer/index.d.ts +50 -30
  318. package/dist/statsAnalyzer/index.js +436 -511
  319. package/dist/statsAnalyzer/index.js.map +1 -1
  320. package/dist/statsAnalyzer/mqaUtil.d.ts +8 -6
  321. package/dist/statsAnalyzer/mqaUtil.js +130 -90
  322. package/dist/statsAnalyzer/mqaUtil.js.map +1 -1
  323. package/dist/transcription/index.js +1 -2
  324. package/dist/transcription/index.js.map +1 -1
  325. package/dist/webinar/collection.d.ts +16 -0
  326. package/dist/webinar/collection.js +43 -0
  327. package/dist/webinar/collection.js.map +1 -0
  328. package/dist/webinar/index.d.ts +5 -0
  329. package/dist/webinar/index.js +68 -0
  330. package/dist/webinar/index.js.map +1 -0
  331. package/package.json +39 -26
  332. package/src/annotation/annotation.types.ts +50 -0
  333. package/src/annotation/constants.ts +36 -0
  334. package/src/annotation/index.ts +328 -0
  335. package/src/breakouts/README.md +220 -0
  336. package/src/breakouts/breakout.ts +188 -0
  337. package/src/breakouts/collection.ts +19 -0
  338. package/src/breakouts/edit-lock-error.ts +25 -0
  339. package/src/breakouts/events.ts +56 -0
  340. package/src/breakouts/index.ts +925 -0
  341. package/src/breakouts/request.ts +55 -0
  342. package/src/breakouts/utils.ts +57 -0
  343. package/src/common/errors/no-meeting-info.ts +24 -0
  344. package/src/common/errors/webex-errors.ts +36 -12
  345. package/src/common/logs/logger-proxy.ts +1 -1
  346. package/src/common/logs/request.ts +5 -1
  347. package/src/common/queue.ts +22 -8
  348. package/src/config.ts +6 -7
  349. package/src/constants.ts +265 -100
  350. package/src/controls-options-manager/enums.ts +12 -0
  351. package/src/controls-options-manager/index.ts +116 -21
  352. package/src/controls-options-manager/types.ts +59 -0
  353. package/src/controls-options-manager/util.ts +294 -14
  354. package/src/index.ts +44 -0
  355. package/src/interceptors/index.ts +3 -0
  356. package/src/interceptors/locusRetry.ts +67 -0
  357. package/src/interpretation/README.md +60 -0
  358. package/src/interpretation/collection.ts +19 -0
  359. package/src/interpretation/index.ts +332 -0
  360. package/src/interpretation/siLanguage.ts +18 -0
  361. package/src/locus-info/controlsUtils.ts +110 -0
  362. package/src/locus-info/index.ts +450 -61
  363. package/src/locus-info/infoUtils.ts +14 -2
  364. package/src/locus-info/mediaSharesUtils.ts +64 -0
  365. package/src/locus-info/parser.ts +258 -47
  366. package/src/locus-info/selfUtils.ts +85 -2
  367. package/src/media/index.ts +153 -370
  368. package/src/media/properties.ts +106 -136
  369. package/src/media/util.ts +0 -21
  370. package/src/mediaQualityMetrics/config.ts +244 -377
  371. package/src/meeting/in-meeting-actions.ts +176 -0
  372. package/src/meeting/index.ts +4306 -2581
  373. package/src/meeting/locusMediaRequest.ts +313 -0
  374. package/src/meeting/muteState.ts +224 -138
  375. package/src/meeting/request.ts +214 -127
  376. package/src/meeting/request.type.ts +13 -0
  377. package/src/meeting/util.ts +687 -423
  378. package/src/meeting/voicea-meeting.ts +161 -0
  379. package/src/meeting-info/index.ts +81 -8
  380. package/src/meeting-info/meeting-info-v2.ts +163 -13
  381. package/src/meeting-info/util.ts +1 -1
  382. package/src/meeting-info/utilv2.ts +28 -28
  383. package/src/meetings/collection.ts +33 -0
  384. package/src/meetings/index.ts +529 -127
  385. package/src/meetings/meetings.types.ts +12 -0
  386. package/src/meetings/request.ts +2 -0
  387. package/src/meetings/util.ts +116 -5
  388. package/src/member/index.ts +43 -1
  389. package/src/member/types.ts +38 -0
  390. package/src/member/util.ts +125 -28
  391. package/src/members/collection.ts +8 -0
  392. package/src/members/index.ts +187 -52
  393. package/src/members/request.ts +87 -27
  394. package/src/members/util.ts +332 -291
  395. package/src/metrics/constants.ts +15 -6
  396. package/src/metrics/index.ts +1 -471
  397. package/src/multistream/mediaRequestManager.ts +440 -0
  398. package/src/multistream/receiveSlot.ts +184 -0
  399. package/src/multistream/receiveSlotManager.ts +166 -0
  400. package/src/multistream/remoteMedia.ts +254 -0
  401. package/src/multistream/remoteMediaGroup.ts +284 -0
  402. package/src/multistream/remoteMediaManager.ts +1145 -0
  403. package/src/multistream/sendSlotManager.ts +170 -0
  404. package/src/networkQualityMonitor/index.ts +6 -6
  405. package/src/reachability/clusterReachability.ts +320 -0
  406. package/src/reachability/index.ts +246 -347
  407. package/src/reachability/request.ts +17 -8
  408. package/src/reachability/util.ts +24 -0
  409. package/src/reactions/constants.ts +4 -0
  410. package/src/reactions/reactions.ts +4 -4
  411. package/src/reactions/reactions.type.ts +30 -4
  412. package/src/reconnection-manager/index.ts +168 -156
  413. package/src/recording-controller/index.ts +20 -3
  414. package/src/recording-controller/util.ts +26 -9
  415. package/src/roap/index.ts +96 -241
  416. package/src/roap/request.ts +74 -148
  417. package/src/roap/turnDiscovery.ts +62 -56
  418. package/src/rtcMetrics/constants.ts +3 -0
  419. package/src/rtcMetrics/index.ts +124 -0
  420. package/src/statsAnalyzer/global.ts +1 -84
  421. package/src/statsAnalyzer/index.ts +479 -645
  422. package/src/statsAnalyzer/mqaUtil.ts +128 -126
  423. package/src/webinar/collection.ts +31 -0
  424. package/src/webinar/index.ts +62 -0
  425. package/test/integration/spec/converged-space-meetings.js +233 -0
  426. package/test/integration/spec/journey.js +320 -264
  427. package/test/integration/spec/space-meeting.js +77 -4
  428. package/test/unit/spec/annotation/index.ts +418 -0
  429. package/test/unit/spec/breakouts/breakout.ts +237 -0
  430. package/test/unit/spec/breakouts/collection.ts +15 -0
  431. package/test/unit/spec/breakouts/edit-lock-error.ts +30 -0
  432. package/test/unit/spec/breakouts/events.ts +89 -0
  433. package/test/unit/spec/breakouts/index.ts +1790 -0
  434. package/test/unit/spec/breakouts/request.ts +104 -0
  435. package/test/unit/spec/breakouts/utils.js +72 -0
  436. package/test/unit/spec/common/queue.js +31 -2
  437. package/test/unit/spec/controls-options-manager/index.js +163 -0
  438. package/test/unit/spec/controls-options-manager/util.js +576 -60
  439. package/test/unit/spec/fixture/locus.js +1 -0
  440. package/test/unit/spec/interceptors/locusRetry.ts +131 -0
  441. package/test/unit/spec/interpretation/collection.ts +15 -0
  442. package/test/unit/spec/interpretation/index.ts +589 -0
  443. package/test/unit/spec/interpretation/siLanguage.ts +28 -0
  444. package/test/unit/spec/locus-info/controlsUtils.js +323 -30
  445. package/test/unit/spec/locus-info/index.js +1438 -16
  446. package/test/unit/spec/locus-info/infoUtils.js +54 -16
  447. package/test/unit/spec/locus-info/lib/SeqCmp.json +16 -0
  448. package/test/unit/spec/locus-info/lib/selfConstant.js +48 -0
  449. package/test/unit/spec/locus-info/mediaSharesUtils.ts +32 -0
  450. package/test/unit/spec/locus-info/parser.js +116 -35
  451. package/test/unit/spec/locus-info/selfUtils.js +275 -0
  452. package/test/unit/spec/media/index.ts +290 -0
  453. package/test/unit/spec/media/properties.ts +75 -84
  454. package/test/unit/spec/meeting/in-meeting-actions.ts +86 -0
  455. package/test/unit/spec/meeting/index.js +8886 -2815
  456. package/test/unit/spec/meeting/locusMediaRequest.ts +442 -0
  457. package/test/unit/spec/meeting/muteState.js +409 -213
  458. package/test/unit/spec/meeting/request.js +523 -43
  459. package/test/unit/spec/meeting/utils.js +834 -24
  460. package/test/unit/spec/meeting-info/index.js +300 -0
  461. package/test/unit/spec/meeting-info/meetinginfov2.js +527 -5
  462. package/test/unit/spec/meeting-info/utilv2.js +21 -0
  463. package/test/unit/spec/meetings/collection.js +26 -0
  464. package/test/unit/spec/meetings/index.js +1446 -217
  465. package/test/unit/spec/meetings/utils.js +202 -2
  466. package/test/unit/spec/member/index.js +32 -9
  467. package/test/unit/spec/member/util.js +499 -61
  468. package/test/unit/spec/members/index.js +394 -5
  469. package/test/unit/spec/members/request.js +206 -27
  470. package/test/unit/spec/members/utils.js +173 -38
  471. package/test/unit/spec/metrics/index.js +1 -50
  472. package/test/unit/spec/multistream/mediaRequestManager.ts +1418 -0
  473. package/test/unit/spec/multistream/receiveSlot.ts +163 -0
  474. package/test/unit/spec/multistream/receiveSlotManager.ts +203 -0
  475. package/test/unit/spec/multistream/remoteMedia.ts +255 -0
  476. package/test/unit/spec/multistream/remoteMediaGroup.ts +662 -0
  477. package/test/unit/spec/multistream/remoteMediaManager.ts +1924 -0
  478. package/test/unit/spec/multistream/sendSlotManager.ts +242 -0
  479. package/test/unit/spec/networkQualityMonitor/index.js +4 -4
  480. package/test/unit/spec/reachability/clusterReachability.ts +279 -0
  481. package/test/unit/spec/reachability/index.ts +532 -24
  482. package/test/unit/spec/reachability/request.js +68 -0
  483. package/test/unit/spec/reachability/util.ts +40 -0
  484. package/test/unit/spec/reconnection-manager/index.js +163 -24
  485. package/test/unit/spec/recording-controller/index.js +293 -218
  486. package/test/unit/spec/recording-controller/util.js +223 -96
  487. package/test/unit/spec/roap/index.ts +187 -77
  488. package/test/unit/spec/roap/request.ts +255 -0
  489. package/test/unit/spec/roap/turnDiscovery.ts +86 -48
  490. package/test/unit/spec/rtcMetrics/index.ts +93 -0
  491. package/test/unit/spec/stats-analyzer/index.js +644 -165
  492. package/test/unit/spec/webinar/collection.ts +13 -0
  493. package/test/unit/spec/webinar/index.ts +60 -0
  494. package/test/utils/constants.js +9 -0
  495. package/test/utils/integrationTestUtils.js +46 -0
  496. package/test/utils/testUtils.js +0 -45
  497. package/test/utils/webex-config.js +4 -0
  498. package/test/utils/webex-test-users.js +7 -3
  499. package/dist/common/errors/reclaim-host-role-error.js.map +0 -1
  500. package/dist/meeting/effectsState.d.ts +0 -42
  501. package/dist/meeting/effectsState.js +0 -260
  502. package/dist/meeting/effectsState.js.map +0 -1
  503. package/dist/metrics/config.d.ts +0 -169
  504. package/dist/metrics/config.js +0 -289
  505. package/dist/metrics/config.js.map +0 -1
  506. package/dist/peer-connection-manager/index.d.ts +0 -6
  507. package/dist/peer-connection-manager/index.js +0 -671
  508. package/dist/peer-connection-manager/index.js.map +0 -1
  509. package/dist/peer-connection-manager/util.d.ts +0 -6
  510. package/dist/peer-connection-manager/util.js +0 -110
  511. package/dist/peer-connection-manager/util.js.map +0 -1
  512. package/dist/roap/collection.d.ts +0 -10
  513. package/dist/roap/collection.js +0 -63
  514. package/dist/roap/collection.js.map +0 -1
  515. package/dist/roap/handler.d.ts +0 -47
  516. package/dist/roap/handler.js +0 -279
  517. package/dist/roap/handler.js.map +0 -1
  518. package/dist/roap/state.d.ts +0 -9
  519. package/dist/roap/state.js +0 -127
  520. package/dist/roap/state.js.map +0 -1
  521. package/dist/roap/util.d.ts +0 -2
  522. package/dist/roap/util.js +0 -76
  523. package/dist/roap/util.js.map +0 -1
  524. package/src/index.js +0 -15
  525. package/src/meeting/effectsState.ts +0 -209
  526. package/src/metrics/config.ts +0 -485
  527. package/src/peer-connection-manager/index.ts +0 -847
  528. package/src/peer-connection-manager/util.ts +0 -119
  529. package/src/roap/collection.ts +0 -62
  530. package/src/roap/handler.ts +0 -294
  531. package/src/roap/state.ts +0 -156
  532. package/src/roap/util.ts +0 -100
  533. package/test/unit/spec/meeting/effectsState.js +0 -281
  534. package/test/unit/spec/peerconnection-manager/index.js +0 -218
  535. package/test/unit/spec/peerconnection-manager/utils.js +0 -49
  536. package/test/unit/spec/peerconnection-manager/utils.test-fixtures.ts +0 -388
  537. package/test/unit/spec/roap/util.js +0 -30
  538. /package/dist/common/errors/{reclaim-host-role-error.d.ts → reclaim-host-role-errors.d.ts} +0 -0
  539. /package/src/common/errors/{reclaim-host-role-error.ts → reclaim-host-role-errors.ts} +0 -0
@@ -0,0 +1,166 @@
1
+ /* eslint-disable valid-jsdoc */
2
+ /* eslint-disable import/prefer-default-export */
3
+ import {MediaType, ReceiveSlot as WcmeReceiveSlot} from '@webex/internal-media-core';
4
+ import LoggerProxy from '../common/logs/logger-proxy';
5
+
6
+ import {FindMemberIdCallback, ReceiveSlot} from './receiveSlot';
7
+
8
+ export type CreateSlotCallback = (mediaType: MediaType) => Promise<WcmeReceiveSlot>;
9
+
10
+ export type {CSI, FindMemberIdCallback} from './receiveSlot';
11
+
12
+ /**
13
+ * Manages all receive slots used by a meeting. WMCE receive slots cannot be ever deleted,
14
+ * so this manager has a pool in order to re-use the slots that were released earlier.
15
+ */
16
+ export class ReceiveSlotManager {
17
+ private allocatedSlots: {[key in MediaType]: ReceiveSlot[]};
18
+
19
+ private freeSlots: {[key in MediaType]: ReceiveSlot[]};
20
+
21
+ private createSlotCallback: CreateSlotCallback;
22
+
23
+ private findMemberIdByCsiCallback: FindMemberIdCallback;
24
+
25
+ /**
26
+ * Constructor
27
+ * @param {Meeting} meeting
28
+ */
29
+ constructor(
30
+ createSlotCallback: CreateSlotCallback,
31
+ findMemberIdByCsiCallback: FindMemberIdCallback
32
+ ) {
33
+ this.allocatedSlots = {
34
+ [MediaType.AudioMain]: [],
35
+ [MediaType.VideoMain]: [],
36
+ [MediaType.AudioSlides]: [],
37
+ [MediaType.VideoSlides]: [],
38
+ };
39
+ this.freeSlots = {
40
+ [MediaType.AudioMain]: [],
41
+ [MediaType.VideoMain]: [],
42
+ [MediaType.AudioSlides]: [],
43
+ [MediaType.VideoSlides]: [],
44
+ };
45
+ this.createSlotCallback = createSlotCallback;
46
+ this.findMemberIdByCsiCallback = findMemberIdByCsiCallback;
47
+ }
48
+
49
+ /**
50
+ * Creates a new receive slot or returns one from the existing pool of free slots
51
+ *
52
+ * @param {MediaType} mediaType
53
+ * @returns {Promise<ReceiveSlot>}
54
+ */
55
+ async allocateSlot(mediaType: MediaType): Promise<ReceiveSlot> {
56
+ // try to use one of the free ones
57
+ const availableSlot = this.freeSlots[mediaType].pop();
58
+
59
+ if (availableSlot) {
60
+ this.allocatedSlots[mediaType].push(availableSlot);
61
+
62
+ LoggerProxy.logger.log(`${mediaType}: receive slot re-used: ${availableSlot.id}`);
63
+
64
+ return availableSlot;
65
+ }
66
+
67
+ // we have to create a new one
68
+ const wcmeReceiveSlot = await this.createSlotCallback(mediaType);
69
+
70
+ const receiveSlot = new ReceiveSlot(mediaType, wcmeReceiveSlot, this.findMemberIdByCsiCallback);
71
+
72
+ this.allocatedSlots[mediaType].push(receiveSlot);
73
+ LoggerProxy.logger.log(`${mediaType}: new receive slot allocated: ${receiveSlot.id}`);
74
+
75
+ return receiveSlot;
76
+ }
77
+
78
+ /**
79
+ * Releases the slot back to the pool so it can be re-used by others in the future
80
+ * @param {ReceiveSlot} slot
81
+ */
82
+ releaseSlot(slot: ReceiveSlot) {
83
+ const idx = this.allocatedSlots[slot.mediaType].findIndex(
84
+ (allocatedSlot) => allocatedSlot === slot
85
+ );
86
+
87
+ if (idx >= 0) {
88
+ this.allocatedSlots[slot.mediaType].splice(idx, 1);
89
+ this.freeSlots[slot.mediaType].push(slot);
90
+ LoggerProxy.logger.log(`${slot.mediaType}: receive slot released: ${slot.id}`);
91
+ } else {
92
+ LoggerProxy.logger.warn(
93
+ `ReceiveSlotManager#releaseSlot --> trying to release a ${slot.mediaType}} slot that is not managed by this ReceiveSlotManager`
94
+ );
95
+ }
96
+ }
97
+
98
+ /**
99
+ * Resets the slot manager - this method should be called when the media connection is torn down
100
+ */
101
+ reset() {
102
+ this.allocatedSlots = {
103
+ [MediaType.AudioMain]: [],
104
+ [MediaType.VideoMain]: [],
105
+ [MediaType.AudioSlides]: [],
106
+ [MediaType.VideoSlides]: [],
107
+ };
108
+ this.freeSlots = {
109
+ [MediaType.AudioMain]: [],
110
+ [MediaType.VideoMain]: [],
111
+ [MediaType.AudioSlides]: [],
112
+ [MediaType.VideoSlides]: [],
113
+ };
114
+ }
115
+
116
+ /**
117
+ * Returns statistics about the managed slots
118
+ *
119
+ * @returns {Object}
120
+ */
121
+ getStats() {
122
+ const numAllocatedSlots = {};
123
+ const numFreeSlots = {};
124
+
125
+ Object.keys(this.allocatedSlots).forEach((key) => {
126
+ if (this.allocatedSlots[key].length > 0) {
127
+ numAllocatedSlots[key] = this.allocatedSlots[key].length;
128
+ }
129
+ });
130
+
131
+ Object.keys(this.freeSlots).forEach((key) => {
132
+ if (this.freeSlots[key].length > 0) {
133
+ numFreeSlots[key] = this.freeSlots[key].length;
134
+ }
135
+ });
136
+
137
+ return {
138
+ numAllocatedSlots,
139
+ numFreeSlots,
140
+ };
141
+ }
142
+
143
+ /**
144
+ * Tries to find the member id on all allocated receive slots
145
+ * This function should be called when new members are added to the meeting.
146
+ */
147
+ updateMemberIds() {
148
+ Object.keys(this.allocatedSlots).forEach((key) => {
149
+ this.allocatedSlots[key].forEach((slot: ReceiveSlot) => {
150
+ slot.findMemberId();
151
+ });
152
+ });
153
+ }
154
+
155
+ /**
156
+ * Find a receive slot by a ssrc.
157
+ *
158
+ * @param ssrc - The ssrc of the receive slot to find.
159
+ * @returns - The receive slot with this ssrc, undefined if not found.
160
+ */
161
+ findReceiveSlotBySsrc(ssrc: number): ReceiveSlot | undefined {
162
+ return Object.values(this.allocatedSlots)
163
+ .flat()
164
+ .find((r) => ssrc && r.wcmeReceiveSlot?.id?.ssrc === ssrc);
165
+ }
166
+ }
@@ -0,0 +1,254 @@
1
+ /* eslint-disable valid-jsdoc */
2
+ import {MediaType, StreamState} from '@webex/internal-media-core';
3
+ import LoggerProxy from '../common/logs/logger-proxy';
4
+ import EventsScope from '../common/events/events-scope';
5
+
6
+ import {MediaRequestId, MediaRequestManager} from './mediaRequestManager';
7
+ import {CSI, ReceiveSlot, ReceiveSlotEvents} from './receiveSlot';
8
+
9
+ export const RemoteMediaEvents = {
10
+ SourceUpdate: ReceiveSlotEvents.SourceUpdate,
11
+ Stopped: 'stopped',
12
+ };
13
+
14
+ export type RemoteVideoResolution =
15
+ | 'thumbnail' // the smallest possible resolution, 90p or less
16
+ | 'very small' // 180p or less
17
+ | 'small' // 360p or less
18
+ | 'medium' // 720p or less
19
+ | 'large' // 1080p or less
20
+ | 'best'; // highest possible resolution
21
+
22
+ /**
23
+ * Converts pane size into h264 maxFs
24
+ * @param {PaneSize} paneSize
25
+ * @returns {number}
26
+ */
27
+ export function getMaxFs(paneSize: RemoteVideoResolution): number {
28
+ let maxFs;
29
+
30
+ switch (paneSize) {
31
+ case 'thumbnail':
32
+ maxFs = 60;
33
+ break;
34
+ case 'very small':
35
+ maxFs = 240;
36
+ break;
37
+ case 'small':
38
+ maxFs = 920;
39
+ break;
40
+ case 'medium':
41
+ maxFs = 3600;
42
+ break;
43
+ case 'large':
44
+ maxFs = 8192;
45
+ break;
46
+ case 'best':
47
+ maxFs = 8192; // for now 'best' is 1080p, so same as 'large'
48
+ break;
49
+ default:
50
+ LoggerProxy.logger.warn(
51
+ `RemoteMedia#getMaxFs --> unsupported paneSize: ${paneSize}, using "medium" instead`
52
+ );
53
+ maxFs = 3600;
54
+ }
55
+
56
+ return maxFs;
57
+ }
58
+
59
+ type Options = {
60
+ resolution?: RemoteVideoResolution; // applies only to groups of type MediaType.VideoMain and MediaType.VideoSlides
61
+ };
62
+
63
+ export type RemoteMediaId = string;
64
+
65
+ let remoteMediaCounter = 0;
66
+
67
+ /**
68
+ * Class representing a remote audio/video stream.
69
+ *
70
+ * Internally it is associated with a specific receive slot
71
+ * and a media request for it.
72
+ */
73
+ export class RemoteMedia extends EventsScope {
74
+ private receiveSlot?: ReceiveSlot;
75
+
76
+ private readonly mediaRequestManager: MediaRequestManager;
77
+
78
+ private readonly options: Options;
79
+
80
+ private mediaRequestId?: MediaRequestId;
81
+
82
+ public readonly id: RemoteMediaId;
83
+
84
+ /**
85
+ * Constructs RemoteMedia instance
86
+ *
87
+ * @param receiveSlot
88
+ * @param mediaRequestManager
89
+ * @param options
90
+ */
91
+ constructor(
92
+ receiveSlot: ReceiveSlot,
93
+ mediaRequestManager: MediaRequestManager,
94
+ options?: Options
95
+ ) {
96
+ super();
97
+ remoteMediaCounter += 1;
98
+ this.receiveSlot = receiveSlot;
99
+ this.mediaRequestManager = mediaRequestManager;
100
+ this.options = options || {};
101
+ this.setupEventListeners();
102
+ this.id = `RM${remoteMediaCounter}-${this.receiveSlot.id}`;
103
+ }
104
+
105
+ /**
106
+ * Supply the width and height of the video element
107
+ * to restrict the requested resolution to this size
108
+ * @param width width of the video element
109
+ * @param height height of the video element
110
+ */
111
+ public setSizeHint(width, height) {
112
+ // only base on height for now
113
+ let fs: number;
114
+
115
+ if (height < 135) {
116
+ fs = 60;
117
+ } else if (height < 270) {
118
+ fs = 240;
119
+ } else if (height < 540) {
120
+ fs = 920;
121
+ } else if (height <= 720) {
122
+ fs = 3600;
123
+ } else {
124
+ fs = 8192;
125
+ }
126
+
127
+ this.receiveSlot?.setMaxFs(fs);
128
+ }
129
+
130
+ /**
131
+ * Invalidates the remote media by clearing the reference to a receive slot and
132
+ * cancelling the media request.
133
+ * After this call the remote media is unusable.
134
+ *
135
+ * @param {boolean} commit - whether to commit the cancellation of the media request
136
+ * @internal
137
+ */
138
+ public stop(commit = true) {
139
+ this.cancelMediaRequest(commit);
140
+ this.receiveSlot?.removeAllListeners();
141
+ this.receiveSlot = undefined;
142
+ this.emit(
143
+ {
144
+ file: 'multistream/remoteMedia',
145
+ function: 'stop',
146
+ },
147
+ RemoteMediaEvents.Stopped,
148
+ {}
149
+ );
150
+ }
151
+
152
+ /**
153
+ * Sends a new media request. This method can only be used for receiver-selected policy,
154
+ * because only in that policy we have a 1-1 relationship between RemoteMedia and MediaRequest
155
+ * and the request id is then stored in this RemoteMedia instance.
156
+ * For active-speaker policy, the same request is shared among many RemoteMedia instances,
157
+ * so it's managed through RemoteMediaGroup
158
+ *
159
+ * @internal
160
+ */
161
+ public sendMediaRequest(csi: CSI, commit: boolean) {
162
+ if (this.mediaRequestId) {
163
+ this.cancelMediaRequest(false);
164
+ }
165
+
166
+ if (!this.receiveSlot) {
167
+ throw new Error('sendMediaRequest() called on an invalidated RemoteMedia instance');
168
+ }
169
+
170
+ this.mediaRequestId = this.mediaRequestManager.addRequest(
171
+ {
172
+ policyInfo: {
173
+ policy: 'receiver-selected',
174
+ csi,
175
+ },
176
+ receiveSlots: [this.receiveSlot],
177
+ codecInfo: this.options.resolution && {
178
+ codec: 'h264',
179
+ maxFs: getMaxFs(this.options.resolution),
180
+ },
181
+ },
182
+ commit
183
+ );
184
+ }
185
+
186
+ /**
187
+ * @internal
188
+ */
189
+ public cancelMediaRequest(commit: boolean) {
190
+ if (this.mediaRequestId) {
191
+ this.mediaRequestManager.cancelRequest(this.mediaRequestId, commit);
192
+ this.mediaRequestId = undefined;
193
+ }
194
+ }
195
+
196
+ /**
197
+ * registers event listeners on the receive slot and forwards all the events
198
+ */
199
+ private setupEventListeners() {
200
+ if (this.receiveSlot) {
201
+ const scope = {
202
+ file: 'multistream/remoteMedia',
203
+ function: 'setupEventListeners',
204
+ };
205
+
206
+ this.receiveSlot.on(ReceiveSlotEvents.SourceUpdate, (data) => {
207
+ this.emit(scope, RemoteMediaEvents.SourceUpdate, data);
208
+ });
209
+ }
210
+ }
211
+
212
+ /**
213
+ * Getter for mediaType
214
+ */
215
+ public get mediaType(): MediaType {
216
+ return this.receiveSlot?.mediaType;
217
+ }
218
+
219
+ /**
220
+ * Getter for memberId
221
+ */
222
+ public get memberId() {
223
+ return this.receiveSlot?.memberId;
224
+ }
225
+
226
+ /**
227
+ * Getter for csi
228
+ */
229
+ public get csi() {
230
+ return this.receiveSlot?.csi;
231
+ }
232
+
233
+ /**
234
+ * Getter for source state
235
+ */
236
+ public get sourceState(): StreamState {
237
+ return this.receiveSlot?.sourceState;
238
+ }
239
+
240
+ /**
241
+ * Getter for remote media stream
242
+ */
243
+ public get stream() {
244
+ return this.receiveSlot?.stream;
245
+ }
246
+
247
+ /**
248
+ * @internal
249
+ * @returns {ReceiveSlot}
250
+ */
251
+ public getUnderlyingReceiveSlot() {
252
+ return this.receiveSlot;
253
+ }
254
+ }
@@ -0,0 +1,284 @@
1
+ /* eslint-disable valid-jsdoc */
2
+ /* eslint-disable require-jsdoc */
3
+ /* eslint-disable import/prefer-default-export */
4
+ import {forEach} from 'lodash';
5
+ import LoggerProxy from '../common/logs/logger-proxy';
6
+
7
+ import {getMaxFs, RemoteMedia, RemoteVideoResolution} from './remoteMedia';
8
+ import {MediaRequestId, MediaRequestManager} from './mediaRequestManager';
9
+ import {CSI, ReceiveSlot} from './receiveSlot';
10
+
11
+ type Options = {
12
+ resolution?: RemoteVideoResolution; // applies only to groups of type MediaType.VideoMain and MediaType.VideoSlides
13
+ preferLiveVideo?: boolean; // applies only to groups of type MediaType.VideoMain and MediaType.VideoSlides
14
+ };
15
+
16
+ export class RemoteMediaGroup {
17
+ private mediaRequestManager: MediaRequestManager;
18
+
19
+ private priority: number;
20
+
21
+ private options: Options;
22
+
23
+ private unpinnedRemoteMedia: RemoteMedia[];
24
+
25
+ private mediaRequestId?: MediaRequestId; // id of the "active-speaker" media request id
26
+
27
+ private pinnedRemoteMedia: RemoteMedia[];
28
+
29
+ constructor(
30
+ mediaRequestManager: MediaRequestManager,
31
+ receiveSlots: ReceiveSlot[],
32
+ priority: number,
33
+ commitMediaRequest: boolean,
34
+ options: Options = {}
35
+ ) {
36
+ this.mediaRequestManager = mediaRequestManager;
37
+ this.priority = priority;
38
+ this.options = options;
39
+
40
+ this.unpinnedRemoteMedia = receiveSlots.map(
41
+ (slot) =>
42
+ new RemoteMedia(slot, this.mediaRequestManager, {
43
+ resolution: this.options.resolution,
44
+ })
45
+ );
46
+ this.pinnedRemoteMedia = [];
47
+
48
+ this.sendActiveSpeakerMediaRequest(commitMediaRequest);
49
+ }
50
+
51
+ /**
52
+ * Gets the array of remote media elements from the group
53
+ *
54
+ * @param {string} filter - 'all' (default) returns both pinned and unpinned
55
+ * @returns {Array<RemoteMedia>}
56
+ */
57
+ public getRemoteMedia(filter: 'all' | 'pinned' | 'unpinned' = 'all') {
58
+ if (filter === 'unpinned') {
59
+ // return a shallow copy so that the client cannot modify this.unpinnedRemoteMedia array
60
+ return [...this.unpinnedRemoteMedia];
61
+ }
62
+ if (filter === 'pinned') {
63
+ // return a shallow copy so that the client cannot modify this.pinnedRemoteMedia array
64
+ return [...this.pinnedRemoteMedia];
65
+ }
66
+
67
+ return [...this.unpinnedRemoteMedia, ...this.pinnedRemoteMedia];
68
+ }
69
+
70
+ /**
71
+ * Sets CSIs for multiple RemoteMedia instances belonging to this RemoteMediaGroup.
72
+ * For each entry in the remoteMediaCsis array:
73
+ * - if csi is specified, the RemoteMedia instance is pinned to that CSI
74
+ * - if csi is undefined, the RemoteMedia instance is unpinned
75
+ * @internal
76
+ */
77
+ public setActiveSpeakerCsis(
78
+ remoteMediaCsis: {remoteMedia: RemoteMedia; csi?: number}[],
79
+ commit = true
80
+ ): void {
81
+ forEach(remoteMediaCsis, ({csi, remoteMedia}) => {
82
+ if (csi) {
83
+ if (!(this.pinnedRemoteMedia.indexOf(remoteMedia) >= 0)) {
84
+ const unpinId = this.unpinnedRemoteMedia.indexOf(remoteMedia);
85
+ if (unpinId >= 0) {
86
+ this.unpinnedRemoteMedia.splice(unpinId, 1);
87
+ this.pinnedRemoteMedia.push(remoteMedia);
88
+ } else {
89
+ throw new Error(
90
+ `failed to pin a remote media object ${remoteMedia.id}, because it is not found in this remote media group`
91
+ );
92
+ }
93
+ }
94
+ remoteMedia.sendMediaRequest(csi, false);
95
+ } else {
96
+ if (!(this.unpinnedRemoteMedia.indexOf(remoteMedia) >= 0)) {
97
+ const pinId = this.pinnedRemoteMedia.indexOf(remoteMedia);
98
+ if (pinId >= 0) {
99
+ this.pinnedRemoteMedia.splice(pinId, 1);
100
+ this.unpinnedRemoteMedia.push(remoteMedia);
101
+ } else {
102
+ throw new Error(
103
+ `failed to unpin a remote media object ${remoteMedia.id}, because it is not found in this remote media group`
104
+ );
105
+ }
106
+ }
107
+ remoteMedia.cancelMediaRequest(false);
108
+ }
109
+ });
110
+ this.cancelActiveSpeakerMediaRequest(false);
111
+ this.sendActiveSpeakerMediaRequest(false);
112
+ if (commit) {
113
+ this.mediaRequestManager.commit();
114
+ }
115
+ }
116
+
117
+ /**
118
+ * Pins a specific remote media instance to a specfic CSI, so the media will
119
+ * no longer come from active speaker, but from that CSI.
120
+ * If no CSI is given, the current CSI value is used.
121
+ *
122
+ */
123
+ public pin(remoteMedia: RemoteMedia, csi?: CSI): void {
124
+ // if csi is not specified, use the current one
125
+ const targetCsi = csi || remoteMedia.csi;
126
+
127
+ if (!targetCsi) {
128
+ throw new Error(
129
+ `failed to pin a remote media object ${remoteMedia.id}, because it has no CSI set and no CSI value was given`
130
+ );
131
+ }
132
+
133
+ if (this.pinnedRemoteMedia.indexOf(remoteMedia) >= 0) {
134
+ if (targetCsi === remoteMedia.csi) {
135
+ // remote media already pinned to target CSI, nothing to do
136
+ LoggerProxy.logger.log(
137
+ `RemoteMediaGroup#pin --> remote media ${remoteMedia.id} already pinned`
138
+ );
139
+
140
+ return;
141
+ }
142
+ } else {
143
+ const idx = this.unpinnedRemoteMedia.indexOf(remoteMedia);
144
+
145
+ if (idx < 0) {
146
+ throw new Error(
147
+ `failed to pin a remote media object ${remoteMedia.id}, because it is not found in this remote media group`
148
+ );
149
+ }
150
+
151
+ this.unpinnedRemoteMedia.splice(idx, 1);
152
+ this.pinnedRemoteMedia.push(remoteMedia);
153
+
154
+ this.cancelActiveSpeakerMediaRequest(false);
155
+ this.sendActiveSpeakerMediaRequest(false);
156
+ }
157
+
158
+ remoteMedia.sendMediaRequest(targetCsi, false);
159
+ this.mediaRequestManager.commit();
160
+ }
161
+
162
+ /**
163
+ * Unpins a remote media instance, so that it will again provide media from active speakers
164
+ *
165
+ */
166
+ public unpin(remoteMedia: RemoteMedia) {
167
+ if (this.unpinnedRemoteMedia.indexOf(remoteMedia) >= 0) {
168
+ LoggerProxy.logger.log(
169
+ `RemoteMediaGroup#pin --> remote media ${remoteMedia.id} already unpinned`
170
+ );
171
+
172
+ return;
173
+ }
174
+ const idx = this.pinnedRemoteMedia.indexOf(remoteMedia);
175
+
176
+ if (idx < 0) {
177
+ throw new Error(
178
+ `failed to unpin a remote media object ${remoteMedia.id}, because it is not found in this remote media group`
179
+ );
180
+ }
181
+
182
+ this.pinnedRemoteMedia.splice(idx, 1);
183
+ this.unpinnedRemoteMedia.push(remoteMedia);
184
+
185
+ remoteMedia.cancelMediaRequest(false);
186
+ this.cancelActiveSpeakerMediaRequest(false);
187
+ this.sendActiveSpeakerMediaRequest(false);
188
+ this.mediaRequestManager.commit();
189
+ }
190
+
191
+ public isPinned(remoteMedia: RemoteMedia) {
192
+ if (this.unpinnedRemoteMedia.indexOf(remoteMedia) >= 0) {
193
+ return false;
194
+ }
195
+ if (this.pinnedRemoteMedia.indexOf(remoteMedia) >= 0) {
196
+ return true;
197
+ }
198
+
199
+ throw new Error(`remote media object ${remoteMedia.id} not found in the group`);
200
+ }
201
+
202
+ /**
203
+ * setPreferLiveVideo - sets preferLiveVideo to true/false
204
+ * @internal
205
+ */
206
+ public setPreferLiveVideo(preferLiveVideo: boolean, commit: boolean) {
207
+ if (this.options.preferLiveVideo !== preferLiveVideo) {
208
+ this.options.preferLiveVideo = preferLiveVideo;
209
+ this.sendActiveSpeakerMediaRequest(commit);
210
+ }
211
+ }
212
+
213
+ private sendActiveSpeakerMediaRequest(commit: boolean) {
214
+ this.cancelActiveSpeakerMediaRequest(false);
215
+
216
+ this.mediaRequestId = this.mediaRequestManager.addRequest(
217
+ {
218
+ policyInfo: {
219
+ policy: 'active-speaker',
220
+ priority: this.priority,
221
+ crossPriorityDuplication: false,
222
+ crossPolicyDuplication: false,
223
+ preferLiveVideo: !!this.options?.preferLiveVideo,
224
+ },
225
+ receiveSlots: this.unpinnedRemoteMedia.map((remoteMedia) =>
226
+ remoteMedia.getUnderlyingReceiveSlot()
227
+ ) as ReceiveSlot[],
228
+ codecInfo: this.options.resolution && {
229
+ codec: 'h264',
230
+ maxFs: getMaxFs(this.options.resolution),
231
+ },
232
+ },
233
+ commit
234
+ );
235
+ }
236
+
237
+ private cancelActiveSpeakerMediaRequest(commit: boolean) {
238
+ if (this.mediaRequestId) {
239
+ this.mediaRequestManager.cancelRequest(this.mediaRequestId, commit);
240
+ this.mediaRequestId = undefined;
241
+ }
242
+ }
243
+
244
+ /**
245
+ * Invalidates the remote media group by clearing the references to the receive slots
246
+ * used by all remote media from that group and cancelling all media requests.
247
+ * After this call the remote media group is unusable.
248
+ *
249
+ * @param{boolean} commit whether to commit the cancellation of media requests
250
+ * @internal
251
+ */
252
+ public stop(commit = true) {
253
+ this.unpinnedRemoteMedia.forEach((remoteMedia) => remoteMedia.stop(false));
254
+ this.pinnedRemoteMedia.forEach((remoteMedia) => remoteMedia.stop(false));
255
+ this.cancelActiveSpeakerMediaRequest(false);
256
+
257
+ if (commit) {
258
+ this.mediaRequestManager.commit();
259
+ }
260
+ }
261
+
262
+ /**
263
+ * Checks if a given RemoteMedia instance belongs to this group.
264
+ *
265
+ * @param remoteMedia RemoteMedia instance to check
266
+ * @param filter controls which remote media from the group to check
267
+ * @returns true if remote media is found
268
+ */
269
+ public includes(
270
+ remoteMedia: RemoteMedia,
271
+ filter: 'all' | 'pinned' | 'unpinned' = 'all'
272
+ ): boolean {
273
+ if (filter === 'pinned') {
274
+ return this.pinnedRemoteMedia.includes(remoteMedia);
275
+ }
276
+ if (filter === 'unpinned') {
277
+ return this.unpinnedRemoteMedia.includes(remoteMedia);
278
+ }
279
+
280
+ return (
281
+ this.unpinnedRemoteMedia.includes(remoteMedia) || this.pinnedRemoteMedia.includes(remoteMedia)
282
+ );
283
+ }
284
+ }