@webex/plugin-meetings 2.59.1 → 2.59.3-next.1
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/.eslintrc.js +6 -6
- package/LICENSE +1 -1
- package/README.md +1204 -1204
- package/UPGRADING.md +287 -287
- package/babel.config.js +3 -3
- package/browsers.js +108 -108
- package/dist/common/browser-detection.d.ts +9 -9
- package/dist/common/browser-detection.js.map +1 -1
- package/dist/common/collection.d.ts +48 -48
- package/dist/common/collection.js +43 -43
- package/dist/common/collection.js.map +1 -1
- package/dist/common/config.d.ts +2 -2
- package/dist/common/config.js.map +1 -1
- package/dist/common/errors/captcha-error.d.ts +15 -15
- package/dist/common/errors/captcha-error.js +7 -7
- package/dist/common/errors/captcha-error.js.map +1 -1
- package/dist/common/errors/intent-to-join.d.ts +16 -16
- package/dist/common/errors/intent-to-join.js +7 -7
- package/dist/common/errors/intent-to-join.js.map +1 -1
- package/dist/common/errors/join-meeting.d.ts +17 -17
- package/dist/common/errors/join-meeting.js +8 -8
- package/dist/common/errors/join-meeting.js.map +1 -1
- package/dist/common/errors/media.d.ts +15 -15
- package/dist/common/errors/media.js +7 -7
- package/dist/common/errors/media.js.map +1 -1
- package/dist/common/errors/parameter.d.ts +15 -15
- package/dist/common/errors/parameter.js +7 -7
- package/dist/common/errors/parameter.js.map +1 -1
- package/dist/common/errors/password-error.d.ts +15 -15
- package/dist/common/errors/password-error.js +7 -7
- package/dist/common/errors/password-error.js.map +1 -1
- package/dist/common/errors/permission.d.ts +14 -14
- package/dist/common/errors/permission.js +6 -6
- package/dist/common/errors/permission.js.map +1 -1
- package/dist/common/errors/reconnection-in-progress.d.ts +9 -9
- package/dist/common/errors/reconnection-in-progress.js +6 -6
- package/dist/common/errors/reconnection-in-progress.js.map +1 -1
- package/dist/common/errors/reconnection.d.ts +15 -15
- package/dist/common/errors/reconnection.js +7 -7
- package/dist/common/errors/reconnection.js.map +1 -1
- package/dist/common/errors/stats.d.ts +15 -15
- package/dist/common/errors/stats.js +7 -7
- package/dist/common/errors/stats.js.map +1 -1
- package/dist/common/errors/webex-errors.d.ts +81 -81
- package/dist/common/errors/webex-errors.js +42 -42
- package/dist/common/errors/webex-errors.js.map +1 -1
- package/dist/common/errors/webex-meetings-error.d.ts +20 -20
- package/dist/common/errors/webex-meetings-error.js +12 -12
- package/dist/common/errors/webex-meetings-error.js.map +1 -1
- package/dist/common/events/events-scope.d.ts +17 -17
- package/dist/common/events/events-scope.js +10 -10
- package/dist/common/events/events-scope.js.map +1 -1
- package/dist/common/events/events.d.ts +12 -12
- package/dist/common/events/events.js +4 -4
- package/dist/common/events/events.js.map +1 -1
- package/dist/common/events/trigger-proxy.d.ts +2 -2
- package/dist/common/events/trigger-proxy.js.map +1 -1
- package/dist/common/events/util.d.ts +2 -2
- package/dist/common/events/util.js.map +1 -1
- package/dist/common/logs/logger-config.d.ts +2 -2
- package/dist/common/logs/logger-config.js.map +1 -1
- package/dist/common/logs/logger-proxy.d.ts +2 -2
- package/dist/common/logs/logger-proxy.js.map +1 -1
- package/dist/common/logs/request.d.ts +34 -34
- package/dist/common/logs/request.js +18 -18
- package/dist/common/logs/request.js.map +1 -1
- package/dist/common/queue.d.ts +32 -32
- package/dist/common/queue.js +18 -18
- package/dist/common/queue.js.map +1 -1
- package/dist/config.d.ts +73 -73
- package/dist/config.js.map +1 -1
- package/dist/constants.d.ts +924 -924
- package/dist/constants.js +9 -9
- package/dist/constants.js.map +1 -1
- package/dist/controls-options-manager/constants.d.ts +4 -4
- package/dist/controls-options-manager/constants.js.map +1 -1
- package/dist/controls-options-manager/enums.d.ts +5 -5
- package/dist/controls-options-manager/enums.js.map +1 -1
- package/dist/controls-options-manager/index.d.ts +120 -120
- package/dist/controls-options-manager/index.js +81 -81
- package/dist/controls-options-manager/index.js.map +1 -1
- package/dist/controls-options-manager/util.d.ts +7 -7
- package/dist/controls-options-manager/util.js.map +1 -1
- package/dist/index.d.ts +4 -4
- package/dist/index.js.map +1 -1
- package/dist/locus-info/controlsUtils.d.ts +2 -2
- package/dist/locus-info/controlsUtils.js +21 -21
- package/dist/locus-info/controlsUtils.js.map +1 -1
- package/dist/locus-info/embeddedAppsUtils.d.ts +2 -2
- package/dist/locus-info/embeddedAppsUtils.js +14 -14
- package/dist/locus-info/embeddedAppsUtils.js.map +1 -1
- package/dist/locus-info/fullState.d.ts +2 -2
- package/dist/locus-info/fullState.js.map +1 -1
- package/dist/locus-info/hostUtils.d.ts +2 -2
- package/dist/locus-info/hostUtils.js +19 -19
- package/dist/locus-info/hostUtils.js.map +1 -1
- package/dist/locus-info/index.d.ts +269 -269
- package/dist/locus-info/index.js +180 -180
- package/dist/locus-info/index.js.map +1 -1
- package/dist/locus-info/infoUtils.d.ts +2 -2
- package/dist/locus-info/infoUtils.js.map +1 -1
- package/dist/locus-info/mediaSharesUtils.d.ts +2 -2
- package/dist/locus-info/mediaSharesUtils.js +50 -50
- package/dist/locus-info/mediaSharesUtils.js.map +1 -1
- package/dist/locus-info/parser.d.ts +212 -212
- package/dist/locus-info/parser.js +136 -136
- package/dist/locus-info/parser.js.map +1 -1
- package/dist/locus-info/selfUtils.d.ts +2 -2
- package/dist/locus-info/selfUtils.js +52 -52
- package/dist/locus-info/selfUtils.js.map +1 -1
- package/dist/media/index.d.ts +32 -32
- package/dist/media/index.js +145 -145
- package/dist/media/index.js.map +1 -1
- package/dist/media/properties.d.ts +108 -108
- package/dist/media/properties.js +49 -49
- package/dist/media/properties.js.map +1 -1
- package/dist/media/util.d.ts +2 -2
- package/dist/media/util.js.map +1 -1
- package/dist/mediaQualityMetrics/config.d.ts +233 -233
- package/dist/mediaQualityMetrics/config.js.map +1 -1
- package/dist/meeting/effectsState.d.ts +42 -42
- package/dist/meeting/effectsState.js +24 -24
- package/dist/meeting/effectsState.js.map +1 -1
- package/dist/meeting/in-meeting-actions.d.ts +79 -79
- package/dist/meeting/in-meeting-actions.js +11 -11
- package/dist/meeting/in-meeting-actions.js.map +1 -1
- package/dist/meeting/index.d.ts +1621 -1621
- package/dist/meeting/index.js +1502 -1505
- package/dist/meeting/index.js.map +1 -1
- package/dist/meeting/muteState.d.ts +116 -116
- package/dist/meeting/muteState.js +85 -85
- package/dist/meeting/muteState.js.map +1 -1
- package/dist/meeting/request.d.ts +255 -255
- package/dist/meeting/request.js +141 -141
- package/dist/meeting/request.js.map +1 -1
- package/dist/meeting/state.d.ts +9 -9
- package/dist/meeting/state.js +30 -30
- package/dist/meeting/state.js.map +1 -1
- package/dist/meeting/util.d.ts +2 -2
- package/dist/meeting/util.js.map +1 -1
- package/dist/meeting-info/collection.d.ts +20 -20
- package/dist/meeting-info/collection.js +11 -11
- package/dist/meeting-info/collection.js.map +1 -1
- package/dist/meeting-info/index.d.ts +57 -57
- package/dist/meeting-info/index.js +50 -50
- package/dist/meeting-info/index.js.map +1 -1
- package/dist/meeting-info/meeting-info-v2.d.ts +93 -93
- package/dist/meeting-info/meeting-info-v2.js +52 -52
- package/dist/meeting-info/meeting-info-v2.js.map +1 -1
- package/dist/meeting-info/request.d.ts +22 -22
- package/dist/meeting-info/request.js +14 -14
- package/dist/meeting-info/request.js.map +1 -1
- package/dist/meeting-info/util.d.ts +2 -2
- package/dist/meeting-info/util.js +9 -9
- package/dist/meeting-info/util.js.map +1 -1
- package/dist/meeting-info/utilv2.d.ts +2 -2
- package/dist/meeting-info/utilv2.js +20 -20
- package/dist/meeting-info/utilv2.js.map +1 -1
- package/dist/meetings/collection.d.ts +23 -23
- package/dist/meetings/collection.js +14 -14
- package/dist/meetings/collection.js.map +1 -1
- package/dist/meetings/index.d.ts +296 -296
- package/dist/meetings/index.js +259 -259
- package/dist/meetings/index.js.map +1 -1
- package/dist/meetings/request.d.ts +27 -27
- package/dist/meetings/request.js +15 -15
- package/dist/meetings/request.js.map +1 -1
- package/dist/meetings/util.d.ts +18 -18
- package/dist/meetings/util.js +29 -29
- package/dist/meetings/util.js.map +1 -1
- package/dist/member/index.d.ts +147 -147
- package/dist/member/index.js +214 -214
- package/dist/member/index.js.map +1 -1
- package/dist/member/member.types.d.ts +11 -11
- package/dist/member/member.types.js.map +1 -1
- package/dist/member/util.d.ts +2 -2
- package/dist/member/util.js +60 -60
- package/dist/member/util.js.map +1 -1
- package/dist/members/collection.d.ts +24 -24
- package/dist/members/collection.js +11 -11
- package/dist/members/collection.js.map +1 -1
- package/dist/members/index.d.ts +298 -298
- package/dist/members/index.js +275 -275
- package/dist/members/index.js.map +1 -1
- package/dist/members/request.d.ts +50 -50
- package/dist/members/request.js +27 -27
- package/dist/members/request.js.map +1 -1
- package/dist/members/util.d.ts +2 -2
- package/dist/members/util.js +21 -21
- package/dist/members/util.js.map +1 -1
- package/dist/metrics/config.d.ts +169 -169
- package/dist/metrics/config.js.map +1 -1
- package/dist/metrics/constants.d.ts +57 -57
- package/dist/metrics/constants.js.map +1 -1
- package/dist/metrics/index.d.ts +152 -152
- package/dist/metrics/index.js +90 -90
- package/dist/metrics/index.js.map +1 -1
- package/dist/networkQualityMonitor/index.d.ts +70 -70
- package/dist/networkQualityMonitor/index.js +65 -65
- package/dist/networkQualityMonitor/index.js.map +1 -1
- package/dist/peer-connection-manager/index.d.ts +6 -6
- package/dist/peer-connection-manager/index.js +87 -87
- package/dist/peer-connection-manager/index.js.map +1 -1
- package/dist/peer-connection-manager/util.d.ts +6 -6
- package/dist/peer-connection-manager/util.js +9 -9
- package/dist/peer-connection-manager/util.js.map +1 -1
- package/dist/personal-meeting-room/index.d.ts +47 -47
- package/dist/personal-meeting-room/index.js +67 -67
- package/dist/personal-meeting-room/index.js.map +1 -1
- package/dist/personal-meeting-room/request.d.ts +14 -14
- package/dist/personal-meeting-room/request.js +7 -7
- package/dist/personal-meeting-room/request.js.map +1 -1
- package/dist/personal-meeting-room/util.d.ts +2 -2
- package/dist/personal-meeting-room/util.js.map +1 -1
- package/dist/reachability/index.d.ts +139 -139
- package/dist/reachability/index.js +110 -110
- package/dist/reachability/index.js.map +1 -1
- package/dist/reachability/request.d.ts +35 -35
- package/dist/reachability/request.js +15 -15
- package/dist/reachability/request.js.map +1 -1
- package/dist/reactions/reactions.d.ts +4 -4
- package/dist/reactions/reactions.js.map +1 -1
- package/dist/reactions/reactions.type.d.ts +32 -32
- package/dist/reactions/reactions.type.js.map +1 -1
- package/dist/reconnection-manager/index.d.ts +112 -112
- package/dist/reconnection-manager/index.js +112 -112
- package/dist/reconnection-manager/index.js.map +1 -1
- package/dist/recording-controller/enums.d.ts +7 -7
- package/dist/recording-controller/enums.js.map +1 -1
- package/dist/recording-controller/index.d.ts +193 -193
- package/dist/recording-controller/index.js +127 -127
- package/dist/recording-controller/index.js.map +1 -1
- package/dist/recording-controller/util.d.ts +13 -13
- package/dist/recording-controller/util.js.map +1 -1
- package/dist/roap/collection.d.ts +10 -10
- package/dist/roap/collection.js.map +1 -1
- package/dist/roap/handler.d.ts +47 -47
- package/dist/roap/handler.js +27 -27
- package/dist/roap/handler.js.map +1 -1
- package/dist/roap/index.d.ts +116 -116
- package/dist/roap/index.js +111 -111
- package/dist/roap/index.js.map +1 -1
- package/dist/roap/request.d.ts +35 -35
- package/dist/roap/request.js +17 -17
- package/dist/roap/request.js.map +1 -1
- package/dist/roap/state.d.ts +9 -9
- package/dist/roap/state.js +14 -14
- package/dist/roap/state.js.map +1 -1
- package/dist/roap/turnDiscovery.d.ts +67 -67
- package/dist/roap/turnDiscovery.js +46 -46
- package/dist/roap/turnDiscovery.js.map +1 -1
- package/dist/roap/util.d.ts +2 -2
- package/dist/roap/util.js.map +1 -1
- package/dist/statsAnalyzer/global.d.ts +126 -126
- package/dist/statsAnalyzer/global.js.map +1 -1
- package/dist/statsAnalyzer/index.d.ts +190 -190
- package/dist/statsAnalyzer/index.js +128 -128
- package/dist/statsAnalyzer/index.js.map +1 -1
- package/dist/statsAnalyzer/mqaUtil.d.ts +22 -22
- package/dist/statsAnalyzer/mqaUtil.js.map +1 -1
- package/dist/transcription/index.d.ts +64 -64
- package/dist/transcription/index.js +42 -42
- package/dist/transcription/index.js.map +1 -1
- package/internal-README.md +172 -172
- package/jest.config.js +3 -3
- package/package.json +22 -21
- package/process +1 -1
- package/src/common/browser-detection.ts +39 -39
- package/src/common/collection.ts +94 -94
- package/src/common/config.ts +9 -9
- package/src/common/errors/captcha-error.ts +25 -25
- package/src/common/errors/intent-to-join.ts +27 -27
- package/src/common/errors/join-meeting.ts +32 -32
- package/src/common/errors/media.ts +25 -25
- package/src/common/errors/parameter.ts +33 -33
- package/src/common/errors/password-error.ts +25 -25
- package/src/common/errors/permission.ts +24 -24
- package/src/common/errors/reconnection-in-progress.ts +8 -8
- package/src/common/errors/reconnection.ts +25 -25
- package/src/common/errors/stats.ts +25 -25
- package/src/common/errors/webex-errors.ts +140 -140
- package/src/common/errors/webex-meetings-error.ts +35 -35
- package/src/common/events/events-scope.ts +30 -30
- package/src/common/events/events.ts +25 -25
- package/src/common/events/trigger-proxy.ts +25 -25
- package/src/common/events/util.ts +39 -39
- package/src/common/logs/logger-config.ts +8 -8
- package/src/common/logs/logger-proxy.ts +44 -44
- package/src/common/logs/request.ts +65 -65
- package/src/common/queue.ts +50 -50
- package/src/config.ts +96 -96
- package/src/constants.ts +1121 -1121
- package/src/controls-options-manager/constants.ts +5 -5
- package/src/controls-options-manager/enums.ts +6 -6
- package/src/controls-options-manager/index.ts +183 -183
- package/src/controls-options-manager/util.ts +20 -20
- package/src/index.js +15 -15
- package/src/locus-info/controlsUtils.ts +112 -112
- package/src/locus-info/embeddedAppsUtils.ts +57 -57
- package/src/locus-info/fullState.ts +69 -69
- package/src/locus-info/hostUtils.ts +60 -60
- package/src/locus-info/index.ts +1303 -1303
- package/src/locus-info/infoUtils.ts +101 -101
- package/src/locus-info/mediaSharesUtils.ts +173 -173
- package/src/locus-info/parser.ts +680 -680
- package/src/locus-info/selfUtils.ts +428 -428
- package/src/media/index.ts +675 -675
- package/src/media/properties.ts +313 -313
- package/src/media/util.ts +37 -37
- package/src/mediaQualityMetrics/config.ts +382 -382
- package/src/meeting/effectsState.ts +209 -209
- package/src/meeting/in-meeting-actions.ts +153 -153
- package/src/meeting/index.ts +6537 -6543
- package/src/meeting/muteState.ts +365 -365
- package/src/meeting/request.ts +810 -810
- package/src/meeting/state.ts +194 -194
- package/src/meeting/util.ts +530 -530
- package/src/meeting-info/collection.ts +41 -41
- package/src/meeting-info/index.ts +137 -137
- package/src/meeting-info/meeting-info-v2.ts +273 -273
- package/src/meeting-info/request.ts +46 -46
- package/src/meeting-info/util.ts +314 -314
- package/src/meeting-info/utilv2.ts +324 -324
- package/src/meetings/collection.ts +43 -43
- package/src/meetings/index.ts +1128 -1128
- package/src/meetings/request.ts +81 -81
- package/src/meetings/util.ts +181 -181
- package/src/member/index.ts +446 -446
- package/src/member/member.types.ts +13 -13
- package/src/member/util.ts +286 -286
- package/src/members/collection.ts +40 -40
- package/src/members/index.ts +900 -900
- package/src/members/request.ts +175 -175
- package/src/members/util.ts +260 -260
- package/src/metrics/config.ts +485 -485
- package/src/metrics/constants.ts +61 -61
- package/src/metrics/index.ts +543 -543
- package/src/networkQualityMonitor/index.ts +211 -211
- package/src/peer-connection-manager/index.ts +847 -847
- package/src/peer-connection-manager/util.ts +119 -119
- package/src/personal-meeting-room/index.ts +157 -157
- package/src/personal-meeting-room/request.ts +48 -48
- package/src/personal-meeting-room/util.ts +49 -49
- package/src/reachability/index.ts +478 -478
- package/src/reachability/request.ts +81 -81
- package/src/reactions/reactions.ts +104 -104
- package/src/reactions/reactions.type.ts +36 -36
- package/src/reconnection-manager/index.ts +622 -622
- package/src/recording-controller/enums.ts +8 -8
- package/src/recording-controller/index.ts +315 -315
- package/src/recording-controller/util.ts +58 -58
- package/src/roap/collection.ts +62 -62
- package/src/roap/handler.ts +294 -294
- package/src/roap/index.ts +413 -413
- package/src/roap/request.ts +229 -229
- package/src/roap/state.ts +156 -156
- package/src/roap/turnDiscovery.ts +283 -283
- package/src/roap/util.ts +100 -100
- package/src/statsAnalyzer/global.ts +128 -128
- package/src/statsAnalyzer/index.ts +1266 -1266
- package/src/statsAnalyzer/mqaUtil.ts +290 -290
- package/src/transcription/index.ts +154 -154
- package/test/integration/spec/journey.js +941 -941
- package/test/integration/spec/space-meeting.js +457 -457
- package/test/integration/spec/transcription.js +55 -55
- package/test/unit/spec/common/browser-detection.js +119 -119
- package/test/unit/spec/common/queue.js +69 -69
- package/test/unit/spec/controls-options-manager/index.js +123 -123
- package/test/unit/spec/controls-options-manager/util.js +65 -65
- package/test/unit/spec/fixture/locus.js +406 -406
- package/test/unit/spec/locus-info/controlsUtils.js +82 -82
- package/test/unit/spec/locus-info/embeddedAppsUtils.js +104 -104
- package/test/unit/spec/locus-info/index.js +1272 -1272
- package/test/unit/spec/locus-info/infoUtils.js +138 -138
- package/test/unit/spec/locus-info/lib/BasicSeqCmp.json +975 -975
- package/test/unit/spec/locus-info/lib/SeqCmp.json +522 -522
- package/test/unit/spec/locus-info/lib/selfConstant.js +286 -286
- package/test/unit/spec/locus-info/parser.js +298 -298
- package/test/unit/spec/locus-info/selfUtils.js +185 -185
- package/test/unit/spec/media/properties.ts +305 -305
- package/test/unit/spec/meeting/effectsState.js +281 -281
- package/test/unit/spec/meeting/in-meeting-actions.ts +90 -90
- package/test/unit/spec/meeting/index.js +5227 -5227
- package/test/unit/spec/meeting/muteState.js +430 -430
- package/test/unit/spec/meeting/request.js +317 -317
- package/test/unit/spec/meeting/utils.js +319 -319
- package/test/unit/spec/meeting-info/meetinginfov2.js +376 -376
- package/test/unit/spec/meeting-info/request.js +64 -64
- package/test/unit/spec/meeting-info/util.js +37 -37
- package/test/unit/spec/meeting-info/utilv2.js +330 -330
- package/test/unit/spec/meetings/collection.js +52 -52
- package/test/unit/spec/meetings/index.js +1375 -1375
- package/test/unit/spec/meetings/utils.js +66 -66
- package/test/unit/spec/member/index.js +47 -47
- package/test/unit/spec/member/util.js +80 -80
- package/test/unit/spec/members/index.js +364 -364
- package/test/unit/spec/members/request.js +200 -200
- package/test/unit/spec/members/utils.js +42 -42
- package/test/unit/spec/metrics/index.js +111 -111
- package/test/unit/spec/networkQualityMonitor/index.js +99 -99
- package/test/unit/spec/peerconnection-manager/index.js +218 -218
- package/test/unit/spec/peerconnection-manager/utils.js +49 -49
- package/test/unit/spec/peerconnection-manager/utils.test-fixtures.ts +388 -388
- package/test/unit/spec/personal-meeting-room/personal-meeting-room.js +29 -29
- package/test/unit/spec/reachability/index.ts +50 -50
- package/test/unit/spec/reconnection-manager/index.js +206 -206
- package/test/unit/spec/recording-controller/index.js +230 -230
- package/test/unit/spec/recording-controller/util.js +101 -101
- package/test/unit/spec/roap/index.ts +128 -128
- package/test/unit/spec/roap/turnDiscovery.ts +372 -372
- package/test/unit/spec/roap/util.js +30 -30
- package/test/unit/spec/stats-analyzer/index.js +287 -287
- package/test/utils/cmr.js +104 -104
- package/test/utils/testUtils.js +287 -287
- package/test/utils/webex-config.js +77 -77
- package/test/utils/webex-test-users.js +82 -82
- package/tsconfig.json +5 -5
package/src/metrics/index.ts
CHANGED
|
@@ -1,543 +1,543 @@
|
|
|
1
|
-
/*!
|
|
2
|
-
* Copyright (c) 2015-2020 Cisco Systems, Inc. See LICENSE file.
|
|
3
|
-
*/
|
|
4
|
-
import util from 'util';
|
|
5
|
-
|
|
6
|
-
import {includes} from 'lodash';
|
|
7
|
-
import uuid from 'uuid';
|
|
8
|
-
// @ts-ignore
|
|
9
|
-
import window from 'global/window';
|
|
10
|
-
import anonymize from 'ip-anonymize';
|
|
11
|
-
|
|
12
|
-
// @ts-ignore
|
|
13
|
-
import {getOSNameInternal} from '@webex/internal-plugin-metrics';
|
|
14
|
-
import LoggerProxy from '../common/logs/logger-proxy';
|
|
15
|
-
import {MEETING_ERRORS} from '../constants';
|
|
16
|
-
import BrowserDetection from '../common/browser-detection';
|
|
17
|
-
|
|
18
|
-
import {
|
|
19
|
-
error,
|
|
20
|
-
eventType,
|
|
21
|
-
errorCodes as ERROR_CODE,
|
|
22
|
-
UNKNOWN,
|
|
23
|
-
CLIENT_NAME,
|
|
24
|
-
mediaType,
|
|
25
|
-
} from './config';
|
|
26
|
-
|
|
27
|
-
const {getOSName, getOSVersion, getBrowserName, getBrowserVersion} = BrowserDetection();
|
|
28
|
-
|
|
29
|
-
// Apply a CIDR /28 format to the IPV4 and /96 to the IPV6 addresses
|
|
30
|
-
// For reference : https://en.wikipedia.org/wiki/Classless_Inter-Domain_Routing
|
|
31
|
-
const anonymizeIPAddress = (localIp) => anonymize(localIp, 28, 96);
|
|
32
|
-
|
|
33
|
-
const triggerTimers = ({event, meeting, data}) => {
|
|
34
|
-
switch (event) {
|
|
35
|
-
case eventType.CALL_INITIATED:
|
|
36
|
-
meeting.setStartCallInitiateJoinReq();
|
|
37
|
-
break;
|
|
38
|
-
case eventType.LOCUS_JOIN_REQUEST:
|
|
39
|
-
meeting.setEndCallInitiateJoinReq();
|
|
40
|
-
meeting.setStartJoinReqResp();
|
|
41
|
-
break;
|
|
42
|
-
case eventType.LOCUS_JOIN_RESPONSE:
|
|
43
|
-
meeting.setEndJoinReqResp();
|
|
44
|
-
meeting.setStartSetupDelay(mediaType.AUDIO);
|
|
45
|
-
meeting.setStartSetupDelay(mediaType.VIDEO);
|
|
46
|
-
meeting.setStartSendingMediaDelay(mediaType.AUDIO);
|
|
47
|
-
meeting.setStartSendingMediaDelay(mediaType.VIDEO);
|
|
48
|
-
break;
|
|
49
|
-
case eventType.RECEIVING_MEDIA_START:
|
|
50
|
-
meeting.setEndSetupDelay(data.mediaType);
|
|
51
|
-
break;
|
|
52
|
-
case eventType.SENDING_MEDIA_START:
|
|
53
|
-
meeting.setEndSendingMediaDelay(data.mediaType);
|
|
54
|
-
break;
|
|
55
|
-
case eventType.LOCAL_SDP_GENERATED:
|
|
56
|
-
meeting.setStartLocalSDPGenRemoteSDPRecvDelay();
|
|
57
|
-
break;
|
|
58
|
-
case eventType.REMOTE_SDP_RECEIVED:
|
|
59
|
-
meeting.setEndLocalSDPGenRemoteSDPRecvDelay();
|
|
60
|
-
break;
|
|
61
|
-
default:
|
|
62
|
-
break;
|
|
63
|
-
}
|
|
64
|
-
};
|
|
65
|
-
|
|
66
|
-
/**
|
|
67
|
-
* @description Metrics handles all the call metrics events
|
|
68
|
-
* @export
|
|
69
|
-
* @class Metrics
|
|
70
|
-
*/
|
|
71
|
-
class Metrics {
|
|
72
|
-
static instance: Metrics;
|
|
73
|
-
|
|
74
|
-
_events: any;
|
|
75
|
-
keys: any;
|
|
76
|
-
meetingCollection: any;
|
|
77
|
-
webex: any;
|
|
78
|
-
|
|
79
|
-
/**
|
|
80
|
-
* Create Metrics Object
|
|
81
|
-
* @constructor
|
|
82
|
-
* @public
|
|
83
|
-
* @memberof Meetings
|
|
84
|
-
*/
|
|
85
|
-
constructor() {
|
|
86
|
-
if (!Metrics.instance) {
|
|
87
|
-
/**
|
|
88
|
-
* @instance
|
|
89
|
-
* @type {Array}
|
|
90
|
-
* @private
|
|
91
|
-
* @memberof Metrics
|
|
92
|
-
*/
|
|
93
|
-
this._events = [];
|
|
94
|
-
/**
|
|
95
|
-
* @instance
|
|
96
|
-
* @type {MeetingCollection}
|
|
97
|
-
* @private
|
|
98
|
-
* @memberof Metrics
|
|
99
|
-
*/
|
|
100
|
-
this.meetingCollection = null;
|
|
101
|
-
/**
|
|
102
|
-
* @instance
|
|
103
|
-
* @type {MeetingCollection}
|
|
104
|
-
* @private
|
|
105
|
-
* @memberof Metrics
|
|
106
|
-
*/
|
|
107
|
-
this.keys = Object.values(eventType);
|
|
108
|
-
/**
|
|
109
|
-
* @instance
|
|
110
|
-
* @type {Metrics}
|
|
111
|
-
* @private
|
|
112
|
-
* @memberof Metrics
|
|
113
|
-
*/
|
|
114
|
-
Metrics.instance = this;
|
|
115
|
-
}
|
|
116
|
-
|
|
117
|
-
// eslint-disable-next-line no-constructor-return
|
|
118
|
-
return Metrics.instance;
|
|
119
|
-
}
|
|
120
|
-
|
|
121
|
-
/**
|
|
122
|
-
* Initializes the Metrics singleton with a meeting Collection.
|
|
123
|
-
*
|
|
124
|
-
* @param {Object} meetingCollection meetings object
|
|
125
|
-
* @param {Object} webex webex SDK object
|
|
126
|
-
*
|
|
127
|
-
* @returns {void}
|
|
128
|
-
*/
|
|
129
|
-
initialSetup(meetingCollection: object, webex: object) {
|
|
130
|
-
this.meetingCollection = meetingCollection;
|
|
131
|
-
this.webex = webex;
|
|
132
|
-
}
|
|
133
|
-
|
|
134
|
-
/**
|
|
135
|
-
* poste Meeting event metrics
|
|
136
|
-
* @param {object} options {meetingId/meeting} as a json object
|
|
137
|
-
* @param {Meeting} options.meeting Meeting object
|
|
138
|
-
* @param {String} options.meetingId
|
|
139
|
-
* @param {object} options.data
|
|
140
|
-
* @param {object} options.event
|
|
141
|
-
* @returns {object} null
|
|
142
|
-
*/
|
|
143
|
-
postEvent(options: {meeting?: any; meetingId?: string; data?: object; event?: any} | any) {
|
|
144
|
-
const {meetingId, data = {}, event} = options;
|
|
145
|
-
let {meeting} = options;
|
|
146
|
-
|
|
147
|
-
if (this.keys.indexOf(event) === -1) {
|
|
148
|
-
LoggerProxy.logger.error(
|
|
149
|
-
`Metrics:index#postEvent --> Event ${event} doesn't exist in dictionary`
|
|
150
|
-
);
|
|
151
|
-
}
|
|
152
|
-
|
|
153
|
-
if (!meeting && meetingId) {
|
|
154
|
-
meeting = this.meetingCollection.get(meetingId);
|
|
155
|
-
options.meeting = meeting;
|
|
156
|
-
}
|
|
157
|
-
|
|
158
|
-
if (meeting) {
|
|
159
|
-
triggerTimers(options);
|
|
160
|
-
|
|
161
|
-
if (!meeting.callEvents) {
|
|
162
|
-
meeting.callEvents = [];
|
|
163
|
-
}
|
|
164
|
-
if (event === eventType.MEDIA_QUALITY) {
|
|
165
|
-
data.event = event;
|
|
166
|
-
meeting.sendMediaQualityAnalyzerMetrics(data);
|
|
167
|
-
} else {
|
|
168
|
-
meeting.callEvents.push(event);
|
|
169
|
-
data.event = event;
|
|
170
|
-
meeting.sendCallAnalyzerMetrics(data);
|
|
171
|
-
}
|
|
172
|
-
} else {
|
|
173
|
-
LoggerProxy.logger.info(
|
|
174
|
-
`Metrics:index#postEvent --> Event received for meetingId:${meetingId}, but meeting not found in collection.`
|
|
175
|
-
);
|
|
176
|
-
}
|
|
177
|
-
}
|
|
178
|
-
|
|
179
|
-
/**
|
|
180
|
-
* Docs for Call analyzer metrics
|
|
181
|
-
* https://sqbu-github.cisco.com/WebExSquared/call-analyzer/wiki
|
|
182
|
-
* https://sqbu-github.cisco.com/WebExSquared/event-dictionary/blob/master/diagnostic-events.raml
|
|
183
|
-
*/
|
|
184
|
-
|
|
185
|
-
// eslint-disable-next-line @typescript-eslint/no-shadow
|
|
186
|
-
initPayload(eventType, identifiers, options) {
|
|
187
|
-
const payload: any = {
|
|
188
|
-
eventId: uuid.v4(),
|
|
189
|
-
version: 1,
|
|
190
|
-
origin: {
|
|
191
|
-
name: 'endpoint',
|
|
192
|
-
networkType: 'unknown',
|
|
193
|
-
userAgent: this.userAgentToString(),
|
|
194
|
-
clientInfo: {
|
|
195
|
-
clientType: options.clientType,
|
|
196
|
-
clientVersion: `${CLIENT_NAME}/${this.webex.version}`,
|
|
197
|
-
localNetworkPrefix: anonymizeIPAddress(this.webex.meetings.geoHintInfo?.clientAddress),
|
|
198
|
-
osVersion: getOSVersion() || 'unknown',
|
|
199
|
-
subClientType: options.subClientType,
|
|
200
|
-
os: getOSNameInternal(),
|
|
201
|
-
browser: getBrowserName(),
|
|
202
|
-
browserVersion: getBrowserVersion(),
|
|
203
|
-
},
|
|
204
|
-
},
|
|
205
|
-
originTime: {
|
|
206
|
-
triggered: new Date().toISOString(),
|
|
207
|
-
},
|
|
208
|
-
senderCountryCode: this.webex.meetings.geoHintInfo?.countryCode,
|
|
209
|
-
event: {
|
|
210
|
-
name: eventType,
|
|
211
|
-
canProceed: true,
|
|
212
|
-
identifiers,
|
|
213
|
-
eventData: {webClientDomain: window.location.hostname},
|
|
214
|
-
},
|
|
215
|
-
};
|
|
216
|
-
|
|
217
|
-
// TODO: more options should be checked and some of them should be mandatory in certain conditions
|
|
218
|
-
if (options) {
|
|
219
|
-
if (Object.prototype.hasOwnProperty.call(options, 'canProceed')) {
|
|
220
|
-
payload.event.canProceed = options.canProceed;
|
|
221
|
-
}
|
|
222
|
-
if (options.errors) {
|
|
223
|
-
payload.event.errors = options.errors;
|
|
224
|
-
}
|
|
225
|
-
if (options.mediaType) {
|
|
226
|
-
payload.event.mediaType = options.mediaType;
|
|
227
|
-
}
|
|
228
|
-
if (options.trigger) {
|
|
229
|
-
payload.event.trigger = options.trigger;
|
|
230
|
-
}
|
|
231
|
-
if (options.pstnAudioType) {
|
|
232
|
-
payload.event.pstnAudioType = options.pstnAudioType;
|
|
233
|
-
}
|
|
234
|
-
if (options.mediaCapabilities) {
|
|
235
|
-
payload.event.mediaCapabilities = options.mediaCapabilities;
|
|
236
|
-
}
|
|
237
|
-
if (options.recoveredBy) {
|
|
238
|
-
payload.event.recoveredBy = options.recoveredBy;
|
|
239
|
-
}
|
|
240
|
-
if (options.joinTimes) {
|
|
241
|
-
payload.event.joinTimes = options.joinTimes;
|
|
242
|
-
}
|
|
243
|
-
if (options.isRoapCallEnabled) {
|
|
244
|
-
payload.event.isRoapCallEnabled = options.isRoapCallEnabled;
|
|
245
|
-
}
|
|
246
|
-
}
|
|
247
|
-
|
|
248
|
-
return payload;
|
|
249
|
-
}
|
|
250
|
-
|
|
251
|
-
/**
|
|
252
|
-
* get the payload specific for a media quality event through call analyzer
|
|
253
|
-
* @param {String} eventType the event name
|
|
254
|
-
* @param {Object} identifiers contains the identifiers needed for CA
|
|
255
|
-
* @param {String} identifiers.correlationId
|
|
256
|
-
* @param {String} identifiers.locusUrl
|
|
257
|
-
* @param {String} identifiers.locusId
|
|
258
|
-
* @param {Object} options
|
|
259
|
-
* @param {Object} options.intervalData
|
|
260
|
-
* @param {String} options.clientType
|
|
261
|
-
* @returns {Object}
|
|
262
|
-
* @public
|
|
263
|
-
* @memberof Metrics
|
|
264
|
-
*/
|
|
265
|
-
public initMediaPayload(
|
|
266
|
-
// eslint-disable-next-line @typescript-eslint/no-shadow
|
|
267
|
-
eventType: string,
|
|
268
|
-
identifiers: {
|
|
269
|
-
correlationId: string;
|
|
270
|
-
locusUrl: string;
|
|
271
|
-
locusId: string;
|
|
272
|
-
},
|
|
273
|
-
options:
|
|
274
|
-
| {
|
|
275
|
-
intervalData: object;
|
|
276
|
-
clientType: string;
|
|
277
|
-
}
|
|
278
|
-
| any = {}
|
|
279
|
-
) {
|
|
280
|
-
const {audioSetupDelay, videoSetupDelay, joinTimes} = options;
|
|
281
|
-
|
|
282
|
-
const payload = {
|
|
283
|
-
eventId: uuid.v4(),
|
|
284
|
-
version: 1,
|
|
285
|
-
origin: {
|
|
286
|
-
audioSetupDelay,
|
|
287
|
-
videoSetupDelay,
|
|
288
|
-
name: 'endpoint',
|
|
289
|
-
networkType: options.networkType || UNKNOWN,
|
|
290
|
-
userAgent: this.userAgentToString(),
|
|
291
|
-
clientInfo: {
|
|
292
|
-
clientType: options.clientType, // TODO: Only clientType: 'TEAMS_CLIENT' is whitelisted
|
|
293
|
-
clientVersion: `${CLIENT_NAME}/${this.webex.version}`,
|
|
294
|
-
localNetworkPrefix: anonymizeIPAddress(this.webex.meetings.geoHintInfo?.clientAddress),
|
|
295
|
-
os: getOSNameInternal(),
|
|
296
|
-
osVersion: getOSVersion() || UNKNOWN,
|
|
297
|
-
subClientType: options.subClientType,
|
|
298
|
-
browser: getBrowserName(),
|
|
299
|
-
browserVersion: getBrowserVersion(),
|
|
300
|
-
},
|
|
301
|
-
},
|
|
302
|
-
originTime: {
|
|
303
|
-
triggered: new Date().toISOString(),
|
|
304
|
-
},
|
|
305
|
-
senderCountryCode: this.webex.meetings.geoHintInfo?.countryCode,
|
|
306
|
-
event: {
|
|
307
|
-
name: eventType,
|
|
308
|
-
canProceed: true,
|
|
309
|
-
identifiers,
|
|
310
|
-
intervals: [options.intervalData],
|
|
311
|
-
joinTimes,
|
|
312
|
-
eventData: {
|
|
313
|
-
webClientDomain: window.location.hostname,
|
|
314
|
-
},
|
|
315
|
-
sourceMetadata: {
|
|
316
|
-
applicationSoftwareType: CLIENT_NAME,
|
|
317
|
-
applicationSoftwareVersion: this.webex.version,
|
|
318
|
-
mediaEngineSoftwareType: getBrowserName() || 'browser',
|
|
319
|
-
mediaEngineSoftwareVersion: getOSVersion() || UNKNOWN,
|
|
320
|
-
startTime: new Date().toISOString(),
|
|
321
|
-
},
|
|
322
|
-
},
|
|
323
|
-
};
|
|
324
|
-
|
|
325
|
-
return payload;
|
|
326
|
-
}
|
|
327
|
-
|
|
328
|
-
/**
|
|
329
|
-
* This function Parses a Locus error and returns a diagnostic event payload.
|
|
330
|
-
* It should keep updating from:
|
|
331
|
-
* https://sqbu-github.cisco.com/WebExSquared/spark-client-framework/blob/master/spark-client-framework/Adapters/TelephonyAdapter/TelephonyAdapter.cpp#L920
|
|
332
|
-
*
|
|
333
|
-
* @param {Object} err the error Object from Locus response
|
|
334
|
-
* @param {boolean} showToUser true if a toast is shown to user
|
|
335
|
-
* @returns {{showToUser: boolean, category: string, errorDescription: string,
|
|
336
|
-
* errorCode: number, errorData: *, fatal: boolean, name: string}}
|
|
337
|
-
*/
|
|
338
|
-
parseLocusError(err: any, showToUser: boolean) {
|
|
339
|
-
let errorCode;
|
|
340
|
-
|
|
341
|
-
if (err && err.statusCode && err.statusCode >= 500) {
|
|
342
|
-
errorCode = 1003;
|
|
343
|
-
} else if (err && err.body && err.body.errorCode) {
|
|
344
|
-
// locus error codes: https://sqbu-github.cisco.com/WebExSquared/locus/blob/master/server/src/main/resources/locus-error-codes.properties
|
|
345
|
-
switch (ERROR_CODE[err.body.errorCode]) {
|
|
346
|
-
case MEETING_ERRORS.FREE_USER_MAX_PARTICIPANTS_EXCEEDED:
|
|
347
|
-
errorCode = 3007;
|
|
348
|
-
break;
|
|
349
|
-
case MEETING_ERRORS.PAID_USER_MAX_PARTICIPANTS_EXCEEDED:
|
|
350
|
-
case MEETING_ERRORS.SERVICE_MAX_PARTICIPANTS_EXCEEDED:
|
|
351
|
-
errorCode = 3002;
|
|
352
|
-
break;
|
|
353
|
-
case MEETING_ERRORS.INACTIVE:
|
|
354
|
-
errorCode = 4001;
|
|
355
|
-
break;
|
|
356
|
-
case MEETING_ERRORS.EXCEEDED_MAX_JOINED_PARTICIPANTS:
|
|
357
|
-
case MEETING_ERRORS.EXCEEDED_SERVICE_MAX_PARTICIPANTS:
|
|
358
|
-
errorCode = 3001;
|
|
359
|
-
break;
|
|
360
|
-
case MEETING_ERRORS.MEETING_IS_LOCKED:
|
|
361
|
-
errorCode = 4002;
|
|
362
|
-
break;
|
|
363
|
-
case MEETING_ERRORS.MEETING_IS_TERMINATING:
|
|
364
|
-
errorCode = 4003;
|
|
365
|
-
break;
|
|
366
|
-
case MEETING_ERRORS.MEETING_REQUIRE_MODERATOR_PIN_INTENT:
|
|
367
|
-
errorCode = 4004;
|
|
368
|
-
break;
|
|
369
|
-
case MEETING_ERRORS.MEETING_REQUIRE_MODERATOR_PIN:
|
|
370
|
-
errorCode = 4005;
|
|
371
|
-
break;
|
|
372
|
-
case MEETING_ERRORS.MEETING_REQUIRE_MODERATOR_ROLE:
|
|
373
|
-
errorCode = 4006;
|
|
374
|
-
break;
|
|
375
|
-
case MEETING_ERRORS.JOIN_RESTRICTED_USER:
|
|
376
|
-
case MEETING_ERRORS.GET_RESTRICTED_USER:
|
|
377
|
-
case MEETING_ERRORS.CREATE_MEDIA_RESTRICTED_USER:
|
|
378
|
-
errorCode = 3005;
|
|
379
|
-
break;
|
|
380
|
-
case MEETING_ERRORS.JOIN_RESTRICTED_USER_NOT_IN_ROOM:
|
|
381
|
-
errorCode = 4007;
|
|
382
|
-
break;
|
|
383
|
-
case MEETING_ERRORS.MEETING_NOT_FOUND:
|
|
384
|
-
errorCode = 4011;
|
|
385
|
-
break;
|
|
386
|
-
case MEETING_ERRORS.NOT_WEBEX_SITE:
|
|
387
|
-
errorCode = 4012;
|
|
388
|
-
break;
|
|
389
|
-
case MEETING_ERRORS.INVALID_JOIN_TIME:
|
|
390
|
-
errorCode = 4013;
|
|
391
|
-
break;
|
|
392
|
-
case MEETING_ERRORS.PHONE_NUMBER_NOT_A_NUMBER:
|
|
393
|
-
errorCode = 4016;
|
|
394
|
-
break;
|
|
395
|
-
case MEETING_ERRORS.PHONE_NUMBER_TOO_LONG:
|
|
396
|
-
errorCode = 4017;
|
|
397
|
-
break;
|
|
398
|
-
case MEETING_ERRORS.INVALID_DIALABLE_KEY:
|
|
399
|
-
errorCode = 4018;
|
|
400
|
-
break;
|
|
401
|
-
case MEETING_ERRORS.ONE_ON_ONE_TO_SELF_NOT_ALLOWED:
|
|
402
|
-
errorCode = 4019;
|
|
403
|
-
break;
|
|
404
|
-
case MEETING_ERRORS.REMOVED_PARTICIPANT:
|
|
405
|
-
errorCode = 4020;
|
|
406
|
-
break;
|
|
407
|
-
case MEETING_ERRORS.MEETING_LINK_NOT_FOUND:
|
|
408
|
-
errorCode = 4021;
|
|
409
|
-
break;
|
|
410
|
-
case MEETING_ERRORS.PHONE_NUMBER_TOO_SHORT_AFTER_IDD:
|
|
411
|
-
errorCode = 4022;
|
|
412
|
-
break;
|
|
413
|
-
case MEETING_ERRORS.INVALID_INVITEE_ADDRESS:
|
|
414
|
-
errorCode = 4023;
|
|
415
|
-
break;
|
|
416
|
-
case MEETING_ERRORS.PMR_ACCOUNT_LOCKED:
|
|
417
|
-
errorCode = 4024;
|
|
418
|
-
break;
|
|
419
|
-
case MEETING_ERRORS.RESOURCE_GUEST_FORBIDDEN:
|
|
420
|
-
errorCode = 4025;
|
|
421
|
-
break;
|
|
422
|
-
case MEETING_ERRORS.PMR_ACCOUNT_SUSPENDED:
|
|
423
|
-
errorCode = 4026;
|
|
424
|
-
break;
|
|
425
|
-
case MEETING_ERRORS.EMPTY_PHONE_NUMBER_OR_COUNTRY_CODE:
|
|
426
|
-
errorCode = 4027;
|
|
427
|
-
break;
|
|
428
|
-
case MEETING_ERRORS.INVALID_SINCE_OR_SEQUENCE_HASH_IN_REQUEST:
|
|
429
|
-
errorCode = 1006;
|
|
430
|
-
break;
|
|
431
|
-
case MEETING_ERRORS.CONVERSATION_NOT_FOUND:
|
|
432
|
-
errorCode = 4028;
|
|
433
|
-
break;
|
|
434
|
-
case MEETING_ERRORS.RECORDING_CONTROL_NOT_SUPPORTED:
|
|
435
|
-
case MEETING_ERRORS.RECORDING_NOT_STARTED:
|
|
436
|
-
case MEETING_ERRORS.RECORDING_NOT_ENABLED:
|
|
437
|
-
errorCode = 4029;
|
|
438
|
-
break;
|
|
439
|
-
default:
|
|
440
|
-
errorCode = 4008;
|
|
441
|
-
}
|
|
442
|
-
} else {
|
|
443
|
-
errorCode = 4008;
|
|
444
|
-
}
|
|
445
|
-
|
|
446
|
-
return this.generateErrorPayload(errorCode, showToUser, error.name.LOCUS_RESPONSE, err);
|
|
447
|
-
}
|
|
448
|
-
|
|
449
|
-
generateErrorPayload(errorCode, shownToUser, name, err) {
|
|
450
|
-
if (error.errors[errorCode]) {
|
|
451
|
-
const errorPayload: any = {
|
|
452
|
-
shownToUser: shownToUser || false,
|
|
453
|
-
category: error.errors[errorCode][2],
|
|
454
|
-
errorDescription: error.errors[errorCode][0],
|
|
455
|
-
errorCode,
|
|
456
|
-
fatal: !includes(error.notFatalErrorList, errorCode),
|
|
457
|
-
name: name || error.name.OTHER,
|
|
458
|
-
};
|
|
459
|
-
|
|
460
|
-
if (err && err.body) {
|
|
461
|
-
errorPayload.errorData = err.body;
|
|
462
|
-
}
|
|
463
|
-
|
|
464
|
-
if (err && err.statusCode) {
|
|
465
|
-
errorPayload.httpCode = err.statusCode;
|
|
466
|
-
}
|
|
467
|
-
|
|
468
|
-
return errorPayload;
|
|
469
|
-
}
|
|
470
|
-
|
|
471
|
-
return null;
|
|
472
|
-
}
|
|
473
|
-
|
|
474
|
-
/**
|
|
475
|
-
* Returns a formated string of the user agent.
|
|
476
|
-
*
|
|
477
|
-
* @returns {string} formatted user agent information
|
|
478
|
-
*/
|
|
479
|
-
userAgentToString() {
|
|
480
|
-
let userAgentOption;
|
|
481
|
-
let browserInfo;
|
|
482
|
-
const clientInfo = util.format('client=%s', `${this.webex.meetings?.metrics?.clientName}`);
|
|
483
|
-
|
|
484
|
-
if (
|
|
485
|
-
['chrome', 'firefox', 'msie', 'msedge', 'safari'].indexOf(getBrowserName().toLowerCase()) !==
|
|
486
|
-
-1
|
|
487
|
-
) {
|
|
488
|
-
browserInfo = util.format(
|
|
489
|
-
'browser=%s',
|
|
490
|
-
`${getBrowserName().toLowerCase()}/${getBrowserVersion().split('.')[0]}`
|
|
491
|
-
);
|
|
492
|
-
}
|
|
493
|
-
const osInfo = util.format('os=%s', `${getOSName()}/${getOSVersion().split('.')[0]}`);
|
|
494
|
-
|
|
495
|
-
if (browserInfo) {
|
|
496
|
-
userAgentOption = `(${browserInfo}`;
|
|
497
|
-
}
|
|
498
|
-
if (osInfo) {
|
|
499
|
-
userAgentOption = userAgentOption
|
|
500
|
-
? `${userAgentOption}; ${clientInfo}; ${osInfo}`
|
|
501
|
-
: `${clientInfo}; (${osInfo}`;
|
|
502
|
-
}
|
|
503
|
-
if (userAgentOption) {
|
|
504
|
-
userAgentOption += ')';
|
|
505
|
-
|
|
506
|
-
return util.format(
|
|
507
|
-
'webex-js-sdk/%s %s',
|
|
508
|
-
`${process.env.NODE_ENV}-${this.webex.version}`,
|
|
509
|
-
userAgentOption
|
|
510
|
-
);
|
|
511
|
-
}
|
|
512
|
-
|
|
513
|
-
return util.format('webex-js-sdk/%s', `${process.env.NODE_ENV}-${this.webex.version}`);
|
|
514
|
-
}
|
|
515
|
-
|
|
516
|
-
/**
|
|
517
|
-
* Uploads given metric to the Metrics service as an Behavioral metric.
|
|
518
|
-
* Metadata about the environment such as browser, OS, SDK and their versions
|
|
519
|
-
* are automatically added when the metric is sent.
|
|
520
|
-
*
|
|
521
|
-
* The Metrics service will send an Behavioral metric to InfluxDB for
|
|
522
|
-
* aggregation.
|
|
523
|
-
* See https://confluence-eng-gpk2.cisco.com/conf/display/WBXT/Getting+started+with+Metrics+Service.
|
|
524
|
-
*
|
|
525
|
-
* @param {string} metricName Name of the metric (measurement) to send
|
|
526
|
-
* @param {Object} metricFields Key-valye pairs of data or values about this metric
|
|
527
|
-
* @param {Object} metricTags Key-value pairs of metric metadata
|
|
528
|
-
*
|
|
529
|
-
* @returns {void}
|
|
530
|
-
*/
|
|
531
|
-
sendBehavioralMetric(metricName: string, metricFields: object = {}, metricTags: object = {}) {
|
|
532
|
-
this.webex.internal.metrics.submitClientMetrics(metricName, {
|
|
533
|
-
type: this.webex.config.metrics.type,
|
|
534
|
-
fields: metricFields,
|
|
535
|
-
tags: metricTags,
|
|
536
|
-
});
|
|
537
|
-
}
|
|
538
|
-
}
|
|
539
|
-
|
|
540
|
-
// Export Metrics singleton ---------------------------------------------------
|
|
541
|
-
const instance = new Metrics();
|
|
542
|
-
|
|
543
|
-
export default instance;
|
|
1
|
+
/*!
|
|
2
|
+
* Copyright (c) 2015-2020 Cisco Systems, Inc. See LICENSE file.
|
|
3
|
+
*/
|
|
4
|
+
import util from 'util';
|
|
5
|
+
|
|
6
|
+
import {includes} from 'lodash';
|
|
7
|
+
import uuid from 'uuid';
|
|
8
|
+
// @ts-ignore
|
|
9
|
+
import window from 'global/window';
|
|
10
|
+
import anonymize from 'ip-anonymize';
|
|
11
|
+
|
|
12
|
+
// @ts-ignore
|
|
13
|
+
import {getOSNameInternal} from '@webex/internal-plugin-metrics';
|
|
14
|
+
import LoggerProxy from '../common/logs/logger-proxy';
|
|
15
|
+
import {MEETING_ERRORS} from '../constants';
|
|
16
|
+
import BrowserDetection from '../common/browser-detection';
|
|
17
|
+
|
|
18
|
+
import {
|
|
19
|
+
error,
|
|
20
|
+
eventType,
|
|
21
|
+
errorCodes as ERROR_CODE,
|
|
22
|
+
UNKNOWN,
|
|
23
|
+
CLIENT_NAME,
|
|
24
|
+
mediaType,
|
|
25
|
+
} from './config';
|
|
26
|
+
|
|
27
|
+
const {getOSName, getOSVersion, getBrowserName, getBrowserVersion} = BrowserDetection();
|
|
28
|
+
|
|
29
|
+
// Apply a CIDR /28 format to the IPV4 and /96 to the IPV6 addresses
|
|
30
|
+
// For reference : https://en.wikipedia.org/wiki/Classless_Inter-Domain_Routing
|
|
31
|
+
const anonymizeIPAddress = (localIp) => anonymize(localIp, 28, 96);
|
|
32
|
+
|
|
33
|
+
const triggerTimers = ({event, meeting, data}) => {
|
|
34
|
+
switch (event) {
|
|
35
|
+
case eventType.CALL_INITIATED:
|
|
36
|
+
meeting.setStartCallInitiateJoinReq();
|
|
37
|
+
break;
|
|
38
|
+
case eventType.LOCUS_JOIN_REQUEST:
|
|
39
|
+
meeting.setEndCallInitiateJoinReq();
|
|
40
|
+
meeting.setStartJoinReqResp();
|
|
41
|
+
break;
|
|
42
|
+
case eventType.LOCUS_JOIN_RESPONSE:
|
|
43
|
+
meeting.setEndJoinReqResp();
|
|
44
|
+
meeting.setStartSetupDelay(mediaType.AUDIO);
|
|
45
|
+
meeting.setStartSetupDelay(mediaType.VIDEO);
|
|
46
|
+
meeting.setStartSendingMediaDelay(mediaType.AUDIO);
|
|
47
|
+
meeting.setStartSendingMediaDelay(mediaType.VIDEO);
|
|
48
|
+
break;
|
|
49
|
+
case eventType.RECEIVING_MEDIA_START:
|
|
50
|
+
meeting.setEndSetupDelay(data.mediaType);
|
|
51
|
+
break;
|
|
52
|
+
case eventType.SENDING_MEDIA_START:
|
|
53
|
+
meeting.setEndSendingMediaDelay(data.mediaType);
|
|
54
|
+
break;
|
|
55
|
+
case eventType.LOCAL_SDP_GENERATED:
|
|
56
|
+
meeting.setStartLocalSDPGenRemoteSDPRecvDelay();
|
|
57
|
+
break;
|
|
58
|
+
case eventType.REMOTE_SDP_RECEIVED:
|
|
59
|
+
meeting.setEndLocalSDPGenRemoteSDPRecvDelay();
|
|
60
|
+
break;
|
|
61
|
+
default:
|
|
62
|
+
break;
|
|
63
|
+
}
|
|
64
|
+
};
|
|
65
|
+
|
|
66
|
+
/**
|
|
67
|
+
* @description Metrics handles all the call metrics events
|
|
68
|
+
* @export
|
|
69
|
+
* @class Metrics
|
|
70
|
+
*/
|
|
71
|
+
class Metrics {
|
|
72
|
+
static instance: Metrics;
|
|
73
|
+
|
|
74
|
+
_events: any;
|
|
75
|
+
keys: any;
|
|
76
|
+
meetingCollection: any;
|
|
77
|
+
webex: any;
|
|
78
|
+
|
|
79
|
+
/**
|
|
80
|
+
* Create Metrics Object
|
|
81
|
+
* @constructor
|
|
82
|
+
* @public
|
|
83
|
+
* @memberof Meetings
|
|
84
|
+
*/
|
|
85
|
+
constructor() {
|
|
86
|
+
if (!Metrics.instance) {
|
|
87
|
+
/**
|
|
88
|
+
* @instance
|
|
89
|
+
* @type {Array}
|
|
90
|
+
* @private
|
|
91
|
+
* @memberof Metrics
|
|
92
|
+
*/
|
|
93
|
+
this._events = [];
|
|
94
|
+
/**
|
|
95
|
+
* @instance
|
|
96
|
+
* @type {MeetingCollection}
|
|
97
|
+
* @private
|
|
98
|
+
* @memberof Metrics
|
|
99
|
+
*/
|
|
100
|
+
this.meetingCollection = null;
|
|
101
|
+
/**
|
|
102
|
+
* @instance
|
|
103
|
+
* @type {MeetingCollection}
|
|
104
|
+
* @private
|
|
105
|
+
* @memberof Metrics
|
|
106
|
+
*/
|
|
107
|
+
this.keys = Object.values(eventType);
|
|
108
|
+
/**
|
|
109
|
+
* @instance
|
|
110
|
+
* @type {Metrics}
|
|
111
|
+
* @private
|
|
112
|
+
* @memberof Metrics
|
|
113
|
+
*/
|
|
114
|
+
Metrics.instance = this;
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
// eslint-disable-next-line no-constructor-return
|
|
118
|
+
return Metrics.instance;
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
/**
|
|
122
|
+
* Initializes the Metrics singleton with a meeting Collection.
|
|
123
|
+
*
|
|
124
|
+
* @param {Object} meetingCollection meetings object
|
|
125
|
+
* @param {Object} webex webex SDK object
|
|
126
|
+
*
|
|
127
|
+
* @returns {void}
|
|
128
|
+
*/
|
|
129
|
+
initialSetup(meetingCollection: object, webex: object) {
|
|
130
|
+
this.meetingCollection = meetingCollection;
|
|
131
|
+
this.webex = webex;
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
/**
|
|
135
|
+
* poste Meeting event metrics
|
|
136
|
+
* @param {object} options {meetingId/meeting} as a json object
|
|
137
|
+
* @param {Meeting} options.meeting Meeting object
|
|
138
|
+
* @param {String} options.meetingId
|
|
139
|
+
* @param {object} options.data
|
|
140
|
+
* @param {object} options.event
|
|
141
|
+
* @returns {object} null
|
|
142
|
+
*/
|
|
143
|
+
postEvent(options: {meeting?: any; meetingId?: string; data?: object; event?: any} | any) {
|
|
144
|
+
const {meetingId, data = {}, event} = options;
|
|
145
|
+
let {meeting} = options;
|
|
146
|
+
|
|
147
|
+
if (this.keys.indexOf(event) === -1) {
|
|
148
|
+
LoggerProxy.logger.error(
|
|
149
|
+
`Metrics:index#postEvent --> Event ${event} doesn't exist in dictionary`
|
|
150
|
+
);
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
if (!meeting && meetingId) {
|
|
154
|
+
meeting = this.meetingCollection.get(meetingId);
|
|
155
|
+
options.meeting = meeting;
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
if (meeting) {
|
|
159
|
+
triggerTimers(options);
|
|
160
|
+
|
|
161
|
+
if (!meeting.callEvents) {
|
|
162
|
+
meeting.callEvents = [];
|
|
163
|
+
}
|
|
164
|
+
if (event === eventType.MEDIA_QUALITY) {
|
|
165
|
+
data.event = event;
|
|
166
|
+
meeting.sendMediaQualityAnalyzerMetrics(data);
|
|
167
|
+
} else {
|
|
168
|
+
meeting.callEvents.push(event);
|
|
169
|
+
data.event = event;
|
|
170
|
+
meeting.sendCallAnalyzerMetrics(data);
|
|
171
|
+
}
|
|
172
|
+
} else {
|
|
173
|
+
LoggerProxy.logger.info(
|
|
174
|
+
`Metrics:index#postEvent --> Event received for meetingId:${meetingId}, but meeting not found in collection.`
|
|
175
|
+
);
|
|
176
|
+
}
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
/**
|
|
180
|
+
* Docs for Call analyzer metrics
|
|
181
|
+
* https://sqbu-github.cisco.com/WebExSquared/call-analyzer/wiki
|
|
182
|
+
* https://sqbu-github.cisco.com/WebExSquared/event-dictionary/blob/master/diagnostic-events.raml
|
|
183
|
+
*/
|
|
184
|
+
|
|
185
|
+
// eslint-disable-next-line @typescript-eslint/no-shadow
|
|
186
|
+
initPayload(eventType, identifiers, options) {
|
|
187
|
+
const payload: any = {
|
|
188
|
+
eventId: uuid.v4(),
|
|
189
|
+
version: 1,
|
|
190
|
+
origin: {
|
|
191
|
+
name: 'endpoint',
|
|
192
|
+
networkType: 'unknown',
|
|
193
|
+
userAgent: this.userAgentToString(),
|
|
194
|
+
clientInfo: {
|
|
195
|
+
clientType: options.clientType,
|
|
196
|
+
clientVersion: `${CLIENT_NAME}/${this.webex.version}`,
|
|
197
|
+
localNetworkPrefix: anonymizeIPAddress(this.webex.meetings.geoHintInfo?.clientAddress),
|
|
198
|
+
osVersion: getOSVersion() || 'unknown',
|
|
199
|
+
subClientType: options.subClientType,
|
|
200
|
+
os: getOSNameInternal(),
|
|
201
|
+
browser: getBrowserName(),
|
|
202
|
+
browserVersion: getBrowserVersion(),
|
|
203
|
+
},
|
|
204
|
+
},
|
|
205
|
+
originTime: {
|
|
206
|
+
triggered: new Date().toISOString(),
|
|
207
|
+
},
|
|
208
|
+
senderCountryCode: this.webex.meetings.geoHintInfo?.countryCode,
|
|
209
|
+
event: {
|
|
210
|
+
name: eventType,
|
|
211
|
+
canProceed: true,
|
|
212
|
+
identifiers,
|
|
213
|
+
eventData: {webClientDomain: window.location.hostname},
|
|
214
|
+
},
|
|
215
|
+
};
|
|
216
|
+
|
|
217
|
+
// TODO: more options should be checked and some of them should be mandatory in certain conditions
|
|
218
|
+
if (options) {
|
|
219
|
+
if (Object.prototype.hasOwnProperty.call(options, 'canProceed')) {
|
|
220
|
+
payload.event.canProceed = options.canProceed;
|
|
221
|
+
}
|
|
222
|
+
if (options.errors) {
|
|
223
|
+
payload.event.errors = options.errors;
|
|
224
|
+
}
|
|
225
|
+
if (options.mediaType) {
|
|
226
|
+
payload.event.mediaType = options.mediaType;
|
|
227
|
+
}
|
|
228
|
+
if (options.trigger) {
|
|
229
|
+
payload.event.trigger = options.trigger;
|
|
230
|
+
}
|
|
231
|
+
if (options.pstnAudioType) {
|
|
232
|
+
payload.event.pstnAudioType = options.pstnAudioType;
|
|
233
|
+
}
|
|
234
|
+
if (options.mediaCapabilities) {
|
|
235
|
+
payload.event.mediaCapabilities = options.mediaCapabilities;
|
|
236
|
+
}
|
|
237
|
+
if (options.recoveredBy) {
|
|
238
|
+
payload.event.recoveredBy = options.recoveredBy;
|
|
239
|
+
}
|
|
240
|
+
if (options.joinTimes) {
|
|
241
|
+
payload.event.joinTimes = options.joinTimes;
|
|
242
|
+
}
|
|
243
|
+
if (options.isRoapCallEnabled) {
|
|
244
|
+
payload.event.isRoapCallEnabled = options.isRoapCallEnabled;
|
|
245
|
+
}
|
|
246
|
+
}
|
|
247
|
+
|
|
248
|
+
return payload;
|
|
249
|
+
}
|
|
250
|
+
|
|
251
|
+
/**
|
|
252
|
+
* get the payload specific for a media quality event through call analyzer
|
|
253
|
+
* @param {String} eventType the event name
|
|
254
|
+
* @param {Object} identifiers contains the identifiers needed for CA
|
|
255
|
+
* @param {String} identifiers.correlationId
|
|
256
|
+
* @param {String} identifiers.locusUrl
|
|
257
|
+
* @param {String} identifiers.locusId
|
|
258
|
+
* @param {Object} options
|
|
259
|
+
* @param {Object} options.intervalData
|
|
260
|
+
* @param {String} options.clientType
|
|
261
|
+
* @returns {Object}
|
|
262
|
+
* @public
|
|
263
|
+
* @memberof Metrics
|
|
264
|
+
*/
|
|
265
|
+
public initMediaPayload(
|
|
266
|
+
// eslint-disable-next-line @typescript-eslint/no-shadow
|
|
267
|
+
eventType: string,
|
|
268
|
+
identifiers: {
|
|
269
|
+
correlationId: string;
|
|
270
|
+
locusUrl: string;
|
|
271
|
+
locusId: string;
|
|
272
|
+
},
|
|
273
|
+
options:
|
|
274
|
+
| {
|
|
275
|
+
intervalData: object;
|
|
276
|
+
clientType: string;
|
|
277
|
+
}
|
|
278
|
+
| any = {}
|
|
279
|
+
) {
|
|
280
|
+
const {audioSetupDelay, videoSetupDelay, joinTimes} = options;
|
|
281
|
+
|
|
282
|
+
const payload = {
|
|
283
|
+
eventId: uuid.v4(),
|
|
284
|
+
version: 1,
|
|
285
|
+
origin: {
|
|
286
|
+
audioSetupDelay,
|
|
287
|
+
videoSetupDelay,
|
|
288
|
+
name: 'endpoint',
|
|
289
|
+
networkType: options.networkType || UNKNOWN,
|
|
290
|
+
userAgent: this.userAgentToString(),
|
|
291
|
+
clientInfo: {
|
|
292
|
+
clientType: options.clientType, // TODO: Only clientType: 'TEAMS_CLIENT' is whitelisted
|
|
293
|
+
clientVersion: `${CLIENT_NAME}/${this.webex.version}`,
|
|
294
|
+
localNetworkPrefix: anonymizeIPAddress(this.webex.meetings.geoHintInfo?.clientAddress),
|
|
295
|
+
os: getOSNameInternal(),
|
|
296
|
+
osVersion: getOSVersion() || UNKNOWN,
|
|
297
|
+
subClientType: options.subClientType,
|
|
298
|
+
browser: getBrowserName(),
|
|
299
|
+
browserVersion: getBrowserVersion(),
|
|
300
|
+
},
|
|
301
|
+
},
|
|
302
|
+
originTime: {
|
|
303
|
+
triggered: new Date().toISOString(),
|
|
304
|
+
},
|
|
305
|
+
senderCountryCode: this.webex.meetings.geoHintInfo?.countryCode,
|
|
306
|
+
event: {
|
|
307
|
+
name: eventType,
|
|
308
|
+
canProceed: true,
|
|
309
|
+
identifiers,
|
|
310
|
+
intervals: [options.intervalData],
|
|
311
|
+
joinTimes,
|
|
312
|
+
eventData: {
|
|
313
|
+
webClientDomain: window.location.hostname,
|
|
314
|
+
},
|
|
315
|
+
sourceMetadata: {
|
|
316
|
+
applicationSoftwareType: CLIENT_NAME,
|
|
317
|
+
applicationSoftwareVersion: this.webex.version,
|
|
318
|
+
mediaEngineSoftwareType: getBrowserName() || 'browser',
|
|
319
|
+
mediaEngineSoftwareVersion: getOSVersion() || UNKNOWN,
|
|
320
|
+
startTime: new Date().toISOString(),
|
|
321
|
+
},
|
|
322
|
+
},
|
|
323
|
+
};
|
|
324
|
+
|
|
325
|
+
return payload;
|
|
326
|
+
}
|
|
327
|
+
|
|
328
|
+
/**
|
|
329
|
+
* This function Parses a Locus error and returns a diagnostic event payload.
|
|
330
|
+
* It should keep updating from:
|
|
331
|
+
* https://sqbu-github.cisco.com/WebExSquared/spark-client-framework/blob/master/spark-client-framework/Adapters/TelephonyAdapter/TelephonyAdapter.cpp#L920
|
|
332
|
+
*
|
|
333
|
+
* @param {Object} err the error Object from Locus response
|
|
334
|
+
* @param {boolean} showToUser true if a toast is shown to user
|
|
335
|
+
* @returns {{showToUser: boolean, category: string, errorDescription: string,
|
|
336
|
+
* errorCode: number, errorData: *, fatal: boolean, name: string}}
|
|
337
|
+
*/
|
|
338
|
+
parseLocusError(err: any, showToUser: boolean) {
|
|
339
|
+
let errorCode;
|
|
340
|
+
|
|
341
|
+
if (err && err.statusCode && err.statusCode >= 500) {
|
|
342
|
+
errorCode = 1003;
|
|
343
|
+
} else if (err && err.body && err.body.errorCode) {
|
|
344
|
+
// locus error codes: https://sqbu-github.cisco.com/WebExSquared/locus/blob/master/server/src/main/resources/locus-error-codes.properties
|
|
345
|
+
switch (ERROR_CODE[err.body.errorCode]) {
|
|
346
|
+
case MEETING_ERRORS.FREE_USER_MAX_PARTICIPANTS_EXCEEDED:
|
|
347
|
+
errorCode = 3007;
|
|
348
|
+
break;
|
|
349
|
+
case MEETING_ERRORS.PAID_USER_MAX_PARTICIPANTS_EXCEEDED:
|
|
350
|
+
case MEETING_ERRORS.SERVICE_MAX_PARTICIPANTS_EXCEEDED:
|
|
351
|
+
errorCode = 3002;
|
|
352
|
+
break;
|
|
353
|
+
case MEETING_ERRORS.INACTIVE:
|
|
354
|
+
errorCode = 4001;
|
|
355
|
+
break;
|
|
356
|
+
case MEETING_ERRORS.EXCEEDED_MAX_JOINED_PARTICIPANTS:
|
|
357
|
+
case MEETING_ERRORS.EXCEEDED_SERVICE_MAX_PARTICIPANTS:
|
|
358
|
+
errorCode = 3001;
|
|
359
|
+
break;
|
|
360
|
+
case MEETING_ERRORS.MEETING_IS_LOCKED:
|
|
361
|
+
errorCode = 4002;
|
|
362
|
+
break;
|
|
363
|
+
case MEETING_ERRORS.MEETING_IS_TERMINATING:
|
|
364
|
+
errorCode = 4003;
|
|
365
|
+
break;
|
|
366
|
+
case MEETING_ERRORS.MEETING_REQUIRE_MODERATOR_PIN_INTENT:
|
|
367
|
+
errorCode = 4004;
|
|
368
|
+
break;
|
|
369
|
+
case MEETING_ERRORS.MEETING_REQUIRE_MODERATOR_PIN:
|
|
370
|
+
errorCode = 4005;
|
|
371
|
+
break;
|
|
372
|
+
case MEETING_ERRORS.MEETING_REQUIRE_MODERATOR_ROLE:
|
|
373
|
+
errorCode = 4006;
|
|
374
|
+
break;
|
|
375
|
+
case MEETING_ERRORS.JOIN_RESTRICTED_USER:
|
|
376
|
+
case MEETING_ERRORS.GET_RESTRICTED_USER:
|
|
377
|
+
case MEETING_ERRORS.CREATE_MEDIA_RESTRICTED_USER:
|
|
378
|
+
errorCode = 3005;
|
|
379
|
+
break;
|
|
380
|
+
case MEETING_ERRORS.JOIN_RESTRICTED_USER_NOT_IN_ROOM:
|
|
381
|
+
errorCode = 4007;
|
|
382
|
+
break;
|
|
383
|
+
case MEETING_ERRORS.MEETING_NOT_FOUND:
|
|
384
|
+
errorCode = 4011;
|
|
385
|
+
break;
|
|
386
|
+
case MEETING_ERRORS.NOT_WEBEX_SITE:
|
|
387
|
+
errorCode = 4012;
|
|
388
|
+
break;
|
|
389
|
+
case MEETING_ERRORS.INVALID_JOIN_TIME:
|
|
390
|
+
errorCode = 4013;
|
|
391
|
+
break;
|
|
392
|
+
case MEETING_ERRORS.PHONE_NUMBER_NOT_A_NUMBER:
|
|
393
|
+
errorCode = 4016;
|
|
394
|
+
break;
|
|
395
|
+
case MEETING_ERRORS.PHONE_NUMBER_TOO_LONG:
|
|
396
|
+
errorCode = 4017;
|
|
397
|
+
break;
|
|
398
|
+
case MEETING_ERRORS.INVALID_DIALABLE_KEY:
|
|
399
|
+
errorCode = 4018;
|
|
400
|
+
break;
|
|
401
|
+
case MEETING_ERRORS.ONE_ON_ONE_TO_SELF_NOT_ALLOWED:
|
|
402
|
+
errorCode = 4019;
|
|
403
|
+
break;
|
|
404
|
+
case MEETING_ERRORS.REMOVED_PARTICIPANT:
|
|
405
|
+
errorCode = 4020;
|
|
406
|
+
break;
|
|
407
|
+
case MEETING_ERRORS.MEETING_LINK_NOT_FOUND:
|
|
408
|
+
errorCode = 4021;
|
|
409
|
+
break;
|
|
410
|
+
case MEETING_ERRORS.PHONE_NUMBER_TOO_SHORT_AFTER_IDD:
|
|
411
|
+
errorCode = 4022;
|
|
412
|
+
break;
|
|
413
|
+
case MEETING_ERRORS.INVALID_INVITEE_ADDRESS:
|
|
414
|
+
errorCode = 4023;
|
|
415
|
+
break;
|
|
416
|
+
case MEETING_ERRORS.PMR_ACCOUNT_LOCKED:
|
|
417
|
+
errorCode = 4024;
|
|
418
|
+
break;
|
|
419
|
+
case MEETING_ERRORS.RESOURCE_GUEST_FORBIDDEN:
|
|
420
|
+
errorCode = 4025;
|
|
421
|
+
break;
|
|
422
|
+
case MEETING_ERRORS.PMR_ACCOUNT_SUSPENDED:
|
|
423
|
+
errorCode = 4026;
|
|
424
|
+
break;
|
|
425
|
+
case MEETING_ERRORS.EMPTY_PHONE_NUMBER_OR_COUNTRY_CODE:
|
|
426
|
+
errorCode = 4027;
|
|
427
|
+
break;
|
|
428
|
+
case MEETING_ERRORS.INVALID_SINCE_OR_SEQUENCE_HASH_IN_REQUEST:
|
|
429
|
+
errorCode = 1006;
|
|
430
|
+
break;
|
|
431
|
+
case MEETING_ERRORS.CONVERSATION_NOT_FOUND:
|
|
432
|
+
errorCode = 4028;
|
|
433
|
+
break;
|
|
434
|
+
case MEETING_ERRORS.RECORDING_CONTROL_NOT_SUPPORTED:
|
|
435
|
+
case MEETING_ERRORS.RECORDING_NOT_STARTED:
|
|
436
|
+
case MEETING_ERRORS.RECORDING_NOT_ENABLED:
|
|
437
|
+
errorCode = 4029;
|
|
438
|
+
break;
|
|
439
|
+
default:
|
|
440
|
+
errorCode = 4008;
|
|
441
|
+
}
|
|
442
|
+
} else {
|
|
443
|
+
errorCode = 4008;
|
|
444
|
+
}
|
|
445
|
+
|
|
446
|
+
return this.generateErrorPayload(errorCode, showToUser, error.name.LOCUS_RESPONSE, err);
|
|
447
|
+
}
|
|
448
|
+
|
|
449
|
+
generateErrorPayload(errorCode, shownToUser, name, err) {
|
|
450
|
+
if (error.errors[errorCode]) {
|
|
451
|
+
const errorPayload: any = {
|
|
452
|
+
shownToUser: shownToUser || false,
|
|
453
|
+
category: error.errors[errorCode][2],
|
|
454
|
+
errorDescription: error.errors[errorCode][0],
|
|
455
|
+
errorCode,
|
|
456
|
+
fatal: !includes(error.notFatalErrorList, errorCode),
|
|
457
|
+
name: name || error.name.OTHER,
|
|
458
|
+
};
|
|
459
|
+
|
|
460
|
+
if (err && err.body) {
|
|
461
|
+
errorPayload.errorData = err.body;
|
|
462
|
+
}
|
|
463
|
+
|
|
464
|
+
if (err && err.statusCode) {
|
|
465
|
+
errorPayload.httpCode = err.statusCode;
|
|
466
|
+
}
|
|
467
|
+
|
|
468
|
+
return errorPayload;
|
|
469
|
+
}
|
|
470
|
+
|
|
471
|
+
return null;
|
|
472
|
+
}
|
|
473
|
+
|
|
474
|
+
/**
|
|
475
|
+
* Returns a formated string of the user agent.
|
|
476
|
+
*
|
|
477
|
+
* @returns {string} formatted user agent information
|
|
478
|
+
*/
|
|
479
|
+
userAgentToString() {
|
|
480
|
+
let userAgentOption;
|
|
481
|
+
let browserInfo;
|
|
482
|
+
const clientInfo = util.format('client=%s', `${this.webex.meetings?.metrics?.clientName}`);
|
|
483
|
+
|
|
484
|
+
if (
|
|
485
|
+
['chrome', 'firefox', 'msie', 'msedge', 'safari'].indexOf(getBrowserName().toLowerCase()) !==
|
|
486
|
+
-1
|
|
487
|
+
) {
|
|
488
|
+
browserInfo = util.format(
|
|
489
|
+
'browser=%s',
|
|
490
|
+
`${getBrowserName().toLowerCase()}/${getBrowserVersion().split('.')[0]}`
|
|
491
|
+
);
|
|
492
|
+
}
|
|
493
|
+
const osInfo = util.format('os=%s', `${getOSName()}/${getOSVersion().split('.')[0]}`);
|
|
494
|
+
|
|
495
|
+
if (browserInfo) {
|
|
496
|
+
userAgentOption = `(${browserInfo}`;
|
|
497
|
+
}
|
|
498
|
+
if (osInfo) {
|
|
499
|
+
userAgentOption = userAgentOption
|
|
500
|
+
? `${userAgentOption}; ${clientInfo}; ${osInfo}`
|
|
501
|
+
: `${clientInfo}; (${osInfo}`;
|
|
502
|
+
}
|
|
503
|
+
if (userAgentOption) {
|
|
504
|
+
userAgentOption += ')';
|
|
505
|
+
|
|
506
|
+
return util.format(
|
|
507
|
+
'webex-js-sdk/%s %s',
|
|
508
|
+
`${process.env.NODE_ENV}-${this.webex.version}`,
|
|
509
|
+
userAgentOption
|
|
510
|
+
);
|
|
511
|
+
}
|
|
512
|
+
|
|
513
|
+
return util.format('webex-js-sdk/%s', `${process.env.NODE_ENV}-${this.webex.version}`);
|
|
514
|
+
}
|
|
515
|
+
|
|
516
|
+
/**
|
|
517
|
+
* Uploads given metric to the Metrics service as an Behavioral metric.
|
|
518
|
+
* Metadata about the environment such as browser, OS, SDK and their versions
|
|
519
|
+
* are automatically added when the metric is sent.
|
|
520
|
+
*
|
|
521
|
+
* The Metrics service will send an Behavioral metric to InfluxDB for
|
|
522
|
+
* aggregation.
|
|
523
|
+
* See https://confluence-eng-gpk2.cisco.com/conf/display/WBXT/Getting+started+with+Metrics+Service.
|
|
524
|
+
*
|
|
525
|
+
* @param {string} metricName Name of the metric (measurement) to send
|
|
526
|
+
* @param {Object} metricFields Key-valye pairs of data or values about this metric
|
|
527
|
+
* @param {Object} metricTags Key-value pairs of metric metadata
|
|
528
|
+
*
|
|
529
|
+
* @returns {void}
|
|
530
|
+
*/
|
|
531
|
+
sendBehavioralMetric(metricName: string, metricFields: object = {}, metricTags: object = {}) {
|
|
532
|
+
this.webex.internal.metrics.submitClientMetrics(metricName, {
|
|
533
|
+
type: this.webex.config.metrics.type,
|
|
534
|
+
fields: metricFields,
|
|
535
|
+
tags: metricTags,
|
|
536
|
+
});
|
|
537
|
+
}
|
|
538
|
+
}
|
|
539
|
+
|
|
540
|
+
// Export Metrics singleton ---------------------------------------------------
|
|
541
|
+
const instance = new Metrics();
|
|
542
|
+
|
|
543
|
+
export default instance;
|