@rocket.chat/apps-engine
Version:
The engine code for the Rocket.Chat Apps which manages, runs, translates, coordinates and all of that.
123 lines (98 loc) • 5.33 kB
text/typescript
import { Defined, JsonRpcError } from 'jsonrpc-lite';
import type { App } from "@rocket.chat/apps-engine/definition/App.ts";
import type { IRoom } from '@rocket.chat/apps-engine/definition/rooms/IRoom.ts';
import type { ISlashCommand } from '@rocket.chat/apps-engine/definition/slashcommands/ISlashCommand.ts';
import type { SlashCommandContext as _SlashCommandContext } from '@rocket.chat/apps-engine/definition/slashcommands/SlashCommandContext.ts';
import type { Room as _Room } from '@rocket.chat/apps-engine/server/rooms/Room.ts';
import { AppObjectRegistry } from '../AppObjectRegistry.ts';
import { AppAccessors, AppAccessorsInstance } from '../lib/accessors/mod.ts';
import { require } from '../lib/require.ts';
import createRoom from '../lib/roomFactory.ts';
// For some reason Deno couldn't understand the typecast to the original interfaces and said it wasn't a constructor type
const { SlashCommandContext } = require('@rocket.chat/apps-engine/definition/slashcommands/SlashCommandContext.js') as {
SlashCommandContext: typeof _SlashCommandContext;
};
export default async function slashCommandHandler(call: string, params: unknown): Promise<JsonRpcError | Defined> {
const [, commandName, method] = call.split(':');
const command = AppObjectRegistry.get<ISlashCommand>(`slashcommand:${commandName}`);
if (!command) {
return new JsonRpcError(`Slashcommand ${commandName} not found`, -32000);
}
let result: Awaited<ReturnType<typeof handleExecutor>> | Awaited<ReturnType<typeof handlePreviewItem>>;
// If the command is registered, we're pretty safe to assume the app is not undefined
const app = AppObjectRegistry.get<App>('app')!;
app.getLogger().debug(`${commandName}'s ${method} is being executed...`, params);
try {
if (method === 'executor' || method === 'previewer') {
result = await handleExecutor({ AppAccessorsInstance }, command, method, params);
} else if (method === 'executePreviewItem') {
result = await handlePreviewItem({ AppAccessorsInstance }, command, params);
} else {
return new JsonRpcError(`Method ${method} not found on slashcommand ${commandName}`, -32000);
}
app.getLogger().debug(`${commandName}'s ${method} was successfully executed.`);
} catch (error) {
app.getLogger().debug(`${commandName}'s ${method} was unsuccessful.`);
return new JsonRpcError(error.message, -32000);
}
return result;
}
/**
* @param deps Dependencies that need to be injected into the slashcommand
* @param command The slashcommand that is being executed
* @param method The method that is being executed
* @param params The parameters that are being passed to the method
*/
export function handleExecutor(deps: { AppAccessorsInstance: AppAccessors }, command: ISlashCommand, method: 'executor' | 'previewer', params: unknown) {
const executor = command[method];
if (typeof executor !== 'function') {
throw new Error(`Method ${method} not found on slashcommand ${command.command}`);
}
if (!Array.isArray(params) || typeof params[0] !== 'object' || !params[0]) {
throw new Error(`First parameter must be an object`);
}
const { sender, room, params: args, threadId, triggerId } = params[0] as Record<string, unknown>;
const context = new SlashCommandContext(
sender as _SlashCommandContext['sender'],
createRoom(room as IRoom, deps.AppAccessorsInstance.getSenderFn()),
args as _SlashCommandContext['params'],
threadId as _SlashCommandContext['threadId'],
triggerId as _SlashCommandContext['triggerId'],
);
return executor.apply(command, [
context,
deps.AppAccessorsInstance.getReader(),
deps.AppAccessorsInstance.getModifier(),
deps.AppAccessorsInstance.getHttp(),
deps.AppAccessorsInstance.getPersistence(),
]);
}
/**
* @param deps Dependencies that need to be injected into the slashcommand
* @param command The slashcommand that is being executed
* @param params The parameters that are being passed to the method
*/
export function handlePreviewItem(deps: { AppAccessorsInstance: AppAccessors }, command: ISlashCommand, params: unknown) {
if (typeof command.executePreviewItem !== 'function') {
throw new Error(`Method not found on slashcommand ${command.command}`);
}
if (!Array.isArray(params) || typeof params[0] !== 'object' || !params[0]) {
throw new Error(`First parameter must be an object`);
}
const [previewItem, { sender, room, params: args, threadId, triggerId }] = params as [Record<string, unknown>, Record<string, unknown>];
const context = new SlashCommandContext(
sender as _SlashCommandContext['sender'],
createRoom(room as IRoom, deps.AppAccessorsInstance.getSenderFn()),
args as _SlashCommandContext['params'],
threadId as _SlashCommandContext['threadId'],
triggerId as _SlashCommandContext['triggerId'],
);
return command.executePreviewItem(
previewItem,
context,
deps.AppAccessorsInstance.getReader(),
deps.AppAccessorsInstance.getModifier(),
deps.AppAccessorsInstance.getHttp(),
deps.AppAccessorsInstance.getPersistence(),
);
}