GoRag: A Go Library for Retrieval-Augmented Generation (RAG) Development with Multi-Vector Database Support 📜
GoRag provides an intuitive Go interface for developing Retrieval-Augmented Generation (RAG) applications. It supports multiple vector database types, enabling efficient data retrieval for enhanced generation augmentation.
Language Model backends including Ollama and OpenAI, along with an embeddings interface for RAG using a local embeddings model (mxbai-embed-large) or a hosted such type (such as text-embedding-ada-002 from OpenAI)
- Interact with Ollama & OpenAI: Generate responses from multiple AI backends.
- RAG / Embeddings Generation: Generate text embeddings store / load to a vector database for RAG.
- Multiple Vector Database Support: Currently Postgres with pgvector is supported, along with qdrant (others to follow, open an issue if you want to see something included).
gorag needs to be installed as a dependency in your project. You can do it by importing it in your codebase:
import "github.com/stackloklabs/gorag"
Then make sure that you have Go installed, and run:
go mod tidy
You'll need to have an Ollama server running and accessible.
Install Ollama Server: Download the server from the official Ollama website
Pull and run a model
ollama run llama3.2
Ollama should run on port 11434
and localhost
, if you change this, don't
forget to update your config.
You'll need an OpenAI API key to use the OpenAI backend.
Currently Postgres is supported, and the database should be created before
running the application, with the schema provided in db/init.sql
Should you prefer, the docker-compose will automate the setup of the database.
Best bet is to see /examples/*
for reference, this explains how to use
the library with examples for generation, embeddings and implementing RAG for pgvector or qdrant.
There are currently two backend systems supported, Ollama and OpenAI, with the ability to generate embeddings for RAG on both.
First create a Backend object
generationBackend := backend.NewOllamaBackend("http://localhost:11434", "llama3", time.Duration(10*time.Second))
Create a prompt
prompt := backend.NewPrompt().
AddMessage("system", "You are an AI assistant. Use the provided context to answer the user's question as accurately as possible.").
AddMessage("user", "What is love?").
SetParameters(backend.Parameters{
MaxTokens: 150,
Temperature: 0.7,
TopP: 0.9,
})
Call the Generations API
response, err := generationBackend.Generate(ctx, prompt)
if err != nil {
log.Fatalf("Failed to generate response: %v", err)
}
First create a Backend object
generationBackend = backend.NewOpenAIBackend("API_KEY", "gpt-3.5-turbo", 10*time.Second)
Create a prompt
prompt := backend.NewPrompt().
AddMessage("system", "You are an AI assistant. Use the provided context to answer the user's question as accurately as possible.").
AddMessage("user", "How much is too much?").
SetParameters(backend.Parameters{
MaxTokens: 150,
Temperature: 0.7,
TopP: 0.9,
FrequencyPenalty: 0.5,
PresencePenalty: 0.6,
})
Call the Generations API and get a response
response, err := generationBackend.Generate(ctx, prompt)
if err != nil {
log.Fatalf("Failed to generate response: %v", err)
}
To generate embeddings for RAG, you can use the Embeddings
interface in both
Ollama and OpenAI backends.
embedding, err := embeddingBackend.Embed(ctx, "Mickey mouse is a real human being")
if err != nil {
log.Fatalf("Error generating embedding: %v", err)
}
log.Println("Embedding generated")
// Embed the query using the specified embedding backend
queryEmbedding, err := embeddingBackend.Embed(ctx, query)
if err != nil {
log.Fatalf("Error generating query embedding: %v", err)
}
log.Println("Vector embeddings generated")
// Retrieve relevant documents for the query embedding
retrievedDocs, err := vectorDB.QueryRelevantDocuments(ctx, queryEmbedding, "ollama")
if err != nil {
log.Fatalf("Error retrieving relevant documents: %v", err)
}
// Log the retrieved documents to see if they include the inserted content
for _, doc := range retrievedDocs {
log.Printf("Retrieved Document: %v", doc)
}
// Augment the query with retrieved context
augmentedQuery := db.CombineQueryWithContext(query, retrievedDocs)
prompt := backend.NewPrompt().
AddMessage("system", "You are an AI assistant. Use the provided context to answer the user's question as accurately as possible.").
AddMessage("user", augmentedQuery).
SetParameters(backend.Parameters{
MaxTokens: 150, // Supported by LLaMa
Temperature: 0.7, // Supported by LLaMa
TopP: 0.9, // Supported by LLaMa
})
Example output:
2024/10/28 15:08:25 Embedding backend LLM: mxbai-embed-large
2024/10/28 15:08:25 Generation backend: llama3
2024/10/28 15:08:25 Vector database initialized
2024/10/28 15:08:26 Embedding generated
2024/10/28 15:08:26 Vector Document generated
2024/10/28 15:08:26 Vector embeddings generated
2024/10/28 15:08:26 Retrieved Document: {doc-5630d3f2-bf61-4e13-8ec9-9e863bc1a962 map[content:Mickey mouse is a real human being]}
2024/10/28 15:08:34 Retrieval-Augmented Generation influenced output from LLM model: Mickey Mouse is indeed a human!
We welcome contributions! Please submit a pull request or raise an issue if you want to see something included or hit a bug.