mirror of
https://github.com/zulip/zulip.git
synced 2025-11-18 12:54:58 +00:00
submessage: Convert module to TypeScript.
This commit is contained in:
207
web/src/submessage.ts
Normal file
207
web/src/submessage.ts
Normal file
@@ -0,0 +1,207 @@
|
||||
import z from "zod";
|
||||
|
||||
import * as blueslip from "./blueslip";
|
||||
import * as channel from "./channel";
|
||||
import * as message_store from "./message_store";
|
||||
import type {Message} from "./message_store";
|
||||
import type {Submessage} from "./types";
|
||||
import * as widgetize from "./widgetize";
|
||||
|
||||
export const zform_widget_extra_data_schema = z.object({
|
||||
choices: z.array(
|
||||
z.object({
|
||||
type: z.string(),
|
||||
long_name: z.string(),
|
||||
reply: z.string(),
|
||||
short_name: z.string(),
|
||||
}),
|
||||
),
|
||||
heading: z.string(),
|
||||
type: z.literal("choices"),
|
||||
});
|
||||
|
||||
const poll_widget_extra_data_schema = z.object({
|
||||
question: z.string().optional(),
|
||||
options: z.array(z.string()).optional(),
|
||||
});
|
||||
|
||||
const widget_data_event_schema = z.object({
|
||||
sender_id: z.number(),
|
||||
data: z.object({
|
||||
widget_type: z.string().optional(),
|
||||
extra_data: z
|
||||
.union([zform_widget_extra_data_schema, poll_widget_extra_data_schema])
|
||||
.nullable(),
|
||||
}),
|
||||
});
|
||||
|
||||
const inbound_data_event_schema = z.object({
|
||||
sender_id: z.number(),
|
||||
data: z.intersection(
|
||||
z.object({
|
||||
type: z.string(),
|
||||
}),
|
||||
z.record(z.string(), z.unknown()),
|
||||
),
|
||||
});
|
||||
|
||||
const submessages_event_schema = z
|
||||
.tuple([widget_data_event_schema])
|
||||
.rest(inbound_data_event_schema);
|
||||
|
||||
type SubmessageEvents = z.infer<typeof submessages_event_schema>;
|
||||
|
||||
export function get_message_events(message: Message): SubmessageEvents | undefined {
|
||||
if (message.locally_echoed) {
|
||||
return undefined;
|
||||
}
|
||||
|
||||
if (!message.submessages) {
|
||||
return undefined;
|
||||
}
|
||||
|
||||
if (message.submessages.length === 0) {
|
||||
return undefined;
|
||||
}
|
||||
|
||||
message.submessages.sort((m1, m2) => m1.id - m2.id);
|
||||
|
||||
const events = message.submessages.map((obj) => ({
|
||||
sender_id: obj.sender_id,
|
||||
data: JSON.parse(obj.content),
|
||||
}));
|
||||
const clean_events = submessages_event_schema.parse(events);
|
||||
return clean_events;
|
||||
}
|
||||
|
||||
export function process_submessages(in_opts: {$row: JQuery; message_id: number}): void {
|
||||
// This happens in our rendering path, so we try to limit any
|
||||
// damage that may be triggered by one rogue message.
|
||||
try {
|
||||
do_process_submessages(in_opts);
|
||||
return;
|
||||
} catch (error) {
|
||||
blueslip.error("Failed to do_process_submessages", undefined, error);
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
export function do_process_submessages(in_opts: {$row: JQuery; message_id: number}): void {
|
||||
const message_id = in_opts.message_id;
|
||||
const message = message_store.get(message_id);
|
||||
|
||||
if (!message) {
|
||||
return;
|
||||
}
|
||||
|
||||
const events = get_message_events(message);
|
||||
|
||||
if (!events) {
|
||||
return;
|
||||
}
|
||||
const [widget_event, ...inbound_events] = events;
|
||||
|
||||
if (widget_event.sender_id !== message.sender_id) {
|
||||
blueslip.warn(`User ${widget_event.sender_id} tried to hijack message ${message.id}`);
|
||||
return;
|
||||
}
|
||||
|
||||
const $row = in_opts.$row;
|
||||
|
||||
// Right now, our only use of submessages is widgets.
|
||||
|
||||
const data = widget_event.data;
|
||||
|
||||
if (data === undefined) {
|
||||
return;
|
||||
}
|
||||
|
||||
const widget_type = data.widget_type;
|
||||
|
||||
if (widget_type === undefined) {
|
||||
return;
|
||||
}
|
||||
|
||||
const post_to_server = make_server_callback(message_id);
|
||||
|
||||
widgetize.activate({
|
||||
widget_type,
|
||||
extra_data: data.extra_data,
|
||||
events: inbound_events,
|
||||
$row,
|
||||
message,
|
||||
post_to_server,
|
||||
});
|
||||
}
|
||||
|
||||
export function update_message(submsg: Submessage): void {
|
||||
const message = message_store.get(submsg.message_id);
|
||||
|
||||
if (message === undefined) {
|
||||
// This is generally not a problem--the server
|
||||
// can send us events without us having received
|
||||
// the original message, since the server doesn't
|
||||
// track that.
|
||||
return;
|
||||
}
|
||||
|
||||
if (message.submessages === undefined) {
|
||||
message.submessages = [];
|
||||
}
|
||||
|
||||
const existing = message.submessages.find((sm) => sm.id === submsg.id);
|
||||
|
||||
if (existing !== undefined) {
|
||||
blueslip.warn("Got submessage multiple times: " + submsg.id);
|
||||
return;
|
||||
}
|
||||
|
||||
message.submessages.push(submsg);
|
||||
}
|
||||
|
||||
export function handle_event(submsg: Submessage): void {
|
||||
// Update message.submessages in case we haven't actually
|
||||
// activated the widget yet, so that when the message does
|
||||
// come in view, the data will be complete.
|
||||
update_message(submsg);
|
||||
|
||||
// Right now, our only use of submessages is widgets.
|
||||
const msg_type = submsg.msg_type;
|
||||
|
||||
if (msg_type !== "widget") {
|
||||
blueslip.warn("unknown msg_type: " + msg_type);
|
||||
return;
|
||||
}
|
||||
|
||||
let data;
|
||||
|
||||
try {
|
||||
data = JSON.parse(submsg.content);
|
||||
} catch {
|
||||
blueslip.warn("server sent us invalid json in handle_event: " + submsg.content);
|
||||
return;
|
||||
}
|
||||
|
||||
widgetize.handle_event({
|
||||
sender_id: submsg.sender_id,
|
||||
message_id: submsg.message_id,
|
||||
data,
|
||||
});
|
||||
}
|
||||
|
||||
export function make_server_callback(
|
||||
message_id: number,
|
||||
): (opts: {msg_type: string; data: string}) => void {
|
||||
return function (opts: {msg_type: string; data: string}) {
|
||||
const url = "/json/submessage";
|
||||
|
||||
void channel.post({
|
||||
url,
|
||||
data: {
|
||||
message_id,
|
||||
msg_type: opts.msg_type,
|
||||
content: JSON.stringify(opts.data),
|
||||
},
|
||||
});
|
||||
};
|
||||
}
|
||||
Reference in New Issue
Block a user