2023-07-05 13:12:37 +01:00
|
|
|
/*
|
|
|
|
|
Copyright 2023 New Vector Ltd
|
|
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
|
limitations under the License.
|
|
|
|
|
*/
|
|
|
|
|
|
2023-07-12 17:57:54 +01:00
|
|
|
import { GroupCall, IOpenIDToken, MatrixClient } from "matrix-js-sdk";
|
2023-07-05 13:12:37 +01:00
|
|
|
import { logger } from "matrix-js-sdk/src/logger";
|
|
|
|
|
|
2023-07-12 17:57:54 +01:00
|
|
|
import { Config } from "../config/Config";
|
|
|
|
|
|
2023-07-05 13:12:37 +01:00
|
|
|
export interface SFUConfig {
|
|
|
|
|
url: string;
|
|
|
|
|
jwt: string;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// The bits we need from MatrixClient
|
|
|
|
|
export type OpenIDClientParts = Pick<
|
|
|
|
|
MatrixClient,
|
|
|
|
|
"getOpenIdToken" | "getDeviceId"
|
|
|
|
|
>;
|
|
|
|
|
|
|
|
|
|
export async function getSFUConfigWithOpenID(
|
|
|
|
|
client: OpenIDClientParts,
|
2023-07-12 17:57:54 +01:00
|
|
|
groupCall: GroupCall,
|
2023-07-05 13:12:37 +01:00
|
|
|
roomName: string
|
|
|
|
|
): Promise<SFUConfig> {
|
|
|
|
|
const openIdToken = await client.getOpenIdToken();
|
|
|
|
|
logger.debug("Got openID token", openIdToken);
|
|
|
|
|
|
2023-07-12 17:57:54 +01:00
|
|
|
// if the call has a livekit service URL, try it.
|
|
|
|
|
if (groupCall.livekitServiceURL) {
|
|
|
|
|
try {
|
2023-07-14 18:42:31 +01:00
|
|
|
logger.info(
|
|
|
|
|
`Trying to get JWT from call's configured URL of ${groupCall.livekitServiceURL}...`
|
|
|
|
|
);
|
2023-07-12 17:57:54 +01:00
|
|
|
const sfuConfig = await getLiveKitJWT(
|
|
|
|
|
client,
|
|
|
|
|
groupCall.livekitServiceURL,
|
|
|
|
|
roomName,
|
|
|
|
|
openIdToken
|
|
|
|
|
);
|
2023-07-14 18:42:31 +01:00
|
|
|
logger.info(`Got JWT from call state event URL.`);
|
2023-07-12 17:57:54 +01:00
|
|
|
|
|
|
|
|
return sfuConfig;
|
|
|
|
|
} catch (e) {
|
|
|
|
|
logger.warn(
|
|
|
|
|
`Failed to get JWT from group call's configured URL of ${groupCall.livekitServiceURL}.`,
|
|
|
|
|
e
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// otherwise, try our configured one and, if it works, update the call's service URL in the state event
|
|
|
|
|
// NB. This wuill update it for everyone so we may end up with multiple clients updating this when they
|
|
|
|
|
// join at similar times, but we don't have a huge number of options here.
|
|
|
|
|
const urlFromConf = Config.get().livekit!.livekit_service_url;
|
|
|
|
|
logger.info(`Trying livekit service URL from our config: ${urlFromConf}...`);
|
|
|
|
|
try {
|
|
|
|
|
const sfuConfig = await getLiveKitJWT(
|
|
|
|
|
client,
|
|
|
|
|
urlFromConf,
|
|
|
|
|
roomName,
|
|
|
|
|
openIdToken
|
|
|
|
|
);
|
|
|
|
|
|
2023-07-14 18:42:31 +01:00
|
|
|
logger.info(
|
|
|
|
|
`Got JWT, updating call livekit service URL with: ${urlFromConf}...`
|
|
|
|
|
);
|
2023-07-12 17:57:54 +01:00
|
|
|
try {
|
|
|
|
|
await groupCall.updateLivekitServiceURL(urlFromConf);
|
2023-07-14 18:42:31 +01:00
|
|
|
logger.info(`Call livekit service URL updated.`);
|
2023-07-12 17:57:54 +01:00
|
|
|
} catch (e) {
|
|
|
|
|
logger.warn(
|
|
|
|
|
`Failed to update call livekit service URL: continuing anyway.`
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return sfuConfig;
|
|
|
|
|
} catch (e) {
|
|
|
|
|
logger.error("Failed to get JWT from URL defined in Config.", e);
|
|
|
|
|
throw e;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function getLiveKitJWT(
|
|
|
|
|
client: OpenIDClientParts,
|
|
|
|
|
livekitServiceURL: string,
|
|
|
|
|
roomName: string,
|
|
|
|
|
openIDToken: IOpenIDToken
|
|
|
|
|
): Promise<SFUConfig> {
|
|
|
|
|
try {
|
|
|
|
|
const res = await fetch(livekitServiceURL + "/sfu/get", {
|
|
|
|
|
method: "POST",
|
|
|
|
|
headers: {
|
|
|
|
|
"Content-Type": "application/json",
|
|
|
|
|
},
|
|
|
|
|
body: JSON.stringify({
|
|
|
|
|
room: roomName,
|
|
|
|
|
openid_token: openIDToken,
|
|
|
|
|
device_id: client.getDeviceId(),
|
|
|
|
|
}),
|
|
|
|
|
});
|
|
|
|
|
if (!res.ok) {
|
|
|
|
|
throw new Error("SFU Config fetch failed with status code " + res.status);
|
|
|
|
|
}
|
|
|
|
|
return await res.json();
|
|
|
|
|
} catch (e) {
|
|
|
|
|
throw new Error("SFU Config fetch failed with exception " + e);
|
2023-07-05 13:12:37 +01:00
|
|
|
}
|
|
|
|
|
}
|