blob: 738a37a606f5d29d7efca70dee35237fdbf863e8 [file] [log] [blame]
// Copyright 2024 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
// Package ollama implements access to offline Ollama model.
//
// [Client] implements [llm.Embedder]. Use [NewClient] to connect.
package ollama
import (
"bytes"
"context"
"encoding/json"
"fmt"
"io"
"log/slog"
"net/http"
"net/url"
"os"
"slices"
"golang.org/x/oscar/internal/llm"
)
// NOTE: This package does not use third party packages for
// querying ollama models to avoid bringing in their many dependencies.
// A Client represents a connection to Ollama.
type Client struct {
slog *slog.Logger
hc *http.Client
url *url.URL // url of the ollama server
}
// NewClient returns a connection to Ollama server. If empty, the
// server is assumed to be hosted at http://127.0.0.1:11434.
func NewClient(lg *slog.Logger, hc *http.Client, server string) (*Client, error) {
if server == "" {
host := os.Getenv("OLLAMA_HOST")
if host == "" {
host = "127.0.0.1"
}
server = "http://" + host + ":11434"
}
u, err := url.Parse(server)
if err != nil {
return nil, err
}
return &Client{slog: lg, hc: hc, url: u}, nil
}
const maxBatch = 512 // default physical batch size in ollama
// EmbedDocs returns the vector embeddings for the docs,
// implementing [llm.Embedder].
func (c *Client) EmbedDocs(ctx context.Context, docs []llm.EmbedDoc) ([]llm.Vector, error) {
embedURL := c.url.JoinPath("/api/embed") // ollama embed endpoint
var vecs []llm.Vector
for docs := range slices.Chunk(docs, maxBatch) {
var inputs []string
for _, doc := range docs {
// ollama does not support adding content with title
input := doc.Title + "\n\n" + doc.Text
inputs = append(inputs, input)
}
vs, err := embed(ctx, c.hc, embedURL, inputs)
if err != nil {
return nil, err
}
vecs = append(vecs, vs...)
}
return vecs, nil
}
func embed(ctx context.Context, hc *http.Client, embedURL *url.URL, inputs []string) ([]llm.Vector, error) {
embReq := struct {
Model string `json:"model"`
Input []string `json:"input"`
}{
Model: "mxbai-embed-large", // use largest ollama embedding model
Input: inputs,
}
erj, err := json.Marshal(embReq)
if err != nil {
return nil, err
}
request, err := http.NewRequestWithContext(ctx, http.MethodPost, embedURL.String(), bytes.NewReader(erj))
if err != nil {
return nil, err
}
request.Header.Set("Content-Type", "application/json")
request.Header.Set("Accept", "application/json")
response, err := hc.Do(request)
if err != nil {
return nil, err
}
defer response.Body.Close()
embResp, err := io.ReadAll(response.Body)
if err != nil {
return nil, err
}
if err := embedError(response, embResp); err != nil {
return nil, err
}
return embeddings(embResp)
}
// embedError extracts error from ollama's response, if any.
func embedError(resp *http.Response, embResp []byte) error {
if resp.StatusCode == 200 {
return nil
}
if resp.StatusCode == 400 {
var e struct {
Error string `json:"error"`
}
// ollama returns JSON with error field set for bad requests.
if err := json.Unmarshal(embResp, &e); err != nil {
return err
}
return fmt.Errorf("ollama response error: %s", e.Error)
}
return fmt.Errorf("ollama response error: %s", resp.Status)
}
func embeddings(embResp []byte) ([]llm.Vector, error) {
// In case there are no errors, ollama returns
// a JSON with Embeddings field set.
var e struct {
Embeddings []llm.Vector `json:"embeddings"`
}
if err := json.Unmarshal(embResp, &e); err != nil {
return nil, err
}
return e.Embeddings, nil
}