Use random widget IDs for video rooms (#8739)
This commit is contained in:
parent
abfc66a34e
commit
f152310c08
6 changed files with 15 additions and 17 deletions
|
@ -1035,8 +1035,7 @@ export default class CallHandler extends EventEmitter {
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const userId = client.credentials.userId;
|
await WidgetUtils.addJitsiWidget(roomId, type, 'Jitsi', false);
|
||||||
await WidgetUtils.addJitsiWidget(roomId, type, 'Jitsi', `jitsi_${userId}_${Date.now()}`);
|
|
||||||
logger.log('Jitsi widget added');
|
logger.log('Jitsi widget added');
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
if (e.errcode === 'M_FORBIDDEN') {
|
if (e.errcode === 'M_FORBIDDEN') {
|
||||||
|
|
|
@ -35,16 +35,15 @@ interface IVideoChannelMemberContent {
|
||||||
devices: string[];
|
devices: string[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export const VIDEO_CHANNEL = "io.element.video";
|
|
||||||
export const VIDEO_CHANNEL_MEMBER = "io.element.video.member";
|
export const VIDEO_CHANNEL_MEMBER = "io.element.video.member";
|
||||||
|
|
||||||
export const getVideoChannel = (roomId: string): IApp => {
|
export const getVideoChannel = (roomId: string): IApp => {
|
||||||
const apps = WidgetStore.instance.getApps(roomId);
|
const apps = WidgetStore.instance.getApps(roomId);
|
||||||
return apps.find(app => WidgetType.JITSI.matches(app.type) && app.id === VIDEO_CHANNEL);
|
return apps.find(app => WidgetType.JITSI.matches(app.type) && app.data.isVideoChannel);
|
||||||
};
|
};
|
||||||
|
|
||||||
export const addVideoChannel = async (roomId: string, roomName: string) => {
|
export const addVideoChannel = async (roomId: string, roomName: string) => {
|
||||||
await WidgetUtils.addJitsiWidget(roomId, CallType.Video, "Video channel", VIDEO_CHANNEL, roomName);
|
await WidgetUtils.addJitsiWidget(roomId, CallType.Video, "Video channel", true, roomName);
|
||||||
};
|
};
|
||||||
|
|
||||||
export const getConnectedMembers = (room: Room, connectedLocalEcho: boolean): Set<RoomMember> => {
|
export const getConnectedMembers = (room: Room, connectedLocalEcho: boolean): Set<RoomMember> => {
|
||||||
|
|
|
@ -23,7 +23,7 @@ import { MatrixEvent } from "matrix-js-sdk/src/models/event";
|
||||||
import { logger } from "matrix-js-sdk/src/logger";
|
import { logger } from "matrix-js-sdk/src/logger";
|
||||||
import { ClientEvent, RoomStateEvent } from "matrix-js-sdk/src/matrix";
|
import { ClientEvent, RoomStateEvent } from "matrix-js-sdk/src/matrix";
|
||||||
import { CallType } from "matrix-js-sdk/src/webrtc/call";
|
import { CallType } from "matrix-js-sdk/src/webrtc/call";
|
||||||
import { randomLowercaseString, randomUppercaseString } from "matrix-js-sdk/src/randomstring";
|
import { randomString, randomLowercaseString, randomUppercaseString } from "matrix-js-sdk/src/randomstring";
|
||||||
|
|
||||||
import { MatrixClientPeg } from '../MatrixClientPeg';
|
import { MatrixClientPeg } from '../MatrixClientPeg';
|
||||||
import SdkConfig from "../SdkConfig";
|
import SdkConfig from "../SdkConfig";
|
||||||
|
@ -35,7 +35,6 @@ import { Jitsi } from "../widgets/Jitsi";
|
||||||
import { objectClone } from "./objects";
|
import { objectClone } from "./objects";
|
||||||
import { _t } from "../languageHandler";
|
import { _t } from "../languageHandler";
|
||||||
import { IApp } from "../stores/WidgetStore";
|
import { IApp } from "../stores/WidgetStore";
|
||||||
import { VIDEO_CHANNEL } from "./VideoChannelUtils";
|
|
||||||
|
|
||||||
// How long we wait for the state event echo to come back from the server
|
// How long we wait for the state event echo to come back from the server
|
||||||
// before waitFor[Room/User]Widget rejects its promise
|
// before waitFor[Room/User]Widget rejects its promise
|
||||||
|
@ -444,11 +443,12 @@ export default class WidgetUtils {
|
||||||
roomId: string,
|
roomId: string,
|
||||||
type: CallType,
|
type: CallType,
|
||||||
name: string,
|
name: string,
|
||||||
widgetId: string,
|
isVideoChannel: boolean,
|
||||||
oobRoomName?: string,
|
oobRoomName?: string,
|
||||||
): Promise<void> {
|
): Promise<void> {
|
||||||
const domain = Jitsi.getInstance().preferredDomain;
|
const domain = Jitsi.getInstance().preferredDomain;
|
||||||
const auth = await Jitsi.getInstance().getJitsiAuth();
|
const auth = await Jitsi.getInstance().getJitsiAuth();
|
||||||
|
const widgetId = randomString(24); // Must be globally unique
|
||||||
|
|
||||||
let confId;
|
let confId;
|
||||||
if (auth === 'openidtoken-jwt') {
|
if (auth === 'openidtoken-jwt') {
|
||||||
|
@ -471,7 +471,7 @@ export default class WidgetUtils {
|
||||||
conferenceId: confId,
|
conferenceId: confId,
|
||||||
roomName: oobRoomName ?? MatrixClientPeg.get().getRoom(roomId)?.name,
|
roomName: oobRoomName ?? MatrixClientPeg.get().getRoom(roomId)?.name,
|
||||||
isAudioOnly: type === CallType.Voice,
|
isAudioOnly: type === CallType.Voice,
|
||||||
isVideoChannel: widgetId === VIDEO_CHANNEL,
|
isVideoChannel,
|
||||||
domain,
|
domain,
|
||||||
auth,
|
auth,
|
||||||
});
|
});
|
||||||
|
|
|
@ -32,7 +32,7 @@ import {
|
||||||
mkVideoChannelMember,
|
mkVideoChannelMember,
|
||||||
} from "../../test-utils";
|
} from "../../test-utils";
|
||||||
import { MatrixClientPeg } from "../../../src/MatrixClientPeg";
|
import { MatrixClientPeg } from "../../../src/MatrixClientPeg";
|
||||||
import { VIDEO_CHANNEL, VIDEO_CHANNEL_MEMBER } from "../../../src/utils/VideoChannelUtils";
|
import { VIDEO_CHANNEL_MEMBER } from "../../../src/utils/VideoChannelUtils";
|
||||||
import WidgetStore from "../../../src/stores/WidgetStore";
|
import WidgetStore from "../../../src/stores/WidgetStore";
|
||||||
import _VideoRoomView from "../../../src/components/structures/VideoRoomView";
|
import _VideoRoomView from "../../../src/components/structures/VideoRoomView";
|
||||||
import VideoLobby from "../../../src/components/views/voip/VideoLobby";
|
import VideoLobby from "../../../src/components/views/voip/VideoLobby";
|
||||||
|
@ -42,7 +42,7 @@ const VideoRoomView = wrapInMatrixClientContext(_VideoRoomView);
|
||||||
|
|
||||||
describe("VideoRoomView", () => {
|
describe("VideoRoomView", () => {
|
||||||
jest.spyOn(WidgetStore.instance, "getApps").mockReturnValue([{
|
jest.spyOn(WidgetStore.instance, "getApps").mockReturnValue([{
|
||||||
id: VIDEO_CHANNEL,
|
id: "1",
|
||||||
eventId: "$1:example.org",
|
eventId: "$1:example.org",
|
||||||
roomId: "!1:example.org",
|
roomId: "!1:example.org",
|
||||||
type: MatrixWidgetType.JitsiMeet,
|
type: MatrixWidgetType.JitsiMeet,
|
||||||
|
@ -50,6 +50,7 @@ describe("VideoRoomView", () => {
|
||||||
name: "Video channel",
|
name: "Video channel",
|
||||||
creatorUserId: "@alice:example.org",
|
creatorUserId: "@alice:example.org",
|
||||||
avatar_url: null,
|
avatar_url: null,
|
||||||
|
data: { isVideoChannel: true },
|
||||||
}]);
|
}]);
|
||||||
Object.defineProperty(navigator, "mediaDevices", {
|
Object.defineProperty(navigator, "mediaDevices", {
|
||||||
value: { enumerateDevices: () => [] },
|
value: { enumerateDevices: () => [] },
|
||||||
|
|
|
@ -23,7 +23,7 @@ import { stubClient, setupAsyncStoreWithClient } from "./test-utils";
|
||||||
import { MatrixClientPeg } from "../src/MatrixClientPeg";
|
import { MatrixClientPeg } from "../src/MatrixClientPeg";
|
||||||
import WidgetStore from "../src/stores/WidgetStore";
|
import WidgetStore from "../src/stores/WidgetStore";
|
||||||
import WidgetUtils from "../src/utils/WidgetUtils";
|
import WidgetUtils from "../src/utils/WidgetUtils";
|
||||||
import { VIDEO_CHANNEL, VIDEO_CHANNEL_MEMBER } from "../src/utils/VideoChannelUtils";
|
import { VIDEO_CHANNEL_MEMBER } from "../src/utils/VideoChannelUtils";
|
||||||
import createRoom, { canEncryptToAllUsers } from '../src/createRoom';
|
import createRoom, { canEncryptToAllUsers } from '../src/createRoom';
|
||||||
|
|
||||||
describe("createRoom", () => {
|
describe("createRoom", () => {
|
||||||
|
@ -43,12 +43,11 @@ describe("createRoom", () => {
|
||||||
events: { [VIDEO_CHANNEL_MEMBER]: videoMemberPower },
|
events: { [VIDEO_CHANNEL_MEMBER]: videoMemberPower },
|
||||||
},
|
},
|
||||||
}]] = mocked(client.createRoom).mock.calls as any; // no good type
|
}]] = mocked(client.createRoom).mock.calls as any; // no good type
|
||||||
const [[widgetRoomId, widgetStateKey, , widgetId]] = mocked(client.sendStateEvent).mock.calls;
|
const [[widgetRoomId, widgetStateKey]] = mocked(client.sendStateEvent).mock.calls;
|
||||||
|
|
||||||
// We should have set up the Jitsi widget
|
// We should have set up the Jitsi widget
|
||||||
expect(widgetRoomId).toEqual(roomId);
|
expect(widgetRoomId).toEqual(roomId);
|
||||||
expect(widgetStateKey).toEqual("im.vector.modular.widgets");
|
expect(widgetStateKey).toEqual("im.vector.modular.widgets");
|
||||||
expect(widgetId).toEqual(VIDEO_CHANNEL);
|
|
||||||
|
|
||||||
// All members should be able to update their connected devices
|
// All members should be able to update their connected devices
|
||||||
expect(videoMemberPower).toEqual(0);
|
expect(videoMemberPower).toEqual(0);
|
||||||
|
|
|
@ -23,14 +23,13 @@ import WidgetStore, { IApp } from "../../src/stores/WidgetStore";
|
||||||
import { WidgetMessagingStore } from "../../src/stores/widgets/WidgetMessagingStore";
|
import { WidgetMessagingStore } from "../../src/stores/widgets/WidgetMessagingStore";
|
||||||
import { ElementWidgetActions } from "../../src/stores/widgets/ElementWidgetActions";
|
import { ElementWidgetActions } from "../../src/stores/widgets/ElementWidgetActions";
|
||||||
import VideoChannelStore, { VideoChannelEvent } from "../../src/stores/VideoChannelStore";
|
import VideoChannelStore, { VideoChannelEvent } from "../../src/stores/VideoChannelStore";
|
||||||
import { VIDEO_CHANNEL } from "../../src/utils/VideoChannelUtils";
|
|
||||||
|
|
||||||
describe("VideoChannelStore", () => {
|
describe("VideoChannelStore", () => {
|
||||||
const store = VideoChannelStore.instance;
|
const store = VideoChannelStore.instance;
|
||||||
|
|
||||||
const widget = { id: VIDEO_CHANNEL } as unknown as Widget;
|
const widget = { id: "1" } as unknown as Widget;
|
||||||
const app = {
|
const app = {
|
||||||
id: VIDEO_CHANNEL,
|
id: "1",
|
||||||
eventId: "$1:example.org",
|
eventId: "$1:example.org",
|
||||||
roomId: "!1:example.org",
|
roomId: "!1:example.org",
|
||||||
type: MatrixWidgetType.JitsiMeet,
|
type: MatrixWidgetType.JitsiMeet,
|
||||||
|
@ -38,6 +37,7 @@ describe("VideoChannelStore", () => {
|
||||||
name: "Video channel",
|
name: "Video channel",
|
||||||
creatorUserId: "@alice:example.org",
|
creatorUserId: "@alice:example.org",
|
||||||
avatar_url: null,
|
avatar_url: null,
|
||||||
|
data: { isVideoChannel: true },
|
||||||
} as IApp;
|
} as IApp;
|
||||||
|
|
||||||
// Set up mocks to simulate the remote end of the widget API
|
// Set up mocks to simulate the remote end of the widget API
|
||||||
|
|
Loading…
Reference in a new issue