element-call/src/room/GroupCallInspector.jsx

446 lines
12 KiB
React
Raw Normal View History

import { Resizable } from "re-resizable";
2022-01-13 22:11:06 +00:00
import React, { useEffect, useState, useReducer, useRef } from "react";
2021-10-15 23:41:23 +00:00
import ReactJson from "react-json-view";
2022-01-13 22:11:06 +00:00
import mermaid from "mermaid";
import styles from "./GroupCallInspector.module.css";
import { SelectInput } from "../input/SelectInput";
import { Item } from "@react-stately/collections";
2021-10-15 23:41:23 +00:00
function getCallUserId(call) {
return call.getOpponentMember()?.userId || call.invitee || null;
}
function getCallState(call) {
return {
2021-10-21 19:53:17 +00:00
id: call.callId,
2021-10-15 23:41:23 +00:00
opponentMemberId: getCallUserId(call),
state: call.state,
direction: call.direction,
};
}
function getHangupCallState(call) {
return {
...getCallState(call),
hangupReason: call.hangupReason,
};
}
2022-01-12 21:47:46 +00:00
const dateFormatter = new Intl.DateTimeFormat([], {
hour: "2-digit",
minute: "2-digit",
second: "2-digit",
2022-01-13 22:11:06 +00:00
fractionalSecondDigits: 3,
2022-01-12 21:47:46 +00:00
});
const defaultCollapsedFields = [
"org.matrix.msc3401.call",
"org.matrix.msc3401.call.member",
"calls",
"callStats",
"hangupCalls",
"toDeviceEvents",
"sentVoipEvents",
"content",
];
function shouldCollapse({ name, src, type, namespace }) {
return defaultCollapsedFields.includes(name);
}
2022-01-13 22:11:06 +00:00
function getUserName(userId) {
2022-01-18 21:52:16 +00:00
const match = userId.match(/@([^\:]+):/);
2021-10-15 23:41:23 +00:00
2022-01-18 21:52:16 +00:00
return match && match.length > 0
? match[1].replace("-", " ").replace("W", "")
: userId.replace("W", "");
2022-01-13 22:11:06 +00:00
}
2021-10-15 23:41:23 +00:00
2022-01-13 22:11:06 +00:00
function formatContent(type, content) {
2022-01-14 21:40:02 +00:00
if (type === "m.call.hangup") {
2022-01-13 22:11:06 +00:00
return `callId: ${content.call_id.slice(-4)} reason: ${
content.reason
2022-01-14 21:40:02 +00:00
} senderSID: ${content.sender_session_id} destSID: ${
content.dest_session_id
}`;
}
if (type.startsWith("m.call.")) {
return `callId: ${content.call_id?.slice(-4)} senderSID: ${
content.sender_session_id
} destSID: ${content.dest_session_id}`;
2022-01-13 22:11:06 +00:00
} else if (type === "org.matrix.msc3401.call.member") {
const call =
content["m.calls"] &&
content["m.calls"].length > 0 &&
content["m.calls"][0];
const device =
call &&
call["m.devices"] &&
call["m.devices"].length > 0 &&
call["m.devices"][0];
2022-01-14 21:40:02 +00:00
return `callId: ${call && call["m.call_id"].slice(-4)} sessionId: ${
device && device.session_id
}`;
2022-01-13 22:11:06 +00:00
} else {
return "";
}
}
2021-10-15 23:41:23 +00:00
2022-01-13 22:11:06 +00:00
function formatTimestamp(timestamp) {
return dateFormatter.format(timestamp);
}
2021-10-15 23:41:23 +00:00
2022-01-13 22:11:06 +00:00
function SequenceDiagramViewer({
localUserId,
remoteUserIds,
selectedUserId,
onSelectUserId,
events,
}) {
const mermaidElRef = useRef();
2021-10-15 23:41:23 +00:00
2022-01-13 22:11:06 +00:00
useEffect(() => {
mermaid.initialize({
startOnLoad: true,
theme: "dark",
sequence: {
showSequenceNumbers: true,
},
});
}, []);
2021-10-15 23:41:23 +00:00
2022-01-13 22:11:06 +00:00
useEffect(() => {
const graphDefinition = `sequenceDiagram
participant ${getUserName(localUserId)}
participant Room
participant ${selectedUserId ? getUserName(selectedUserId) : "unknown"}
${
events
? events
.map(
2022-01-14 21:40:02 +00:00
({ to, from, timestamp, type, content, ignored }) =>
`${getUserName(from)} ${ignored ? "-x" : "->>"} ${getUserName(
2022-01-13 22:11:06 +00:00
to
)}: ${formatTimestamp(timestamp)} ${type} ${formatContent(
type,
content
)}`
)
.join("\n ")
: ""
}
`;
2021-10-15 23:41:23 +00:00
2022-01-13 22:11:06 +00:00
mermaid.mermaidAPI.render("mermaid", graphDefinition, (svgCode) => {
mermaidElRef.current.innerHTML = svgCode;
});
}, [events, localUserId, selectedUserId]);
2021-10-15 23:41:23 +00:00
2022-01-13 22:11:06 +00:00
return (
<div className={styles.scrollContainer}>
<div className={styles.sequenceDiagramViewer}>
<SelectInput
className={styles.selectInput}
label="Remote User"
selectedKey={selectedUserId}
onSelectionChange={onSelectUserId}
>
{remoteUserIds.map((userId) => (
<Item key={userId}>{userId}</Item>
))}
</SelectInput>
<div id="mermaid" />
<div ref={mermaidElRef} />
</div>
</div>
);
}
2021-10-15 23:41:23 +00:00
2022-01-13 22:11:06 +00:00
function reducer(state, action) {
switch (action.type) {
case "receive_room_state_event": {
const { event, callStateEvent, memberStateEvents } = action;
2021-10-15 23:41:23 +00:00
2022-01-13 22:11:06 +00:00
let eventsByUserId = state.eventsByUserId;
let remoteUserIds = state.remoteUserIds;
2021-10-15 23:41:23 +00:00
2022-01-13 22:11:06 +00:00
if (event) {
const fromId = event.getStateKey();
2022-01-14 21:52:33 +00:00
remoteUserIds =
fromId === state.localUserId || eventsByUserId.has(fromId)
? state.remoteUserIds
: [...state.remoteUserIds, fromId];
2022-01-13 22:11:06 +00:00
eventsByUserId = new Map(state.eventsByUserId);
if (event.getStateKey() === state.localUserId) {
for (const userId in eventsByUserId) {
eventsByUserId.set(userId, [
...(eventsByUserId.get(userId) || []),
{
from: fromId,
to: "Room",
type: event.getType(),
content: event.getContent(),
timestamp: event.getTs() || Date.now(),
2022-01-14 21:40:02 +00:00
ignored: false,
2022-01-13 22:11:06 +00:00
},
]);
}
} else {
eventsByUserId.set(fromId, [
...(eventsByUserId.get(fromId) || []),
{
from: fromId,
to: "Room",
type: event.getType(),
content: event.getContent(),
timestamp: event.getTs() || Date.now(),
2022-01-14 21:40:02 +00:00
ignored: false,
2022-01-13 22:11:06 +00:00
},
]);
}
2021-10-15 23:41:23 +00:00
}
2022-01-13 22:11:06 +00:00
return {
...state,
eventsByUserId,
remoteUserIds,
callStateEvent: callStateEvent.getContent(),
memberStateEvents: Object.fromEntries(
memberStateEvents.map((e) => [e.getStateKey(), e.getContent()])
),
};
}
case "receive_to_device_event": {
const event = action.event;
const eventsByUserId = new Map(state.eventsByUserId);
const fromId = event.getSender();
const toId = state.localUserId;
2022-01-14 21:40:02 +00:00
const content = event.getContent();
2022-01-13 22:11:06 +00:00
const remoteUserIds = eventsByUserId.has(fromId)
? state.remoteUserIds
: [...state.remoteUserIds, fromId];
eventsByUserId.set(fromId, [
...(eventsByUserId.get(fromId) || []),
2022-01-12 21:47:46 +00:00
{
2022-01-13 22:11:06 +00:00
from: fromId,
to: toId,
type: event.getType(),
2022-01-14 21:40:02 +00:00
content,
2022-01-13 22:11:06 +00:00
timestamp: event.getTs() || Date.now(),
2022-01-14 21:40:02 +00:00
ignored: state.localSessionId !== content.dest_session_id,
2022-01-12 21:47:46 +00:00
},
2021-11-01 20:39:30 +00:00
]);
2021-10-15 23:41:23 +00:00
2022-01-13 22:11:06 +00:00
return { ...state, eventsByUserId, remoteUserIds };
2022-01-06 23:24:35 +00:00
}
2022-01-13 22:11:06 +00:00
case "send_voip_event": {
const event = action.event;
const eventsByUserId = new Map(state.eventsByUserId);
const fromId = state.localUserId;
const toId = event.userId;
const remoteUserIds = eventsByUserId.has(toId)
? state.remoteUserIds
: [...state.remoteUserIds, toId];
eventsByUserId.set(toId, [
...(eventsByUserId.get(toId) || []),
{
from: fromId,
to: toId,
type: event.eventType,
content: event.content,
timestamp: Date.now(),
2022-01-14 21:40:02 +00:00
ignored: false,
2022-01-13 22:11:06 +00:00
},
]);
2022-01-06 23:24:35 +00:00
2022-01-13 22:11:06 +00:00
return { ...state, eventsByUserId, remoteUserIds };
2022-01-06 23:24:35 +00:00
}
2022-01-13 22:11:06 +00:00
default:
return state;
}
}
2022-01-06 23:24:35 +00:00
2022-01-13 22:11:06 +00:00
function useGroupCallState(client, groupCall, pollCallStats) {
const [state, dispatch] = useReducer(reducer, {
localUserId: client.getUserId(),
2022-01-14 21:40:02 +00:00
localSessionId: client.getSessionId(),
2022-01-13 22:11:06 +00:00
eventsByUserId: new Map(),
remoteUserIds: [],
callStateEvent: null,
memberStateEvents: {},
});
2022-01-06 23:24:35 +00:00
useEffect(() => {
2022-01-13 22:11:06 +00:00
function onUpdateRoomState(event) {
const callStateEvent = groupCall.room.currentState.getStateEvents(
"org.matrix.msc3401.call",
groupCall.groupCallId
);
2022-01-13 22:11:06 +00:00
const memberStateEvents = groupCall.room.currentState.getStateEvents(
"org.matrix.msc3401.call.member"
2021-11-01 19:37:45 +00:00
);
2022-01-13 22:11:06 +00:00
dispatch({
type: "receive_room_state_event",
event,
callStateEvent,
memberStateEvents,
});
}
2021-11-01 19:37:45 +00:00
2022-01-13 22:11:06 +00:00
// function onCallsChanged() {
// const calls = groupCall.calls.reduce((obj, call) => {
// obj[
// `${call.callId} (${call.getOpponentMember()?.userId || call.sender})`
// ] = getCallState(call);
// return obj;
// }, {});
// updateState({ calls });
// }
// function onCallHangup(call) {
// setState(({ hangupCalls, ...rest }) => ({
// ...rest,
// hangupCalls: {
// ...hangupCalls,
// [`${call.callId} (${
// call.getOpponentMember()?.userId || call.sender
// })`]: getHangupCallState(call),
// },
// }));
// dispatch({ type: "call_hangup", call });
// }
2021-11-01 19:37:45 +00:00
2022-01-13 22:11:06 +00:00
function onToDeviceEvent(event) {
dispatch({ type: "receive_to_device_event", event });
2021-11-01 19:37:45 +00:00
}
2022-01-13 22:11:06 +00:00
function onSendVoipEvent(event) {
dispatch({ type: "send_voip_event", event });
}
2021-11-01 19:37:45 +00:00
2022-01-13 22:11:06 +00:00
client.on("RoomState.events", onUpdateRoomState);
//groupCall.on("calls_changed", onCallsChanged);
groupCall.on("send_voip_event", onSendVoipEvent);
//client.on("state", onCallsChanged);
//client.on("hangup", onCallHangup);
client.on("toDeviceEvent", onToDeviceEvent);
onUpdateRoomState();
2021-11-01 19:37:45 +00:00
return () => {
2022-01-13 22:11:06 +00:00
client.removeListener("RoomState.events", onUpdateRoomState);
//groupCall.removeListener("calls_changed", onCallsChanged);
groupCall.removeListener("send_voip_event", onSendVoipEvent);
//client.removeListener("state", onCallsChanged);
//client.removeListener("hangup", onCallHangup);
client.removeListener("toDeviceEvent", onToDeviceEvent);
2021-11-01 19:37:45 +00:00
};
2022-01-13 22:11:06 +00:00
}, [client, groupCall]);
// useEffect(() => {
// let timeout;
// async function updateCallStats() {
// const callIds = groupCall.calls.map(
// (call) =>
// `${call.callId} (${call.getOpponentMember()?.userId || call.sender})`
// );
// const stats = await Promise.all(
// groupCall.calls.map((call) =>
// call.peerConn
// ? call.peerConn
// .getStats(null)
// .then((stats) =>
// Object.fromEntries(
// Array.from(stats).map(([_id, report], i) => [
// report.type + i,
// report,
// ])
// )
// )
// : Promise.resolve(null)
// )
// );
// const callStats = {};
// for (let i = 0; i < groupCall.calls.length; i++) {
// callStats[callIds[i]] = stats[i];
// }
// dispatch({ type: "callStats", callStats });
// timeout = setTimeout(updateCallStats, 1000);
// }
// if (pollCallStats) {
// updateCallStats();
// }
// return () => {
// clearTimeout(timeout);
// };
// }, [pollCallStats]);
return state;
}
export function GroupCallInspector({ client, groupCall, show }) {
2022-01-14 21:40:02 +00:00
const [currentTab, setCurrentTab] = useState("sequence-diagrams");
2022-01-13 22:11:06 +00:00
const [selectedUserId, setSelectedUserId] = useState();
const state = useGroupCallState(client, groupCall, show);
2021-10-15 23:41:23 +00:00
2021-11-01 19:49:26 +00:00
if (!show) {
return null;
}
2021-10-15 23:41:23 +00:00
return (
2022-01-13 22:11:06 +00:00
<Resizable
enable={{ top: true }}
defaultSize={{ height: 200 }}
className={styles.inspector}
>
<div className={styles.toolbar}>
<button onClick={() => setCurrentTab("sequence-diagrams")}>
Sequence Diagrams
</button>
2022-01-14 21:40:02 +00:00
<button onClick={() => setCurrentTab("inspector")}>Inspector</button>
2022-01-13 22:11:06 +00:00
</div>
{currentTab === "sequence-diagrams" && (
<SequenceDiagramViewer
localUserId={state.localUserId}
selectedUserId={selectedUserId}
onSelectUserId={setSelectedUserId}
remoteUserIds={state.remoteUserIds}
events={state.eventsByUserId.get(selectedUserId)}
/>
)}
{currentTab === "inspector" && (
<ReactJson
theme="monokai"
src={{
...state,
eventsByUserId: Object.fromEntries(state.eventsByUserId),
}}
name={null}
indentWidth={2}
shouldCollapse={shouldCollapse}
displayDataTypes={false}
displayObjectSize={false}
enableClipboard
style={{ height: "100%", overflowY: "scroll" }}
/>
)}
</Resizable>
2021-10-15 23:41:23 +00:00
);
}