@techsee/techsee-media-service 999.0.17-switch → 999.0.18-switch
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/lib/LocalStreamManager.d.ts +1 -0
- package/lib/LocalStreamManager.d.ts.map +1 -1
- package/lib/LocalStreamManager.js +7 -1
- package/lib/LocalStreamManager.js.map +1 -1
- package/lib/MediaServiceBase.d.ts +2 -2
- package/lib/MediaServiceBase.d.ts.map +1 -1
- package/lib/MediaServiceBase.js +16 -14
- package/lib/MediaServiceBase.js.map +1 -1
- package/package.json +1 -1
|
@@ -4,6 +4,7 @@ declare global {
|
|
|
4
4
|
interface Window {
|
|
5
5
|
latestUserMediaStream: Nullable<MediaRequestSuccessResult>;
|
|
6
6
|
latestDesktopMediaStream: Nullable<MediaRequestSuccessResult>;
|
|
7
|
+
latestLocalMediaConstraints: Nullable<LocalMediaConstraints>;
|
|
7
8
|
}
|
|
8
9
|
}
|
|
9
10
|
export declare class LocalStreamManager {
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/LocalStreamManager.ts"],"names":[],"mappings":"AAMA,OAAO,EAAC,QAAQ,EAAC,MAAM,yBAAyB,CAAC;AAEjD,OAAO,EACH,iBAAiB,EACjB,qBAAqB,EACrB,2BAA2B,EAE3B,yBAAyB,EAE5B,MAAM,kBAAkB,CAAC;AAW1B,OAAO,CAAC,MAAM,CAAC;IACX,UAAU,MAAM;QACZ,qBAAqB,EAAE,QAAQ,CAAC,yBAAyB,CAAC,CAAC;QAC3D,wBAAwB,EAAE,QAAQ,CAAC,yBAAyB,CAAC,CAAC;KACjE;CACJ;AAYD,qBAAa,kBAAkB;IAC3B,OAAO,CAAC,WAAW,CAAoB;IACvC,OAAO,CAAC,eAAe,CACoF;gBAE/F,WAAW,EAAE,iBAAiB;aAOtC,cAAc;;;;;;;;IAIlB,kBAAkB,CAAC,WAAW,EAAE,qBAAqB,GAAG,OAAO,CAAC,yBAAyB,CAAC;IAwC1F,qBAAqB,CAAC,WAAW,EAAE,2BAA2B,GAAG,OAAO,CAAC,yBAAyB,CAAC;IA4CnG,sBAAsB,IAAI,OAAO,CAAC,IAAI,CAAC;IAiBvC,yBAAyB,IAAI,OAAO,CAAC,IAAI,CAAC;IAe1C,eAAe,IAAI,OAAO,CAAC,IAAI,CAAC;IAShC,OAAO,CAAC,0BAA0B;IAwDlC,OAAO,CAAC,6BAA6B;IAiBrC,OAAO,CAAC,mCAAmC;IAgD3C,iBAAiB,CAAC,cAAc,EAAE;QAAC,KAAK,EAAE,eAAe,EAAE,CAAC;QAAC,KAAK,EAAE,eAAe,EAAE,CAAA;KAAC;IAStF,OAAO,CAAC,oBAAoB;IAiE5B,OAAO,CAAC,wBAAwB;CAwCnC","file":"LocalStreamManager.d.ts","sourcesContent":["import assign from 'lodash/assign';\nimport get from 'lodash/get';\nimport last from 'lodash/last';\nimport first from 'lodash/first';\nimport {Promise} from 'bluebird';\n\nimport {Nullable} from '@techsee/techsee-common';\n\nimport {\n IMediaEnvironment,\n LocalMediaConstraints,\n LocalVideoStreamConstraints,\n MediaRequestFailResult,\n MediaRequestSuccessResult,\n VideoStreamResolution\n} from './MediaContracts';\n\nimport {\n ConstraintType,\n DEFAULT_VIDEO_CONSTRAINT_TYPE,\n DEFAULT_VIDEO_RESOLUTION,\n LocalVideoSourceType,\n MediaRequestErrorCode\n} from './MediaConstants';\nimport {getMediaTracer} from './MediaUtils/MediaTracer';\n\ndeclare global {\n interface Window {\n latestUserMediaStream: Nullable<MediaRequestSuccessResult>;\n latestDesktopMediaStream: Nullable<MediaRequestSuccessResult>;\n }\n}\n\ndeclare const window: Window;\n\nwindow.latestUserMediaStream = null;\nlet userMediaPromise: Nullable<Promise<MediaRequestSuccessResult>> = null;\n\nwindow.latestDesktopMediaStream = null;\nlet desktopMediaPromise: Nullable<Promise<MediaRequestSuccessResult>> = null;\n\nconst trace = getMediaTracer('LocalStreamManager');\n\nexport class LocalStreamManager {\n private environment: IMediaEnvironment;\n private _groupedDevices: {video: Nullable<{back?: MediaDeviceInfo; front?: MediaDeviceInfo}>;\n audio: Nullable<MediaDeviceInfo>; camerasCount: number;} = {video: null, audio: null, camerasCount: 0};\n\n constructor(environment: IMediaEnvironment) {\n this.environment = environment;\n\n this.parseMediaErrorThenThrow = this.parseMediaErrorThenThrow.bind(this);\n this.clearAllStreams = this.clearAllStreams.bind(this);\n }\n\n get groupedDevices() {\n return this._groupedDevices;\n }\n\n getUserMediaStream(constraints: LocalMediaConstraints): Promise<MediaRequestSuccessResult> {\n trace.info('getUserMediaStream', constraints);\n if (userMediaPromise) {\n return userMediaPromise;\n }\n\n if (window.latestUserMediaStream) {\n const videoTrack = window.latestUserMediaStream.mediaStream.getVideoTracks();\n\n if (videoTrack.length > 0 && videoTrack[0].readyState === 'live') {\n return Promise.resolve({...window.latestUserMediaStream, isNew: false});\n }\n\n this.destroyUserMediaStream();\n }\n\n userMediaPromise = new Promise((resolve, reject) => {\n const catchHandler = (err: any): void => {\n userMediaPromise = null;\n window.latestUserMediaStream = null;\n trace.error('getUserMedia error', err);\n reject(err);\n };\n\n try {\n this.getHighestResolutionStream(false, constraints)\n .then((streamResult: MediaRequestSuccessResult) => {\n userMediaPromise = null;\n window.latestUserMediaStream = streamResult;\n resolve({...window.latestUserMediaStream, isNew: true});\n })\n .catch(catchHandler);\n } catch (err) {\n catchHandler(err);\n }\n });\n\n return userMediaPromise;\n }\n\n getDesktopMediaStream(constraints: LocalVideoStreamConstraints): Promise<MediaRequestSuccessResult> {\n if (desktopMediaPromise) {\n return desktopMediaPromise;\n }\n\n trace.info('getDesktopMediaStream', constraints);\n\n if (window.latestDesktopMediaStream) {\n const track = window.latestDesktopMediaStream.mediaStream.getVideoTracks();\n\n if (track.length > 0 && track[0].readyState === 'live') {\n //return Promise.resolve();\n return Promise.resolve({...window.latestDesktopMediaStream, isNew: false});\n }\n this.destroyDesktopMediaStream().catch((error: any) => {\n trace.error('Error destroying desktop media stream', error);\n });\n\n }\n\n desktopMediaPromise = new Promise((resolve, reject) => {\n const mediaCatchHandler = (err: any): void => {\n desktopMediaPromise = null;\n window.latestDesktopMediaStream = null;\n trace.error('getDisplayMedia error', err);\n reject(err);\n };\n\n try {\n this.getHighestResolutionStream(true, {video: constraints, audio: false})\n .then((streamResult: MediaRequestSuccessResult) => {\n desktopMediaPromise = null;\n window.latestDesktopMediaStream = streamResult;\n\n resolve({...window.latestDesktopMediaStream, isNew: true});\n }).catch(mediaCatchHandler);\n } catch (err) {\n mediaCatchHandler(err);\n }\n });\n\n return desktopMediaPromise;\n }\n\n destroyUserMediaStream(): Promise<void> {\n const streamResult = window.latestUserMediaStream;\n\n window.latestUserMediaStream = null;\n userMediaPromise = null;\n\n if (streamResult && streamResult.mediaStream) {\n streamResult.mediaStream.getTracks().forEach((track: MediaStreamTrack) => {\n track.stop();\n });\n\n trace.info('destroyUserMediaStream - stop all tracks');\n }\n\n return Promise.resolve();\n }\n\n destroyDesktopMediaStream(): Promise<void> {\n const streamResult = window.latestDesktopMediaStream;\n\n window.latestDesktopMediaStream = null;\n desktopMediaPromise = null;\n\n if (streamResult && streamResult.mediaStream) {\n streamResult.mediaStream.getTracks().forEach((track: MediaStreamTrack) => {\n track.stop();\n });\n }\n\n return Promise.resolve();\n }\n\n clearAllStreams(): Promise<void> {\n trace.info('LocalStreamManager Clearing all streams');\n\n return Promise.all([\n this.destroyDesktopMediaStream(),\n this.destroyUserMediaStream()\n ]).then(() => undefined);\n }\n\n private getHighestResolutionStream(isDesktopMedia: boolean, constraints?: LocalMediaConstraints): Promise<MediaRequestSuccessResult> {\n const constraintsCandidates = this.getParsedConstraints(isDesktopMedia, constraints);\n\n trace.info('getHighestResolutionStream - constraintsCandidates:', constraintsCandidates);\n\n if (constraintsCandidates.length === 0) {\n return Promise.reject(new Error('No constraints to retrieve the stream'));\n }\n\n return new Promise((resolve, reject) => {\n let lastSuccessfulResult: Nullable<MediaRequestSuccessResult> = null;\n let lastFailedResult: Nullable<MediaRequestFailResult> = null;\n let isStreamRequestFulfilled = false;\n\n (Promise as any).reduce(constraintsCandidates, (total: any, candidate: MediaStreamConstraints) => {\n if (isStreamRequestFulfilled) {\n return;\n }\n\n const constraintType = get(constraints, 'video.videoResolution.constraintType') || get(constraints, 'video.videoResolution[0].constraintType');\n\n return this.getStreamPromiseByParsedConstraints(candidate, isDesktopMedia)\n .then((stream: MediaStream) => {\n lastSuccessfulResult = {\n mediaStream: stream,\n constraint: candidate,\n constraintType,\n isNew: true\n };\n trace.info('getHighestResolutionStream Success', lastSuccessfulResult);\n isStreamRequestFulfilled = true;\n resolve(lastSuccessfulResult);\n })\n .catch((failResult: MediaRequestFailResult) => {\n lastFailedResult = failResult;\n\n if (lastFailedResult.errorCode === MediaRequestErrorCode.PermissionDenied) {\n isStreamRequestFulfilled = true;\n reject(lastFailedResult);\n }\n });\n }, 0)\n .then(() => {\n if (!isStreamRequestFulfilled) {\n if (!lastFailedResult) {\n lastFailedResult = {\n errorCode: MediaRequestErrorCode.GeneralError,\n message: 'Suitable stream cannot be created'\n };\n }\n reject(lastFailedResult);\n }\n });\n });\n }\n\n private getUserMediaStreamMediaDevice(constraints: MediaStreamConstraints, isDesktopMedia: boolean) {\n let streamPromise = null;\n\n if (!isDesktopMedia) {\n streamPromise = navigator.mediaDevices.getUserMedia(constraints);\n } else {\n //@ts-ignore\n streamPromise = navigator.mediaDevices.getDisplayMedia\n //@ts-ignore\n ? navigator.mediaDevices.getDisplayMedia(constraints)\n : navigator.getDisplayMedia(constraints);\n }\n\n return streamPromise;\n }\n\n //@ts-ignore\n private getStreamPromiseByParsedConstraints(constraints: MediaStreamConstraints, isDesktopMedia: boolean): Promise<MediaStream> {\n trace.info('getStreamPromiseByParsedConstraints', constraints);\n\n try {\n return this.getUserMediaStreamMediaDevice(constraints, isDesktopMedia).catch((err: any) => {\n let streamPromise = null;\n\n trace.info('getStreamPromiseByParsedConstraints - get user stream error', err.name);\n\n // Android (Galaxy esp.) have a bug that when using facingMode it might cause NotReadableError. So\n // we should try again but without facingMode. The exception to the rule is when we don't have specific\n // device to use (like in IOS13 bug, as can be seen in MediaServiceBase.ts), in that case, we might end\n // up using the front camera, which is a privacy concern and the preference is to fallback from video\n if (err && (err.name === 'NotReadableError') && get(constraints, 'video.facingMode') &&\n get(constraints, 'video.deviceId')) {\n\n const newConstraints = constraints;\n\n trace.info(`getStreamPromiseByParsedConstraints - NotReadableError - Failed to get video user media stream with facingMode=${get(constraints, 'video.facingMode')}`, err.name);\n\n // @ts-ignore\n delete newConstraints.video.facingMode;\n\n streamPromise = this.getStreamPromiseByParsedConstraints(newConstraints, isDesktopMedia);\n } else if (err && err.name && constraints.audio && !constraints.video) {\n trace.info('getStreamPromiseByParsedConstraints - audioStreamFailed - Failed to get audio user media stream', err.name);\n\n return this.parseMediaErrorThenThrow({name: 'audioStreamFailed'});\n } else if (err && err.name && constraints.audio) {\n const newConstraints = constraints;\n\n trace.info('getStreamPromiseByParsedConstraints - Failed to get video and audio user media stream, try to get video media stream only', err.name);\n\n newConstraints.audio = false;\n streamPromise = this.getStreamPromiseByParsedConstraints(newConstraints, isDesktopMedia);\n } else if (err && !streamPromise) {\n trace.info('getStreamPromiseByParsedConstraints - Failed to get video user media stream', err && err.name);\n\n streamPromise = Promise.reject(err);\n }\n\n return streamPromise && streamPromise.catch(this.parseMediaErrorThenThrow);\n });\n } catch (e) {\n this.parseMediaErrorThenThrow(e);\n }\n }\n\n setGroupedDevices(groupedDevices: {video: MediaDeviceInfo[]; audio: MediaDeviceInfo[]}) {\n trace.info('setGroupedDevices', groupedDevices);\n this._groupedDevices = {video: groupedDevices.video\n ? {back: last(groupedDevices.video), front: first(groupedDevices.video)}\n : null,\n audio: last(groupedDevices.audio) || null,\n camerasCount: groupedDevices.video.length};\n }\n\n private getParsedConstraints(isDesktopMedia: boolean, streamConstraints?: LocalMediaConstraints): MediaStreamConstraints[] {\n const constraintCandidates: MediaStreamConstraints[] = [];\n\n //TODO - Alex: need to understand which constraints not compatible with screen share\n if (isDesktopMedia) {\n return [{video: true}];\n }\n\n if (!streamConstraints) {\n return constraintCandidates;\n }\n\n const baseConstraints: MediaStreamConstraints = {};\n\n baseConstraints.audio = typeof streamConstraints.audio === 'boolean' ? streamConstraints.audio : undefined;\n\n if (!streamConstraints.video) {\n return [baseConstraints];\n }\n\n const cameraType = (streamConstraints.video !== 'boolean') &&\n (streamConstraints.video as LocalVideoStreamConstraints).videoSourceType;\n\n baseConstraints.video = {\n facingMode: cameraType && cameraType === LocalVideoSourceType.CAMERA_FRONT\n ? 'user'\n : 'environment',\n frameRate: {ideal: 15, max: 30}\n };\n\n const deviceId = get(this._groupedDevices, `video.${cameraType === LocalVideoSourceType.CAMERA_FRONT ? 'front' : 'back'}.deviceId`);\n\n if (deviceId) {\n baseConstraints.video.deviceId = {exact: deviceId};\n }\n\n const assignConstraint = (videoResolution: VideoStreamResolution): void => {\n const [width, height] = videoResolution.resolution.split('x');\n const constraintCandidate = assign({}, baseConstraints);\n\n constraintCandidate.video = assign({}, baseConstraints.video, {\n width: _constraintByType(width, videoResolution.constraintType),\n height: _constraintByType(height, videoResolution.constraintType)\n });\n constraintCandidates.push(constraintCandidate);\n };\n\n if ((baseConstraints.video && typeof streamConstraints.video === 'boolean') ||\n !((streamConstraints.video as LocalVideoStreamConstraints).videoResolution instanceof Array)) {\n\n const defaultResolution: VideoStreamResolution = {\n resolution: DEFAULT_VIDEO_RESOLUTION,\n constraintType: DEFAULT_VIDEO_CONSTRAINT_TYPE\n };\n\n assignConstraint(defaultResolution);\n\n return constraintCandidates;\n }\n\n (streamConstraints.video as any).videoResolution.forEach(assignConstraint);\n\n return constraintCandidates;\n }\n\n private parseMediaErrorThenThrow(error: any): MediaRequestFailResult {\n if (error && error.errorCode) {\n throw error;\n }\n\n trace.error('Get media stream error.', error);\n const errorName = error && error.name ? error.name : '';\n\n const permissionErrorNames = [\n 'NotAllowedError',\n 'PermissionDismissedError',\n 'PermissionDeniedError'\n\n ];\n\n // const generalError = [\n // 'NotFoundError',\n // 'DevicesNotFoundError',\n // 'NotReadableError',\n // 'TrackStartError'\n // ];\n\n const constraintsErrors = [\n 'OverconstrainedError',\n 'ConstraintNotSatisfiedError'\n\n ];\n\n const isErrorOfType = (errorsArr: string[]) => errorsArr.filter((err: string) => err === errorName).length > 0;\n\n if (isErrorOfType(permissionErrorNames)) {\n throw {errorCode: MediaRequestErrorCode.PermissionDenied, message: errorName};\n }\n\n if (isErrorOfType(constraintsErrors)) {\n throw {errorCode: MediaRequestErrorCode.Overconstrained, message: errorName};\n }\n\n throw {errorCode: MediaRequestErrorCode.GeneralError, message: errorName};\n }\n}\n\nfunction _constraintByType(value: any, type: ConstraintType): any {\n switch (type) {\n case ConstraintType.MIN:\n return {min: value};\n default:\n return {ideal: value};\n }\n}\n"]}
|
|
1
|
+
{"version":3,"sources":["../src/LocalStreamManager.ts"],"names":[],"mappings":"AAOA,OAAO,EAAC,QAAQ,EAAC,MAAM,yBAAyB,CAAC;AAEjD,OAAO,EACH,iBAAiB,EACjB,qBAAqB,EACrB,2BAA2B,EAE3B,yBAAyB,EAE5B,MAAM,kBAAkB,CAAC;AAW1B,OAAO,CAAC,MAAM,CAAC;IACX,UAAU,MAAM;QACZ,qBAAqB,EAAE,QAAQ,CAAC,yBAAyB,CAAC,CAAC;QAC3D,wBAAwB,EAAE,QAAQ,CAAC,yBAAyB,CAAC,CAAC;QAC9D,2BAA2B,EAAE,QAAQ,CAAC,qBAAqB,CAAC,CAAC;KAChE;CACJ;AAYD,qBAAa,kBAAkB;IAC3B,OAAO,CAAC,WAAW,CAAoB;IACvC,OAAO,CAAC,eAAe,CACoF;gBAE/F,WAAW,EAAE,iBAAiB;aAOtC,cAAc;;;;;;;;IAIlB,kBAAkB,CAAC,WAAW,EAAE,qBAAqB,GAAG,OAAO,CAAC,yBAAyB,CAAC;IA8C1F,qBAAqB,CAAC,WAAW,EAAE,2BAA2B,GAAG,OAAO,CAAC,yBAAyB,CAAC;IA4CnG,sBAAsB,IAAI,OAAO,CAAC,IAAI,CAAC;IAkBvC,yBAAyB,IAAI,OAAO,CAAC,IAAI,CAAC;IAe1C,eAAe,IAAI,OAAO,CAAC,IAAI,CAAC;IAShC,OAAO,CAAC,0BAA0B;IAwDlC,OAAO,CAAC,6BAA6B;IAiBrC,OAAO,CAAC,mCAAmC;IAgD3C,iBAAiB,CAAC,cAAc,EAAE;QAAC,KAAK,EAAE,eAAe,EAAE,CAAC;QAAC,KAAK,EAAE,eAAe,EAAE,CAAA;KAAC;IAStF,OAAO,CAAC,oBAAoB;IAiE5B,OAAO,CAAC,wBAAwB;CAwCnC","file":"LocalStreamManager.d.ts","sourcesContent":["import assign from 'lodash/assign';\nimport get from 'lodash/get';\nimport last from 'lodash/last';\nimport first from 'lodash/first';\nimport cloneDeep from 'lodash/cloneDeep';\nimport {Promise} from 'bluebird';\n\nimport {Nullable} from '@techsee/techsee-common';\n\nimport {\n IMediaEnvironment,\n LocalMediaConstraints,\n LocalVideoStreamConstraints,\n MediaRequestFailResult,\n MediaRequestSuccessResult,\n VideoStreamResolution\n} from './MediaContracts';\n\nimport {\n ConstraintType,\n DEFAULT_VIDEO_CONSTRAINT_TYPE,\n DEFAULT_VIDEO_RESOLUTION,\n LocalVideoSourceType,\n MediaRequestErrorCode\n} from './MediaConstants';\nimport {getMediaTracer} from './MediaUtils/MediaTracer';\n\ndeclare global {\n interface Window {\n latestUserMediaStream: Nullable<MediaRequestSuccessResult>;\n latestDesktopMediaStream: Nullable<MediaRequestSuccessResult>;\n latestLocalMediaConstraints: Nullable<LocalMediaConstraints>;\n }\n}\n\ndeclare const window: Window;\n\nwindow.latestUserMediaStream = null;\nlet userMediaPromise: Nullable<Promise<MediaRequestSuccessResult>> = null;\n\nwindow.latestDesktopMediaStream = null;\nlet desktopMediaPromise: Nullable<Promise<MediaRequestSuccessResult>> = null;\n\nconst trace = getMediaTracer('LocalStreamManager');\n\nexport class LocalStreamManager {\n private environment: IMediaEnvironment;\n private _groupedDevices: {video: Nullable<{back?: MediaDeviceInfo; front?: MediaDeviceInfo}>;\n audio: Nullable<MediaDeviceInfo>; camerasCount: number;} = {video: null, audio: null, camerasCount: 0};\n\n constructor(environment: IMediaEnvironment) {\n this.environment = environment;\n\n this.parseMediaErrorThenThrow = this.parseMediaErrorThenThrow.bind(this);\n this.clearAllStreams = this.clearAllStreams.bind(this);\n }\n\n get groupedDevices() {\n return this._groupedDevices;\n }\n\n getUserMediaStream(constraints: LocalMediaConstraints): Promise<MediaRequestSuccessResult> {\n trace.info('getUserMediaStream', constraints);\n if (userMediaPromise) {\n return userMediaPromise;\n }\n\n if (window.latestUserMediaStream) {\n const videoTrack = window.latestUserMediaStream.mediaStream.getVideoTracks();\n\n if (videoTrack.length > 0 && videoTrack[0].readyState === 'live') {\n return Promise.resolve({...window.latestUserMediaStream, isNew: false});\n }\n\n this.destroyUserMediaStream();\n }\n\n userMediaPromise = new Promise((resolve, reject) => {\n const catchHandler = (err: any): void => {\n userMediaPromise = null;\n window.latestUserMediaStream = null;\n trace.error('getUserMedia error', err);\n reject(err);\n };\n\n try {\n this.getHighestResolutionStream(false, constraints)\n .then((streamResult: MediaRequestSuccessResult) => {\n userMediaPromise = null;\n window.latestUserMediaStream = streamResult;\n window.latestLocalMediaConstraints = cloneDeep(constraints);\n\n if (window.latestLocalMediaConstraints && window.latestLocalMediaConstraints.audio) {\n window.latestLocalMediaConstraints.audio = (streamResult.constraint && streamResult.constraint.audio) as boolean;\n }\n\n resolve({...window.latestUserMediaStream, isNew: true});\n })\n .catch(catchHandler);\n } catch (err) {\n catchHandler(err);\n }\n });\n\n return userMediaPromise;\n }\n\n getDesktopMediaStream(constraints: LocalVideoStreamConstraints): Promise<MediaRequestSuccessResult> {\n if (desktopMediaPromise) {\n return desktopMediaPromise;\n }\n\n trace.info('getDesktopMediaStream', constraints);\n\n if (window.latestDesktopMediaStream) {\n const track = window.latestDesktopMediaStream.mediaStream.getVideoTracks();\n\n if (track.length > 0 && track[0].readyState === 'live') {\n //return Promise.resolve();\n return Promise.resolve({...window.latestDesktopMediaStream, isNew: false});\n }\n this.destroyDesktopMediaStream().catch((error: any) => {\n trace.error('Error destroying desktop media stream', error);\n });\n\n }\n\n desktopMediaPromise = new Promise((resolve, reject) => {\n const mediaCatchHandler = (err: any): void => {\n desktopMediaPromise = null;\n window.latestDesktopMediaStream = null;\n trace.error('getDisplayMedia error', err);\n reject(err);\n };\n\n try {\n this.getHighestResolutionStream(true, {video: constraints, audio: false})\n .then((streamResult: MediaRequestSuccessResult) => {\n desktopMediaPromise = null;\n window.latestDesktopMediaStream = streamResult;\n\n resolve({...window.latestDesktopMediaStream, isNew: true});\n }).catch(mediaCatchHandler);\n } catch (err) {\n mediaCatchHandler(err);\n }\n });\n\n return desktopMediaPromise;\n }\n\n destroyUserMediaStream(): Promise<void> {\n const streamResult = window.latestUserMediaStream;\n\n window.latestUserMediaStream = null;\n window.latestLocalMediaConstraints = null;\n userMediaPromise = null;\n\n if (streamResult && streamResult.mediaStream) {\n streamResult.mediaStream.getTracks().forEach((track: MediaStreamTrack) => {\n track.stop();\n });\n\n trace.info('destroyUserMediaStream - stop all tracks');\n }\n\n return Promise.resolve();\n }\n\n destroyDesktopMediaStream(): Promise<void> {\n const streamResult = window.latestDesktopMediaStream;\n\n window.latestDesktopMediaStream = null;\n desktopMediaPromise = null;\n\n if (streamResult && streamResult.mediaStream) {\n streamResult.mediaStream.getTracks().forEach((track: MediaStreamTrack) => {\n track.stop();\n });\n }\n\n return Promise.resolve();\n }\n\n clearAllStreams(): Promise<void> {\n trace.info('LocalStreamManager Clearing all streams');\n\n return Promise.all([\n this.destroyDesktopMediaStream(),\n this.destroyUserMediaStream()\n ]).then(() => undefined);\n }\n\n private getHighestResolutionStream(isDesktopMedia: boolean, constraints?: LocalMediaConstraints): Promise<MediaRequestSuccessResult> {\n const constraintsCandidates = this.getParsedConstraints(isDesktopMedia, constraints);\n\n trace.info('getHighestResolutionStream - constraintsCandidates:', constraintsCandidates);\n\n if (constraintsCandidates.length === 0) {\n return Promise.reject(new Error('No constraints to retrieve the stream'));\n }\n\n return new Promise((resolve, reject) => {\n let lastSuccessfulResult: Nullable<MediaRequestSuccessResult> = null;\n let lastFailedResult: Nullable<MediaRequestFailResult> = null;\n let isStreamRequestFulfilled = false;\n\n (Promise as any).reduce(constraintsCandidates, (total: any, candidate: MediaStreamConstraints) => {\n if (isStreamRequestFulfilled) {\n return;\n }\n\n const constraintType = get(constraints, 'video.videoResolution.constraintType') || get(constraints, 'video.videoResolution[0].constraintType');\n\n return this.getStreamPromiseByParsedConstraints(candidate, isDesktopMedia)\n .then((stream: MediaStream) => {\n lastSuccessfulResult = {\n mediaStream: stream,\n constraint: candidate,\n constraintType,\n isNew: true\n };\n trace.info('getHighestResolutionStream Success', lastSuccessfulResult);\n isStreamRequestFulfilled = true;\n resolve(lastSuccessfulResult);\n })\n .catch((failResult: MediaRequestFailResult) => {\n lastFailedResult = failResult;\n\n if (lastFailedResult.errorCode === MediaRequestErrorCode.PermissionDenied) {\n isStreamRequestFulfilled = true;\n reject(lastFailedResult);\n }\n });\n }, 0)\n .then(() => {\n if (!isStreamRequestFulfilled) {\n if (!lastFailedResult) {\n lastFailedResult = {\n errorCode: MediaRequestErrorCode.GeneralError,\n message: 'Suitable stream cannot be created'\n };\n }\n reject(lastFailedResult);\n }\n });\n });\n }\n\n private getUserMediaStreamMediaDevice(constraints: MediaStreamConstraints, isDesktopMedia: boolean) {\n let streamPromise = null;\n\n if (!isDesktopMedia) {\n streamPromise = navigator.mediaDevices.getUserMedia(constraints);\n } else {\n //@ts-ignore\n streamPromise = navigator.mediaDevices.getDisplayMedia\n //@ts-ignore\n ? navigator.mediaDevices.getDisplayMedia(constraints)\n : navigator.getDisplayMedia(constraints);\n }\n\n return streamPromise;\n }\n\n //@ts-ignore\n private getStreamPromiseByParsedConstraints(constraints: MediaStreamConstraints, isDesktopMedia: boolean): Promise<MediaStream> {\n trace.info('getStreamPromiseByParsedConstraints', constraints);\n\n try {\n return this.getUserMediaStreamMediaDevice(constraints, isDesktopMedia).catch((err: any) => {\n let streamPromise = null;\n\n trace.info('getStreamPromiseByParsedConstraints - get user stream error', err.name);\n\n // Android (Galaxy esp.) have a bug that when using facingMode it might cause NotReadableError. So\n // we should try again but without facingMode. The exception to the rule is when we don't have specific\n // device to use (like in IOS13 bug, as can be seen in MediaServiceBase.ts), in that case, we might end\n // up using the front camera, which is a privacy concern and the preference is to fallback from video\n if (err && (err.name === 'NotReadableError') && get(constraints, 'video.facingMode') &&\n get(constraints, 'video.deviceId')) {\n\n const newConstraints = constraints;\n\n trace.info(`getStreamPromiseByParsedConstraints - NotReadableError - Failed to get video user media stream with facingMode=${get(constraints, 'video.facingMode')}`, err.name);\n\n // @ts-ignore\n delete newConstraints.video.facingMode;\n\n streamPromise = this.getStreamPromiseByParsedConstraints(newConstraints, isDesktopMedia);\n } else if (err && err.name && constraints.audio && !constraints.video) {\n trace.info('getStreamPromiseByParsedConstraints - audioStreamFailed - Failed to get audio user media stream', err.name);\n\n return this.parseMediaErrorThenThrow({name: 'audioStreamFailed'});\n } else if (err && err.name && constraints.audio) {\n const newConstraints = constraints;\n\n trace.info('getStreamPromiseByParsedConstraints - Failed to get video and audio user media stream, try to get video media stream only', err.name);\n\n newConstraints.audio = false;\n streamPromise = this.getStreamPromiseByParsedConstraints(newConstraints, isDesktopMedia);\n } else if (err && !streamPromise) {\n trace.info('getStreamPromiseByParsedConstraints - Failed to get video user media stream', err && err.name);\n\n streamPromise = Promise.reject(err);\n }\n\n return streamPromise && streamPromise.catch(this.parseMediaErrorThenThrow);\n });\n } catch (e) {\n this.parseMediaErrorThenThrow(e);\n }\n }\n\n setGroupedDevices(groupedDevices: {video: MediaDeviceInfo[]; audio: MediaDeviceInfo[]}) {\n trace.info('setGroupedDevices', groupedDevices);\n this._groupedDevices = {video: groupedDevices.video\n ? {back: last(groupedDevices.video), front: first(groupedDevices.video)}\n : null,\n audio: last(groupedDevices.audio) || null,\n camerasCount: groupedDevices.video.length};\n }\n\n private getParsedConstraints(isDesktopMedia: boolean, streamConstraints?: LocalMediaConstraints): MediaStreamConstraints[] {\n const constraintCandidates: MediaStreamConstraints[] = [];\n\n //TODO - Alex: need to understand which constraints not compatible with screen share\n if (isDesktopMedia) {\n return [{video: true}];\n }\n\n if (!streamConstraints) {\n return constraintCandidates;\n }\n\n const baseConstraints: MediaStreamConstraints = {};\n\n baseConstraints.audio = typeof streamConstraints.audio === 'boolean' ? streamConstraints.audio : undefined;\n\n if (!streamConstraints.video) {\n return [baseConstraints];\n }\n\n const cameraType = (streamConstraints.video !== 'boolean') &&\n (streamConstraints.video as LocalVideoStreamConstraints).videoSourceType;\n\n baseConstraints.video = {\n facingMode: cameraType && cameraType === LocalVideoSourceType.CAMERA_FRONT\n ? 'user'\n : 'environment',\n frameRate: {ideal: 15, max: 30}\n };\n\n const deviceId = get(this._groupedDevices, `video.${cameraType === LocalVideoSourceType.CAMERA_FRONT ? 'front' : 'back'}.deviceId`);\n\n if (deviceId) {\n baseConstraints.video.deviceId = deviceId;\n }\n\n const assignConstraint = (videoResolution: VideoStreamResolution): void => {\n const [width, height] = videoResolution.resolution.split('x');\n const constraintCandidate = assign({}, baseConstraints);\n\n constraintCandidate.video = assign({}, baseConstraints.video, {\n width: _constraintByType(width, videoResolution.constraintType),\n height: _constraintByType(height, videoResolution.constraintType)\n });\n constraintCandidates.push(constraintCandidate);\n };\n\n if ((baseConstraints.video && typeof streamConstraints.video === 'boolean') ||\n !((streamConstraints.video as LocalVideoStreamConstraints).videoResolution instanceof Array)) {\n\n const defaultResolution: VideoStreamResolution = {\n resolution: DEFAULT_VIDEO_RESOLUTION,\n constraintType: DEFAULT_VIDEO_CONSTRAINT_TYPE\n };\n\n assignConstraint(defaultResolution);\n\n return constraintCandidates;\n }\n\n (streamConstraints.video as any).videoResolution.forEach(assignConstraint);\n\n return constraintCandidates;\n }\n\n private parseMediaErrorThenThrow(error: any): MediaRequestFailResult {\n if (error && error.errorCode) {\n throw error;\n }\n\n trace.error('Get media stream error.', error);\n const errorName = error && error.name ? error.name : '';\n\n const permissionErrorNames = [\n 'NotAllowedError',\n 'PermissionDismissedError',\n 'PermissionDeniedError'\n\n ];\n\n // const generalError = [\n // 'NotFoundError',\n // 'DevicesNotFoundError',\n // 'NotReadableError',\n // 'TrackStartError'\n // ];\n\n const constraintsErrors = [\n 'OverconstrainedError',\n 'ConstraintNotSatisfiedError'\n\n ];\n\n const isErrorOfType = (errorsArr: string[]) => errorsArr.filter((err: string) => err === errorName).length > 0;\n\n if (isErrorOfType(permissionErrorNames)) {\n throw {errorCode: MediaRequestErrorCode.PermissionDenied, message: errorName};\n }\n\n if (isErrorOfType(constraintsErrors)) {\n throw {errorCode: MediaRequestErrorCode.Overconstrained, message: errorName};\n }\n\n throw {errorCode: MediaRequestErrorCode.GeneralError, message: errorName};\n }\n}\n\nfunction _constraintByType(value: any, type: ConstraintType): any {\n switch (type) {\n case ConstraintType.MIN:\n return {min: value};\n default:\n return {ideal: value};\n }\n}\n"]}
|
|
@@ -20,6 +20,7 @@ var assign_1 = __importDefault(require("lodash/assign"));
|
|
|
20
20
|
var get_1 = __importDefault(require("lodash/get"));
|
|
21
21
|
var last_1 = __importDefault(require("lodash/last"));
|
|
22
22
|
var first_1 = __importDefault(require("lodash/first"));
|
|
23
|
+
var cloneDeep_1 = __importDefault(require("lodash/cloneDeep"));
|
|
23
24
|
var bluebird_1 = require("bluebird");
|
|
24
25
|
var MediaConstants_1 = require("./MediaConstants");
|
|
25
26
|
var MediaTracer_1 = require("./MediaUtils/MediaTracer");
|
|
@@ -66,6 +67,10 @@ var LocalStreamManager = /** @class */function () {
|
|
|
66
67
|
_this.getHighestResolutionStream(false, constraints).then(function (streamResult) {
|
|
67
68
|
userMediaPromise = null;
|
|
68
69
|
window.latestUserMediaStream = streamResult;
|
|
70
|
+
window.latestLocalMediaConstraints = cloneDeep_1.default(constraints);
|
|
71
|
+
if (window.latestLocalMediaConstraints && window.latestLocalMediaConstraints.audio) {
|
|
72
|
+
window.latestLocalMediaConstraints.audio = streamResult.constraint && streamResult.constraint.audio;
|
|
73
|
+
}
|
|
69
74
|
resolve(__assign({}, window.latestUserMediaStream, { isNew: true }));
|
|
70
75
|
}).catch(catchHandler);
|
|
71
76
|
} catch (err) {
|
|
@@ -112,6 +117,7 @@ var LocalStreamManager = /** @class */function () {
|
|
|
112
117
|
LocalStreamManager.prototype.destroyUserMediaStream = function () {
|
|
113
118
|
var streamResult = window.latestUserMediaStream;
|
|
114
119
|
window.latestUserMediaStream = null;
|
|
120
|
+
window.latestLocalMediaConstraints = null;
|
|
115
121
|
userMediaPromise = null;
|
|
116
122
|
if (streamResult && streamResult.mediaStream) {
|
|
117
123
|
streamResult.mediaStream.getTracks().forEach(function (track) {
|
|
@@ -259,7 +265,7 @@ var LocalStreamManager = /** @class */function () {
|
|
|
259
265
|
};
|
|
260
266
|
var deviceId = get_1.default(this._groupedDevices, "video." + (cameraType === MediaConstants_1.LocalVideoSourceType.CAMERA_FRONT ? 'front' : 'back') + ".deviceId");
|
|
261
267
|
if (deviceId) {
|
|
262
|
-
baseConstraints.video.deviceId =
|
|
268
|
+
baseConstraints.video.deviceId = deviceId;
|
|
263
269
|
}
|
|
264
270
|
var assignConstraint = function assignConstraint(videoResolution) {
|
|
265
271
|
var _a = videoResolution.resolution.split('x'),
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["LocalStreamManager.js","../src/LocalStreamManager.ts"],"names":["__assign","Object","assign","t","s","i","n","arguments","length","p","prototype","hasOwnProperty","call","apply","__importDefault","mod","__esModule","defineProperty","exports","value","assign_1","require","get_1","last_1","first_1","bluebird_1","MediaConstants_1","MediaTracer_1","window","latestUserMediaStream","userMediaPromise","latestDesktopMediaStream","desktopMediaPromise","trace","getMediaTracer","LocalStreamManager","environment","_groupedDevices","video","audio","camerasCount","parseMediaErrorThenThrow","bind","clearAllStreams","get","enumerable","configurable","getUserMediaStream","constraints","_this","info","videoTrack","mediaStream","getVideoTracks","readyState","Promise","resolve","isNew","destroyUserMediaStream","reject","catchHandler","err","error","getHighestResolutionStream","then","streamResult","catch","getDesktopMediaStream","track","destroyDesktopMediaStream","mediaCatchHandler","getTracks","forEach","stop","all","undefined","isDesktopMedia","constraintsCandidates","getParsedConstraints","Error","lastSuccessfulResult","lastFailedResult","isStreamRequestFulfilled","reduce","total","candidate","constraintType","default","getStreamPromiseByParsedConstraints","stream","constraint","failResult","errorCode","MediaRequestErrorCode","PermissionDenied","GeneralError","message","getUserMediaStreamMediaDevice","streamPromise","navigator","mediaDevices","getUserMedia","getDisplayMedia","name","newConstraints","facingMode","e","setGroupedDevices","groupedDevices","back","front","streamConstraints","constraintCandidates","baseConstraints","cameraType","videoSourceType","LocalVideoSourceType","CAMERA_FRONT","frameRate","ideal","max","deviceId","exact","assignConstraint","videoResolution","_a","resolution","split","width","height","constraintCandidate","_constraintByType","push","Array","defaultResolution","DEFAULT_VIDEO_RESOLUTION","DEFAULT_VIDEO_CONSTRAINT_TYPE","errorName","permissionErrorNames","constraintsErrors","isErrorOfType","errorsArr","filter","Overconstrained","type","ConstraintType","MIN","min"],"mappings":"AAAA;;AACA,IAAIA,WAAY,aAAQ,UAAKA,QAAd,IAA2B,YAAY;AAClDA,eAAWC,OAAOC,MAAP,IAAiB,UAASC,CAAT,EAAY;AACpC,aAAK,IAAIC,CAAJ,EAAOC,IAAI,CAAX,EAAcC,IAAIC,UAAUC,MAAjC,EAAyCH,IAAIC,CAA7C,EAAgDD,GAAhD,EAAqD;AACjDD,gBAAIG,UAAUF,CAAV,CAAJ;AACA,iBAAK,IAAII,CAAT,IAAcL,CAAd;AAAiB,oBAAIH,OAAOS,SAAP,CAAiBC,cAAjB,CAAgCC,IAAhC,CAAqCR,CAArC,EAAwCK,CAAxC,CAAJ,EACbN,EAAEM,CAAF,IAAOL,EAAEK,CAAF,CAAP;AADJ;AAEH;AACD,eAAON,CAAP;AACH,KAPD;AAQA,WAAOH,SAASa,KAAT,CAAe,IAAf,EAAqBN,SAArB,CAAP;AACH,CAVD;AAWA,IAAIO,kBAAmB,aAAQ,UAAKA,eAAd,IAAkC,UAAUC,GAAV,EAAe;AACnE,WAAQA,OAAOA,IAAIC,UAAZ,GAA0BD,GAA1B,GAAgC,EAAE,WAAWA,GAAb,EAAvC;AACH,CAFD;AAGAd,OAAOgB,cAAP,CAAsBC,OAAtB,EAA+B,YAA/B,EAA6C,EAAEC,OAAO,IAAT,EAA7C;ACfA,IAAAC,WAAAN,gBAAAO,QAAA,eAAA,CAAA,CAAA;AACA,IAAAC,QAAAR,gBAAAO,QAAA,YAAA,CAAA,CAAA;AACA,IAAAE,SAAAT,gBAAAO,QAAA,aAAA,CAAA,CAAA;AACA,IAAAG,UAAAV,gBAAAO,QAAA,cAAA,CAAA,CAAA;AACA,IAAAI,aAAAJ,QAAA,UAAA,CAAA;AAaA,IAAAK,mBAAAL,QAAA,kBAAA,CAAA;AAOA,IAAAM,gBAAAN,QAAA,0BAAA,CAAA;AAWAO,OAAOC,qBAAP,GAA+B,IAA/B;AACA,IAAIC,mBAAiE,IAArE;AAEAF,OAAOG,wBAAP,GAAkC,IAAlC;AACA,IAAIC,sBAAoE,IAAxE;AAEA,IAAMC,QAAQN,cAAAO,cAAA,CAAe,oBAAf,CAAd;AAEA,IAAAC,qBAAA,aAAA,YAAA;AAKI,aAAAA,kBAAA,CAAYC,WAAZ,EAA0C;AAHlC,aAAAC,eAAA,GACuD,EAACC,OAAO,IAAR,EAAcC,OAAO,IAArB,EAA2BC,cAAc,CAAzC,EADvD;AAIJ,aAAKJ,WAAL,GAAmBA,WAAnB;AAEA,aAAKK,wBAAL,GAAgC,KAAKA,wBAAL,CAA8BC,IAA9B,CAAmC,IAAnC,CAAhC;AACA,aAAKC,eAAL,GAAuB,KAAKA,eAAL,CAAqBD,IAArB,CAA0B,IAA1B,CAAvB;AACH;AAEDzC,WAAAgB,cAAA,CAAIkB,mBAAAzB,SAAJ,EAAI,gBAAJ,EAAkB;ADnBdkC,aCmBJ,eAAA;AACI,mBAAO,KAAKP,eAAZ;AACH,SAFiB;ADhBdQ,oBAAY,ICgBE;ADfdC,sBAAc;ACeA,KAAlB;AAIAX,uBAAAzB,SAAA,CAAAqC,kBAAA,GAAA,UAAmBC,WAAnB,EAAqD;AAArD,YAAAC,QAAA,IAAA;AACIhB,cAAMiB,IAAN,CAAW,oBAAX,EAAiCF,WAAjC;AACA,YAAIlB,gBAAJ,EAAsB;AAClB,mBAAOA,gBAAP;AACH;AAED,YAAIF,OAAOC,qBAAX,EAAkC;AAC9B,gBAAMsB,aAAavB,OAAOC,qBAAP,CAA6BuB,WAA7B,CAAyCC,cAAzC,EAAnB;AAEA,gBAAIF,WAAW3C,MAAX,GAAoB,CAApB,IAAyB2C,WAAW,CAAX,EAAcG,UAAd,KAA6B,MAA1D,EAAkE;AAC9D,uBAAO7B,WAAA8B,OAAA,CAAQC,OAAR,CAAexD,SAAA,EAAA,EAAK4B,OAAOC,qBAAZ,EAAiC,EAAE4B,OAAO,KAAT,EAAjC,CAAf,CAAP;AACH;AAED,iBAAKC,sBAAL;AACH;AAED5B,2BAAmB,IAAIL,WAAA8B,OAAJ,CAAY,UAACC,OAAD,EAAUG,MAAV,EAAgB;AAC3C,gBAAMC,eAAe,SAAfA,YAAe,CAACC,GAAD,EAAS;AAC1B/B,mCAAmB,IAAnB;AACAF,uBAAOC,qBAAP,GAA+B,IAA/B;AACAI,sBAAM6B,KAAN,CAAY,oBAAZ,EAAkCD,GAAlC;AACAF,uBAAOE,GAAP;AACH,aALD;AAOA,gBAAI;AACAZ,sBAAKc,0BAAL,CAAgC,KAAhC,EAAuCf,WAAvC,EACKgB,IADL,CACU,UAACC,YAAD,EAAwC;AAC1CnC,uCAAmB,IAAnB;AACAF,2BAAOC,qBAAP,GAA+BoC,YAA/B;AACAT,4BAAOxD,SAAA,EAAA,EAAK4B,OAAOC,qBAAZ,EAAiC,EAAE4B,OAAO,IAAT,EAAjC,CAAP;AACH,iBALL,EAMKS,KANL,CAMWN,YANX;AAOH,aARD,CAQE,OAAOC,GAAP,EAAY;AACVD,6BAAaC,GAAb;AACH;AACJ,SAnBkB,CAAnB;AAqBA,eAAO/B,gBAAP;AACH,KAtCD;AAwCAK,uBAAAzB,SAAA,CAAAyD,qBAAA,GAAA,UAAsBnB,WAAtB,EAA8D;AAA9D,YAAAC,QAAA,IAAA;AACI,YAAIjB,mBAAJ,EAAyB;AACrB,mBAAOA,mBAAP;AACH;AAEDC,cAAMiB,IAAN,CAAW,uBAAX,EAAoCF,WAApC;AAEA,YAAIpB,OAAOG,wBAAX,EAAqC;AACjC,gBAAMqC,QAAQxC,OAAOG,wBAAP,CAAgCqB,WAAhC,CAA4CC,cAA5C,EAAd;AAEA,gBAAIe,MAAM5D,MAAN,GAAe,CAAf,IAAoB4D,MAAM,CAAN,EAASd,UAAT,KAAwB,MAAhD,EAAwD;AACpD;AACA,uBAAO7B,WAAA8B,OAAA,CAAQC,OAAR,CAAexD,SAAA,EAAA,EAAK4B,OAAOG,wBAAZ,EAAoC,EAAE0B,OAAO,KAAT,EAApC,CAAf,CAAP;AACH;AACD,iBAAKY,yBAAL,GAAiCH,KAAjC,CAAuC,UAACJ,KAAD,EAAW;AAC9C7B,sBAAM6B,KAAN,CAAY,uCAAZ,EAAqDA,KAArD;AACH,aAFD;AAIH;AAED9B,8BAAsB,IAAIP,WAAA8B,OAAJ,CAAY,UAACC,OAAD,EAAUG,MAAV,EAAgB;AAC9C,gBAAMW,oBAAoB,SAApBA,iBAAoB,CAACT,GAAD,EAAS;AAC/B7B,sCAAsB,IAAtB;AACAJ,uBAAOG,wBAAP,GAAkC,IAAlC;AACAE,sBAAM6B,KAAN,CAAY,uBAAZ,EAAqCD,GAArC;AACAF,uBAAOE,GAAP;AACH,aALD;AAOA,gBAAI;AACAZ,sBAAKc,0BAAL,CAAgC,IAAhC,EAAsC,EAACzB,OAAOU,WAAR,EAAqBT,OAAO,KAA5B,EAAtC,EACKyB,IADL,CACU,UAACC,YAAD,EAAwC;AAC1CjC,0CAAsB,IAAtB;AACAJ,2BAAOG,wBAAP,GAAkCkC,YAAlC;AAEAT,4BAAOxD,SAAA,EAAA,EAAK4B,OAAOG,wBAAZ,EAAoC,EAAE0B,OAAO,IAAT,EAApC,CAAP;AACH,iBANL,EAMOS,KANP,CAMaI,iBANb;AAOH,aARD,CAQE,OAAOT,GAAP,EAAY;AACVS,kCAAkBT,GAAlB;AACH;AACJ,SAnBqB,CAAtB;AAqBA,eAAO7B,mBAAP;AACH,KA1CD;AA4CAG,uBAAAzB,SAAA,CAAAgD,sBAAA,GAAA,YAAA;AACI,YAAMO,eAAerC,OAAOC,qBAA5B;AAEAD,eAAOC,qBAAP,GAA+B,IAA/B;AACAC,2BAAmB,IAAnB;AAEA,YAAImC,gBAAgBA,aAAab,WAAjC,EAA8C;AAC1Ca,yBAAab,WAAb,CAAyBmB,SAAzB,GAAqCC,OAArC,CAA6C,UAACJ,KAAD,EAAwB;AACjEA,sBAAMK,IAAN;AACH,aAFD;AAIAxC,kBAAMiB,IAAN,CAAW,0CAAX;AACH;AAED,eAAOzB,WAAA8B,OAAA,CAAQC,OAAR,EAAP;AACH,KAfD;AAiBArB,uBAAAzB,SAAA,CAAA2D,yBAAA,GAAA,YAAA;AACI,YAAMJ,eAAerC,OAAOG,wBAA5B;AAEAH,eAAOG,wBAAP,GAAkC,IAAlC;AACAC,8BAAsB,IAAtB;AAEA,YAAIiC,gBAAgBA,aAAab,WAAjC,EAA8C;AAC1Ca,yBAAab,WAAb,CAAyBmB,SAAzB,GAAqCC,OAArC,CAA6C,UAACJ,KAAD,EAAwB;AACjEA,sBAAMK,IAAN;AACH,aAFD;AAGH;AAED,eAAOhD,WAAA8B,OAAA,CAAQC,OAAR,EAAP;AACH,KAbD;AAeArB,uBAAAzB,SAAA,CAAAiC,eAAA,GAAA,YAAA;AACIV,cAAMiB,IAAN,CAAW,yCAAX;AAEA,eAAOzB,WAAA8B,OAAA,CAAQmB,GAAR,CAAY,CACf,KAAKL,yBAAL,EADe,EAEf,KAAKX,sBAAL,EAFe,CAAZ,EAGJM,IAHI,CAGC,YAAA;AAAM,mBAAAW,SAAA;AAAS,SAHhB,CAAP;AAIH,KAPD;AASQxC,uBAAAzB,SAAA,CAAAqD,0BAAA,GAAR,UAAmCa,cAAnC,EAA4D5B,WAA5D,EAA+F;AAA/F,YAAAC,QAAA,IAAA;AACI,YAAM4B,wBAAwB,KAAKC,oBAAL,CAA0BF,cAA1B,EAA0C5B,WAA1C,CAA9B;AAEAf,cAAMiB,IAAN,CAAW,qDAAX,EAAkE2B,qBAAlE;AAEA,YAAIA,sBAAsBrE,MAAtB,KAAiC,CAArC,EAAwC;AACpC,mBAAOiB,WAAA8B,OAAA,CAAQI,MAAR,CAAe,IAAIoB,KAAJ,CAAU,uCAAV,CAAf,CAAP;AACH;AAED,eAAO,IAAItD,WAAA8B,OAAJ,CAAY,UAACC,OAAD,EAAUG,MAAV,EAAgB;AAC/B,gBAAIqB,uBAA4D,IAAhE;AACA,gBAAIC,mBAAqD,IAAzD;AACA,gBAAIC,2BAA2B,KAA/B;AAECzD,uBAAA8B,OAAA,CAAgB4B,MAAhB,CAAuBN,qBAAvB,EAA8C,UAACO,KAAD,EAAaC,SAAb,EAA8C;AACzF,oBAAIH,wBAAJ,EAA8B;AAC1B;AACH;AAED,oBAAMI,iBAAiBhE,MAAAiE,OAAA,CAAIvC,WAAJ,EAAiB,sCAAjB,KAA4D1B,MAAAiE,OAAA,CAAIvC,WAAJ,EAAiB,yCAAjB,CAAnF;AAEA,uBAAOC,MAAKuC,mCAAL,CAAyCH,SAAzC,EAAoDT,cAApD,EACFZ,IADE,CACG,UAACyB,MAAD,EAAoB;AACtBT,2CAAuB;AACnB5B,qCAAaqC,MADM;AAEnBC,oCAAYL,SAFO;AAGnBC,wCAAcA,cAHK;AAInB7B,+BAAO;AAJY,qBAAvB;AAMAxB,0BAAMiB,IAAN,CAAW,oCAAX,EAAiD8B,oBAAjD;AACAE,+CAA2B,IAA3B;AACA1B,4BAAQwB,oBAAR;AACH,iBAXE,EAYFd,KAZE,CAYI,UAACyB,UAAD,EAAmC;AACtCV,uCAAmBU,UAAnB;AAEA,wBAAIV,iBAAiBW,SAAjB,KAA+BlE,iBAAAmE,qBAAA,CAAsBC,gBAAzD,EAA2E;AACvEZ,mDAA2B,IAA3B;AACAvB,+BAAOsB,gBAAP;AACH;AACJ,iBAnBE,CAAP;AAoBH,aA3BA,EA2BE,CA3BF,EA4BIjB,IA5BJ,CA4BS,YAAA;AACF,oBAAI,CAACkB,wBAAL,EAA+B;AAC3B,wBAAI,CAACD,gBAAL,EAAuB;AACnBA,2CAAmB;AACfW,uCAAWlE,iBAAAmE,qBAAA,CAAsBE,YADlB;AAEfC,qCAAS;AAFM,yBAAnB;AAIH;AACDrC,2BAAOsB,gBAAP;AACH;AACJ,aAtCJ;AAuCJ,SA5CM,CAAP;AA6CH,KAtDO;AAwDA9C,uBAAAzB,SAAA,CAAAuF,6BAAA,GAAR,UAAsCjD,WAAtC,EAA2E4B,cAA3E,EAAkG;AAC9F,YAAIsB,gBAAgB,IAApB;AAEA,YAAI,CAACtB,cAAL,EAAqB;AACjBsB,4BAAgBC,UAAUC,YAAV,CAAuBC,YAAvB,CAAoCrD,WAApC,CAAhB;AACH,SAFD,MAEO;AACH;AACAkD,4BAAgBC,UAAUC,YAAV,CAAuBE;AACnC;AADY,cAEVH,UAAUC,YAAV,CAAuBE,eAAvB,CAAuCtD,WAAvC,CAFU,GAGVmD,UAAUG,eAAV,CAA0BtD,WAA1B,CAHN;AAIH;AAED,eAAOkD,aAAP;AACH,KAdO;AAgBR;AACQ/D,uBAAAzB,SAAA,CAAA8E,mCAAA,GAAR,UAA4CxC,WAA5C,EAAiF4B,cAAjF,EAAwG;AAAxG,YAAA3B,QAAA,IAAA;AACIhB,cAAMiB,IAAN,CAAW,qCAAX,EAAkDF,WAAlD;AAEA,YAAI;AACA,mBAAO,KAAKiD,6BAAL,CAAmCjD,WAAnC,EAAgD4B,cAAhD,EAAgEV,KAAhE,CAAsE,UAACL,GAAD,EAAS;AAClF,oBAAIqC,gBAAgB,IAApB;AAEAjE,sBAAMiB,IAAN,CAAW,6DAAX,EAA0EW,IAAI0C,IAA9E;AAEA;AACA;AACA;AACA;AACA,oBAAI1C,OAAQA,IAAI0C,IAAJ,KAAa,kBAArB,IAA4CjF,MAAAiE,OAAA,CAAIvC,WAAJ,EAAiB,kBAAjB,CAA5C,IACA1B,MAAAiE,OAAA,CAAIvC,WAAJ,EAAiB,gBAAjB,CADJ,EACwC;AAEpC,wBAAMwD,iBAAiBxD,WAAvB;AAEAf,0BAAMiB,IAAN,CAAW,oHAAkH5B,MAAAiE,OAAA,CAAIvC,WAAJ,EAAiB,kBAAjB,CAA7H,EAAqKa,IAAI0C,IAAzK;AAEA;AACA,2BAAOC,eAAelE,KAAf,CAAqBmE,UAA5B;AAEAP,oCAAgBjD,MAAKuC,mCAAL,CAAyCgB,cAAzC,EAAyD5B,cAAzD,CAAhB;AACH,iBAXD,MAWO,IAAIf,OAAOA,IAAI0C,IAAX,IAAmBvD,YAAYT,KAA/B,IAAwC,CAACS,YAAYV,KAAzD,EAAgE;AACnEL,0BAAMiB,IAAN,CAAW,iGAAX,EAA8GW,IAAI0C,IAAlH;AAEA,2BAAOtD,MAAKR,wBAAL,CAA8B,EAAC8D,MAAM,mBAAP,EAA9B,CAAP;AACH,iBAJM,MAIA,IAAI1C,OAAOA,IAAI0C,IAAX,IAAmBvD,YAAYT,KAAnC,EAA0C;AAC7C,wBAAMiE,iBAAiBxD,WAAvB;AAEAf,0BAAMiB,IAAN,CAAW,2HAAX,EAAwIW,IAAI0C,IAA5I;AAEAC,mCAAejE,KAAf,GAAuB,KAAvB;AACA2D,oCAAgBjD,MAAKuC,mCAAL,CAAyCgB,cAAzC,EAAyD5B,cAAzD,CAAhB;AACH,iBAPM,MAOA,IAAIf,OAAO,CAACqC,aAAZ,EAA2B;AAC9BjE,0BAAMiB,IAAN,CAAW,6EAAX,EAA0FW,OAAOA,IAAI0C,IAArG;AAEAL,oCAAgBzE,WAAA8B,OAAA,CAAQI,MAAR,CAAeE,GAAf,CAAhB;AACH;AAED,uBAAOqC,iBAAiBA,cAAchC,KAAd,CAAoBjB,MAAKR,wBAAzB,CAAxB;AACH,aAtCM,CAAP;AAuCH,SAxCD,CAwCE,OAAOiE,CAAP,EAAU;AACR,iBAAKjE,wBAAL,CAA8BiE,CAA9B;AACH;AACJ,KA9CO;AAgDRvE,uBAAAzB,SAAA,CAAAiG,iBAAA,GAAA,UAAkBC,cAAlB,EAAsF;AAClF3E,cAAMiB,IAAN,CAAW,mBAAX,EAAgC0D,cAAhC;AACA,aAAKvE,eAAL,GAAuB,EAACC,OAAOsE,eAAetE,KAAf,GACzB,EAACuE,MAAMtF,OAAAgE,OAAA,CAAKqB,eAAetE,KAApB,CAAP,EAAmCwE,OAAOtF,QAAA+D,OAAA,CAAMqB,eAAetE,KAArB,CAA1C,EADyB,GAEzB,IAFiB;AAGvBC,mBAAOhB,OAAAgE,OAAA,CAAKqB,eAAerE,KAApB,KAA8B,IAHd;AAIvBC,0BAAcoE,eAAetE,KAAf,CAAqB9B,MAJZ,EAAvB;AAKH,KAPD;AASQ2B,uBAAAzB,SAAA,CAAAoE,oBAAA,GAAR,UAA6BF,cAA7B,EAAsDmC,iBAAtD,EAA+F;AAC3F,YAAMC,uBAAiD,EAAvD;AAEA;AACA,YAAIpC,cAAJ,EAAoB;AAChB,mBAAO,CAAC,EAACtC,OAAO,IAAR,EAAD,CAAP;AACH;AAED,YAAI,CAACyE,iBAAL,EAAwB;AACpB,mBAAOC,oBAAP;AACH;AAED,YAAMC,kBAA0C,EAAhD;AAEAA,wBAAgB1E,KAAhB,GAAwB,OAAOwE,kBAAkBxE,KAAzB,KAAmC,SAAnC,GAA+CwE,kBAAkBxE,KAAjE,GAAyEoC,SAAjG;AAEA,YAAI,CAACoC,kBAAkBzE,KAAvB,EAA8B;AAC1B,mBAAO,CAAC2E,eAAD,CAAP;AACH;AAED,YAAMC,aAAcH,kBAAkBzE,KAAlB,KAA4B,SAA7B,IACdyE,kBAAkBzE,KAAlB,CAAwD6E,eAD7D;AAGAF,wBAAgB3E,KAAhB,GAAwB;AACpBmE,wBAAYS,cAAcA,eAAexF,iBAAA0F,oBAAA,CAAqBC,YAAlD,GACN,MADM,GAEN,aAHc;AAIpBC,uBAAW,EAACC,OAAO,EAAR,EAAYC,KAAK,EAAjB;AAJS,SAAxB;AAOA,YAAMC,WAAWnG,MAAAiE,OAAA,CAAI,KAAKlD,eAAT,EAA0B,YAAS6E,eAAexF,iBAAA0F,oBAAA,CAAqBC,YAApC,GAAmD,OAAnD,GAA6D,MAAtE,IAA4E,WAAtG,CAAjB;AAEA,YAAII,QAAJ,EAAc;AACVR,4BAAgB3E,KAAhB,CAAsBmF,QAAtB,GAAiC,EAACC,OAAOD,QAAR,EAAjC;AACH;AAED,YAAME,mBAAmB,SAAnBA,gBAAmB,CAACC,eAAD,EAAuC;AACtD,gBAAAC,KAAAD,gBAAAE,UAAA,CAAAC,KAAA,CAAA,GAAA,CAAA;AAAA,gBAACC,QAAAH,GAAA,CAAA,CAAD;AAAA,gBAAQI,SAAAJ,GAAA,CAAA,CAAR;AACN,gBAAMK,sBAAsB9G,SAAAmE,OAAA,CAAO,EAAP,EAAW0B,eAAX,CAA5B;AAEAiB,gCAAoB5F,KAApB,GAA4BlB,SAAAmE,OAAA,CAAO,EAAP,EAAW0B,gBAAgB3E,KAA3B,EAAkC;AAC1D0F,uBAAOG,kBAAkBH,KAAlB,EAAyBJ,gBAAgBtC,cAAzC,CADmD;AAE1D2C,wBAAQE,kBAAkBF,MAAlB,EAA0BL,gBAAgBtC,cAA1C;AAFkD,aAAlC,CAA5B;AAIA0B,iCAAqBoB,IAArB,CAA0BF,mBAA1B;AACH,SATD;AAWA,YAAKjB,gBAAgB3E,KAAhB,IAAyB,OAAOyE,kBAAkBzE,KAAzB,KAAmC,SAA7D,IACA,EAAGyE,kBAAkBzE,KAAlB,CAAwDsF,eAAxD,YAAmFS,KAAtF,CADJ,EACkG;AAE9F,gBAAMC,oBAA2C;AAC7CR,4BAAYpG,iBAAA6G,wBADiC;AAE7CjD,gCAAgB5D,iBAAA8G;AAF6B,aAAjD;AAKAb,6BAAiBW,iBAAjB;AAEA,mBAAOtB,oBAAP;AACH;AAEAD,0BAAkBzE,KAAlB,CAAgCsF,eAAhC,CAAgDpD,OAAhD,CAAwDmD,gBAAxD;AAED,eAAOX,oBAAP;AACH,KA/DO;AAiEA7E,uBAAAzB,SAAA,CAAA+B,wBAAA,GAAR,UAAiCqB,KAAjC,EAA2C;AACvC,YAAIA,SAASA,MAAM8B,SAAnB,EAA8B;AAC1B,kBAAM9B,KAAN;AACH;AAED7B,cAAM6B,KAAN,CAAY,yBAAZ,EAAuCA,KAAvC;AACA,YAAM2E,YAAY3E,SAASA,MAAMyC,IAAf,GAAsBzC,MAAMyC,IAA5B,GAAmC,EAArD;AAEA,YAAMmC,uBAAuB,CACzB,iBADyB,EAEzB,0BAFyB,EAGzB,uBAHyB,CAA7B;AAOA;AACA;AACA;AACA;AACA;AACA;AAEA,YAAMC,oBAAoB,CACtB,sBADsB,EAEtB,6BAFsB,CAA1B;AAMA,YAAMC,gBAAgB,SAAhBA,aAAgB,CAACC,SAAD,EAAoB;AAAK,mBAAAA,UAAUC,MAAV,CAAiB,UAACjF,GAAD,EAAY;AAAK,uBAAAA,QAAQ4E,SAAR;AAAiB,aAAnD,EAAqDjI,MAArD,GAA8D,CAA9D;AAA+D,SAA9G;AAEA,YAAIoI,cAAcF,oBAAd,CAAJ,EAAyC;AACrC,kBAAM,EAAC9C,WAAWlE,iBAAAmE,qBAAA,CAAsBC,gBAAlC,EAAoDE,SAASyC,SAA7D,EAAN;AACH;AAED,YAAIG,cAAcD,iBAAd,CAAJ,EAAsC;AAClC,kBAAM,EAAC/C,WAAWlE,iBAAAmE,qBAAA,CAAsBkD,eAAlC,EAAmD/C,SAASyC,SAA5D,EAAN;AACH;AAED,cAAM,EAAC7C,WAAWlE,iBAAAmE,qBAAA,CAAsBE,YAAlC,EAAgDC,SAASyC,SAAzD,EAAN;AACH,KAvCO;AAwCZ,WAAAtG,kBAAA;AAxXA,CAAA,EAAA;AAAajB,QAAAiB,kBAAA,GAAAA,kBAAA;AA0Xb,SAASgG,iBAAT,CAA2BhH,KAA3B,EAAuC6H,IAAvC,EAA2D;AACvD,YAAQA,IAAR;AACI,aAAKtH,iBAAAuH,cAAA,CAAeC,GAApB;AACI,mBAAO,EAACC,KAAKhI,KAAN,EAAP;AACJ;AACI,mBAAO,EAACoG,OAAOpG,KAAR,EAAP;AAJR;AAMH;;ADnFD","file":"LocalStreamManager.js","sourcesContent":["\"use strict\";\nvar __assign = (this && this.__assign) || function () {\n __assign = Object.assign || function(t) {\n for (var s, i = 1, n = arguments.length; i < n; i++) {\n s = arguments[i];\n for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p))\n t[p] = s[p];\n }\n return t;\n };\n return __assign.apply(this, arguments);\n};\nvar __importDefault = (this && this.__importDefault) || function (mod) {\n return (mod && mod.__esModule) ? mod : { \"default\": mod };\n};\nObject.defineProperty(exports, \"__esModule\", { value: true });\nvar assign_1 = __importDefault(require(\"lodash/assign\"));\nvar get_1 = __importDefault(require(\"lodash/get\"));\nvar last_1 = __importDefault(require(\"lodash/last\"));\nvar first_1 = __importDefault(require(\"lodash/first\"));\nvar bluebird_1 = require(\"bluebird\");\nvar MediaConstants_1 = require(\"./MediaConstants\");\nvar MediaTracer_1 = require(\"./MediaUtils/MediaTracer\");\nwindow.latestUserMediaStream = null;\nvar userMediaPromise = null;\nwindow.latestDesktopMediaStream = null;\nvar desktopMediaPromise = null;\nvar trace = MediaTracer_1.getMediaTracer('LocalStreamManager');\nvar LocalStreamManager = /** @class */ (function () {\n function LocalStreamManager(environment) {\n this._groupedDevices = { video: null, audio: null, camerasCount: 0 };\n this.environment = environment;\n this.parseMediaErrorThenThrow = this.parseMediaErrorThenThrow.bind(this);\n this.clearAllStreams = this.clearAllStreams.bind(this);\n }\n Object.defineProperty(LocalStreamManager.prototype, \"groupedDevices\", {\n get: function () {\n return this._groupedDevices;\n },\n enumerable: true,\n configurable: true\n });\n LocalStreamManager.prototype.getUserMediaStream = function (constraints) {\n var _this = this;\n trace.info('getUserMediaStream', constraints);\n if (userMediaPromise) {\n return userMediaPromise;\n }\n if (window.latestUserMediaStream) {\n var videoTrack = window.latestUserMediaStream.mediaStream.getVideoTracks();\n if (videoTrack.length > 0 && videoTrack[0].readyState === 'live') {\n return bluebird_1.Promise.resolve(__assign({}, window.latestUserMediaStream, { isNew: false }));\n }\n this.destroyUserMediaStream();\n }\n userMediaPromise = new bluebird_1.Promise(function (resolve, reject) {\n var catchHandler = function (err) {\n userMediaPromise = null;\n window.latestUserMediaStream = null;\n trace.error('getUserMedia error', err);\n reject(err);\n };\n try {\n _this.getHighestResolutionStream(false, constraints)\n .then(function (streamResult) {\n userMediaPromise = null;\n window.latestUserMediaStream = streamResult;\n resolve(__assign({}, window.latestUserMediaStream, { isNew: true }));\n })\n .catch(catchHandler);\n }\n catch (err) {\n catchHandler(err);\n }\n });\n return userMediaPromise;\n };\n LocalStreamManager.prototype.getDesktopMediaStream = function (constraints) {\n var _this = this;\n if (desktopMediaPromise) {\n return desktopMediaPromise;\n }\n trace.info('getDesktopMediaStream', constraints);\n if (window.latestDesktopMediaStream) {\n var track = window.latestDesktopMediaStream.mediaStream.getVideoTracks();\n if (track.length > 0 && track[0].readyState === 'live') {\n //return Promise.resolve();\n return bluebird_1.Promise.resolve(__assign({}, window.latestDesktopMediaStream, { isNew: false }));\n }\n this.destroyDesktopMediaStream().catch(function (error) {\n trace.error('Error destroying desktop media stream', error);\n });\n }\n desktopMediaPromise = new bluebird_1.Promise(function (resolve, reject) {\n var mediaCatchHandler = function (err) {\n desktopMediaPromise = null;\n window.latestDesktopMediaStream = null;\n trace.error('getDisplayMedia error', err);\n reject(err);\n };\n try {\n _this.getHighestResolutionStream(true, { video: constraints, audio: false })\n .then(function (streamResult) {\n desktopMediaPromise = null;\n window.latestDesktopMediaStream = streamResult;\n resolve(__assign({}, window.latestDesktopMediaStream, { isNew: true }));\n }).catch(mediaCatchHandler);\n }\n catch (err) {\n mediaCatchHandler(err);\n }\n });\n return desktopMediaPromise;\n };\n LocalStreamManager.prototype.destroyUserMediaStream = function () {\n var streamResult = window.latestUserMediaStream;\n window.latestUserMediaStream = null;\n userMediaPromise = null;\n if (streamResult && streamResult.mediaStream) {\n streamResult.mediaStream.getTracks().forEach(function (track) {\n track.stop();\n });\n trace.info('destroyUserMediaStream - stop all tracks');\n }\n return bluebird_1.Promise.resolve();\n };\n LocalStreamManager.prototype.destroyDesktopMediaStream = function () {\n var streamResult = window.latestDesktopMediaStream;\n window.latestDesktopMediaStream = null;\n desktopMediaPromise = null;\n if (streamResult && streamResult.mediaStream) {\n streamResult.mediaStream.getTracks().forEach(function (track) {\n track.stop();\n });\n }\n return bluebird_1.Promise.resolve();\n };\n LocalStreamManager.prototype.clearAllStreams = function () {\n trace.info('LocalStreamManager Clearing all streams');\n return bluebird_1.Promise.all([\n this.destroyDesktopMediaStream(),\n this.destroyUserMediaStream()\n ]).then(function () { return undefined; });\n };\n LocalStreamManager.prototype.getHighestResolutionStream = function (isDesktopMedia, constraints) {\n var _this = this;\n var constraintsCandidates = this.getParsedConstraints(isDesktopMedia, constraints);\n trace.info('getHighestResolutionStream - constraintsCandidates:', constraintsCandidates);\n if (constraintsCandidates.length === 0) {\n return bluebird_1.Promise.reject(new Error('No constraints to retrieve the stream'));\n }\n return new bluebird_1.Promise(function (resolve, reject) {\n var lastSuccessfulResult = null;\n var lastFailedResult = null;\n var isStreamRequestFulfilled = false;\n bluebird_1.Promise.reduce(constraintsCandidates, function (total, candidate) {\n if (isStreamRequestFulfilled) {\n return;\n }\n var constraintType = get_1.default(constraints, 'video.videoResolution.constraintType') || get_1.default(constraints, 'video.videoResolution[0].constraintType');\n return _this.getStreamPromiseByParsedConstraints(candidate, isDesktopMedia)\n .then(function (stream) {\n lastSuccessfulResult = {\n mediaStream: stream,\n constraint: candidate,\n constraintType: constraintType,\n isNew: true\n };\n trace.info('getHighestResolutionStream Success', lastSuccessfulResult);\n isStreamRequestFulfilled = true;\n resolve(lastSuccessfulResult);\n })\n .catch(function (failResult) {\n lastFailedResult = failResult;\n if (lastFailedResult.errorCode === MediaConstants_1.MediaRequestErrorCode.PermissionDenied) {\n isStreamRequestFulfilled = true;\n reject(lastFailedResult);\n }\n });\n }, 0)\n .then(function () {\n if (!isStreamRequestFulfilled) {\n if (!lastFailedResult) {\n lastFailedResult = {\n errorCode: MediaConstants_1.MediaRequestErrorCode.GeneralError,\n message: 'Suitable stream cannot be created'\n };\n }\n reject(lastFailedResult);\n }\n });\n });\n };\n LocalStreamManager.prototype.getUserMediaStreamMediaDevice = function (constraints, isDesktopMedia) {\n var streamPromise = null;\n if (!isDesktopMedia) {\n streamPromise = navigator.mediaDevices.getUserMedia(constraints);\n }\n else {\n //@ts-ignore\n streamPromise = navigator.mediaDevices.getDisplayMedia\n //@ts-ignore\n ? navigator.mediaDevices.getDisplayMedia(constraints)\n : navigator.getDisplayMedia(constraints);\n }\n return streamPromise;\n };\n //@ts-ignore\n LocalStreamManager.prototype.getStreamPromiseByParsedConstraints = function (constraints, isDesktopMedia) {\n var _this = this;\n trace.info('getStreamPromiseByParsedConstraints', constraints);\n try {\n return this.getUserMediaStreamMediaDevice(constraints, isDesktopMedia).catch(function (err) {\n var streamPromise = null;\n trace.info('getStreamPromiseByParsedConstraints - get user stream error', err.name);\n // Android (Galaxy esp.) have a bug that when using facingMode it might cause NotReadableError. So\n // we should try again but without facingMode. The exception to the rule is when we don't have specific\n // device to use (like in IOS13 bug, as can be seen in MediaServiceBase.ts), in that case, we might end\n // up using the front camera, which is a privacy concern and the preference is to fallback from video\n if (err && (err.name === 'NotReadableError') && get_1.default(constraints, 'video.facingMode') &&\n get_1.default(constraints, 'video.deviceId')) {\n var newConstraints = constraints;\n trace.info(\"getStreamPromiseByParsedConstraints - NotReadableError - Failed to get video user media stream with facingMode=\" + get_1.default(constraints, 'video.facingMode'), err.name);\n // @ts-ignore\n delete newConstraints.video.facingMode;\n streamPromise = _this.getStreamPromiseByParsedConstraints(newConstraints, isDesktopMedia);\n }\n else if (err && err.name && constraints.audio && !constraints.video) {\n trace.info('getStreamPromiseByParsedConstraints - audioStreamFailed - Failed to get audio user media stream', err.name);\n return _this.parseMediaErrorThenThrow({ name: 'audioStreamFailed' });\n }\n else if (err && err.name && constraints.audio) {\n var newConstraints = constraints;\n trace.info('getStreamPromiseByParsedConstraints - Failed to get video and audio user media stream, try to get video media stream only', err.name);\n newConstraints.audio = false;\n streamPromise = _this.getStreamPromiseByParsedConstraints(newConstraints, isDesktopMedia);\n }\n else if (err && !streamPromise) {\n trace.info('getStreamPromiseByParsedConstraints - Failed to get video user media stream', err && err.name);\n streamPromise = bluebird_1.Promise.reject(err);\n }\n return streamPromise && streamPromise.catch(_this.parseMediaErrorThenThrow);\n });\n }\n catch (e) {\n this.parseMediaErrorThenThrow(e);\n }\n };\n LocalStreamManager.prototype.setGroupedDevices = function (groupedDevices) {\n trace.info('setGroupedDevices', groupedDevices);\n this._groupedDevices = { video: groupedDevices.video\n ? { back: last_1.default(groupedDevices.video), front: first_1.default(groupedDevices.video) }\n : null,\n audio: last_1.default(groupedDevices.audio) || null,\n camerasCount: groupedDevices.video.length };\n };\n LocalStreamManager.prototype.getParsedConstraints = function (isDesktopMedia, streamConstraints) {\n var constraintCandidates = [];\n //TODO - Alex: need to understand which constraints not compatible with screen share\n if (isDesktopMedia) {\n return [{ video: true }];\n }\n if (!streamConstraints) {\n return constraintCandidates;\n }\n var baseConstraints = {};\n baseConstraints.audio = typeof streamConstraints.audio === 'boolean' ? streamConstraints.audio : undefined;\n if (!streamConstraints.video) {\n return [baseConstraints];\n }\n var cameraType = (streamConstraints.video !== 'boolean') &&\n streamConstraints.video.videoSourceType;\n baseConstraints.video = {\n facingMode: cameraType && cameraType === MediaConstants_1.LocalVideoSourceType.CAMERA_FRONT\n ? 'user'\n : 'environment',\n frameRate: { ideal: 15, max: 30 }\n };\n var deviceId = get_1.default(this._groupedDevices, \"video.\" + (cameraType === MediaConstants_1.LocalVideoSourceType.CAMERA_FRONT ? 'front' : 'back') + \".deviceId\");\n if (deviceId) {\n baseConstraints.video.deviceId = { exact: deviceId };\n }\n var assignConstraint = function (videoResolution) {\n var _a = videoResolution.resolution.split('x'), width = _a[0], height = _a[1];\n var constraintCandidate = assign_1.default({}, baseConstraints);\n constraintCandidate.video = assign_1.default({}, baseConstraints.video, {\n width: _constraintByType(width, videoResolution.constraintType),\n height: _constraintByType(height, videoResolution.constraintType)\n });\n constraintCandidates.push(constraintCandidate);\n };\n if ((baseConstraints.video && typeof streamConstraints.video === 'boolean') ||\n !(streamConstraints.video.videoResolution instanceof Array)) {\n var defaultResolution = {\n resolution: MediaConstants_1.DEFAULT_VIDEO_RESOLUTION,\n constraintType: MediaConstants_1.DEFAULT_VIDEO_CONSTRAINT_TYPE\n };\n assignConstraint(defaultResolution);\n return constraintCandidates;\n }\n streamConstraints.video.videoResolution.forEach(assignConstraint);\n return constraintCandidates;\n };\n LocalStreamManager.prototype.parseMediaErrorThenThrow = function (error) {\n if (error && error.errorCode) {\n throw error;\n }\n trace.error('Get media stream error.', error);\n var errorName = error && error.name ? error.name : '';\n var permissionErrorNames = [\n 'NotAllowedError',\n 'PermissionDismissedError',\n 'PermissionDeniedError'\n ];\n // const generalError = [\n // 'NotFoundError',\n // 'DevicesNotFoundError',\n // 'NotReadableError',\n // 'TrackStartError'\n // ];\n var constraintsErrors = [\n 'OverconstrainedError',\n 'ConstraintNotSatisfiedError'\n ];\n var isErrorOfType = function (errorsArr) { return errorsArr.filter(function (err) { return err === errorName; }).length > 0; };\n if (isErrorOfType(permissionErrorNames)) {\n throw { errorCode: MediaConstants_1.MediaRequestErrorCode.PermissionDenied, message: errorName };\n }\n if (isErrorOfType(constraintsErrors)) {\n throw { errorCode: MediaConstants_1.MediaRequestErrorCode.Overconstrained, message: errorName };\n }\n throw { errorCode: MediaConstants_1.MediaRequestErrorCode.GeneralError, message: errorName };\n };\n return LocalStreamManager;\n}());\nexports.LocalStreamManager = LocalStreamManager;\nfunction _constraintByType(value, type) {\n switch (type) {\n case MediaConstants_1.ConstraintType.MIN:\n return { min: value };\n default:\n return { ideal: value };\n }\n}\n\n//# sourceMappingURL=LocalStreamManager.js.map\n","import assign from 'lodash/assign';\nimport get from 'lodash/get';\nimport last from 'lodash/last';\nimport first from 'lodash/first';\nimport {Promise} from 'bluebird';\n\nimport {Nullable} from '@techsee/techsee-common';\n\nimport {\n IMediaEnvironment,\n LocalMediaConstraints,\n LocalVideoStreamConstraints,\n MediaRequestFailResult,\n MediaRequestSuccessResult,\n VideoStreamResolution\n} from './MediaContracts';\n\nimport {\n ConstraintType,\n DEFAULT_VIDEO_CONSTRAINT_TYPE,\n DEFAULT_VIDEO_RESOLUTION,\n LocalVideoSourceType,\n MediaRequestErrorCode\n} from './MediaConstants';\nimport {getMediaTracer} from './MediaUtils/MediaTracer';\n\ndeclare global {\n interface Window {\n latestUserMediaStream: Nullable<MediaRequestSuccessResult>;\n latestDesktopMediaStream: Nullable<MediaRequestSuccessResult>;\n }\n}\n\ndeclare const window: Window;\n\nwindow.latestUserMediaStream = null;\nlet userMediaPromise: Nullable<Promise<MediaRequestSuccessResult>> = null;\n\nwindow.latestDesktopMediaStream = null;\nlet desktopMediaPromise: Nullable<Promise<MediaRequestSuccessResult>> = null;\n\nconst trace = getMediaTracer('LocalStreamManager');\n\nexport class LocalStreamManager {\n private environment: IMediaEnvironment;\n private _groupedDevices: {video: Nullable<{back?: MediaDeviceInfo; front?: MediaDeviceInfo}>;\n audio: Nullable<MediaDeviceInfo>; camerasCount: number;} = {video: null, audio: null, camerasCount: 0};\n\n constructor(environment: IMediaEnvironment) {\n this.environment = environment;\n\n this.parseMediaErrorThenThrow = this.parseMediaErrorThenThrow.bind(this);\n this.clearAllStreams = this.clearAllStreams.bind(this);\n }\n\n get groupedDevices() {\n return this._groupedDevices;\n }\n\n getUserMediaStream(constraints: LocalMediaConstraints): Promise<MediaRequestSuccessResult> {\n trace.info('getUserMediaStream', constraints);\n if (userMediaPromise) {\n return userMediaPromise;\n }\n\n if (window.latestUserMediaStream) {\n const videoTrack = window.latestUserMediaStream.mediaStream.getVideoTracks();\n\n if (videoTrack.length > 0 && videoTrack[0].readyState === 'live') {\n return Promise.resolve({...window.latestUserMediaStream, isNew: false});\n }\n\n this.destroyUserMediaStream();\n }\n\n userMediaPromise = new Promise((resolve, reject) => {\n const catchHandler = (err: any): void => {\n userMediaPromise = null;\n window.latestUserMediaStream = null;\n trace.error('getUserMedia error', err);\n reject(err);\n };\n\n try {\n this.getHighestResolutionStream(false, constraints)\n .then((streamResult: MediaRequestSuccessResult) => {\n userMediaPromise = null;\n window.latestUserMediaStream = streamResult;\n resolve({...window.latestUserMediaStream, isNew: true});\n })\n .catch(catchHandler);\n } catch (err) {\n catchHandler(err);\n }\n });\n\n return userMediaPromise;\n }\n\n getDesktopMediaStream(constraints: LocalVideoStreamConstraints): Promise<MediaRequestSuccessResult> {\n if (desktopMediaPromise) {\n return desktopMediaPromise;\n }\n\n trace.info('getDesktopMediaStream', constraints);\n\n if (window.latestDesktopMediaStream) {\n const track = window.latestDesktopMediaStream.mediaStream.getVideoTracks();\n\n if (track.length > 0 && track[0].readyState === 'live') {\n //return Promise.resolve();\n return Promise.resolve({...window.latestDesktopMediaStream, isNew: false});\n }\n this.destroyDesktopMediaStream().catch((error: any) => {\n trace.error('Error destroying desktop media stream', error);\n });\n\n }\n\n desktopMediaPromise = new Promise((resolve, reject) => {\n const mediaCatchHandler = (err: any): void => {\n desktopMediaPromise = null;\n window.latestDesktopMediaStream = null;\n trace.error('getDisplayMedia error', err);\n reject(err);\n };\n\n try {\n this.getHighestResolutionStream(true, {video: constraints, audio: false})\n .then((streamResult: MediaRequestSuccessResult) => {\n desktopMediaPromise = null;\n window.latestDesktopMediaStream = streamResult;\n\n resolve({...window.latestDesktopMediaStream, isNew: true});\n }).catch(mediaCatchHandler);\n } catch (err) {\n mediaCatchHandler(err);\n }\n });\n\n return desktopMediaPromise;\n }\n\n destroyUserMediaStream(): Promise<void> {\n const streamResult = window.latestUserMediaStream;\n\n window.latestUserMediaStream = null;\n userMediaPromise = null;\n\n if (streamResult && streamResult.mediaStream) {\n streamResult.mediaStream.getTracks().forEach((track: MediaStreamTrack) => {\n track.stop();\n });\n\n trace.info('destroyUserMediaStream - stop all tracks');\n }\n\n return Promise.resolve();\n }\n\n destroyDesktopMediaStream(): Promise<void> {\n const streamResult = window.latestDesktopMediaStream;\n\n window.latestDesktopMediaStream = null;\n desktopMediaPromise = null;\n\n if (streamResult && streamResult.mediaStream) {\n streamResult.mediaStream.getTracks().forEach((track: MediaStreamTrack) => {\n track.stop();\n });\n }\n\n return Promise.resolve();\n }\n\n clearAllStreams(): Promise<void> {\n trace.info('LocalStreamManager Clearing all streams');\n\n return Promise.all([\n this.destroyDesktopMediaStream(),\n this.destroyUserMediaStream()\n ]).then(() => undefined);\n }\n\n private getHighestResolutionStream(isDesktopMedia: boolean, constraints?: LocalMediaConstraints): Promise<MediaRequestSuccessResult> {\n const constraintsCandidates = this.getParsedConstraints(isDesktopMedia, constraints);\n\n trace.info('getHighestResolutionStream - constraintsCandidates:', constraintsCandidates);\n\n if (constraintsCandidates.length === 0) {\n return Promise.reject(new Error('No constraints to retrieve the stream'));\n }\n\n return new Promise((resolve, reject) => {\n let lastSuccessfulResult: Nullable<MediaRequestSuccessResult> = null;\n let lastFailedResult: Nullable<MediaRequestFailResult> = null;\n let isStreamRequestFulfilled = false;\n\n (Promise as any).reduce(constraintsCandidates, (total: any, candidate: MediaStreamConstraints) => {\n if (isStreamRequestFulfilled) {\n return;\n }\n\n const constraintType = get(constraints, 'video.videoResolution.constraintType') || get(constraints, 'video.videoResolution[0].constraintType');\n\n return this.getStreamPromiseByParsedConstraints(candidate, isDesktopMedia)\n .then((stream: MediaStream) => {\n lastSuccessfulResult = {\n mediaStream: stream,\n constraint: candidate,\n constraintType,\n isNew: true\n };\n trace.info('getHighestResolutionStream Success', lastSuccessfulResult);\n isStreamRequestFulfilled = true;\n resolve(lastSuccessfulResult);\n })\n .catch((failResult: MediaRequestFailResult) => {\n lastFailedResult = failResult;\n\n if (lastFailedResult.errorCode === MediaRequestErrorCode.PermissionDenied) {\n isStreamRequestFulfilled = true;\n reject(lastFailedResult);\n }\n });\n }, 0)\n .then(() => {\n if (!isStreamRequestFulfilled) {\n if (!lastFailedResult) {\n lastFailedResult = {\n errorCode: MediaRequestErrorCode.GeneralError,\n message: 'Suitable stream cannot be created'\n };\n }\n reject(lastFailedResult);\n }\n });\n });\n }\n\n private getUserMediaStreamMediaDevice(constraints: MediaStreamConstraints, isDesktopMedia: boolean) {\n let streamPromise = null;\n\n if (!isDesktopMedia) {\n streamPromise = navigator.mediaDevices.getUserMedia(constraints);\n } else {\n //@ts-ignore\n streamPromise = navigator.mediaDevices.getDisplayMedia\n //@ts-ignore\n ? navigator.mediaDevices.getDisplayMedia(constraints)\n : navigator.getDisplayMedia(constraints);\n }\n\n return streamPromise;\n }\n\n //@ts-ignore\n private getStreamPromiseByParsedConstraints(constraints: MediaStreamConstraints, isDesktopMedia: boolean): Promise<MediaStream> {\n trace.info('getStreamPromiseByParsedConstraints', constraints);\n\n try {\n return this.getUserMediaStreamMediaDevice(constraints, isDesktopMedia).catch((err: any) => {\n let streamPromise = null;\n\n trace.info('getStreamPromiseByParsedConstraints - get user stream error', err.name);\n\n // Android (Galaxy esp.) have a bug that when using facingMode it might cause NotReadableError. So\n // we should try again but without facingMode. The exception to the rule is when we don't have specific\n // device to use (like in IOS13 bug, as can be seen in MediaServiceBase.ts), in that case, we might end\n // up using the front camera, which is a privacy concern and the preference is to fallback from video\n if (err && (err.name === 'NotReadableError') && get(constraints, 'video.facingMode') &&\n get(constraints, 'video.deviceId')) {\n\n const newConstraints = constraints;\n\n trace.info(`getStreamPromiseByParsedConstraints - NotReadableError - Failed to get video user media stream with facingMode=${get(constraints, 'video.facingMode')}`, err.name);\n\n // @ts-ignore\n delete newConstraints.video.facingMode;\n\n streamPromise = this.getStreamPromiseByParsedConstraints(newConstraints, isDesktopMedia);\n } else if (err && err.name && constraints.audio && !constraints.video) {\n trace.info('getStreamPromiseByParsedConstraints - audioStreamFailed - Failed to get audio user media stream', err.name);\n\n return this.parseMediaErrorThenThrow({name: 'audioStreamFailed'});\n } else if (err && err.name && constraints.audio) {\n const newConstraints = constraints;\n\n trace.info('getStreamPromiseByParsedConstraints - Failed to get video and audio user media stream, try to get video media stream only', err.name);\n\n newConstraints.audio = false;\n streamPromise = this.getStreamPromiseByParsedConstraints(newConstraints, isDesktopMedia);\n } else if (err && !streamPromise) {\n trace.info('getStreamPromiseByParsedConstraints - Failed to get video user media stream', err && err.name);\n\n streamPromise = Promise.reject(err);\n }\n\n return streamPromise && streamPromise.catch(this.parseMediaErrorThenThrow);\n });\n } catch (e) {\n this.parseMediaErrorThenThrow(e);\n }\n }\n\n setGroupedDevices(groupedDevices: {video: MediaDeviceInfo[]; audio: MediaDeviceInfo[]}) {\n trace.info('setGroupedDevices', groupedDevices);\n this._groupedDevices = {video: groupedDevices.video\n ? {back: last(groupedDevices.video), front: first(groupedDevices.video)}\n : null,\n audio: last(groupedDevices.audio) || null,\n camerasCount: groupedDevices.video.length};\n }\n\n private getParsedConstraints(isDesktopMedia: boolean, streamConstraints?: LocalMediaConstraints): MediaStreamConstraints[] {\n const constraintCandidates: MediaStreamConstraints[] = [];\n\n //TODO - Alex: need to understand which constraints not compatible with screen share\n if (isDesktopMedia) {\n return [{video: true}];\n }\n\n if (!streamConstraints) {\n return constraintCandidates;\n }\n\n const baseConstraints: MediaStreamConstraints = {};\n\n baseConstraints.audio = typeof streamConstraints.audio === 'boolean' ? streamConstraints.audio : undefined;\n\n if (!streamConstraints.video) {\n return [baseConstraints];\n }\n\n const cameraType = (streamConstraints.video !== 'boolean') &&\n (streamConstraints.video as LocalVideoStreamConstraints).videoSourceType;\n\n baseConstraints.video = {\n facingMode: cameraType && cameraType === LocalVideoSourceType.CAMERA_FRONT\n ? 'user'\n : 'environment',\n frameRate: {ideal: 15, max: 30}\n };\n\n const deviceId = get(this._groupedDevices, `video.${cameraType === LocalVideoSourceType.CAMERA_FRONT ? 'front' : 'back'}.deviceId`);\n\n if (deviceId) {\n baseConstraints.video.deviceId = {exact: deviceId};\n }\n\n const assignConstraint = (videoResolution: VideoStreamResolution): void => {\n const [width, height] = videoResolution.resolution.split('x');\n const constraintCandidate = assign({}, baseConstraints);\n\n constraintCandidate.video = assign({}, baseConstraints.video, {\n width: _constraintByType(width, videoResolution.constraintType),\n height: _constraintByType(height, videoResolution.constraintType)\n });\n constraintCandidates.push(constraintCandidate);\n };\n\n if ((baseConstraints.video && typeof streamConstraints.video === 'boolean') ||\n !((streamConstraints.video as LocalVideoStreamConstraints).videoResolution instanceof Array)) {\n\n const defaultResolution: VideoStreamResolution = {\n resolution: DEFAULT_VIDEO_RESOLUTION,\n constraintType: DEFAULT_VIDEO_CONSTRAINT_TYPE\n };\n\n assignConstraint(defaultResolution);\n\n return constraintCandidates;\n }\n\n (streamConstraints.video as any).videoResolution.forEach(assignConstraint);\n\n return constraintCandidates;\n }\n\n private parseMediaErrorThenThrow(error: any): MediaRequestFailResult {\n if (error && error.errorCode) {\n throw error;\n }\n\n trace.error('Get media stream error.', error);\n const errorName = error && error.name ? error.name : '';\n\n const permissionErrorNames = [\n 'NotAllowedError',\n 'PermissionDismissedError',\n 'PermissionDeniedError'\n\n ];\n\n // const generalError = [\n // 'NotFoundError',\n // 'DevicesNotFoundError',\n // 'NotReadableError',\n // 'TrackStartError'\n // ];\n\n const constraintsErrors = [\n 'OverconstrainedError',\n 'ConstraintNotSatisfiedError'\n\n ];\n\n const isErrorOfType = (errorsArr: string[]) => errorsArr.filter((err: string) => err === errorName).length > 0;\n\n if (isErrorOfType(permissionErrorNames)) {\n throw {errorCode: MediaRequestErrorCode.PermissionDenied, message: errorName};\n }\n\n if (isErrorOfType(constraintsErrors)) {\n throw {errorCode: MediaRequestErrorCode.Overconstrained, message: errorName};\n }\n\n throw {errorCode: MediaRequestErrorCode.GeneralError, message: errorName};\n }\n}\n\nfunction _constraintByType(value: any, type: ConstraintType): any {\n switch (type) {\n case ConstraintType.MIN:\n return {min: value};\n default:\n return {ideal: value};\n }\n}\n"]}
|
|
1
|
+
{"version":3,"sources":["LocalStreamManager.js","../src/LocalStreamManager.ts"],"names":["__assign","Object","assign","t","s","i","n","arguments","length","p","prototype","hasOwnProperty","call","apply","__importDefault","mod","__esModule","defineProperty","exports","value","assign_1","require","get_1","last_1","first_1","cloneDeep_1","bluebird_1","MediaConstants_1","MediaTracer_1","window","latestUserMediaStream","userMediaPromise","latestDesktopMediaStream","desktopMediaPromise","trace","getMediaTracer","LocalStreamManager","environment","_groupedDevices","video","audio","camerasCount","parseMediaErrorThenThrow","bind","clearAllStreams","get","enumerable","configurable","getUserMediaStream","constraints","_this","info","videoTrack","mediaStream","getVideoTracks","readyState","Promise","resolve","isNew","destroyUserMediaStream","reject","catchHandler","err","error","getHighestResolutionStream","then","streamResult","latestLocalMediaConstraints","default","constraint","catch","getDesktopMediaStream","track","destroyDesktopMediaStream","mediaCatchHandler","getTracks","forEach","stop","all","undefined","isDesktopMedia","constraintsCandidates","getParsedConstraints","Error","lastSuccessfulResult","lastFailedResult","isStreamRequestFulfilled","reduce","total","candidate","constraintType","getStreamPromiseByParsedConstraints","stream","failResult","errorCode","MediaRequestErrorCode","PermissionDenied","GeneralError","message","getUserMediaStreamMediaDevice","streamPromise","navigator","mediaDevices","getUserMedia","getDisplayMedia","name","newConstraints","facingMode","e","setGroupedDevices","groupedDevices","back","front","streamConstraints","constraintCandidates","baseConstraints","cameraType","videoSourceType","LocalVideoSourceType","CAMERA_FRONT","frameRate","ideal","max","deviceId","assignConstraint","videoResolution","_a","resolution","split","width","height","constraintCandidate","_constraintByType","push","Array","defaultResolution","DEFAULT_VIDEO_RESOLUTION","DEFAULT_VIDEO_CONSTRAINT_TYPE","errorName","permissionErrorNames","constraintsErrors","isErrorOfType","errorsArr","filter","Overconstrained","type","ConstraintType","MIN","min"],"mappings":"AAAA;;AACA,IAAIA,WAAY,aAAQ,UAAKA,QAAd,IAA2B,YAAY;AAClDA,eAAWC,OAAOC,MAAP,IAAiB,UAASC,CAAT,EAAY;AACpC,aAAK,IAAIC,CAAJ,EAAOC,IAAI,CAAX,EAAcC,IAAIC,UAAUC,MAAjC,EAAyCH,IAAIC,CAA7C,EAAgDD,GAAhD,EAAqD;AACjDD,gBAAIG,UAAUF,CAAV,CAAJ;AACA,iBAAK,IAAII,CAAT,IAAcL,CAAd;AAAiB,oBAAIH,OAAOS,SAAP,CAAiBC,cAAjB,CAAgCC,IAAhC,CAAqCR,CAArC,EAAwCK,CAAxC,CAAJ,EACbN,EAAEM,CAAF,IAAOL,EAAEK,CAAF,CAAP;AADJ;AAEH;AACD,eAAON,CAAP;AACH,KAPD;AAQA,WAAOH,SAASa,KAAT,CAAe,IAAf,EAAqBN,SAArB,CAAP;AACH,CAVD;AAWA,IAAIO,kBAAmB,aAAQ,UAAKA,eAAd,IAAkC,UAAUC,GAAV,EAAe;AACnE,WAAQA,OAAOA,IAAIC,UAAZ,GAA0BD,GAA1B,GAAgC,EAAE,WAAWA,GAAb,EAAvC;AACH,CAFD;AAGAd,OAAOgB,cAAP,CAAsBC,OAAtB,EAA+B,YAA/B,EAA6C,EAAEC,OAAO,IAAT,EAA7C;ACfA,IAAAC,WAAAN,gBAAAO,QAAA,eAAA,CAAA,CAAA;AACA,IAAAC,QAAAR,gBAAAO,QAAA,YAAA,CAAA,CAAA;AACA,IAAAE,SAAAT,gBAAAO,QAAA,aAAA,CAAA,CAAA;AACA,IAAAG,UAAAV,gBAAAO,QAAA,cAAA,CAAA,CAAA;AACA,IAAAI,cAAAX,gBAAAO,QAAA,kBAAA,CAAA,CAAA;AACA,IAAAK,aAAAL,QAAA,UAAA,CAAA;AAaA,IAAAM,mBAAAN,QAAA,kBAAA,CAAA;AAOA,IAAAO,gBAAAP,QAAA,0BAAA,CAAA;AAYAQ,OAAOC,qBAAP,GAA+B,IAA/B;AACA,IAAIC,mBAAiE,IAArE;AAEAF,OAAOG,wBAAP,GAAkC,IAAlC;AACA,IAAIC,sBAAoE,IAAxE;AAEA,IAAMC,QAAQN,cAAAO,cAAA,CAAe,oBAAf,CAAd;AAEA,IAAAC,qBAAA,aAAA,YAAA;AAKI,aAAAA,kBAAA,CAAYC,WAAZ,EAA0C;AAHlC,aAAAC,eAAA,GACuD,EAACC,OAAO,IAAR,EAAcC,OAAO,IAArB,EAA2BC,cAAc,CAAzC,EADvD;AAIJ,aAAKJ,WAAL,GAAmBA,WAAnB;AAEA,aAAKK,wBAAL,GAAgC,KAAKA,wBAAL,CAA8BC,IAA9B,CAAmC,IAAnC,CAAhC;AACA,aAAKC,eAAL,GAAuB,KAAKA,eAAL,CAAqBD,IAArB,CAA0B,IAA1B,CAAvB;AACH;AAED1C,WAAAgB,cAAA,CAAImB,mBAAA1B,SAAJ,EAAI,gBAAJ,EAAkB;ADpBdmC,aCoBJ,eAAA;AACI,mBAAO,KAAKP,eAAZ;AACH,SAFiB;ADjBdQ,oBAAY,ICiBE;ADhBdC,sBAAc;ACgBA,KAAlB;AAIAX,uBAAA1B,SAAA,CAAAsC,kBAAA,GAAA,UAAmBC,WAAnB,EAAqD;AAArD,YAAAC,QAAA,IAAA;AACIhB,cAAMiB,IAAN,CAAW,oBAAX,EAAiCF,WAAjC;AACA,YAAIlB,gBAAJ,EAAsB;AAClB,mBAAOA,gBAAP;AACH;AAED,YAAIF,OAAOC,qBAAX,EAAkC;AAC9B,gBAAMsB,aAAavB,OAAOC,qBAAP,CAA6BuB,WAA7B,CAAyCC,cAAzC,EAAnB;AAEA,gBAAIF,WAAW5C,MAAX,GAAoB,CAApB,IAAyB4C,WAAW,CAAX,EAAcG,UAAd,KAA6B,MAA1D,EAAkE;AAC9D,uBAAO7B,WAAA8B,OAAA,CAAQC,OAAR,CAAezD,SAAA,EAAA,EAAK6B,OAAOC,qBAAZ,EAAiC,EAAE4B,OAAO,KAAT,EAAjC,CAAf,CAAP;AACH;AAED,iBAAKC,sBAAL;AACH;AAED5B,2BAAmB,IAAIL,WAAA8B,OAAJ,CAAY,UAACC,OAAD,EAAUG,MAAV,EAAgB;AAC3C,gBAAMC,eAAe,SAAfA,YAAe,CAACC,GAAD,EAAS;AAC1B/B,mCAAmB,IAAnB;AACAF,uBAAOC,qBAAP,GAA+B,IAA/B;AACAI,sBAAM6B,KAAN,CAAY,oBAAZ,EAAkCD,GAAlC;AACAF,uBAAOE,GAAP;AACH,aALD;AAOA,gBAAI;AACAZ,sBAAKc,0BAAL,CAAgC,KAAhC,EAAuCf,WAAvC,EACKgB,IADL,CACU,UAACC,YAAD,EAAwC;AAC1CnC,uCAAmB,IAAnB;AACAF,2BAAOC,qBAAP,GAA+BoC,YAA/B;AACArC,2BAAOsC,2BAAP,GAAqC1C,YAAA2C,OAAA,CAAUnB,WAAV,CAArC;AAEA,wBAAIpB,OAAOsC,2BAAP,IAAsCtC,OAAOsC,2BAAP,CAAmC3B,KAA7E,EAAoF;AAChFX,+BAAOsC,2BAAP,CAAmC3B,KAAnC,GAA4C0B,aAAaG,UAAb,IAA2BH,aAAaG,UAAb,CAAwB7B,KAA/F;AACH;AAEDiB,4BAAOzD,SAAA,EAAA,EAAK6B,OAAOC,qBAAZ,EAAiC,EAAE4B,OAAO,IAAT,EAAjC,CAAP;AACH,iBAXL,EAYKY,KAZL,CAYWT,YAZX;AAaH,aAdD,CAcE,OAAOC,GAAP,EAAY;AACVD,6BAAaC,GAAb;AACH;AACJ,SAzBkB,CAAnB;AA2BA,eAAO/B,gBAAP;AACH,KA5CD;AA8CAK,uBAAA1B,SAAA,CAAA6D,qBAAA,GAAA,UAAsBtB,WAAtB,EAA8D;AAA9D,YAAAC,QAAA,IAAA;AACI,YAAIjB,mBAAJ,EAAyB;AACrB,mBAAOA,mBAAP;AACH;AAEDC,cAAMiB,IAAN,CAAW,uBAAX,EAAoCF,WAApC;AAEA,YAAIpB,OAAOG,wBAAX,EAAqC;AACjC,gBAAMwC,QAAQ3C,OAAOG,wBAAP,CAAgCqB,WAAhC,CAA4CC,cAA5C,EAAd;AAEA,gBAAIkB,MAAMhE,MAAN,GAAe,CAAf,IAAoBgE,MAAM,CAAN,EAASjB,UAAT,KAAwB,MAAhD,EAAwD;AACpD;AACA,uBAAO7B,WAAA8B,OAAA,CAAQC,OAAR,CAAezD,SAAA,EAAA,EAAK6B,OAAOG,wBAAZ,EAAoC,EAAE0B,OAAO,KAAT,EAApC,CAAf,CAAP;AACH;AACD,iBAAKe,yBAAL,GAAiCH,KAAjC,CAAuC,UAACP,KAAD,EAAW;AAC9C7B,sBAAM6B,KAAN,CAAY,uCAAZ,EAAqDA,KAArD;AACH,aAFD;AAIH;AAED9B,8BAAsB,IAAIP,WAAA8B,OAAJ,CAAY,UAACC,OAAD,EAAUG,MAAV,EAAgB;AAC9C,gBAAMc,oBAAoB,SAApBA,iBAAoB,CAACZ,GAAD,EAAS;AAC/B7B,sCAAsB,IAAtB;AACAJ,uBAAOG,wBAAP,GAAkC,IAAlC;AACAE,sBAAM6B,KAAN,CAAY,uBAAZ,EAAqCD,GAArC;AACAF,uBAAOE,GAAP;AACH,aALD;AAOA,gBAAI;AACAZ,sBAAKc,0BAAL,CAAgC,IAAhC,EAAsC,EAACzB,OAAOU,WAAR,EAAqBT,OAAO,KAA5B,EAAtC,EACKyB,IADL,CACU,UAACC,YAAD,EAAwC;AAC1CjC,0CAAsB,IAAtB;AACAJ,2BAAOG,wBAAP,GAAkCkC,YAAlC;AAEAT,4BAAOzD,SAAA,EAAA,EAAK6B,OAAOG,wBAAZ,EAAoC,EAAE0B,OAAO,IAAT,EAApC,CAAP;AACH,iBANL,EAMOY,KANP,CAMaI,iBANb;AAOH,aARD,CAQE,OAAOZ,GAAP,EAAY;AACVY,kCAAkBZ,GAAlB;AACH;AACJ,SAnBqB,CAAtB;AAqBA,eAAO7B,mBAAP;AACH,KA1CD;AA4CAG,uBAAA1B,SAAA,CAAAiD,sBAAA,GAAA,YAAA;AACI,YAAMO,eAAerC,OAAOC,qBAA5B;AAEAD,eAAOC,qBAAP,GAA+B,IAA/B;AACAD,eAAOsC,2BAAP,GAAqC,IAArC;AACApC,2BAAmB,IAAnB;AAEA,YAAImC,gBAAgBA,aAAab,WAAjC,EAA8C;AAC1Ca,yBAAab,WAAb,CAAyBsB,SAAzB,GAAqCC,OAArC,CAA6C,UAACJ,KAAD,EAAwB;AACjEA,sBAAMK,IAAN;AACH,aAFD;AAIA3C,kBAAMiB,IAAN,CAAW,0CAAX;AACH;AAED,eAAOzB,WAAA8B,OAAA,CAAQC,OAAR,EAAP;AACH,KAhBD;AAkBArB,uBAAA1B,SAAA,CAAA+D,yBAAA,GAAA,YAAA;AACI,YAAMP,eAAerC,OAAOG,wBAA5B;AAEAH,eAAOG,wBAAP,GAAkC,IAAlC;AACAC,8BAAsB,IAAtB;AAEA,YAAIiC,gBAAgBA,aAAab,WAAjC,EAA8C;AAC1Ca,yBAAab,WAAb,CAAyBsB,SAAzB,GAAqCC,OAArC,CAA6C,UAACJ,KAAD,EAAwB;AACjEA,sBAAMK,IAAN;AACH,aAFD;AAGH;AAED,eAAOnD,WAAA8B,OAAA,CAAQC,OAAR,EAAP;AACH,KAbD;AAeArB,uBAAA1B,SAAA,CAAAkC,eAAA,GAAA,YAAA;AACIV,cAAMiB,IAAN,CAAW,yCAAX;AAEA,eAAOzB,WAAA8B,OAAA,CAAQsB,GAAR,CAAY,CACf,KAAKL,yBAAL,EADe,EAEf,KAAKd,sBAAL,EAFe,CAAZ,EAGJM,IAHI,CAGC,YAAA;AAAM,mBAAAc,SAAA;AAAS,SAHhB,CAAP;AAIH,KAPD;AASQ3C,uBAAA1B,SAAA,CAAAsD,0BAAA,GAAR,UAAmCgB,cAAnC,EAA4D/B,WAA5D,EAA+F;AAA/F,YAAAC,QAAA,IAAA;AACI,YAAM+B,wBAAwB,KAAKC,oBAAL,CAA0BF,cAA1B,EAA0C/B,WAA1C,CAA9B;AAEAf,cAAMiB,IAAN,CAAW,qDAAX,EAAkE8B,qBAAlE;AAEA,YAAIA,sBAAsBzE,MAAtB,KAAiC,CAArC,EAAwC;AACpC,mBAAOkB,WAAA8B,OAAA,CAAQI,MAAR,CAAe,IAAIuB,KAAJ,CAAU,uCAAV,CAAf,CAAP;AACH;AAED,eAAO,IAAIzD,WAAA8B,OAAJ,CAAY,UAACC,OAAD,EAAUG,MAAV,EAAgB;AAC/B,gBAAIwB,uBAA4D,IAAhE;AACA,gBAAIC,mBAAqD,IAAzD;AACA,gBAAIC,2BAA2B,KAA/B;AAEC5D,uBAAA8B,OAAA,CAAgB+B,MAAhB,CAAuBN,qBAAvB,EAA8C,UAACO,KAAD,EAAaC,SAAb,EAA8C;AACzF,oBAAIH,wBAAJ,EAA8B;AAC1B;AACH;AAED,oBAAMI,iBAAiBpE,MAAA8C,OAAA,CAAInB,WAAJ,EAAiB,sCAAjB,KAA4D3B,MAAA8C,OAAA,CAAInB,WAAJ,EAAiB,yCAAjB,CAAnF;AAEA,uBAAOC,MAAKyC,mCAAL,CAAyCF,SAAzC,EAAoDT,cAApD,EACFf,IADE,CACG,UAAC2B,MAAD,EAAoB;AACtBR,2CAAuB;AACnB/B,qCAAauC,MADM;AAEnBvB,oCAAYoB,SAFO;AAGnBC,wCAAcA,cAHK;AAInBhC,+BAAO;AAJY,qBAAvB;AAMAxB,0BAAMiB,IAAN,CAAW,oCAAX,EAAiDiC,oBAAjD;AACAE,+CAA2B,IAA3B;AACA7B,4BAAQ2B,oBAAR;AACH,iBAXE,EAYFd,KAZE,CAYI,UAACuB,UAAD,EAAmC;AACtCR,uCAAmBQ,UAAnB;AAEA,wBAAIR,iBAAiBS,SAAjB,KAA+BnE,iBAAAoE,qBAAA,CAAsBC,gBAAzD,EAA2E;AACvEV,mDAA2B,IAA3B;AACA1B,+BAAOyB,gBAAP;AACH;AACJ,iBAnBE,CAAP;AAoBH,aA3BA,EA2BE,CA3BF,EA4BIpB,IA5BJ,CA4BS,YAAA;AACF,oBAAI,CAACqB,wBAAL,EAA+B;AAC3B,wBAAI,CAACD,gBAAL,EAAuB;AACnBA,2CAAmB;AACfS,uCAAWnE,iBAAAoE,qBAAA,CAAsBE,YADlB;AAEfC,qCAAS;AAFM,yBAAnB;AAIH;AACDtC,2BAAOyB,gBAAP;AACH;AACJ,aAtCJ;AAuCJ,SA5CM,CAAP;AA6CH,KAtDO;AAwDAjD,uBAAA1B,SAAA,CAAAyF,6BAAA,GAAR,UAAsClD,WAAtC,EAA2E+B,cAA3E,EAAkG;AAC9F,YAAIoB,gBAAgB,IAApB;AAEA,YAAI,CAACpB,cAAL,EAAqB;AACjBoB,4BAAgBC,UAAUC,YAAV,CAAuBC,YAAvB,CAAoCtD,WAApC,CAAhB;AACH,SAFD,MAEO;AACH;AACAmD,4BAAgBC,UAAUC,YAAV,CAAuBE;AACnC;AADY,cAEVH,UAAUC,YAAV,CAAuBE,eAAvB,CAAuCvD,WAAvC,CAFU,GAGVoD,UAAUG,eAAV,CAA0BvD,WAA1B,CAHN;AAIH;AAED,eAAOmD,aAAP;AACH,KAdO;AAgBR;AACQhE,uBAAA1B,SAAA,CAAAiF,mCAAA,GAAR,UAA4C1C,WAA5C,EAAiF+B,cAAjF,EAAwG;AAAxG,YAAA9B,QAAA,IAAA;AACIhB,cAAMiB,IAAN,CAAW,qCAAX,EAAkDF,WAAlD;AAEA,YAAI;AACA,mBAAO,KAAKkD,6BAAL,CAAmClD,WAAnC,EAAgD+B,cAAhD,EAAgEV,KAAhE,CAAsE,UAACR,GAAD,EAAS;AAClF,oBAAIsC,gBAAgB,IAApB;AAEAlE,sBAAMiB,IAAN,CAAW,6DAAX,EAA0EW,IAAI2C,IAA9E;AAEA;AACA;AACA;AACA;AACA,oBAAI3C,OAAQA,IAAI2C,IAAJ,KAAa,kBAArB,IAA4CnF,MAAA8C,OAAA,CAAInB,WAAJ,EAAiB,kBAAjB,CAA5C,IACA3B,MAAA8C,OAAA,CAAInB,WAAJ,EAAiB,gBAAjB,CADJ,EACwC;AAEpC,wBAAMyD,iBAAiBzD,WAAvB;AAEAf,0BAAMiB,IAAN,CAAW,oHAAkH7B,MAAA8C,OAAA,CAAInB,WAAJ,EAAiB,kBAAjB,CAA7H,EAAqKa,IAAI2C,IAAzK;AAEA;AACA,2BAAOC,eAAenE,KAAf,CAAqBoE,UAA5B;AAEAP,oCAAgBlD,MAAKyC,mCAAL,CAAyCe,cAAzC,EAAyD1B,cAAzD,CAAhB;AACH,iBAXD,MAWO,IAAIlB,OAAOA,IAAI2C,IAAX,IAAmBxD,YAAYT,KAA/B,IAAwC,CAACS,YAAYV,KAAzD,EAAgE;AACnEL,0BAAMiB,IAAN,CAAW,iGAAX,EAA8GW,IAAI2C,IAAlH;AAEA,2BAAOvD,MAAKR,wBAAL,CAA8B,EAAC+D,MAAM,mBAAP,EAA9B,CAAP;AACH,iBAJM,MAIA,IAAI3C,OAAOA,IAAI2C,IAAX,IAAmBxD,YAAYT,KAAnC,EAA0C;AAC7C,wBAAMkE,iBAAiBzD,WAAvB;AAEAf,0BAAMiB,IAAN,CAAW,2HAAX,EAAwIW,IAAI2C,IAA5I;AAEAC,mCAAelE,KAAf,GAAuB,KAAvB;AACA4D,oCAAgBlD,MAAKyC,mCAAL,CAAyCe,cAAzC,EAAyD1B,cAAzD,CAAhB;AACH,iBAPM,MAOA,IAAIlB,OAAO,CAACsC,aAAZ,EAA2B;AAC9BlE,0BAAMiB,IAAN,CAAW,6EAAX,EAA0FW,OAAOA,IAAI2C,IAArG;AAEAL,oCAAgB1E,WAAA8B,OAAA,CAAQI,MAAR,CAAeE,GAAf,CAAhB;AACH;AAED,uBAAOsC,iBAAiBA,cAAc9B,KAAd,CAAoBpB,MAAKR,wBAAzB,CAAxB;AACH,aAtCM,CAAP;AAuCH,SAxCD,CAwCE,OAAOkE,CAAP,EAAU;AACR,iBAAKlE,wBAAL,CAA8BkE,CAA9B;AACH;AACJ,KA9CO;AAgDRxE,uBAAA1B,SAAA,CAAAmG,iBAAA,GAAA,UAAkBC,cAAlB,EAAsF;AAClF5E,cAAMiB,IAAN,CAAW,mBAAX,EAAgC2D,cAAhC;AACA,aAAKxE,eAAL,GAAuB,EAACC,OAAOuE,eAAevE,KAAf,GACzB,EAACwE,MAAMxF,OAAA6C,OAAA,CAAK0C,eAAevE,KAApB,CAAP,EAAmCyE,OAAOxF,QAAA4C,OAAA,CAAM0C,eAAevE,KAArB,CAA1C,EADyB,GAEzB,IAFiB;AAGvBC,mBAAOjB,OAAA6C,OAAA,CAAK0C,eAAetE,KAApB,KAA8B,IAHd;AAIvBC,0BAAcqE,eAAevE,KAAf,CAAqB/B,MAJZ,EAAvB;AAKH,KAPD;AASQ4B,uBAAA1B,SAAA,CAAAwE,oBAAA,GAAR,UAA6BF,cAA7B,EAAsDiC,iBAAtD,EAA+F;AAC3F,YAAMC,uBAAiD,EAAvD;AAEA;AACA,YAAIlC,cAAJ,EAAoB;AAChB,mBAAO,CAAC,EAACzC,OAAO,IAAR,EAAD,CAAP;AACH;AAED,YAAI,CAAC0E,iBAAL,EAAwB;AACpB,mBAAOC,oBAAP;AACH;AAED,YAAMC,kBAA0C,EAAhD;AAEAA,wBAAgB3E,KAAhB,GAAwB,OAAOyE,kBAAkBzE,KAAzB,KAAmC,SAAnC,GAA+CyE,kBAAkBzE,KAAjE,GAAyEuC,SAAjG;AAEA,YAAI,CAACkC,kBAAkB1E,KAAvB,EAA8B;AAC1B,mBAAO,CAAC4E,eAAD,CAAP;AACH;AAED,YAAMC,aAAcH,kBAAkB1E,KAAlB,KAA4B,SAA7B,IACd0E,kBAAkB1E,KAAlB,CAAwD8E,eAD7D;AAGAF,wBAAgB5E,KAAhB,GAAwB;AACpBoE,wBAAYS,cAAcA,eAAezF,iBAAA2F,oBAAA,CAAqBC,YAAlD,GACN,MADM,GAEN,aAHc;AAIpBC,uBAAW,EAACC,OAAO,EAAR,EAAYC,KAAK,EAAjB;AAJS,SAAxB;AAOA,YAAMC,WAAWrG,MAAA8C,OAAA,CAAI,KAAK9B,eAAT,EAA0B,YAAS8E,eAAezF,iBAAA2F,oBAAA,CAAqBC,YAApC,GAAmD,OAAnD,GAA6D,MAAtE,IAA4E,WAAtG,CAAjB;AAEA,YAAII,QAAJ,EAAc;AACVR,4BAAgB5E,KAAhB,CAAsBoF,QAAtB,GAAiCA,QAAjC;AACH;AAED,YAAMC,mBAAmB,SAAnBA,gBAAmB,CAACC,eAAD,EAAuC;AACtD,gBAAAC,KAAAD,gBAAAE,UAAA,CAAAC,KAAA,CAAA,GAAA,CAAA;AAAA,gBAACC,QAAAH,GAAA,CAAA,CAAD;AAAA,gBAAQI,SAAAJ,GAAA,CAAA,CAAR;AACN,gBAAMK,sBAAsB/G,SAAAgD,OAAA,CAAO,EAAP,EAAW+C,eAAX,CAA5B;AAEAgB,gCAAoB5F,KAApB,GAA4BnB,SAAAgD,OAAA,CAAO,EAAP,EAAW+C,gBAAgB5E,KAA3B,EAAkC;AAC1D0F,uBAAOG,kBAAkBH,KAAlB,EAAyBJ,gBAAgBnC,cAAzC,CADmD;AAE1DwC,wBAAQE,kBAAkBF,MAAlB,EAA0BL,gBAAgBnC,cAA1C;AAFkD,aAAlC,CAA5B;AAIAwB,iCAAqBmB,IAArB,CAA0BF,mBAA1B;AACH,SATD;AAWA,YAAKhB,gBAAgB5E,KAAhB,IAAyB,OAAO0E,kBAAkB1E,KAAzB,KAAmC,SAA7D,IACA,EAAG0E,kBAAkB1E,KAAlB,CAAwDsF,eAAxD,YAAmFS,KAAtF,CADJ,EACkG;AAE9F,gBAAMC,oBAA2C;AAC7CR,4BAAYpG,iBAAA6G,wBADiC;AAE7C9C,gCAAgB/D,iBAAA8G;AAF6B,aAAjD;AAKAb,6BAAiBW,iBAAjB;AAEA,mBAAOrB,oBAAP;AACH;AAEAD,0BAAkB1E,KAAlB,CAAgCsF,eAAhC,CAAgDjD,OAAhD,CAAwDgD,gBAAxD;AAED,eAAOV,oBAAP;AACH,KA/DO;AAiEA9E,uBAAA1B,SAAA,CAAAgC,wBAAA,GAAR,UAAiCqB,KAAjC,EAA2C;AACvC,YAAIA,SAASA,MAAM+B,SAAnB,EAA8B;AAC1B,kBAAM/B,KAAN;AACH;AAED7B,cAAM6B,KAAN,CAAY,yBAAZ,EAAuCA,KAAvC;AACA,YAAM2E,YAAY3E,SAASA,MAAM0C,IAAf,GAAsB1C,MAAM0C,IAA5B,GAAmC,EAArD;AAEA,YAAMkC,uBAAuB,CACzB,iBADyB,EAEzB,0BAFyB,EAGzB,uBAHyB,CAA7B;AAOA;AACA;AACA;AACA;AACA;AACA;AAEA,YAAMC,oBAAoB,CACtB,sBADsB,EAEtB,6BAFsB,CAA1B;AAMA,YAAMC,gBAAgB,SAAhBA,aAAgB,CAACC,SAAD,EAAoB;AAAK,mBAAAA,UAAUC,MAAV,CAAiB,UAACjF,GAAD,EAAY;AAAK,uBAAAA,QAAQ4E,SAAR;AAAiB,aAAnD,EAAqDlI,MAArD,GAA8D,CAA9D;AAA+D,SAA9G;AAEA,YAAIqI,cAAcF,oBAAd,CAAJ,EAAyC;AACrC,kBAAM,EAAC7C,WAAWnE,iBAAAoE,qBAAA,CAAsBC,gBAAlC,EAAoDE,SAASwC,SAA7D,EAAN;AACH;AAED,YAAIG,cAAcD,iBAAd,CAAJ,EAAsC;AAClC,kBAAM,EAAC9C,WAAWnE,iBAAAoE,qBAAA,CAAsBiD,eAAlC,EAAmD9C,SAASwC,SAA5D,EAAN;AACH;AAED,cAAM,EAAC5C,WAAWnE,iBAAAoE,qBAAA,CAAsBE,YAAlC,EAAgDC,SAASwC,SAAzD,EAAN;AACH,KAvCO;AAwCZ,WAAAtG,kBAAA;AA/XA,CAAA,EAAA;AAAalB,QAAAkB,kBAAA,GAAAA,kBAAA;AAiYb,SAASgG,iBAAT,CAA2BjH,KAA3B,EAAuC8H,IAAvC,EAA2D;AACvD,YAAQA,IAAR;AACI,aAAKtH,iBAAAuH,cAAA,CAAeC,GAApB;AACI,mBAAO,EAACC,KAAKjI,KAAN,EAAP;AACJ;AACI,mBAAO,EAACsG,OAAOtG,KAAR,EAAP;AAJR;AAMH;;ADtFD","file":"LocalStreamManager.js","sourcesContent":["\"use strict\";\nvar __assign = (this && this.__assign) || function () {\n __assign = Object.assign || function(t) {\n for (var s, i = 1, n = arguments.length; i < n; i++) {\n s = arguments[i];\n for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p))\n t[p] = s[p];\n }\n return t;\n };\n return __assign.apply(this, arguments);\n};\nvar __importDefault = (this && this.__importDefault) || function (mod) {\n return (mod && mod.__esModule) ? mod : { \"default\": mod };\n};\nObject.defineProperty(exports, \"__esModule\", { value: true });\nvar assign_1 = __importDefault(require(\"lodash/assign\"));\nvar get_1 = __importDefault(require(\"lodash/get\"));\nvar last_1 = __importDefault(require(\"lodash/last\"));\nvar first_1 = __importDefault(require(\"lodash/first\"));\nvar cloneDeep_1 = __importDefault(require(\"lodash/cloneDeep\"));\nvar bluebird_1 = require(\"bluebird\");\nvar MediaConstants_1 = require(\"./MediaConstants\");\nvar MediaTracer_1 = require(\"./MediaUtils/MediaTracer\");\nwindow.latestUserMediaStream = null;\nvar userMediaPromise = null;\nwindow.latestDesktopMediaStream = null;\nvar desktopMediaPromise = null;\nvar trace = MediaTracer_1.getMediaTracer('LocalStreamManager');\nvar LocalStreamManager = /** @class */ (function () {\n function LocalStreamManager(environment) {\n this._groupedDevices = { video: null, audio: null, camerasCount: 0 };\n this.environment = environment;\n this.parseMediaErrorThenThrow = this.parseMediaErrorThenThrow.bind(this);\n this.clearAllStreams = this.clearAllStreams.bind(this);\n }\n Object.defineProperty(LocalStreamManager.prototype, \"groupedDevices\", {\n get: function () {\n return this._groupedDevices;\n },\n enumerable: true,\n configurable: true\n });\n LocalStreamManager.prototype.getUserMediaStream = function (constraints) {\n var _this = this;\n trace.info('getUserMediaStream', constraints);\n if (userMediaPromise) {\n return userMediaPromise;\n }\n if (window.latestUserMediaStream) {\n var videoTrack = window.latestUserMediaStream.mediaStream.getVideoTracks();\n if (videoTrack.length > 0 && videoTrack[0].readyState === 'live') {\n return bluebird_1.Promise.resolve(__assign({}, window.latestUserMediaStream, { isNew: false }));\n }\n this.destroyUserMediaStream();\n }\n userMediaPromise = new bluebird_1.Promise(function (resolve, reject) {\n var catchHandler = function (err) {\n userMediaPromise = null;\n window.latestUserMediaStream = null;\n trace.error('getUserMedia error', err);\n reject(err);\n };\n try {\n _this.getHighestResolutionStream(false, constraints)\n .then(function (streamResult) {\n userMediaPromise = null;\n window.latestUserMediaStream = streamResult;\n window.latestLocalMediaConstraints = cloneDeep_1.default(constraints);\n if (window.latestLocalMediaConstraints && window.latestLocalMediaConstraints.audio) {\n window.latestLocalMediaConstraints.audio = (streamResult.constraint && streamResult.constraint.audio);\n }\n resolve(__assign({}, window.latestUserMediaStream, { isNew: true }));\n })\n .catch(catchHandler);\n }\n catch (err) {\n catchHandler(err);\n }\n });\n return userMediaPromise;\n };\n LocalStreamManager.prototype.getDesktopMediaStream = function (constraints) {\n var _this = this;\n if (desktopMediaPromise) {\n return desktopMediaPromise;\n }\n trace.info('getDesktopMediaStream', constraints);\n if (window.latestDesktopMediaStream) {\n var track = window.latestDesktopMediaStream.mediaStream.getVideoTracks();\n if (track.length > 0 && track[0].readyState === 'live') {\n //return Promise.resolve();\n return bluebird_1.Promise.resolve(__assign({}, window.latestDesktopMediaStream, { isNew: false }));\n }\n this.destroyDesktopMediaStream().catch(function (error) {\n trace.error('Error destroying desktop media stream', error);\n });\n }\n desktopMediaPromise = new bluebird_1.Promise(function (resolve, reject) {\n var mediaCatchHandler = function (err) {\n desktopMediaPromise = null;\n window.latestDesktopMediaStream = null;\n trace.error('getDisplayMedia error', err);\n reject(err);\n };\n try {\n _this.getHighestResolutionStream(true, { video: constraints, audio: false })\n .then(function (streamResult) {\n desktopMediaPromise = null;\n window.latestDesktopMediaStream = streamResult;\n resolve(__assign({}, window.latestDesktopMediaStream, { isNew: true }));\n }).catch(mediaCatchHandler);\n }\n catch (err) {\n mediaCatchHandler(err);\n }\n });\n return desktopMediaPromise;\n };\n LocalStreamManager.prototype.destroyUserMediaStream = function () {\n var streamResult = window.latestUserMediaStream;\n window.latestUserMediaStream = null;\n window.latestLocalMediaConstraints = null;\n userMediaPromise = null;\n if (streamResult && streamResult.mediaStream) {\n streamResult.mediaStream.getTracks().forEach(function (track) {\n track.stop();\n });\n trace.info('destroyUserMediaStream - stop all tracks');\n }\n return bluebird_1.Promise.resolve();\n };\n LocalStreamManager.prototype.destroyDesktopMediaStream = function () {\n var streamResult = window.latestDesktopMediaStream;\n window.latestDesktopMediaStream = null;\n desktopMediaPromise = null;\n if (streamResult && streamResult.mediaStream) {\n streamResult.mediaStream.getTracks().forEach(function (track) {\n track.stop();\n });\n }\n return bluebird_1.Promise.resolve();\n };\n LocalStreamManager.prototype.clearAllStreams = function () {\n trace.info('LocalStreamManager Clearing all streams');\n return bluebird_1.Promise.all([\n this.destroyDesktopMediaStream(),\n this.destroyUserMediaStream()\n ]).then(function () { return undefined; });\n };\n LocalStreamManager.prototype.getHighestResolutionStream = function (isDesktopMedia, constraints) {\n var _this = this;\n var constraintsCandidates = this.getParsedConstraints(isDesktopMedia, constraints);\n trace.info('getHighestResolutionStream - constraintsCandidates:', constraintsCandidates);\n if (constraintsCandidates.length === 0) {\n return bluebird_1.Promise.reject(new Error('No constraints to retrieve the stream'));\n }\n return new bluebird_1.Promise(function (resolve, reject) {\n var lastSuccessfulResult = null;\n var lastFailedResult = null;\n var isStreamRequestFulfilled = false;\n bluebird_1.Promise.reduce(constraintsCandidates, function (total, candidate) {\n if (isStreamRequestFulfilled) {\n return;\n }\n var constraintType = get_1.default(constraints, 'video.videoResolution.constraintType') || get_1.default(constraints, 'video.videoResolution[0].constraintType');\n return _this.getStreamPromiseByParsedConstraints(candidate, isDesktopMedia)\n .then(function (stream) {\n lastSuccessfulResult = {\n mediaStream: stream,\n constraint: candidate,\n constraintType: constraintType,\n isNew: true\n };\n trace.info('getHighestResolutionStream Success', lastSuccessfulResult);\n isStreamRequestFulfilled = true;\n resolve(lastSuccessfulResult);\n })\n .catch(function (failResult) {\n lastFailedResult = failResult;\n if (lastFailedResult.errorCode === MediaConstants_1.MediaRequestErrorCode.PermissionDenied) {\n isStreamRequestFulfilled = true;\n reject(lastFailedResult);\n }\n });\n }, 0)\n .then(function () {\n if (!isStreamRequestFulfilled) {\n if (!lastFailedResult) {\n lastFailedResult = {\n errorCode: MediaConstants_1.MediaRequestErrorCode.GeneralError,\n message: 'Suitable stream cannot be created'\n };\n }\n reject(lastFailedResult);\n }\n });\n });\n };\n LocalStreamManager.prototype.getUserMediaStreamMediaDevice = function (constraints, isDesktopMedia) {\n var streamPromise = null;\n if (!isDesktopMedia) {\n streamPromise = navigator.mediaDevices.getUserMedia(constraints);\n }\n else {\n //@ts-ignore\n streamPromise = navigator.mediaDevices.getDisplayMedia\n //@ts-ignore\n ? navigator.mediaDevices.getDisplayMedia(constraints)\n : navigator.getDisplayMedia(constraints);\n }\n return streamPromise;\n };\n //@ts-ignore\n LocalStreamManager.prototype.getStreamPromiseByParsedConstraints = function (constraints, isDesktopMedia) {\n var _this = this;\n trace.info('getStreamPromiseByParsedConstraints', constraints);\n try {\n return this.getUserMediaStreamMediaDevice(constraints, isDesktopMedia).catch(function (err) {\n var streamPromise = null;\n trace.info('getStreamPromiseByParsedConstraints - get user stream error', err.name);\n // Android (Galaxy esp.) have a bug that when using facingMode it might cause NotReadableError. So\n // we should try again but without facingMode. The exception to the rule is when we don't have specific\n // device to use (like in IOS13 bug, as can be seen in MediaServiceBase.ts), in that case, we might end\n // up using the front camera, which is a privacy concern and the preference is to fallback from video\n if (err && (err.name === 'NotReadableError') && get_1.default(constraints, 'video.facingMode') &&\n get_1.default(constraints, 'video.deviceId')) {\n var newConstraints = constraints;\n trace.info(\"getStreamPromiseByParsedConstraints - NotReadableError - Failed to get video user media stream with facingMode=\" + get_1.default(constraints, 'video.facingMode'), err.name);\n // @ts-ignore\n delete newConstraints.video.facingMode;\n streamPromise = _this.getStreamPromiseByParsedConstraints(newConstraints, isDesktopMedia);\n }\n else if (err && err.name && constraints.audio && !constraints.video) {\n trace.info('getStreamPromiseByParsedConstraints - audioStreamFailed - Failed to get audio user media stream', err.name);\n return _this.parseMediaErrorThenThrow({ name: 'audioStreamFailed' });\n }\n else if (err && err.name && constraints.audio) {\n var newConstraints = constraints;\n trace.info('getStreamPromiseByParsedConstraints - Failed to get video and audio user media stream, try to get video media stream only', err.name);\n newConstraints.audio = false;\n streamPromise = _this.getStreamPromiseByParsedConstraints(newConstraints, isDesktopMedia);\n }\n else if (err && !streamPromise) {\n trace.info('getStreamPromiseByParsedConstraints - Failed to get video user media stream', err && err.name);\n streamPromise = bluebird_1.Promise.reject(err);\n }\n return streamPromise && streamPromise.catch(_this.parseMediaErrorThenThrow);\n });\n }\n catch (e) {\n this.parseMediaErrorThenThrow(e);\n }\n };\n LocalStreamManager.prototype.setGroupedDevices = function (groupedDevices) {\n trace.info('setGroupedDevices', groupedDevices);\n this._groupedDevices = { video: groupedDevices.video\n ? { back: last_1.default(groupedDevices.video), front: first_1.default(groupedDevices.video) }\n : null,\n audio: last_1.default(groupedDevices.audio) || null,\n camerasCount: groupedDevices.video.length };\n };\n LocalStreamManager.prototype.getParsedConstraints = function (isDesktopMedia, streamConstraints) {\n var constraintCandidates = [];\n //TODO - Alex: need to understand which constraints not compatible with screen share\n if (isDesktopMedia) {\n return [{ video: true }];\n }\n if (!streamConstraints) {\n return constraintCandidates;\n }\n var baseConstraints = {};\n baseConstraints.audio = typeof streamConstraints.audio === 'boolean' ? streamConstraints.audio : undefined;\n if (!streamConstraints.video) {\n return [baseConstraints];\n }\n var cameraType = (streamConstraints.video !== 'boolean') &&\n streamConstraints.video.videoSourceType;\n baseConstraints.video = {\n facingMode: cameraType && cameraType === MediaConstants_1.LocalVideoSourceType.CAMERA_FRONT\n ? 'user'\n : 'environment',\n frameRate: { ideal: 15, max: 30 }\n };\n var deviceId = get_1.default(this._groupedDevices, \"video.\" + (cameraType === MediaConstants_1.LocalVideoSourceType.CAMERA_FRONT ? 'front' : 'back') + \".deviceId\");\n if (deviceId) {\n baseConstraints.video.deviceId = deviceId;\n }\n var assignConstraint = function (videoResolution) {\n var _a = videoResolution.resolution.split('x'), width = _a[0], height = _a[1];\n var constraintCandidate = assign_1.default({}, baseConstraints);\n constraintCandidate.video = assign_1.default({}, baseConstraints.video, {\n width: _constraintByType(width, videoResolution.constraintType),\n height: _constraintByType(height, videoResolution.constraintType)\n });\n constraintCandidates.push(constraintCandidate);\n };\n if ((baseConstraints.video && typeof streamConstraints.video === 'boolean') ||\n !(streamConstraints.video.videoResolution instanceof Array)) {\n var defaultResolution = {\n resolution: MediaConstants_1.DEFAULT_VIDEO_RESOLUTION,\n constraintType: MediaConstants_1.DEFAULT_VIDEO_CONSTRAINT_TYPE\n };\n assignConstraint(defaultResolution);\n return constraintCandidates;\n }\n streamConstraints.video.videoResolution.forEach(assignConstraint);\n return constraintCandidates;\n };\n LocalStreamManager.prototype.parseMediaErrorThenThrow = function (error) {\n if (error && error.errorCode) {\n throw error;\n }\n trace.error('Get media stream error.', error);\n var errorName = error && error.name ? error.name : '';\n var permissionErrorNames = [\n 'NotAllowedError',\n 'PermissionDismissedError',\n 'PermissionDeniedError'\n ];\n // const generalError = [\n // 'NotFoundError',\n // 'DevicesNotFoundError',\n // 'NotReadableError',\n // 'TrackStartError'\n // ];\n var constraintsErrors = [\n 'OverconstrainedError',\n 'ConstraintNotSatisfiedError'\n ];\n var isErrorOfType = function (errorsArr) { return errorsArr.filter(function (err) { return err === errorName; }).length > 0; };\n if (isErrorOfType(permissionErrorNames)) {\n throw { errorCode: MediaConstants_1.MediaRequestErrorCode.PermissionDenied, message: errorName };\n }\n if (isErrorOfType(constraintsErrors)) {\n throw { errorCode: MediaConstants_1.MediaRequestErrorCode.Overconstrained, message: errorName };\n }\n throw { errorCode: MediaConstants_1.MediaRequestErrorCode.GeneralError, message: errorName };\n };\n return LocalStreamManager;\n}());\nexports.LocalStreamManager = LocalStreamManager;\nfunction _constraintByType(value, type) {\n switch (type) {\n case MediaConstants_1.ConstraintType.MIN:\n return { min: value };\n default:\n return { ideal: value };\n }\n}\n\n//# sourceMappingURL=LocalStreamManager.js.map\n","import assign from 'lodash/assign';\nimport get from 'lodash/get';\nimport last from 'lodash/last';\nimport first from 'lodash/first';\nimport cloneDeep from 'lodash/cloneDeep';\nimport {Promise} from 'bluebird';\n\nimport {Nullable} from '@techsee/techsee-common';\n\nimport {\n IMediaEnvironment,\n LocalMediaConstraints,\n LocalVideoStreamConstraints,\n MediaRequestFailResult,\n MediaRequestSuccessResult,\n VideoStreamResolution\n} from './MediaContracts';\n\nimport {\n ConstraintType,\n DEFAULT_VIDEO_CONSTRAINT_TYPE,\n DEFAULT_VIDEO_RESOLUTION,\n LocalVideoSourceType,\n MediaRequestErrorCode\n} from './MediaConstants';\nimport {getMediaTracer} from './MediaUtils/MediaTracer';\n\ndeclare global {\n interface Window {\n latestUserMediaStream: Nullable<MediaRequestSuccessResult>;\n latestDesktopMediaStream: Nullable<MediaRequestSuccessResult>;\n latestLocalMediaConstraints: Nullable<LocalMediaConstraints>;\n }\n}\n\ndeclare const window: Window;\n\nwindow.latestUserMediaStream = null;\nlet userMediaPromise: Nullable<Promise<MediaRequestSuccessResult>> = null;\n\nwindow.latestDesktopMediaStream = null;\nlet desktopMediaPromise: Nullable<Promise<MediaRequestSuccessResult>> = null;\n\nconst trace = getMediaTracer('LocalStreamManager');\n\nexport class LocalStreamManager {\n private environment: IMediaEnvironment;\n private _groupedDevices: {video: Nullable<{back?: MediaDeviceInfo; front?: MediaDeviceInfo}>;\n audio: Nullable<MediaDeviceInfo>; camerasCount: number;} = {video: null, audio: null, camerasCount: 0};\n\n constructor(environment: IMediaEnvironment) {\n this.environment = environment;\n\n this.parseMediaErrorThenThrow = this.parseMediaErrorThenThrow.bind(this);\n this.clearAllStreams = this.clearAllStreams.bind(this);\n }\n\n get groupedDevices() {\n return this._groupedDevices;\n }\n\n getUserMediaStream(constraints: LocalMediaConstraints): Promise<MediaRequestSuccessResult> {\n trace.info('getUserMediaStream', constraints);\n if (userMediaPromise) {\n return userMediaPromise;\n }\n\n if (window.latestUserMediaStream) {\n const videoTrack = window.latestUserMediaStream.mediaStream.getVideoTracks();\n\n if (videoTrack.length > 0 && videoTrack[0].readyState === 'live') {\n return Promise.resolve({...window.latestUserMediaStream, isNew: false});\n }\n\n this.destroyUserMediaStream();\n }\n\n userMediaPromise = new Promise((resolve, reject) => {\n const catchHandler = (err: any): void => {\n userMediaPromise = null;\n window.latestUserMediaStream = null;\n trace.error('getUserMedia error', err);\n reject(err);\n };\n\n try {\n this.getHighestResolutionStream(false, constraints)\n .then((streamResult: MediaRequestSuccessResult) => {\n userMediaPromise = null;\n window.latestUserMediaStream = streamResult;\n window.latestLocalMediaConstraints = cloneDeep(constraints);\n\n if (window.latestLocalMediaConstraints && window.latestLocalMediaConstraints.audio) {\n window.latestLocalMediaConstraints.audio = (streamResult.constraint && streamResult.constraint.audio) as boolean;\n }\n\n resolve({...window.latestUserMediaStream, isNew: true});\n })\n .catch(catchHandler);\n } catch (err) {\n catchHandler(err);\n }\n });\n\n return userMediaPromise;\n }\n\n getDesktopMediaStream(constraints: LocalVideoStreamConstraints): Promise<MediaRequestSuccessResult> {\n if (desktopMediaPromise) {\n return desktopMediaPromise;\n }\n\n trace.info('getDesktopMediaStream', constraints);\n\n if (window.latestDesktopMediaStream) {\n const track = window.latestDesktopMediaStream.mediaStream.getVideoTracks();\n\n if (track.length > 0 && track[0].readyState === 'live') {\n //return Promise.resolve();\n return Promise.resolve({...window.latestDesktopMediaStream, isNew: false});\n }\n this.destroyDesktopMediaStream().catch((error: any) => {\n trace.error('Error destroying desktop media stream', error);\n });\n\n }\n\n desktopMediaPromise = new Promise((resolve, reject) => {\n const mediaCatchHandler = (err: any): void => {\n desktopMediaPromise = null;\n window.latestDesktopMediaStream = null;\n trace.error('getDisplayMedia error', err);\n reject(err);\n };\n\n try {\n this.getHighestResolutionStream(true, {video: constraints, audio: false})\n .then((streamResult: MediaRequestSuccessResult) => {\n desktopMediaPromise = null;\n window.latestDesktopMediaStream = streamResult;\n\n resolve({...window.latestDesktopMediaStream, isNew: true});\n }).catch(mediaCatchHandler);\n } catch (err) {\n mediaCatchHandler(err);\n }\n });\n\n return desktopMediaPromise;\n }\n\n destroyUserMediaStream(): Promise<void> {\n const streamResult = window.latestUserMediaStream;\n\n window.latestUserMediaStream = null;\n window.latestLocalMediaConstraints = null;\n userMediaPromise = null;\n\n if (streamResult && streamResult.mediaStream) {\n streamResult.mediaStream.getTracks().forEach((track: MediaStreamTrack) => {\n track.stop();\n });\n\n trace.info('destroyUserMediaStream - stop all tracks');\n }\n\n return Promise.resolve();\n }\n\n destroyDesktopMediaStream(): Promise<void> {\n const streamResult = window.latestDesktopMediaStream;\n\n window.latestDesktopMediaStream = null;\n desktopMediaPromise = null;\n\n if (streamResult && streamResult.mediaStream) {\n streamResult.mediaStream.getTracks().forEach((track: MediaStreamTrack) => {\n track.stop();\n });\n }\n\n return Promise.resolve();\n }\n\n clearAllStreams(): Promise<void> {\n trace.info('LocalStreamManager Clearing all streams');\n\n return Promise.all([\n this.destroyDesktopMediaStream(),\n this.destroyUserMediaStream()\n ]).then(() => undefined);\n }\n\n private getHighestResolutionStream(isDesktopMedia: boolean, constraints?: LocalMediaConstraints): Promise<MediaRequestSuccessResult> {\n const constraintsCandidates = this.getParsedConstraints(isDesktopMedia, constraints);\n\n trace.info('getHighestResolutionStream - constraintsCandidates:', constraintsCandidates);\n\n if (constraintsCandidates.length === 0) {\n return Promise.reject(new Error('No constraints to retrieve the stream'));\n }\n\n return new Promise((resolve, reject) => {\n let lastSuccessfulResult: Nullable<MediaRequestSuccessResult> = null;\n let lastFailedResult: Nullable<MediaRequestFailResult> = null;\n let isStreamRequestFulfilled = false;\n\n (Promise as any).reduce(constraintsCandidates, (total: any, candidate: MediaStreamConstraints) => {\n if (isStreamRequestFulfilled) {\n return;\n }\n\n const constraintType = get(constraints, 'video.videoResolution.constraintType') || get(constraints, 'video.videoResolution[0].constraintType');\n\n return this.getStreamPromiseByParsedConstraints(candidate, isDesktopMedia)\n .then((stream: MediaStream) => {\n lastSuccessfulResult = {\n mediaStream: stream,\n constraint: candidate,\n constraintType,\n isNew: true\n };\n trace.info('getHighestResolutionStream Success', lastSuccessfulResult);\n isStreamRequestFulfilled = true;\n resolve(lastSuccessfulResult);\n })\n .catch((failResult: MediaRequestFailResult) => {\n lastFailedResult = failResult;\n\n if (lastFailedResult.errorCode === MediaRequestErrorCode.PermissionDenied) {\n isStreamRequestFulfilled = true;\n reject(lastFailedResult);\n }\n });\n }, 0)\n .then(() => {\n if (!isStreamRequestFulfilled) {\n if (!lastFailedResult) {\n lastFailedResult = {\n errorCode: MediaRequestErrorCode.GeneralError,\n message: 'Suitable stream cannot be created'\n };\n }\n reject(lastFailedResult);\n }\n });\n });\n }\n\n private getUserMediaStreamMediaDevice(constraints: MediaStreamConstraints, isDesktopMedia: boolean) {\n let streamPromise = null;\n\n if (!isDesktopMedia) {\n streamPromise = navigator.mediaDevices.getUserMedia(constraints);\n } else {\n //@ts-ignore\n streamPromise = navigator.mediaDevices.getDisplayMedia\n //@ts-ignore\n ? navigator.mediaDevices.getDisplayMedia(constraints)\n : navigator.getDisplayMedia(constraints);\n }\n\n return streamPromise;\n }\n\n //@ts-ignore\n private getStreamPromiseByParsedConstraints(constraints: MediaStreamConstraints, isDesktopMedia: boolean): Promise<MediaStream> {\n trace.info('getStreamPromiseByParsedConstraints', constraints);\n\n try {\n return this.getUserMediaStreamMediaDevice(constraints, isDesktopMedia).catch((err: any) => {\n let streamPromise = null;\n\n trace.info('getStreamPromiseByParsedConstraints - get user stream error', err.name);\n\n // Android (Galaxy esp.) have a bug that when using facingMode it might cause NotReadableError. So\n // we should try again but without facingMode. The exception to the rule is when we don't have specific\n // device to use (like in IOS13 bug, as can be seen in MediaServiceBase.ts), in that case, we might end\n // up using the front camera, which is a privacy concern and the preference is to fallback from video\n if (err && (err.name === 'NotReadableError') && get(constraints, 'video.facingMode') &&\n get(constraints, 'video.deviceId')) {\n\n const newConstraints = constraints;\n\n trace.info(`getStreamPromiseByParsedConstraints - NotReadableError - Failed to get video user media stream with facingMode=${get(constraints, 'video.facingMode')}`, err.name);\n\n // @ts-ignore\n delete newConstraints.video.facingMode;\n\n streamPromise = this.getStreamPromiseByParsedConstraints(newConstraints, isDesktopMedia);\n } else if (err && err.name && constraints.audio && !constraints.video) {\n trace.info('getStreamPromiseByParsedConstraints - audioStreamFailed - Failed to get audio user media stream', err.name);\n\n return this.parseMediaErrorThenThrow({name: 'audioStreamFailed'});\n } else if (err && err.name && constraints.audio) {\n const newConstraints = constraints;\n\n trace.info('getStreamPromiseByParsedConstraints - Failed to get video and audio user media stream, try to get video media stream only', err.name);\n\n newConstraints.audio = false;\n streamPromise = this.getStreamPromiseByParsedConstraints(newConstraints, isDesktopMedia);\n } else if (err && !streamPromise) {\n trace.info('getStreamPromiseByParsedConstraints - Failed to get video user media stream', err && err.name);\n\n streamPromise = Promise.reject(err);\n }\n\n return streamPromise && streamPromise.catch(this.parseMediaErrorThenThrow);\n });\n } catch (e) {\n this.parseMediaErrorThenThrow(e);\n }\n }\n\n setGroupedDevices(groupedDevices: {video: MediaDeviceInfo[]; audio: MediaDeviceInfo[]}) {\n trace.info('setGroupedDevices', groupedDevices);\n this._groupedDevices = {video: groupedDevices.video\n ? {back: last(groupedDevices.video), front: first(groupedDevices.video)}\n : null,\n audio: last(groupedDevices.audio) || null,\n camerasCount: groupedDevices.video.length};\n }\n\n private getParsedConstraints(isDesktopMedia: boolean, streamConstraints?: LocalMediaConstraints): MediaStreamConstraints[] {\n const constraintCandidates: MediaStreamConstraints[] = [];\n\n //TODO - Alex: need to understand which constraints not compatible with screen share\n if (isDesktopMedia) {\n return [{video: true}];\n }\n\n if (!streamConstraints) {\n return constraintCandidates;\n }\n\n const baseConstraints: MediaStreamConstraints = {};\n\n baseConstraints.audio = typeof streamConstraints.audio === 'boolean' ? streamConstraints.audio : undefined;\n\n if (!streamConstraints.video) {\n return [baseConstraints];\n }\n\n const cameraType = (streamConstraints.video !== 'boolean') &&\n (streamConstraints.video as LocalVideoStreamConstraints).videoSourceType;\n\n baseConstraints.video = {\n facingMode: cameraType && cameraType === LocalVideoSourceType.CAMERA_FRONT\n ? 'user'\n : 'environment',\n frameRate: {ideal: 15, max: 30}\n };\n\n const deviceId = get(this._groupedDevices, `video.${cameraType === LocalVideoSourceType.CAMERA_FRONT ? 'front' : 'back'}.deviceId`);\n\n if (deviceId) {\n baseConstraints.video.deviceId = deviceId;\n }\n\n const assignConstraint = (videoResolution: VideoStreamResolution): void => {\n const [width, height] = videoResolution.resolution.split('x');\n const constraintCandidate = assign({}, baseConstraints);\n\n constraintCandidate.video = assign({}, baseConstraints.video, {\n width: _constraintByType(width, videoResolution.constraintType),\n height: _constraintByType(height, videoResolution.constraintType)\n });\n constraintCandidates.push(constraintCandidate);\n };\n\n if ((baseConstraints.video && typeof streamConstraints.video === 'boolean') ||\n !((streamConstraints.video as LocalVideoStreamConstraints).videoResolution instanceof Array)) {\n\n const defaultResolution: VideoStreamResolution = {\n resolution: DEFAULT_VIDEO_RESOLUTION,\n constraintType: DEFAULT_VIDEO_CONSTRAINT_TYPE\n };\n\n assignConstraint(defaultResolution);\n\n return constraintCandidates;\n }\n\n (streamConstraints.video as any).videoResolution.forEach(assignConstraint);\n\n return constraintCandidates;\n }\n\n private parseMediaErrorThenThrow(error: any): MediaRequestFailResult {\n if (error && error.errorCode) {\n throw error;\n }\n\n trace.error('Get media stream error.', error);\n const errorName = error && error.name ? error.name : '';\n\n const permissionErrorNames = [\n 'NotAllowedError',\n 'PermissionDismissedError',\n 'PermissionDeniedError'\n\n ];\n\n // const generalError = [\n // 'NotFoundError',\n // 'DevicesNotFoundError',\n // 'NotReadableError',\n // 'TrackStartError'\n // ];\n\n const constraintsErrors = [\n 'OverconstrainedError',\n 'ConstraintNotSatisfiedError'\n\n ];\n\n const isErrorOfType = (errorsArr: string[]) => errorsArr.filter((err: string) => err === errorName).length > 0;\n\n if (isErrorOfType(permissionErrorNames)) {\n throw {errorCode: MediaRequestErrorCode.PermissionDenied, message: errorName};\n }\n\n if (isErrorOfType(constraintsErrors)) {\n throw {errorCode: MediaRequestErrorCode.Overconstrained, message: errorName};\n }\n\n throw {errorCode: MediaRequestErrorCode.GeneralError, message: errorName};\n }\n}\n\nfunction _constraintByType(value: any, type: ConstraintType): any {\n switch (type) {\n case ConstraintType.MIN:\n return {min: value};\n default:\n return {ideal: value};\n }\n}\n"]}
|
|
@@ -36,8 +36,8 @@ export declare abstract class TechseeMediaServiceBase {
|
|
|
36
36
|
onStreamDestroyed(callback: (eventArgs: StreamDestroyedEventArgs) => void): void;
|
|
37
37
|
onStreamCreated(callback: (eventArgs: StreamCreatedEventArgs) => void): void;
|
|
38
38
|
private getSwitchCameraConstraints;
|
|
39
|
-
switchCamera(
|
|
40
|
-
|
|
39
|
+
switchCamera(isFailed?: boolean): Promise<{
|
|
40
|
+
constraints: LocalMediaConstraints;
|
|
41
41
|
streamResult: MediaRequestSuccessResult;
|
|
42
42
|
}>;
|
|
43
43
|
private replaceStreamTracks;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/MediaServiceBase.ts"],"names":[],"mappings":"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","file":"MediaServiceBase.d.ts","sourcesContent":["import filter from 'lodash/filter';\nimport cloneDeep from 'lodash/cloneDeep';\nimport values from 'lodash/values';\nimport debounce from 'lodash/debounce';\nimport includes from 'lodash/includes';\nimport {EventEmitter} from 'events';\nimport {Nullable} from '@techsee/techsee-common';\nimport {TechseeMediaSubscriber} from './MediaSubscriber';\nimport {TechseeMediaPublisher} from './MediaPublisher';\nimport {LocalStreamManager} from './LocalStreamManager';\nimport {TechseeMediaStream} from './TechseeMediaStream';\nimport {TurnWebRtcSession} from './MediaSession/SessionTurn';\nimport {getSnapshotFromMediaStream} from './MediaUtils/MediaDomUtils';\nimport {\n ClientWebRtcInfo,\n DeviceSupportInfo,\n IMediaEnvironment,\n IMediaSession,\n IMediaSubscriber,\n ISessionStreamsManager, LocalMediaConstraints,\n LocalVideoStreamConstraints,\n MediaRequestFailResult, MediaRequestSuccessResult,\n MediaServiceOptions,\n MediaSessionConfiguration,\n MediaSessionHandlers,\n MediaSessionIceCredentials,\n MediaSessionParams,\n MediaSubscriberParams,\n RemoteMediaTrack,\n RemoteTrackStats,\n SnapshotOptions,\n SnapshotResult,\n StreamCreatedEventArgs,\n StreamDestroyedEventArgs\n} from './MediaContracts';\nimport {enumerateMediaDevices, isVideoPlaySupportedOnDevice} from './MediaUtils/Compatibility';\nimport {OpentokSession} from './MediaSession/SessionOpentok';\nimport {\n DEFAULT_PEER_CONNECTIVITY_TIMEOUT_SECONDS,\n KnownMediaStream,\n KnownMediaStreamKind,\n LocalVideoSourceType,\n MediaServiceType,\n MediaSessionDisconnectReason,\n MediaStreamUnregisterReason,\n SessionClientRole\n} from './MediaConstants';\nimport {throwableGuard} from '@techsee/techsee-common/lib/core/guards';\n\nimport {getMediaTracer} from './MediaUtils/MediaTracer';\n\nconst trace = getMediaTracer('MediaServiceBase');\nconst traceStatsInfo = debounce(trace.info, 1000 * 5, {leading: true, maxWait: 1000 * 30});\n\nenum privateEvents {\n STREAM_CREATED = 'STREAM_CREATED',\n STREAM_DESTROYED = 'STREAM_DESTROYED'\n}\n\nexport abstract class TechseeMediaServiceBase {\n //This method should not be called. The only place that allowed to use it, is initLocalMediaStreams method.\n //Everyone who need to initialize MediaStream should use initLocalMediaStreams method.\n protected abstract getLocalMediaImplementation(): Promise<void>;\n\n protected abstract createMediaPublisher(destinationRole: SessionClientRole): Promise<Nullable<TechseeMediaPublisher>>;\n\n protected readonly _environment: IMediaEnvironment;\n protected readonly _sessionStreamsManager: ISessionStreamsManager;\n protected readonly _localStreamsManager: LocalStreamManager;\n\n private readonly _deviceSupportFlags: DeviceSupportInfo;\n private readonly _subscribers: Map<HTMLDivElement, TechseeMediaSubscriber> = new Map();\n private readonly _publishers: Map<SessionClientRole, TechseeMediaPublisher> = new Map();\n private readonly _publisherPromises: Map<SessionClientRole, Promise<Nullable<MediaStream>>> = new Map();\n protected _serviceOptions: Nullable<MediaServiceOptions> = null;\n private _isIOS_13_orLater: boolean = false;\n private _autoReconnectEnabled: boolean = true;\n private readonly _emitter: EventEmitter;\n private _initLocalStreamsPromise: Nullable<Promise<void>> = null;\n private _initServicePromise: Nullable<Promise<void>> = null;\n private _registeredStreams: Map<KnownMediaStream, TechseeMediaStream> = new Map();\n private _isVoipEnabled: boolean = false;\n private _session: Nullable<{\n params: MediaSessionParams;\n connectPromise?: Promise<void>;\n instance?: IMediaSession;\n }> = null;\n\n protected constructor(environment: IMediaEnvironment, webRtcSupportInfo: ClientWebRtcInfo) {\n trace.info('TechseeMediaServiceBase created');\n\n this.bindClassMethods();\n\n this._emitter = new EventEmitter();\n this._environment = environment;\n this._deviceSupportFlags = {\n videoPlayback: false,\n hasCamera: false,\n hasMicrophone: false,\n webRtcSupportInfo: webRtcSupportInfo\n };\n\n this._localStreamsManager = new LocalStreamManager(this._environment);\n\n // TODO: Hack as IOS 13.3 and above sometimes do not return the video/audio device. This\n // is a hack to work around the issue until Apple resolve it\n this._isIOS_13_orLater = this._environment.isIOS() &&\n (!this._environment.majorVersion() || (this._environment.majorVersion() >= 13));\n\n trace.info(`TechseeMediaServiceBase: isIOS_13_orLater: ${this._isIOS_13_orLater}`);\n\n this._sessionStreamsManager = {\n getMediaStreamForRole: this.getStreamForDestinationRole,\n addRemoteMediaTrack: this.registerRemoteMediaTrack,\n removeMediaTrack: this.unregisterRemoteMediaTrack\n };\n }\n\n get deviceSupportInfo(): DeviceSupportInfo {\n this.serviceInitGuard(false);\n\n return cloneDeep(this._deviceSupportFlags);\n }\n\n get isVoipEnabled(): boolean {\n return this._isVoipEnabled;\n }\n\n get isSessionActive(): boolean {\n return this._session !== null;\n }\n\n get isLocalStreamInitialized(): boolean {\n return this._initLocalStreamsPromise !== null;\n }\n\n /*\n Initializes media service. Before using anything from this service it should be initialized first.\n All of serviceOptions should be retrieved from account settings and by evaluating application state.\n For example, when initializing service, accountSettings is already loaded, and we know if FS session is audio enabled.\n */\n initMediaService(serviceOptions: MediaServiceOptions): Promise<void> {\n trace.info('TechseeMediaServiceBase.initMediaService', serviceOptions);\n const warnMessage = 'Multiple initialization does not supposed to happen. Not rejecting, but check the flow for correctness.';\n\n if (this._initServicePromise || this._serviceOptions) {\n console.warn(warnMessage);\n\n return this._initServicePromise ? this._initServicePromise : Promise.resolve();\n }\n\n this._serviceOptions = cloneDeep(serviceOptions);\n const isOpenTok = this._serviceOptions.mediaServiceType === MediaServiceType.OPENTOK;\n\n this._initServicePromise = Promise.all([\n isVideoPlaySupportedOnDevice(isOpenTok)\n .then((isSupported: boolean) => this._deviceSupportFlags.videoPlayback = isSupported),\n enumerateMediaDevices()\n .catch((error) => {\n trace.warn(error);\n\n return [];\n })\n .then((localDevicesList: MediaDeviceInfo[]) => {\n const groupedDevices = {\n video: filter(localDevicesList, (device) => device.kind.toLowerCase() === 'videoinput'),\n audio: filter(localDevicesList, (device) => device.kind.toLowerCase() === 'audioinput')\n };\n\n this._localStreamsManager.setGroupedDevices(groupedDevices);\n this._deviceSupportFlags.hasCamera = this._isIOS_13_orLater || (groupedDevices.video.length > 0);\n this._deviceSupportFlags.hasMicrophone = this._isIOS_13_orLater || (groupedDevices.audio.length > 0);\n\n })\n ]).then(() => undefined);\n\n\n return this._initServicePromise;\n }\n\n //#region Media Streams Management\n\n /*\n The main method to start the initialization of local media streams.\n This method can be executed many times and it will insure all calls are synced and only relevant stream will be created.\n DO NOT USE ANY OTHER METHOD THAT FORCES \"navigator.getUserMedia\" OTHER THAN THIS ONE.\n */\n initLocalMediaStreams(): Promise<void> {\n this.serviceInitGuard();\n\n if (!this._initLocalStreamsPromise) {\n trace.info('initLocalMediaStreams creating media streams');\n this._initLocalStreamsPromise = this.getLocalMediaImplementation()\n .catch((mediaRequestFailResult: MediaRequestFailResult) => {\n trace.error('initLocalMediaStreams failure: ', mediaRequestFailResult);\n\n throw mediaRequestFailResult;\n });\n } else {\n trace.info('initLocalMediaStreams already done or in progress');\n }\n\n return this._initLocalStreamsPromise;\n }\n\n /*\n Creates an instance of subscriber that will \"listen\" to the requested stream.\n Whenever requested stream is available, subscriber will render it.\n */\n createSubscriber(subscriberParams: MediaSubscriberParams): Promise<IMediaSubscriber> {\n if (this._subscribers.has(subscriberParams.container)) {\n return Promise.reject('Subscriber for provided DIV element already exists');\n }\n\n const subscriber = new TechseeMediaSubscriber(subscriberParams);\n\n this._subscribers.set(subscriberParams.container, subscriber);\n const streamForSubscriber = this.getRegisteredStreamByType(subscriber.streamType);\n\n if (streamForSubscriber) {\n trace.info(`Stream for ${subscriberParams.streamType} is ready.`);\n subscriber.renderStream(streamForSubscriber);\n } else {\n trace.info(`Stream for ${subscriberParams.streamType} not created yet`);\n }\n\n return Promise.resolve(subscriber);\n }\n\n /*\n Destroys the subscriber for provided HTML DIV container\n */\n destroySubscriber(container: HTMLDivElement): Promise<void> {\n if (this._subscribers.has(container)) {\n const subscriber = this._subscribers.get(container);\n\n subscriber!.dispose();\n }\n\n this._subscribers.delete(container);\n\n return Promise.resolve();\n }\n\n onStreamDestroyed(callback: (eventArgs: StreamDestroyedEventArgs) => void): void {\n this.registerEventCallback(privateEvents.STREAM_DESTROYED, callback);\n }\n\n onStreamCreated(callback: (eventArgs: StreamCreatedEventArgs) => void): void {\n this.registerEventCallback(privateEvents.STREAM_CREATED, callback);\n }\n\n private getSwitchCameraConstraints(constraints: Nullable<LocalMediaConstraints>) {\n const newConstraints = cloneDeep(constraints);\n\n if (!newConstraints) {\n throw new Error('getSwitchCameraConstraints: unexpected use case constraints is null.');\n }\n\n trace.info('getSwitchCameraConstraints - Start switch camera with constraints:', newConstraints);\n\n const videoSourceType = (newConstraints.video as LocalVideoStreamConstraints).videoSourceType === LocalVideoSourceType.CAMERA\n ? LocalVideoSourceType.CAMERA_FRONT\n : LocalVideoSourceType.CAMERA;\n\n trace.info(`getSwitchCameraConstraints - switch to videoSourceType: ${videoSourceType}`);\n\n (newConstraints.video as LocalVideoStreamConstraints).videoSourceType = videoSourceType;\n\n return constraints;\n }\n\n switchCamera(constraints: Nullable<LocalMediaConstraints>, isFailed?: boolean) {\n trace.info('switchCamera: start');\n\n const newConstraints = !isFailed ? this.getSwitchCameraConstraints(constraints) : cloneDeep(constraints);\n\n if (!newConstraints) {\n throw new Error('switchCamera - unexpected use case constraints is null.');\n }\n\n this._localStreamsManager.destroyUserMediaStream();\n\n return this._localStreamsManager.getUserMediaStream(newConstraints)\n .then((streamResult: MediaRequestSuccessResult) => {\n const stream = streamResult.mediaStream;\n\n trace.info('switchCamera: new stream: ', stream);\n\n this.replaceStreamTracks(stream);\n\n return this.registerStreamResult(newConstraints, streamResult, true).then(() => ({newConstraints, streamResult}));\n })\n .catch((err) => {\n trace.error('switchCamera: Failed to switch camera: ', err);\n\n if (!isFailed) {\n this.switchCamera(constraints, true);\n }\n\n return Promise.reject('Failed to switch camera.');\n });\n }\n\n private replaceStreamTracks(mediaStream: Nullable<MediaStream>) {\n this._session!.instance && this._session!.instance.replaceStreamTracks(mediaStream);\n }\n\n protected registerStreamResult(constraints: Nullable<LocalMediaConstraints>, streamResult: MediaRequestSuccessResult, switchCamera?: boolean, addStreamType?: KnownMediaStream) {\n const regPromises: any[] = [];\n\n if (streamResult.isNew) {\n trace.info('registerStreamResult: stream result from getUserMediaStream:', streamResult.mediaStream);\n streamResult.mediaStream.getTracks().forEach((mediaTrack) => {\n trace.info('registerStreamResult: stream result from mediaTrack:', mediaTrack);\n const streamType = addStreamType || (mediaTrack.kind === 'video'\n ? KnownMediaStream.USER_VIDEO_STREAM : KnownMediaStream.USER_AUDIO_STREAM);\n const newDedicatedStream = new TechseeMediaStream(mediaTrack, streamType, false);\n\n regPromises.push(switchCamera ? this.registerTrack(newDedicatedStream) : this.registerLocalMediaStream(newDedicatedStream));\n });\n }\n\n return Promise.all(regPromises);\n }\n\n get supportSwitchCameras() {\n return this._isIOS_13_orLater || this._localStreamsManager.groupedDevices.camerasCount > 1;\n }\n\n //#endregion Media Streams Management\n\n //#region Session Management\n\n /*\n Creates instance of WebRTC session which connects to (signaling server), and begins to listen to WebRTC events.\n */\n connectToSession(sessionParams: MediaSessionParams): Promise<void> {\n this.serviceInitGuard();\n\n if (!this._session) {\n this._session = {params: cloneDeep(sessionParams)};\n this._session.connectPromise = new Promise((resolve, reject) => {\n const doAsyncReject = (error: any): void => {\n\n /*\n Reject with timeout, to change JS 'execution flow'.\n If timeout removed, session is assigned to null, before promise returns, and \"catch\" will not\n work on promise that is returned.\n */\n setTimeout(() => {\n this._session = null;\n reject(error);\n });\n };\n\n const sessionHandlers: MediaSessionHandlers = {\n onDisconnectedHandler: this.onSessionDisconnectHandler\n };\n const mediaSessionParams: MediaSessionParams & MediaSessionConfiguration = {\n ...sessionParams,\n peerConnectivityTimeoutSeconds: this._serviceOptions!.peerConnectivityTimeoutSeconds || DEFAULT_PEER_CONNECTIVITY_TIMEOUT_SECONDS\n };\n\n trace.info('connectToSession', sessionParams);\n if (this._serviceOptions!.mediaServiceType === MediaServiceType.TURNSERVER) {\n this._session!.instance = new TurnWebRtcSession(mediaSessionParams, sessionHandlers, this._sessionStreamsManager);\n\n } else if (this._serviceOptions!.mediaServiceType === MediaServiceType.OPENTOK) {\n this._session!.instance = new OpentokSession(mediaSessionParams, this._sessionStreamsManager);\n } else {\n const allowedValues = values(MediaServiceType);\n\n trace.error(`mediaServiceType '${this._serviceOptions!.mediaServiceType}' is not supported.`, {allowedValues});\n doAsyncReject(new Error('mediaServiceType is not supported'));\n }\n\n if (this._session && this._session.instance) {\n this._session!.instance.connect()\n .then(() => {\n trace.info('Session is connected');\n resolve();\n })\n .catch((error: any) => {\n trace.error('Failed to connect to session', error);\n doAsyncReject(error);\n });\n }\n\n });\n } else {\n trace.warn('Session connected already. Disconnect before connect to new session again');\n }\n\n //TODO - Alex: Change a structure of this function to be more safe\n //(when we return, there should not be case when session or promise is null)\n //To fix, need to return promise immediately, then execute process in setTimeout(fn, 0);\n return this._session!.connectPromise!;\n }\n\n /*\n Disconnects from WebRTC session.\n */\n disconnectFromSession(): Promise<void> {\n return this.disconnectFromSessionInternal(MediaSessionDisconnectReason.ForcedByConsumer);\n }\n\n /*\n Updates a credentials for turn server\n */\n updateSessionCredentials(credentials: MediaSessionIceCredentials): Promise<void> {\n trace.info('updateSessionCredentials');\n if (!this.sessionExistsGuard(false)) {\n return Promise.reject('There no session to update credentials');\n }\n\n this._session!.params.credentials = cloneDeep(credentials);\n\n return Promise.resolve();\n }\n\n enableVoipDuringSession(): void {\n if (!this._isVoipEnabled && this.isLocalStreamInitialized) {\n throw new Error('Voip support cannot be enabled after local stream where initialized');\n }\n\n this._isVoipEnabled = true;\n }\n\n protected disconnectFromSessionInternal(reason: MediaSessionDisconnectReason): Promise<void> {\n if (!this._session) {\n return Promise.resolve();\n }\n\n trace.info('Disconnecting from session', reason);\n\n const lastSession = this._session;\n\n this._session = null;\n\n return Promise.all([\n lastSession.instance!.disconnect(),\n this.clearPublishers()\n ]).then(() => undefined);\n }\n\n protected getStatsForRemoteTrack(streamType: KnownMediaStream): Promise<RemoteTrackStats> {\n if (!this.sessionExistsGuard(false)) {\n return Promise.reject(new Error('Media session not started'));\n }\n\n const streamForStats = this.getRegisteredStreamByType(streamType);\n\n if (!streamForStats) {\n return Promise.reject(new Error('Stream for requested type was not found'));\n }\n\n return this._session!.instance!.getRemoteTrackStats(streamForStats.mediaTrack)\n .then((trackStats: RemoteTrackStats) => {\n traceStatsInfo(`MediaTrackStats for ${streamType}: ${JSON.stringify(trackStats)}`);\n\n return {\n trackId: streamForStats.mediaTrack.id,\n trackStats: trackStats\n };\n });\n }\n\n private onSessionDisconnectHandler(reason: MediaSessionDisconnectReason): void {\n trace.info('onSessionDisconnectHandler', reason);\n if (reason !== MediaSessionDisconnectReason.ForcedByConsumer) {\n const lastParams = this._session && this._session.params ? this._session.params : null;\n\n this.disconnectFromSessionInternal(reason).then(() => {\n const reconnectReasons = [\n MediaSessionDisconnectReason.InitiatorPeerReconnected,\n MediaSessionDisconnectReason.SignalingChannelDisconnect,\n MediaSessionDisconnectReason.PeerConnectionInterrupted,\n MediaSessionDisconnectReason.PeerConnectionStateChangeTimeout\n ];\n\n if (lastParams && includes(reconnectReasons, reason) && this._autoReconnectEnabled) {\n this.reconnectToSession(lastParams);\n } else if (!this._autoReconnectEnabled) {\n trace.info('No reconnection- auto reconnect disabled');\n } else {\n trace.info('No params for reconnection to media session');\n }\n });\n } else {\n trace.info('Ignore Session disconnect event');\n }\n }\n\n setAutoReconnect(state: boolean) {\n this._autoReconnectEnabled = state;\n trace.info('_autoReconnectEnabled:', state);\n\n if (this._autoReconnectEnabled && !this.isSessionActive) {\n const lastParams = this._session && this._session.params ? this._session.params : null;\n\n if (lastParams) {\n trace.info('setAutoReconnect - reconnect to session');\n this.reconnectToSession(lastParams);\n }\n }\n }\n\n reconnectToSession(lastParams: MediaSessionParams) {\n trace.info('Reconnecting to media session, sessionParams:', lastParams);\n\n this.connectToSession(lastParams)\n .then(() => {\n trace.info('Media session reconnected');\n })\n .catch((error: any) => {\n trace.error('Error while reconnecting to media session', error);\n });\n }\n\n //#endregion Session Management\n\n //#region Utils\n\n getSnapshotFromKnownStream(sourceStream: KnownMediaStream, snapshotOptions?: SnapshotOptions): Promise<SnapshotResult> {\n return new Promise((resolve, reject) => {\n if (sourceStream !== KnownMediaStream.USER_VIDEO_STREAM &&\n sourceStream !== KnownMediaStream.USER_SCREEN_SHARE_STREAM) {\n trace.error('The requested stream is not video stream, and cannot be used for snapshot');\n reject(new Error('INCOMPATIBLE_STREAM_FOR_SNAPSHOT'));\n\n return;\n }\n\n const snapshotStream = this.getRegisteredStreamByType(sourceStream);\n\n if (!snapshotStream) {\n trace.error('Cannot make snapshot: The requested stream not exists yet.');\n reject(new Error('NO_REQUESTED_STREAM'));\n\n return;\n }\n\n getSnapshotFromMediaStream(snapshotStream.mediaStream, snapshotOptions)\n .then((imageData: any) => {\n const urlComponents = imageData.split(';base64,');\n const mimeType = urlComponents[0].split(':')[1];\n const bytes = atob(urlComponents[1]);\n const buffer = new ArrayBuffer(bytes.length);\n const rawData = new Uint8Array(buffer);\n\n for (let i = 0; i < bytes.length; i++) {\n rawData[i] = bytes.charCodeAt(i);\n }\n\n const blob = new Blob([rawData], {type: mimeType});\n const objectUrl = window.URL.createObjectURL(blob);\n\n trace.info('Snapshot created successfully');\n\n const result: SnapshotResult = {\n base64img: imageData,\n objectUrl: objectUrl,\n imageBlob: blob,\n mimeType: mimeType\n };\n\n resolve(result);\n })\n .catch((error: any) => {\n trace.error('Error creating snapshot', error);\n reject(error);\n });\n });\n }\n\n //Will clean streams, publishers and subscribers, but will not remove event listeners\n clearService(): Promise<void> {\n trace.info('MediaService clearing all resources');\n\n return this.disconnectFromSessionInternal(MediaSessionDisconnectReason.ForcedByConsumer)\n .then(this.clearSubscribers)\n .then(this.clearRegisteredStreams)\n .then(this._localStreamsManager.clearAllStreams)\n .then(() => {\n this._initLocalStreamsPromise = null;\n this._isVoipEnabled = false;\n }).then(() => undefined);\n }\n\n //Will clear the service and remove all event listeners\n dispose(): Promise<void> {\n return this.clearService().catch(() => undefined).then(() => {\n this._emitter.removeAllListeners();\n });\n }\n\n //#endregion\n\n //#region Protected Methods\n\n protected get subscribers(): Map<HTMLDivElement, TechseeMediaSubscriber> {\n return this._subscribers;\n }\n\n protected registerLocalMediaStream(tsMediaStream: TechseeMediaStream): Promise<void> {\n trace.info('Registering local stream', tsMediaStream.streamType);\n if (this._registeredStreams.has(tsMediaStream.streamType)) {\n return Promise.reject(new Error(`Stream ${tsMediaStream.streamType} already registered`));\n }\n\n return this.registerStream(tsMediaStream);\n }\n\n protected getRegisteredStreamByType(streamType: KnownMediaStream): Nullable<TechseeMediaStream> {\n return this._registeredStreams.get(streamType) || null;\n }\n\n protected changeEnableForKnownStream(streamType: KnownMediaStream, isPaused: boolean): void {\n const streamToChangeState = this.getRegisteredStreamByType(streamType);\n\n if (!streamToChangeState) {\n trace.warn('There no stream found to change enable state', streamType);\n } else {\n streamToChangeState.mediaTrack.enabled = !isPaused;\n\n if (streamToChangeState.isRemote && streamToChangeState.streamKind === KnownMediaStreamKind.Audio) {\n this.subscribers.forEach((subscriber) => {\n if (subscriber.streamType === streamType) {\n subscriber.muteSound(isPaused);\n }\n });\n }\n\n trace.info('Local stream enable state is changed', {streamType, isPaused});\n }\n }\n\n protected registerEventCallback(event: string, callback: (eventArgs?: any) => void): void {\n this._emitter.on(event, callback);\n }\n\n protected emitEvent(event: string, eventArgs?: any): void {\n setTimeout(() => {\n this._emitter.emit(event, eventArgs);\n });\n }\n\n //#endregion Protected Methods\n\n //#region Private Methods\n\n private registerRemoteMediaTrack(remoteMediaTrack: RemoteMediaTrack): Promise<void> {\n trace.info(`Registering remote ${remoteMediaTrack.trackType} MediaStreamTrack`, remoteMediaTrack.mediaTrack);\n const currentStream = this._registeredStreams.get(remoteMediaTrack.trackType);\n\n if (currentStream && !currentStream.isRemote) {\n return Promise.reject(new Error('Cannot register remote stream with the same type as local stream'));\n }\n\n const newDedicatedStream = new TechseeMediaStream(remoteMediaTrack.mediaTrack, remoteMediaTrack.trackType, true);\n\n return this.registerStream(newDedicatedStream);\n }\n\n private registerTrack(mediaStream: TechseeMediaStream): Promise<void> {\n mediaStream.mediaTrack.onended = () =>\n this.unregisterTechseeMediaStream(mediaStream, MediaStreamUnregisterReason.NativeEvent);\n\n this._registeredStreams.set(mediaStream.streamType, mediaStream);\n\n trace.info(`TechseeMediaStream registered - ${mediaStream.streamType}`, TechseeMediaStream);\n\n return this.updateSubscribersWithNewStream(mediaStream).then(() => {\n const eventArgs: StreamCreatedEventArgs = {streamType: mediaStream.streamType};\n\n this.emitEvent(privateEvents.STREAM_CREATED, eventArgs);\n });\n }\n\n private registerStream(mediaStream: TechseeMediaStream): Promise<void> {\n const currentStream = this._registeredStreams.get(mediaStream.streamType);\n\n if (currentStream) {\n return this.unregisterTechseeMediaStream(currentStream, MediaStreamUnregisterReason.ReplacingStream).then(() => this.registerTrack(mediaStream));\n }\n\n return this.registerTrack(mediaStream);\n }\n\n private unregisterTechseeMediaStream(streamToUnregister: TechseeMediaStream, reason: MediaStreamUnregisterReason): Promise<void> {\n const traceError = (error: any): void => {\n trace.warn('Unregister TechseeMediaStream error:', error);\n };\n\n const promises: any = [];\n\n trace.info('Unregister TechseeMediaStream: ', streamToUnregister.streamType);\n\n this._subscribers.forEach((subscriber) => {\n if (subscriber.streamType === streamToUnregister.streamType) {\n promises.push(subscriber.stopRendering().catch(traceError));\n }\n });\n\n if (!streamToUnregister.isRemote) {\n this._publishers.forEach((publisher: TechseeMediaPublisher) => {\n if (publisher.streamTypes.find((streamType) => streamType === streamToUnregister.streamType)) {\n if (this._session && this._session!.instance) {\n promises.push(this._session.instance.onMediaStreamDestroyed(publisher.destinationRole).catch(traceError));\n }\n }\n });\n }\n\n return Promise.all(promises)\n .then(() => {\n this._registeredStreams.delete(streamToUnregister.streamType);\n trace.info('Stream deleted', streamToUnregister.streamType);\n\n const eventArgs: StreamDestroyedEventArgs = {streamType: streamToUnregister.streamType, reason: reason};\n\n this.emitEvent(privateEvents.STREAM_DESTROYED, eventArgs);\n });\n }\n\n private unregisterRemoteMediaTrack(mediaTrack: MediaStreamTrack): Promise<void> {\n const promises: any = [];\n\n this._registeredStreams.forEach((registeredStream: TechseeMediaStream) => {\n if (registeredStream.mediaTrack.id === mediaTrack.id) {\n promises.push(this.unregisterTechseeMediaStream(registeredStream, MediaStreamUnregisterReason.ClosedRemotely));\n }\n });\n\n return Promise.all(promises).then(() => undefined);\n }\n\n private removePublisher(publisher: TechseeMediaPublisher): Promise<void> {\n this._publishers.delete(publisher.destinationRole);\n if (this._session && this._session.instance) {\n trace.info('Removing publisher', publisher.destinationRole);\n\n return this._session.instance.onMediaStreamDestroyed(publisher.destinationRole)\n .catch((error: any) => {\n trace.warn('Error while removing publisher', error);\n });\n }\n\n return Promise.resolve();\n }\n\n private getStreamForDestinationRole(destinationRole: SessionClientRole): Promise<Nullable<MediaStream>> {\n this.serviceInitGuard();\n\n if (this._publishers.has(destinationRole)) {\n trace.info(`Publisher for ${destinationRole} already exists`);\n\n return Promise.resolve(this._publishers.get(destinationRole)!.mediaStream);\n }\n\n if (!this._publisherPromises.get(destinationRole)) {\n trace.info(`Creating publisher for ${destinationRole}`);\n const publisherPromise = this.initLocalMediaStreams()\n .then(() => this.createMediaPublisher(destinationRole))\n .then((mediaPublisher: Nullable<TechseeMediaPublisher>) => {\n if (mediaPublisher) {\n this._publishers.set(destinationRole, mediaPublisher);\n }\n\n this._publisherPromises.delete(destinationRole);\n\n return mediaPublisher ? mediaPublisher.mediaStream : null;\n })\n .catch((ex: any) => {\n if (ex && ex.message === 'audioStreamFailed') {\n this._isVoipEnabled = false;\n\n return null;\n }\n\n this._publisherPromises.delete(destinationRole);\n throw ex;\n });\n\n this._publisherPromises.set(destinationRole, publisherPromise);\n } else {\n trace.info(`Create publisher promise for ${destinationRole} already exists`);\n }\n\n return this._publisherPromises.get(destinationRole)!;\n }\n\n private updateSubscribersWithNewStream(registeredStream: TechseeMediaStream): Promise<void> {\n trace.info(`Updating subscribers of ${registeredStream.streamType} with new stream`);\n\n if (this._subscribers.size === 0) {\n trace.warn(`No subscribers exists for the ${registeredStream.streamType}.`);\n } else {\n trace.info(`Total ${this._subscribers.size} subscribers exists, will check if rerender needed.`);\n }\n\n this._subscribers.forEach((subscriber) => {\n if (subscriber.streamType === registeredStream.streamType) {\n trace.info(`${registeredStream.streamType} rendering on subscriber`);\n subscriber.renderStream(registeredStream);\n }\n });\n\n return Promise.resolve();\n }\n\n private bindClassMethods(): void {\n this.updateSessionCredentials = this.updateSessionCredentials.bind(this);\n this.getStreamForDestinationRole = this.getStreamForDestinationRole.bind(this);\n this.registerRemoteMediaTrack = this.registerRemoteMediaTrack.bind(this);\n this.unregisterRemoteMediaTrack = this.unregisterRemoteMediaTrack.bind(this);\n this.createMediaPublisher = this.createMediaPublisher.bind(this);\n this.updateSubscribersWithNewStream = this.updateSubscribersWithNewStream.bind(this);\n this.initLocalMediaStreams = this.initLocalMediaStreams.bind(this);\n this.onSessionDisconnectHandler = this.onSessionDisconnectHandler.bind(this);\n this.clearService = this.clearService.bind(this);\n this.clearRegisteredStreams = this.clearRegisteredStreams.bind(this);\n this.clearPublishers = this.clearPublishers.bind(this);\n this.clearSubscribers = this.clearSubscribers.bind(this);\n }\n\n //region Cleanup\n\n private clearRegisteredStreams(): Promise<void> {\n trace.info('Clearing registered streams');\n const promises: any = [];\n\n this._registeredStreams.forEach((streamToUnregister: TechseeMediaStream) => {\n promises.push(this.unregisterTechseeMediaStream(streamToUnregister, MediaStreamUnregisterReason.ServiceCleanUp));\n });\n\n return Promise.all(promises).then(() => undefined);\n }\n\n private clearSubscribers(): Promise<void> {\n trace.info('Clearing subscribers');\n const promises: any = [];\n\n this._subscribers.forEach((subscriber) => {\n promises.push(this.destroySubscriber(subscriber.container));\n });\n\n return Promise.all(promises).then(() => undefined);\n }\n\n private clearPublishers(): Promise<void> {\n trace.info('Clearing publishers');\n const promises: any = [];\n\n this._publishers.forEach((publisher) => {\n promises.push(this.removePublisher(publisher));\n });\n\n return Promise.all(promises).then(() => undefined);\n }\n\n //#endregion Cleanup\n\n //#region Simple Validation Methods\n\n protected serviceInitGuard(shouldThrow = true): boolean {\n return throwableGuard(!!this._serviceOptions, 'Media service is not initialized', shouldThrow);\n }\n\n protected sessionExistsGuard(shouldThrow = true): boolean {\n return throwableGuard(!!this._session, 'There no active session', shouldThrow);\n }\n\n //#endregion\n\n //#endregion Private Methods\n}\n"]}
|
|
1
|
+
{"version":3,"sources":["../src/MediaServiceBase.ts"],"names":[],"mappings":"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","file":"MediaServiceBase.d.ts","sourcesContent":["import filter from 'lodash/filter';\nimport cloneDeep from 'lodash/cloneDeep';\nimport values from 'lodash/values';\nimport debounce from 'lodash/debounce';\nimport includes from 'lodash/includes';\nimport {EventEmitter} from 'events';\nimport {Nullable} from '@techsee/techsee-common';\nimport {TechseeMediaSubscriber} from './MediaSubscriber';\nimport {TechseeMediaPublisher} from './MediaPublisher';\nimport {LocalStreamManager} from './LocalStreamManager';\nimport {TechseeMediaStream} from './TechseeMediaStream';\nimport {TurnWebRtcSession} from './MediaSession/SessionTurn';\nimport {getSnapshotFromMediaStream} from './MediaUtils/MediaDomUtils';\nimport {\n ClientWebRtcInfo,\n DeviceSupportInfo,\n IMediaEnvironment,\n IMediaSession,\n IMediaSubscriber,\n ISessionStreamsManager, LocalMediaConstraints,\n LocalVideoStreamConstraints,\n MediaRequestFailResult, MediaRequestSuccessResult,\n MediaServiceOptions,\n MediaSessionConfiguration,\n MediaSessionHandlers,\n MediaSessionIceCredentials,\n MediaSessionParams,\n MediaSubscriberParams,\n RemoteMediaTrack,\n RemoteTrackStats,\n SnapshotOptions,\n SnapshotResult,\n StreamCreatedEventArgs,\n StreamDestroyedEventArgs\n} from './MediaContracts';\nimport {enumerateMediaDevices, isVideoPlaySupportedOnDevice} from './MediaUtils/Compatibility';\nimport {OpentokSession} from './MediaSession/SessionOpentok';\nimport {\n DEFAULT_PEER_CONNECTIVITY_TIMEOUT_SECONDS,\n KnownMediaStream,\n KnownMediaStreamKind,\n LocalVideoSourceType,\n MediaServiceType,\n MediaSessionDisconnectReason,\n MediaStreamUnregisterReason,\n SessionClientRole\n} from './MediaConstants';\nimport {throwableGuard} from '@techsee/techsee-common/lib/core/guards';\n\nimport {getMediaTracer} from './MediaUtils/MediaTracer';\n\nconst trace = getMediaTracer('MediaServiceBase');\nconst traceStatsInfo = debounce(trace.info, 1000 * 5, {leading: true, maxWait: 1000 * 30});\n\nenum privateEvents {\n STREAM_CREATED = 'STREAM_CREATED',\n STREAM_DESTROYED = 'STREAM_DESTROYED'\n}\n\nexport abstract class TechseeMediaServiceBase {\n //This method should not be called. The only place that allowed to use it, is initLocalMediaStreams method.\n //Everyone who need to initialize MediaStream should use initLocalMediaStreams method.\n protected abstract getLocalMediaImplementation(): Promise<void>;\n\n protected abstract createMediaPublisher(destinationRole: SessionClientRole): Promise<Nullable<TechseeMediaPublisher>>;\n\n protected readonly _environment: IMediaEnvironment;\n protected readonly _sessionStreamsManager: ISessionStreamsManager;\n protected readonly _localStreamsManager: LocalStreamManager;\n\n private readonly _deviceSupportFlags: DeviceSupportInfo;\n private readonly _subscribers: Map<HTMLDivElement, TechseeMediaSubscriber> = new Map();\n private readonly _publishers: Map<SessionClientRole, TechseeMediaPublisher> = new Map();\n private readonly _publisherPromises: Map<SessionClientRole, Promise<Nullable<MediaStream>>> = new Map();\n protected _serviceOptions: Nullable<MediaServiceOptions> = null;\n private _isIOS_13_orLater: boolean = false;\n private _autoReconnectEnabled: boolean = true;\n private readonly _emitter: EventEmitter;\n private _initLocalStreamsPromise: Nullable<Promise<void>> = null;\n private _initServicePromise: Nullable<Promise<void>> = null;\n private _registeredStreams: Map<KnownMediaStream, TechseeMediaStream> = new Map();\n private _isVoipEnabled: boolean = false;\n private _session: Nullable<{\n params: MediaSessionParams;\n connectPromise?: Promise<void>;\n instance?: IMediaSession;\n }> = null;\n\n protected constructor(environment: IMediaEnvironment, webRtcSupportInfo: ClientWebRtcInfo) {\n trace.info('TechseeMediaServiceBase created');\n\n this.bindClassMethods();\n\n this._emitter = new EventEmitter();\n this._environment = environment;\n this._deviceSupportFlags = {\n videoPlayback: false,\n hasCamera: false,\n hasMicrophone: false,\n webRtcSupportInfo: webRtcSupportInfo\n };\n\n this._localStreamsManager = new LocalStreamManager(this._environment);\n\n // TODO: Hack as IOS 13.3 and above sometimes do not return the video/audio device. This\n // is a hack to work around the issue until Apple resolve it\n this._isIOS_13_orLater = this._environment.isIOS() &&\n (!this._environment.majorVersion() || (this._environment.majorVersion() >= 13));\n\n trace.info(`XXXXX: this._isIOS_13_orLater: ${this._isIOS_13_orLater}`);\n trace.info(`XXXXX: this._environment.isIOS: ${this._environment.isIOS()}`);\n trace.info(`XXXXX: this._environment.majorVersion(): ${this._environment.majorVersion()}`);\n\n this._sessionStreamsManager = {\n getMediaStreamForRole: this.getStreamForDestinationRole,\n addRemoteMediaTrack: this.registerRemoteMediaTrack,\n removeMediaTrack: this.unregisterRemoteMediaTrack\n };\n }\n\n get deviceSupportInfo(): DeviceSupportInfo {\n this.serviceInitGuard(false);\n\n return cloneDeep(this._deviceSupportFlags);\n }\n\n get isVoipEnabled(): boolean {\n return this._isVoipEnabled;\n }\n\n get isSessionActive(): boolean {\n return this._session !== null;\n }\n\n get isLocalStreamInitialized(): boolean {\n return this._initLocalStreamsPromise !== null;\n }\n\n /*\n Initializes media service. Before using anything from this service it should be initialized first.\n All of serviceOptions should be retrieved from account settings and by evaluating application state.\n For example, when initializing service, accountSettings is already loaded, and we know if FS session is audio enabled.\n */\n initMediaService(serviceOptions: MediaServiceOptions): Promise<void> {\n trace.info('TechseeMediaServiceBase.initMediaService', serviceOptions);\n const warnMessage = 'Multiple initialization does not supposed to happen. Not rejecting, but check the flow for correctness.';\n\n if (this._initServicePromise || this._serviceOptions) {\n console.warn(warnMessage);\n\n return this._initServicePromise ? this._initServicePromise : Promise.resolve();\n }\n\n this._serviceOptions = cloneDeep(serviceOptions);\n const isOpenTok = this._serviceOptions.mediaServiceType === MediaServiceType.OPENTOK;\n\n this._initServicePromise = Promise.all([\n isVideoPlaySupportedOnDevice(isOpenTok)\n .then((isSupported: boolean) => this._deviceSupportFlags.videoPlayback = isSupported),\n enumerateMediaDevices()\n .catch((error) => {\n trace.warn(error);\n\n return [];\n })\n .then((localDevicesList: MediaDeviceInfo[]) => {\n const groupedDevices = {\n video: filter(localDevicesList, (device) => device.kind.toLowerCase() === 'videoinput'),\n audio: filter(localDevicesList, (device) => device.kind.toLowerCase() === 'audioinput')\n };\n\n this._localStreamsManager.setGroupedDevices(groupedDevices);\n this._deviceSupportFlags.hasCamera = this._isIOS_13_orLater || (groupedDevices.video.length > 0);\n this._deviceSupportFlags.hasMicrophone = this._isIOS_13_orLater || (groupedDevices.audio.length > 0);\n\n })\n ]).then(() => undefined);\n\n\n return this._initServicePromise;\n }\n\n //#region Media Streams Management\n\n /*\n The main method to start the initialization of local media streams.\n This method can be executed many times and it will insure all calls are synced and only relevant stream will be created.\n DO NOT USE ANY OTHER METHOD THAT FORCES \"navigator.getUserMedia\" OTHER THAN THIS ONE.\n */\n initLocalMediaStreams(): Promise<void> {\n this.serviceInitGuard();\n\n if (!this._initLocalStreamsPromise) {\n trace.info('initLocalMediaStreams creating media streams');\n this._initLocalStreamsPromise = this.getLocalMediaImplementation()\n .catch((mediaRequestFailResult: MediaRequestFailResult) => {\n trace.error('initLocalMediaStreams failure: ', mediaRequestFailResult);\n\n throw mediaRequestFailResult;\n });\n } else {\n trace.info('initLocalMediaStreams already done or in progress');\n }\n\n return this._initLocalStreamsPromise;\n }\n\n /*\n Creates an instance of subscriber that will \"listen\" to the requested stream.\n Whenever requested stream is available, subscriber will render it.\n */\n createSubscriber(subscriberParams: MediaSubscriberParams): Promise<IMediaSubscriber> {\n if (this._subscribers.has(subscriberParams.container)) {\n return Promise.reject('Subscriber for provided DIV element already exists');\n }\n\n const subscriber = new TechseeMediaSubscriber(subscriberParams);\n\n this._subscribers.set(subscriberParams.container, subscriber);\n const streamForSubscriber = this.getRegisteredStreamByType(subscriber.streamType);\n\n if (streamForSubscriber) {\n trace.info(`Stream for ${subscriberParams.streamType} is ready.`);\n subscriber.renderStream(streamForSubscriber);\n } else {\n trace.info(`Stream for ${subscriberParams.streamType} not created yet`);\n }\n\n return Promise.resolve(subscriber);\n }\n\n /*\n Destroys the subscriber for provided HTML DIV container\n */\n destroySubscriber(container: HTMLDivElement): Promise<void> {\n if (this._subscribers.has(container)) {\n const subscriber = this._subscribers.get(container);\n\n subscriber!.dispose();\n }\n\n this._subscribers.delete(container);\n\n return Promise.resolve();\n }\n\n onStreamDestroyed(callback: (eventArgs: StreamDestroyedEventArgs) => void): void {\n this.registerEventCallback(privateEvents.STREAM_DESTROYED, callback);\n }\n\n onStreamCreated(callback: (eventArgs: StreamCreatedEventArgs) => void): void {\n this.registerEventCallback(privateEvents.STREAM_CREATED, callback);\n }\n\n private getSwitchCameraConstraints() {\n const constraints = cloneDeep(window.latestLocalMediaConstraints);\n\n if (!constraints) {\n throw new Error('getSwitchCameraConstraints: unexpected use case constraints is null.');\n }\n\n trace.info('getSwitchCameraConstraints - Start switch camera with constraints:', constraints);\n\n const videoSourceType = (constraints.video as LocalVideoStreamConstraints).videoSourceType === LocalVideoSourceType.CAMERA\n ? LocalVideoSourceType.CAMERA_FRONT\n : LocalVideoSourceType.CAMERA;\n\n trace.info(`getSwitchCameraConstraints - switch to videoSourceType: ${videoSourceType}`);\n\n (constraints.video as LocalVideoStreamConstraints).videoSourceType = videoSourceType;\n\n return constraints;\n }\n\n switchCamera(isFailed?: boolean) {\n trace.info('switchCamera: start');\n\n const constraints = !isFailed ? this.getSwitchCameraConstraints() : cloneDeep(window.latestLocalMediaConstraints);\n\n if (!constraints) {\n throw new Error('switchCamera - unexpected use case constraints is null.');\n }\n\n this._localStreamsManager.destroyUserMediaStream();\n\n return this._localStreamsManager.getUserMediaStream(constraints)\n .then((streamResult: MediaRequestSuccessResult) => {\n const stream = streamResult.mediaStream;\n\n trace.info('switchCamera: new stream: ', stream);\n\n this.replaceStreamTracks(stream);\n\n return this.registerStreamResult(constraints, streamResult, true).then(() => ({constraints, streamResult}));\n })\n .catch((err) => {\n trace.error('switchCamera: Failed to switch camera: ', err);\n\n if (!isFailed) {\n this.switchCamera(true);\n }\n\n return Promise.reject('Failed to switch camera.');\n });\n }\n\n private replaceStreamTracks(mediaStream: Nullable<MediaStream>) {\n this._session!.instance && this._session!.instance.replaceStreamTracks(mediaStream);\n }\n\n protected registerStreamResult(constraints: Nullable<LocalMediaConstraints>, streamResult: MediaRequestSuccessResult, switchCamera?: boolean, addStreamType?: KnownMediaStream) {\n const regPromises: any[] = [];\n\n if (streamResult.isNew) {\n trace.info('registerStreamResult: stream result from getUserMediaStream:', streamResult.mediaStream);\n streamResult.mediaStream.getTracks().forEach((mediaTrack) => {\n trace.info('registerStreamResult: stream result from mediaTrack:', mediaTrack);\n const streamType = addStreamType || (mediaTrack.kind === 'video'\n ? KnownMediaStream.USER_VIDEO_STREAM : KnownMediaStream.USER_AUDIO_STREAM);\n const newDedicatedStream = new TechseeMediaStream(mediaTrack, streamType, false);\n\n regPromises.push(switchCamera ? this.registerTrack(newDedicatedStream) : this.registerLocalMediaStream(newDedicatedStream));\n });\n }\n\n return Promise.all(regPromises);\n }\n\n get supportSwitchCameras() {\n return this._isIOS_13_orLater || this._localStreamsManager.groupedDevices.camerasCount > 1;\n }\n\n //#endregion Media Streams Management\n\n //#region Session Management\n\n /*\n Creates instance of WebRTC session which connects to (signaling server), and begins to listen to WebRTC events.\n */\n connectToSession(sessionParams: MediaSessionParams): Promise<void> {\n this.serviceInitGuard();\n\n if (!this._session) {\n this._session = {params: cloneDeep(sessionParams)};\n this._session.connectPromise = new Promise((resolve, reject) => {\n const doAsyncReject = (error: any): void => {\n\n /*\n Reject with timeout, to change JS 'execution flow'.\n If timeout removed, session is assigned to null, before promise returns, and \"catch\" will not\n work on promise that is returned.\n */\n setTimeout(() => {\n this._session = null;\n reject(error);\n });\n };\n\n const sessionHandlers: MediaSessionHandlers = {\n onDisconnectedHandler: this.onSessionDisconnectHandler\n };\n const mediaSessionParams: MediaSessionParams & MediaSessionConfiguration = {\n ...sessionParams,\n peerConnectivityTimeoutSeconds: this._serviceOptions!.peerConnectivityTimeoutSeconds || DEFAULT_PEER_CONNECTIVITY_TIMEOUT_SECONDS\n };\n\n trace.info('connectToSession', sessionParams);\n if (this._serviceOptions!.mediaServiceType === MediaServiceType.TURNSERVER) {\n this._session!.instance = new TurnWebRtcSession(mediaSessionParams, sessionHandlers, this._sessionStreamsManager);\n\n } else if (this._serviceOptions!.mediaServiceType === MediaServiceType.OPENTOK) {\n this._session!.instance = new OpentokSession(mediaSessionParams, this._sessionStreamsManager);\n } else {\n const allowedValues = values(MediaServiceType);\n\n trace.error(`mediaServiceType '${this._serviceOptions!.mediaServiceType}' is not supported.`, {allowedValues});\n doAsyncReject(new Error('mediaServiceType is not supported'));\n }\n\n if (this._session && this._session.instance) {\n this._session!.instance.connect()\n .then(() => {\n trace.info('Session is connected');\n resolve();\n })\n .catch((error: any) => {\n trace.error('Failed to connect to session', error);\n doAsyncReject(error);\n });\n }\n\n });\n } else {\n trace.warn('Session connected already. Disconnect before connect to new session again');\n }\n\n //TODO - Alex: Change a structure of this function to be more safe\n //(when we return, there should not be case when session or promise is null)\n //To fix, need to return promise immediately, then execute process in setTimeout(fn, 0);\n return this._session!.connectPromise!;\n }\n\n /*\n Disconnects from WebRTC session.\n */\n disconnectFromSession(): Promise<void> {\n return this.disconnectFromSessionInternal(MediaSessionDisconnectReason.ForcedByConsumer);\n }\n\n /*\n Updates a credentials for turn server\n */\n updateSessionCredentials(credentials: MediaSessionIceCredentials): Promise<void> {\n trace.info('updateSessionCredentials');\n if (!this.sessionExistsGuard(false)) {\n return Promise.reject('There no session to update credentials');\n }\n\n this._session!.params.credentials = cloneDeep(credentials);\n\n return Promise.resolve();\n }\n\n enableVoipDuringSession(): void {\n if (!this._isVoipEnabled && this.isLocalStreamInitialized) {\n throw new Error('Voip support cannot be enabled after local stream where initialized');\n }\n\n this._isVoipEnabled = true;\n }\n\n protected disconnectFromSessionInternal(reason: MediaSessionDisconnectReason): Promise<void> {\n if (!this._session) {\n return Promise.resolve();\n }\n\n trace.info('Disconnecting from session', reason);\n\n const lastSession = this._session;\n\n this._session = null;\n\n return Promise.all([\n lastSession.instance!.disconnect(),\n this.clearPublishers()\n ]).then(() => undefined);\n }\n\n protected getStatsForRemoteTrack(streamType: KnownMediaStream): Promise<RemoteTrackStats> {\n if (!this.sessionExistsGuard(false)) {\n return Promise.reject(new Error('Media session not started'));\n }\n\n const streamForStats = this.getRegisteredStreamByType(streamType);\n\n if (!streamForStats) {\n return Promise.reject(new Error('Stream for requested type was not found'));\n }\n\n return this._session!.instance!.getRemoteTrackStats(streamForStats.mediaTrack)\n .then((trackStats: RemoteTrackStats) => {\n traceStatsInfo(`MediaTrackStats for ${streamType}: ${JSON.stringify(trackStats)}`);\n\n return {\n trackId: streamForStats.mediaTrack.id,\n trackStats: trackStats\n };\n });\n }\n\n private onSessionDisconnectHandler(reason: MediaSessionDisconnectReason): void {\n trace.info('onSessionDisconnectHandler', reason);\n if (reason !== MediaSessionDisconnectReason.ForcedByConsumer) {\n const lastParams = this._session && this._session.params ? this._session.params : null;\n\n this.disconnectFromSessionInternal(reason).then(() => {\n const reconnectReasons = [\n MediaSessionDisconnectReason.InitiatorPeerReconnected,\n MediaSessionDisconnectReason.SignalingChannelDisconnect,\n MediaSessionDisconnectReason.PeerConnectionInterrupted,\n MediaSessionDisconnectReason.PeerConnectionStateChangeTimeout\n ];\n\n if (lastParams && includes(reconnectReasons, reason) && this._autoReconnectEnabled) {\n this.reconnectToSession(lastParams);\n } else if (!this._autoReconnectEnabled) {\n trace.info('No reconnection- auto reconnect disabled');\n } else {\n trace.info('No params for reconnection to media session');\n }\n });\n } else {\n trace.info('Ignore Session disconnect event');\n }\n }\n\n setAutoReconnect(state: boolean) {\n this._autoReconnectEnabled = state;\n trace.info('_autoReconnectEnabled:', state);\n\n if (this._autoReconnectEnabled && !this.isSessionActive) {\n const lastParams = this._session && this._session.params ? this._session.params : null;\n\n if (lastParams) {\n trace.info('setAutoReconnect - reconnect to session');\n this.reconnectToSession(lastParams);\n }\n }\n }\n\n reconnectToSession(lastParams: MediaSessionParams) {\n trace.info('Reconnecting to media session, sessionParams:', lastParams);\n\n this.connectToSession(lastParams)\n .then(() => {\n trace.info('Media session reconnected');\n })\n .catch((error: any) => {\n trace.error('Error while reconnecting to media session', error);\n });\n }\n\n //#endregion Session Management\n\n //#region Utils\n\n getSnapshotFromKnownStream(sourceStream: KnownMediaStream, snapshotOptions?: SnapshotOptions): Promise<SnapshotResult> {\n return new Promise((resolve, reject) => {\n if (sourceStream !== KnownMediaStream.USER_VIDEO_STREAM &&\n sourceStream !== KnownMediaStream.USER_SCREEN_SHARE_STREAM) {\n trace.error('The requested stream is not video stream, and cannot be used for snapshot');\n reject(new Error('INCOMPATIBLE_STREAM_FOR_SNAPSHOT'));\n\n return;\n }\n\n const snapshotStream = this.getRegisteredStreamByType(sourceStream);\n\n if (!snapshotStream) {\n trace.error('Cannot make snapshot: The requested stream not exists yet.');\n reject(new Error('NO_REQUESTED_STREAM'));\n\n return;\n }\n\n getSnapshotFromMediaStream(snapshotStream.mediaStream, snapshotOptions)\n .then((imageData: any) => {\n const urlComponents = imageData.split(';base64,');\n const mimeType = urlComponents[0].split(':')[1];\n const bytes = atob(urlComponents[1]);\n const buffer = new ArrayBuffer(bytes.length);\n const rawData = new Uint8Array(buffer);\n\n for (let i = 0; i < bytes.length; i++) {\n rawData[i] = bytes.charCodeAt(i);\n }\n\n const blob = new Blob([rawData], {type: mimeType});\n const objectUrl = window.URL.createObjectURL(blob);\n\n trace.info('Snapshot created successfully');\n\n const result: SnapshotResult = {\n base64img: imageData,\n objectUrl: objectUrl,\n imageBlob: blob,\n mimeType: mimeType\n };\n\n resolve(result);\n })\n .catch((error: any) => {\n trace.error('Error creating snapshot', error);\n reject(error);\n });\n });\n }\n\n //Will clean streams, publishers and subscribers, but will not remove event listeners\n clearService(): Promise<void> {\n trace.info('MediaService clearing all resources');\n\n return this.disconnectFromSessionInternal(MediaSessionDisconnectReason.ForcedByConsumer)\n .then(this.clearSubscribers)\n .then(this.clearRegisteredStreams)\n .then(this._localStreamsManager.clearAllStreams)\n .then(() => {\n this._initLocalStreamsPromise = null;\n this._isVoipEnabled = false;\n }).then(() => undefined);\n }\n\n //Will clear the service and remove all event listeners\n dispose(): Promise<void> {\n return this.clearService().catch(() => undefined).then(() => {\n this._emitter.removeAllListeners();\n });\n }\n\n //#endregion\n\n //#region Protected Methods\n\n protected get subscribers(): Map<HTMLDivElement, TechseeMediaSubscriber> {\n return this._subscribers;\n }\n\n protected registerLocalMediaStream(tsMediaStream: TechseeMediaStream): Promise<void> {\n trace.info('Registering local stream', tsMediaStream.streamType);\n if (this._registeredStreams.has(tsMediaStream.streamType)) {\n return Promise.reject(new Error(`Stream ${tsMediaStream.streamType} already registered`));\n }\n\n return this.registerStream(tsMediaStream);\n }\n\n protected getRegisteredStreamByType(streamType: KnownMediaStream): Nullable<TechseeMediaStream> {\n return this._registeredStreams.get(streamType) || null;\n }\n\n protected changeEnableForKnownStream(streamType: KnownMediaStream, isPaused: boolean): void {\n const streamToChangeState = this.getRegisteredStreamByType(streamType);\n\n if (!streamToChangeState) {\n trace.warn('There no stream found to change enable state', streamType);\n } else {\n streamToChangeState.mediaTrack.enabled = !isPaused;\n\n if (streamToChangeState.isRemote && streamToChangeState.streamKind === KnownMediaStreamKind.Audio) {\n this.subscribers.forEach((subscriber) => {\n if (subscriber.streamType === streamType) {\n subscriber.muteSound(isPaused);\n }\n });\n }\n\n trace.info('Local stream enable state is changed', {streamType, isPaused});\n }\n }\n\n protected registerEventCallback(event: string, callback: (eventArgs?: any) => void): void {\n this._emitter.on(event, callback);\n }\n\n protected emitEvent(event: string, eventArgs?: any): void {\n setTimeout(() => {\n this._emitter.emit(event, eventArgs);\n });\n }\n\n //#endregion Protected Methods\n\n //#region Private Methods\n\n private registerRemoteMediaTrack(remoteMediaTrack: RemoteMediaTrack): Promise<void> {\n trace.info(`Registering remote ${remoteMediaTrack.trackType} MediaStreamTrack`, remoteMediaTrack.mediaTrack);\n const currentStream = this._registeredStreams.get(remoteMediaTrack.trackType);\n\n if (currentStream && !currentStream.isRemote) {\n return Promise.reject(new Error('Cannot register remote stream with the same type as local stream'));\n }\n\n const newDedicatedStream = new TechseeMediaStream(remoteMediaTrack.mediaTrack, remoteMediaTrack.trackType, true);\n\n return this.registerStream(newDedicatedStream);\n }\n\n private registerTrack(mediaStream: TechseeMediaStream): Promise<void> {\n mediaStream.mediaTrack.onended = () =>\n this.unregisterTechseeMediaStream(mediaStream, MediaStreamUnregisterReason.NativeEvent);\n\n this._registeredStreams.set(mediaStream.streamType, mediaStream);\n\n trace.info(`TechseeMediaStream registered - ${mediaStream.streamType}`, TechseeMediaStream);\n\n return this.updateSubscribersWithNewStream(mediaStream).then(() => {\n const eventArgs: StreamCreatedEventArgs = {streamType: mediaStream.streamType};\n\n this.emitEvent(privateEvents.STREAM_CREATED, eventArgs);\n });\n }\n\n private registerStream(mediaStream: TechseeMediaStream): Promise<void> {\n const currentStream = this._registeredStreams.get(mediaStream.streamType);\n\n if (currentStream) {\n return this.unregisterTechseeMediaStream(currentStream, MediaStreamUnregisterReason.ReplacingStream).then(() => this.registerTrack(mediaStream));\n }\n\n return this.registerTrack(mediaStream);\n }\n\n private unregisterTechseeMediaStream(streamToUnregister: TechseeMediaStream, reason: MediaStreamUnregisterReason): Promise<void> {\n const traceError = (error: any): void => {\n trace.warn('Unregister TechseeMediaStream error:', error);\n };\n\n const promises: any = [];\n\n trace.info('Unregister TechseeMediaStream: ', streamToUnregister.streamType);\n\n this._subscribers.forEach((subscriber) => {\n if (subscriber.streamType === streamToUnregister.streamType) {\n promises.push(subscriber.stopRendering().catch(traceError));\n }\n });\n\n if (!streamToUnregister.isRemote) {\n this._publishers.forEach((publisher: TechseeMediaPublisher) => {\n if (publisher.streamTypes.find((streamType) => streamType === streamToUnregister.streamType)) {\n if (this._session && this._session!.instance) {\n promises.push(this._session.instance.onMediaStreamDestroyed(publisher.destinationRole).catch(traceError));\n }\n }\n });\n }\n\n return Promise.all(promises)\n .then(() => {\n this._registeredStreams.delete(streamToUnregister.streamType);\n trace.info('Stream deleted', streamToUnregister.streamType);\n\n const eventArgs: StreamDestroyedEventArgs = {streamType: streamToUnregister.streamType, reason: reason};\n\n this.emitEvent(privateEvents.STREAM_DESTROYED, eventArgs);\n });\n }\n\n private unregisterRemoteMediaTrack(mediaTrack: MediaStreamTrack): Promise<void> {\n const promises: any = [];\n\n this._registeredStreams.forEach((registeredStream: TechseeMediaStream) => {\n if (registeredStream.mediaTrack.id === mediaTrack.id) {\n promises.push(this.unregisterTechseeMediaStream(registeredStream, MediaStreamUnregisterReason.ClosedRemotely));\n }\n });\n\n return Promise.all(promises).then(() => undefined);\n }\n\n private removePublisher(publisher: TechseeMediaPublisher): Promise<void> {\n this._publishers.delete(publisher.destinationRole);\n if (this._session && this._session.instance) {\n trace.info('Removing publisher', publisher.destinationRole);\n\n return this._session.instance.onMediaStreamDestroyed(publisher.destinationRole)\n .catch((error: any) => {\n trace.warn('Error while removing publisher', error);\n });\n }\n\n return Promise.resolve();\n }\n\n private getStreamForDestinationRole(destinationRole: SessionClientRole): Promise<Nullable<MediaStream>> {\n this.serviceInitGuard();\n\n if (this._publishers.has(destinationRole)) {\n trace.info(`Publisher for ${destinationRole} already exists`);\n\n return Promise.resolve(this._publishers.get(destinationRole)!.mediaStream);\n }\n\n if (!this._publisherPromises.get(destinationRole)) {\n trace.info(`Creating publisher for ${destinationRole}`);\n const publisherPromise = this.initLocalMediaStreams()\n .then(() => this.createMediaPublisher(destinationRole))\n .then((mediaPublisher: Nullable<TechseeMediaPublisher>) => {\n if (mediaPublisher) {\n this._publishers.set(destinationRole, mediaPublisher);\n }\n\n this._publisherPromises.delete(destinationRole);\n\n return mediaPublisher ? mediaPublisher.mediaStream : null;\n })\n .catch((ex: any) => {\n if (ex && ex.message === 'audioStreamFailed') {\n this._isVoipEnabled = false;\n\n return null;\n }\n\n this._publisherPromises.delete(destinationRole);\n throw ex;\n });\n\n this._publisherPromises.set(destinationRole, publisherPromise);\n } else {\n trace.info(`Create publisher promise for ${destinationRole} already exists`);\n }\n\n return this._publisherPromises.get(destinationRole)!;\n }\n\n private updateSubscribersWithNewStream(registeredStream: TechseeMediaStream): Promise<void> {\n trace.info(`Updating subscribers of ${registeredStream.streamType} with new stream`);\n\n if (this._subscribers.size === 0) {\n trace.warn(`No subscribers exists for the ${registeredStream.streamType}.`);\n } else {\n trace.info(`Total ${this._subscribers.size} subscribers exists, will check if rerender needed.`);\n }\n\n this._subscribers.forEach((subscriber) => {\n if (subscriber.streamType === registeredStream.streamType) {\n trace.info(`${registeredStream.streamType} rendering on subscriber`);\n subscriber.renderStream(registeredStream);\n }\n });\n\n return Promise.resolve();\n }\n\n private bindClassMethods(): void {\n this.updateSessionCredentials = this.updateSessionCredentials.bind(this);\n this.getStreamForDestinationRole = this.getStreamForDestinationRole.bind(this);\n this.registerRemoteMediaTrack = this.registerRemoteMediaTrack.bind(this);\n this.unregisterRemoteMediaTrack = this.unregisterRemoteMediaTrack.bind(this);\n this.createMediaPublisher = this.createMediaPublisher.bind(this);\n this.updateSubscribersWithNewStream = this.updateSubscribersWithNewStream.bind(this);\n this.initLocalMediaStreams = this.initLocalMediaStreams.bind(this);\n this.onSessionDisconnectHandler = this.onSessionDisconnectHandler.bind(this);\n this.clearService = this.clearService.bind(this);\n this.clearRegisteredStreams = this.clearRegisteredStreams.bind(this);\n this.clearPublishers = this.clearPublishers.bind(this);\n this.clearSubscribers = this.clearSubscribers.bind(this);\n }\n\n //region Cleanup\n\n private clearRegisteredStreams(): Promise<void> {\n trace.info('Clearing registered streams');\n const promises: any = [];\n\n this._registeredStreams.forEach((streamToUnregister: TechseeMediaStream) => {\n promises.push(this.unregisterTechseeMediaStream(streamToUnregister, MediaStreamUnregisterReason.ServiceCleanUp));\n });\n\n return Promise.all(promises).then(() => undefined);\n }\n\n private clearSubscribers(): Promise<void> {\n trace.info('Clearing subscribers');\n const promises: any = [];\n\n this._subscribers.forEach((subscriber) => {\n promises.push(this.destroySubscriber(subscriber.container));\n });\n\n return Promise.all(promises).then(() => undefined);\n }\n\n private clearPublishers(): Promise<void> {\n trace.info('Clearing publishers');\n const promises: any = [];\n\n this._publishers.forEach((publisher) => {\n promises.push(this.removePublisher(publisher));\n });\n\n return Promise.all(promises).then(() => undefined);\n }\n\n //#endregion Cleanup\n\n //#region Simple Validation Methods\n\n protected serviceInitGuard(shouldThrow = true): boolean {\n return throwableGuard(!!this._serviceOptions, 'Media service is not initialized', shouldThrow);\n }\n\n protected sessionExistsGuard(shouldThrow = true): boolean {\n return throwableGuard(!!this._session, 'There no active session', shouldThrow);\n }\n\n //#endregion\n\n //#endregion Private Methods\n}\n"]}
|
package/lib/MediaServiceBase.js
CHANGED
|
@@ -66,7 +66,9 @@ var TechseeMediaServiceBase = /** @class */function () {
|
|
|
66
66
|
// TODO: Hack as IOS 13.3 and above sometimes do not return the video/audio device. This
|
|
67
67
|
// is a hack to work around the issue until Apple resolve it
|
|
68
68
|
this._isIOS_13_orLater = this._environment.isIOS() && (!this._environment.majorVersion() || this._environment.majorVersion() >= 13);
|
|
69
|
-
trace.info("
|
|
69
|
+
trace.info("XXXXX: this._isIOS_13_orLater: " + this._isIOS_13_orLater);
|
|
70
|
+
trace.info("XXXXX: this._environment.isIOS: " + this._environment.isIOS());
|
|
71
|
+
trace.info("XXXXX: this._environment.majorVersion(): " + this._environment.majorVersion());
|
|
70
72
|
this._sessionStreamsManager = {
|
|
71
73
|
getMediaStreamForRole: this.getStreamForDestinationRole,
|
|
72
74
|
addRemoteMediaTrack: this.registerRemoteMediaTrack,
|
|
@@ -194,36 +196,36 @@ var TechseeMediaServiceBase = /** @class */function () {
|
|
|
194
196
|
TechseeMediaServiceBase.prototype.onStreamCreated = function (callback) {
|
|
195
197
|
this.registerEventCallback(privateEvents.STREAM_CREATED, callback);
|
|
196
198
|
};
|
|
197
|
-
TechseeMediaServiceBase.prototype.getSwitchCameraConstraints = function (
|
|
198
|
-
var
|
|
199
|
-
if (!
|
|
199
|
+
TechseeMediaServiceBase.prototype.getSwitchCameraConstraints = function () {
|
|
200
|
+
var constraints = cloneDeep_1.default(window.latestLocalMediaConstraints);
|
|
201
|
+
if (!constraints) {
|
|
200
202
|
throw new Error('getSwitchCameraConstraints: unexpected use case constraints is null.');
|
|
201
203
|
}
|
|
202
|
-
trace.info('getSwitchCameraConstraints - Start switch camera with constraints:',
|
|
203
|
-
var videoSourceType =
|
|
204
|
+
trace.info('getSwitchCameraConstraints - Start switch camera with constraints:', constraints);
|
|
205
|
+
var videoSourceType = constraints.video.videoSourceType === MediaConstants_1.LocalVideoSourceType.CAMERA ? MediaConstants_1.LocalVideoSourceType.CAMERA_FRONT : MediaConstants_1.LocalVideoSourceType.CAMERA;
|
|
204
206
|
trace.info("getSwitchCameraConstraints - switch to videoSourceType: " + videoSourceType);
|
|
205
|
-
|
|
207
|
+
constraints.video.videoSourceType = videoSourceType;
|
|
206
208
|
return constraints;
|
|
207
209
|
};
|
|
208
|
-
TechseeMediaServiceBase.prototype.switchCamera = function (
|
|
210
|
+
TechseeMediaServiceBase.prototype.switchCamera = function (isFailed) {
|
|
209
211
|
var _this = this;
|
|
210
212
|
trace.info('switchCamera: start');
|
|
211
|
-
var
|
|
212
|
-
if (!
|
|
213
|
+
var constraints = !isFailed ? this.getSwitchCameraConstraints() : cloneDeep_1.default(window.latestLocalMediaConstraints);
|
|
214
|
+
if (!constraints) {
|
|
213
215
|
throw new Error('switchCamera - unexpected use case constraints is null.');
|
|
214
216
|
}
|
|
215
217
|
this._localStreamsManager.destroyUserMediaStream();
|
|
216
|
-
return this._localStreamsManager.getUserMediaStream(
|
|
218
|
+
return this._localStreamsManager.getUserMediaStream(constraints).then(function (streamResult) {
|
|
217
219
|
var stream = streamResult.mediaStream;
|
|
218
220
|
trace.info('switchCamera: new stream: ', stream);
|
|
219
221
|
_this.replaceStreamTracks(stream);
|
|
220
|
-
return _this.registerStreamResult(
|
|
221
|
-
return {
|
|
222
|
+
return _this.registerStreamResult(constraints, streamResult, true).then(function () {
|
|
223
|
+
return { constraints: constraints, streamResult: streamResult };
|
|
222
224
|
});
|
|
223
225
|
}).catch(function (err) {
|
|
224
226
|
trace.error('switchCamera: Failed to switch camera: ', err);
|
|
225
227
|
if (!isFailed) {
|
|
226
|
-
_this.switchCamera(
|
|
228
|
+
_this.switchCamera(true);
|
|
227
229
|
}
|
|
228
230
|
return Promise.reject('Failed to switch camera.');
|
|
229
231
|
});
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["MediaServiceBase.js","../src/MediaServiceBase.ts"],"names":["__assign","Object","assign","t","s","i","n","arguments","length","p","prototype","hasOwnProperty","call","apply","__importDefault","mod","__esModule","defineProperty","exports","value","filter_1","require","cloneDeep_1","values_1","debounce_1","includes_1","events_1","MediaSubscriber_1","LocalStreamManager_1","TechseeMediaStream_1","SessionTurn_1","MediaDomUtils_1","Compatibility_1","SessionOpentok_1","MediaConstants_1","guards_1","MediaTracer_1","trace","getMediaTracer","traceStatsInfo","default","info","leading","maxWait","privateEvents","TechseeMediaServiceBase","environment","webRtcSupportInfo","_subscribers","Map","_publishers","_publisherPromises","_serviceOptions","_isIOS_13_orLater","_autoReconnectEnabled","_initLocalStreamsPromise","_initServicePromise","_registeredStreams","_isVoipEnabled","_session","bindClassMethods","_emitter","EventEmitter","_environment","_deviceSupportFlags","videoPlayback","hasCamera","hasMicrophone","_localStreamsManager","LocalStreamManager","isIOS","majorVersion","_sessionStreamsManager","getMediaStreamForRole","getStreamForDestinationRole","addRemoteMediaTrack","registerRemoteMediaTrack","removeMediaTrack","unregisterRemoteMediaTrack","get","serviceInitGuard","enumerable","configurable","initMediaService","serviceOptions","_this","warnMessage","console","warn","Promise","resolve","isOpenTok","mediaServiceType","MediaServiceType","OPENTOK","all","isVideoPlaySupportedOnDevice","then","isSupported","enumerateMediaDevices","catch","error","localDevicesList","groupedDevices","video","device","kind","toLowerCase","audio","setGroupedDevices","undefined","initLocalMediaStreams","getLocalMediaImplementation","mediaRequestFailResult","createSubscriber","subscriberParams","has","container","reject","subscriber","TechseeMediaSubscriber","set","streamForSubscriber","getRegisteredStreamByType","streamType","renderStream","destroySubscriber","dispose","delete","onStreamDestroyed","callback","registerEventCallback","STREAM_DESTROYED","onStreamCreated","STREAM_CREATED","getSwitchCameraConstraints","constraints","newConstraints","Error","videoSourceType","LocalVideoSourceType","CAMERA","CAMERA_FRONT","switchCamera","isFailed","destroyUserMediaStream","getUserMediaStream","streamResult","stream","mediaStream","replaceStreamTracks","registerStreamResult","err","instance","addStreamType","regPromises","isNew","getTracks","forEach","mediaTrack","KnownMediaStream","USER_VIDEO_STREAM","USER_AUDIO_STREAM","newDedicatedStream","TechseeMediaStream","push","registerTrack","registerLocalMediaStream","camerasCount","connectToSession","sessionParams","params","connectPromise","doAsyncReject","setTimeout","sessionHandlers","onDisconnectedHandler","onSessionDisconnectHandler","mediaSessionParams","peerConnectivityTimeoutSeconds","DEFAULT_PEER_CONNECTIVITY_TIMEOUT_SECONDS","TURNSERVER","TurnWebRtcSession","OpentokSession","allowedValues","connect","disconnectFromSession","disconnectFromSessionInternal","MediaSessionDisconnectReason","ForcedByConsumer","updateSessionCredentials","credentials","sessionExistsGuard","enableVoipDuringSession","isLocalStreamInitialized","reason","lastSession","disconnect","clearPublishers","getStatsForRemoteTrack","streamForStats","getRemoteTrackStats","trackStats","JSON","stringify","trackId","id","lastParams_1","reconnectReasons","InitiatorPeerReconnected","SignalingChannelDisconnect","PeerConnectionInterrupted","PeerConnectionStateChangeTimeout","reconnectToSession","setAutoReconnect","state","isSessionActive","lastParams","getSnapshotFromKnownStream","sourceStream","snapshotOptions","USER_SCREEN_SHARE_STREAM","snapshotStream","getSnapshotFromMediaStream","imageData","urlComponents","split","mimeType","bytes","atob","buffer","ArrayBuffer","rawData","Uint8Array","charCodeAt","blob","Blob","type","objectUrl","window","URL","createObjectURL","result","base64img","imageBlob","clearService","clearSubscribers","clearRegisteredStreams","clearAllStreams","removeAllListeners","tsMediaStream","registerStream","changeEnableForKnownStream","isPaused","streamToChangeState","enabled","isRemote","streamKind","KnownMediaStreamKind","Audio","subscribers","muteSound","event","on","emitEvent","eventArgs","emit","remoteMediaTrack","trackType","currentStream","onended","unregisterTechseeMediaStream","MediaStreamUnregisterReason","NativeEvent","updateSubscribersWithNewStream","ReplacingStream","streamToUnregister","traceError","promises","stopRendering","publisher","streamTypes","find","onMediaStreamDestroyed","destinationRole","registeredStream","ClosedRemotely","removePublisher","publisherPromise","createMediaPublisher","mediaPublisher","ex","message","size","bind","ServiceCleanUp","shouldThrow","throwableGuard"],"mappings":"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","file":"MediaServiceBase.js","sourcesContent":["\"use strict\";\nvar __assign = (this && this.__assign) || function () {\n __assign = Object.assign || function(t) {\n for (var s, i = 1, n = arguments.length; i < n; i++) {\n s = arguments[i];\n for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p))\n t[p] = s[p];\n }\n return t;\n };\n return __assign.apply(this, arguments);\n};\nvar __importDefault = (this && this.__importDefault) || function (mod) {\n return (mod && mod.__esModule) ? mod : { \"default\": mod };\n};\nObject.defineProperty(exports, \"__esModule\", { value: true });\nvar filter_1 = __importDefault(require(\"lodash/filter\"));\nvar cloneDeep_1 = __importDefault(require(\"lodash/cloneDeep\"));\nvar values_1 = __importDefault(require(\"lodash/values\"));\nvar debounce_1 = __importDefault(require(\"lodash/debounce\"));\nvar includes_1 = __importDefault(require(\"lodash/includes\"));\nvar events_1 = require(\"events\");\nvar MediaSubscriber_1 = require(\"./MediaSubscriber\");\nvar LocalStreamManager_1 = require(\"./LocalStreamManager\");\nvar TechseeMediaStream_1 = require(\"./TechseeMediaStream\");\nvar SessionTurn_1 = require(\"./MediaSession/SessionTurn\");\nvar MediaDomUtils_1 = require(\"./MediaUtils/MediaDomUtils\");\nvar Compatibility_1 = require(\"./MediaUtils/Compatibility\");\nvar SessionOpentok_1 = require(\"./MediaSession/SessionOpentok\");\nvar MediaConstants_1 = require(\"./MediaConstants\");\nvar guards_1 = require(\"@techsee/techsee-common/lib/core/guards\");\nvar MediaTracer_1 = require(\"./MediaUtils/MediaTracer\");\nvar trace = MediaTracer_1.getMediaTracer('MediaServiceBase');\nvar traceStatsInfo = debounce_1.default(trace.info, 1000 * 5, { leading: true, maxWait: 1000 * 30 });\nvar privateEvents;\n(function (privateEvents) {\n privateEvents[\"STREAM_CREATED\"] = \"STREAM_CREATED\";\n privateEvents[\"STREAM_DESTROYED\"] = \"STREAM_DESTROYED\";\n})(privateEvents || (privateEvents = {}));\nvar TechseeMediaServiceBase = /** @class */ (function () {\n function TechseeMediaServiceBase(environment, webRtcSupportInfo) {\n this._subscribers = new Map();\n this._publishers = new Map();\n this._publisherPromises = new Map();\n this._serviceOptions = null;\n this._isIOS_13_orLater = false;\n this._autoReconnectEnabled = true;\n this._initLocalStreamsPromise = null;\n this._initServicePromise = null;\n this._registeredStreams = new Map();\n this._isVoipEnabled = false;\n this._session = null;\n trace.info('TechseeMediaServiceBase created');\n this.bindClassMethods();\n this._emitter = new events_1.EventEmitter();\n this._environment = environment;\n this._deviceSupportFlags = {\n videoPlayback: false,\n hasCamera: false,\n hasMicrophone: false,\n webRtcSupportInfo: webRtcSupportInfo\n };\n this._localStreamsManager = new LocalStreamManager_1.LocalStreamManager(this._environment);\n // TODO: Hack as IOS 13.3 and above sometimes do not return the video/audio device. This\n // is a hack to work around the issue until Apple resolve it\n this._isIOS_13_orLater = this._environment.isIOS() &&\n (!this._environment.majorVersion() || (this._environment.majorVersion() >= 13));\n trace.info(\"TechseeMediaServiceBase: isIOS_13_orLater: \" + this._isIOS_13_orLater);\n this._sessionStreamsManager = {\n getMediaStreamForRole: this.getStreamForDestinationRole,\n addRemoteMediaTrack: this.registerRemoteMediaTrack,\n removeMediaTrack: this.unregisterRemoteMediaTrack\n };\n }\n Object.defineProperty(TechseeMediaServiceBase.prototype, \"deviceSupportInfo\", {\n get: function () {\n this.serviceInitGuard(false);\n return cloneDeep_1.default(this._deviceSupportFlags);\n },\n enumerable: true,\n configurable: true\n });\n Object.defineProperty(TechseeMediaServiceBase.prototype, \"isVoipEnabled\", {\n get: function () {\n return this._isVoipEnabled;\n },\n enumerable: true,\n configurable: true\n });\n Object.defineProperty(TechseeMediaServiceBase.prototype, \"isSessionActive\", {\n get: function () {\n return this._session !== null;\n },\n enumerable: true,\n configurable: true\n });\n Object.defineProperty(TechseeMediaServiceBase.prototype, \"isLocalStreamInitialized\", {\n get: function () {\n return this._initLocalStreamsPromise !== null;\n },\n enumerable: true,\n configurable: true\n });\n /*\n Initializes media service. Before using anything from this service it should be initialized first.\n All of serviceOptions should be retrieved from account settings and by evaluating application state.\n For example, when initializing service, accountSettings is already loaded, and we know if FS session is audio enabled.\n */\n TechseeMediaServiceBase.prototype.initMediaService = function (serviceOptions) {\n var _this = this;\n trace.info('TechseeMediaServiceBase.initMediaService', serviceOptions);\n var warnMessage = 'Multiple initialization does not supposed to happen. Not rejecting, but check the flow for correctness.';\n if (this._initServicePromise || this._serviceOptions) {\n console.warn(warnMessage);\n return this._initServicePromise ? this._initServicePromise : Promise.resolve();\n }\n this._serviceOptions = cloneDeep_1.default(serviceOptions);\n var isOpenTok = this._serviceOptions.mediaServiceType === MediaConstants_1.MediaServiceType.OPENTOK;\n this._initServicePromise = Promise.all([\n Compatibility_1.isVideoPlaySupportedOnDevice(isOpenTok)\n .then(function (isSupported) { return _this._deviceSupportFlags.videoPlayback = isSupported; }),\n Compatibility_1.enumerateMediaDevices()\n .catch(function (error) {\n trace.warn(error);\n return [];\n })\n .then(function (localDevicesList) {\n var groupedDevices = {\n video: filter_1.default(localDevicesList, function (device) { return device.kind.toLowerCase() === 'videoinput'; }),\n audio: filter_1.default(localDevicesList, function (device) { return device.kind.toLowerCase() === 'audioinput'; })\n };\n _this._localStreamsManager.setGroupedDevices(groupedDevices);\n _this._deviceSupportFlags.hasCamera = _this._isIOS_13_orLater || (groupedDevices.video.length > 0);\n _this._deviceSupportFlags.hasMicrophone = _this._isIOS_13_orLater || (groupedDevices.audio.length > 0);\n })\n ]).then(function () { return undefined; });\n return this._initServicePromise;\n };\n //#region Media Streams Management\n /*\n The main method to start the initialization of local media streams.\n This method can be executed many times and it will insure all calls are synced and only relevant stream will be created.\n DO NOT USE ANY OTHER METHOD THAT FORCES \"navigator.getUserMedia\" OTHER THAN THIS ONE.\n */\n TechseeMediaServiceBase.prototype.initLocalMediaStreams = function () {\n this.serviceInitGuard();\n if (!this._initLocalStreamsPromise) {\n trace.info('initLocalMediaStreams creating media streams');\n this._initLocalStreamsPromise = this.getLocalMediaImplementation()\n .catch(function (mediaRequestFailResult) {\n trace.error('initLocalMediaStreams failure: ', mediaRequestFailResult);\n throw mediaRequestFailResult;\n });\n }\n else {\n trace.info('initLocalMediaStreams already done or in progress');\n }\n return this._initLocalStreamsPromise;\n };\n /*\n Creates an instance of subscriber that will \"listen\" to the requested stream.\n Whenever requested stream is available, subscriber will render it.\n */\n TechseeMediaServiceBase.prototype.createSubscriber = function (subscriberParams) {\n if (this._subscribers.has(subscriberParams.container)) {\n return Promise.reject('Subscriber for provided DIV element already exists');\n }\n var subscriber = new MediaSubscriber_1.TechseeMediaSubscriber(subscriberParams);\n this._subscribers.set(subscriberParams.container, subscriber);\n var streamForSubscriber = this.getRegisteredStreamByType(subscriber.streamType);\n if (streamForSubscriber) {\n trace.info(\"Stream for \" + subscriberParams.streamType + \" is ready.\");\n subscriber.renderStream(streamForSubscriber);\n }\n else {\n trace.info(\"Stream for \" + subscriberParams.streamType + \" not created yet\");\n }\n return Promise.resolve(subscriber);\n };\n /*\n Destroys the subscriber for provided HTML DIV container\n */\n TechseeMediaServiceBase.prototype.destroySubscriber = function (container) {\n if (this._subscribers.has(container)) {\n var subscriber = this._subscribers.get(container);\n subscriber.dispose();\n }\n this._subscribers.delete(container);\n return Promise.resolve();\n };\n TechseeMediaServiceBase.prototype.onStreamDestroyed = function (callback) {\n this.registerEventCallback(privateEvents.STREAM_DESTROYED, callback);\n };\n TechseeMediaServiceBase.prototype.onStreamCreated = function (callback) {\n this.registerEventCallback(privateEvents.STREAM_CREATED, callback);\n };\n TechseeMediaServiceBase.prototype.getSwitchCameraConstraints = function (constraints) {\n var newConstraints = cloneDeep_1.default(constraints);\n if (!newConstraints) {\n throw new Error('getSwitchCameraConstraints: unexpected use case constraints is null.');\n }\n trace.info('getSwitchCameraConstraints - Start switch camera with constraints:', newConstraints);\n var videoSourceType = newConstraints.video.videoSourceType === MediaConstants_1.LocalVideoSourceType.CAMERA\n ? MediaConstants_1.LocalVideoSourceType.CAMERA_FRONT\n : MediaConstants_1.LocalVideoSourceType.CAMERA;\n trace.info(\"getSwitchCameraConstraints - switch to videoSourceType: \" + videoSourceType);\n newConstraints.video.videoSourceType = videoSourceType;\n return constraints;\n };\n TechseeMediaServiceBase.prototype.switchCamera = function (constraints, isFailed) {\n var _this = this;\n trace.info('switchCamera: start');\n var newConstraints = !isFailed ? this.getSwitchCameraConstraints(constraints) : cloneDeep_1.default(constraints);\n if (!newConstraints) {\n throw new Error('switchCamera - unexpected use case constraints is null.');\n }\n this._localStreamsManager.destroyUserMediaStream();\n return this._localStreamsManager.getUserMediaStream(newConstraints)\n .then(function (streamResult) {\n var stream = streamResult.mediaStream;\n trace.info('switchCamera: new stream: ', stream);\n _this.replaceStreamTracks(stream);\n return _this.registerStreamResult(newConstraints, streamResult, true).then(function () { return ({ newConstraints: newConstraints, streamResult: streamResult }); });\n })\n .catch(function (err) {\n trace.error('switchCamera: Failed to switch camera: ', err);\n if (!isFailed) {\n _this.switchCamera(constraints, true);\n }\n return Promise.reject('Failed to switch camera.');\n });\n };\n TechseeMediaServiceBase.prototype.replaceStreamTracks = function (mediaStream) {\n this._session.instance && this._session.instance.replaceStreamTracks(mediaStream);\n };\n TechseeMediaServiceBase.prototype.registerStreamResult = function (constraints, streamResult, switchCamera, addStreamType) {\n var _this = this;\n var regPromises = [];\n if (streamResult.isNew) {\n trace.info('registerStreamResult: stream result from getUserMediaStream:', streamResult.mediaStream);\n streamResult.mediaStream.getTracks().forEach(function (mediaTrack) {\n trace.info('registerStreamResult: stream result from mediaTrack:', mediaTrack);\n var streamType = addStreamType || (mediaTrack.kind === 'video'\n ? MediaConstants_1.KnownMediaStream.USER_VIDEO_STREAM : MediaConstants_1.KnownMediaStream.USER_AUDIO_STREAM);\n var newDedicatedStream = new TechseeMediaStream_1.TechseeMediaStream(mediaTrack, streamType, false);\n regPromises.push(switchCamera ? _this.registerTrack(newDedicatedStream) : _this.registerLocalMediaStream(newDedicatedStream));\n });\n }\n return Promise.all(regPromises);\n };\n Object.defineProperty(TechseeMediaServiceBase.prototype, \"supportSwitchCameras\", {\n get: function () {\n return this._isIOS_13_orLater || this._localStreamsManager.groupedDevices.camerasCount > 1;\n },\n enumerable: true,\n configurable: true\n });\n //#endregion Media Streams Management\n //#region Session Management\n /*\n Creates instance of WebRTC session which connects to (signaling server), and begins to listen to WebRTC events.\n */\n TechseeMediaServiceBase.prototype.connectToSession = function (sessionParams) {\n var _this = this;\n this.serviceInitGuard();\n if (!this._session) {\n this._session = { params: cloneDeep_1.default(sessionParams) };\n this._session.connectPromise = new Promise(function (resolve, reject) {\n var doAsyncReject = function (error) {\n /*\n Reject with timeout, to change JS 'execution flow'.\n If timeout removed, session is assigned to null, before promise returns, and \"catch\" will not\n work on promise that is returned.\n */\n setTimeout(function () {\n _this._session = null;\n reject(error);\n });\n };\n var sessionHandlers = {\n onDisconnectedHandler: _this.onSessionDisconnectHandler\n };\n var mediaSessionParams = __assign({}, sessionParams, { peerConnectivityTimeoutSeconds: _this._serviceOptions.peerConnectivityTimeoutSeconds || MediaConstants_1.DEFAULT_PEER_CONNECTIVITY_TIMEOUT_SECONDS });\n trace.info('connectToSession', sessionParams);\n if (_this._serviceOptions.mediaServiceType === MediaConstants_1.MediaServiceType.TURNSERVER) {\n _this._session.instance = new SessionTurn_1.TurnWebRtcSession(mediaSessionParams, sessionHandlers, _this._sessionStreamsManager);\n }\n else if (_this._serviceOptions.mediaServiceType === MediaConstants_1.MediaServiceType.OPENTOK) {\n _this._session.instance = new SessionOpentok_1.OpentokSession(mediaSessionParams, _this._sessionStreamsManager);\n }\n else {\n var allowedValues = values_1.default(MediaConstants_1.MediaServiceType);\n trace.error(\"mediaServiceType '\" + _this._serviceOptions.mediaServiceType + \"' is not supported.\", { allowedValues: allowedValues });\n doAsyncReject(new Error('mediaServiceType is not supported'));\n }\n if (_this._session && _this._session.instance) {\n _this._session.instance.connect()\n .then(function () {\n trace.info('Session is connected');\n resolve();\n })\n .catch(function (error) {\n trace.error('Failed to connect to session', error);\n doAsyncReject(error);\n });\n }\n });\n }\n else {\n trace.warn('Session connected already. Disconnect before connect to new session again');\n }\n //TODO - Alex: Change a structure of this function to be more safe\n //(when we return, there should not be case when session or promise is null)\n //To fix, need to return promise immediately, then execute process in setTimeout(fn, 0);\n return this._session.connectPromise;\n };\n /*\n Disconnects from WebRTC session.\n */\n TechseeMediaServiceBase.prototype.disconnectFromSession = function () {\n return this.disconnectFromSessionInternal(MediaConstants_1.MediaSessionDisconnectReason.ForcedByConsumer);\n };\n /*\n Updates a credentials for turn server\n */\n TechseeMediaServiceBase.prototype.updateSessionCredentials = function (credentials) {\n trace.info('updateSessionCredentials');\n if (!this.sessionExistsGuard(false)) {\n return Promise.reject('There no session to update credentials');\n }\n this._session.params.credentials = cloneDeep_1.default(credentials);\n return Promise.resolve();\n };\n TechseeMediaServiceBase.prototype.enableVoipDuringSession = function () {\n if (!this._isVoipEnabled && this.isLocalStreamInitialized) {\n throw new Error('Voip support cannot be enabled after local stream where initialized');\n }\n this._isVoipEnabled = true;\n };\n TechseeMediaServiceBase.prototype.disconnectFromSessionInternal = function (reason) {\n if (!this._session) {\n return Promise.resolve();\n }\n trace.info('Disconnecting from session', reason);\n var lastSession = this._session;\n this._session = null;\n return Promise.all([\n lastSession.instance.disconnect(),\n this.clearPublishers()\n ]).then(function () { return undefined; });\n };\n TechseeMediaServiceBase.prototype.getStatsForRemoteTrack = function (streamType) {\n if (!this.sessionExistsGuard(false)) {\n return Promise.reject(new Error('Media session not started'));\n }\n var streamForStats = this.getRegisteredStreamByType(streamType);\n if (!streamForStats) {\n return Promise.reject(new Error('Stream for requested type was not found'));\n }\n return this._session.instance.getRemoteTrackStats(streamForStats.mediaTrack)\n .then(function (trackStats) {\n traceStatsInfo(\"MediaTrackStats for \" + streamType + \": \" + JSON.stringify(trackStats));\n return {\n trackId: streamForStats.mediaTrack.id,\n trackStats: trackStats\n };\n });\n };\n TechseeMediaServiceBase.prototype.onSessionDisconnectHandler = function (reason) {\n var _this = this;\n trace.info('onSessionDisconnectHandler', reason);\n if (reason !== MediaConstants_1.MediaSessionDisconnectReason.ForcedByConsumer) {\n var lastParams_1 = this._session && this._session.params ? this._session.params : null;\n this.disconnectFromSessionInternal(reason).then(function () {\n var reconnectReasons = [\n MediaConstants_1.MediaSessionDisconnectReason.InitiatorPeerReconnected,\n MediaConstants_1.MediaSessionDisconnectReason.SignalingChannelDisconnect,\n MediaConstants_1.MediaSessionDisconnectReason.PeerConnectionInterrupted,\n MediaConstants_1.MediaSessionDisconnectReason.PeerConnectionStateChangeTimeout\n ];\n if (lastParams_1 && includes_1.default(reconnectReasons, reason) && _this._autoReconnectEnabled) {\n _this.reconnectToSession(lastParams_1);\n }\n else if (!_this._autoReconnectEnabled) {\n trace.info('No reconnection- auto reconnect disabled');\n }\n else {\n trace.info('No params for reconnection to media session');\n }\n });\n }\n else {\n trace.info('Ignore Session disconnect event');\n }\n };\n TechseeMediaServiceBase.prototype.setAutoReconnect = function (state) {\n this._autoReconnectEnabled = state;\n trace.info('_autoReconnectEnabled:', state);\n if (this._autoReconnectEnabled && !this.isSessionActive) {\n var lastParams = this._session && this._session.params ? this._session.params : null;\n if (lastParams) {\n trace.info('setAutoReconnect - reconnect to session');\n this.reconnectToSession(lastParams);\n }\n }\n };\n TechseeMediaServiceBase.prototype.reconnectToSession = function (lastParams) {\n trace.info('Reconnecting to media session, sessionParams:', lastParams);\n this.connectToSession(lastParams)\n .then(function () {\n trace.info('Media session reconnected');\n })\n .catch(function (error) {\n trace.error('Error while reconnecting to media session', error);\n });\n };\n //#endregion Session Management\n //#region Utils\n TechseeMediaServiceBase.prototype.getSnapshotFromKnownStream = function (sourceStream, snapshotOptions) {\n var _this = this;\n return new Promise(function (resolve, reject) {\n if (sourceStream !== MediaConstants_1.KnownMediaStream.USER_VIDEO_STREAM &&\n sourceStream !== MediaConstants_1.KnownMediaStream.USER_SCREEN_SHARE_STREAM) {\n trace.error('The requested stream is not video stream, and cannot be used for snapshot');\n reject(new Error('INCOMPATIBLE_STREAM_FOR_SNAPSHOT'));\n return;\n }\n var snapshotStream = _this.getRegisteredStreamByType(sourceStream);\n if (!snapshotStream) {\n trace.error('Cannot make snapshot: The requested stream not exists yet.');\n reject(new Error('NO_REQUESTED_STREAM'));\n return;\n }\n MediaDomUtils_1.getSnapshotFromMediaStream(snapshotStream.mediaStream, snapshotOptions)\n .then(function (imageData) {\n var urlComponents = imageData.split(';base64,');\n var mimeType = urlComponents[0].split(':')[1];\n var bytes = atob(urlComponents[1]);\n var buffer = new ArrayBuffer(bytes.length);\n var rawData = new Uint8Array(buffer);\n for (var i = 0; i < bytes.length; i++) {\n rawData[i] = bytes.charCodeAt(i);\n }\n var blob = new Blob([rawData], { type: mimeType });\n var objectUrl = window.URL.createObjectURL(blob);\n trace.info('Snapshot created successfully');\n var result = {\n base64img: imageData,\n objectUrl: objectUrl,\n imageBlob: blob,\n mimeType: mimeType\n };\n resolve(result);\n })\n .catch(function (error) {\n trace.error('Error creating snapshot', error);\n reject(error);\n });\n });\n };\n //Will clean streams, publishers and subscribers, but will not remove event listeners\n TechseeMediaServiceBase.prototype.clearService = function () {\n var _this = this;\n trace.info('MediaService clearing all resources');\n return this.disconnectFromSessionInternal(MediaConstants_1.MediaSessionDisconnectReason.ForcedByConsumer)\n .then(this.clearSubscribers)\n .then(this.clearRegisteredStreams)\n .then(this._localStreamsManager.clearAllStreams)\n .then(function () {\n _this._initLocalStreamsPromise = null;\n _this._isVoipEnabled = false;\n }).then(function () { return undefined; });\n };\n //Will clear the service and remove all event listeners\n TechseeMediaServiceBase.prototype.dispose = function () {\n var _this = this;\n return this.clearService().catch(function () { return undefined; }).then(function () {\n _this._emitter.removeAllListeners();\n });\n };\n Object.defineProperty(TechseeMediaServiceBase.prototype, \"subscribers\", {\n //#endregion\n //#region Protected Methods\n get: function () {\n return this._subscribers;\n },\n enumerable: true,\n configurable: true\n });\n TechseeMediaServiceBase.prototype.registerLocalMediaStream = function (tsMediaStream) {\n trace.info('Registering local stream', tsMediaStream.streamType);\n if (this._registeredStreams.has(tsMediaStream.streamType)) {\n return Promise.reject(new Error(\"Stream \" + tsMediaStream.streamType + \" already registered\"));\n }\n return this.registerStream(tsMediaStream);\n };\n TechseeMediaServiceBase.prototype.getRegisteredStreamByType = function (streamType) {\n return this._registeredStreams.get(streamType) || null;\n };\n TechseeMediaServiceBase.prototype.changeEnableForKnownStream = function (streamType, isPaused) {\n var streamToChangeState = this.getRegisteredStreamByType(streamType);\n if (!streamToChangeState) {\n trace.warn('There no stream found to change enable state', streamType);\n }\n else {\n streamToChangeState.mediaTrack.enabled = !isPaused;\n if (streamToChangeState.isRemote && streamToChangeState.streamKind === MediaConstants_1.KnownMediaStreamKind.Audio) {\n this.subscribers.forEach(function (subscriber) {\n if (subscriber.streamType === streamType) {\n subscriber.muteSound(isPaused);\n }\n });\n }\n trace.info('Local stream enable state is changed', { streamType: streamType, isPaused: isPaused });\n }\n };\n TechseeMediaServiceBase.prototype.registerEventCallback = function (event, callback) {\n this._emitter.on(event, callback);\n };\n TechseeMediaServiceBase.prototype.emitEvent = function (event, eventArgs) {\n var _this = this;\n setTimeout(function () {\n _this._emitter.emit(event, eventArgs);\n });\n };\n //#endregion Protected Methods\n //#region Private Methods\n TechseeMediaServiceBase.prototype.registerRemoteMediaTrack = function (remoteMediaTrack) {\n trace.info(\"Registering remote \" + remoteMediaTrack.trackType + \" MediaStreamTrack\", remoteMediaTrack.mediaTrack);\n var currentStream = this._registeredStreams.get(remoteMediaTrack.trackType);\n if (currentStream && !currentStream.isRemote) {\n return Promise.reject(new Error('Cannot register remote stream with the same type as local stream'));\n }\n var newDedicatedStream = new TechseeMediaStream_1.TechseeMediaStream(remoteMediaTrack.mediaTrack, remoteMediaTrack.trackType, true);\n return this.registerStream(newDedicatedStream);\n };\n TechseeMediaServiceBase.prototype.registerTrack = function (mediaStream) {\n var _this = this;\n mediaStream.mediaTrack.onended = function () {\n return _this.unregisterTechseeMediaStream(mediaStream, MediaConstants_1.MediaStreamUnregisterReason.NativeEvent);\n };\n this._registeredStreams.set(mediaStream.streamType, mediaStream);\n trace.info(\"TechseeMediaStream registered - \" + mediaStream.streamType, TechseeMediaStream_1.TechseeMediaStream);\n return this.updateSubscribersWithNewStream(mediaStream).then(function () {\n var eventArgs = { streamType: mediaStream.streamType };\n _this.emitEvent(privateEvents.STREAM_CREATED, eventArgs);\n });\n };\n TechseeMediaServiceBase.prototype.registerStream = function (mediaStream) {\n var _this = this;\n var currentStream = this._registeredStreams.get(mediaStream.streamType);\n if (currentStream) {\n return this.unregisterTechseeMediaStream(currentStream, MediaConstants_1.MediaStreamUnregisterReason.ReplacingStream).then(function () { return _this.registerTrack(mediaStream); });\n }\n return this.registerTrack(mediaStream);\n };\n TechseeMediaServiceBase.prototype.unregisterTechseeMediaStream = function (streamToUnregister, reason) {\n var _this = this;\n var traceError = function (error) {\n trace.warn('Unregister TechseeMediaStream error:', error);\n };\n var promises = [];\n trace.info('Unregister TechseeMediaStream: ', streamToUnregister.streamType);\n this._subscribers.forEach(function (subscriber) {\n if (subscriber.streamType === streamToUnregister.streamType) {\n promises.push(subscriber.stopRendering().catch(traceError));\n }\n });\n if (!streamToUnregister.isRemote) {\n this._publishers.forEach(function (publisher) {\n if (publisher.streamTypes.find(function (streamType) { return streamType === streamToUnregister.streamType; })) {\n if (_this._session && _this._session.instance) {\n promises.push(_this._session.instance.onMediaStreamDestroyed(publisher.destinationRole).catch(traceError));\n }\n }\n });\n }\n return Promise.all(promises)\n .then(function () {\n _this._registeredStreams.delete(streamToUnregister.streamType);\n trace.info('Stream deleted', streamToUnregister.streamType);\n var eventArgs = { streamType: streamToUnregister.streamType, reason: reason };\n _this.emitEvent(privateEvents.STREAM_DESTROYED, eventArgs);\n });\n };\n TechseeMediaServiceBase.prototype.unregisterRemoteMediaTrack = function (mediaTrack) {\n var _this = this;\n var promises = [];\n this._registeredStreams.forEach(function (registeredStream) {\n if (registeredStream.mediaTrack.id === mediaTrack.id) {\n promises.push(_this.unregisterTechseeMediaStream(registeredStream, MediaConstants_1.MediaStreamUnregisterReason.ClosedRemotely));\n }\n });\n return Promise.all(promises).then(function () { return undefined; });\n };\n TechseeMediaServiceBase.prototype.removePublisher = function (publisher) {\n this._publishers.delete(publisher.destinationRole);\n if (this._session && this._session.instance) {\n trace.info('Removing publisher', publisher.destinationRole);\n return this._session.instance.onMediaStreamDestroyed(publisher.destinationRole)\n .catch(function (error) {\n trace.warn('Error while removing publisher', error);\n });\n }\n return Promise.resolve();\n };\n TechseeMediaServiceBase.prototype.getStreamForDestinationRole = function (destinationRole) {\n var _this = this;\n this.serviceInitGuard();\n if (this._publishers.has(destinationRole)) {\n trace.info(\"Publisher for \" + destinationRole + \" already exists\");\n return Promise.resolve(this._publishers.get(destinationRole).mediaStream);\n }\n if (!this._publisherPromises.get(destinationRole)) {\n trace.info(\"Creating publisher for \" + destinationRole);\n var publisherPromise = this.initLocalMediaStreams()\n .then(function () { return _this.createMediaPublisher(destinationRole); })\n .then(function (mediaPublisher) {\n if (mediaPublisher) {\n _this._publishers.set(destinationRole, mediaPublisher);\n }\n _this._publisherPromises.delete(destinationRole);\n return mediaPublisher ? mediaPublisher.mediaStream : null;\n })\n .catch(function (ex) {\n if (ex && ex.message === 'audioStreamFailed') {\n _this._isVoipEnabled = false;\n return null;\n }\n _this._publisherPromises.delete(destinationRole);\n throw ex;\n });\n this._publisherPromises.set(destinationRole, publisherPromise);\n }\n else {\n trace.info(\"Create publisher promise for \" + destinationRole + \" already exists\");\n }\n return this._publisherPromises.get(destinationRole);\n };\n TechseeMediaServiceBase.prototype.updateSubscribersWithNewStream = function (registeredStream) {\n trace.info(\"Updating subscribers of \" + registeredStream.streamType + \" with new stream\");\n if (this._subscribers.size === 0) {\n trace.warn(\"No subscribers exists for the \" + registeredStream.streamType + \".\");\n }\n else {\n trace.info(\"Total \" + this._subscribers.size + \" subscribers exists, will check if rerender needed.\");\n }\n this._subscribers.forEach(function (subscriber) {\n if (subscriber.streamType === registeredStream.streamType) {\n trace.info(registeredStream.streamType + \" rendering on subscriber\");\n subscriber.renderStream(registeredStream);\n }\n });\n return Promise.resolve();\n };\n TechseeMediaServiceBase.prototype.bindClassMethods = function () {\n this.updateSessionCredentials = this.updateSessionCredentials.bind(this);\n this.getStreamForDestinationRole = this.getStreamForDestinationRole.bind(this);\n this.registerRemoteMediaTrack = this.registerRemoteMediaTrack.bind(this);\n this.unregisterRemoteMediaTrack = this.unregisterRemoteMediaTrack.bind(this);\n this.createMediaPublisher = this.createMediaPublisher.bind(this);\n this.updateSubscribersWithNewStream = this.updateSubscribersWithNewStream.bind(this);\n this.initLocalMediaStreams = this.initLocalMediaStreams.bind(this);\n this.onSessionDisconnectHandler = this.onSessionDisconnectHandler.bind(this);\n this.clearService = this.clearService.bind(this);\n this.clearRegisteredStreams = this.clearRegisteredStreams.bind(this);\n this.clearPublishers = this.clearPublishers.bind(this);\n this.clearSubscribers = this.clearSubscribers.bind(this);\n };\n //region Cleanup\n TechseeMediaServiceBase.prototype.clearRegisteredStreams = function () {\n var _this = this;\n trace.info('Clearing registered streams');\n var promises = [];\n this._registeredStreams.forEach(function (streamToUnregister) {\n promises.push(_this.unregisterTechseeMediaStream(streamToUnregister, MediaConstants_1.MediaStreamUnregisterReason.ServiceCleanUp));\n });\n return Promise.all(promises).then(function () { return undefined; });\n };\n TechseeMediaServiceBase.prototype.clearSubscribers = function () {\n var _this = this;\n trace.info('Clearing subscribers');\n var promises = [];\n this._subscribers.forEach(function (subscriber) {\n promises.push(_this.destroySubscriber(subscriber.container));\n });\n return Promise.all(promises).then(function () { return undefined; });\n };\n TechseeMediaServiceBase.prototype.clearPublishers = function () {\n var _this = this;\n trace.info('Clearing publishers');\n var promises = [];\n this._publishers.forEach(function (publisher) {\n promises.push(_this.removePublisher(publisher));\n });\n return Promise.all(promises).then(function () { return undefined; });\n };\n //#endregion Cleanup\n //#region Simple Validation Methods\n TechseeMediaServiceBase.prototype.serviceInitGuard = function (shouldThrow) {\n if (shouldThrow === void 0) { shouldThrow = true; }\n return guards_1.throwableGuard(!!this._serviceOptions, 'Media service is not initialized', shouldThrow);\n };\n TechseeMediaServiceBase.prototype.sessionExistsGuard = function (shouldThrow) {\n if (shouldThrow === void 0) { shouldThrow = true; }\n return guards_1.throwableGuard(!!this._session, 'There no active session', shouldThrow);\n };\n return TechseeMediaServiceBase;\n}());\nexports.TechseeMediaServiceBase = TechseeMediaServiceBase;\n\n//# sourceMappingURL=MediaServiceBase.js.map\n","import filter from 'lodash/filter';\nimport cloneDeep from 'lodash/cloneDeep';\nimport values from 'lodash/values';\nimport debounce from 'lodash/debounce';\nimport includes from 'lodash/includes';\nimport {EventEmitter} from 'events';\nimport {Nullable} from '@techsee/techsee-common';\nimport {TechseeMediaSubscriber} from './MediaSubscriber';\nimport {TechseeMediaPublisher} from './MediaPublisher';\nimport {LocalStreamManager} from './LocalStreamManager';\nimport {TechseeMediaStream} from './TechseeMediaStream';\nimport {TurnWebRtcSession} from './MediaSession/SessionTurn';\nimport {getSnapshotFromMediaStream} from './MediaUtils/MediaDomUtils';\nimport {\n ClientWebRtcInfo,\n DeviceSupportInfo,\n IMediaEnvironment,\n IMediaSession,\n IMediaSubscriber,\n ISessionStreamsManager, LocalMediaConstraints,\n LocalVideoStreamConstraints,\n MediaRequestFailResult, MediaRequestSuccessResult,\n MediaServiceOptions,\n MediaSessionConfiguration,\n MediaSessionHandlers,\n MediaSessionIceCredentials,\n MediaSessionParams,\n MediaSubscriberParams,\n RemoteMediaTrack,\n RemoteTrackStats,\n SnapshotOptions,\n SnapshotResult,\n StreamCreatedEventArgs,\n StreamDestroyedEventArgs\n} from './MediaContracts';\nimport {enumerateMediaDevices, isVideoPlaySupportedOnDevice} from './MediaUtils/Compatibility';\nimport {OpentokSession} from './MediaSession/SessionOpentok';\nimport {\n DEFAULT_PEER_CONNECTIVITY_TIMEOUT_SECONDS,\n KnownMediaStream,\n KnownMediaStreamKind,\n LocalVideoSourceType,\n MediaServiceType,\n MediaSessionDisconnectReason,\n MediaStreamUnregisterReason,\n SessionClientRole\n} from './MediaConstants';\nimport {throwableGuard} from '@techsee/techsee-common/lib/core/guards';\n\nimport {getMediaTracer} from './MediaUtils/MediaTracer';\n\nconst trace = getMediaTracer('MediaServiceBase');\nconst traceStatsInfo = debounce(trace.info, 1000 * 5, {leading: true, maxWait: 1000 * 30});\n\nenum privateEvents {\n STREAM_CREATED = 'STREAM_CREATED',\n STREAM_DESTROYED = 'STREAM_DESTROYED'\n}\n\nexport abstract class TechseeMediaServiceBase {\n //This method should not be called. The only place that allowed to use it, is initLocalMediaStreams method.\n //Everyone who need to initialize MediaStream should use initLocalMediaStreams method.\n protected abstract getLocalMediaImplementation(): Promise<void>;\n\n protected abstract createMediaPublisher(destinationRole: SessionClientRole): Promise<Nullable<TechseeMediaPublisher>>;\n\n protected readonly _environment: IMediaEnvironment;\n protected readonly _sessionStreamsManager: ISessionStreamsManager;\n protected readonly _localStreamsManager: LocalStreamManager;\n\n private readonly _deviceSupportFlags: DeviceSupportInfo;\n private readonly _subscribers: Map<HTMLDivElement, TechseeMediaSubscriber> = new Map();\n private readonly _publishers: Map<SessionClientRole, TechseeMediaPublisher> = new Map();\n private readonly _publisherPromises: Map<SessionClientRole, Promise<Nullable<MediaStream>>> = new Map();\n protected _serviceOptions: Nullable<MediaServiceOptions> = null;\n private _isIOS_13_orLater: boolean = false;\n private _autoReconnectEnabled: boolean = true;\n private readonly _emitter: EventEmitter;\n private _initLocalStreamsPromise: Nullable<Promise<void>> = null;\n private _initServicePromise: Nullable<Promise<void>> = null;\n private _registeredStreams: Map<KnownMediaStream, TechseeMediaStream> = new Map();\n private _isVoipEnabled: boolean = false;\n private _session: Nullable<{\n params: MediaSessionParams;\n connectPromise?: Promise<void>;\n instance?: IMediaSession;\n }> = null;\n\n protected constructor(environment: IMediaEnvironment, webRtcSupportInfo: ClientWebRtcInfo) {\n trace.info('TechseeMediaServiceBase created');\n\n this.bindClassMethods();\n\n this._emitter = new EventEmitter();\n this._environment = environment;\n this._deviceSupportFlags = {\n videoPlayback: false,\n hasCamera: false,\n hasMicrophone: false,\n webRtcSupportInfo: webRtcSupportInfo\n };\n\n this._localStreamsManager = new LocalStreamManager(this._environment);\n\n // TODO: Hack as IOS 13.3 and above sometimes do not return the video/audio device. This\n // is a hack to work around the issue until Apple resolve it\n this._isIOS_13_orLater = this._environment.isIOS() &&\n (!this._environment.majorVersion() || (this._environment.majorVersion() >= 13));\n\n trace.info(`TechseeMediaServiceBase: isIOS_13_orLater: ${this._isIOS_13_orLater}`);\n\n this._sessionStreamsManager = {\n getMediaStreamForRole: this.getStreamForDestinationRole,\n addRemoteMediaTrack: this.registerRemoteMediaTrack,\n removeMediaTrack: this.unregisterRemoteMediaTrack\n };\n }\n\n get deviceSupportInfo(): DeviceSupportInfo {\n this.serviceInitGuard(false);\n\n return cloneDeep(this._deviceSupportFlags);\n }\n\n get isVoipEnabled(): boolean {\n return this._isVoipEnabled;\n }\n\n get isSessionActive(): boolean {\n return this._session !== null;\n }\n\n get isLocalStreamInitialized(): boolean {\n return this._initLocalStreamsPromise !== null;\n }\n\n /*\n Initializes media service. Before using anything from this service it should be initialized first.\n All of serviceOptions should be retrieved from account settings and by evaluating application state.\n For example, when initializing service, accountSettings is already loaded, and we know if FS session is audio enabled.\n */\n initMediaService(serviceOptions: MediaServiceOptions): Promise<void> {\n trace.info('TechseeMediaServiceBase.initMediaService', serviceOptions);\n const warnMessage = 'Multiple initialization does not supposed to happen. Not rejecting, but check the flow for correctness.';\n\n if (this._initServicePromise || this._serviceOptions) {\n console.warn(warnMessage);\n\n return this._initServicePromise ? this._initServicePromise : Promise.resolve();\n }\n\n this._serviceOptions = cloneDeep(serviceOptions);\n const isOpenTok = this._serviceOptions.mediaServiceType === MediaServiceType.OPENTOK;\n\n this._initServicePromise = Promise.all([\n isVideoPlaySupportedOnDevice(isOpenTok)\n .then((isSupported: boolean) => this._deviceSupportFlags.videoPlayback = isSupported),\n enumerateMediaDevices()\n .catch((error) => {\n trace.warn(error);\n\n return [];\n })\n .then((localDevicesList: MediaDeviceInfo[]) => {\n const groupedDevices = {\n video: filter(localDevicesList, (device) => device.kind.toLowerCase() === 'videoinput'),\n audio: filter(localDevicesList, (device) => device.kind.toLowerCase() === 'audioinput')\n };\n\n this._localStreamsManager.setGroupedDevices(groupedDevices);\n this._deviceSupportFlags.hasCamera = this._isIOS_13_orLater || (groupedDevices.video.length > 0);\n this._deviceSupportFlags.hasMicrophone = this._isIOS_13_orLater || (groupedDevices.audio.length > 0);\n\n })\n ]).then(() => undefined);\n\n\n return this._initServicePromise;\n }\n\n //#region Media Streams Management\n\n /*\n The main method to start the initialization of local media streams.\n This method can be executed many times and it will insure all calls are synced and only relevant stream will be created.\n DO NOT USE ANY OTHER METHOD THAT FORCES \"navigator.getUserMedia\" OTHER THAN THIS ONE.\n */\n initLocalMediaStreams(): Promise<void> {\n this.serviceInitGuard();\n\n if (!this._initLocalStreamsPromise) {\n trace.info('initLocalMediaStreams creating media streams');\n this._initLocalStreamsPromise = this.getLocalMediaImplementation()\n .catch((mediaRequestFailResult: MediaRequestFailResult) => {\n trace.error('initLocalMediaStreams failure: ', mediaRequestFailResult);\n\n throw mediaRequestFailResult;\n });\n } else {\n trace.info('initLocalMediaStreams already done or in progress');\n }\n\n return this._initLocalStreamsPromise;\n }\n\n /*\n Creates an instance of subscriber that will \"listen\" to the requested stream.\n Whenever requested stream is available, subscriber will render it.\n */\n createSubscriber(subscriberParams: MediaSubscriberParams): Promise<IMediaSubscriber> {\n if (this._subscribers.has(subscriberParams.container)) {\n return Promise.reject('Subscriber for provided DIV element already exists');\n }\n\n const subscriber = new TechseeMediaSubscriber(subscriberParams);\n\n this._subscribers.set(subscriberParams.container, subscriber);\n const streamForSubscriber = this.getRegisteredStreamByType(subscriber.streamType);\n\n if (streamForSubscriber) {\n trace.info(`Stream for ${subscriberParams.streamType} is ready.`);\n subscriber.renderStream(streamForSubscriber);\n } else {\n trace.info(`Stream for ${subscriberParams.streamType} not created yet`);\n }\n\n return Promise.resolve(subscriber);\n }\n\n /*\n Destroys the subscriber for provided HTML DIV container\n */\n destroySubscriber(container: HTMLDivElement): Promise<void> {\n if (this._subscribers.has(container)) {\n const subscriber = this._subscribers.get(container);\n\n subscriber!.dispose();\n }\n\n this._subscribers.delete(container);\n\n return Promise.resolve();\n }\n\n onStreamDestroyed(callback: (eventArgs: StreamDestroyedEventArgs) => void): void {\n this.registerEventCallback(privateEvents.STREAM_DESTROYED, callback);\n }\n\n onStreamCreated(callback: (eventArgs: StreamCreatedEventArgs) => void): void {\n this.registerEventCallback(privateEvents.STREAM_CREATED, callback);\n }\n\n private getSwitchCameraConstraints(constraints: Nullable<LocalMediaConstraints>) {\n const newConstraints = cloneDeep(constraints);\n\n if (!newConstraints) {\n throw new Error('getSwitchCameraConstraints: unexpected use case constraints is null.');\n }\n\n trace.info('getSwitchCameraConstraints - Start switch camera with constraints:', newConstraints);\n\n const videoSourceType = (newConstraints.video as LocalVideoStreamConstraints).videoSourceType === LocalVideoSourceType.CAMERA\n ? LocalVideoSourceType.CAMERA_FRONT\n : LocalVideoSourceType.CAMERA;\n\n trace.info(`getSwitchCameraConstraints - switch to videoSourceType: ${videoSourceType}`);\n\n (newConstraints.video as LocalVideoStreamConstraints).videoSourceType = videoSourceType;\n\n return constraints;\n }\n\n switchCamera(constraints: Nullable<LocalMediaConstraints>, isFailed?: boolean) {\n trace.info('switchCamera: start');\n\n const newConstraints = !isFailed ? this.getSwitchCameraConstraints(constraints) : cloneDeep(constraints);\n\n if (!newConstraints) {\n throw new Error('switchCamera - unexpected use case constraints is null.');\n }\n\n this._localStreamsManager.destroyUserMediaStream();\n\n return this._localStreamsManager.getUserMediaStream(newConstraints)\n .then((streamResult: MediaRequestSuccessResult) => {\n const stream = streamResult.mediaStream;\n\n trace.info('switchCamera: new stream: ', stream);\n\n this.replaceStreamTracks(stream);\n\n return this.registerStreamResult(newConstraints, streamResult, true).then(() => ({newConstraints, streamResult}));\n })\n .catch((err) => {\n trace.error('switchCamera: Failed to switch camera: ', err);\n\n if (!isFailed) {\n this.switchCamera(constraints, true);\n }\n\n return Promise.reject('Failed to switch camera.');\n });\n }\n\n private replaceStreamTracks(mediaStream: Nullable<MediaStream>) {\n this._session!.instance && this._session!.instance.replaceStreamTracks(mediaStream);\n }\n\n protected registerStreamResult(constraints: Nullable<LocalMediaConstraints>, streamResult: MediaRequestSuccessResult, switchCamera?: boolean, addStreamType?: KnownMediaStream) {\n const regPromises: any[] = [];\n\n if (streamResult.isNew) {\n trace.info('registerStreamResult: stream result from getUserMediaStream:', streamResult.mediaStream);\n streamResult.mediaStream.getTracks().forEach((mediaTrack) => {\n trace.info('registerStreamResult: stream result from mediaTrack:', mediaTrack);\n const streamType = addStreamType || (mediaTrack.kind === 'video'\n ? KnownMediaStream.USER_VIDEO_STREAM : KnownMediaStream.USER_AUDIO_STREAM);\n const newDedicatedStream = new TechseeMediaStream(mediaTrack, streamType, false);\n\n regPromises.push(switchCamera ? this.registerTrack(newDedicatedStream) : this.registerLocalMediaStream(newDedicatedStream));\n });\n }\n\n return Promise.all(regPromises);\n }\n\n get supportSwitchCameras() {\n return this._isIOS_13_orLater || this._localStreamsManager.groupedDevices.camerasCount > 1;\n }\n\n //#endregion Media Streams Management\n\n //#region Session Management\n\n /*\n Creates instance of WebRTC session which connects to (signaling server), and begins to listen to WebRTC events.\n */\n connectToSession(sessionParams: MediaSessionParams): Promise<void> {\n this.serviceInitGuard();\n\n if (!this._session) {\n this._session = {params: cloneDeep(sessionParams)};\n this._session.connectPromise = new Promise((resolve, reject) => {\n const doAsyncReject = (error: any): void => {\n\n /*\n Reject with timeout, to change JS 'execution flow'.\n If timeout removed, session is assigned to null, before promise returns, and \"catch\" will not\n work on promise that is returned.\n */\n setTimeout(() => {\n this._session = null;\n reject(error);\n });\n };\n\n const sessionHandlers: MediaSessionHandlers = {\n onDisconnectedHandler: this.onSessionDisconnectHandler\n };\n const mediaSessionParams: MediaSessionParams & MediaSessionConfiguration = {\n ...sessionParams,\n peerConnectivityTimeoutSeconds: this._serviceOptions!.peerConnectivityTimeoutSeconds || DEFAULT_PEER_CONNECTIVITY_TIMEOUT_SECONDS\n };\n\n trace.info('connectToSession', sessionParams);\n if (this._serviceOptions!.mediaServiceType === MediaServiceType.TURNSERVER) {\n this._session!.instance = new TurnWebRtcSession(mediaSessionParams, sessionHandlers, this._sessionStreamsManager);\n\n } else if (this._serviceOptions!.mediaServiceType === MediaServiceType.OPENTOK) {\n this._session!.instance = new OpentokSession(mediaSessionParams, this._sessionStreamsManager);\n } else {\n const allowedValues = values(MediaServiceType);\n\n trace.error(`mediaServiceType '${this._serviceOptions!.mediaServiceType}' is not supported.`, {allowedValues});\n doAsyncReject(new Error('mediaServiceType is not supported'));\n }\n\n if (this._session && this._session.instance) {\n this._session!.instance.connect()\n .then(() => {\n trace.info('Session is connected');\n resolve();\n })\n .catch((error: any) => {\n trace.error('Failed to connect to session', error);\n doAsyncReject(error);\n });\n }\n\n });\n } else {\n trace.warn('Session connected already. Disconnect before connect to new session again');\n }\n\n //TODO - Alex: Change a structure of this function to be more safe\n //(when we return, there should not be case when session or promise is null)\n //To fix, need to return promise immediately, then execute process in setTimeout(fn, 0);\n return this._session!.connectPromise!;\n }\n\n /*\n Disconnects from WebRTC session.\n */\n disconnectFromSession(): Promise<void> {\n return this.disconnectFromSessionInternal(MediaSessionDisconnectReason.ForcedByConsumer);\n }\n\n /*\n Updates a credentials for turn server\n */\n updateSessionCredentials(credentials: MediaSessionIceCredentials): Promise<void> {\n trace.info('updateSessionCredentials');\n if (!this.sessionExistsGuard(false)) {\n return Promise.reject('There no session to update credentials');\n }\n\n this._session!.params.credentials = cloneDeep(credentials);\n\n return Promise.resolve();\n }\n\n enableVoipDuringSession(): void {\n if (!this._isVoipEnabled && this.isLocalStreamInitialized) {\n throw new Error('Voip support cannot be enabled after local stream where initialized');\n }\n\n this._isVoipEnabled = true;\n }\n\n protected disconnectFromSessionInternal(reason: MediaSessionDisconnectReason): Promise<void> {\n if (!this._session) {\n return Promise.resolve();\n }\n\n trace.info('Disconnecting from session', reason);\n\n const lastSession = this._session;\n\n this._session = null;\n\n return Promise.all([\n lastSession.instance!.disconnect(),\n this.clearPublishers()\n ]).then(() => undefined);\n }\n\n protected getStatsForRemoteTrack(streamType: KnownMediaStream): Promise<RemoteTrackStats> {\n if (!this.sessionExistsGuard(false)) {\n return Promise.reject(new Error('Media session not started'));\n }\n\n const streamForStats = this.getRegisteredStreamByType(streamType);\n\n if (!streamForStats) {\n return Promise.reject(new Error('Stream for requested type was not found'));\n }\n\n return this._session!.instance!.getRemoteTrackStats(streamForStats.mediaTrack)\n .then((trackStats: RemoteTrackStats) => {\n traceStatsInfo(`MediaTrackStats for ${streamType}: ${JSON.stringify(trackStats)}`);\n\n return {\n trackId: streamForStats.mediaTrack.id,\n trackStats: trackStats\n };\n });\n }\n\n private onSessionDisconnectHandler(reason: MediaSessionDisconnectReason): void {\n trace.info('onSessionDisconnectHandler', reason);\n if (reason !== MediaSessionDisconnectReason.ForcedByConsumer) {\n const lastParams = this._session && this._session.params ? this._session.params : null;\n\n this.disconnectFromSessionInternal(reason).then(() => {\n const reconnectReasons = [\n MediaSessionDisconnectReason.InitiatorPeerReconnected,\n MediaSessionDisconnectReason.SignalingChannelDisconnect,\n MediaSessionDisconnectReason.PeerConnectionInterrupted,\n MediaSessionDisconnectReason.PeerConnectionStateChangeTimeout\n ];\n\n if (lastParams && includes(reconnectReasons, reason) && this._autoReconnectEnabled) {\n this.reconnectToSession(lastParams);\n } else if (!this._autoReconnectEnabled) {\n trace.info('No reconnection- auto reconnect disabled');\n } else {\n trace.info('No params for reconnection to media session');\n }\n });\n } else {\n trace.info('Ignore Session disconnect event');\n }\n }\n\n setAutoReconnect(state: boolean) {\n this._autoReconnectEnabled = state;\n trace.info('_autoReconnectEnabled:', state);\n\n if (this._autoReconnectEnabled && !this.isSessionActive) {\n const lastParams = this._session && this._session.params ? this._session.params : null;\n\n if (lastParams) {\n trace.info('setAutoReconnect - reconnect to session');\n this.reconnectToSession(lastParams);\n }\n }\n }\n\n reconnectToSession(lastParams: MediaSessionParams) {\n trace.info('Reconnecting to media session, sessionParams:', lastParams);\n\n this.connectToSession(lastParams)\n .then(() => {\n trace.info('Media session reconnected');\n })\n .catch((error: any) => {\n trace.error('Error while reconnecting to media session', error);\n });\n }\n\n //#endregion Session Management\n\n //#region Utils\n\n getSnapshotFromKnownStream(sourceStream: KnownMediaStream, snapshotOptions?: SnapshotOptions): Promise<SnapshotResult> {\n return new Promise((resolve, reject) => {\n if (sourceStream !== KnownMediaStream.USER_VIDEO_STREAM &&\n sourceStream !== KnownMediaStream.USER_SCREEN_SHARE_STREAM) {\n trace.error('The requested stream is not video stream, and cannot be used for snapshot');\n reject(new Error('INCOMPATIBLE_STREAM_FOR_SNAPSHOT'));\n\n return;\n }\n\n const snapshotStream = this.getRegisteredStreamByType(sourceStream);\n\n if (!snapshotStream) {\n trace.error('Cannot make snapshot: The requested stream not exists yet.');\n reject(new Error('NO_REQUESTED_STREAM'));\n\n return;\n }\n\n getSnapshotFromMediaStream(snapshotStream.mediaStream, snapshotOptions)\n .then((imageData: any) => {\n const urlComponents = imageData.split(';base64,');\n const mimeType = urlComponents[0].split(':')[1];\n const bytes = atob(urlComponents[1]);\n const buffer = new ArrayBuffer(bytes.length);\n const rawData = new Uint8Array(buffer);\n\n for (let i = 0; i < bytes.length; i++) {\n rawData[i] = bytes.charCodeAt(i);\n }\n\n const blob = new Blob([rawData], {type: mimeType});\n const objectUrl = window.URL.createObjectURL(blob);\n\n trace.info('Snapshot created successfully');\n\n const result: SnapshotResult = {\n base64img: imageData,\n objectUrl: objectUrl,\n imageBlob: blob,\n mimeType: mimeType\n };\n\n resolve(result);\n })\n .catch((error: any) => {\n trace.error('Error creating snapshot', error);\n reject(error);\n });\n });\n }\n\n //Will clean streams, publishers and subscribers, but will not remove event listeners\n clearService(): Promise<void> {\n trace.info('MediaService clearing all resources');\n\n return this.disconnectFromSessionInternal(MediaSessionDisconnectReason.ForcedByConsumer)\n .then(this.clearSubscribers)\n .then(this.clearRegisteredStreams)\n .then(this._localStreamsManager.clearAllStreams)\n .then(() => {\n this._initLocalStreamsPromise = null;\n this._isVoipEnabled = false;\n }).then(() => undefined);\n }\n\n //Will clear the service and remove all event listeners\n dispose(): Promise<void> {\n return this.clearService().catch(() => undefined).then(() => {\n this._emitter.removeAllListeners();\n });\n }\n\n //#endregion\n\n //#region Protected Methods\n\n protected get subscribers(): Map<HTMLDivElement, TechseeMediaSubscriber> {\n return this._subscribers;\n }\n\n protected registerLocalMediaStream(tsMediaStream: TechseeMediaStream): Promise<void> {\n trace.info('Registering local stream', tsMediaStream.streamType);\n if (this._registeredStreams.has(tsMediaStream.streamType)) {\n return Promise.reject(new Error(`Stream ${tsMediaStream.streamType} already registered`));\n }\n\n return this.registerStream(tsMediaStream);\n }\n\n protected getRegisteredStreamByType(streamType: KnownMediaStream): Nullable<TechseeMediaStream> {\n return this._registeredStreams.get(streamType) || null;\n }\n\n protected changeEnableForKnownStream(streamType: KnownMediaStream, isPaused: boolean): void {\n const streamToChangeState = this.getRegisteredStreamByType(streamType);\n\n if (!streamToChangeState) {\n trace.warn('There no stream found to change enable state', streamType);\n } else {\n streamToChangeState.mediaTrack.enabled = !isPaused;\n\n if (streamToChangeState.isRemote && streamToChangeState.streamKind === KnownMediaStreamKind.Audio) {\n this.subscribers.forEach((subscriber) => {\n if (subscriber.streamType === streamType) {\n subscriber.muteSound(isPaused);\n }\n });\n }\n\n trace.info('Local stream enable state is changed', {streamType, isPaused});\n }\n }\n\n protected registerEventCallback(event: string, callback: (eventArgs?: any) => void): void {\n this._emitter.on(event, callback);\n }\n\n protected emitEvent(event: string, eventArgs?: any): void {\n setTimeout(() => {\n this._emitter.emit(event, eventArgs);\n });\n }\n\n //#endregion Protected Methods\n\n //#region Private Methods\n\n private registerRemoteMediaTrack(remoteMediaTrack: RemoteMediaTrack): Promise<void> {\n trace.info(`Registering remote ${remoteMediaTrack.trackType} MediaStreamTrack`, remoteMediaTrack.mediaTrack);\n const currentStream = this._registeredStreams.get(remoteMediaTrack.trackType);\n\n if (currentStream && !currentStream.isRemote) {\n return Promise.reject(new Error('Cannot register remote stream with the same type as local stream'));\n }\n\n const newDedicatedStream = new TechseeMediaStream(remoteMediaTrack.mediaTrack, remoteMediaTrack.trackType, true);\n\n return this.registerStream(newDedicatedStream);\n }\n\n private registerTrack(mediaStream: TechseeMediaStream): Promise<void> {\n mediaStream.mediaTrack.onended = () =>\n this.unregisterTechseeMediaStream(mediaStream, MediaStreamUnregisterReason.NativeEvent);\n\n this._registeredStreams.set(mediaStream.streamType, mediaStream);\n\n trace.info(`TechseeMediaStream registered - ${mediaStream.streamType}`, TechseeMediaStream);\n\n return this.updateSubscribersWithNewStream(mediaStream).then(() => {\n const eventArgs: StreamCreatedEventArgs = {streamType: mediaStream.streamType};\n\n this.emitEvent(privateEvents.STREAM_CREATED, eventArgs);\n });\n }\n\n private registerStream(mediaStream: TechseeMediaStream): Promise<void> {\n const currentStream = this._registeredStreams.get(mediaStream.streamType);\n\n if (currentStream) {\n return this.unregisterTechseeMediaStream(currentStream, MediaStreamUnregisterReason.ReplacingStream).then(() => this.registerTrack(mediaStream));\n }\n\n return this.registerTrack(mediaStream);\n }\n\n private unregisterTechseeMediaStream(streamToUnregister: TechseeMediaStream, reason: MediaStreamUnregisterReason): Promise<void> {\n const traceError = (error: any): void => {\n trace.warn('Unregister TechseeMediaStream error:', error);\n };\n\n const promises: any = [];\n\n trace.info('Unregister TechseeMediaStream: ', streamToUnregister.streamType);\n\n this._subscribers.forEach((subscriber) => {\n if (subscriber.streamType === streamToUnregister.streamType) {\n promises.push(subscriber.stopRendering().catch(traceError));\n }\n });\n\n if (!streamToUnregister.isRemote) {\n this._publishers.forEach((publisher: TechseeMediaPublisher) => {\n if (publisher.streamTypes.find((streamType) => streamType === streamToUnregister.streamType)) {\n if (this._session && this._session!.instance) {\n promises.push(this._session.instance.onMediaStreamDestroyed(publisher.destinationRole).catch(traceError));\n }\n }\n });\n }\n\n return Promise.all(promises)\n .then(() => {\n this._registeredStreams.delete(streamToUnregister.streamType);\n trace.info('Stream deleted', streamToUnregister.streamType);\n\n const eventArgs: StreamDestroyedEventArgs = {streamType: streamToUnregister.streamType, reason: reason};\n\n this.emitEvent(privateEvents.STREAM_DESTROYED, eventArgs);\n });\n }\n\n private unregisterRemoteMediaTrack(mediaTrack: MediaStreamTrack): Promise<void> {\n const promises: any = [];\n\n this._registeredStreams.forEach((registeredStream: TechseeMediaStream) => {\n if (registeredStream.mediaTrack.id === mediaTrack.id) {\n promises.push(this.unregisterTechseeMediaStream(registeredStream, MediaStreamUnregisterReason.ClosedRemotely));\n }\n });\n\n return Promise.all(promises).then(() => undefined);\n }\n\n private removePublisher(publisher: TechseeMediaPublisher): Promise<void> {\n this._publishers.delete(publisher.destinationRole);\n if (this._session && this._session.instance) {\n trace.info('Removing publisher', publisher.destinationRole);\n\n return this._session.instance.onMediaStreamDestroyed(publisher.destinationRole)\n .catch((error: any) => {\n trace.warn('Error while removing publisher', error);\n });\n }\n\n return Promise.resolve();\n }\n\n private getStreamForDestinationRole(destinationRole: SessionClientRole): Promise<Nullable<MediaStream>> {\n this.serviceInitGuard();\n\n if (this._publishers.has(destinationRole)) {\n trace.info(`Publisher for ${destinationRole} already exists`);\n\n return Promise.resolve(this._publishers.get(destinationRole)!.mediaStream);\n }\n\n if (!this._publisherPromises.get(destinationRole)) {\n trace.info(`Creating publisher for ${destinationRole}`);\n const publisherPromise = this.initLocalMediaStreams()\n .then(() => this.createMediaPublisher(destinationRole))\n .then((mediaPublisher: Nullable<TechseeMediaPublisher>) => {\n if (mediaPublisher) {\n this._publishers.set(destinationRole, mediaPublisher);\n }\n\n this._publisherPromises.delete(destinationRole);\n\n return mediaPublisher ? mediaPublisher.mediaStream : null;\n })\n .catch((ex: any) => {\n if (ex && ex.message === 'audioStreamFailed') {\n this._isVoipEnabled = false;\n\n return null;\n }\n\n this._publisherPromises.delete(destinationRole);\n throw ex;\n });\n\n this._publisherPromises.set(destinationRole, publisherPromise);\n } else {\n trace.info(`Create publisher promise for ${destinationRole} already exists`);\n }\n\n return this._publisherPromises.get(destinationRole)!;\n }\n\n private updateSubscribersWithNewStream(registeredStream: TechseeMediaStream): Promise<void> {\n trace.info(`Updating subscribers of ${registeredStream.streamType} with new stream`);\n\n if (this._subscribers.size === 0) {\n trace.warn(`No subscribers exists for the ${registeredStream.streamType}.`);\n } else {\n trace.info(`Total ${this._subscribers.size} subscribers exists, will check if rerender needed.`);\n }\n\n this._subscribers.forEach((subscriber) => {\n if (subscriber.streamType === registeredStream.streamType) {\n trace.info(`${registeredStream.streamType} rendering on subscriber`);\n subscriber.renderStream(registeredStream);\n }\n });\n\n return Promise.resolve();\n }\n\n private bindClassMethods(): void {\n this.updateSessionCredentials = this.updateSessionCredentials.bind(this);\n this.getStreamForDestinationRole = this.getStreamForDestinationRole.bind(this);\n this.registerRemoteMediaTrack = this.registerRemoteMediaTrack.bind(this);\n this.unregisterRemoteMediaTrack = this.unregisterRemoteMediaTrack.bind(this);\n this.createMediaPublisher = this.createMediaPublisher.bind(this);\n this.updateSubscribersWithNewStream = this.updateSubscribersWithNewStream.bind(this);\n this.initLocalMediaStreams = this.initLocalMediaStreams.bind(this);\n this.onSessionDisconnectHandler = this.onSessionDisconnectHandler.bind(this);\n this.clearService = this.clearService.bind(this);\n this.clearRegisteredStreams = this.clearRegisteredStreams.bind(this);\n this.clearPublishers = this.clearPublishers.bind(this);\n this.clearSubscribers = this.clearSubscribers.bind(this);\n }\n\n //region Cleanup\n\n private clearRegisteredStreams(): Promise<void> {\n trace.info('Clearing registered streams');\n const promises: any = [];\n\n this._registeredStreams.forEach((streamToUnregister: TechseeMediaStream) => {\n promises.push(this.unregisterTechseeMediaStream(streamToUnregister, MediaStreamUnregisterReason.ServiceCleanUp));\n });\n\n return Promise.all(promises).then(() => undefined);\n }\n\n private clearSubscribers(): Promise<void> {\n trace.info('Clearing subscribers');\n const promises: any = [];\n\n this._subscribers.forEach((subscriber) => {\n promises.push(this.destroySubscriber(subscriber.container));\n });\n\n return Promise.all(promises).then(() => undefined);\n }\n\n private clearPublishers(): Promise<void> {\n trace.info('Clearing publishers');\n const promises: any = [];\n\n this._publishers.forEach((publisher) => {\n promises.push(this.removePublisher(publisher));\n });\n\n return Promise.all(promises).then(() => undefined);\n }\n\n //#endregion Cleanup\n\n //#region Simple Validation Methods\n\n protected serviceInitGuard(shouldThrow = true): boolean {\n return throwableGuard(!!this._serviceOptions, 'Media service is not initialized', shouldThrow);\n }\n\n protected sessionExistsGuard(shouldThrow = true): boolean {\n return throwableGuard(!!this._session, 'There no active session', shouldThrow);\n }\n\n //#endregion\n\n //#endregion Private Methods\n}\n"]}
|
|
1
|
+
{"version":3,"sources":["MediaServiceBase.js","../src/MediaServiceBase.ts"],"names":["__assign","Object","assign","t","s","i","n","arguments","length","p","prototype","hasOwnProperty","call","apply","__importDefault","mod","__esModule","defineProperty","exports","value","filter_1","require","cloneDeep_1","values_1","debounce_1","includes_1","events_1","MediaSubscriber_1","LocalStreamManager_1","TechseeMediaStream_1","SessionTurn_1","MediaDomUtils_1","Compatibility_1","SessionOpentok_1","MediaConstants_1","guards_1","MediaTracer_1","trace","getMediaTracer","traceStatsInfo","default","info","leading","maxWait","privateEvents","TechseeMediaServiceBase","environment","webRtcSupportInfo","_subscribers","Map","_publishers","_publisherPromises","_serviceOptions","_isIOS_13_orLater","_autoReconnectEnabled","_initLocalStreamsPromise","_initServicePromise","_registeredStreams","_isVoipEnabled","_session","bindClassMethods","_emitter","EventEmitter","_environment","_deviceSupportFlags","videoPlayback","hasCamera","hasMicrophone","_localStreamsManager","LocalStreamManager","isIOS","majorVersion","_sessionStreamsManager","getMediaStreamForRole","getStreamForDestinationRole","addRemoteMediaTrack","registerRemoteMediaTrack","removeMediaTrack","unregisterRemoteMediaTrack","get","serviceInitGuard","enumerable","configurable","initMediaService","serviceOptions","_this","warnMessage","console","warn","Promise","resolve","isOpenTok","mediaServiceType","MediaServiceType","OPENTOK","all","isVideoPlaySupportedOnDevice","then","isSupported","enumerateMediaDevices","catch","error","localDevicesList","groupedDevices","video","device","kind","toLowerCase","audio","setGroupedDevices","undefined","initLocalMediaStreams","getLocalMediaImplementation","mediaRequestFailResult","createSubscriber","subscriberParams","has","container","reject","subscriber","TechseeMediaSubscriber","set","streamForSubscriber","getRegisteredStreamByType","streamType","renderStream","destroySubscriber","dispose","delete","onStreamDestroyed","callback","registerEventCallback","STREAM_DESTROYED","onStreamCreated","STREAM_CREATED","getSwitchCameraConstraints","constraints","window","latestLocalMediaConstraints","Error","videoSourceType","LocalVideoSourceType","CAMERA","CAMERA_FRONT","switchCamera","isFailed","destroyUserMediaStream","getUserMediaStream","streamResult","stream","mediaStream","replaceStreamTracks","registerStreamResult","err","instance","addStreamType","regPromises","isNew","getTracks","forEach","mediaTrack","KnownMediaStream","USER_VIDEO_STREAM","USER_AUDIO_STREAM","newDedicatedStream","TechseeMediaStream","push","registerTrack","registerLocalMediaStream","camerasCount","connectToSession","sessionParams","params","connectPromise","doAsyncReject","setTimeout","sessionHandlers","onDisconnectedHandler","onSessionDisconnectHandler","mediaSessionParams","peerConnectivityTimeoutSeconds","DEFAULT_PEER_CONNECTIVITY_TIMEOUT_SECONDS","TURNSERVER","TurnWebRtcSession","OpentokSession","allowedValues","connect","disconnectFromSession","disconnectFromSessionInternal","MediaSessionDisconnectReason","ForcedByConsumer","updateSessionCredentials","credentials","sessionExistsGuard","enableVoipDuringSession","isLocalStreamInitialized","reason","lastSession","disconnect","clearPublishers","getStatsForRemoteTrack","streamForStats","getRemoteTrackStats","trackStats","JSON","stringify","trackId","id","lastParams_1","reconnectReasons","InitiatorPeerReconnected","SignalingChannelDisconnect","PeerConnectionInterrupted","PeerConnectionStateChangeTimeout","reconnectToSession","setAutoReconnect","state","isSessionActive","lastParams","getSnapshotFromKnownStream","sourceStream","snapshotOptions","USER_SCREEN_SHARE_STREAM","snapshotStream","getSnapshotFromMediaStream","imageData","urlComponents","split","mimeType","bytes","atob","buffer","ArrayBuffer","rawData","Uint8Array","charCodeAt","blob","Blob","type","objectUrl","URL","createObjectURL","result","base64img","imageBlob","clearService","clearSubscribers","clearRegisteredStreams","clearAllStreams","removeAllListeners","tsMediaStream","registerStream","changeEnableForKnownStream","isPaused","streamToChangeState","enabled","isRemote","streamKind","KnownMediaStreamKind","Audio","subscribers","muteSound","event","on","emitEvent","eventArgs","emit","remoteMediaTrack","trackType","currentStream","onended","unregisterTechseeMediaStream","MediaStreamUnregisterReason","NativeEvent","updateSubscribersWithNewStream","ReplacingStream","streamToUnregister","traceError","promises","stopRendering","publisher","streamTypes","find","onMediaStreamDestroyed","destinationRole","registeredStream","ClosedRemotely","removePublisher","publisherPromise","createMediaPublisher","mediaPublisher","ex","message","size","bind","ServiceCleanUp","shouldThrow","throwableGuard"],"mappings":"AAAA;;AACA,IAAIA,WAAY,aAAQ,UAAKA,QAAd,IAA2B,YAAY;AAClDA,eAAWC,OAAOC,MAAP,IAAiB,UAASC,CAAT,EAAY;AACpC,aAAK,IAAIC,CAAJ,EAAOC,IAAI,CAAX,EAAcC,IAAIC,UAAUC,MAAjC,EAAyCH,IAAIC,CAA7C,EAAgDD,GAAhD,EAAqD;AACjDD,gBAAIG,UAAUF,CAAV,CAAJ;AACA,iBAAK,IAAII,CAAT,IAAcL,CAAd;AAAiB,oBAAIH,OAAOS,SAAP,CAAiBC,cAAjB,CAAgCC,IAAhC,CAAqCR,CAArC,EAAwCK,CAAxC,CAAJ,EACbN,EAAEM,CAAF,IAAOL,EAAEK,CAAF,CAAP;AADJ;AAEH;AACD,eAAON,CAAP;AACH,KAPD;AAQA,WAAOH,SAASa,KAAT,CAAe,IAAf,EAAqBN,SAArB,CAAP;AACH,CAVD;AAWA,IAAIO,kBAAmB,aAAQ,UAAKA,eAAd,IAAkC,UAAUC,GAAV,EAAe;AACnE,WAAQA,OAAOA,IAAIC,UAAZ,GAA0BD,GAA1B,GAAgC,EAAE,WAAWA,GAAb,EAAvC;AACH,CAFD;AAGAd,OAAOgB,cAAP,CAAsBC,OAAtB,EAA+B,YAA/B,EAA6C,EAAEC,OAAO,IAAT,EAA7C;ACfA,IAAAC,WAAAN,gBAAAO,QAAA,eAAA,CAAA,CAAA;AACA,IAAAC,cAAAR,gBAAAO,QAAA,kBAAA,CAAA,CAAA;AACA,IAAAE,WAAAT,gBAAAO,QAAA,eAAA,CAAA,CAAA;AACA,IAAAG,aAAAV,gBAAAO,QAAA,iBAAA,CAAA,CAAA;AACA,IAAAI,aAAAX,gBAAAO,QAAA,iBAAA,CAAA,CAAA;AACA,IAAAK,WAAAL,QAAA,QAAA,CAAA;AAEA,IAAAM,oBAAAN,QAAA,mBAAA,CAAA;AAEA,IAAAO,uBAAAP,QAAA,sBAAA,CAAA;AACA,IAAAQ,uBAAAR,QAAA,sBAAA,CAAA;AACA,IAAAS,gBAAAT,QAAA,4BAAA,CAAA;AACA,IAAAU,kBAAAV,QAAA,4BAAA,CAAA;AAuBA,IAAAW,kBAAAX,QAAA,4BAAA,CAAA;AACA,IAAAY,mBAAAZ,QAAA,+BAAA,CAAA;AACA,IAAAa,mBAAAb,QAAA,kBAAA,CAAA;AAUA,IAAAc,WAAAd,QAAA,yCAAA,CAAA;AAEA,IAAAe,gBAAAf,QAAA,0BAAA,CAAA;AAEA,IAAMgB,QAAQD,cAAAE,cAAA,CAAe,kBAAf,CAAd;AACA,IAAMC,iBAAiBf,WAAAgB,OAAA,CAASH,MAAMI,IAAf,EAAqB,OAAO,CAA5B,EAA+B,EAACC,SAAS,IAAV,EAAgBC,SAAS,OAAO,EAAhC,EAA/B,CAAvB;AAEA,IAAKC,aAAL;AAAA,CAAA,UAAKA,aAAL,EAAkB;AACdA,kBAAA,gBAAA,IAAA,gBAAA;AACAA,kBAAA,kBAAA,IAAA,kBAAA;AACH,CAHD,EAAKA,kBAAAA,gBAAa,EAAb,CAAL;AAKA,IAAAC,0BAAA,aAAA,YAAA;AA6BI,aAAAA,uBAAA,CAAsBC,WAAtB,EAAsDC,iBAAtD,EAAyF;AAjBxE,aAAAC,YAAA,GAA4D,IAAIC,GAAJ,EAA5D;AACA,aAAAC,WAAA,GAA6D,IAAID,GAAJ,EAA7D;AACA,aAAAE,kBAAA,GAA6E,IAAIF,GAAJ,EAA7E;AACP,aAAAG,eAAA,GAAiD,IAAjD;AACF,aAAAC,iBAAA,GAA6B,KAA7B;AACA,aAAAC,qBAAA,GAAiC,IAAjC;AAEA,aAAAC,wBAAA,GAAoD,IAApD;AACA,aAAAC,mBAAA,GAA+C,IAA/C;AACA,aAAAC,kBAAA,GAAgE,IAAIR,GAAJ,EAAhE;AACA,aAAAS,cAAA,GAA0B,KAA1B;AACA,aAAAC,QAAA,GAIH,IAJG;AAOJtB,cAAMI,IAAN,CAAW,iCAAX;AAEA,aAAKmB,gBAAL;AAEA,aAAKC,QAAL,GAAgB,IAAInC,SAAAoC,YAAJ,EAAhB;AACA,aAAKC,YAAL,GAAoBjB,WAApB;AACA,aAAKkB,mBAAL,GAA2B;AACvBC,2BAAe,KADQ;AAEvBC,uBAAW,KAFY;AAGvBC,2BAAe,KAHQ;AAIvBpB,+BAAmBA;AAJI,SAA3B;AAOA,aAAKqB,oBAAL,GAA4B,IAAIxC,qBAAAyC,kBAAJ,CAAuB,KAAKN,YAA5B,CAA5B;AAEA;AACA;AACA,aAAKV,iBAAL,GAAyB,KAAKU,YAAL,CAAkBO,KAAlB,OACpB,CAAC,KAAKP,YAAL,CAAkBQ,YAAlB,EAAD,IAAsC,KAAKR,YAAL,CAAkBQ,YAAlB,MAAoC,EADtD,CAAzB;AAGAlC,cAAMI,IAAN,CAAW,oCAAkC,KAAKY,iBAAlD;AACAhB,cAAMI,IAAN,CAAW,qCAAmC,KAAKsB,YAAL,CAAkBO,KAAlB,EAA9C;AACAjC,cAAMI,IAAN,CAAW,8CAA4C,KAAKsB,YAAL,CAAkBQ,YAAlB,EAAvD;AAEA,aAAKC,sBAAL,GAA8B;AAC1BC,mCAAuB,KAAKC,2BADF;AAE1BC,iCAAqB,KAAKC,wBAFA;AAG1BC,8BAAkB,KAAKC;AAHG,SAA9B;AAKH;AAED7E,WAAAgB,cAAA,CAAI4B,wBAAAnC,SAAJ,EAAI,mBAAJ,EAAqB;AD3CjBqE,aC2CJ,eAAA;AACI,iBAAKC,gBAAL,CAAsB,KAAtB;AAEA,mBAAO1D,YAAAkB,OAAA,CAAU,KAAKwB,mBAAf,CAAP;AACH,SAJoB;ADvCjBiB,oBAAY,ICuCK;ADtCjBC,sBAAc;ACsCG,KAArB;AAMAjF,WAAAgB,cAAA,CAAI4B,wBAAAnC,SAAJ,EAAI,eAAJ,EAAiB;ADzCbqE,aCyCJ,eAAA;AACI,mBAAO,KAAKrB,cAAZ;AACH,SAFgB;ADtCbuB,oBAAY,ICsCC;ADrCbC,sBAAc;ACqCD,KAAjB;AAIAjF,WAAAgB,cAAA,CAAI4B,wBAAAnC,SAAJ,EAAI,iBAAJ,EAAmB;ADtCfqE,aCsCJ,eAAA;AACI,mBAAO,KAAKpB,QAAL,KAAkB,IAAzB;AACH,SAFkB;ADnCfsB,oBAAY,ICmCG;ADlCfC,sBAAc;ACkCC,KAAnB;AAIAjF,WAAAgB,cAAA,CAAI4B,wBAAAnC,SAAJ,EAAI,0BAAJ,EAA4B;ADnCxBqE,aCmCJ,eAAA;AACI,mBAAO,KAAKxB,wBAAL,KAAkC,IAAzC;AACH,SAF2B;ADhCxB0B,oBAAY,ICgCY;AD/BxBC,sBAAc;AC+BU,KAA5B;AAIA;;;;;AAKArC,4BAAAnC,SAAA,CAAAyE,gBAAA,GAAA,UAAiBC,cAAjB,EAAoD;AAApD,YAAAC,QAAA,IAAA;AACIhD,cAAMI,IAAN,CAAW,0CAAX,EAAuD2C,cAAvD;AACA,YAAME,cAAc,yGAApB;AAEA,YAAI,KAAK9B,mBAAL,IAA4B,KAAKJ,eAArC,EAAsD;AAClDmC,oBAAQC,IAAR,CAAaF,WAAb;AAEA,mBAAO,KAAK9B,mBAAL,GAA2B,KAAKA,mBAAhC,GAAsDiC,QAAQC,OAAR,EAA7D;AACH;AAED,aAAKtC,eAAL,GAAuB9B,YAAAkB,OAAA,CAAU4C,cAAV,CAAvB;AACA,YAAMO,YAAY,KAAKvC,eAAL,CAAqBwC,gBAArB,KAA0C1D,iBAAA2D,gBAAA,CAAiBC,OAA7E;AAEA,aAAKtC,mBAAL,GAA2BiC,QAAQM,GAAR,CAAY,CACnC/D,gBAAAgE,4BAAA,CAA6BL,SAA7B,EACKM,IADL,CACU,UAACC,WAAD,EAAqB;AAAK,mBAAAb,MAAKrB,mBAAL,CAAyBC,aAAzB,GAAyCiC,WAAzC;AAAoD,SADxF,CADmC,EAGnClE,gBAAAmE,qBAAA,GACKC,KADL,CACW,UAACC,KAAD,EAAM;AACThE,kBAAMmD,IAAN,CAAWa,KAAX;AAEA,mBAAO,EAAP;AACH,SALL,EAMKJ,IANL,CAMU,UAACK,gBAAD,EAAoC;AACtC,gBAAMC,iBAAiB;AACnBC,uBAAOpF,SAAAoB,OAAA,CAAO8D,gBAAP,EAAyB,UAACG,MAAD,EAAO;AAAK,2BAAAA,OAAOC,IAAP,CAAYC,WAAZ,OAA8B,YAA9B;AAA0C,iBAA/E,CADY;AAEnBC,uBAAOxF,SAAAoB,OAAA,CAAO8D,gBAAP,EAAyB,UAACG,MAAD,EAAO;AAAK,2BAAAA,OAAOC,IAAP,CAAYC,WAAZ,OAA8B,YAA9B;AAA0C,iBAA/E;AAFY,aAAvB;AAKAtB,kBAAKjB,oBAAL,CAA0ByC,iBAA1B,CAA4CN,cAA5C;AACAlB,kBAAKrB,mBAAL,CAAyBE,SAAzB,GAAqCmB,MAAKhC,iBAAL,IAA2BkD,eAAeC,KAAf,CAAqBhG,MAArB,GAA8B,CAA9F;AACA6E,kBAAKrB,mBAAL,CAAyBG,aAAzB,GAAyCkB,MAAKhC,iBAAL,IAA2BkD,eAAeK,KAAf,CAAqBpG,MAArB,GAA8B,CAAlG;AAEH,SAhBL,CAHmC,CAAZ,EAoBxByF,IApBwB,CAoBnB,YAAA;AAAM,mBAAAa,SAAA;AAAS,SApBI,CAA3B;AAuBA,eAAO,KAAKtD,mBAAZ;AACH,KArCD;AAuCA;AAEA;;;;;AAKAX,4BAAAnC,SAAA,CAAAqG,qBAAA,GAAA,YAAA;AACI,aAAK/B,gBAAL;AAEA,YAAI,CAAC,KAAKzB,wBAAV,EAAoC;AAChClB,kBAAMI,IAAN,CAAW,8CAAX;AACA,iBAAKc,wBAAL,GAAgC,KAAKyD,2BAAL,GAC3BZ,KAD2B,CACrB,UAACa,sBAAD,EAA+C;AAClD5E,sBAAMgE,KAAN,CAAY,iCAAZ,EAA+CY,sBAA/C;AAEA,sBAAMA,sBAAN;AACH,aAL2B,CAAhC;AAMH,SARD,MAQO;AACH5E,kBAAMI,IAAN,CAAW,mDAAX;AACH;AAED,eAAO,KAAKc,wBAAZ;AACH,KAhBD;AAkBA;;;;AAIAV,4BAAAnC,SAAA,CAAAwG,gBAAA,GAAA,UAAiBC,gBAAjB,EAAwD;AACpD,YAAI,KAAKnE,YAAL,CAAkBoE,GAAlB,CAAsBD,iBAAiBE,SAAvC,CAAJ,EAAuD;AACnD,mBAAO5B,QAAQ6B,MAAR,CAAe,oDAAf,CAAP;AACH;AAED,YAAMC,aAAa,IAAI5F,kBAAA6F,sBAAJ,CAA2BL,gBAA3B,CAAnB;AAEA,aAAKnE,YAAL,CAAkByE,GAAlB,CAAsBN,iBAAiBE,SAAvC,EAAkDE,UAAlD;AACA,YAAMG,sBAAsB,KAAKC,yBAAL,CAA+BJ,WAAWK,UAA1C,CAA5B;AAEA,YAAIF,mBAAJ,EAAyB;AACrBrF,kBAAMI,IAAN,CAAW,gBAAc0E,iBAAiBS,UAA/B,GAAyC,YAApD;AACAL,uBAAWM,YAAX,CAAwBH,mBAAxB;AACH,SAHD,MAGO;AACHrF,kBAAMI,IAAN,CAAW,gBAAc0E,iBAAiBS,UAA/B,GAAyC,kBAApD;AACH;AAED,eAAOnC,QAAQC,OAAR,CAAgB6B,UAAhB,CAAP;AACH,KAlBD;AAoBA;;;AAGA1E,4BAAAnC,SAAA,CAAAoH,iBAAA,GAAA,UAAkBT,SAAlB,EAA2C;AACvC,YAAI,KAAKrE,YAAL,CAAkBoE,GAAlB,CAAsBC,SAAtB,CAAJ,EAAsC;AAClC,gBAAME,aAAa,KAAKvE,YAAL,CAAkB+B,GAAlB,CAAsBsC,SAAtB,CAAnB;AAEAE,uBAAYQ,OAAZ;AACH;AAED,aAAK/E,YAAL,CAAkBgF,MAAlB,CAAyBX,SAAzB;AAEA,eAAO5B,QAAQC,OAAR,EAAP;AACH,KAVD;AAYA7C,4BAAAnC,SAAA,CAAAuH,iBAAA,GAAA,UAAkBC,QAAlB,EAAyE;AACrE,aAAKC,qBAAL,CAA2BvF,cAAcwF,gBAAzC,EAA2DF,QAA3D;AACH,KAFD;AAIArF,4BAAAnC,SAAA,CAAA2H,eAAA,GAAA,UAAgBH,QAAhB,EAAqE;AACjE,aAAKC,qBAAL,CAA2BvF,cAAc0F,cAAzC,EAAyDJ,QAAzD;AACH,KAFD;AAIQrF,4BAAAnC,SAAA,CAAA6H,0BAAA,GAAR,YAAA;AACI,YAAMC,cAAclH,YAAAkB,OAAA,CAAUiG,OAAOC,2BAAjB,CAApB;AAEA,YAAI,CAACF,WAAL,EAAkB;AACd,kBAAM,IAAIG,KAAJ,CAAU,sEAAV,CAAN;AACH;AAEDtG,cAAMI,IAAN,CAAW,oEAAX,EAAiF+F,WAAjF;AAEA,YAAMI,kBAAmBJ,YAAYhC,KAAZ,CAAkDoC,eAAlD,KAAsE1G,iBAAA2G,oBAAA,CAAqBC,MAA3F,GACnB5G,iBAAA2G,oBAAA,CAAqBE,YADF,GAEnB7G,iBAAA2G,oBAAA,CAAqBC,MAF3B;AAIAzG,cAAMI,IAAN,CAAW,6DAA2DmG,eAAtE;AAECJ,oBAAYhC,KAAZ,CAAkDoC,eAAlD,GAAoEA,eAApE;AAED,eAAOJ,WAAP;AACH,KAlBO;AAoBR3F,4BAAAnC,SAAA,CAAAsI,YAAA,GAAA,UAAaC,QAAb,EAA+B;AAA/B,YAAA5D,QAAA,IAAA;AACIhD,cAAMI,IAAN,CAAW,qBAAX;AAEA,YAAM+F,cAAc,CAACS,QAAD,GAAY,KAAKV,0BAAL,EAAZ,GAAgDjH,YAAAkB,OAAA,CAAUiG,OAAOC,2BAAjB,CAApE;AAEA,YAAI,CAACF,WAAL,EAAkB;AACd,kBAAM,IAAIG,KAAJ,CAAU,yDAAV,CAAN;AACH;AAED,aAAKvE,oBAAL,CAA0B8E,sBAA1B;AAEA,eAAO,KAAK9E,oBAAL,CAA0B+E,kBAA1B,CAA6CX,WAA7C,EACFvC,IADE,CACG,UAACmD,YAAD,EAAwC;AAC1C,gBAAMC,SAASD,aAAaE,WAA5B;AAEAjH,kBAAMI,IAAN,CAAW,4BAAX,EAAyC4G,MAAzC;AAEAhE,kBAAKkE,mBAAL,CAAyBF,MAAzB;AAEA,mBAAOhE,MAAKmE,oBAAL,CAA0BhB,WAA1B,EAAuCY,YAAvC,EAAqD,IAArD,EAA2DnD,IAA3D,CAAgE,YAAA;AAAM,uBAAC,EAACuC,aAAWA,WAAZ,EAAcY,cAAYA,YAA1B,EAAD;AAA6B,aAAnG,CAAP;AACH,SATE,EAUFhD,KAVE,CAUI,UAACqD,GAAD,EAAI;AACPpH,kBAAMgE,KAAN,CAAY,yCAAZ,EAAuDoD,GAAvD;AAEA,gBAAI,CAACR,QAAL,EAAe;AACX5D,sBAAK2D,YAAL,CAAkB,IAAlB;AACH;AAED,mBAAOvD,QAAQ6B,MAAR,CAAe,0BAAf,CAAP;AACH,SAlBE,CAAP;AAmBH,KA9BD;AAgCQzE,4BAAAnC,SAAA,CAAA6I,mBAAA,GAAR,UAA4BD,WAA5B,EAA8D;AAC1D,aAAK3F,QAAL,CAAe+F,QAAf,IAA2B,KAAK/F,QAAL,CAAe+F,QAAf,CAAwBH,mBAAxB,CAA4CD,WAA5C,CAA3B;AACH,KAFO;AAIEzG,4BAAAnC,SAAA,CAAA8I,oBAAA,GAAV,UAA+BhB,WAA/B,EAA6EY,YAA7E,EAAsHJ,YAAtH,EAA8IW,aAA9I,EAA8K;AAA9K,YAAAtE,QAAA,IAAA;AACI,YAAMuE,cAAqB,EAA3B;AAEA,YAAIR,aAAaS,KAAjB,EAAwB;AACpBxH,kBAAMI,IAAN,CAAW,8DAAX,EAA2E2G,aAAaE,WAAxF;AACAF,yBAAaE,WAAb,CAAyBQ,SAAzB,GAAqCC,OAArC,CAA6C,UAACC,UAAD,EAAW;AACpD3H,sBAAMI,IAAN,CAAW,sDAAX,EAAmEuH,UAAnE;AACA,oBAAMpC,aAAa+B,kBAAkBK,WAAWtD,IAAX,KAAoB,OAApB,GAC/BxE,iBAAA+H,gBAAA,CAAiBC,iBADc,GACMhI,iBAAA+H,gBAAA,CAAiBE,iBADzC,CAAnB;AAEA,oBAAMC,qBAAqB,IAAIvI,qBAAAwI,kBAAJ,CAAuBL,UAAvB,EAAmCpC,UAAnC,EAA+C,KAA/C,CAA3B;AAEAgC,4BAAYU,IAAZ,CAAiBtB,eAAe3D,MAAKkF,aAAL,CAAmBH,kBAAnB,CAAf,GAAwD/E,MAAKmF,wBAAL,CAA8BJ,kBAA9B,CAAzE;AACH,aAPD;AAQH;AAED,eAAO3E,QAAQM,GAAR,CAAY6D,WAAZ,CAAP;AACH,KAhBS;AAkBV3J,WAAAgB,cAAA,CAAI4B,wBAAAnC,SAAJ,EAAI,sBAAJ,EAAwB;AD3EpBqE,aC2EJ,eAAA;AACI,mBAAO,KAAK1B,iBAAL,IAA0B,KAAKe,oBAAL,CAA0BmC,cAA1B,CAAyCkE,YAAzC,GAAwD,CAAzF;AACH,SAFuB;ADxEpBxF,oBAAY,ICwEQ;ADvEpBC,sBAAc;ACuEM,KAAxB;AAIA;AAEA;AAEA;;;AAGArC,4BAAAnC,SAAA,CAAAgK,gBAAA,GAAA,UAAiBC,aAAjB,EAAkD;AAAlD,YAAAtF,QAAA,IAAA;AACI,aAAKL,gBAAL;AAEA,YAAI,CAAC,KAAKrB,QAAV,EAAoB;AAChB,iBAAKA,QAAL,GAAgB,EAACiH,QAAQtJ,YAAAkB,OAAA,CAAUmI,aAAV,CAAT,EAAhB;AACA,iBAAKhH,QAAL,CAAckH,cAAd,GAA+B,IAAIpF,OAAJ,CAAY,UAACC,OAAD,EAAU4B,MAAV,EAAgB;AACvD,oBAAMwD,gBAAgB,SAAhBA,aAAgB,CAACzE,KAAD,EAAW;AAE7B;;;;;AAKA0E,+BAAW,YAAA;AACP1F,8BAAK1B,QAAL,GAAgB,IAAhB;AACA2D,+BAAOjB,KAAP;AACH,qBAHD;AAIH,iBAXD;AAaA,oBAAM2E,kBAAwC;AAC1CC,2CAAuB5F,MAAK6F;AADc,iBAA9C;AAGA,oBAAMC,qBAAkBnL,SAAA,EAAA,EACjB2K,aADiB,EACJ,EAChBS,gCAAgC/F,MAAKjC,eAAL,CAAsBgI,8BAAtB,IAAwDlJ,iBAAAmJ,yCADxE,EADI,CAAxB;AAKAhJ,sBAAMI,IAAN,CAAW,kBAAX,EAA+BkI,aAA/B;AACA,oBAAItF,MAAKjC,eAAL,CAAsBwC,gBAAtB,KAA2C1D,iBAAA2D,gBAAA,CAAiByF,UAAhE,EAA4E;AACxEjG,0BAAK1B,QAAL,CAAe+F,QAAf,GAA0B,IAAI5H,cAAAyJ,iBAAJ,CAAsBJ,kBAAtB,EAA0CH,eAA1C,EAA2D3F,MAAKb,sBAAhE,CAA1B;AAEH,iBAHD,MAGO,IAAIa,MAAKjC,eAAL,CAAsBwC,gBAAtB,KAA2C1D,iBAAA2D,gBAAA,CAAiBC,OAAhE,EAAyE;AAC5ET,0BAAK1B,QAAL,CAAe+F,QAAf,GAA0B,IAAIzH,iBAAAuJ,cAAJ,CAAmBL,kBAAnB,EAAuC9F,MAAKb,sBAA5C,CAA1B;AACH,iBAFM,MAEA;AACH,wBAAMiH,gBAAgBlK,SAAAiB,OAAA,CAAON,iBAAA2D,gBAAP,CAAtB;AAEAxD,0BAAMgE,KAAN,CAAY,uBAAqBhB,MAAKjC,eAAL,CAAsBwC,gBAA3C,GAA2D,qBAAvE,EAA8F,EAAC6F,eAAaA,aAAd,EAA9F;AACAX,kCAAc,IAAInC,KAAJ,CAAU,mCAAV,CAAd;AACH;AAED,oBAAItD,MAAK1B,QAAL,IAAiB0B,MAAK1B,QAAL,CAAc+F,QAAnC,EAA6C;AACzCrE,0BAAK1B,QAAL,CAAe+F,QAAf,CAAwBgC,OAAxB,GACKzF,IADL,CACU,YAAA;AACF5D,8BAAMI,IAAN,CAAW,sBAAX;AACAiD;AACH,qBAJL,EAKKU,KALL,CAKW,UAACC,KAAD,EAAW;AACdhE,8BAAMgE,KAAN,CAAY,8BAAZ,EAA4CA,KAA5C;AACAyE,sCAAczE,KAAd;AACH,qBARL;AASH;AAEJ,aA/C8B,CAA/B;AAgDH,SAlDD,MAkDO;AACHhE,kBAAMmD,IAAN,CAAW,2EAAX;AACH;AAED;AACA;AACA;AACA,eAAO,KAAK7B,QAAL,CAAekH,cAAtB;AACH,KA7DD;AA+DA;;;AAGAhI,4BAAAnC,SAAA,CAAAiL,qBAAA,GAAA,YAAA;AACI,eAAO,KAAKC,6BAAL,CAAmC1J,iBAAA2J,4BAAA,CAA6BC,gBAAhE,CAAP;AACH,KAFD;AAIA;;;AAGAjJ,4BAAAnC,SAAA,CAAAqL,wBAAA,GAAA,UAAyBC,WAAzB,EAAgE;AAC5D3J,cAAMI,IAAN,CAAW,0BAAX;AACA,YAAI,CAAC,KAAKwJ,kBAAL,CAAwB,KAAxB,CAAL,EAAqC;AACjC,mBAAOxG,QAAQ6B,MAAR,CAAe,wCAAf,CAAP;AACH;AAED,aAAK3D,QAAL,CAAeiH,MAAf,CAAsBoB,WAAtB,GAAoC1K,YAAAkB,OAAA,CAAUwJ,WAAV,CAApC;AAEA,eAAOvG,QAAQC,OAAR,EAAP;AACH,KATD;AAWA7C,4BAAAnC,SAAA,CAAAwL,uBAAA,GAAA,YAAA;AACI,YAAI,CAAC,KAAKxI,cAAN,IAAwB,KAAKyI,wBAAjC,EAA2D;AACvD,kBAAM,IAAIxD,KAAJ,CAAU,qEAAV,CAAN;AACH;AAED,aAAKjF,cAAL,GAAsB,IAAtB;AACH,KAND;AAQUb,4BAAAnC,SAAA,CAAAkL,6BAAA,GAAV,UAAwCQ,MAAxC,EAA4E;AACxE,YAAI,CAAC,KAAKzI,QAAV,EAAoB;AAChB,mBAAO8B,QAAQC,OAAR,EAAP;AACH;AAEDrD,cAAMI,IAAN,CAAW,4BAAX,EAAyC2J,MAAzC;AAEA,YAAMC,cAAc,KAAK1I,QAAzB;AAEA,aAAKA,QAAL,GAAgB,IAAhB;AAEA,eAAO8B,QAAQM,GAAR,CAAY,CACfsG,YAAY3C,QAAZ,CAAsB4C,UAAtB,EADe,EAEf,KAAKC,eAAL,EAFe,CAAZ,EAGJtG,IAHI,CAGC,YAAA;AAAM,mBAAAa,SAAA;AAAS,SAHhB,CAAP;AAIH,KAfS;AAiBAjE,4BAAAnC,SAAA,CAAA8L,sBAAA,GAAV,UAAiC5E,UAAjC,EAA6D;AACzD,YAAI,CAAC,KAAKqE,kBAAL,CAAwB,KAAxB,CAAL,EAAqC;AACjC,mBAAOxG,QAAQ6B,MAAR,CAAe,IAAIqB,KAAJ,CAAU,2BAAV,CAAf,CAAP;AACH;AAED,YAAM8D,iBAAiB,KAAK9E,yBAAL,CAA+BC,UAA/B,CAAvB;AAEA,YAAI,CAAC6E,cAAL,EAAqB;AACjB,mBAAOhH,QAAQ6B,MAAR,CAAe,IAAIqB,KAAJ,CAAU,yCAAV,CAAf,CAAP;AACH;AAED,eAAO,KAAKhF,QAAL,CAAe+F,QAAf,CAAyBgD,mBAAzB,CAA6CD,eAAezC,UAA5D,EACF/D,IADE,CACG,UAAC0G,UAAD,EAA6B;AAC/BpK,2BAAe,yBAAuBqF,UAAvB,GAAiC,IAAjC,GAAsCgF,KAAKC,SAAL,CAAeF,UAAf,CAArD;AAEA,mBAAO;AACHG,yBAASL,eAAezC,UAAf,CAA0B+C,EADhC;AAEHJ,4BAAYA;AAFT,aAAP;AAIH,SARE,CAAP;AASH,KApBS;AAsBF9J,4BAAAnC,SAAA,CAAAwK,0BAAA,GAAR,UAAmCkB,MAAnC,EAAuE;AAAvE,YAAA/G,QAAA,IAAA;AACIhD,cAAMI,IAAN,CAAW,4BAAX,EAAyC2J,MAAzC;AACA,YAAIA,WAAWlK,iBAAA2J,4BAAA,CAA6BC,gBAA5C,EAA8D;AAC1D,gBAAMkB,eAAa,KAAKrJ,QAAL,IAAiB,KAAKA,QAAL,CAAciH,MAA/B,GAAwC,KAAKjH,QAAL,CAAciH,MAAtD,GAA+D,IAAlF;AAEA,iBAAKgB,6BAAL,CAAmCQ,MAAnC,EAA2CnG,IAA3C,CAAgD,YAAA;AAC5C,oBAAMgH,mBAAmB,CACrB/K,iBAAA2J,4BAAA,CAA6BqB,wBADR,EAErBhL,iBAAA2J,4BAAA,CAA6BsB,0BAFR,EAGrBjL,iBAAA2J,4BAAA,CAA6BuB,yBAHR,EAIrBlL,iBAAA2J,4BAAA,CAA6BwB,gCAJR,CAAzB;AAOA,oBAAIL,gBAAcvL,WAAAe,OAAA,CAASyK,gBAAT,EAA2Bb,MAA3B,CAAd,IAAoD/G,MAAK/B,qBAA7D,EAAoF;AAChF+B,0BAAKiI,kBAAL,CAAwBN,YAAxB;AACH,iBAFD,MAEO,IAAI,CAAC3H,MAAK/B,qBAAV,EAAiC;AACpCjB,0BAAMI,IAAN,CAAW,0CAAX;AACH,iBAFM,MAEA;AACHJ,0BAAMI,IAAN,CAAW,6CAAX;AACH;AACJ,aAfD;AAgBH,SAnBD,MAmBO;AACHJ,kBAAMI,IAAN,CAAW,iCAAX;AACH;AACJ,KAxBO;AA0BRI,4BAAAnC,SAAA,CAAA6M,gBAAA,GAAA,UAAiBC,KAAjB,EAA+B;AAC3B,aAAKlK,qBAAL,GAA6BkK,KAA7B;AACAnL,cAAMI,IAAN,CAAW,wBAAX,EAAqC+K,KAArC;AAEA,YAAI,KAAKlK,qBAAL,IAA8B,CAAC,KAAKmK,eAAxC,EAAyD;AACrD,gBAAMC,aAAa,KAAK/J,QAAL,IAAiB,KAAKA,QAAL,CAAciH,MAA/B,GAAwC,KAAKjH,QAAL,CAAciH,MAAtD,GAA+D,IAAlF;AAEA,gBAAI8C,UAAJ,EAAgB;AACZrL,sBAAMI,IAAN,CAAW,yCAAX;AACA,qBAAK6K,kBAAL,CAAwBI,UAAxB;AACH;AACJ;AACJ,KAZD;AAcA7K,4BAAAnC,SAAA,CAAA4M,kBAAA,GAAA,UAAmBI,UAAnB,EAAiD;AAC7CrL,cAAMI,IAAN,CAAW,+CAAX,EAA4DiL,UAA5D;AAEA,aAAKhD,gBAAL,CAAsBgD,UAAtB,EACKzH,IADL,CACU,YAAA;AACF5D,kBAAMI,IAAN,CAAW,2BAAX;AACH,SAHL,EAIK2D,KAJL,CAIW,UAACC,KAAD,EAAW;AACdhE,kBAAMgE,KAAN,CAAY,2CAAZ,EAAyDA,KAAzD;AACH,SANL;AAOH,KAVD;AAYA;AAEA;AAEAxD,4BAAAnC,SAAA,CAAAiN,0BAAA,GAAA,UAA2BC,YAA3B,EAA2DC,eAA3D,EAA4F;AAA5F,YAAAxI,QAAA,IAAA;AACI,eAAO,IAAII,OAAJ,CAAY,UAACC,OAAD,EAAU4B,MAAV,EAAgB;AAC/B,gBAAIsG,iBAAiB1L,iBAAA+H,gBAAA,CAAiBC,iBAAlC,IACA0D,iBAAiB1L,iBAAA+H,gBAAA,CAAiB6D,wBADtC,EACgE;AAC5DzL,sBAAMgE,KAAN,CAAY,2EAAZ;AACAiB,uBAAO,IAAIqB,KAAJ,CAAU,kCAAV,CAAP;AAEA;AACH;AAED,gBAAMoF,iBAAiB1I,MAAKsC,yBAAL,CAA+BiG,YAA/B,CAAvB;AAEA,gBAAI,CAACG,cAAL,EAAqB;AACjB1L,sBAAMgE,KAAN,CAAY,4DAAZ;AACAiB,uBAAO,IAAIqB,KAAJ,CAAU,qBAAV,CAAP;AAEA;AACH;AAED5G,4BAAAiM,0BAAA,CAA2BD,eAAezE,WAA1C,EAAuDuE,eAAvD,EACK5H,IADL,CACU,UAACgI,SAAD,EAAe;AACjB,oBAAMC,gBAAgBD,UAAUE,KAAV,CAAgB,UAAhB,CAAtB;AACA,oBAAMC,WAAWF,cAAc,CAAd,EAAiBC,KAAjB,CAAuB,GAAvB,EAA4B,CAA5B,CAAjB;AACA,oBAAME,QAAQC,KAAKJ,cAAc,CAAd,CAAL,CAAd;AACA,oBAAMK,SAAS,IAAIC,WAAJ,CAAgBH,MAAM7N,MAAtB,CAAf;AACA,oBAAMiO,UAAU,IAAIC,UAAJ,CAAeH,MAAf,CAAhB;AAEA,qBAAK,IAAIlO,IAAI,CAAb,EAAgBA,IAAIgO,MAAM7N,MAA1B,EAAkCH,GAAlC,EAAuC;AACnCoO,4BAAQpO,CAAR,IAAagO,MAAMM,UAAN,CAAiBtO,CAAjB,CAAb;AACH;AAED,oBAAMuO,OAAO,IAAIC,IAAJ,CAAS,CAACJ,OAAD,CAAT,EAAoB,EAACK,MAAMV,QAAP,EAApB,CAAb;AACA,oBAAMW,YAAYtG,OAAOuG,GAAP,CAAWC,eAAX,CAA2BL,IAA3B,CAAlB;AAEAvM,sBAAMI,IAAN,CAAW,+BAAX;AAEA,oBAAMyM,SAAyB;AAC3BC,+BAAWlB,SADgB;AAE3Bc,+BAAWA,SAFgB;AAG3BK,+BAAWR,IAHgB;AAI3BR,8BAAUA;AAJiB,iBAA/B;AAOA1I,wBAAQwJ,MAAR;AACH,aAzBL,EA0BK9I,KA1BL,CA0BW,UAACC,KAAD,EAAW;AACdhE,sBAAMgE,KAAN,CAAY,yBAAZ,EAAuCA,KAAvC;AACAiB,uBAAOjB,KAAP;AACH,aA7BL;AA8BH,SAhDM,CAAP;AAiDH,KAlDD;AAoDA;AACAxD,4BAAAnC,SAAA,CAAA2O,YAAA,GAAA,YAAA;AAAA,YAAAhK,QAAA,IAAA;AACIhD,cAAMI,IAAN,CAAW,qCAAX;AAEA,eAAO,KAAKmJ,6BAAL,CAAmC1J,iBAAA2J,4BAAA,CAA6BC,gBAAhE,EACF7F,IADE,CACG,KAAKqJ,gBADR,EAEFrJ,IAFE,CAEG,KAAKsJ,sBAFR,EAGFtJ,IAHE,CAGG,KAAK7B,oBAAL,CAA0BoL,eAH7B,EAIFvJ,IAJE,CAIG,YAAA;AACFZ,kBAAK9B,wBAAL,GAAgC,IAAhC;AACA8B,kBAAK3B,cAAL,GAAsB,KAAtB;AACH,SAPE,EAOAuC,IAPA,CAOK,YAAA;AAAM,mBAAAa,SAAA;AAAS,SAPpB,CAAP;AAQH,KAXD;AAaA;AACAjE,4BAAAnC,SAAA,CAAAqH,OAAA,GAAA,YAAA;AAAA,YAAA1C,QAAA,IAAA;AACI,eAAO,KAAKgK,YAAL,GAAoBjJ,KAApB,CAA0B,YAAA;AAAM,mBAAAU,SAAA;AAAS,SAAzC,EAA2Cb,IAA3C,CAAgD,YAAA;AACnDZ,kBAAKxB,QAAL,CAAc4L,kBAAd;AACH,SAFM,CAAP;AAGH,KAJD;AAUAxP,WAAAgB,cAAA,CAAc4B,wBAAAnC,SAAd,EAAc,aAAd,EAAyB;AAJzB;AAEA;ADpHIqE,aCsHJ,eAAA;AACI,mBAAO,KAAK/B,YAAZ;AACH,SAFwB;ADnHrBiC,oBAAY,ICmHS;ADlHrBC,sBAAc;ACkHO,KAAzB;AAIUrC,4BAAAnC,SAAA,CAAA8J,wBAAA,GAAV,UAAmCkF,aAAnC,EAAoE;AAChErN,cAAMI,IAAN,CAAW,0BAAX,EAAuCiN,cAAc9H,UAArD;AACA,YAAI,KAAKnE,kBAAL,CAAwB2D,GAAxB,CAA4BsI,cAAc9H,UAA1C,CAAJ,EAA2D;AACvD,mBAAOnC,QAAQ6B,MAAR,CAAe,IAAIqB,KAAJ,CAAU,YAAU+G,cAAc9H,UAAxB,GAAkC,qBAA5C,CAAf,CAAP;AACH;AAED,eAAO,KAAK+H,cAAL,CAAoBD,aAApB,CAAP;AACH,KAPS;AASA7M,4BAAAnC,SAAA,CAAAiH,yBAAA,GAAV,UAAoCC,UAApC,EAAgE;AAC5D,eAAO,KAAKnE,kBAAL,CAAwBsB,GAAxB,CAA4B6C,UAA5B,KAA2C,IAAlD;AACH,KAFS;AAIA/E,4BAAAnC,SAAA,CAAAkP,0BAAA,GAAV,UAAqChI,UAArC,EAAmEiI,QAAnE,EAAoF;AAChF,YAAMC,sBAAsB,KAAKnI,yBAAL,CAA+BC,UAA/B,CAA5B;AAEA,YAAI,CAACkI,mBAAL,EAA0B;AACtBzN,kBAAMmD,IAAN,CAAW,8CAAX,EAA2DoC,UAA3D;AACH,SAFD,MAEO;AACHkI,gCAAoB9F,UAApB,CAA+B+F,OAA/B,GAAyC,CAACF,QAA1C;AAEA,gBAAIC,oBAAoBE,QAApB,IAAgCF,oBAAoBG,UAApB,KAAmC/N,iBAAAgO,oBAAA,CAAqBC,KAA5F,EAAmG;AAC/F,qBAAKC,WAAL,CAAiBrG,OAAjB,CAAyB,UAACxC,UAAD,EAAW;AAChC,wBAAIA,WAAWK,UAAX,KAA0BA,UAA9B,EAA0C;AACtCL,mCAAW8I,SAAX,CAAqBR,QAArB;AACH;AACJ,iBAJD;AAKH;AAEDxN,kBAAMI,IAAN,CAAW,sCAAX,EAAmD,EAACmF,YAAUA,UAAX,EAAaiI,UAAQA,QAArB,EAAnD;AACH;AACJ,KAlBS;AAoBAhN,4BAAAnC,SAAA,CAAAyH,qBAAA,GAAV,UAAgCmI,KAAhC,EAA+CpI,QAA/C,EAAkF;AAC9E,aAAKrE,QAAL,CAAc0M,EAAd,CAAiBD,KAAjB,EAAwBpI,QAAxB;AACH,KAFS;AAIArF,4BAAAnC,SAAA,CAAA8P,SAAA,GAAV,UAAoBF,KAApB,EAAmCG,SAAnC,EAAkD;AAAlD,YAAApL,QAAA,IAAA;AACI0F,mBAAW,YAAA;AACP1F,kBAAKxB,QAAL,CAAc6M,IAAd,CAAmBJ,KAAnB,EAA0BG,SAA1B;AACH,SAFD;AAGH,KAJS;AAMV;AAEA;AAEQ5N,4BAAAnC,SAAA,CAAAkE,wBAAA,GAAR,UAAiC+L,gBAAjC,EAAmE;AAC/DtO,cAAMI,IAAN,CAAW,wBAAsBkO,iBAAiBC,SAAvC,GAAgD,mBAA3D,EAAgFD,iBAAiB3G,UAAjG;AACA,YAAM6G,gBAAgB,KAAKpN,kBAAL,CAAwBsB,GAAxB,CAA4B4L,iBAAiBC,SAA7C,CAAtB;AAEA,YAAIC,iBAAiB,CAACA,cAAcb,QAApC,EAA8C;AAC1C,mBAAOvK,QAAQ6B,MAAR,CAAe,IAAIqB,KAAJ,CAAU,kEAAV,CAAf,CAAP;AACH;AAED,YAAMyB,qBAAqB,IAAIvI,qBAAAwI,kBAAJ,CAAuBsG,iBAAiB3G,UAAxC,EAAoD2G,iBAAiBC,SAArE,EAAgF,IAAhF,CAA3B;AAEA,eAAO,KAAKjB,cAAL,CAAoBvF,kBAApB,CAAP;AACH,KAXO;AAaAvH,4BAAAnC,SAAA,CAAA6J,aAAA,GAAR,UAAsBjB,WAAtB,EAAqD;AAArD,YAAAjE,QAAA,IAAA;AACIiE,oBAAYU,UAAZ,CAAuB8G,OAAvB,GAAiC,YAAA;AAC7B,mBAAAzL,MAAK0L,4BAAL,CAAkCzH,WAAlC,EAA+CpH,iBAAA8O,2BAAA,CAA4BC,WAA3E,CAAA;AAAuF,SAD3F;AAGA,aAAKxN,kBAAL,CAAwBgE,GAAxB,CAA4B6B,YAAY1B,UAAxC,EAAoD0B,WAApD;AAEAjH,cAAMI,IAAN,CAAW,qCAAmC6G,YAAY1B,UAA1D,EAAwE/F,qBAAAwI,kBAAxE;AAEA,eAAO,KAAK6G,8BAAL,CAAoC5H,WAApC,EAAiDrD,IAAjD,CAAsD,YAAA;AACzD,gBAAMwK,YAAoC,EAAC7I,YAAY0B,YAAY1B,UAAzB,EAA1C;AAEAvC,kBAAKmL,SAAL,CAAe5N,cAAc0F,cAA7B,EAA6CmI,SAA7C;AACH,SAJM,CAAP;AAKH,KAbO;AAeA5N,4BAAAnC,SAAA,CAAAiP,cAAA,GAAR,UAAuBrG,WAAvB,EAAsD;AAAtD,YAAAjE,QAAA,IAAA;AACI,YAAMwL,gBAAgB,KAAKpN,kBAAL,CAAwBsB,GAAxB,CAA4BuE,YAAY1B,UAAxC,CAAtB;AAEA,YAAIiJ,aAAJ,EAAmB;AACf,mBAAO,KAAKE,4BAAL,CAAkCF,aAAlC,EAAiD3O,iBAAA8O,2BAAA,CAA4BG,eAA7E,EAA8FlL,IAA9F,CAAmG,YAAA;AAAM,uBAAAZ,MAAKkF,aAAL,CAAmBjB,WAAnB,CAAA;AAA+B,aAAxI,CAAP;AACH;AAED,eAAO,KAAKiB,aAAL,CAAmBjB,WAAnB,CAAP;AACH,KARO;AAUAzG,4BAAAnC,SAAA,CAAAqQ,4BAAA,GAAR,UAAqCK,kBAArC,EAA6EhF,MAA7E,EAAgH;AAAhH,YAAA/G,QAAA,IAAA;AACI,YAAMgM,aAAa,SAAbA,UAAa,CAAChL,KAAD,EAAW;AAC1BhE,kBAAMmD,IAAN,CAAW,sCAAX,EAAmDa,KAAnD;AACH,SAFD;AAIA,YAAMiL,WAAgB,EAAtB;AAEAjP,cAAMI,IAAN,CAAW,iCAAX,EAA8C2O,mBAAmBxJ,UAAjE;AAEA,aAAK5E,YAAL,CAAkB+G,OAAlB,CAA0B,UAACxC,UAAD,EAAW;AACjC,gBAAIA,WAAWK,UAAX,KAA0BwJ,mBAAmBxJ,UAAjD,EAA6D;AACzD0J,yBAAShH,IAAT,CAAc/C,WAAWgK,aAAX,GAA2BnL,KAA3B,CAAiCiL,UAAjC,CAAd;AACH;AACJ,SAJD;AAMA,YAAI,CAACD,mBAAmBpB,QAAxB,EAAkC;AAC9B,iBAAK9M,WAAL,CAAiB6G,OAAjB,CAAyB,UAACyH,SAAD,EAAiC;AACtD,oBAAIA,UAAUC,WAAV,CAAsBC,IAAtB,CAA2B,UAAC9J,UAAD,EAAW;AAAK,2BAAAA,eAAewJ,mBAAmBxJ,UAAlC;AAA4C,iBAAvF,CAAJ,EAA8F;AAC1F,wBAAIvC,MAAK1B,QAAL,IAAiB0B,MAAK1B,QAAL,CAAe+F,QAApC,EAA8C;AAC1C4H,iCAAShH,IAAT,CAAcjF,MAAK1B,QAAL,CAAc+F,QAAd,CAAuBiI,sBAAvB,CAA8CH,UAAUI,eAAxD,EAAyExL,KAAzE,CAA+EiL,UAA/E,CAAd;AACH;AACJ;AACJ,aAND;AAOH;AAED,eAAO5L,QAAQM,GAAR,CAAYuL,QAAZ,EACFrL,IADE,CACG,YAAA;AACFZ,kBAAK5B,kBAAL,CAAwBuE,MAAxB,CAA+BoJ,mBAAmBxJ,UAAlD;AACAvF,kBAAMI,IAAN,CAAW,gBAAX,EAA6B2O,mBAAmBxJ,UAAhD;AAEA,gBAAM6I,YAAsC,EAAC7I,YAAYwJ,mBAAmBxJ,UAAhC,EAA4CwE,QAAQA,MAApD,EAA5C;AAEA/G,kBAAKmL,SAAL,CAAe5N,cAAcwF,gBAA7B,EAA+CqI,SAA/C;AACH,SARE,CAAP;AASH,KAlCO;AAoCA5N,4BAAAnC,SAAA,CAAAoE,0BAAA,GAAR,UAAmCkF,UAAnC,EAA+D;AAA/D,YAAA3E,QAAA,IAAA;AACI,YAAMiM,WAAgB,EAAtB;AAEA,aAAK7N,kBAAL,CAAwBsG,OAAxB,CAAgC,UAAC8H,gBAAD,EAAqC;AACjE,gBAAIA,iBAAiB7H,UAAjB,CAA4B+C,EAA5B,KAAmC/C,WAAW+C,EAAlD,EAAsD;AAClDuE,yBAAShH,IAAT,CAAcjF,MAAK0L,4BAAL,CAAkCc,gBAAlC,EAAoD3P,iBAAA8O,2BAAA,CAA4Bc,cAAhF,CAAd;AACH;AACJ,SAJD;AAMA,eAAOrM,QAAQM,GAAR,CAAYuL,QAAZ,EAAsBrL,IAAtB,CAA2B,YAAA;AAAM,mBAAAa,SAAA;AAAS,SAA1C,CAAP;AACH,KAVO;AAYAjE,4BAAAnC,SAAA,CAAAqR,eAAA,GAAR,UAAwBP,SAAxB,EAAwD;AACpD,aAAKtO,WAAL,CAAiB8E,MAAjB,CAAwBwJ,UAAUI,eAAlC;AACA,YAAI,KAAKjO,QAAL,IAAiB,KAAKA,QAAL,CAAc+F,QAAnC,EAA6C;AACzCrH,kBAAMI,IAAN,CAAW,oBAAX,EAAiC+O,UAAUI,eAA3C;AAEA,mBAAO,KAAKjO,QAAL,CAAc+F,QAAd,CAAuBiI,sBAAvB,CAA8CH,UAAUI,eAAxD,EACFxL,KADE,CACI,UAACC,KAAD,EAAW;AACdhE,sBAAMmD,IAAN,CAAW,gCAAX,EAA6Ca,KAA7C;AACH,aAHE,CAAP;AAIH;AAED,eAAOZ,QAAQC,OAAR,EAAP;AACH,KAZO;AAcA7C,4BAAAnC,SAAA,CAAAgE,2BAAA,GAAR,UAAoCkN,eAApC,EAAsE;AAAtE,YAAAvM,QAAA,IAAA;AACI,aAAKL,gBAAL;AAEA,YAAI,KAAK9B,WAAL,CAAiBkE,GAAjB,CAAqBwK,eAArB,CAAJ,EAA2C;AACvCvP,kBAAMI,IAAN,CAAW,mBAAiBmP,eAAjB,GAAgC,iBAA3C;AAEA,mBAAOnM,QAAQC,OAAR,CAAgB,KAAKxC,WAAL,CAAiB6B,GAAjB,CAAqB6M,eAArB,EAAuCtI,WAAvD,CAAP;AACH;AAED,YAAI,CAAC,KAAKnG,kBAAL,CAAwB4B,GAAxB,CAA4B6M,eAA5B,CAAL,EAAmD;AAC/CvP,kBAAMI,IAAN,CAAW,4BAA0BmP,eAArC;AACA,gBAAMI,mBAAmB,KAAKjL,qBAAL,GACpBd,IADoB,CACf,YAAA;AAAM,uBAAAZ,MAAK4M,oBAAL,CAA0BL,eAA1B,CAAA;AAA0C,aADjC,EAEpB3L,IAFoB,CAEf,UAACiM,cAAD,EAAgD;AAClD,oBAAIA,cAAJ,EAAoB;AAChB7M,0BAAKnC,WAAL,CAAiBuE,GAAjB,CAAqBmK,eAArB,EAAsCM,cAAtC;AACH;AAED7M,sBAAKlC,kBAAL,CAAwB6E,MAAxB,CAA+B4J,eAA/B;AAEA,uBAAOM,iBAAiBA,eAAe5I,WAAhC,GAA8C,IAArD;AACH,aAVoB,EAWpBlD,KAXoB,CAWd,UAAC+L,EAAD,EAAQ;AACX,oBAAIA,MAAMA,GAAGC,OAAH,KAAe,mBAAzB,EAA8C;AAC1C/M,0BAAK3B,cAAL,GAAsB,KAAtB;AAEA,2BAAO,IAAP;AACH;AAED2B,sBAAKlC,kBAAL,CAAwB6E,MAAxB,CAA+B4J,eAA/B;AACA,sBAAMO,EAAN;AACH,aApBoB,CAAzB;AAsBA,iBAAKhP,kBAAL,CAAwBsE,GAAxB,CAA4BmK,eAA5B,EAA6CI,gBAA7C;AACH,SAzBD,MAyBO;AACH3P,kBAAMI,IAAN,CAAW,kCAAgCmP,eAAhC,GAA+C,iBAA1D;AACH;AAED,eAAO,KAAKzO,kBAAL,CAAwB4B,GAAxB,CAA4B6M,eAA5B,CAAP;AACH,KAvCO;AAyCA/O,4BAAAnC,SAAA,CAAAwQ,8BAAA,GAAR,UAAuCW,gBAAvC,EAA2E;AACvExP,cAAMI,IAAN,CAAW,6BAA2BoP,iBAAiBjK,UAA5C,GAAsD,kBAAjE;AAEA,YAAI,KAAK5E,YAAL,CAAkBqP,IAAlB,KAA2B,CAA/B,EAAkC;AAC9BhQ,kBAAMmD,IAAN,CAAW,mCAAiCqM,iBAAiBjK,UAAlD,GAA4D,GAAvE;AACH,SAFD,MAEO;AACHvF,kBAAMI,IAAN,CAAW,WAAS,KAAKO,YAAL,CAAkBqP,IAA3B,GAA+B,qDAA1C;AACH;AAED,aAAKrP,YAAL,CAAkB+G,OAAlB,CAA0B,UAACxC,UAAD,EAAW;AACjC,gBAAIA,WAAWK,UAAX,KAA0BiK,iBAAiBjK,UAA/C,EAA2D;AACvDvF,sBAAMI,IAAN,CAAcoP,iBAAiBjK,UAAjB,GAA2B,0BAAzC;AACAL,2BAAWM,YAAX,CAAwBgK,gBAAxB;AACH;AACJ,SALD;AAOA,eAAOpM,QAAQC,OAAR,EAAP;AACH,KAjBO;AAmBA7C,4BAAAnC,SAAA,CAAAkD,gBAAA,GAAR,YAAA;AACI,aAAKmI,wBAAL,GAAgC,KAAKA,wBAAL,CAA8BuG,IAA9B,CAAmC,IAAnC,CAAhC;AACA,aAAK5N,2BAAL,GAAmC,KAAKA,2BAAL,CAAiC4N,IAAjC,CAAsC,IAAtC,CAAnC;AACA,aAAK1N,wBAAL,GAAgC,KAAKA,wBAAL,CAA8B0N,IAA9B,CAAmC,IAAnC,CAAhC;AACA,aAAKxN,0BAAL,GAAkC,KAAKA,0BAAL,CAAgCwN,IAAhC,CAAqC,IAArC,CAAlC;AACA,aAAKL,oBAAL,GAA4B,KAAKA,oBAAL,CAA0BK,IAA1B,CAA+B,IAA/B,CAA5B;AACA,aAAKpB,8BAAL,GAAsC,KAAKA,8BAAL,CAAoCoB,IAApC,CAAyC,IAAzC,CAAtC;AACA,aAAKvL,qBAAL,GAA6B,KAAKA,qBAAL,CAA2BuL,IAA3B,CAAgC,IAAhC,CAA7B;AACA,aAAKpH,0BAAL,GAAkC,KAAKA,0BAAL,CAAgCoH,IAAhC,CAAqC,IAArC,CAAlC;AACA,aAAKjD,YAAL,GAAoB,KAAKA,YAAL,CAAkBiD,IAAlB,CAAuB,IAAvB,CAApB;AACA,aAAK/C,sBAAL,GAA8B,KAAKA,sBAAL,CAA4B+C,IAA5B,CAAiC,IAAjC,CAA9B;AACA,aAAK/F,eAAL,GAAuB,KAAKA,eAAL,CAAqB+F,IAArB,CAA0B,IAA1B,CAAvB;AACA,aAAKhD,gBAAL,GAAwB,KAAKA,gBAAL,CAAsBgD,IAAtB,CAA2B,IAA3B,CAAxB;AACH,KAbO;AAeR;AAEQzP,4BAAAnC,SAAA,CAAA6O,sBAAA,GAAR,YAAA;AAAA,YAAAlK,QAAA,IAAA;AACIhD,cAAMI,IAAN,CAAW,6BAAX;AACA,YAAM6O,WAAgB,EAAtB;AAEA,aAAK7N,kBAAL,CAAwBsG,OAAxB,CAAgC,UAACqH,kBAAD,EAAuC;AACnEE,qBAAShH,IAAT,CAAcjF,MAAK0L,4BAAL,CAAkCK,kBAAlC,EAAsDlP,iBAAA8O,2BAAA,CAA4BuB,cAAlF,CAAd;AACH,SAFD;AAIA,eAAO9M,QAAQM,GAAR,CAAYuL,QAAZ,EAAsBrL,IAAtB,CAA2B,YAAA;AAAM,mBAAAa,SAAA;AAAS,SAA1C,CAAP;AACH,KATO;AAWAjE,4BAAAnC,SAAA,CAAA4O,gBAAA,GAAR,YAAA;AAAA,YAAAjK,QAAA,IAAA;AACIhD,cAAMI,IAAN,CAAW,sBAAX;AACA,YAAM6O,WAAgB,EAAtB;AAEA,aAAKtO,YAAL,CAAkB+G,OAAlB,CAA0B,UAACxC,UAAD,EAAW;AACjC+J,qBAAShH,IAAT,CAAcjF,MAAKyC,iBAAL,CAAuBP,WAAWF,SAAlC,CAAd;AACH,SAFD;AAIA,eAAO5B,QAAQM,GAAR,CAAYuL,QAAZ,EAAsBrL,IAAtB,CAA2B,YAAA;AAAM,mBAAAa,SAAA;AAAS,SAA1C,CAAP;AACH,KATO;AAWAjE,4BAAAnC,SAAA,CAAA6L,eAAA,GAAR,YAAA;AAAA,YAAAlH,QAAA,IAAA;AACIhD,cAAMI,IAAN,CAAW,qBAAX;AACA,YAAM6O,WAAgB,EAAtB;AAEA,aAAKpO,WAAL,CAAiB6G,OAAjB,CAAyB,UAACyH,SAAD,EAAU;AAC/BF,qBAAShH,IAAT,CAAcjF,MAAK0M,eAAL,CAAqBP,SAArB,CAAd;AACH,SAFD;AAIA,eAAO/L,QAAQM,GAAR,CAAYuL,QAAZ,EAAsBrL,IAAtB,CAA2B,YAAA;AAAM,mBAAAa,SAAA;AAAS,SAA1C,CAAP;AACH,KATO;AAWR;AAEA;AAEUjE,4BAAAnC,SAAA,CAAAsE,gBAAA,GAAV,UAA2BwN,WAA3B,EAA6C;AAAlB,YAAAA,gBAAA,KAAA,CAAA,EAAA;AAAAA,0BAAA,IAAA;AAAkB;AACzC,eAAOrQ,SAAAsQ,cAAA,CAAe,CAAC,CAAC,KAAKrP,eAAtB,EAAuC,kCAAvC,EAA2EoP,WAA3E,CAAP;AACH,KAFS;AAIA3P,4BAAAnC,SAAA,CAAAuL,kBAAA,GAAV,UAA6BuG,WAA7B,EAA+C;AAAlB,YAAAA,gBAAA,KAAA,CAAA,EAAA;AAAAA,0BAAA,IAAA;AAAkB;AAC3C,eAAOrQ,SAAAsQ,cAAA,CAAe,CAAC,CAAC,KAAK9O,QAAtB,EAAgC,yBAAhC,EAA2D6O,WAA3D,CAAP;AACH,KAFS;AAOd,WAAA3P,uBAAA;AApzBA,CAAA,EAAA;AAAsB3B,QAAA2B,uBAAA,GAAAA,uBAAA;;AD8oBtB","file":"MediaServiceBase.js","sourcesContent":["\"use strict\";\nvar __assign = (this && this.__assign) || function () {\n __assign = Object.assign || function(t) {\n for (var s, i = 1, n = arguments.length; i < n; i++) {\n s = arguments[i];\n for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p))\n t[p] = s[p];\n }\n return t;\n };\n return __assign.apply(this, arguments);\n};\nvar __importDefault = (this && this.__importDefault) || function (mod) {\n return (mod && mod.__esModule) ? mod : { \"default\": mod };\n};\nObject.defineProperty(exports, \"__esModule\", { value: true });\nvar filter_1 = __importDefault(require(\"lodash/filter\"));\nvar cloneDeep_1 = __importDefault(require(\"lodash/cloneDeep\"));\nvar values_1 = __importDefault(require(\"lodash/values\"));\nvar debounce_1 = __importDefault(require(\"lodash/debounce\"));\nvar includes_1 = __importDefault(require(\"lodash/includes\"));\nvar events_1 = require(\"events\");\nvar MediaSubscriber_1 = require(\"./MediaSubscriber\");\nvar LocalStreamManager_1 = require(\"./LocalStreamManager\");\nvar TechseeMediaStream_1 = require(\"./TechseeMediaStream\");\nvar SessionTurn_1 = require(\"./MediaSession/SessionTurn\");\nvar MediaDomUtils_1 = require(\"./MediaUtils/MediaDomUtils\");\nvar Compatibility_1 = require(\"./MediaUtils/Compatibility\");\nvar SessionOpentok_1 = require(\"./MediaSession/SessionOpentok\");\nvar MediaConstants_1 = require(\"./MediaConstants\");\nvar guards_1 = require(\"@techsee/techsee-common/lib/core/guards\");\nvar MediaTracer_1 = require(\"./MediaUtils/MediaTracer\");\nvar trace = MediaTracer_1.getMediaTracer('MediaServiceBase');\nvar traceStatsInfo = debounce_1.default(trace.info, 1000 * 5, { leading: true, maxWait: 1000 * 30 });\nvar privateEvents;\n(function (privateEvents) {\n privateEvents[\"STREAM_CREATED\"] = \"STREAM_CREATED\";\n privateEvents[\"STREAM_DESTROYED\"] = \"STREAM_DESTROYED\";\n})(privateEvents || (privateEvents = {}));\nvar TechseeMediaServiceBase = /** @class */ (function () {\n function TechseeMediaServiceBase(environment, webRtcSupportInfo) {\n this._subscribers = new Map();\n this._publishers = new Map();\n this._publisherPromises = new Map();\n this._serviceOptions = null;\n this._isIOS_13_orLater = false;\n this._autoReconnectEnabled = true;\n this._initLocalStreamsPromise = null;\n this._initServicePromise = null;\n this._registeredStreams = new Map();\n this._isVoipEnabled = false;\n this._session = null;\n trace.info('TechseeMediaServiceBase created');\n this.bindClassMethods();\n this._emitter = new events_1.EventEmitter();\n this._environment = environment;\n this._deviceSupportFlags = {\n videoPlayback: false,\n hasCamera: false,\n hasMicrophone: false,\n webRtcSupportInfo: webRtcSupportInfo\n };\n this._localStreamsManager = new LocalStreamManager_1.LocalStreamManager(this._environment);\n // TODO: Hack as IOS 13.3 and above sometimes do not return the video/audio device. This\n // is a hack to work around the issue until Apple resolve it\n this._isIOS_13_orLater = this._environment.isIOS() &&\n (!this._environment.majorVersion() || (this._environment.majorVersion() >= 13));\n trace.info(\"XXXXX: this._isIOS_13_orLater: \" + this._isIOS_13_orLater);\n trace.info(\"XXXXX: this._environment.isIOS: \" + this._environment.isIOS());\n trace.info(\"XXXXX: this._environment.majorVersion(): \" + this._environment.majorVersion());\n this._sessionStreamsManager = {\n getMediaStreamForRole: this.getStreamForDestinationRole,\n addRemoteMediaTrack: this.registerRemoteMediaTrack,\n removeMediaTrack: this.unregisterRemoteMediaTrack\n };\n }\n Object.defineProperty(TechseeMediaServiceBase.prototype, \"deviceSupportInfo\", {\n get: function () {\n this.serviceInitGuard(false);\n return cloneDeep_1.default(this._deviceSupportFlags);\n },\n enumerable: true,\n configurable: true\n });\n Object.defineProperty(TechseeMediaServiceBase.prototype, \"isVoipEnabled\", {\n get: function () {\n return this._isVoipEnabled;\n },\n enumerable: true,\n configurable: true\n });\n Object.defineProperty(TechseeMediaServiceBase.prototype, \"isSessionActive\", {\n get: function () {\n return this._session !== null;\n },\n enumerable: true,\n configurable: true\n });\n Object.defineProperty(TechseeMediaServiceBase.prototype, \"isLocalStreamInitialized\", {\n get: function () {\n return this._initLocalStreamsPromise !== null;\n },\n enumerable: true,\n configurable: true\n });\n /*\n Initializes media service. Before using anything from this service it should be initialized first.\n All of serviceOptions should be retrieved from account settings and by evaluating application state.\n For example, when initializing service, accountSettings is already loaded, and we know if FS session is audio enabled.\n */\n TechseeMediaServiceBase.prototype.initMediaService = function (serviceOptions) {\n var _this = this;\n trace.info('TechseeMediaServiceBase.initMediaService', serviceOptions);\n var warnMessage = 'Multiple initialization does not supposed to happen. Not rejecting, but check the flow for correctness.';\n if (this._initServicePromise || this._serviceOptions) {\n console.warn(warnMessage);\n return this._initServicePromise ? this._initServicePromise : Promise.resolve();\n }\n this._serviceOptions = cloneDeep_1.default(serviceOptions);\n var isOpenTok = this._serviceOptions.mediaServiceType === MediaConstants_1.MediaServiceType.OPENTOK;\n this._initServicePromise = Promise.all([\n Compatibility_1.isVideoPlaySupportedOnDevice(isOpenTok)\n .then(function (isSupported) { return _this._deviceSupportFlags.videoPlayback = isSupported; }),\n Compatibility_1.enumerateMediaDevices()\n .catch(function (error) {\n trace.warn(error);\n return [];\n })\n .then(function (localDevicesList) {\n var groupedDevices = {\n video: filter_1.default(localDevicesList, function (device) { return device.kind.toLowerCase() === 'videoinput'; }),\n audio: filter_1.default(localDevicesList, function (device) { return device.kind.toLowerCase() === 'audioinput'; })\n };\n _this._localStreamsManager.setGroupedDevices(groupedDevices);\n _this._deviceSupportFlags.hasCamera = _this._isIOS_13_orLater || (groupedDevices.video.length > 0);\n _this._deviceSupportFlags.hasMicrophone = _this._isIOS_13_orLater || (groupedDevices.audio.length > 0);\n })\n ]).then(function () { return undefined; });\n return this._initServicePromise;\n };\n //#region Media Streams Management\n /*\n The main method to start the initialization of local media streams.\n This method can be executed many times and it will insure all calls are synced and only relevant stream will be created.\n DO NOT USE ANY OTHER METHOD THAT FORCES \"navigator.getUserMedia\" OTHER THAN THIS ONE.\n */\n TechseeMediaServiceBase.prototype.initLocalMediaStreams = function () {\n this.serviceInitGuard();\n if (!this._initLocalStreamsPromise) {\n trace.info('initLocalMediaStreams creating media streams');\n this._initLocalStreamsPromise = this.getLocalMediaImplementation()\n .catch(function (mediaRequestFailResult) {\n trace.error('initLocalMediaStreams failure: ', mediaRequestFailResult);\n throw mediaRequestFailResult;\n });\n }\n else {\n trace.info('initLocalMediaStreams already done or in progress');\n }\n return this._initLocalStreamsPromise;\n };\n /*\n Creates an instance of subscriber that will \"listen\" to the requested stream.\n Whenever requested stream is available, subscriber will render it.\n */\n TechseeMediaServiceBase.prototype.createSubscriber = function (subscriberParams) {\n if (this._subscribers.has(subscriberParams.container)) {\n return Promise.reject('Subscriber for provided DIV element already exists');\n }\n var subscriber = new MediaSubscriber_1.TechseeMediaSubscriber(subscriberParams);\n this._subscribers.set(subscriberParams.container, subscriber);\n var streamForSubscriber = this.getRegisteredStreamByType(subscriber.streamType);\n if (streamForSubscriber) {\n trace.info(\"Stream for \" + subscriberParams.streamType + \" is ready.\");\n subscriber.renderStream(streamForSubscriber);\n }\n else {\n trace.info(\"Stream for \" + subscriberParams.streamType + \" not created yet\");\n }\n return Promise.resolve(subscriber);\n };\n /*\n Destroys the subscriber for provided HTML DIV container\n */\n TechseeMediaServiceBase.prototype.destroySubscriber = function (container) {\n if (this._subscribers.has(container)) {\n var subscriber = this._subscribers.get(container);\n subscriber.dispose();\n }\n this._subscribers.delete(container);\n return Promise.resolve();\n };\n TechseeMediaServiceBase.prototype.onStreamDestroyed = function (callback) {\n this.registerEventCallback(privateEvents.STREAM_DESTROYED, callback);\n };\n TechseeMediaServiceBase.prototype.onStreamCreated = function (callback) {\n this.registerEventCallback(privateEvents.STREAM_CREATED, callback);\n };\n TechseeMediaServiceBase.prototype.getSwitchCameraConstraints = function () {\n var constraints = cloneDeep_1.default(window.latestLocalMediaConstraints);\n if (!constraints) {\n throw new Error('getSwitchCameraConstraints: unexpected use case constraints is null.');\n }\n trace.info('getSwitchCameraConstraints - Start switch camera with constraints:', constraints);\n var videoSourceType = constraints.video.videoSourceType === MediaConstants_1.LocalVideoSourceType.CAMERA\n ? MediaConstants_1.LocalVideoSourceType.CAMERA_FRONT\n : MediaConstants_1.LocalVideoSourceType.CAMERA;\n trace.info(\"getSwitchCameraConstraints - switch to videoSourceType: \" + videoSourceType);\n constraints.video.videoSourceType = videoSourceType;\n return constraints;\n };\n TechseeMediaServiceBase.prototype.switchCamera = function (isFailed) {\n var _this = this;\n trace.info('switchCamera: start');\n var constraints = !isFailed ? this.getSwitchCameraConstraints() : cloneDeep_1.default(window.latestLocalMediaConstraints);\n if (!constraints) {\n throw new Error('switchCamera - unexpected use case constraints is null.');\n }\n this._localStreamsManager.destroyUserMediaStream();\n return this._localStreamsManager.getUserMediaStream(constraints)\n .then(function (streamResult) {\n var stream = streamResult.mediaStream;\n trace.info('switchCamera: new stream: ', stream);\n _this.replaceStreamTracks(stream);\n return _this.registerStreamResult(constraints, streamResult, true).then(function () { return ({ constraints: constraints, streamResult: streamResult }); });\n })\n .catch(function (err) {\n trace.error('switchCamera: Failed to switch camera: ', err);\n if (!isFailed) {\n _this.switchCamera(true);\n }\n return Promise.reject('Failed to switch camera.');\n });\n };\n TechseeMediaServiceBase.prototype.replaceStreamTracks = function (mediaStream) {\n this._session.instance && this._session.instance.replaceStreamTracks(mediaStream);\n };\n TechseeMediaServiceBase.prototype.registerStreamResult = function (constraints, streamResult, switchCamera, addStreamType) {\n var _this = this;\n var regPromises = [];\n if (streamResult.isNew) {\n trace.info('registerStreamResult: stream result from getUserMediaStream:', streamResult.mediaStream);\n streamResult.mediaStream.getTracks().forEach(function (mediaTrack) {\n trace.info('registerStreamResult: stream result from mediaTrack:', mediaTrack);\n var streamType = addStreamType || (mediaTrack.kind === 'video'\n ? MediaConstants_1.KnownMediaStream.USER_VIDEO_STREAM : MediaConstants_1.KnownMediaStream.USER_AUDIO_STREAM);\n var newDedicatedStream = new TechseeMediaStream_1.TechseeMediaStream(mediaTrack, streamType, false);\n regPromises.push(switchCamera ? _this.registerTrack(newDedicatedStream) : _this.registerLocalMediaStream(newDedicatedStream));\n });\n }\n return Promise.all(regPromises);\n };\n Object.defineProperty(TechseeMediaServiceBase.prototype, \"supportSwitchCameras\", {\n get: function () {\n return this._isIOS_13_orLater || this._localStreamsManager.groupedDevices.camerasCount > 1;\n },\n enumerable: true,\n configurable: true\n });\n //#endregion Media Streams Management\n //#region Session Management\n /*\n Creates instance of WebRTC session which connects to (signaling server), and begins to listen to WebRTC events.\n */\n TechseeMediaServiceBase.prototype.connectToSession = function (sessionParams) {\n var _this = this;\n this.serviceInitGuard();\n if (!this._session) {\n this._session = { params: cloneDeep_1.default(sessionParams) };\n this._session.connectPromise = new Promise(function (resolve, reject) {\n var doAsyncReject = function (error) {\n /*\n Reject with timeout, to change JS 'execution flow'.\n If timeout removed, session is assigned to null, before promise returns, and \"catch\" will not\n work on promise that is returned.\n */\n setTimeout(function () {\n _this._session = null;\n reject(error);\n });\n };\n var sessionHandlers = {\n onDisconnectedHandler: _this.onSessionDisconnectHandler\n };\n var mediaSessionParams = __assign({}, sessionParams, { peerConnectivityTimeoutSeconds: _this._serviceOptions.peerConnectivityTimeoutSeconds || MediaConstants_1.DEFAULT_PEER_CONNECTIVITY_TIMEOUT_SECONDS });\n trace.info('connectToSession', sessionParams);\n if (_this._serviceOptions.mediaServiceType === MediaConstants_1.MediaServiceType.TURNSERVER) {\n _this._session.instance = new SessionTurn_1.TurnWebRtcSession(mediaSessionParams, sessionHandlers, _this._sessionStreamsManager);\n }\n else if (_this._serviceOptions.mediaServiceType === MediaConstants_1.MediaServiceType.OPENTOK) {\n _this._session.instance = new SessionOpentok_1.OpentokSession(mediaSessionParams, _this._sessionStreamsManager);\n }\n else {\n var allowedValues = values_1.default(MediaConstants_1.MediaServiceType);\n trace.error(\"mediaServiceType '\" + _this._serviceOptions.mediaServiceType + \"' is not supported.\", { allowedValues: allowedValues });\n doAsyncReject(new Error('mediaServiceType is not supported'));\n }\n if (_this._session && _this._session.instance) {\n _this._session.instance.connect()\n .then(function () {\n trace.info('Session is connected');\n resolve();\n })\n .catch(function (error) {\n trace.error('Failed to connect to session', error);\n doAsyncReject(error);\n });\n }\n });\n }\n else {\n trace.warn('Session connected already. Disconnect before connect to new session again');\n }\n //TODO - Alex: Change a structure of this function to be more safe\n //(when we return, there should not be case when session or promise is null)\n //To fix, need to return promise immediately, then execute process in setTimeout(fn, 0);\n return this._session.connectPromise;\n };\n /*\n Disconnects from WebRTC session.\n */\n TechseeMediaServiceBase.prototype.disconnectFromSession = function () {\n return this.disconnectFromSessionInternal(MediaConstants_1.MediaSessionDisconnectReason.ForcedByConsumer);\n };\n /*\n Updates a credentials for turn server\n */\n TechseeMediaServiceBase.prototype.updateSessionCredentials = function (credentials) {\n trace.info('updateSessionCredentials');\n if (!this.sessionExistsGuard(false)) {\n return Promise.reject('There no session to update credentials');\n }\n this._session.params.credentials = cloneDeep_1.default(credentials);\n return Promise.resolve();\n };\n TechseeMediaServiceBase.prototype.enableVoipDuringSession = function () {\n if (!this._isVoipEnabled && this.isLocalStreamInitialized) {\n throw new Error('Voip support cannot be enabled after local stream where initialized');\n }\n this._isVoipEnabled = true;\n };\n TechseeMediaServiceBase.prototype.disconnectFromSessionInternal = function (reason) {\n if (!this._session) {\n return Promise.resolve();\n }\n trace.info('Disconnecting from session', reason);\n var lastSession = this._session;\n this._session = null;\n return Promise.all([\n lastSession.instance.disconnect(),\n this.clearPublishers()\n ]).then(function () { return undefined; });\n };\n TechseeMediaServiceBase.prototype.getStatsForRemoteTrack = function (streamType) {\n if (!this.sessionExistsGuard(false)) {\n return Promise.reject(new Error('Media session not started'));\n }\n var streamForStats = this.getRegisteredStreamByType(streamType);\n if (!streamForStats) {\n return Promise.reject(new Error('Stream for requested type was not found'));\n }\n return this._session.instance.getRemoteTrackStats(streamForStats.mediaTrack)\n .then(function (trackStats) {\n traceStatsInfo(\"MediaTrackStats for \" + streamType + \": \" + JSON.stringify(trackStats));\n return {\n trackId: streamForStats.mediaTrack.id,\n trackStats: trackStats\n };\n });\n };\n TechseeMediaServiceBase.prototype.onSessionDisconnectHandler = function (reason) {\n var _this = this;\n trace.info('onSessionDisconnectHandler', reason);\n if (reason !== MediaConstants_1.MediaSessionDisconnectReason.ForcedByConsumer) {\n var lastParams_1 = this._session && this._session.params ? this._session.params : null;\n this.disconnectFromSessionInternal(reason).then(function () {\n var reconnectReasons = [\n MediaConstants_1.MediaSessionDisconnectReason.InitiatorPeerReconnected,\n MediaConstants_1.MediaSessionDisconnectReason.SignalingChannelDisconnect,\n MediaConstants_1.MediaSessionDisconnectReason.PeerConnectionInterrupted,\n MediaConstants_1.MediaSessionDisconnectReason.PeerConnectionStateChangeTimeout\n ];\n if (lastParams_1 && includes_1.default(reconnectReasons, reason) && _this._autoReconnectEnabled) {\n _this.reconnectToSession(lastParams_1);\n }\n else if (!_this._autoReconnectEnabled) {\n trace.info('No reconnection- auto reconnect disabled');\n }\n else {\n trace.info('No params for reconnection to media session');\n }\n });\n }\n else {\n trace.info('Ignore Session disconnect event');\n }\n };\n TechseeMediaServiceBase.prototype.setAutoReconnect = function (state) {\n this._autoReconnectEnabled = state;\n trace.info('_autoReconnectEnabled:', state);\n if (this._autoReconnectEnabled && !this.isSessionActive) {\n var lastParams = this._session && this._session.params ? this._session.params : null;\n if (lastParams) {\n trace.info('setAutoReconnect - reconnect to session');\n this.reconnectToSession(lastParams);\n }\n }\n };\n TechseeMediaServiceBase.prototype.reconnectToSession = function (lastParams) {\n trace.info('Reconnecting to media session, sessionParams:', lastParams);\n this.connectToSession(lastParams)\n .then(function () {\n trace.info('Media session reconnected');\n })\n .catch(function (error) {\n trace.error('Error while reconnecting to media session', error);\n });\n };\n //#endregion Session Management\n //#region Utils\n TechseeMediaServiceBase.prototype.getSnapshotFromKnownStream = function (sourceStream, snapshotOptions) {\n var _this = this;\n return new Promise(function (resolve, reject) {\n if (sourceStream !== MediaConstants_1.KnownMediaStream.USER_VIDEO_STREAM &&\n sourceStream !== MediaConstants_1.KnownMediaStream.USER_SCREEN_SHARE_STREAM) {\n trace.error('The requested stream is not video stream, and cannot be used for snapshot');\n reject(new Error('INCOMPATIBLE_STREAM_FOR_SNAPSHOT'));\n return;\n }\n var snapshotStream = _this.getRegisteredStreamByType(sourceStream);\n if (!snapshotStream) {\n trace.error('Cannot make snapshot: The requested stream not exists yet.');\n reject(new Error('NO_REQUESTED_STREAM'));\n return;\n }\n MediaDomUtils_1.getSnapshotFromMediaStream(snapshotStream.mediaStream, snapshotOptions)\n .then(function (imageData) {\n var urlComponents = imageData.split(';base64,');\n var mimeType = urlComponents[0].split(':')[1];\n var bytes = atob(urlComponents[1]);\n var buffer = new ArrayBuffer(bytes.length);\n var rawData = new Uint8Array(buffer);\n for (var i = 0; i < bytes.length; i++) {\n rawData[i] = bytes.charCodeAt(i);\n }\n var blob = new Blob([rawData], { type: mimeType });\n var objectUrl = window.URL.createObjectURL(blob);\n trace.info('Snapshot created successfully');\n var result = {\n base64img: imageData,\n objectUrl: objectUrl,\n imageBlob: blob,\n mimeType: mimeType\n };\n resolve(result);\n })\n .catch(function (error) {\n trace.error('Error creating snapshot', error);\n reject(error);\n });\n });\n };\n //Will clean streams, publishers and subscribers, but will not remove event listeners\n TechseeMediaServiceBase.prototype.clearService = function () {\n var _this = this;\n trace.info('MediaService clearing all resources');\n return this.disconnectFromSessionInternal(MediaConstants_1.MediaSessionDisconnectReason.ForcedByConsumer)\n .then(this.clearSubscribers)\n .then(this.clearRegisteredStreams)\n .then(this._localStreamsManager.clearAllStreams)\n .then(function () {\n _this._initLocalStreamsPromise = null;\n _this._isVoipEnabled = false;\n }).then(function () { return undefined; });\n };\n //Will clear the service and remove all event listeners\n TechseeMediaServiceBase.prototype.dispose = function () {\n var _this = this;\n return this.clearService().catch(function () { return undefined; }).then(function () {\n _this._emitter.removeAllListeners();\n });\n };\n Object.defineProperty(TechseeMediaServiceBase.prototype, \"subscribers\", {\n //#endregion\n //#region Protected Methods\n get: function () {\n return this._subscribers;\n },\n enumerable: true,\n configurable: true\n });\n TechseeMediaServiceBase.prototype.registerLocalMediaStream = function (tsMediaStream) {\n trace.info('Registering local stream', tsMediaStream.streamType);\n if (this._registeredStreams.has(tsMediaStream.streamType)) {\n return Promise.reject(new Error(\"Stream \" + tsMediaStream.streamType + \" already registered\"));\n }\n return this.registerStream(tsMediaStream);\n };\n TechseeMediaServiceBase.prototype.getRegisteredStreamByType = function (streamType) {\n return this._registeredStreams.get(streamType) || null;\n };\n TechseeMediaServiceBase.prototype.changeEnableForKnownStream = function (streamType, isPaused) {\n var streamToChangeState = this.getRegisteredStreamByType(streamType);\n if (!streamToChangeState) {\n trace.warn('There no stream found to change enable state', streamType);\n }\n else {\n streamToChangeState.mediaTrack.enabled = !isPaused;\n if (streamToChangeState.isRemote && streamToChangeState.streamKind === MediaConstants_1.KnownMediaStreamKind.Audio) {\n this.subscribers.forEach(function (subscriber) {\n if (subscriber.streamType === streamType) {\n subscriber.muteSound(isPaused);\n }\n });\n }\n trace.info('Local stream enable state is changed', { streamType: streamType, isPaused: isPaused });\n }\n };\n TechseeMediaServiceBase.prototype.registerEventCallback = function (event, callback) {\n this._emitter.on(event, callback);\n };\n TechseeMediaServiceBase.prototype.emitEvent = function (event, eventArgs) {\n var _this = this;\n setTimeout(function () {\n _this._emitter.emit(event, eventArgs);\n });\n };\n //#endregion Protected Methods\n //#region Private Methods\n TechseeMediaServiceBase.prototype.registerRemoteMediaTrack = function (remoteMediaTrack) {\n trace.info(\"Registering remote \" + remoteMediaTrack.trackType + \" MediaStreamTrack\", remoteMediaTrack.mediaTrack);\n var currentStream = this._registeredStreams.get(remoteMediaTrack.trackType);\n if (currentStream && !currentStream.isRemote) {\n return Promise.reject(new Error('Cannot register remote stream with the same type as local stream'));\n }\n var newDedicatedStream = new TechseeMediaStream_1.TechseeMediaStream(remoteMediaTrack.mediaTrack, remoteMediaTrack.trackType, true);\n return this.registerStream(newDedicatedStream);\n };\n TechseeMediaServiceBase.prototype.registerTrack = function (mediaStream) {\n var _this = this;\n mediaStream.mediaTrack.onended = function () {\n return _this.unregisterTechseeMediaStream(mediaStream, MediaConstants_1.MediaStreamUnregisterReason.NativeEvent);\n };\n this._registeredStreams.set(mediaStream.streamType, mediaStream);\n trace.info(\"TechseeMediaStream registered - \" + mediaStream.streamType, TechseeMediaStream_1.TechseeMediaStream);\n return this.updateSubscribersWithNewStream(mediaStream).then(function () {\n var eventArgs = { streamType: mediaStream.streamType };\n _this.emitEvent(privateEvents.STREAM_CREATED, eventArgs);\n });\n };\n TechseeMediaServiceBase.prototype.registerStream = function (mediaStream) {\n var _this = this;\n var currentStream = this._registeredStreams.get(mediaStream.streamType);\n if (currentStream) {\n return this.unregisterTechseeMediaStream(currentStream, MediaConstants_1.MediaStreamUnregisterReason.ReplacingStream).then(function () { return _this.registerTrack(mediaStream); });\n }\n return this.registerTrack(mediaStream);\n };\n TechseeMediaServiceBase.prototype.unregisterTechseeMediaStream = function (streamToUnregister, reason) {\n var _this = this;\n var traceError = function (error) {\n trace.warn('Unregister TechseeMediaStream error:', error);\n };\n var promises = [];\n trace.info('Unregister TechseeMediaStream: ', streamToUnregister.streamType);\n this._subscribers.forEach(function (subscriber) {\n if (subscriber.streamType === streamToUnregister.streamType) {\n promises.push(subscriber.stopRendering().catch(traceError));\n }\n });\n if (!streamToUnregister.isRemote) {\n this._publishers.forEach(function (publisher) {\n if (publisher.streamTypes.find(function (streamType) { return streamType === streamToUnregister.streamType; })) {\n if (_this._session && _this._session.instance) {\n promises.push(_this._session.instance.onMediaStreamDestroyed(publisher.destinationRole).catch(traceError));\n }\n }\n });\n }\n return Promise.all(promises)\n .then(function () {\n _this._registeredStreams.delete(streamToUnregister.streamType);\n trace.info('Stream deleted', streamToUnregister.streamType);\n var eventArgs = { streamType: streamToUnregister.streamType, reason: reason };\n _this.emitEvent(privateEvents.STREAM_DESTROYED, eventArgs);\n });\n };\n TechseeMediaServiceBase.prototype.unregisterRemoteMediaTrack = function (mediaTrack) {\n var _this = this;\n var promises = [];\n this._registeredStreams.forEach(function (registeredStream) {\n if (registeredStream.mediaTrack.id === mediaTrack.id) {\n promises.push(_this.unregisterTechseeMediaStream(registeredStream, MediaConstants_1.MediaStreamUnregisterReason.ClosedRemotely));\n }\n });\n return Promise.all(promises).then(function () { return undefined; });\n };\n TechseeMediaServiceBase.prototype.removePublisher = function (publisher) {\n this._publishers.delete(publisher.destinationRole);\n if (this._session && this._session.instance) {\n trace.info('Removing publisher', publisher.destinationRole);\n return this._session.instance.onMediaStreamDestroyed(publisher.destinationRole)\n .catch(function (error) {\n trace.warn('Error while removing publisher', error);\n });\n }\n return Promise.resolve();\n };\n TechseeMediaServiceBase.prototype.getStreamForDestinationRole = function (destinationRole) {\n var _this = this;\n this.serviceInitGuard();\n if (this._publishers.has(destinationRole)) {\n trace.info(\"Publisher for \" + destinationRole + \" already exists\");\n return Promise.resolve(this._publishers.get(destinationRole).mediaStream);\n }\n if (!this._publisherPromises.get(destinationRole)) {\n trace.info(\"Creating publisher for \" + destinationRole);\n var publisherPromise = this.initLocalMediaStreams()\n .then(function () { return _this.createMediaPublisher(destinationRole); })\n .then(function (mediaPublisher) {\n if (mediaPublisher) {\n _this._publishers.set(destinationRole, mediaPublisher);\n }\n _this._publisherPromises.delete(destinationRole);\n return mediaPublisher ? mediaPublisher.mediaStream : null;\n })\n .catch(function (ex) {\n if (ex && ex.message === 'audioStreamFailed') {\n _this._isVoipEnabled = false;\n return null;\n }\n _this._publisherPromises.delete(destinationRole);\n throw ex;\n });\n this._publisherPromises.set(destinationRole, publisherPromise);\n }\n else {\n trace.info(\"Create publisher promise for \" + destinationRole + \" already exists\");\n }\n return this._publisherPromises.get(destinationRole);\n };\n TechseeMediaServiceBase.prototype.updateSubscribersWithNewStream = function (registeredStream) {\n trace.info(\"Updating subscribers of \" + registeredStream.streamType + \" with new stream\");\n if (this._subscribers.size === 0) {\n trace.warn(\"No subscribers exists for the \" + registeredStream.streamType + \".\");\n }\n else {\n trace.info(\"Total \" + this._subscribers.size + \" subscribers exists, will check if rerender needed.\");\n }\n this._subscribers.forEach(function (subscriber) {\n if (subscriber.streamType === registeredStream.streamType) {\n trace.info(registeredStream.streamType + \" rendering on subscriber\");\n subscriber.renderStream(registeredStream);\n }\n });\n return Promise.resolve();\n };\n TechseeMediaServiceBase.prototype.bindClassMethods = function () {\n this.updateSessionCredentials = this.updateSessionCredentials.bind(this);\n this.getStreamForDestinationRole = this.getStreamForDestinationRole.bind(this);\n this.registerRemoteMediaTrack = this.registerRemoteMediaTrack.bind(this);\n this.unregisterRemoteMediaTrack = this.unregisterRemoteMediaTrack.bind(this);\n this.createMediaPublisher = this.createMediaPublisher.bind(this);\n this.updateSubscribersWithNewStream = this.updateSubscribersWithNewStream.bind(this);\n this.initLocalMediaStreams = this.initLocalMediaStreams.bind(this);\n this.onSessionDisconnectHandler = this.onSessionDisconnectHandler.bind(this);\n this.clearService = this.clearService.bind(this);\n this.clearRegisteredStreams = this.clearRegisteredStreams.bind(this);\n this.clearPublishers = this.clearPublishers.bind(this);\n this.clearSubscribers = this.clearSubscribers.bind(this);\n };\n //region Cleanup\n TechseeMediaServiceBase.prototype.clearRegisteredStreams = function () {\n var _this = this;\n trace.info('Clearing registered streams');\n var promises = [];\n this._registeredStreams.forEach(function (streamToUnregister) {\n promises.push(_this.unregisterTechseeMediaStream(streamToUnregister, MediaConstants_1.MediaStreamUnregisterReason.ServiceCleanUp));\n });\n return Promise.all(promises).then(function () { return undefined; });\n };\n TechseeMediaServiceBase.prototype.clearSubscribers = function () {\n var _this = this;\n trace.info('Clearing subscribers');\n var promises = [];\n this._subscribers.forEach(function (subscriber) {\n promises.push(_this.destroySubscriber(subscriber.container));\n });\n return Promise.all(promises).then(function () { return undefined; });\n };\n TechseeMediaServiceBase.prototype.clearPublishers = function () {\n var _this = this;\n trace.info('Clearing publishers');\n var promises = [];\n this._publishers.forEach(function (publisher) {\n promises.push(_this.removePublisher(publisher));\n });\n return Promise.all(promises).then(function () { return undefined; });\n };\n //#endregion Cleanup\n //#region Simple Validation Methods\n TechseeMediaServiceBase.prototype.serviceInitGuard = function (shouldThrow) {\n if (shouldThrow === void 0) { shouldThrow = true; }\n return guards_1.throwableGuard(!!this._serviceOptions, 'Media service is not initialized', shouldThrow);\n };\n TechseeMediaServiceBase.prototype.sessionExistsGuard = function (shouldThrow) {\n if (shouldThrow === void 0) { shouldThrow = true; }\n return guards_1.throwableGuard(!!this._session, 'There no active session', shouldThrow);\n };\n return TechseeMediaServiceBase;\n}());\nexports.TechseeMediaServiceBase = TechseeMediaServiceBase;\n\n//# sourceMappingURL=MediaServiceBase.js.map\n","import filter from 'lodash/filter';\nimport cloneDeep from 'lodash/cloneDeep';\nimport values from 'lodash/values';\nimport debounce from 'lodash/debounce';\nimport includes from 'lodash/includes';\nimport {EventEmitter} from 'events';\nimport {Nullable} from '@techsee/techsee-common';\nimport {TechseeMediaSubscriber} from './MediaSubscriber';\nimport {TechseeMediaPublisher} from './MediaPublisher';\nimport {LocalStreamManager} from './LocalStreamManager';\nimport {TechseeMediaStream} from './TechseeMediaStream';\nimport {TurnWebRtcSession} from './MediaSession/SessionTurn';\nimport {getSnapshotFromMediaStream} from './MediaUtils/MediaDomUtils';\nimport {\n ClientWebRtcInfo,\n DeviceSupportInfo,\n IMediaEnvironment,\n IMediaSession,\n IMediaSubscriber,\n ISessionStreamsManager, LocalMediaConstraints,\n LocalVideoStreamConstraints,\n MediaRequestFailResult, MediaRequestSuccessResult,\n MediaServiceOptions,\n MediaSessionConfiguration,\n MediaSessionHandlers,\n MediaSessionIceCredentials,\n MediaSessionParams,\n MediaSubscriberParams,\n RemoteMediaTrack,\n RemoteTrackStats,\n SnapshotOptions,\n SnapshotResult,\n StreamCreatedEventArgs,\n StreamDestroyedEventArgs\n} from './MediaContracts';\nimport {enumerateMediaDevices, isVideoPlaySupportedOnDevice} from './MediaUtils/Compatibility';\nimport {OpentokSession} from './MediaSession/SessionOpentok';\nimport {\n DEFAULT_PEER_CONNECTIVITY_TIMEOUT_SECONDS,\n KnownMediaStream,\n KnownMediaStreamKind,\n LocalVideoSourceType,\n MediaServiceType,\n MediaSessionDisconnectReason,\n MediaStreamUnregisterReason,\n SessionClientRole\n} from './MediaConstants';\nimport {throwableGuard} from '@techsee/techsee-common/lib/core/guards';\n\nimport {getMediaTracer} from './MediaUtils/MediaTracer';\n\nconst trace = getMediaTracer('MediaServiceBase');\nconst traceStatsInfo = debounce(trace.info, 1000 * 5, {leading: true, maxWait: 1000 * 30});\n\nenum privateEvents {\n STREAM_CREATED = 'STREAM_CREATED',\n STREAM_DESTROYED = 'STREAM_DESTROYED'\n}\n\nexport abstract class TechseeMediaServiceBase {\n //This method should not be called. The only place that allowed to use it, is initLocalMediaStreams method.\n //Everyone who need to initialize MediaStream should use initLocalMediaStreams method.\n protected abstract getLocalMediaImplementation(): Promise<void>;\n\n protected abstract createMediaPublisher(destinationRole: SessionClientRole): Promise<Nullable<TechseeMediaPublisher>>;\n\n protected readonly _environment: IMediaEnvironment;\n protected readonly _sessionStreamsManager: ISessionStreamsManager;\n protected readonly _localStreamsManager: LocalStreamManager;\n\n private readonly _deviceSupportFlags: DeviceSupportInfo;\n private readonly _subscribers: Map<HTMLDivElement, TechseeMediaSubscriber> = new Map();\n private readonly _publishers: Map<SessionClientRole, TechseeMediaPublisher> = new Map();\n private readonly _publisherPromises: Map<SessionClientRole, Promise<Nullable<MediaStream>>> = new Map();\n protected _serviceOptions: Nullable<MediaServiceOptions> = null;\n private _isIOS_13_orLater: boolean = false;\n private _autoReconnectEnabled: boolean = true;\n private readonly _emitter: EventEmitter;\n private _initLocalStreamsPromise: Nullable<Promise<void>> = null;\n private _initServicePromise: Nullable<Promise<void>> = null;\n private _registeredStreams: Map<KnownMediaStream, TechseeMediaStream> = new Map();\n private _isVoipEnabled: boolean = false;\n private _session: Nullable<{\n params: MediaSessionParams;\n connectPromise?: Promise<void>;\n instance?: IMediaSession;\n }> = null;\n\n protected constructor(environment: IMediaEnvironment, webRtcSupportInfo: ClientWebRtcInfo) {\n trace.info('TechseeMediaServiceBase created');\n\n this.bindClassMethods();\n\n this._emitter = new EventEmitter();\n this._environment = environment;\n this._deviceSupportFlags = {\n videoPlayback: false,\n hasCamera: false,\n hasMicrophone: false,\n webRtcSupportInfo: webRtcSupportInfo\n };\n\n this._localStreamsManager = new LocalStreamManager(this._environment);\n\n // TODO: Hack as IOS 13.3 and above sometimes do not return the video/audio device. This\n // is a hack to work around the issue until Apple resolve it\n this._isIOS_13_orLater = this._environment.isIOS() &&\n (!this._environment.majorVersion() || (this._environment.majorVersion() >= 13));\n\n trace.info(`XXXXX: this._isIOS_13_orLater: ${this._isIOS_13_orLater}`);\n trace.info(`XXXXX: this._environment.isIOS: ${this._environment.isIOS()}`);\n trace.info(`XXXXX: this._environment.majorVersion(): ${this._environment.majorVersion()}`);\n\n this._sessionStreamsManager = {\n getMediaStreamForRole: this.getStreamForDestinationRole,\n addRemoteMediaTrack: this.registerRemoteMediaTrack,\n removeMediaTrack: this.unregisterRemoteMediaTrack\n };\n }\n\n get deviceSupportInfo(): DeviceSupportInfo {\n this.serviceInitGuard(false);\n\n return cloneDeep(this._deviceSupportFlags);\n }\n\n get isVoipEnabled(): boolean {\n return this._isVoipEnabled;\n }\n\n get isSessionActive(): boolean {\n return this._session !== null;\n }\n\n get isLocalStreamInitialized(): boolean {\n return this._initLocalStreamsPromise !== null;\n }\n\n /*\n Initializes media service. Before using anything from this service it should be initialized first.\n All of serviceOptions should be retrieved from account settings and by evaluating application state.\n For example, when initializing service, accountSettings is already loaded, and we know if FS session is audio enabled.\n */\n initMediaService(serviceOptions: MediaServiceOptions): Promise<void> {\n trace.info('TechseeMediaServiceBase.initMediaService', serviceOptions);\n const warnMessage = 'Multiple initialization does not supposed to happen. Not rejecting, but check the flow for correctness.';\n\n if (this._initServicePromise || this._serviceOptions) {\n console.warn(warnMessage);\n\n return this._initServicePromise ? this._initServicePromise : Promise.resolve();\n }\n\n this._serviceOptions = cloneDeep(serviceOptions);\n const isOpenTok = this._serviceOptions.mediaServiceType === MediaServiceType.OPENTOK;\n\n this._initServicePromise = Promise.all([\n isVideoPlaySupportedOnDevice(isOpenTok)\n .then((isSupported: boolean) => this._deviceSupportFlags.videoPlayback = isSupported),\n enumerateMediaDevices()\n .catch((error) => {\n trace.warn(error);\n\n return [];\n })\n .then((localDevicesList: MediaDeviceInfo[]) => {\n const groupedDevices = {\n video: filter(localDevicesList, (device) => device.kind.toLowerCase() === 'videoinput'),\n audio: filter(localDevicesList, (device) => device.kind.toLowerCase() === 'audioinput')\n };\n\n this._localStreamsManager.setGroupedDevices(groupedDevices);\n this._deviceSupportFlags.hasCamera = this._isIOS_13_orLater || (groupedDevices.video.length > 0);\n this._deviceSupportFlags.hasMicrophone = this._isIOS_13_orLater || (groupedDevices.audio.length > 0);\n\n })\n ]).then(() => undefined);\n\n\n return this._initServicePromise;\n }\n\n //#region Media Streams Management\n\n /*\n The main method to start the initialization of local media streams.\n This method can be executed many times and it will insure all calls are synced and only relevant stream will be created.\n DO NOT USE ANY OTHER METHOD THAT FORCES \"navigator.getUserMedia\" OTHER THAN THIS ONE.\n */\n initLocalMediaStreams(): Promise<void> {\n this.serviceInitGuard();\n\n if (!this._initLocalStreamsPromise) {\n trace.info('initLocalMediaStreams creating media streams');\n this._initLocalStreamsPromise = this.getLocalMediaImplementation()\n .catch((mediaRequestFailResult: MediaRequestFailResult) => {\n trace.error('initLocalMediaStreams failure: ', mediaRequestFailResult);\n\n throw mediaRequestFailResult;\n });\n } else {\n trace.info('initLocalMediaStreams already done or in progress');\n }\n\n return this._initLocalStreamsPromise;\n }\n\n /*\n Creates an instance of subscriber that will \"listen\" to the requested stream.\n Whenever requested stream is available, subscriber will render it.\n */\n createSubscriber(subscriberParams: MediaSubscriberParams): Promise<IMediaSubscriber> {\n if (this._subscribers.has(subscriberParams.container)) {\n return Promise.reject('Subscriber for provided DIV element already exists');\n }\n\n const subscriber = new TechseeMediaSubscriber(subscriberParams);\n\n this._subscribers.set(subscriberParams.container, subscriber);\n const streamForSubscriber = this.getRegisteredStreamByType(subscriber.streamType);\n\n if (streamForSubscriber) {\n trace.info(`Stream for ${subscriberParams.streamType} is ready.`);\n subscriber.renderStream(streamForSubscriber);\n } else {\n trace.info(`Stream for ${subscriberParams.streamType} not created yet`);\n }\n\n return Promise.resolve(subscriber);\n }\n\n /*\n Destroys the subscriber for provided HTML DIV container\n */\n destroySubscriber(container: HTMLDivElement): Promise<void> {\n if (this._subscribers.has(container)) {\n const subscriber = this._subscribers.get(container);\n\n subscriber!.dispose();\n }\n\n this._subscribers.delete(container);\n\n return Promise.resolve();\n }\n\n onStreamDestroyed(callback: (eventArgs: StreamDestroyedEventArgs) => void): void {\n this.registerEventCallback(privateEvents.STREAM_DESTROYED, callback);\n }\n\n onStreamCreated(callback: (eventArgs: StreamCreatedEventArgs) => void): void {\n this.registerEventCallback(privateEvents.STREAM_CREATED, callback);\n }\n\n private getSwitchCameraConstraints() {\n const constraints = cloneDeep(window.latestLocalMediaConstraints);\n\n if (!constraints) {\n throw new Error('getSwitchCameraConstraints: unexpected use case constraints is null.');\n }\n\n trace.info('getSwitchCameraConstraints - Start switch camera with constraints:', constraints);\n\n const videoSourceType = (constraints.video as LocalVideoStreamConstraints).videoSourceType === LocalVideoSourceType.CAMERA\n ? LocalVideoSourceType.CAMERA_FRONT\n : LocalVideoSourceType.CAMERA;\n\n trace.info(`getSwitchCameraConstraints - switch to videoSourceType: ${videoSourceType}`);\n\n (constraints.video as LocalVideoStreamConstraints).videoSourceType = videoSourceType;\n\n return constraints;\n }\n\n switchCamera(isFailed?: boolean) {\n trace.info('switchCamera: start');\n\n const constraints = !isFailed ? this.getSwitchCameraConstraints() : cloneDeep(window.latestLocalMediaConstraints);\n\n if (!constraints) {\n throw new Error('switchCamera - unexpected use case constraints is null.');\n }\n\n this._localStreamsManager.destroyUserMediaStream();\n\n return this._localStreamsManager.getUserMediaStream(constraints)\n .then((streamResult: MediaRequestSuccessResult) => {\n const stream = streamResult.mediaStream;\n\n trace.info('switchCamera: new stream: ', stream);\n\n this.replaceStreamTracks(stream);\n\n return this.registerStreamResult(constraints, streamResult, true).then(() => ({constraints, streamResult}));\n })\n .catch((err) => {\n trace.error('switchCamera: Failed to switch camera: ', err);\n\n if (!isFailed) {\n this.switchCamera(true);\n }\n\n return Promise.reject('Failed to switch camera.');\n });\n }\n\n private replaceStreamTracks(mediaStream: Nullable<MediaStream>) {\n this._session!.instance && this._session!.instance.replaceStreamTracks(mediaStream);\n }\n\n protected registerStreamResult(constraints: Nullable<LocalMediaConstraints>, streamResult: MediaRequestSuccessResult, switchCamera?: boolean, addStreamType?: KnownMediaStream) {\n const regPromises: any[] = [];\n\n if (streamResult.isNew) {\n trace.info('registerStreamResult: stream result from getUserMediaStream:', streamResult.mediaStream);\n streamResult.mediaStream.getTracks().forEach((mediaTrack) => {\n trace.info('registerStreamResult: stream result from mediaTrack:', mediaTrack);\n const streamType = addStreamType || (mediaTrack.kind === 'video'\n ? KnownMediaStream.USER_VIDEO_STREAM : KnownMediaStream.USER_AUDIO_STREAM);\n const newDedicatedStream = new TechseeMediaStream(mediaTrack, streamType, false);\n\n regPromises.push(switchCamera ? this.registerTrack(newDedicatedStream) : this.registerLocalMediaStream(newDedicatedStream));\n });\n }\n\n return Promise.all(regPromises);\n }\n\n get supportSwitchCameras() {\n return this._isIOS_13_orLater || this._localStreamsManager.groupedDevices.camerasCount > 1;\n }\n\n //#endregion Media Streams Management\n\n //#region Session Management\n\n /*\n Creates instance of WebRTC session which connects to (signaling server), and begins to listen to WebRTC events.\n */\n connectToSession(sessionParams: MediaSessionParams): Promise<void> {\n this.serviceInitGuard();\n\n if (!this._session) {\n this._session = {params: cloneDeep(sessionParams)};\n this._session.connectPromise = new Promise((resolve, reject) => {\n const doAsyncReject = (error: any): void => {\n\n /*\n Reject with timeout, to change JS 'execution flow'.\n If timeout removed, session is assigned to null, before promise returns, and \"catch\" will not\n work on promise that is returned.\n */\n setTimeout(() => {\n this._session = null;\n reject(error);\n });\n };\n\n const sessionHandlers: MediaSessionHandlers = {\n onDisconnectedHandler: this.onSessionDisconnectHandler\n };\n const mediaSessionParams: MediaSessionParams & MediaSessionConfiguration = {\n ...sessionParams,\n peerConnectivityTimeoutSeconds: this._serviceOptions!.peerConnectivityTimeoutSeconds || DEFAULT_PEER_CONNECTIVITY_TIMEOUT_SECONDS\n };\n\n trace.info('connectToSession', sessionParams);\n if (this._serviceOptions!.mediaServiceType === MediaServiceType.TURNSERVER) {\n this._session!.instance = new TurnWebRtcSession(mediaSessionParams, sessionHandlers, this._sessionStreamsManager);\n\n } else if (this._serviceOptions!.mediaServiceType === MediaServiceType.OPENTOK) {\n this._session!.instance = new OpentokSession(mediaSessionParams, this._sessionStreamsManager);\n } else {\n const allowedValues = values(MediaServiceType);\n\n trace.error(`mediaServiceType '${this._serviceOptions!.mediaServiceType}' is not supported.`, {allowedValues});\n doAsyncReject(new Error('mediaServiceType is not supported'));\n }\n\n if (this._session && this._session.instance) {\n this._session!.instance.connect()\n .then(() => {\n trace.info('Session is connected');\n resolve();\n })\n .catch((error: any) => {\n trace.error('Failed to connect to session', error);\n doAsyncReject(error);\n });\n }\n\n });\n } else {\n trace.warn('Session connected already. Disconnect before connect to new session again');\n }\n\n //TODO - Alex: Change a structure of this function to be more safe\n //(when we return, there should not be case when session or promise is null)\n //To fix, need to return promise immediately, then execute process in setTimeout(fn, 0);\n return this._session!.connectPromise!;\n }\n\n /*\n Disconnects from WebRTC session.\n */\n disconnectFromSession(): Promise<void> {\n return this.disconnectFromSessionInternal(MediaSessionDisconnectReason.ForcedByConsumer);\n }\n\n /*\n Updates a credentials for turn server\n */\n updateSessionCredentials(credentials: MediaSessionIceCredentials): Promise<void> {\n trace.info('updateSessionCredentials');\n if (!this.sessionExistsGuard(false)) {\n return Promise.reject('There no session to update credentials');\n }\n\n this._session!.params.credentials = cloneDeep(credentials);\n\n return Promise.resolve();\n }\n\n enableVoipDuringSession(): void {\n if (!this._isVoipEnabled && this.isLocalStreamInitialized) {\n throw new Error('Voip support cannot be enabled after local stream where initialized');\n }\n\n this._isVoipEnabled = true;\n }\n\n protected disconnectFromSessionInternal(reason: MediaSessionDisconnectReason): Promise<void> {\n if (!this._session) {\n return Promise.resolve();\n }\n\n trace.info('Disconnecting from session', reason);\n\n const lastSession = this._session;\n\n this._session = null;\n\n return Promise.all([\n lastSession.instance!.disconnect(),\n this.clearPublishers()\n ]).then(() => undefined);\n }\n\n protected getStatsForRemoteTrack(streamType: KnownMediaStream): Promise<RemoteTrackStats> {\n if (!this.sessionExistsGuard(false)) {\n return Promise.reject(new Error('Media session not started'));\n }\n\n const streamForStats = this.getRegisteredStreamByType(streamType);\n\n if (!streamForStats) {\n return Promise.reject(new Error('Stream for requested type was not found'));\n }\n\n return this._session!.instance!.getRemoteTrackStats(streamForStats.mediaTrack)\n .then((trackStats: RemoteTrackStats) => {\n traceStatsInfo(`MediaTrackStats for ${streamType}: ${JSON.stringify(trackStats)}`);\n\n return {\n trackId: streamForStats.mediaTrack.id,\n trackStats: trackStats\n };\n });\n }\n\n private onSessionDisconnectHandler(reason: MediaSessionDisconnectReason): void {\n trace.info('onSessionDisconnectHandler', reason);\n if (reason !== MediaSessionDisconnectReason.ForcedByConsumer) {\n const lastParams = this._session && this._session.params ? this._session.params : null;\n\n this.disconnectFromSessionInternal(reason).then(() => {\n const reconnectReasons = [\n MediaSessionDisconnectReason.InitiatorPeerReconnected,\n MediaSessionDisconnectReason.SignalingChannelDisconnect,\n MediaSessionDisconnectReason.PeerConnectionInterrupted,\n MediaSessionDisconnectReason.PeerConnectionStateChangeTimeout\n ];\n\n if (lastParams && includes(reconnectReasons, reason) && this._autoReconnectEnabled) {\n this.reconnectToSession(lastParams);\n } else if (!this._autoReconnectEnabled) {\n trace.info('No reconnection- auto reconnect disabled');\n } else {\n trace.info('No params for reconnection to media session');\n }\n });\n } else {\n trace.info('Ignore Session disconnect event');\n }\n }\n\n setAutoReconnect(state: boolean) {\n this._autoReconnectEnabled = state;\n trace.info('_autoReconnectEnabled:', state);\n\n if (this._autoReconnectEnabled && !this.isSessionActive) {\n const lastParams = this._session && this._session.params ? this._session.params : null;\n\n if (lastParams) {\n trace.info('setAutoReconnect - reconnect to session');\n this.reconnectToSession(lastParams);\n }\n }\n }\n\n reconnectToSession(lastParams: MediaSessionParams) {\n trace.info('Reconnecting to media session, sessionParams:', lastParams);\n\n this.connectToSession(lastParams)\n .then(() => {\n trace.info('Media session reconnected');\n })\n .catch((error: any) => {\n trace.error('Error while reconnecting to media session', error);\n });\n }\n\n //#endregion Session Management\n\n //#region Utils\n\n getSnapshotFromKnownStream(sourceStream: KnownMediaStream, snapshotOptions?: SnapshotOptions): Promise<SnapshotResult> {\n return new Promise((resolve, reject) => {\n if (sourceStream !== KnownMediaStream.USER_VIDEO_STREAM &&\n sourceStream !== KnownMediaStream.USER_SCREEN_SHARE_STREAM) {\n trace.error('The requested stream is not video stream, and cannot be used for snapshot');\n reject(new Error('INCOMPATIBLE_STREAM_FOR_SNAPSHOT'));\n\n return;\n }\n\n const snapshotStream = this.getRegisteredStreamByType(sourceStream);\n\n if (!snapshotStream) {\n trace.error('Cannot make snapshot: The requested stream not exists yet.');\n reject(new Error('NO_REQUESTED_STREAM'));\n\n return;\n }\n\n getSnapshotFromMediaStream(snapshotStream.mediaStream, snapshotOptions)\n .then((imageData: any) => {\n const urlComponents = imageData.split(';base64,');\n const mimeType = urlComponents[0].split(':')[1];\n const bytes = atob(urlComponents[1]);\n const buffer = new ArrayBuffer(bytes.length);\n const rawData = new Uint8Array(buffer);\n\n for (let i = 0; i < bytes.length; i++) {\n rawData[i] = bytes.charCodeAt(i);\n }\n\n const blob = new Blob([rawData], {type: mimeType});\n const objectUrl = window.URL.createObjectURL(blob);\n\n trace.info('Snapshot created successfully');\n\n const result: SnapshotResult = {\n base64img: imageData,\n objectUrl: objectUrl,\n imageBlob: blob,\n mimeType: mimeType\n };\n\n resolve(result);\n })\n .catch((error: any) => {\n trace.error('Error creating snapshot', error);\n reject(error);\n });\n });\n }\n\n //Will clean streams, publishers and subscribers, but will not remove event listeners\n clearService(): Promise<void> {\n trace.info('MediaService clearing all resources');\n\n return this.disconnectFromSessionInternal(MediaSessionDisconnectReason.ForcedByConsumer)\n .then(this.clearSubscribers)\n .then(this.clearRegisteredStreams)\n .then(this._localStreamsManager.clearAllStreams)\n .then(() => {\n this._initLocalStreamsPromise = null;\n this._isVoipEnabled = false;\n }).then(() => undefined);\n }\n\n //Will clear the service and remove all event listeners\n dispose(): Promise<void> {\n return this.clearService().catch(() => undefined).then(() => {\n this._emitter.removeAllListeners();\n });\n }\n\n //#endregion\n\n //#region Protected Methods\n\n protected get subscribers(): Map<HTMLDivElement, TechseeMediaSubscriber> {\n return this._subscribers;\n }\n\n protected registerLocalMediaStream(tsMediaStream: TechseeMediaStream): Promise<void> {\n trace.info('Registering local stream', tsMediaStream.streamType);\n if (this._registeredStreams.has(tsMediaStream.streamType)) {\n return Promise.reject(new Error(`Stream ${tsMediaStream.streamType} already registered`));\n }\n\n return this.registerStream(tsMediaStream);\n }\n\n protected getRegisteredStreamByType(streamType: KnownMediaStream): Nullable<TechseeMediaStream> {\n return this._registeredStreams.get(streamType) || null;\n }\n\n protected changeEnableForKnownStream(streamType: KnownMediaStream, isPaused: boolean): void {\n const streamToChangeState = this.getRegisteredStreamByType(streamType);\n\n if (!streamToChangeState) {\n trace.warn('There no stream found to change enable state', streamType);\n } else {\n streamToChangeState.mediaTrack.enabled = !isPaused;\n\n if (streamToChangeState.isRemote && streamToChangeState.streamKind === KnownMediaStreamKind.Audio) {\n this.subscribers.forEach((subscriber) => {\n if (subscriber.streamType === streamType) {\n subscriber.muteSound(isPaused);\n }\n });\n }\n\n trace.info('Local stream enable state is changed', {streamType, isPaused});\n }\n }\n\n protected registerEventCallback(event: string, callback: (eventArgs?: any) => void): void {\n this._emitter.on(event, callback);\n }\n\n protected emitEvent(event: string, eventArgs?: any): void {\n setTimeout(() => {\n this._emitter.emit(event, eventArgs);\n });\n }\n\n //#endregion Protected Methods\n\n //#region Private Methods\n\n private registerRemoteMediaTrack(remoteMediaTrack: RemoteMediaTrack): Promise<void> {\n trace.info(`Registering remote ${remoteMediaTrack.trackType} MediaStreamTrack`, remoteMediaTrack.mediaTrack);\n const currentStream = this._registeredStreams.get(remoteMediaTrack.trackType);\n\n if (currentStream && !currentStream.isRemote) {\n return Promise.reject(new Error('Cannot register remote stream with the same type as local stream'));\n }\n\n const newDedicatedStream = new TechseeMediaStream(remoteMediaTrack.mediaTrack, remoteMediaTrack.trackType, true);\n\n return this.registerStream(newDedicatedStream);\n }\n\n private registerTrack(mediaStream: TechseeMediaStream): Promise<void> {\n mediaStream.mediaTrack.onended = () =>\n this.unregisterTechseeMediaStream(mediaStream, MediaStreamUnregisterReason.NativeEvent);\n\n this._registeredStreams.set(mediaStream.streamType, mediaStream);\n\n trace.info(`TechseeMediaStream registered - ${mediaStream.streamType}`, TechseeMediaStream);\n\n return this.updateSubscribersWithNewStream(mediaStream).then(() => {\n const eventArgs: StreamCreatedEventArgs = {streamType: mediaStream.streamType};\n\n this.emitEvent(privateEvents.STREAM_CREATED, eventArgs);\n });\n }\n\n private registerStream(mediaStream: TechseeMediaStream): Promise<void> {\n const currentStream = this._registeredStreams.get(mediaStream.streamType);\n\n if (currentStream) {\n return this.unregisterTechseeMediaStream(currentStream, MediaStreamUnregisterReason.ReplacingStream).then(() => this.registerTrack(mediaStream));\n }\n\n return this.registerTrack(mediaStream);\n }\n\n private unregisterTechseeMediaStream(streamToUnregister: TechseeMediaStream, reason: MediaStreamUnregisterReason): Promise<void> {\n const traceError = (error: any): void => {\n trace.warn('Unregister TechseeMediaStream error:', error);\n };\n\n const promises: any = [];\n\n trace.info('Unregister TechseeMediaStream: ', streamToUnregister.streamType);\n\n this._subscribers.forEach((subscriber) => {\n if (subscriber.streamType === streamToUnregister.streamType) {\n promises.push(subscriber.stopRendering().catch(traceError));\n }\n });\n\n if (!streamToUnregister.isRemote) {\n this._publishers.forEach((publisher: TechseeMediaPublisher) => {\n if (publisher.streamTypes.find((streamType) => streamType === streamToUnregister.streamType)) {\n if (this._session && this._session!.instance) {\n promises.push(this._session.instance.onMediaStreamDestroyed(publisher.destinationRole).catch(traceError));\n }\n }\n });\n }\n\n return Promise.all(promises)\n .then(() => {\n this._registeredStreams.delete(streamToUnregister.streamType);\n trace.info('Stream deleted', streamToUnregister.streamType);\n\n const eventArgs: StreamDestroyedEventArgs = {streamType: streamToUnregister.streamType, reason: reason};\n\n this.emitEvent(privateEvents.STREAM_DESTROYED, eventArgs);\n });\n }\n\n private unregisterRemoteMediaTrack(mediaTrack: MediaStreamTrack): Promise<void> {\n const promises: any = [];\n\n this._registeredStreams.forEach((registeredStream: TechseeMediaStream) => {\n if (registeredStream.mediaTrack.id === mediaTrack.id) {\n promises.push(this.unregisterTechseeMediaStream(registeredStream, MediaStreamUnregisterReason.ClosedRemotely));\n }\n });\n\n return Promise.all(promises).then(() => undefined);\n }\n\n private removePublisher(publisher: TechseeMediaPublisher): Promise<void> {\n this._publishers.delete(publisher.destinationRole);\n if (this._session && this._session.instance) {\n trace.info('Removing publisher', publisher.destinationRole);\n\n return this._session.instance.onMediaStreamDestroyed(publisher.destinationRole)\n .catch((error: any) => {\n trace.warn('Error while removing publisher', error);\n });\n }\n\n return Promise.resolve();\n }\n\n private getStreamForDestinationRole(destinationRole: SessionClientRole): Promise<Nullable<MediaStream>> {\n this.serviceInitGuard();\n\n if (this._publishers.has(destinationRole)) {\n trace.info(`Publisher for ${destinationRole} already exists`);\n\n return Promise.resolve(this._publishers.get(destinationRole)!.mediaStream);\n }\n\n if (!this._publisherPromises.get(destinationRole)) {\n trace.info(`Creating publisher for ${destinationRole}`);\n const publisherPromise = this.initLocalMediaStreams()\n .then(() => this.createMediaPublisher(destinationRole))\n .then((mediaPublisher: Nullable<TechseeMediaPublisher>) => {\n if (mediaPublisher) {\n this._publishers.set(destinationRole, mediaPublisher);\n }\n\n this._publisherPromises.delete(destinationRole);\n\n return mediaPublisher ? mediaPublisher.mediaStream : null;\n })\n .catch((ex: any) => {\n if (ex && ex.message === 'audioStreamFailed') {\n this._isVoipEnabled = false;\n\n return null;\n }\n\n this._publisherPromises.delete(destinationRole);\n throw ex;\n });\n\n this._publisherPromises.set(destinationRole, publisherPromise);\n } else {\n trace.info(`Create publisher promise for ${destinationRole} already exists`);\n }\n\n return this._publisherPromises.get(destinationRole)!;\n }\n\n private updateSubscribersWithNewStream(registeredStream: TechseeMediaStream): Promise<void> {\n trace.info(`Updating subscribers of ${registeredStream.streamType} with new stream`);\n\n if (this._subscribers.size === 0) {\n trace.warn(`No subscribers exists for the ${registeredStream.streamType}.`);\n } else {\n trace.info(`Total ${this._subscribers.size} subscribers exists, will check if rerender needed.`);\n }\n\n this._subscribers.forEach((subscriber) => {\n if (subscriber.streamType === registeredStream.streamType) {\n trace.info(`${registeredStream.streamType} rendering on subscriber`);\n subscriber.renderStream(registeredStream);\n }\n });\n\n return Promise.resolve();\n }\n\n private bindClassMethods(): void {\n this.updateSessionCredentials = this.updateSessionCredentials.bind(this);\n this.getStreamForDestinationRole = this.getStreamForDestinationRole.bind(this);\n this.registerRemoteMediaTrack = this.registerRemoteMediaTrack.bind(this);\n this.unregisterRemoteMediaTrack = this.unregisterRemoteMediaTrack.bind(this);\n this.createMediaPublisher = this.createMediaPublisher.bind(this);\n this.updateSubscribersWithNewStream = this.updateSubscribersWithNewStream.bind(this);\n this.initLocalMediaStreams = this.initLocalMediaStreams.bind(this);\n this.onSessionDisconnectHandler = this.onSessionDisconnectHandler.bind(this);\n this.clearService = this.clearService.bind(this);\n this.clearRegisteredStreams = this.clearRegisteredStreams.bind(this);\n this.clearPublishers = this.clearPublishers.bind(this);\n this.clearSubscribers = this.clearSubscribers.bind(this);\n }\n\n //region Cleanup\n\n private clearRegisteredStreams(): Promise<void> {\n trace.info('Clearing registered streams');\n const promises: any = [];\n\n this._registeredStreams.forEach((streamToUnregister: TechseeMediaStream) => {\n promises.push(this.unregisterTechseeMediaStream(streamToUnregister, MediaStreamUnregisterReason.ServiceCleanUp));\n });\n\n return Promise.all(promises).then(() => undefined);\n }\n\n private clearSubscribers(): Promise<void> {\n trace.info('Clearing subscribers');\n const promises: any = [];\n\n this._subscribers.forEach((subscriber) => {\n promises.push(this.destroySubscriber(subscriber.container));\n });\n\n return Promise.all(promises).then(() => undefined);\n }\n\n private clearPublishers(): Promise<void> {\n trace.info('Clearing publishers');\n const promises: any = [];\n\n this._publishers.forEach((publisher) => {\n promises.push(this.removePublisher(publisher));\n });\n\n return Promise.all(promises).then(() => undefined);\n }\n\n //#endregion Cleanup\n\n //#region Simple Validation Methods\n\n protected serviceInitGuard(shouldThrow = true): boolean {\n return throwableGuard(!!this._serviceOptions, 'Media service is not initialized', shouldThrow);\n }\n\n protected sessionExistsGuard(shouldThrow = true): boolean {\n return throwableGuard(!!this._session, 'There no active session', shouldThrow);\n }\n\n //#endregion\n\n //#endregion Private Methods\n}\n"]}
|