2023-08-30 16:35:03 -04:00
|
|
|
package llm
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"context"
|
2023-11-29 11:00:37 -08:00
|
|
|
_ "embed"
|
2023-08-30 16:35:03 -04:00
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"os/exec"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/jmorganca/ollama/api"
|
2023-10-13 14:45:50 -07:00
|
|
|
"github.com/jmorganca/ollama/format"
|
2023-08-30 16:35:03 -04:00
|
|
|
)
|
|
|
|
|
2023-11-09 16:44:02 -08:00
|
|
|
const jsonGrammar = `
|
|
|
|
root ::= object
|
|
|
|
value ::= object | array | string | number | ("true" | "false" | "null") ws
|
|
|
|
|
|
|
|
object ::=
|
|
|
|
"{" ws (
|
|
|
|
string ":" ws value
|
|
|
|
("," ws string ":" ws value)*
|
|
|
|
)? "}" ws
|
|
|
|
|
|
|
|
array ::=
|
|
|
|
"[" ws (
|
|
|
|
value
|
|
|
|
("," ws value)*
|
|
|
|
)? "]" ws
|
|
|
|
|
|
|
|
string ::=
|
|
|
|
"\"" (
|
|
|
|
[^"\\] |
|
|
|
|
"\\" (["\\/bfnrt] | "u" [0-9a-fA-F] [0-9a-fA-F] [0-9a-fA-F] [0-9a-fA-F]) # escapes
|
|
|
|
)* "\"" ws
|
|
|
|
|
|
|
|
number ::= ("-"? ([0-9] | [1-9] [0-9]*)) ("." [0-9]+)? ([eE] [-+]? [0-9]+)? ws
|
|
|
|
|
|
|
|
# Optional space: by convention, applied in this grammar after literal chars when allowed
|
|
|
|
ws ::= ([ \t\n] ws)?
|
|
|
|
`
|
|
|
|
|
2023-08-30 16:35:03 -04:00
|
|
|
type Running struct {
|
2023-10-18 15:36:56 -04:00
|
|
|
Port int
|
|
|
|
Cmd *exec.Cmd
|
|
|
|
Cancel context.CancelFunc
|
2024-01-08 16:42:00 -05:00
|
|
|
*StatusWriter // captures error messages from the llama runner process
|
2023-08-30 16:35:03 -04:00
|
|
|
}
|
|
|
|
|
2023-12-11 13:56:22 -08:00
|
|
|
type ImageData struct {
|
|
|
|
Data []byte `json:"data"`
|
|
|
|
ID int `json:"id"`
|
|
|
|
}
|
|
|
|
|
2023-11-09 13:16:16 -08:00
|
|
|
var (
|
2023-11-18 21:24:59 -05:00
|
|
|
errNvidiaSMI = errors.New("warning: gpu support may not be enabled, check that you have installed GPU drivers: nvidia-smi command failed")
|
2023-11-09 13:16:16 -08:00
|
|
|
errAvailableVRAM = errors.New("not enough VRAM available, falling back to CPU only")
|
2023-12-20 10:36:01 -08:00
|
|
|
payloadMissing = fmt.Errorf("expected dynamic library payloads not included in this build of ollama")
|
2023-11-09 13:16:16 -08:00
|
|
|
)
|
2023-09-12 11:04:35 -04:00
|
|
|
|
2023-10-12 11:16:37 -04:00
|
|
|
// StatusWriter is a writer that captures error messages from the llama runner process
|
|
|
|
type StatusWriter struct {
|
2023-10-18 15:36:56 -04:00
|
|
|
ErrCh chan error
|
|
|
|
LastErrMsg string
|
2023-10-12 11:16:37 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
func NewStatusWriter() *StatusWriter {
|
|
|
|
return &StatusWriter{
|
|
|
|
ErrCh: make(chan error, 1),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *StatusWriter) Write(b []byte) (int, error) {
|
2023-10-18 15:36:56 -04:00
|
|
|
var errMsg string
|
2023-10-12 11:16:37 -04:00
|
|
|
if _, after, ok := bytes.Cut(b, []byte("error:")); ok {
|
2023-10-18 15:36:56 -04:00
|
|
|
errMsg = string(bytes.TrimSpace(after))
|
|
|
|
} else if _, after, ok := bytes.Cut(b, []byte("CUDA error")); ok {
|
|
|
|
errMsg = string(bytes.TrimSpace(after))
|
2023-10-12 11:16:37 -04:00
|
|
|
}
|
2023-10-18 15:36:56 -04:00
|
|
|
|
|
|
|
if errMsg != "" {
|
|
|
|
w.LastErrMsg = errMsg
|
|
|
|
w.ErrCh <- fmt.Errorf("llama runner: %s", errMsg)
|
|
|
|
}
|
|
|
|
|
2023-10-12 11:16:37 -04:00
|
|
|
return os.Stderr.Write(b)
|
|
|
|
}
|
|
|
|
|
2023-10-16 16:31:29 -07:00
|
|
|
type prediction struct {
|
2023-09-03 17:46:35 -04:00
|
|
|
Content string `json:"content"`
|
|
|
|
Model string `json:"model"`
|
|
|
|
Prompt string `json:"prompt"`
|
|
|
|
Stop bool `json:"stop"`
|
|
|
|
|
2023-10-16 16:31:29 -07:00
|
|
|
Timings struct {
|
|
|
|
PredictedN int `json:"predicted_n"`
|
|
|
|
PredictedMS float64 `json:"predicted_ms"`
|
|
|
|
PromptN int `json:"prompt_n"`
|
|
|
|
PromptMS float64 `json:"prompt_ms"`
|
|
|
|
}
|
2023-08-30 16:35:03 -04:00
|
|
|
}
|
|
|
|
|
2023-10-12 09:34:16 -07:00
|
|
|
const maxBufferSize = 512 * format.KiloByte
|
2023-11-29 11:00:37 -08:00
|
|
|
const maxRetries = 3
|
|
|
|
const retryDelay = 1 * time.Second
|
2023-10-04 14:09:00 -04:00
|
|
|
|
2023-12-05 14:57:33 -05:00
|
|
|
type PredictOpts struct {
|
2024-01-03 12:01:42 -05:00
|
|
|
Prompt string
|
|
|
|
Format string
|
|
|
|
Images []api.ImageData
|
|
|
|
Options api.Options
|
2023-12-05 14:57:33 -05:00
|
|
|
}
|
2023-12-04 21:16:27 -08:00
|
|
|
|
2023-12-05 14:57:33 -05:00
|
|
|
type PredictResult struct {
|
|
|
|
Content string
|
|
|
|
Done bool
|
|
|
|
PromptEvalCount int
|
|
|
|
PromptEvalDuration time.Duration
|
|
|
|
EvalCount int
|
|
|
|
EvalDuration time.Duration
|
|
|
|
}
|
2023-09-03 14:10:03 -04:00
|
|
|
|
2023-08-30 16:35:03 -04:00
|
|
|
type TokenizeRequest struct {
|
|
|
|
Content string `json:"content"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type TokenizeResponse struct {
|
|
|
|
Tokens []int `json:"tokens"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type DetokenizeRequest struct {
|
|
|
|
Tokens []int `json:"tokens"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type DetokenizeResponse struct {
|
|
|
|
Content string `json:"content"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type EmbeddingRequest struct {
|
|
|
|
Content string `json:"content"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type EmbeddingResponse struct {
|
|
|
|
Embedding []float64 `json:"embedding"`
|
|
|
|
}
|