mirror of
https://github.com/discordeno/discordeno.git
synced 2026-06-04 01:40:08 +00:00
Use nacl for signature verification and remove server.ts (included in template)
This commit is contained in:
@@ -1,9 +0,0 @@
|
||||
# Discordeno Interactions
|
||||
|
||||
`interactions` is a standalone submodule that supports the webhooks through HTTP
|
||||
server to add support for Discord's interactions feature. This is a barebones
|
||||
interface that will create and handle requests from Discord API.
|
||||
|
||||
- Complete and extremely fast security and verification checks
|
||||
- First-class TypeScript Support
|
||||
- Minimalistic
|
||||
@@ -1,2 +1 @@
|
||||
export { serve } from "https://deno.land/std@0.90.0/http/server.ts";
|
||||
export { verify } from "https://unpkg.com/@evan/wasm@0.0.50/target/ed25519/deno.js";
|
||||
export { sign } from "https://unpkg.com/@evan/wasm@0.0.50/target/nacl/deno.js";
|
||||
|
||||
@@ -1 +1 @@
|
||||
export * from "./server.ts";
|
||||
export * from "./verify_signature.ts";
|
||||
|
||||
@@ -1,130 +0,0 @@
|
||||
import {
|
||||
DiscordInteractionResponseTypes,
|
||||
DiscordInteractionTypes,
|
||||
} from "../types/mod.ts";
|
||||
import { serve, verify } from "./deps.ts";
|
||||
|
||||
/** This variable is a holder for the public key and other configuration */
|
||||
const serverOptions = {
|
||||
publicKey: "",
|
||||
port: 80,
|
||||
};
|
||||
|
||||
/** Theses are the handlers that you can plug into and customize to your needs. */
|
||||
export const handlers = {
|
||||
handlePayload,
|
||||
handleApplicationCommand,
|
||||
};
|
||||
|
||||
/** Starts the slash command server */
|
||||
export async function startServer(
|
||||
{ port, publicKey, handleApplicationCommand }: StartServerConfig,
|
||||
) {
|
||||
serverOptions.publicKey = publicKey;
|
||||
serverOptions.port = port;
|
||||
if (handleApplicationCommand) {
|
||||
handlers.handleApplicationCommand = handleApplicationCommand;
|
||||
}
|
||||
|
||||
const server = serve({ port: serverOptions.port });
|
||||
|
||||
for await (const req of server) {
|
||||
// TODO: this is going to be removed
|
||||
// deno-lint-ignore no-deprecated-deno-api
|
||||
const buffer = await Deno.readAll(req.body);
|
||||
const signature = req.headers.get("X-Signature-Ed25519");
|
||||
const timestamp = req.headers.get("X-Signature-Timestamp");
|
||||
|
||||
if (!signature || !timestamp) {
|
||||
req.respond({ status: 400, body: "Bad request" });
|
||||
continue;
|
||||
}
|
||||
|
||||
const isVerified = verifySecurity(buffer, signature!, timestamp!);
|
||||
if (!isVerified) {
|
||||
req.respond({ status: 401, body: "Invalid request signature" });
|
||||
continue;
|
||||
}
|
||||
|
||||
try {
|
||||
const data = JSON.parse(new TextDecoder().decode(buffer));
|
||||
const response = await handlers.handlePayload(data);
|
||||
req.respond(
|
||||
{ status: response.status || 200, body: JSON.stringify(response.body) },
|
||||
);
|
||||
} catch (error) {
|
||||
console.error(error);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
function handlePayload(payload: Interaction) {
|
||||
switch (payload.type) {
|
||||
case DiscordInteractionTypes.Ping:
|
||||
return {
|
||||
status: 200,
|
||||
body: { type: DiscordInteractionResponseTypes.Pong },
|
||||
};
|
||||
default: // APPLICATION_COMMAND
|
||||
return handlers.handleApplicationCommand(payload);
|
||||
}
|
||||
}
|
||||
|
||||
/** The function that handles your commands. This command can be overriden by you and you can receive the payload and handle accordingly and respond back. The status if not provided will default to 200. */
|
||||
// deno-lint-ignore require-await
|
||||
async function handleApplicationCommand(
|
||||
payload: Interaction,
|
||||
): Promise<{ status?: number; body: InteractionResponse }> {
|
||||
// Handle the command
|
||||
if (payload.data?.name === "ping") {
|
||||
return {
|
||||
status: 200,
|
||||
body: {
|
||||
type: DiscordInteractionResponseTypes.ChannelMessageWithSource,
|
||||
data: { content: "Pong from Discordeno!" },
|
||||
},
|
||||
};
|
||||
}
|
||||
|
||||
return {
|
||||
status: 200,
|
||||
body: {
|
||||
type: DiscordInteractionResponseTypes.ChannelMessageWithSource,
|
||||
data: {
|
||||
content:
|
||||
"Whoopsies! Seems the handling for this command is missing. Please contact my developers!",
|
||||
},
|
||||
},
|
||||
};
|
||||
}
|
||||
|
||||
/** Internal function to verify security. Discord will send bad and good data and this function is important to verify it. If it is not verified properly, Discord will kill your bot. */
|
||||
export function verifySecurity(
|
||||
buffer: Uint8Array,
|
||||
signature: string,
|
||||
time: string,
|
||||
) {
|
||||
const sig = new Uint8Array(64);
|
||||
const timestamp = new TextEncoder().encode(time);
|
||||
|
||||
let offset = 0;
|
||||
const message = new Uint8Array(buffer.length + timestamp.length);
|
||||
while (offset < 2 * 64) {
|
||||
sig[offset / 2] = parseInt(signature!.substring(offset, offset += 2), 16);
|
||||
}
|
||||
|
||||
const slashKey = new Uint8Array(32);
|
||||
let keyoffset = 0;
|
||||
|
||||
while (keyoffset < 2 * 32) {
|
||||
slashKey[keyoffset / 2] = parseInt(
|
||||
serverOptions.publicKey.substring(keyoffset, keyoffset += 2),
|
||||
16,
|
||||
);
|
||||
}
|
||||
|
||||
message.set(timestamp);
|
||||
message.set(buffer, timestamp.length);
|
||||
|
||||
return verify(slashKey, sig, message);
|
||||
}
|
||||
Reference in New Issue
Block a user