989ab5a432
* Add useDeviceList hook Signed-off-by: Ajay Bura <ajbura@gmail.com> * Add isCrossVerified func to matrixUtil Signed-off-by: Ajay Bura <ajbura@gmail.com> * Add className prop in sidebar avatar comp Signed-off-by: Ajay Bura <ajbura@gmail.com> * Add unverified session indicator in sidebar Signed-off-by: Ajay Bura <ajbura@gmail.com> * Add info card component Signed-off-by: Ajay Bura <ajbura@gmail.com> * Add css variables Signed-off-by: Ajay Bura <ajbura@gmail.com> * Add cross signin status hook Signed-off-by: Ajay Bura <ajbura@gmail.com> * Add hasCrossSigninAccountData function Signed-off-by: Ajay Bura <ajbura@gmail.com> * Add cross signin info card in device manage component Signed-off-by: Ajay Bura <ajbura@gmail.com> * Add cross signing and key backup component Signed-off-by: Ajay Bura <ajbura@gmail.com> * Fix typo Signed-off-by: Ajay Bura <ajbura@gmail.com> * WIP Signed-off-by: Ajay Bura <ajbura@gmail.com> * Add cross singing dialogs Signed-off-by: Ajay Bura <ajbura@gmail.com> * Add cross signing set/reset Signed-off-by: Ajay Bura <ajbura@gmail.com> * Add SecretStorageAccess component Signed-off-by: Ajay Bura <ajbura@gmail.com> * Add key backup Signed-off-by: Ajay Bura <ajbura@gmail.com> * WIP * WIP * WIP * WIP * Show progress when restoring key backup * Add SSSS and key backup
200 lines
5.6 KiB
JavaScript
200 lines
5.6 KiB
JavaScript
import initMatrix from '../client/initMatrix';
|
|
|
|
import HashIC from '../../public/res/ic/outlined/hash.svg';
|
|
import HashGlobeIC from '../../public/res/ic/outlined/hash-globe.svg';
|
|
import HashLockIC from '../../public/res/ic/outlined/hash-lock.svg';
|
|
import SpaceIC from '../../public/res/ic/outlined/space.svg';
|
|
import SpaceGlobeIC from '../../public/res/ic/outlined/space-globe.svg';
|
|
import SpaceLockIC from '../../public/res/ic/outlined/space-lock.svg';
|
|
|
|
const WELL_KNOWN_URI = '/.well-known/matrix/client';
|
|
|
|
export async function getBaseUrl(servername) {
|
|
let protocol = 'https://';
|
|
if (servername.match(/^https?:\/\//) !== null) protocol = '';
|
|
const serverDiscoveryUrl = `${protocol}${servername}${WELL_KNOWN_URI}`;
|
|
try {
|
|
const result = await (await fetch(serverDiscoveryUrl, { method: 'GET' })).json();
|
|
|
|
const baseUrl = result?.['m.homeserver']?.base_url;
|
|
if (baseUrl === undefined) throw new Error();
|
|
return baseUrl;
|
|
} catch (e) {
|
|
throw new Error(`${protocol}${servername}`);
|
|
}
|
|
}
|
|
|
|
export function getUsername(userId) {
|
|
const mx = initMatrix.matrixClient;
|
|
const user = mx.getUser(userId);
|
|
if (user === null) return userId;
|
|
let username = user.displayName;
|
|
if (typeof username === 'undefined') {
|
|
username = userId;
|
|
}
|
|
return username;
|
|
}
|
|
|
|
export function getUsernameOfRoomMember(roomMember) {
|
|
return roomMember.name || roomMember.userId;
|
|
}
|
|
|
|
export async function isRoomAliasAvailable(alias) {
|
|
try {
|
|
const result = await initMatrix.matrixClient.resolveRoomAlias(alias);
|
|
if (result.room_id) return false;
|
|
return false;
|
|
} catch (e) {
|
|
if (e.errcode === 'M_NOT_FOUND') return true;
|
|
return false;
|
|
}
|
|
}
|
|
|
|
export function getPowerLabel(powerLevel) {
|
|
if (powerLevel > 9000) return 'Goku';
|
|
if (powerLevel > 100) return 'Founder';
|
|
if (powerLevel === 100) return 'Admin';
|
|
if (powerLevel >= 50) return 'Mod';
|
|
return null;
|
|
}
|
|
|
|
export function parseReply(rawBody) {
|
|
if (rawBody?.indexOf('>') !== 0) return null;
|
|
let body = rawBody.slice(rawBody.indexOf('<') + 1);
|
|
const user = body.slice(0, body.indexOf('>'));
|
|
|
|
body = body.slice(body.indexOf('>') + 2);
|
|
const replyBody = body.slice(0, body.indexOf('\n\n'));
|
|
body = body.slice(body.indexOf('\n\n') + 2);
|
|
|
|
if (user === '') return null;
|
|
|
|
const isUserId = user.match(/^@.+:.+/);
|
|
|
|
return {
|
|
userId: isUserId ? user : null,
|
|
displayName: isUserId ? null : user,
|
|
replyBody,
|
|
body,
|
|
};
|
|
}
|
|
|
|
export function hasDMWith(userId) {
|
|
const mx = initMatrix.matrixClient;
|
|
const directIds = [...initMatrix.roomList.directs];
|
|
|
|
return directIds.find((roomId) => {
|
|
const dRoom = mx.getRoom(roomId);
|
|
const roomMembers = dRoom.getMembers();
|
|
if (roomMembers.length <= 2 && dRoom.getMember(userId)) {
|
|
return true;
|
|
}
|
|
return false;
|
|
});
|
|
}
|
|
|
|
export function joinRuleToIconSrc(joinRule, isSpace) {
|
|
return ({
|
|
restricted: () => (isSpace ? SpaceIC : HashIC),
|
|
knock: () => (isSpace ? SpaceLockIC : HashLockIC),
|
|
invite: () => (isSpace ? SpaceLockIC : HashLockIC),
|
|
public: () => (isSpace ? SpaceGlobeIC : HashGlobeIC),
|
|
}[joinRule]?.() || null);
|
|
}
|
|
|
|
// NOTE: it gives userId with minimum power level 50;
|
|
function getHighestPowerUserId(room) {
|
|
const userIdToPower = room.currentState.getStateEvents('m.room.power_levels', '')?.getContent().users;
|
|
let powerUserId = null;
|
|
if (!userIdToPower) return powerUserId;
|
|
|
|
Object.keys(userIdToPower).forEach((userId) => {
|
|
if (userIdToPower[userId] < 50) return;
|
|
if (powerUserId === null) {
|
|
powerUserId = userId;
|
|
return;
|
|
}
|
|
if (userIdToPower[userId] > userIdToPower[powerUserId]) {
|
|
powerUserId = userId;
|
|
}
|
|
});
|
|
return powerUserId;
|
|
}
|
|
|
|
export function getIdServer(userId) {
|
|
const idParts = userId.split(':');
|
|
return idParts[1];
|
|
}
|
|
|
|
export function getServerToPopulation(room) {
|
|
const members = room.getMembers();
|
|
const serverToPop = {};
|
|
|
|
members?.forEach((member) => {
|
|
const { userId } = member;
|
|
const server = getIdServer(userId);
|
|
const serverPop = serverToPop[server];
|
|
if (serverPop === undefined) {
|
|
serverToPop[server] = 1;
|
|
return;
|
|
}
|
|
serverToPop[server] = serverPop + 1;
|
|
});
|
|
|
|
return serverToPop;
|
|
}
|
|
|
|
export function genRoomVia(room) {
|
|
const via = [];
|
|
const userId = getHighestPowerUserId(room);
|
|
if (userId) {
|
|
const server = getIdServer(userId);
|
|
if (server) via.push(server);
|
|
}
|
|
const serverToPop = getServerToPopulation(room);
|
|
const sortedServers = Object.keys(serverToPop).sort(
|
|
(svrA, svrB) => serverToPop[svrB] - serverToPop[svrA],
|
|
);
|
|
const mostPop3 = sortedServers.slice(0, 3);
|
|
if (via.length === 0) return mostPop3;
|
|
if (mostPop3.includes(via[0])) {
|
|
mostPop3.splice(mostPop3.indexOf(via[0]), 1);
|
|
}
|
|
return via.concat(mostPop3.slice(0, 2));
|
|
}
|
|
|
|
export function isCrossVerified(deviceId) {
|
|
try {
|
|
const mx = initMatrix.matrixClient;
|
|
const crossSignInfo = mx.getStoredCrossSigningForUser(mx.getUserId());
|
|
const deviceInfo = mx.getStoredDevice(mx.getUserId(), deviceId);
|
|
const deviceTrust = crossSignInfo.checkDeviceTrust(crossSignInfo, deviceInfo, false, true);
|
|
return deviceTrust.isCrossSigningVerified();
|
|
} catch {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
export function hasCrossSigningAccountData() {
|
|
const mx = initMatrix.matrixClient;
|
|
const masterKeyData = mx.getAccountData('m.cross_signing.master');
|
|
return !!masterKeyData;
|
|
}
|
|
|
|
export function getDefaultSSKey() {
|
|
const mx = initMatrix.matrixClient;
|
|
try {
|
|
return mx.getAccountData('m.secret_storage.default_key').getContent().key;
|
|
} catch {
|
|
return undefined;
|
|
}
|
|
}
|
|
|
|
export function getSSKeyInfo(key) {
|
|
const mx = initMatrix.matrixClient;
|
|
try {
|
|
return mx.getAccountData(`m.secret_storage.key.${key}`).getContent();
|
|
} catch {
|
|
return undefined;
|
|
}
|
|
}
|