generated from nhcarrigan/template
feat: add ability to trigger clear message to reset the conversation history (#2)
### Explanation _No response_ ### Issue _No response_ ### Attestations - [ ] I have read and agree to the [Code of Conduct](https://docs.nhcarrigan.com/community/coc/) - [ ] I have read and agree to the [Community Guidelines](https://docs.nhcarrigan.com/community/guide/). - [ ] My contribution complies with the [Contributor Covenant](https://docs.nhcarrigan.com/dev/covenant/). ### Dependencies - [ ] I have pinned the dependencies to a specific patch version. ### Style - [ ] I have run the linter and resolved any errors. - [ ] My pull request uses an appropriate title, matching the conventional commit standards. - [ ] My scope of feat/fix/chore/etc. correctly matches the nature of changes in my pull request. ### Tests - [ ] My contribution adds new code, and I have added tests to cover it. - [ ] My contribution modifies existing code, and I have updated the tests to reflect these changes. - [ ] All new and existing tests pass locally with my changes. - [ ] Code coverage remains at or above the configured threshold. ### Documentation _No response_ ### Versioning _No response_ Reviewed-on: #2 Co-authored-by: Naomi Carrigan <commits@nhcarrigan.com> Co-committed-by: Naomi Carrigan <commits@nhcarrigan.com>
This commit is contained in:
parent
04b18472ba
commit
efc8f904ef
3
dev.env
Normal file
3
dev.env
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
DISCORD_TOKEN="op://Environment Variables - Development/Naomi Dev Bot/token"
|
||||||
|
AI_TOKEN="op://Environment Variables - Development/Naomi Dev Bot/ai token"
|
||||||
|
LOG_TOKEN="op://Environment Variables - Naomi/Alert Server/api_auth"
|
24
src/commands/clear.ts
Normal file
24
src/commands/clear.ts
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
/**
|
||||||
|
* @copyright nhcarrigan
|
||||||
|
* @license Naomi's Public License
|
||||||
|
* @author Naomi Carrigan
|
||||||
|
*/
|
||||||
|
|
||||||
|
import {
|
||||||
|
ApplicationIntegrationType,
|
||||||
|
SlashCommandBuilder,
|
||||||
|
InteractionContextType,
|
||||||
|
} from "discord.js";
|
||||||
|
|
||||||
|
const command = new SlashCommandBuilder().
|
||||||
|
setContexts(
|
||||||
|
InteractionContextType.BotDM,
|
||||||
|
InteractionContextType.Guild,
|
||||||
|
InteractionContextType.PrivateChannel,
|
||||||
|
).
|
||||||
|
setIntegrationTypes(ApplicationIntegrationType.UserInstall).
|
||||||
|
setName("clear").
|
||||||
|
setDescription("Clear your current adventure so you can start a new one!");
|
||||||
|
|
||||||
|
// eslint-disable-next-line no-console -- We don't need our logger here as this never runs in production.
|
||||||
|
console.log(JSON.stringify(command.toJSON()));
|
@ -20,7 +20,7 @@ import type { MessageParam } from "@anthropic-ai/sdk/resources/index.js";
|
|||||||
* Handles the Discord message event.
|
* Handles the Discord message event.
|
||||||
* @param message - The message payload from Discord.
|
* @param message - The message payload from Discord.
|
||||||
*/
|
*/
|
||||||
// eslint-disable-next-line max-lines-per-function -- We're off by one bloody line.
|
// eslint-disable-next-line max-lines-per-function, max-statements -- We're off by one bloody line.
|
||||||
export const onMessage = async(message: Message): Promise<void> => {
|
export const onMessage = async(message: Message): Promise<void> => {
|
||||||
try {
|
try {
|
||||||
if (message.channel.type !== ChannelType.DM) {
|
if (message.channel.type !== ChannelType.DM) {
|
||||||
@ -33,20 +33,32 @@ export const onMessage = async(message: Message): Promise<void> => {
|
|||||||
if (!subbed) {
|
if (!subbed) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const history = await message.channel.messages.fetch({ limit: 6 });
|
const historyRequest = await message.channel.messages.fetch({ limit: 6 });
|
||||||
const context: Array<MessageParam>
|
const history = [ ...historyRequest.values() ];
|
||||||
= history.reverse().map((messageInner) => {
|
const clearMessageIndex = history.findIndex((messageInner) => {
|
||||||
return {
|
return (
|
||||||
content: messageInner.content,
|
messageInner.content === "<Clear History>"
|
||||||
role:
|
&& messageInner.author.id === message.client.user.id
|
||||||
messageInner.author.id === message.client.user.id
|
);
|
||||||
? "assistant"
|
|
||||||
: "user",
|
|
||||||
};
|
|
||||||
});
|
});
|
||||||
|
if (clearMessageIndex !== -1) {
|
||||||
|
// Remove the clear message and everything sent before it, which means everything after in the array because the array is backwards
|
||||||
|
history.splice(0, clearMessageIndex + 1);
|
||||||
|
}
|
||||||
|
const context: Array<MessageParam> = history.
|
||||||
|
reverse().
|
||||||
|
map((messageInner) => {
|
||||||
|
return {
|
||||||
|
content: messageInner.content,
|
||||||
|
role:
|
||||||
|
messageInner.author.id === message.client.user.id
|
||||||
|
? "assistant"
|
||||||
|
: "user",
|
||||||
|
};
|
||||||
|
});
|
||||||
const messages = await ai.messages.create({
|
const messages = await ai.messages.create({
|
||||||
// eslint-disable-next-line @typescript-eslint/naming-convention -- Required key format for SDK.
|
// eslint-disable-next-line @typescript-eslint/naming-convention -- Required key format for SDK.
|
||||||
max_tokens: 3000,
|
max_tokens: 5000,
|
||||||
messages: context,
|
messages: context,
|
||||||
model: "claude-3-5-sonnet-latest",
|
model: "claude-3-5-sonnet-latest",
|
||||||
system: `${personality} Provide a response to the user that continues the story. The user's name is ${message.author.displayName}`,
|
system: `${personality} Provide a response to the user that continues the story. The user's name is ${message.author.displayName}`,
|
||||||
|
@ -6,6 +6,7 @@
|
|||||||
import { Client, Events, GatewayIntentBits, Partials } from "discord.js";
|
import { Client, Events, GatewayIntentBits, Partials } from "discord.js";
|
||||||
import { onMessage } from "./events/message.js";
|
import { onMessage } from "./events/message.js";
|
||||||
import { about } from "./modules/about.js";
|
import { about } from "./modules/about.js";
|
||||||
|
import { clear } from "./modules/clear.js";
|
||||||
import { start } from "./modules/start.js";
|
import { start } from "./modules/start.js";
|
||||||
import { instantiateServer } from "./server/serve.js";
|
import { instantiateServer } from "./server/serve.js";
|
||||||
import { logger } from "./utils/logger.js";
|
import { logger } from "./utils/logger.js";
|
||||||
@ -41,6 +42,9 @@ client.on(Events.InteractionCreate, (interaction) => {
|
|||||||
case "start":
|
case "start":
|
||||||
void start(interaction);
|
void start(interaction);
|
||||||
break;
|
break;
|
||||||
|
case "clear":
|
||||||
|
void clear(interaction);
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
void interaction.reply({
|
void interaction.reply({
|
||||||
content: `I'm sorry, I don't know the ${interaction.commandName} command.`,
|
content: `I'm sorry, I don't know the ${interaction.commandName} command.`,
|
||||||
|
47
src/modules/clear.ts
Normal file
47
src/modules/clear.ts
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
/**
|
||||||
|
* @copyright nhcarrigan
|
||||||
|
* @license Naomi's Public License
|
||||||
|
* @author Naomi Carrigan
|
||||||
|
*/
|
||||||
|
|
||||||
|
import {
|
||||||
|
MessageFlags,
|
||||||
|
type ChatInputCommandInteraction,
|
||||||
|
} from "discord.js";
|
||||||
|
import { isSubscribedInteraction } from "../utils/isSubscribed.js";
|
||||||
|
import { logger } from "../utils/logger.js";
|
||||||
|
import { replyToError } from "../utils/replyToError.js";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sends a clear message in the DMs.
|
||||||
|
* @param interaction -- The interaction payload from Discord.
|
||||||
|
*/
|
||||||
|
export const clear = async(
|
||||||
|
interaction: ChatInputCommandInteraction,
|
||||||
|
): Promise<void> => {
|
||||||
|
try {
|
||||||
|
await interaction.deferReply({ flags: [ MessageFlags.Ephemeral ] });
|
||||||
|
|
||||||
|
const subbed = await isSubscribedInteraction(interaction);
|
||||||
|
if (!subbed) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const sent = await interaction.user.send({
|
||||||
|
content: "<Clear History>",
|
||||||
|
}).catch(() => {
|
||||||
|
return null;
|
||||||
|
});
|
||||||
|
|
||||||
|
await interaction.editReply({
|
||||||
|
content: sent
|
||||||
|
? "I have added a clear history marker to your DMs."
|
||||||
|
// eslint-disable-next-line stylistic/max-len -- This is a long string.
|
||||||
|
: "I was unable to send you a DM. Please ensure your privacy settings allow direct messages.",
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
await replyToError(interaction);
|
||||||
|
if (error instanceof Error) {
|
||||||
|
await logger.error("about command", error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
Loading…
x
Reference in New Issue
Block a user