agsamantha/node_modules/@langchain/community/dist/memory/zep.js

221 lines
7.7 KiB
JavaScript
Raw Normal View History

2024-10-02 20:15:21 +00:00
import { Memory, Message, NotFoundError, ZepClient } from "@getzep/zep-js";
import { getInputValue, getOutputValue, } from "@langchain/core/memory";
import { getBufferString, AIMessage, ChatMessage, HumanMessage, SystemMessage, } from "@langchain/core/messages";
import { BaseChatMemory } from "./chat_memory.js";
/**
* Class used to manage the memory of a chat session, including loading
* and saving the chat history, and clearing the memory when needed. It
* uses the ZepClient to interact with the Zep service for managing the
* chat session's memory.
* @example
* ```typescript
* const sessionId = randomUUID();
* const zepURL = "http://your-zep-url";
*
* // Initialize ZepMemory with session ID, base URL, and API key
* const memory = new ZepMemory({
* sessionId,
* baseURL: zepURL,
* apiKey: "change_this_key",
* });
*
* // Create a ChatOpenAI model instance with specific parameters
* const model = new ChatOpenAI({
* modelName: "gpt-3.5-turbo",
* temperature: 0,
* });
*
* // Create a ConversationChain with the model and memory
* const chain = new ConversationChain({ llm: model, memory });
*
* // Example of calling the chain with an input
* const res1 = await chain.call({ input: "Hi! I'm Jim." });
* console.log({ res1 });
*
* // Follow-up call to the chain to demonstrate memory usage
* const res2 = await chain.call({ input: "What did I just say my name was?" });
* console.log({ res2 });
*
* // Output the session ID and the current state of memory
* console.log("Session ID: ", sessionId);
* console.log("Memory: ", await memory.loadMemoryVariables({}));
*
* ```
*/
export class ZepMemory extends BaseChatMemory {
constructor(fields) {
super({
returnMessages: fields?.returnMessages ?? false,
inputKey: fields?.inputKey,
outputKey: fields?.outputKey,
});
Object.defineProperty(this, "humanPrefix", {
enumerable: true,
configurable: true,
writable: true,
value: "Human"
});
Object.defineProperty(this, "aiPrefix", {
enumerable: true,
configurable: true,
writable: true,
value: "AI"
});
Object.defineProperty(this, "memoryKey", {
enumerable: true,
configurable: true,
writable: true,
value: "history"
});
Object.defineProperty(this, "baseURL", {
enumerable: true,
configurable: true,
writable: true,
value: void 0
});
Object.defineProperty(this, "sessionId", {
enumerable: true,
configurable: true,
writable: true,
value: void 0
});
Object.defineProperty(this, "zepClientPromise", {
enumerable: true,
configurable: true,
writable: true,
value: void 0
});
Object.defineProperty(this, "zepInitFailMsg", {
enumerable: true,
configurable: true,
writable: true,
value: "ZepClient is not initialized"
});
this.humanPrefix = fields.humanPrefix ?? this.humanPrefix;
this.aiPrefix = fields.aiPrefix ?? this.aiPrefix;
this.memoryKey = fields.memoryKey ?? this.memoryKey;
this.baseURL = fields.baseURL;
this.sessionId = fields.sessionId;
this.zepClientPromise = ZepClient.init(this.baseURL, fields.apiKey);
}
get memoryKeys() {
return [this.memoryKey];
}
/**
* Method that retrieves the chat history from the Zep service and formats
* it into a list of messages.
* @param values Input values for the method.
* @returns Promise that resolves with the chat history formatted into a list of messages.
*/
async loadMemoryVariables(values) {
// use either lastN provided by developer or undefined to use the
// server preset.
// Wait for ZepClient to be initialized
const zepClient = await this.zepClientPromise;
if (!zepClient) {
throw new Error(this.zepInitFailMsg);
}
const lastN = values.lastN ?? undefined;
let memory = null;
try {
memory = await zepClient.memory.getMemory(this.sessionId, lastN);
}
catch (error) {
// eslint-disable-next-line no-instanceof/no-instanceof
if (error instanceof NotFoundError) {
const result = this.returnMessages
? { [this.memoryKey]: [] }
: { [this.memoryKey]: "" };
return result;
}
else {
throw error;
}
}
let messages = memory && memory.summary?.content
? [new SystemMessage(memory.summary.content)]
: [];
if (memory) {
messages = messages.concat(memory.messages.map((message) => {
const { content, role } = message;
if (role === this.humanPrefix) {
return new HumanMessage(content);
}
else if (role === this.aiPrefix) {
return new AIMessage(content);
}
else {
// default to generic ChatMessage
return new ChatMessage(content, role);
}
}));
}
if (this.returnMessages) {
return {
[this.memoryKey]: messages,
};
}
return {
[this.memoryKey]: getBufferString(messages, this.humanPrefix, this.aiPrefix),
};
}
/**
* Method that saves the input and output messages to the Zep service.
* @param inputValues Input messages to be saved.
* @param outputValues Output messages to be saved.
* @returns Promise that resolves when the messages have been saved.
*/
async saveContext(inputValues, outputValues) {
const input = getInputValue(inputValues, this.inputKey);
const output = getOutputValue(outputValues, this.outputKey);
// Create new Memory and Message instances
const memory = new Memory({
messages: [
new Message({
role: this.humanPrefix,
content: `${input}`,
}),
new Message({
role: this.aiPrefix,
content: `${output}`,
}),
],
});
// Wait for ZepClient to be initialized
const zepClient = await this.zepClientPromise;
if (!zepClient) {
throw new Error(this.zepInitFailMsg);
}
// Add the new memory to the session using the ZepClient
if (this.sessionId) {
try {
await zepClient.memory.addMemory(this.sessionId, memory);
}
catch (error) {
console.error("Error adding memory: ", error);
}
}
// Call the superclass's saveContext method
await super.saveContext(inputValues, outputValues);
}
/**
* Method that deletes the chat history from the Zep service.
* @returns Promise that resolves when the chat history has been deleted.
*/
async clear() {
// Wait for ZepClient to be initialized
const zepClient = await this.zepClientPromise;
if (!zepClient) {
throw new Error(this.zepInitFailMsg);
}
try {
await zepClient.memory.deleteMemory(this.sessionId);
}
catch (error) {
console.error("Error deleting session: ", error);
}
// Clear the superclass's chat history
await super.clear();
}
}