/[sudobot]/branches/5.x/src/commands/settings/EvalCommand.ts
ViewVC logotype

Annotation of /branches/5.x/src/commands/settings/EvalCommand.ts

Parent Directory Parent Directory | Revision Log Revision Log


Revision 577 - (hide annotations)
Mon Jul 29 18:52:37 2024 UTC (8 months ago) by rakinar2
File MIME type: application/typescript
File size: 5028 byte(s)
chore: add old version archive branches (2.x to 9.x-dev)
1 rakinar2 577 /**
2     * This file is part of SudoBot.
3     *
4     * Copyright (C) 2021-2023 OSN Developers.
5     *
6     * SudoBot is free software; you can redistribute it and/or modify it
7     * under the terms of the GNU Affero General Public License as published by
8     * the Free Software Foundation, either version 3 of the License, or
9     * (at your option) any later version.
10     *
11     * SudoBot is distributed in the hope that it will be useful, but
12     * WITHOUT ANY WARRANTY; without even the implied warranty of
13     * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14     * GNU Affero General Public License for more details.
15     *
16     * You should have received a copy of the GNU Affero General Public License
17     * along with SudoBot. If not, see <https://www.gnu.org/licenses/>.
18     */
19    
20     import { SlashCommandBuilder, escapeCodeBlock, escapeMarkdown } from "discord.js";
21     import Command, { ArgumentType, BasicCommandContext, CommandMessage, CommandReturn, ValidationRule } from "../../core/Command";
22     import { logError } from "../../utils/logger";
23    
24     export default class EvalCommand extends Command {
25     public readonly name = "eval";
26     public readonly validationRules: ValidationRule[] = [
27     {
28     types: [ArgumentType.StringRest],
29     optional: false,
30     name: "code",
31     requiredErrorMessage: "You must provide expression(s) to evaluate!"
32     }
33     ];
34     public readonly systemAdminOnly = true;
35     public errorOccurred: boolean = false;
36    
37     public readonly description = "Execute JavaScript code.";
38     public readonly detailedDescription = "This command executes arbitrary JavaScript code. Must be used with caution.";
39     public readonly argumentSyntaxes = ["<...Code>"];
40    
41     public readonly botRequiredPermissions = [];
42    
43     public readonly slashCommandBuilder = new SlashCommandBuilder().addStringOption(option =>
44     option.setName("code").setDescription("The code to execute").setRequired(true)
45     );
46    
47     createUncaughtExecptionHandler(message: CommandMessage) {
48     return (e: Error) => {
49     this.errorOccurred = true;
50     logError(e);
51    
52     this.deferredReply(message, {
53     embeds: [
54     {
55     description: `${this.emoji("error")} **Exception occurred**\n\n\`\`\`\n${escapeMarkdown(
56     e.message + "\n" + e.stack
57     )}\n\`\`\``,
58     color: 0xf14a60
59     }
60     ]
61     }).catch(logError);
62     };
63     }
64    
65     createUnhandledRejection(message: CommandMessage) {
66     return (e: unknown) => {
67     this.errorOccurred = true;
68     logError(e);
69    
70     this.deferredReply(message, {
71     embeds: [
72     {
73     description: `${this.emoji("error")} **Unhandled promise rejection**\n\n\`\`\`\n${
74     typeof e === "string" || typeof (e as any)?.toString === "function"
75     ? escapeCodeBlock((e as any)?.toString ? (e as any).toString() : (e as any))
76     : e
77     }\n\`\`\``,
78     color: 0xf14a60
79     }
80     ]
81     }).catch(logError);
82     };
83     }
84    
85     async execute(message: CommandMessage, context: BasicCommandContext): Promise<CommandReturn> {
86     this.errorOccurred = false;
87    
88     const code = context.isLegacy ? context.parsedNamedArgs.code : context.options.getString("code", true);
89     await this.deferIfInteraction(message);
90    
91     const exceptionHandler = this.createUncaughtExecptionHandler(message);
92     const rejectionHandler = this.createUnhandledRejection(message);
93    
94     process.on("uncaughtExceptionMonitor", exceptionHandler);
95     process.on("unhandledRejection", rejectionHandler);
96    
97     try {
98     const result = eval(code);
99     const string = `${
100     typeof result === "string" || typeof result?.toString === "function"
101     ? escapeCodeBlock((result as any)?.toString ? (result as any).toString() : (result as any))
102     : result
103     }`;
104    
105     if (!this.errorOccurred) {
106     await this.deferredReply(message, {
107     embeds: [
108     {
109     description: `${this.emoji("check")} **Execution succeeded**\n\n${
110     string.trim() === "" ? "*No output*" : `\`\`\`${string}\`\`\``
111     }`,
112     color: 0x007bff
113     }
114     ]
115     });
116     }
117     } catch (e) {
118     logError("Evaluation failed");
119     logError(e);
120    
121     if ("stack" in (e as any) && "message" in (e as any)) exceptionHandler(e as any);
122     else rejectionHandler(e);
123     }
124    
125     process.off("uncaughtExceptionMonitor", exceptionHandler);
126     process.off("unhandledRejection", rejectionHandler);
127     }
128     }

[email protected]
ViewVC Help
Powered by ViewVC 1.1.26