@techsee/techsee-media-service 999.9.0-switch → 999.9.1-sap
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.
|
@@ -1 +1 @@
|
|
|
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;AAY1B,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,UAAU,eAAe;IACrB,KAAK,EAAE,QAAQ,CAAC;QAAE,IAAI,CAAC,EAAE,eAAe,CAAC;QAAC,KAAK,CAAC,EAAE,eAAe,CAAA;KAAE,CAAC,CAAC;IACrE,KAAK,EAAE,QAAQ,CAAC,eAAe,CAAC,CAAC;IACjC,YAAY,EAAE,MAAM,CAAC;CACxB;AAED,qBAAa,kBAAkB;IAC3B,OAAO,CAAC,WAAW,CAAoB;IACvC,OAAO,CAAC,eAAe,CAAgE;gBAE3E,WAAW,EAAE,iBAAiB;aAOtC,cAAc;IAIlB,kBAAkB,CAAC,WAAW,EAAE,qBAAqB,GAAG,OAAO,CAAC,yBAAyB,CAAC;IA2C1F,qBAAqB,CAAC,WAAW,EAAE,2BAA2B,GAAG,OAAO,CAAC,yBAAyB,CAAC;IA4CnG,sBAAsB,IAAI,OAAO,CAAC,IAAI,CAAC;IAsBvC,yBAAyB,IAAI,OAAO,CAAC,IAAI,CAAC;IAqB1C,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;IAmE5B,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 CameraTypes,\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\ninterface IGroupedDevices {\n video: Nullable<{ back?: MediaDeviceInfo; front?: MediaDeviceInfo }>;\n audio: Nullable<MediaDeviceInfo>;\n camerasCount: number;\n}\n\nexport class LocalStreamManager {\n private environment: IMediaEnvironment;\n private _groupedDevices: IGroupedDevices = {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 window.latestLocalMediaConstraints.audio = (streamResult.constraint && streamResult.constraint.audio) as boolean;\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 try {\n streamResult.mediaStream.getTracks().forEach((track: MediaStreamTrack) => {\n track.stop();\n });\n\n trace.info('destroyUserMediaStream - Failed to stop all tracks');\n } catch (e) {\n trace.info('destroyUserMediaStream - stop all tracks');\n }\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 try {\n streamResult.mediaStream.getTracks().forEach((track: MediaStreamTrack) => {\n track.stop();\n });\n\n trace.info('destroyDesktopMediaStream - stop all tracks');\n } catch (e) {\n trace.info('destroyDesktopMediaStream - Failed to stop all tracks');\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 ? {[CameraTypes.BACK]: last(groupedDevices.video), [CameraTypes.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\n ? CameraTypes.FRONT\n : CameraTypes.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"]}
|
|
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;AAY1B,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,UAAU,eAAe;IACrB,KAAK,EAAE,QAAQ,CAAC;QAAE,IAAI,CAAC,EAAE,eAAe,CAAC;QAAC,KAAK,CAAC,EAAE,eAAe,CAAA;KAAE,CAAC,CAAC;IACrE,KAAK,EAAE,QAAQ,CAAC,eAAe,CAAC,CAAC;IACjC,YAAY,EAAE,MAAM,CAAC;CACxB;AAED,qBAAa,kBAAkB;IAC3B,OAAO,CAAC,WAAW,CAAoB;IACvC,OAAO,CAAC,eAAe,CAAgE;gBAE3E,WAAW,EAAE,iBAAiB;aAOtC,cAAc;IAIlB,kBAAkB,CAAC,WAAW,EAAE,qBAAqB,GAAG,OAAO,CAAC,yBAAyB,CAAC;IA2C1F,qBAAqB,CAAC,WAAW,EAAE,2BAA2B,GAAG,OAAO,CAAC,yBAAyB,CAAC;IA4CnG,sBAAsB,IAAI,OAAO,CAAC,IAAI,CAAC;IAsBvC,yBAAyB,IAAI,OAAO,CAAC,IAAI,CAAC;IAqB1C,eAAe,IAAI,OAAO,CAAC,IAAI,CAAC;IAShC,OAAO,CAAC,0BAA0B;IAwDlC,OAAO,CAAC,6BAA6B;IAiBrC,OAAO,CAAC,mCAAmC;IAmD3C,iBAAiB,CAAC,cAAc,EAAE;QAAC,KAAK,EAAE,eAAe,EAAE,CAAC;QAAC,KAAK,EAAE,eAAe,EAAE,CAAA;KAAC;IAStF,OAAO,CAAC,oBAAoB;IAsE5B,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 CameraTypes,\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\ninterface IGroupedDevices {\n video: Nullable<{ back?: MediaDeviceInfo; front?: MediaDeviceInfo }>;\n audio: Nullable<MediaDeviceInfo>;\n camerasCount: number;\n}\n\nexport class LocalStreamManager {\n private environment: IMediaEnvironment;\n private _groupedDevices: IGroupedDevices = {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 window.latestLocalMediaConstraints.audio = (streamResult.constraint && streamResult.constraint.audio) as boolean;\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 try {\n streamResult.mediaStream.getTracks().forEach((track: MediaStreamTrack) => {\n track.stop();\n });\n\n trace.info('destroyUserMediaStream - stop all tracks');\n } catch (e) {\n trace.info('destroyUserMediaStream - Failed to stop all tracks', e);\n }\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 try {\n streamResult.mediaStream.getTracks().forEach((track: MediaStreamTrack) => {\n track.stop();\n });\n\n trace.info('destroyDesktopMediaStream - stop all tracks');\n } catch (e) {\n trace.info('destroyDesktopMediaStream - Failed to stop all tracks', e);\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 ((newConstraints.video as MediaTrackConstraints).deviceId as ConstrainDOMStringParameters) = {\n exact: (constraints.video as MediaTrackConstraints).deviceId\n } as any as ConstrainDOMStringParameters;\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 ? {[CameraTypes.BACK]: last(groupedDevices.video), [CameraTypes.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 frameRate: {ideal: 15, max: 30}\n };\n\n const facingMode = cameraType && cameraType === LocalVideoSourceType.CAMERA_FRONT\n ? 'user'\n : 'environment';\n const deviceId = get(this._groupedDevices, `video.${cameraType === LocalVideoSourceType.CAMERA_FRONT\n ? CameraTypes.FRONT\n : CameraTypes.BACK}.deviceId`);\n\n if (deviceId) {\n baseConstraints.video.deviceId = deviceId;\n baseConstraints.video.facingMode = {exact: facingMode};\n } else {\n baseConstraints.video.facingMode = facingMode;\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"]}
|
|
@@ -122,9 +122,9 @@ var LocalStreamManager = /** @class */function () {
|
|
|
122
122
|
streamResult.mediaStream.getTracks().forEach(function (track) {
|
|
123
123
|
track.stop();
|
|
124
124
|
});
|
|
125
|
-
trace.info('destroyUserMediaStream - Failed to stop all tracks');
|
|
126
|
-
} catch (e) {
|
|
127
125
|
trace.info('destroyUserMediaStream - stop all tracks');
|
|
126
|
+
} catch (e) {
|
|
127
|
+
trace.info('destroyUserMediaStream - Failed to stop all tracks', e);
|
|
128
128
|
}
|
|
129
129
|
}
|
|
130
130
|
return bluebird_1.Promise.resolve();
|
|
@@ -140,7 +140,7 @@ var LocalStreamManager = /** @class */function () {
|
|
|
140
140
|
});
|
|
141
141
|
trace.info('destroyDesktopMediaStream - stop all tracks');
|
|
142
142
|
} catch (e) {
|
|
143
|
-
trace.info('destroyDesktopMediaStream - Failed to stop all tracks');
|
|
143
|
+
trace.info('destroyDesktopMediaStream - Failed to stop all tracks', e);
|
|
144
144
|
}
|
|
145
145
|
}
|
|
146
146
|
return bluebird_1.Promise.resolve();
|
|
@@ -226,6 +226,9 @@ var LocalStreamManager = /** @class */function () {
|
|
|
226
226
|
trace.info("getStreamPromiseByParsedConstraints - NotReadableError - Failed to get video user media stream with facingMode=" + get_1.default(constraints, 'video.facingMode'), err.name);
|
|
227
227
|
// @ts-ignore
|
|
228
228
|
delete newConstraints.video.facingMode;
|
|
229
|
+
newConstraints.video.deviceId = {
|
|
230
|
+
exact: constraints.video.deviceId
|
|
231
|
+
};
|
|
229
232
|
streamPromise = _this.getStreamPromiseByParsedConstraints(newConstraints, isDesktopMedia);
|
|
230
233
|
} else if (err && err.name && constraints.audio && !constraints.video) {
|
|
231
234
|
trace.info('getStreamPromiseByParsedConstraints - audioStreamFailed - Failed to get audio user media stream', err.name);
|
|
@@ -268,12 +271,15 @@ var LocalStreamManager = /** @class */function () {
|
|
|
268
271
|
}
|
|
269
272
|
var cameraType = streamConstraints.video !== 'boolean' && streamConstraints.video.videoSourceType;
|
|
270
273
|
baseConstraints.video = {
|
|
271
|
-
facingMode: cameraType && cameraType === MediaConstants_1.LocalVideoSourceType.CAMERA_FRONT ? 'user' : 'environment',
|
|
272
274
|
frameRate: { ideal: 15, max: 30 }
|
|
273
275
|
};
|
|
276
|
+
var facingMode = cameraType && cameraType === MediaConstants_1.LocalVideoSourceType.CAMERA_FRONT ? 'user' : 'environment';
|
|
274
277
|
var deviceId = get_1.default(this._groupedDevices, "video." + (cameraType === MediaConstants_1.LocalVideoSourceType.CAMERA_FRONT ? MediaConstants_1.CameraTypes.FRONT : MediaConstants_1.CameraTypes.BACK) + ".deviceId");
|
|
275
278
|
if (deviceId) {
|
|
276
279
|
baseConstraints.video.deviceId = deviceId;
|
|
280
|
+
baseConstraints.video.facingMode = { exact: facingMode };
|
|
281
|
+
} else {
|
|
282
|
+
baseConstraints.video.facingMode = facingMode;
|
|
277
283
|
}
|
|
278
284
|
var assignConstraint = function assignConstraint(videoResolution) {
|
|
279
285
|
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","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","e","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","setGroupedDevices","groupedDevices","_a","CameraTypes","BACK","FRONT","streamConstraints","constraintCandidates","baseConstraints","cameraType","videoSourceType","LocalVideoSourceType","CAMERA_FRONT","frameRate","ideal","max","deviceId","assignConstraint","videoResolution","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":"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","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 window.latestLocalMediaConstraints.audio = (streamResult.constraint && streamResult.constraint.audio);\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 try {\n streamResult.mediaStream.getTracks().forEach(function (track) {\n track.stop();\n });\n trace.info('destroyUserMediaStream - Failed to stop all tracks');\n }\n catch (e) {\n trace.info('destroyUserMediaStream - stop all tracks');\n }\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 try {\n streamResult.mediaStream.getTracks().forEach(function (track) {\n track.stop();\n });\n trace.info('destroyDesktopMediaStream - stop all tracks');\n }\n catch (e) {\n trace.info('destroyDesktopMediaStream - Failed to stop all tracks');\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 var _a;\n trace.info('setGroupedDevices', groupedDevices);\n this._groupedDevices = { video: groupedDevices.video\n ? (_a = {}, _a[MediaConstants_1.CameraTypes.BACK] = last_1.default(groupedDevices.video), _a[MediaConstants_1.CameraTypes.FRONT] = first_1.default(groupedDevices.video), _a) : 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\n ? MediaConstants_1.CameraTypes.FRONT\n : MediaConstants_1.CameraTypes.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 CameraTypes,\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\ninterface IGroupedDevices {\n video: Nullable<{ back?: MediaDeviceInfo; front?: MediaDeviceInfo }>;\n audio: Nullable<MediaDeviceInfo>;\n camerasCount: number;\n}\n\nexport class LocalStreamManager {\n private environment: IMediaEnvironment;\n private _groupedDevices: IGroupedDevices = {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 window.latestLocalMediaConstraints.audio = (streamResult.constraint && streamResult.constraint.audio) as boolean;\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 try {\n streamResult.mediaStream.getTracks().forEach((track: MediaStreamTrack) => {\n track.stop();\n });\n\n trace.info('destroyUserMediaStream - Failed to stop all tracks');\n } catch (e) {\n trace.info('destroyUserMediaStream - stop all tracks');\n }\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 try {\n streamResult.mediaStream.getTracks().forEach((track: MediaStreamTrack) => {\n track.stop();\n });\n\n trace.info('destroyDesktopMediaStream - stop all tracks');\n } catch (e) {\n trace.info('destroyDesktopMediaStream - Failed to stop all tracks');\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 ? {[CameraTypes.BACK]: last(groupedDevices.video), [CameraTypes.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\n ? CameraTypes.FRONT\n : CameraTypes.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"]}
|
|
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","e","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","deviceId","exact","setGroupedDevices","groupedDevices","_a","CameraTypes","BACK","FRONT","streamConstraints","constraintCandidates","baseConstraints","cameraType","videoSourceType","frameRate","ideal","max","LocalVideoSourceType","CAMERA_FRONT","assignConstraint","videoResolution","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":"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","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 window.latestLocalMediaConstraints.audio = (streamResult.constraint && streamResult.constraint.audio);\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 try {\n streamResult.mediaStream.getTracks().forEach(function (track) {\n track.stop();\n });\n trace.info('destroyUserMediaStream - stop all tracks');\n }\n catch (e) {\n trace.info('destroyUserMediaStream - Failed to stop all tracks', e);\n }\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 try {\n streamResult.mediaStream.getTracks().forEach(function (track) {\n track.stop();\n });\n trace.info('destroyDesktopMediaStream - stop all tracks');\n }\n catch (e) {\n trace.info('destroyDesktopMediaStream - Failed to stop all tracks', e);\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 newConstraints.video.deviceId = {\n exact: constraints.video.deviceId\n };\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 var _a;\n trace.info('setGroupedDevices', groupedDevices);\n this._groupedDevices = { video: groupedDevices.video\n ? (_a = {}, _a[MediaConstants_1.CameraTypes.BACK] = last_1.default(groupedDevices.video), _a[MediaConstants_1.CameraTypes.FRONT] = first_1.default(groupedDevices.video), _a) : 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 frameRate: { ideal: 15, max: 30 }\n };\n var facingMode = cameraType && cameraType === MediaConstants_1.LocalVideoSourceType.CAMERA_FRONT\n ? 'user'\n : 'environment';\n var deviceId = get_1.default(this._groupedDevices, \"video.\" + (cameraType === MediaConstants_1.LocalVideoSourceType.CAMERA_FRONT\n ? MediaConstants_1.CameraTypes.FRONT\n : MediaConstants_1.CameraTypes.BACK) + \".deviceId\");\n if (deviceId) {\n baseConstraints.video.deviceId = deviceId;\n baseConstraints.video.facingMode = { exact: facingMode };\n }\n else {\n baseConstraints.video.facingMode = facingMode;\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 CameraTypes,\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\ninterface IGroupedDevices {\n video: Nullable<{ back?: MediaDeviceInfo; front?: MediaDeviceInfo }>;\n audio: Nullable<MediaDeviceInfo>;\n camerasCount: number;\n}\n\nexport class LocalStreamManager {\n private environment: IMediaEnvironment;\n private _groupedDevices: IGroupedDevices = {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 window.latestLocalMediaConstraints.audio = (streamResult.constraint && streamResult.constraint.audio) as boolean;\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 try {\n streamResult.mediaStream.getTracks().forEach((track: MediaStreamTrack) => {\n track.stop();\n });\n\n trace.info('destroyUserMediaStream - stop all tracks');\n } catch (e) {\n trace.info('destroyUserMediaStream - Failed to stop all tracks', e);\n }\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 try {\n streamResult.mediaStream.getTracks().forEach((track: MediaStreamTrack) => {\n track.stop();\n });\n\n trace.info('destroyDesktopMediaStream - stop all tracks');\n } catch (e) {\n trace.info('destroyDesktopMediaStream - Failed to stop all tracks', e);\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 ((newConstraints.video as MediaTrackConstraints).deviceId as ConstrainDOMStringParameters) = {\n exact: (constraints.video as MediaTrackConstraints).deviceId\n } as any as ConstrainDOMStringParameters;\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 ? {[CameraTypes.BACK]: last(groupedDevices.video), [CameraTypes.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 frameRate: {ideal: 15, max: 30}\n };\n\n const facingMode = cameraType && cameraType === LocalVideoSourceType.CAMERA_FRONT\n ? 'user'\n : 'environment';\n const deviceId = get(this._groupedDevices, `video.${cameraType === LocalVideoSourceType.CAMERA_FRONT\n ? CameraTypes.FRONT\n : CameraTypes.BACK}.deviceId`);\n\n if (deviceId) {\n baseConstraints.video.deviceId = deviceId;\n baseConstraints.video.facingMode = {exact: facingMode};\n } else {\n baseConstraints.video.facingMode = facingMode;\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"]}
|