zulip/static/js/submessage.js

156 lines
3.7 KiB
JavaScript
Raw Normal View History

import * as blueslip from "./blueslip";
import * as channel from "./channel";
import * as message_store from "./message_store";
import * as widgetize from "./widgetize";
export function get_message_events(message) {
if (message.locally_echoed) {
return undefined;
}
if (!message.submessages) {
return undefined;
}
if (message.submessages.length === 0) {
return undefined;
}
message.submessages.sort((m1, m2) => Number.parseInt(m1.id, 10) - Number.parseInt(m2.id, 10));
const events = message.submessages.map((obj) => ({
js: Convert _.map(a, …) to a.map(…). And convert the corresponding function expressions to arrow style while we’re here. import * as babelParser from "recast/parsers/babel"; import * as recast from "recast"; import * as tsParser from "recast/parsers/typescript"; import { builders as b, namedTypes as n } from "ast-types"; import K from "ast-types/gen/kinds"; import fs from "fs"; import path from "path"; import process from "process"; const checkExpression = (node: n.Node): node is K.ExpressionKind => n.Expression.check(node); for (const file of process.argv.slice(2)) { console.log("Parsing", file); const ast = recast.parse(fs.readFileSync(file, { encoding: "utf8" }), { parser: path.extname(file) === ".ts" ? tsParser : babelParser, }); let changed = false; recast.visit(ast, { visitCallExpression(path) { const { callee, arguments: args } = path.node; if ( n.MemberExpression.check(callee) && !callee.computed && n.Identifier.check(callee.object) && callee.object.name === "_" && n.Identifier.check(callee.property) && callee.property.name === "map" && args.length === 2 && checkExpression(args[0]) && checkExpression(args[1]) ) { const [arr, fn] = args; path.replace( b.callExpression(b.memberExpression(arr, b.identifier("map")), [ n.FunctionExpression.check(fn) || n.ArrowFunctionExpression.check(fn) ? b.arrowFunctionExpression( fn.params, n.BlockStatement.check(fn.body) && fn.body.body.length === 1 && n.ReturnStatement.check(fn.body.body[0]) ? fn.body.body[0].argument || b.identifier("undefined") : fn.body ) : fn, ]) ); changed = true; } this.traverse(path); }, }); if (changed) { console.log("Writing", file); fs.writeFileSync(file, recast.print(ast).code, { encoding: "utf8" }); } } Signed-off-by: Anders Kaseorg <anders@zulipchat.com>
2020-02-08 02:43:49 +01:00
sender_id: obj.sender_id,
data: JSON.parse(obj.content),
}));
return events;
}
export function process_submessages(in_opts) {
// This happens in our rendering path, so we try to limit any
// damage that may be triggered by one rogue message.
try {
return do_process_submessages(in_opts);
} catch (error) {
blueslip.error("in process_submessages: " + error.message);
return undefined;
}
}
export function do_process_submessages(in_opts) {
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;
}
if (events[0].sender_id !== message.sender_id) {
blueslip.warn(`User ${events[0].sender_id} tried to hijack message ${message.id}`);
return;
}
const row = in_opts.row;
2018-02-23 16:20:33 +01:00
// Right now, our only use of submessages is widgets.
const data = events[0].data;
2018-02-23 16:20:33 +01:00
if (data === undefined) {
return;
}
const widget_type = data.widget_type;
2018-02-23 16:20:33 +01:00
if (widget_type === undefined) {
return;
}
const post_to_server = make_server_callback(message_id);
2018-02-23 16:20:33 +01:00
widgetize.activate({
widget_type,
2018-02-23 16:20:33 +01:00
extra_data: data.extra_data,
events,
row,
message,
post_to_server,
2018-02-23 16:20:33 +01:00
});
}
export function update_message(submsg) {
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) {
// 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);
2018-02-23 16:20:33 +01:00
// Right now, our only use of submessages is widgets.
const msg_type = submsg.msg_type;
2018-02-23 16:20:33 +01:00
if (msg_type !== "widget") {
blueslip.warn("unknown msg_type: " + msg_type);
2018-02-23 16:20:33 +01:00
return;
}
let data;
try {
data = JSON.parse(submsg.content);
} catch {
blueslip.error("server sent us invalid json in handle_event: " + submsg.content);
return;
}
2018-02-23 16:20:33 +01:00
widgetize.handle_event({
sender_id: submsg.sender_id,
message_id: submsg.message_id,
data,
2018-02-23 16:20:33 +01:00
});
}
export function make_server_callback(message_id) {
return function (opts) {
const url = "/json/submessage";
channel.post({
url,
data: {
message_id,
msg_type: opts.msg_type,
content: JSON.stringify(opts.data),
},
});
};
}