A small library in TypeScript for managing different "personas" for OpenAI Chat API.
Use the Persona
class to manage chat context and trim history without losing memories of the persona itself.
npm install gpt-persona
This library includes three features, with Persona
being the main feature.
- A class
Persona
for managing different personas for OpenAI Chat API. - A thin wrapper
OpenAI
around OpenAI Chat API.- You may use other libraries such as
openai
, but this wrapper supports streaming.
- You may use other libraries such as
- A small chat application called
gpt-chat
to quickly test with different personas.
This library is not ready for production, but you may use this to experiment with a chatbot.
import {Persona, OpenAI} from 'gpt-persona';
const persona = new Persona("You end every sentence with '-desu' or '-nya'.");
const api = new OpenAI(process.env.OPENAI_API_KEY);
// Method 1: provide the API object
console.log(await persona.respond(api, "Good morning!"));
// Example response: "Good morning desu!"
// Method 2: use the API directly (with this, you can use other API libraries)
persona.pushMessage("Good afternoon!");
const response = await api.chatCompletion(persona.getAPIMessages());
persona.pushResponse(response);
console.log(response);
// Example response: "Good afternoon nya!"
usage: gpt-chat [-h] [-k KEY] [-p PERSONA_FILE]
A simple client for OpenAI's Chat API.
optional arguments:
-h, --help show this help message and exit
-k KEY, --key KEY API key for OpenAI. If not provided, then the environment variable `OPENAI_API_KEY` will be used.
-p PERSONA_FILE, --persona PERSONA_FILE
A path for the persona JSON/text file.
You can use following commands:
/quit
: Quit chatting./persona
: Override the persona./instruction
: Override the instruction./reset
: Resets the history and persona./clear
: Resets the history. Retains persona and instructions./save
: Save current persona and history.
Start your message with //
if you want to put /
at the beginning.
A Persona
represents a chat agent. It holds memories of who it is and past conversations.
The constructor accepts either the persona (Persona#persona
) or an object (see Persona#toJSON
).
import {Persona} from 'gpt-persona';
const persona = new Persona("... enter detailed persona for this chat agent ...");
Every request to the OpenAI API will be started with this. Default role is system
.
// Could be a string, a `Message`, or a list of those.
persona.persona = "You are a cat.";
Every request to the OpenAI API will be ended with this. Default role is system
.
It's usually better to provide instructions as Persona#persona
and leave Persona#instructions
empty.
# of tokens for current context (== this.persona_token_count + this.history_token_count + this.instruction_token_count
).
Maximum allowed # of tokens for the context.
Create a response given a message; it automatically rollbacks upon an error, so even when an invocation throws an error, the persona can be reused safely.
api
message
options
(optional)- An optional object with following optional fields.
options.request_params
- Same parameters for OpenAI's API, but all fields are optional.
options.additional_instructions
- Additional instructions to be provided.
- Consider it as one-time
persona.instruction
.
options.freeze_history
- History will be not modified when this is set to true.
options.condenser
:(persona: Persona) => void
- The way the history is condensed can be customized. (See
Persona#condense()
for a bit more details.)
- The way the history is condensed can be customized. (See
options.isEphemeral
:(messages: Message[]) => boolean
- Given the persona's responses, if this function returns
true
, then the history will not be updated.
- Given the persona's responses, if this function returns
deltaCallback
:(delta: string) => void
(optional)- An optional function, if provided, will be called for every incremental message.
Here's an example with streaming:
import { stdout } from 'node:process';
import { Persona, OpenAI } from "../index.js";
const api = new OpenAI(process.env.OPENAI_API_KEY);
const persona = new Persona("You are a cat.");
persona.instructions = "You have to talk like a cat while responding.";
const response = await persona.respond(api, "Who are you?", {}, function (delta) { stdout.write(delta); });
// Example response: "Meow, I am just a curious cat. How may I assist you?"
Condenses the history. Currently, it simply wipes out past history which does not fit in max_context_token_count
.
Following alternatives may be used (and provided to options.condenser
above, for example):
- Wipe out random messages.
- Use ellipses to shorten messages.
- Use a summarizer (could be implemented with yet another
Persona
instance) to summarize the history.
Use persona.history = ...
, and do not modify history
in-situ like persona.history.splice(...)
.
Use the exported countTokens
method to count tokens of message(s).
Use persona.token_count - persona.max_context_token_count
to check how much must be trimmed from the history.
Clears history, but retain persona and instructions.
Gives an object which contains all necessary information for the persona.