Add function handling for OpenAI model

for now it's querying only time, but in the future there will be more commands
This commit is contained in:
Wroclaw 2023-07-23 05:50:16 +02:00
parent bebef021fb
commit 0df05e2f06
4 changed files with 169 additions and 33 deletions

View file

@ -8,12 +8,15 @@ datasource db {
} }
model Usage { model Usage {
timestamp DateTime @id @@id([timestamp, functionRan])
timestamp DateTime
user BigInt user BigInt
channel BigInt channel BigInt
guild BigInt? guild BigInt?
usageReguest Int usageReguest Int
usageResponse Int usageResponse Int
functionName String?
functionRan Int @default(0)
} }
model Limits { model Limits {

View file

@ -2,7 +2,8 @@ import { ChatCompletionRequestMessage as OpenAIMessage , CreateChatCompletionReq
// Don't forget to rename the file to config.ts // Don't forget to rename the file to config.ts
const calendarConfig: Intl.DateTimeFormatOptions = { export default class config {
static readonly calendarConfig: Intl.DateTimeFormatOptions = {
weekday: "short", weekday: "short",
year: "numeric", year: "numeric",
month: "short", month: "short",
@ -10,9 +11,8 @@ const calendarConfig: Intl.DateTimeFormatOptions = {
hour: "2-digit", hour: "2-digit",
minute: "2-digit", minute: "2-digit",
hour12: false, hour12: false,
}; };
export default class config {
/** Tokens to authenticate with */ /** Tokens to authenticate with */
static readonly tokens = { static readonly tokens = {
Discord: "Discord token here", Discord: "Discord token here",
@ -22,7 +22,7 @@ export default class config {
/** Messages to append at the start of every chat when sending to API */ /** Messages to append at the start of every chat when sending to API */
static systemPrompt(): OpenAIMessage[] { static systemPrompt(): OpenAIMessage[] {
return [ return [
{ role: "system", content: `You are GPTcord, an AI built on top of ChatGPT (a large language model trained by OpenAI) for Discord. Answer as concisely as possible. Current time (${Intl.DateTimeFormat().resolvedOptions().timeZone}): ${new Date().toLocaleString("en-US", calendarConfig)}` } { role: "system", content: `You are GPTcord, an AI built on top of ChatGPT (a large language model trained by OpenAI) for Discord. Answer as concisely as possible.` }
]; ];
} }

View file

@ -4,6 +4,7 @@ import { database, openai } from "./index";
import Moderation from "./moderation"; import Moderation from "./moderation";
import config from "./config"; import config from "./config";
import toOpenAIMessages from "./toOpenAIMessages"; import toOpenAIMessages from "./toOpenAIMessages";
import FunctionManager from "./funcitonManager";
type NonNullableInObject<T, V> = { [k in keyof T]: k extends V ? NonNullable<T[k]> : T[k] }; type NonNullableInObject<T, V> = { [k in keyof T]: k extends V ? NonNullable<T[k]> : T[k] };
type apiRequest = DiscordApi.Message | DiscordApi.RepliableInteraction; type apiRequest = DiscordApi.Message | DiscordApi.RepliableInteraction;
@ -12,6 +13,9 @@ export type request = apiRequest & NonNullableInObject<apiRequest, "channel" | "
/** Stores the queue requests on the channels. */ /** Stores the queue requests on the channels. */
const channelsRunning: DiscordApi.Collection<string, request[]> = new DiscordApi.Collection(); const channelsRunning: DiscordApi.Collection<string, request[]> = new DiscordApi.Collection();
type ChannelQueue = NonNullable<ReturnType<typeof channelsRunning.get>>;
type RequestMessage = NonNullable<ReturnType<ChannelQueue["at"]>>;
/** /**
* Gets the user that requested the execution * Gets the user that requested the execution
* @param request The request to get the user from * @param request The request to get the user from
@ -158,13 +162,49 @@ export async function queueRequest(request: apiRequest) {
executeFromQueue(request.channelId); executeFromQueue(request.channelId);
} }
/**
* Logs used tokens to the terminal and to the database
* @param answer the response that OpenAI returned
* @param message the message that initiated the execution
* @param functionRan counter of how many function have been ran
*/
function logUsedTokens(
answer: Awaited<ReturnType<typeof openai.createChatCompletion>>,
message: RequestMessage,
functionRan: number,
) {
const usage = answer.data.usage;
const functionName = answer.data.choices[0].message?.function_call?.name;
if (usage != undefined) {
const channelName: string = !message.channel.isDMBased() ? `${message.channel.name} (${message.guild?.name})` : `@${getAuthor(message).tag}`;
console.log(`Used ${usage.total_tokens} (${usage.prompt_tokens} + ${usage.completion_tokens}) tokens for ${getAuthor(message).tag} (${getAuthor(message).id}) in #${channelName}${functionName ? " [Function: " + functionName + "]" : ""}`);
database.usage.create({
data: {
timestamp: message.createdAt,
user: BigInt(getAuthor(message).id),
channel: BigInt(message.channelId),
guild: message.guildId ? BigInt(message.guildId) : null,
usageReguest: usage.prompt_tokens,
usageResponse: usage.completion_tokens,
functionName: functionName ?? null,
functionRan: functionName ? functionRan : 0,
}
}).catch((e => {
console.error("Failed to push to a database");
console.error(e);
}));
}
}
/** /**
* Executes the queue for the channel * Executes the queue for the channel
* @param channel the channel to run the queue for * @param channel the channel to run the queue for
*/ */
async function executeFromQueue(channel: string) { async function executeFromQueue(channel: string) {
const channelQueue = channelsRunning.get(channel) as NonNullable<ReturnType<typeof channelsRunning.get>>; const channelQueue = channelsRunning.get(channel) as ChannelQueue;
const message = channelQueue.at(0) as NonNullable<ReturnType<typeof channelQueue.at>>; const message = channelQueue.at(0) as RequestMessage;
let functionRanCounter = 0;
try { try {
let messages: DiscordApi.Collection<string, DiscordApi.Message> = await message.channel.messages.fetch({ limit: config.limits.messages, cache: false }); let messages: DiscordApi.Collection<string, DiscordApi.Message> = await message.channel.messages.fetch({ limit: config.limits.messages, cache: false });
@ -180,29 +220,39 @@ async function executeFromQueue(channel: string) {
message.deferReply(); message.deferReply();
} }
const answer = await openai.createChatCompletion({ const OpenAImessages = toOpenAIMessages(messages);
let answer = await openai.createChatCompletion({
...config.chatCompletionConfig, ...config.chatCompletionConfig,
messages: toOpenAIMessages(messages), messages: OpenAImessages,
// FIXME: don't use new instance of FunctionManager
functions: new FunctionManager().getFunctions(),
}); });
const usage = answer.data.usage; logUsedTokens(answer, message, ++functionRanCounter);
if (usage != undefined) {
const channelName: string = !message.channel.isDMBased() ? `${message.channel.name} (${message.guild?.name})` : `@${getAuthor(message).tag}`;
console.log(`Used ${usage.total_tokens} (${usage.prompt_tokens} + ${usage.completion_tokens}) tokens for ${getAuthor(message).tag} (${getAuthor(message).id}) in #${channelName}`);
database.usage.create({ let generatedMessage = answer.data.choices[0].message;
data: { if (!generatedMessage) throw new Error("empty message received");
timestamp: message.createdAt,
user: BigInt(getAuthor(message).id), // handle function calls
channel: BigInt(message.channelId), while (generatedMessage.function_call) {
guild: message.guildId ? BigInt(message.guildId) : null, OpenAImessages.push(generatedMessage);
usageReguest: usage.prompt_tokens, OpenAImessages.push({
usageResponse: usage.completion_tokens role: "function",
} name: generatedMessage.function_call.name,
}).catch((e => { // FIXME: don't use new instance of FunctionManager
console.error("Failed to push to a database"); content: new FunctionManager().handleFunction(generatedMessage.function_call),
console.error(e); });
})); answer = await openai.createChatCompletion({
...config.chatCompletionConfig,
messages: OpenAImessages,
// FIXME: don't use new instance of FunctionManager
functions: new FunctionManager().getFunctions(),
});
logUsedTokens(answer, message, ++functionRanCounter);
generatedMessage = answer.data.choices[0].message;
if (!generatedMessage) throw new Error("empty message received");
} }
const answerContent = answer.data.choices[0].message?.content; const answerContent = answer.data.choices[0].message?.content;

83
src/funcitonManager.ts Normal file
View file

@ -0,0 +1,83 @@
import { ChatCompletionFunctions, ChatCompletionRequestMessageFunctionCall } from "openai";
import config from "./config";
type parameterMap = {
string: string,
number: number,
};
type nameTypeMap = {[name: string]: keyof parameterMap} | Record<string, never>;
type OpenAIFunctionRequestData<T extends nameTypeMap> = {
[name in keyof T]: T[name];
};
/**
* Represents the function that can be ran by the OpenAI model
*/
export interface OpenAIFunction<T extends nameTypeMap = nameTypeMap> {
name: string,
description?: string,
parameters: {
type: "object",
properties: T extends Record<string, never> ? Record<string, never> : {
[name in T[string]]: {
type: T[name],
description?: string,
}
},
required?: Array<keyof T>,
},
}
export abstract class OpenAIFunction<T extends nameTypeMap = nameTypeMap> {
getSettings(): ChatCompletionFunctions {
return {
name: this.name,
description: this.description,
parameters: this.parameters,
};
}
abstract execute(data: OpenAIFunctionRequestData<T>): string;
}
/*
* Manages functions for the OpenAI
**/
export default class FunctionManager {
store = new Map<string, OpenAIFunction>();
constructor() {
this.store.set("getTime", new GetTime());
}
public getFunctions(): ChatCompletionFunctions[] {
const rvalue: ChatCompletionFunctions[] = [];
for (const [, value] of this.store) {
rvalue.push(value.getSettings());
}
return rvalue;
}
public handleFunction(request: ChatCompletionRequestMessageFunctionCall) {
const parsedArguments = JSON.parse(request.arguments ?? "");
return this.store.get(request.name ?? "")?.execute(parsedArguments);
}
}
// buildins
class GetTime extends OpenAIFunction<Record<string, never>> {
name = "getTime";
description = "Gets current date and time with a timezone attached";
parameters = {
type: "object" as const,
properties: {} as Record<string, never>,
};
execute(): string {
return `${Intl.DateTimeFormat().resolvedOptions().timeZone}): ${new Date().toLocaleString("en-US", config.calendarConfig)}`;
}
}