86a715dc15
Allows configuring, via bbb-html5, which media server adapter will be used by screen sharing; server wide
162 lines
5.5 KiB
JavaScript
162 lines
5.5 KiB
JavaScript
import Meetings from '/imports/api/meetings';
|
|
import logger from '/imports/startup/client/logger';
|
|
import { fetchWebRTCMappedStunTurnServers, getMappedFallbackStun } from '/imports/utils/fetchStunTurnServers';
|
|
import loadAndPlayMediaStream from '/imports/ui/services/bbb-webrtc-sfu/load-play';
|
|
import { SCREENSHARING_ERRORS } from './errors';
|
|
|
|
const {
|
|
constraints: GDM_CONSTRAINTS,
|
|
mediaTimeouts: MEDIA_TIMEOUTS,
|
|
bitrate: BASE_BITRATE,
|
|
mediaServer: DEFAULT_SCREENSHARE_MEDIA_SERVER,
|
|
} = Meteor.settings.public.kurento.screenshare;
|
|
const {
|
|
baseTimeout: BASE_MEDIA_TIMEOUT,
|
|
maxTimeout: MAX_MEDIA_TIMEOUT,
|
|
maxConnectionAttempts: MAX_CONN_ATTEMPTS,
|
|
timeoutIncreaseFactor: TIMEOUT_INCREASE_FACTOR,
|
|
} = MEDIA_TIMEOUTS;
|
|
|
|
const HAS_DISPLAY_MEDIA = (typeof navigator.getDisplayMedia === 'function'
|
|
|| (navigator.mediaDevices && typeof navigator.mediaDevices.getDisplayMedia === 'function'));
|
|
|
|
const getConferenceBridge = () => Meetings.findOne().voiceProp.voiceConf;
|
|
|
|
const normalizeGetDisplayMediaError = (error) => {
|
|
return SCREENSHARING_ERRORS[error.name] || SCREENSHARING_ERRORS.GetDisplayMediaGenericError;
|
|
};
|
|
|
|
const getBoundGDM = () => {
|
|
if (typeof navigator.getDisplayMedia === 'function') {
|
|
return navigator.getDisplayMedia.bind(navigator);
|
|
} else if (navigator.mediaDevices && typeof navigator.mediaDevices.getDisplayMedia === 'function') {
|
|
return navigator.mediaDevices.getDisplayMedia.bind(navigator.mediaDevices);
|
|
}
|
|
}
|
|
|
|
const getScreenStream = async () => {
|
|
const gDMCallback = (stream) => {
|
|
// Some older Chromium variants choke on gDM when audio: true by NOT generating
|
|
// a promise rejection AND not generating a valid input screen stream, need to
|
|
// work around that manually for now - prlanzarin
|
|
if (stream == null) {
|
|
return Promise.reject(SCREENSHARING_ERRORS.NotSupportedError);
|
|
}
|
|
|
|
if (typeof stream.getVideoTracks === 'function'
|
|
&& typeof GDM_CONSTRAINTS.video === 'object') {
|
|
stream.getVideoTracks().forEach(track => {
|
|
if (typeof track.applyConstraints === 'function') {
|
|
track.applyConstraints(GDM_CONSTRAINTS.video).catch(error => {
|
|
logger.warn({
|
|
logCode: 'screenshare_videoconstraint_failed',
|
|
extraInfo: { errorName: error.name, errorCode: error.code },
|
|
},
|
|
'Error applying screenshare video constraint');
|
|
});
|
|
}
|
|
});
|
|
}
|
|
|
|
if (typeof stream.getAudioTracks === 'function'
|
|
&& typeof GDM_CONSTRAINTS.audio === 'object') {
|
|
stream.getAudioTracks().forEach(track => {
|
|
if (typeof track.applyConstraints === 'function') {
|
|
track.applyConstraints(GDM_CONSTRAINTS.audio).catch(error => {
|
|
logger.warn({
|
|
logCode: 'screenshare_audioconstraint_failed',
|
|
extraInfo: { errorName: error.name, errorCode: error.code },
|
|
}, 'Error applying screenshare audio constraint');
|
|
});
|
|
}
|
|
});
|
|
}
|
|
|
|
return Promise.resolve(stream);
|
|
};
|
|
|
|
const getDisplayMedia = getBoundGDM();
|
|
|
|
if (typeof getDisplayMedia === 'function') {
|
|
return getDisplayMedia(GDM_CONSTRAINTS)
|
|
.then(gDMCallback)
|
|
.catch(error => {
|
|
const normalizedError = normalizeGetDisplayMediaError(error);
|
|
logger.error({
|
|
logCode: 'screenshare_getdisplaymedia_failed',
|
|
extraInfo: { errorCode: normalizedError.errorCode, errorMessage: normalizedError.errorMessage },
|
|
}, 'getDisplayMedia call failed');
|
|
return Promise.reject(normalizedError);
|
|
});
|
|
} else {
|
|
// getDisplayMedia isn't supported, error its way out
|
|
return Promise.reject(SCREENSHARING_ERRORS.NotSupportedError);
|
|
}
|
|
};
|
|
|
|
const getIceServers = (sessionToken) => {
|
|
return fetchWebRTCMappedStunTurnServers(sessionToken).catch(error => {
|
|
logger.error({
|
|
logCode: 'screenshare_fetchstunturninfo_error',
|
|
extraInfo: { error }
|
|
}, 'Screenshare bridge failed to fetch STUN/TURN info');
|
|
return getMappedFallbackStun();
|
|
});
|
|
}
|
|
|
|
const getMediaServerAdapter = () => {
|
|
return DEFAULT_SCREENSHARE_MEDIA_SERVER;
|
|
}
|
|
|
|
const getNextReconnectionInterval = (oldInterval) => {
|
|
return Math.min(
|
|
TIMEOUT_INCREASE_FACTOR * oldInterval,
|
|
MAX_MEDIA_TIMEOUT,
|
|
);
|
|
}
|
|
|
|
const streamHasAudioTrack = (stream) => {
|
|
return stream
|
|
&& typeof stream.getAudioTracks === 'function'
|
|
&& stream.getAudioTracks().length >= 1;
|
|
}
|
|
|
|
const dispatchAutoplayHandlingEvent = (mediaElement) => {
|
|
const tagFailedEvent = new CustomEvent('screensharePlayFailed',
|
|
{ detail: { mediaElement } });
|
|
window.dispatchEvent(tagFailedEvent);
|
|
}
|
|
|
|
const screenshareLoadAndPlayMediaStream = (stream, mediaElement, muted) => {
|
|
return loadAndPlayMediaStream(stream, mediaElement, muted).catch(error => {
|
|
// NotAllowedError equals autoplay issues, fire autoplay handling event.
|
|
// This will be handled in the screenshare react component.
|
|
if (error.name === 'NotAllowedError') {
|
|
logger.error({
|
|
logCode: 'screenshare_error_autoplay',
|
|
extraInfo: { errorName: error.name },
|
|
}, 'Screen share media play failed: autoplay error');
|
|
dispatchAutoplayHandlingEvent(mediaElement);
|
|
} else {
|
|
throw {
|
|
errorCode: SCREENSHARING_ERRORS.SCREENSHARE_PLAY_FAILED.errorCode,
|
|
errorMessage: error.message || SCREENSHARING_ERRORS.SCREENSHARE_PLAY_FAILED.errorMessage,
|
|
};
|
|
}
|
|
});
|
|
}
|
|
|
|
export default {
|
|
HAS_DISPLAY_MEDIA,
|
|
getConferenceBridge,
|
|
getScreenStream,
|
|
getIceServers,
|
|
getNextReconnectionInterval,
|
|
streamHasAudioTrack,
|
|
screenshareLoadAndPlayMediaStream,
|
|
getMediaServerAdapter,
|
|
BASE_MEDIA_TIMEOUT,
|
|
MAX_CONN_ATTEMPTS,
|
|
BASE_BITRATE,
|
|
};
|