mirror of
https://github.com/ollama/ollama.git
synced 2025-05-11 02:16:36 +02:00
272 lines
5.9 KiB
Go
272 lines
5.9 KiB
Go
package server
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"log/slog"
|
|
"net/http"
|
|
"os"
|
|
"slices"
|
|
"strings"
|
|
gotmpl "text/template"
|
|
"text/template/parse"
|
|
|
|
"github.com/ollama/ollama/api"
|
|
"github.com/ollama/ollama/fs/ggml"
|
|
"github.com/ollama/ollama/template"
|
|
"github.com/ollama/ollama/types/model"
|
|
)
|
|
|
|
var intermediateBlobs map[string]string = make(map[string]string)
|
|
|
|
type layerGGML struct {
|
|
Layer
|
|
*ggml.GGML
|
|
}
|
|
|
|
func parseFromModel(ctx context.Context, name model.Name, fn func(api.ProgressResponse)) (layers []*layerGGML, err error) {
|
|
m, err := ParseNamedManifest(name)
|
|
switch {
|
|
case errors.Is(err, os.ErrNotExist):
|
|
if err := PullModel(ctx, name.String(), ®istryOptions{}, fn); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
m, err = ParseNamedManifest(name)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
case err != nil:
|
|
return nil, err
|
|
}
|
|
|
|
for _, layer := range m.Layers {
|
|
layer, err := NewLayerFromLayer(layer.Digest, layer.MediaType, name.DisplayShortest())
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
switch layer.MediaType {
|
|
case "application/vnd.ollama.image.model",
|
|
"application/vnd.ollama.image.projector",
|
|
"application/vnd.ollama.image.adapter":
|
|
blobpath, err := GetBlobsPath(layer.Digest)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
blob, err := os.Open(blobpath)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer blob.Close()
|
|
|
|
f, _, err := ggml.Decode(blob, 1024)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
layers = append(layers, &layerGGML{layer, f})
|
|
default:
|
|
layers = append(layers, &layerGGML{layer, nil})
|
|
}
|
|
}
|
|
|
|
return layers, nil
|
|
}
|
|
|
|
func detectChatTemplate(layers []*layerGGML) ([]*layerGGML, error) {
|
|
for _, layer := range layers {
|
|
if s := layer.GGML.KV().ChatTemplate(); s != "" {
|
|
if t, err := template.Named(s); err != nil {
|
|
slog.Debug("template detection", "error", err, "template", s)
|
|
} else {
|
|
layer, err := NewLayer(t.Reader(), "application/vnd.ollama.image.template")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
layer.status = fmt.Sprintf("using autodetected template %s", t.Name)
|
|
layers = append(layers, &layerGGML{layer, nil})
|
|
|
|
if t.Parameters != nil {
|
|
var b bytes.Buffer
|
|
if err := json.NewEncoder(&b).Encode(t.Parameters); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
layer, err := NewLayer(&b, "application/vnd.ollama.image.params")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
layers = append(layers, &layerGGML{layer, nil})
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return layers, nil
|
|
}
|
|
|
|
func detectContentType(r io.Reader) (string, error) {
|
|
var b bytes.Buffer
|
|
if _, err := io.Copy(&b, r); err != nil {
|
|
return "", err
|
|
}
|
|
|
|
if contentType := ggml.DetectContentType(b.Bytes()); contentType != "" {
|
|
return contentType, nil
|
|
}
|
|
|
|
if contentType := http.DetectContentType(b.Bytes()); contentType != "application/octet-stream" {
|
|
return contentType, nil
|
|
}
|
|
|
|
return "unknown", nil
|
|
}
|
|
|
|
// extractToolCallsTemplate finds the immediate following text after any IfNode containing ".ToolCalls"
|
|
func extractToolCallsTemplate(tmpl *gotmpl.Template) (string, bool) {
|
|
if tmpl == nil || tmpl.Tree == nil {
|
|
slog.Debug("TextAfterToolCalls: template or tree is nil")
|
|
return "", false
|
|
}
|
|
|
|
var result string
|
|
var found bool
|
|
|
|
var walk func(nodes []parse.Node)
|
|
walk = func(nodes []parse.Node) {
|
|
for _, node := range nodes {
|
|
if found {
|
|
return
|
|
}
|
|
|
|
switch n := node.(type) {
|
|
case *parse.IfNode:
|
|
if nodeContainsToolCalls(n) {
|
|
// Collect immediate TextNode(s) at start of IfNode's list
|
|
var sb strings.Builder
|
|
for _, innerNode := range n.List.Nodes {
|
|
if tn, ok := innerNode.(*parse.TextNode); ok {
|
|
sb.Write(tn.Text)
|
|
} else {
|
|
// Stop at first non-text node
|
|
break
|
|
}
|
|
}
|
|
result = sb.String()
|
|
found = true
|
|
return
|
|
}
|
|
// Recurse into child nodes
|
|
walk(n.List.Nodes)
|
|
if n.ElseList != nil {
|
|
walk(n.ElseList.Nodes)
|
|
}
|
|
case *parse.ListNode:
|
|
walk(n.Nodes)
|
|
case *parse.RangeNode:
|
|
walk(n.List.Nodes)
|
|
if n.ElseList != nil {
|
|
walk(n.ElseList.Nodes)
|
|
}
|
|
case *parse.WithNode:
|
|
walk(n.List.Nodes)
|
|
if n.ElseList != nil {
|
|
walk(n.ElseList.Nodes)
|
|
}
|
|
default:
|
|
// Continue to next node
|
|
continue
|
|
}
|
|
|
|
if found {
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
walk(tmpl.Tree.Root.Nodes)
|
|
return result, found
|
|
}
|
|
|
|
// Helper to detect if a node's condition includes ".ToolCalls"
|
|
func nodeContainsToolCalls(n *parse.IfNode) bool {
|
|
for _, cmd := range n.Pipe.Cmds {
|
|
for _, arg := range cmd.Args {
|
|
if field, ok := arg.(*parse.FieldNode); ok {
|
|
if slices.Contains(field.Ident, "ToolCalls") {
|
|
return true
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
func ToolPrefix2(tmpl *gotmpl.Template) (string, bool) {
|
|
tokenText, ok := extractToolCallsTemplate(tmpl)
|
|
if !ok {
|
|
return "", false
|
|
}
|
|
tokenText = strings.TrimSpace(tokenText)
|
|
return tokenText, true
|
|
}
|
|
|
|
func ToolPrefix(tmpl *gotmpl.Template) (string, bool) {
|
|
tokenText, ok := extractToolCallsTemplate(tmpl)
|
|
if !ok {
|
|
return "", false
|
|
}
|
|
tokenText = strings.TrimSpace(tokenText)
|
|
if tokenText == "" {
|
|
return "", false
|
|
}
|
|
first := strings.Fields(tokenText)[0]
|
|
|
|
start := -1
|
|
end := -1
|
|
for i, r := range tokenText {
|
|
if r == '<' || r == '[' {
|
|
start = i
|
|
}
|
|
if (r == '>' || r == ']') && start != -1 {
|
|
end = i
|
|
break
|
|
}
|
|
}
|
|
if start != -1 && end != -1 {
|
|
// return the token including the [ or < and the ] or >
|
|
return tokenText[start : end+1], true
|
|
} else if start != -1 {
|
|
// get until the [ or < - in the case tag was not closed
|
|
return tokenText[:start], true
|
|
} else if end != -1 {
|
|
// get after the ] or > - in the case tag was not opened
|
|
return tokenText[end+1:], true
|
|
}
|
|
return first, true
|
|
}
|
|
|
|
func ToolTemplate(m *Model) (*gotmpl.Template, bool) {
|
|
// create a subtree from the node that ranges over .ToolCalls
|
|
tmpl := m.Template.Subtree(func(n parse.Node) bool {
|
|
if t, ok := n.(*parse.RangeNode); ok {
|
|
return slices.Contains(template.Identifiers(t.Pipe), "ToolCalls")
|
|
}
|
|
|
|
return false
|
|
})
|
|
|
|
if tmpl == nil {
|
|
return nil, false
|
|
}
|
|
|
|
return tmpl, true
|
|
}
|