-
-
Notifications
You must be signed in to change notification settings - Fork 2.3k
Add a devtool for looking at users and their devices #30983
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
base: develop
Are you sure you want to change the base?
Changes from all commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change | ||||
---|---|---|---|---|---|---|
@@ -0,0 +1,356 @@ | ||||||
/* | ||||||
* Copyright 2025 New Vector Ltd. | ||||||
* | ||||||
* SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only OR LicenseRef-Element-Commercial | ||||||
* Please see LICENSE files in the repository root for full details. | ||||||
*/ | ||||||
|
||||||
/** | ||||||
* @file Devtool for viewing room members and their devices. | ||||||
*/ | ||||||
|
||||||
import React, { type JSX, useContext, useState } from "react"; | ||||||
import { type Device, type RoomMember } from "matrix-js-sdk/src/matrix"; | ||||||
import { type CryptoApi } from "matrix-js-sdk/src/crypto-api"; | ||||||
|
||||||
import { _t } from "../../../../languageHandler"; | ||||||
import BaseTool, { DevtoolsContext, type IDevtoolsProps } from "./BaseTool"; | ||||||
import FilteredList from "./FilteredList"; | ||||||
import LabelledToggleSwitch from "../../elements/LabelledToggleSwitch"; | ||||||
import { useAsyncMemo } from "../../../../hooks/useAsyncMemo"; | ||||||
import CopyableText from "../../elements/CopyableText"; | ||||||
import E2EIcon from "../../rooms/E2EIcon"; | ||||||
import { E2EStatus } from "../../../../utils/ShieldUtils"; | ||||||
|
||||||
/** | ||||||
* Replacement function for `<i>` tags in translation strings. | ||||||
*/ | ||||||
function i(sub: string): JSX.Element { | ||||||
return <i>{sub}</i>; | ||||||
} | ||||||
|
||||||
/** | ||||||
* Shows a list of users in the room, and allows selecting a user to view. | ||||||
* | ||||||
* By default, filters to only show joined users. | ||||||
* | ||||||
* If the `member` state is set, delegates to `User` to view a single user. | ||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. what is There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. also: the talk of state seems like an implementation detail which does not belong in the doc-comment?
Suggested change
|
||||||
*/ | ||||||
export const UserList: React.FC<Pick<IDevtoolsProps, "onBack">> = ({ onBack }) => { | ||||||
const context = useContext(DevtoolsContext); | ||||||
const [query, setQuery] = useState(""); | ||||||
// Show only joined users or all users with member events? | ||||||
const [showOnlyJoined, setShowOnlyJoined] = useState(true); | ||||||
// The `RoomMember` for the selected user (if any) | ||||||
const [member, setMember] = useState<RoomMember | null>(null); | ||||||
|
||||||
if (member) { | ||||||
const _onBack = (): void => { | ||||||
setMember(null); | ||||||
}; | ||||||
Comment on lines
+48
to
+50
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I'd be inclined to inline this into the There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I was copying from another file in the directory, but yeah, it's short enough that it can be inlined. |
||||||
return <UserView member={member} onBack={_onBack} />; | ||||||
} | ||||||
|
||||||
const members = showOnlyJoined ? context.room.getJoinedMembers() : context.room.getMembers(); | ||||||
|
||||||
return ( | ||||||
<BaseTool onBack={onBack}> | ||||||
<FilteredList query={query} onChange={setQuery}> | ||||||
{members.map((member) => ( | ||||||
<UserButton key={member.userId} member={member} onClick={() => setMember(member)} /> | ||||||
))} | ||||||
</FilteredList> | ||||||
<LabelledToggleSwitch | ||||||
label={_t("devtools|only_joined_members")} | ||||||
onChange={setShowOnlyJoined} | ||||||
value={showOnlyJoined} | ||||||
/> | ||||||
</BaseTool> | ||||||
); | ||||||
}; | ||||||
|
||||||
interface UserButtonProps { | ||||||
member: RoomMember; | ||||||
onClick(): void; | ||||||
} | ||||||
|
||||||
/** | ||||||
* Button to select a user to view. | ||||||
*/ | ||||||
const UserButton: React.FC<UserButtonProps> = ({ member, onClick }) => { | ||||||
return ( | ||||||
<button className="mx_DevTools_button" onClick={onClick}> | ||||||
{member.userId} | ||||||
</button> | ||||||
); | ||||||
}; | ||||||
|
||||||
interface UserProps extends Pick<IDevtoolsProps, "onBack"> { | ||||||
member: RoomMember; | ||||||
} | ||||||
|
||||||
/** | ||||||
* Shows a single user to view, and allows selecting a device to view. | ||||||
* | ||||||
* If the `device` state is set, delegates to `Device` to show a single device. | ||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. as above, this needs attention |
||||||
*/ | ||||||
const UserView: React.FC<UserProps> = ({ member, onBack }) => { | ||||||
const context = useContext(DevtoolsContext); | ||||||
const crypto = context.room.client.getCrypto(); | ||||||
const verificationStatus = useAsyncMemo( | ||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. it would probably be helpful to add a comment documenting the type and possible values of |
||||||
async () => { | ||||||
if (!crypto) { | ||||||
return null; | ||||||
} | ||||||
const status = await crypto.getUserVerificationStatus(member.userId); | ||||||
if (status.isCrossSigningVerified()) { | ||||||
const e2eIcon = (): JSX.Element => ( | ||||||
<E2EIcon | ||||||
isUser={true} | ||||||
hideTooltip={true} | ||||||
status={E2EStatus.Verified} | ||||||
className="mx_E2EIcon_inline" | ||||||
/> | ||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I'm no react expert, but ... putting this into
... and put the rest outside There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. That's actually how I had it originally, but it all migrated into the |
||||||
); | ||||||
return _t("devtools|user_verification_status|verified", {}, { E2EIcon: e2eIcon }); | ||||||
} else if (status.wasCrossSigningVerified()) { | ||||||
const e2eIcon = (): JSX.Element => ( | ||||||
<E2EIcon | ||||||
isUser={true} | ||||||
hideTooltip={true} | ||||||
status={E2EStatus.Warning} | ||||||
className="mx_E2EIcon_inline" | ||||||
/> | ||||||
); | ||||||
return _t("devtools|user_verification_status|was_verified", {}, { E2EIcon: e2eIcon }); | ||||||
} else if (status.needsUserApproval) { | ||||||
const e2eIcon = (): JSX.Element => ( | ||||||
<E2EIcon | ||||||
isUser={true} | ||||||
hideTooltip={true} | ||||||
status={E2EStatus.Warning} | ||||||
className="mx_E2EIcon_inline" | ||||||
/> | ||||||
); | ||||||
return _t("devtools|user_verification_status|identity_changed", {}, { E2EIcon: e2eIcon }); | ||||||
} else { | ||||||
const e2eIcon = (): JSX.Element => ( | ||||||
<E2EIcon isUser={true} hideTooltip={true} status={E2EStatus.Normal} className="mx_E2EIcon_inline" /> | ||||||
); | ||||||
return _t("devtools|user_verification_status|unverified", {}, { E2EIcon: e2eIcon }); | ||||||
} | ||||||
}, | ||||||
[context], | ||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Since this depends on
Suggested change
|
||||||
_t("common|loading"), | ||||||
); | ||||||
const devices = useAsyncMemo( | ||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. again, a comment recording what the type of |
||||||
async () => { | ||||||
const devices = await crypto?.getUserDeviceInfo([member.userId]); | ||||||
return devices?.get(member.userId) ?? new Map(); | ||||||
}, | ||||||
[context], | ||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Suggested change
|
||||||
new Map(), | ||||||
); | ||||||
// The device to show, if any. | ||||||
const [device, setDevice] = useState<Device | null>(null); | ||||||
|
||||||
if (device) { | ||||||
const _onBack = (): void => { | ||||||
setDevice(null); | ||||||
}; | ||||||
return <DeviceView crypto={crypto!} device={device} onBack={_onBack} />; | ||||||
} | ||||||
|
||||||
const avatarUrl = member.getMxcAvatarUrl(); | ||||||
const memberEventContent = member.events.member?.getContent(); | ||||||
|
||||||
return ( | ||||||
<BaseTool onBack={onBack}> | ||||||
<ul> | ||||||
<li> | ||||||
<CopyableText getTextToCopy={() => member.userId} border={false}> | ||||||
{_t("devtools|user_id", { userId: member.userId })} | ||||||
</CopyableText> | ||||||
</li> | ||||||
<li>{_t("devtools|user_room_membership", { membership: member.membership ?? "leave" })}</li> | ||||||
<li> | ||||||
{memberEventContent && "displayname" in memberEventContent | ||||||
? _t("devtools|user_displayname", { displayname: member.rawDisplayName }) | ||||||
: _t("devtools|user_no_displayname", {}, { i })} | ||||||
</li> | ||||||
<li> | ||||||
{avatarUrl !== undefined ? ( | ||||||
<CopyableText getTextToCopy={() => avatarUrl} border={false}> | ||||||
{_t("devtools|user_avatar", { avatar: avatarUrl })} | ||||||
</CopyableText> | ||||||
) : ( | ||||||
_t("devtools|user_no_avatar", {}, { i }) | ||||||
)} | ||||||
</li> | ||||||
<li>{verificationStatus}</li> | ||||||
</ul> | ||||||
<section> | ||||||
<h2>{_t("devtools|devices", { count: devices.size })}</h2> | ||||||
<ul> | ||||||
{Array.from(devices.values()).map((device) => ( | ||||||
<li key={device.deviceId}> | ||||||
<DeviceButton crypto={crypto!} device={device} onClick={() => setDevice(device)} /> | ||||||
</li> | ||||||
))} | ||||||
</ul> | ||||||
</section> | ||||||
</BaseTool> | ||||||
); | ||||||
}; | ||||||
|
||||||
interface DeviceButtonProps { | ||||||
crypto: CryptoApi; | ||||||
device: Device; | ||||||
onClick(): void; | ||||||
} | ||||||
|
||||||
/** | ||||||
* Button to select a user to view. | ||||||
*/ | ||||||
const DeviceButton: React.FC<DeviceButtonProps> = ({ crypto, device, onClick }) => { | ||||||
const verificationIcon = useAsyncMemo( | ||||||
async () => { | ||||||
const status = await crypto.getDeviceVerificationStatus(device.userId, device.deviceId); | ||||||
if (!status) { | ||||||
return; | ||||||
} else if (status.crossSigningVerified) { | ||||||
return ( | ||||||
<E2EIcon | ||||||
isUser={true} | ||||||
hideTooltip={true} | ||||||
status={E2EStatus.Verified} | ||||||
className="mx_E2EIcon_inline" | ||||||
/> | ||||||
); | ||||||
} else if (status.signedByOwner) { | ||||||
return ( | ||||||
<E2EIcon isUser={true} hideTooltip={true} status={E2EStatus.Normal} className="mx_E2EIcon_inline" /> | ||||||
); | ||||||
} else { | ||||||
return ( | ||||||
<E2EIcon | ||||||
isUser={true} | ||||||
hideTooltip={true} | ||||||
status={E2EStatus.Warning} | ||||||
className="mx_E2EIcon_inline" | ||||||
/> | ||||||
); | ||||||
} | ||||||
}, | ||||||
[], | ||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Suggested change
|
||||||
null, | ||||||
); | ||||||
return ( | ||||||
<button className="mx_DevTools_button" onClick={onClick}> | ||||||
{verificationIcon} | ||||||
{device.deviceId} | ||||||
</button> | ||||||
); | ||||||
}; | ||||||
|
||||||
interface DeviceProps extends Pick<IDevtoolsProps, "onBack"> { | ||||||
crypto: CryptoApi; | ||||||
device: Device; | ||||||
} | ||||||
|
||||||
/** | ||||||
* Show a single device to view. | ||||||
*/ | ||||||
const DeviceView: React.FC<DeviceProps> = ({ crypto, device, onBack }) => { | ||||||
const verificationStatus = useAsyncMemo( | ||||||
async () => { | ||||||
const status = await crypto.getDeviceVerificationStatus(device.userId, device.deviceId); | ||||||
if (!status) { | ||||||
// `status` will be `null` if the device is unknown or if the | ||||||
// device doesn't have device keys. In either case, it's not a | ||||||
// security issue since we won't be sending it decryption keys. | ||||||
return _t("devtools|device_verification_status|unknown"); | ||||||
} else if (status.crossSigningVerified) { | ||||||
const e2eIcon = (): JSX.Element => ( | ||||||
<E2EIcon | ||||||
isUser={true} | ||||||
hideTooltip={true} | ||||||
status={E2EStatus.Verified} | ||||||
className="mx_E2EIcon_inline" | ||||||
/> | ||||||
); | ||||||
return _t("devtools|device_verification_status|verified", {}, { E2EIcon: e2eIcon }); | ||||||
} else if (status.signedByOwner) { | ||||||
const e2eIcon = (): JSX.Element => ( | ||||||
<E2EIcon isUser={true} hideTooltip={true} status={E2EStatus.Normal} className="mx_E2EIcon_inline" /> | ||||||
); | ||||||
return _t("devtools|device_verification_status|signed_by_owner", {}, { E2EIcon: e2eIcon }); | ||||||
} else { | ||||||
const e2eIcon = (): JSX.Element => ( | ||||||
<E2EIcon | ||||||
isUser={true} | ||||||
hideTooltip={true} | ||||||
status={E2EStatus.Warning} | ||||||
className="mx_E2EIcon_inline" | ||||||
/> | ||||||
); | ||||||
return _t("devtools|device_verification_status|unverified", {}, { E2EIcon: e2eIcon }); | ||||||
} | ||||||
}, | ||||||
[], | ||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Suggested change
|
||||||
_t("common|loading"), | ||||||
); | ||||||
|
||||||
const keyIdSuffix = ":" + device.deviceId; | ||||||
const deviceKeys = ( | ||||||
<ul> | ||||||
{Array.from(device.keys.entries()).map(([keyId, key]) => { | ||||||
if (keyId.endsWith(keyIdSuffix)) { | ||||||
return ( | ||||||
<li key={keyId}> | ||||||
<CopyableText getTextToCopy={() => key} border={false}> | ||||||
{keyId.slice(0, -keyIdSuffix.length)}: {key} | ||||||
</CopyableText> | ||||||
</li> | ||||||
); | ||||||
} else { | ||||||
return ( | ||||||
<li key={keyId}> | ||||||
<i>{_t("devtools|invalid_device_key_id")}</i>: {keyId}: {key} | ||||||
</li> | ||||||
); | ||||||
} | ||||||
})} | ||||||
</ul> | ||||||
); | ||||||
|
||||||
return ( | ||||||
<BaseTool onBack={onBack}> | ||||||
<ul> | ||||||
<li> | ||||||
<CopyableText getTextToCopy={() => device.userId} border={false}> | ||||||
{_t("devtools|user_id", { userId: device.userId })} | ||||||
</CopyableText> | ||||||
</li> | ||||||
<li> | ||||||
<CopyableText getTextToCopy={() => device.deviceId} border={false}> | ||||||
{_t("devtools|device_id", { deviceId: device.deviceId })} | ||||||
</CopyableText> | ||||||
</li> | ||||||
<li> | ||||||
{"displayName" in device | ||||||
? _t("devtools|user_displayname", { displayname: device.displayName }) | ||||||
: _t("devtools|user_no_displayname", {}, { i })} | ||||||
</li> | ||||||
<li>{verificationStatus}</li> | ||||||
<li> | ||||||
{device.dehydrated ? _t("devtools|device_dehydrated_yes") : _t("devtools|device_dehydrated_no")} | ||||||
</li> | ||||||
<li> | ||||||
{_t("devtools|device_keys")} | ||||||
{deviceKeys} | ||||||
</li> | ||||||
</ul> | ||||||
</BaseTool> | ||||||
); | ||||||
}; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
how does one override this default, as a user of the component?