#nlp #language-model #agent #openai #conversational #abstraction #memory

myself

Myself is a Rust library for building conversational agents powered by OpenAI's language model, providing a simple Agent abstraction to manage interactions and memory

8 releases (4 breaking)

0.5.0 Apr 27, 2023
0.4.0 Mar 6, 2023
0.3.0 Mar 3, 2023
0.2.4 Mar 3, 2023
0.1.0 Mar 2, 2023

#989 in Machine learning

MIT license

48KB
1K SLoC

Myself

Myself is a Rust library for building conversational agents powered by OpenAI's language model. It provides a simple Agent abstraction to manage interactions and memory, making it easy to create natural language interfaces for various applications.

⚠️ Warning

Please note that Myself is currently under development and is not yet suitable for production use. While you are welcome to try it out and provide feedback, we caution that it may have an incomplete implementation and may not function as intended. Our top priorities at this time are to improve the documentation and complete the list of features.

Features

  • Simple Agent abstraction.
  • Manage interactions and memory.
  • Support SQLite database.
  • Support GPT-3 OpenAI's language model.
  • Support other SQL databases (e.g. PostgreSQL and MySQL).
  • Support other language models (we plan to add support for other open source large language models).
  • Improve documentation, add more examples.

Installation

Add the following to your Cargo.toml file:

[dependencies]
myself = "0.1.0"

Or use cargo:

cargo add myself

Example Usage

You need to set the OPENAI_API_KEY environment variable to your OpenAI API key. You can get one here.

use dotenvy::dotenv;
use myself::agent_builder::AgentBuilder;

#[tokio::main]
async fn main() {
    dotenv().ok();

    let mut agent = AgentBuilder::new()
        .name("Bob".to_string())
        .build()
        .await;

    let message = "Hello World".to_string();
    let response = agent.interact_default(&message).await.unwrap();

    println!("{}", response);
    // Hello there! How can I help you?
}

You can run the other examples in the examples directory. Only clone the repository and run one of minimal, complex, graphql or neural-linux examples:

cargo run --example <example_name>

How it works

The main idea behind Myself is to provide a simple abstraction for building conversational agents. The Agent abstraction manages interactions and memory, making it easy to create natural language interfaces for various applications.

An Agent is a individual that can interact with the world. It has a name, a personality (a.k.a. constitution), and a memory. The name is used to address the agent, the personality is used to generate responses, and the memory is used to keep track of the context of the conversation.

In myself the Agent abstraction is implemented by the Agent struct. The Agent struct is responsible for managing interactions and memory. It is also responsible for generating responses using the OpenAI language model. The Agent struct is initialized with a AgentBuilder struct, which is used to configure the agent.

For example, if you want to create an agent named Bob, you can do the following:

use myself::agent_builder::AgentBuilder;

#[tokio::main]
async fn main() {
    let mut agent = AgentBuilder::new()
        .name("Bob".to_string())
        .build()
        .await;
}

For example, you can set the constitution and the default user name of the agent:

use myself::agent_builder::AgentBuilder;

#[tokio::main]
async fn main() {
    let mut agent = AgentBuilder::new()
        .name("Linux Server".to_string())
        .default_user_name("User".to_string())
        .default_constitution("I want you to act as a linux terminal. I will type commands and you will reply with what the terminal should show. I want you to only reply with the terminal output inside one unique code block, and nothing else. do not write explanations. do not type commands unless I instruct you to do so. When I need to tell you something in English, I will do so by putting text inside curly brackets {like this}.".into())
        .default_memory_size(50)
        .build()
        .await;
}

In the example above, we set the name of the agent to Linux Server, the default user name is User, the constitution to a string that describes the personality of the agent, and the default memory size to 50.

The Agent provides a methods to interact in different instances. For example, you can init (create) a new interaction with the agent:

use myself::agent_builder::AgentBuilder;

#[tokio::main]
async fn main() {
    let mut agent = AgentBuilder::new()
            .name("AI (Agent)".to_string())
            .build()
            .await;

    let interaction = agent
        .init_interaction(
            "Joe (Human)".to_string(),
            "A talkative chatbot conversation".to_string(),
            40,
        )
        .await;
}

And you can interact with it using the interact method:

use myself::agent_builder::AgentBuilder;

#[tokio::main]
async fn main() {
    let mut agent = AgentBuilder::new()
            .name("AI (Agent)".to_string())
            .build()
            .await;

    let interaction = agent
        .init_interaction(
            "Joe (Human)".to_string(),
            "A talkative chatbot conversation".to_string(),
            40,
        )
        .await;

    let message = "How are you?, explain please".to_string();
    let response = agent.interact_with(interaction.id, &message).await.unwrap();

    println!("{}", response);
}

In the example above, we create a new interaction with the agent, and then we interact with it using the interact method. The interact method takes the interaction id and the message as arguments, and returns a response.

The Agent also provides a method to interact with the default interaction:

use myself::agent_builder::AgentBuilder;

#[tokio::main]
async fn main() {
    let mut agent = AgentBuilder::new()
            .name("AI (Agent)".to_string())
            .build()
            .await;

    let message = "How are you?, explain please".to_string();
    let response = agent.interact_default(&message).await.unwrap();

    println!("{}", response);
}

The Interaction structure have the following form:

struct Interaction {
    pub id: Uuid,
    pub created_at: FastDateTime,
    pub updated_at: FastDateTime,

    pub user_name: String,

    pub long_term_memory: String,
    pub short_term_memory: Option<String>,

    pub short_term_memory_size: usize,
}

The user_name field is used to address the user in the conversation. The long_term_memory field is used to store the constitution of the the agent for this interaction. The short_term_memory field is used to store the last messages of the conversation in a buffer represented by a string. The short_term_memory_size field is used to set the size of the buffer measured as the number of lines (separated by '\n') in the dynamic_memory.

Dependencies

~16–31MB
~503K SLoC