2018-06-14 09:15:36 +00:00
|
|
|
// @flow
|
|
|
|
|
2021-06-21 08:36:18 +00:00
|
|
|
import { getMeetingRegion, getRecordingSharingUrl } from '../base/config';
|
2018-09-10 15:39:33 +00:00
|
|
|
import JitsiMeetJS, { JitsiRecordingConstants } from '../base/lib-jitsi-meet';
|
2021-06-21 08:36:18 +00:00
|
|
|
import { getLocalParticipant, getParticipantDisplayName } from '../base/participants';
|
|
|
|
import { copyText } from '../base/util/helpers';
|
2021-07-20 08:58:42 +00:00
|
|
|
import { getVpaasTenant, isVpaasMeeting } from '../jaas/functions';
|
2018-06-14 09:15:36 +00:00
|
|
|
import {
|
2019-03-29 13:37:50 +00:00
|
|
|
NOTIFICATION_TIMEOUT,
|
2018-06-14 09:15:36 +00:00
|
|
|
hideNotification,
|
|
|
|
showErrorNotification,
|
|
|
|
showNotification
|
|
|
|
} from '../notifications';
|
|
|
|
|
|
|
|
import {
|
|
|
|
CLEAR_RECORDING_SESSIONS,
|
|
|
|
RECORDING_SESSION_UPDATED,
|
2018-07-05 11:17:45 +00:00
|
|
|
SET_PENDING_RECORDING_NOTIFICATION_UID,
|
2021-06-23 08:57:11 +00:00
|
|
|
SET_SELECTED_RECORDING_SERVICE,
|
2018-07-05 11:17:45 +00:00
|
|
|
SET_STREAM_KEY
|
2018-06-14 09:15:36 +00:00
|
|
|
} from './actionTypes';
|
2021-06-23 08:57:11 +00:00
|
|
|
import { getRecordingLink, getResourceId, isSavingRecordingOnDropbox } from './functions';
|
2021-06-21 08:36:18 +00:00
|
|
|
import logger from './logger';
|
2018-06-14 09:15:36 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Clears the data of every recording sessions.
|
|
|
|
*
|
|
|
|
* @returns {{
|
|
|
|
* type: CLEAR_RECORDING_SESSIONS
|
|
|
|
* }}
|
|
|
|
*/
|
|
|
|
export function clearRecordingSessions() {
|
|
|
|
return {
|
|
|
|
type: CLEAR_RECORDING_SESSIONS
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Signals that the pending recording notification should be removed from the
|
|
|
|
* screen.
|
|
|
|
*
|
2018-11-08 12:25:02 +00:00
|
|
|
* @param {string} streamType - The type of the stream ({@code 'file'} or
|
|
|
|
* {@code 'stream'}).
|
2018-06-14 09:15:36 +00:00
|
|
|
* @returns {Function}
|
|
|
|
*/
|
2018-07-05 11:17:45 +00:00
|
|
|
export function hidePendingRecordingNotification(streamType: string) {
|
2018-06-14 09:15:36 +00:00
|
|
|
return (dispatch: Function, getState: Function) => {
|
2018-07-05 11:17:45 +00:00
|
|
|
const { pendingNotificationUids } = getState()['features/recording'];
|
|
|
|
const pendingNotificationUid = pendingNotificationUids[streamType];
|
2018-06-14 09:15:36 +00:00
|
|
|
|
|
|
|
if (pendingNotificationUid) {
|
|
|
|
dispatch(hideNotification(pendingNotificationUid));
|
2018-07-05 11:17:45 +00:00
|
|
|
dispatch(
|
|
|
|
_setPendingRecordingNotificationUid(
|
|
|
|
undefined, streamType));
|
2018-06-14 09:15:36 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2018-07-05 11:17:45 +00:00
|
|
|
* Sets the stream key last used by the user for later reuse.
|
2018-06-14 09:15:36 +00:00
|
|
|
*
|
2018-07-05 11:17:45 +00:00
|
|
|
* @param {string} streamKey - The stream key to set.
|
2018-06-14 09:15:36 +00:00
|
|
|
* @returns {{
|
2018-07-05 11:17:45 +00:00
|
|
|
* type: SET_STREAM_KEY,
|
|
|
|
* streamKey: string
|
2018-06-14 09:15:36 +00:00
|
|
|
* }}
|
|
|
|
*/
|
2018-07-05 11:17:45 +00:00
|
|
|
export function setLiveStreamKey(streamKey: string) {
|
2018-06-14 09:15:36 +00:00
|
|
|
return {
|
2018-07-05 11:17:45 +00:00
|
|
|
type: SET_STREAM_KEY,
|
|
|
|
streamKey
|
2018-06-14 09:15:36 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Signals that the pending recording notification should be shown on the
|
|
|
|
* screen.
|
|
|
|
*
|
2018-11-08 12:25:02 +00:00
|
|
|
* @param {string} streamType - The type of the stream ({@code file} or
|
|
|
|
* {@code stream}).
|
2018-06-14 09:15:36 +00:00
|
|
|
* @returns {Function}
|
|
|
|
*/
|
2018-07-05 11:17:45 +00:00
|
|
|
export function showPendingRecordingNotification(streamType: string) {
|
2021-01-11 10:52:52 +00:00
|
|
|
return async (dispatch: Function) => {
|
2018-07-05 11:17:45 +00:00
|
|
|
const isLiveStreaming
|
|
|
|
= streamType === JitsiMeetJS.constants.recording.mode.STREAM;
|
|
|
|
const dialogProps = isLiveStreaming ? {
|
|
|
|
descriptionKey: 'liveStreaming.pending',
|
|
|
|
titleKey: 'dialog.liveStreaming'
|
|
|
|
} : {
|
2018-06-14 09:15:36 +00:00
|
|
|
descriptionKey: 'recording.pending',
|
|
|
|
titleKey: 'dialog.recording'
|
2018-07-05 11:17:45 +00:00
|
|
|
};
|
2021-01-11 10:52:52 +00:00
|
|
|
const notification = await dispatch(showNotification({
|
2018-07-05 11:17:45 +00:00
|
|
|
isDismissAllowed: false,
|
|
|
|
...dialogProps
|
2021-01-11 10:52:52 +00:00
|
|
|
}));
|
2018-06-14 09:15:36 +00:00
|
|
|
|
2021-01-11 10:52:52 +00:00
|
|
|
if (notification) {
|
|
|
|
dispatch(_setPendingRecordingNotificationUid(notification.uid, streamType));
|
|
|
|
}
|
2018-06-14 09:15:36 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Signals that the recording error notification should be shown.
|
|
|
|
*
|
|
|
|
* @param {Object} props - The Props needed to render the notification.
|
|
|
|
* @returns {showErrorNotification}
|
|
|
|
*/
|
|
|
|
export function showRecordingError(props: Object) {
|
|
|
|
return showErrorNotification(props);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Signals that the stopped recording notification should be shown on the
|
|
|
|
* screen for a given period.
|
|
|
|
*
|
2018-11-08 12:25:02 +00:00
|
|
|
* @param {string} streamType - The type of the stream ({@code file} or
|
|
|
|
* {@code stream}).
|
2019-10-03 19:35:21 +00:00
|
|
|
* @param {string?} participantName - The participant name stopping the recording.
|
2018-06-14 09:15:36 +00:00
|
|
|
* @returns {showNotification}
|
|
|
|
*/
|
2019-10-03 19:35:21 +00:00
|
|
|
export function showStoppedRecordingNotification(streamType: string, participantName?: string) {
|
2018-07-05 11:17:45 +00:00
|
|
|
const isLiveStreaming
|
|
|
|
= streamType === JitsiMeetJS.constants.recording.mode.STREAM;
|
2019-10-03 19:35:21 +00:00
|
|
|
const descriptionArguments = { name: participantName };
|
2018-07-05 11:17:45 +00:00
|
|
|
const dialogProps = isLiveStreaming ? {
|
2019-10-03 19:35:21 +00:00
|
|
|
descriptionKey: participantName ? 'liveStreaming.offBy' : 'liveStreaming.off',
|
|
|
|
descriptionArguments,
|
2018-07-05 11:17:45 +00:00
|
|
|
titleKey: 'dialog.liveStreaming'
|
|
|
|
} : {
|
2019-10-03 19:35:21 +00:00
|
|
|
descriptionKey: participantName ? 'recording.offBy' : 'recording.off',
|
|
|
|
descriptionArguments,
|
|
|
|
titleKey: 'dialog.recording'
|
|
|
|
};
|
|
|
|
|
|
|
|
return showNotification(dialogProps, NOTIFICATION_TIMEOUT);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Signals that a started recording notification should be shown on the
|
|
|
|
* screen for a given period.
|
|
|
|
*
|
2021-06-21 08:36:18 +00:00
|
|
|
* @param {string} mode - The type of the recording: Stream of File.
|
|
|
|
* @param {string | Object } initiator - The participant who started recording.
|
|
|
|
* @param {string} sessionId - The recording session id.
|
|
|
|
* @returns {Function}
|
2019-10-03 19:35:21 +00:00
|
|
|
*/
|
2021-06-21 08:36:18 +00:00
|
|
|
export function showStartedRecordingNotification(
|
|
|
|
mode: string,
|
|
|
|
initiator: Object | string,
|
|
|
|
sessionId: string) {
|
|
|
|
return async (dispatch: Function, getState: Function) => {
|
|
|
|
const state = getState();
|
|
|
|
const initiatorId = getResourceId(initiator);
|
|
|
|
const participantName = getParticipantDisplayName(state, initiatorId);
|
|
|
|
let dialogProps = {
|
|
|
|
customActionNameKey: undefined,
|
|
|
|
descriptionKey: participantName ? 'liveStreaming.onBy' : 'liveStreaming.on',
|
|
|
|
descriptionArguments: { name: participantName },
|
|
|
|
isDismissAllowed: true,
|
|
|
|
titleKey: 'dialog.liveStreaming'
|
|
|
|
};
|
2018-07-05 11:17:45 +00:00
|
|
|
|
2021-06-21 08:36:18 +00:00
|
|
|
if (mode !== JitsiMeetJS.constants.recording.mode.STREAM) {
|
|
|
|
const recordingSharingUrl = getRecordingSharingUrl(state);
|
|
|
|
const iAmRecordingInitiator = getLocalParticipant(state).id === initiatorId;
|
|
|
|
|
|
|
|
dialogProps = {
|
|
|
|
customActionHandler: undefined,
|
|
|
|
customActionNameKey: undefined,
|
|
|
|
descriptionKey: participantName ? 'recording.onBy' : 'recording.on',
|
|
|
|
descriptionArguments: { name: participantName },
|
|
|
|
isDismissAllowed: true,
|
|
|
|
titleKey: 'dialog.recording'
|
|
|
|
};
|
|
|
|
|
|
|
|
// fetch the recording link from the server for recording initiators in jaas meetings
|
|
|
|
if (recordingSharingUrl
|
|
|
|
&& isVpaasMeeting(state)
|
2021-06-23 08:57:11 +00:00
|
|
|
&& iAmRecordingInitiator
|
|
|
|
&& !isSavingRecordingOnDropbox(state)) {
|
2021-06-21 08:36:18 +00:00
|
|
|
const region = getMeetingRegion(state);
|
|
|
|
const tenant = getVpaasTenant(state);
|
|
|
|
|
|
|
|
try {
|
|
|
|
const link = await getRecordingLink(recordingSharingUrl, sessionId, region, tenant);
|
|
|
|
|
|
|
|
// add the option to copy recording link
|
|
|
|
dialogProps.customActionNameKey = 'recording.copyLink';
|
|
|
|
dialogProps.customActionHandler = () => copyText(link);
|
|
|
|
dialogProps.titleKey = 'recording.on';
|
|
|
|
dialogProps.descriptionKey = 'recording.linkGenerated';
|
|
|
|
dialogProps.isDismissAllowed = false;
|
|
|
|
} catch (err) {
|
|
|
|
dispatch(showErrorNotification({
|
|
|
|
titleKey: 'recording.errorFetchingLink'
|
|
|
|
}));
|
|
|
|
|
|
|
|
return logger.error('Could not fetch recording link', err);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
dispatch(showNotification(dialogProps));
|
|
|
|
};
|
2018-06-14 09:15:36 +00:00
|
|
|
}
|
2017-08-25 16:45:30 +00:00
|
|
|
|
|
|
|
/**
|
2018-05-16 14:00:16 +00:00
|
|
|
* Updates the known state for a given recording session.
|
2017-08-25 16:45:30 +00:00
|
|
|
*
|
2018-05-16 14:00:16 +00:00
|
|
|
* @param {Object} session - The new state to merge with the existing state in
|
|
|
|
* redux.
|
2017-08-25 16:45:30 +00:00
|
|
|
* @returns {{
|
2018-05-16 14:00:16 +00:00
|
|
|
* type: RECORDING_SESSION_UPDATED,
|
|
|
|
* sessionData: Object
|
2017-08-25 16:45:30 +00:00
|
|
|
* }}
|
|
|
|
*/
|
2018-06-14 09:15:36 +00:00
|
|
|
export function updateRecordingSessionData(session: Object) {
|
2018-09-10 15:39:33 +00:00
|
|
|
const status = session.getStatus();
|
|
|
|
const timestamp
|
|
|
|
= status === JitsiRecordingConstants.status.ON
|
|
|
|
? Date.now() / 1000
|
|
|
|
: undefined;
|
|
|
|
|
2017-08-25 16:45:30 +00:00
|
|
|
return {
|
2018-05-16 14:00:16 +00:00
|
|
|
type: RECORDING_SESSION_UPDATED,
|
|
|
|
sessionData: {
|
|
|
|
error: session.getError(),
|
|
|
|
id: session.getID(),
|
2019-10-03 19:35:21 +00:00
|
|
|
initiator: session.getInitiator(),
|
2018-05-16 14:00:16 +00:00
|
|
|
liveStreamViewURL: session.getLiveStreamViewURL(),
|
|
|
|
mode: session.getMode(),
|
2018-09-10 15:39:33 +00:00
|
|
|
status,
|
2019-10-03 19:35:21 +00:00
|
|
|
terminator: session.getTerminator(),
|
2018-09-10 15:39:33 +00:00
|
|
|
timestamp
|
2018-05-16 14:00:16 +00:00
|
|
|
}
|
2017-08-25 16:45:30 +00:00
|
|
|
};
|
|
|
|
}
|
2018-07-05 11:17:45 +00:00
|
|
|
|
2021-06-23 08:57:11 +00:00
|
|
|
/**
|
|
|
|
* Sets the selected recording service.
|
|
|
|
*
|
|
|
|
* @param {string} selectedRecordingService - The new selected recording service.
|
|
|
|
* @returns {Object}
|
|
|
|
*/
|
|
|
|
export function setSelectedRecordingService(selectedRecordingService: string) {
|
|
|
|
return {
|
|
|
|
type: SET_SELECTED_RECORDING_SERVICE,
|
|
|
|
selectedRecordingService
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2018-07-05 11:17:45 +00:00
|
|
|
/**
|
|
|
|
* Sets UID of the the pending streaming notification to use it when hinding
|
|
|
|
* the notification is necessary, or unsets it when undefined (or no param) is
|
|
|
|
* passed.
|
|
|
|
*
|
|
|
|
* @param {?number} uid - The UID of the notification.
|
2018-11-08 12:25:02 +00:00
|
|
|
* @param {string} streamType - The type of the stream ({@code file} or
|
|
|
|
* {@code stream}).
|
2018-07-05 11:17:45 +00:00
|
|
|
* @returns {{
|
|
|
|
* type: SET_PENDING_RECORDING_NOTIFICATION_UID,
|
|
|
|
* streamType: string,
|
|
|
|
* uid: number
|
|
|
|
* }}
|
|
|
|
*/
|
|
|
|
function _setPendingRecordingNotificationUid(uid: ?number, streamType: string) {
|
|
|
|
return {
|
|
|
|
type: SET_PENDING_RECORDING_NOTIFICATION_UID,
|
|
|
|
streamType,
|
|
|
|
uid
|
|
|
|
};
|
|
|
|
}
|