mirror of
https://github.com/ollama/ollama.git
synced 2025-05-11 02:16:36 +02:00
checkpoint for vscode
This commit is contained in:
parent
128c90d3ac
commit
b4cd1118ab
2 changed files with 101 additions and 203 deletions
279
server/model.go
279
server/model.go
|
@ -154,109 +154,99 @@ func parseObjects(s string) []map[string]any {
|
||||||
return objs
|
return objs
|
||||||
}
|
}
|
||||||
|
|
||||||
// parseToolCalls attempts to parse a JSON string into a slice of ToolCalls.
|
// Get tool call token from model template
|
||||||
// mxyng: this only really works if the input contains tool calls in some JSON format
|
func (m *Model) TemplateToolToken() (string, string, bool) {
|
||||||
func (m *Model) parseToolCalls(s string) ([]api.ToolCall, bool) {
|
// Try to detect the tool call format from the model's template
|
||||||
// create a subtree from the node that ranges over .ToolCalls
|
|
||||||
tmpl := m.Template.Subtree(func(n parse.Node) bool {
|
tmpl := m.Template.Subtree(func(n parse.Node) bool {
|
||||||
if t, ok := n.(*parse.RangeNode); ok {
|
if t, ok := n.(*parse.RangeNode); ok {
|
||||||
return slices.Contains(template.Identifiers(t.Pipe), "ToolCalls")
|
return slices.Contains(template.Identifiers(t.Pipe), "ToolCalls")
|
||||||
}
|
}
|
||||||
|
|
||||||
return false
|
return false
|
||||||
})
|
})
|
||||||
|
|
||||||
if tmpl == nil {
|
// fmt.Println("tool call template", tmpl)
|
||||||
slog.Debug("parseToolCalls: no ToolCalls template found")
|
if tmpl != nil {
|
||||||
return nil, false
|
// Execute template with test data to see the format
|
||||||
}
|
var b bytes.Buffer
|
||||||
|
if err := tmpl.Execute(&b, map[string][]api.ToolCall{
|
||||||
slog.Debug("parseToolCalls: executing template with test data", "input", s)
|
"ToolCalls": {
|
||||||
|
{
|
||||||
var b bytes.Buffer
|
Function: api.ToolCallFunction{
|
||||||
if err := tmpl.Execute(&b, map[string][]api.ToolCall{
|
Name: "function_name",
|
||||||
"ToolCalls": {
|
Arguments: api.ToolCallFunctionArguments{
|
||||||
{
|
"argument1": "value1",
|
||||||
Function: api.ToolCallFunction{
|
// "argument2": "value2",
|
||||||
Name: "@@name@@",
|
},
|
||||||
Arguments: api.ToolCallFunctionArguments{
|
|
||||||
"@@argument@@": 1,
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
}); err == nil {
|
||||||
}); err != nil {
|
// Look for special tokens in the template output
|
||||||
slog.Debug("parseToolCalls: template execution failed", "error", err)
|
output := strings.TrimSpace(b.String())
|
||||||
return nil, false
|
slog.Debug("tool call template output", "output", output)
|
||||||
}
|
if strings.Contains(output, "<") {
|
||||||
|
// Extract the special token between < and >
|
||||||
slog.Debug("parseToolCalls: template executed successfully", "output", b.String())
|
start := strings.Index(output, "<")
|
||||||
|
end := strings.Index(output, ">")
|
||||||
templateObjects := parseObjects(b.String())
|
if start >= 0 && end > start {
|
||||||
if len(templateObjects) == 0 {
|
token := output[start : end+1]
|
||||||
return nil, false
|
return output, token, true
|
||||||
}
|
}
|
||||||
|
} else if strings.Contains(output, "[") {
|
||||||
slog.Debug("parseToolCalls: template objects", "objects", templateObjects)
|
// Check if it's a tool call token rather than JSON array
|
||||||
|
start := strings.Index(output, "[")
|
||||||
// find the keys that correspond to the name and arguments fields
|
end := strings.Index(output, "]")
|
||||||
var name, arguments string
|
if start >= 0 && end > start {
|
||||||
for k, v := range templateObjects[0] {
|
token := output[start : end+1]
|
||||||
switch v.(type) {
|
// Only consider it a token if it's not valid JSON
|
||||||
case string:
|
var jsonTest any
|
||||||
name = k
|
if err := json.Unmarshal([]byte(token), &jsonTest); err != nil {
|
||||||
case map[string]any:
|
return output, token, true
|
||||||
arguments = k
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
if name == "" || arguments == "" {
|
|
||||||
return nil, false
|
|
||||||
}
|
|
||||||
|
|
||||||
responseObjects := parseObjects(s)
|
|
||||||
if len(responseObjects) == 0 {
|
|
||||||
return nil, false
|
|
||||||
}
|
|
||||||
|
|
||||||
// collect all nested objects
|
|
||||||
var collect func(any) []map[string]any
|
|
||||||
collect = func(obj any) (all []map[string]any) {
|
|
||||||
switch o := obj.(type) {
|
|
||||||
case map[string]any:
|
|
||||||
all = append(all, o)
|
|
||||||
for _, v := range o {
|
|
||||||
all = append(all, collect(v)...)
|
|
||||||
}
|
|
||||||
case []any:
|
|
||||||
for _, v := range o {
|
|
||||||
all = append(all, collect(v)...)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return all
|
|
||||||
}
|
}
|
||||||
|
return "", "", false
|
||||||
|
}
|
||||||
|
|
||||||
var objs []map[string]any
|
func parsePythonFunctionCall(s string) ([]api.ToolCall, bool) {
|
||||||
for _, p := range responseObjects {
|
re := regexp.MustCompile(`(\w+)\((.*?)\)`)
|
||||||
objs = append(objs, collect(p)...)
|
matches := re.FindAllStringSubmatchIndex(s, -1)
|
||||||
|
if len(matches) == 0 {
|
||||||
|
return nil, false
|
||||||
}
|
}
|
||||||
|
|
||||||
var toolCalls []api.ToolCall
|
var toolCalls []api.ToolCall
|
||||||
for _, kv := range objs {
|
for _, match := range matches {
|
||||||
n, nok := kv[name].(string)
|
name := s[match[2]:match[3]]
|
||||||
a, aok := kv[arguments].(map[string]any)
|
args := s[match[4]:match[5]]
|
||||||
if nok && aok {
|
|
||||||
|
arguments := make(api.ToolCallFunctionArguments)
|
||||||
|
if strings.Contains(args, "=") { // Keyword args
|
||||||
|
pairs := strings.SplitSeq(args, ",")
|
||||||
|
for pair := range pairs {
|
||||||
|
pair = strings.TrimSpace(pair)
|
||||||
|
kv := strings.Split(pair, "=")
|
||||||
|
if len(kv) == 2 {
|
||||||
|
key := strings.TrimSpace(kv[0])
|
||||||
|
value := strings.TrimSpace(kv[1])
|
||||||
|
arguments[key] = value
|
||||||
|
}
|
||||||
|
}
|
||||||
toolCalls = append(toolCalls, api.ToolCall{
|
toolCalls = append(toolCalls, api.ToolCall{
|
||||||
Function: api.ToolCallFunction{
|
Function: api.ToolCallFunction{
|
||||||
Name: n,
|
Name: name,
|
||||||
Arguments: a,
|
Arguments: arguments,
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return toolCalls, len(toolCalls) > 0
|
if len(toolCalls) > 0 {
|
||||||
|
return toolCalls, true
|
||||||
|
}
|
||||||
|
return nil, false
|
||||||
}
|
}
|
||||||
|
|
||||||
// ToolCallFormat represents different possible formats for tool calls
|
// ToolCallFormat represents different possible formats for tool calls
|
||||||
|
@ -377,100 +367,6 @@ func parseJSONToolCalls(obj map[string]any) ([]api.ToolCall, bool) {
|
||||||
return nil, false
|
return nil, false
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *Model) GetToolCallFormat() (string, string, bool) {
|
|
||||||
// Try to detect the tool call format from the model's template
|
|
||||||
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
|
|
||||||
})
|
|
||||||
|
|
||||||
fmt.Println("tool call template", tmpl)
|
|
||||||
if tmpl != nil {
|
|
||||||
// Execute template with test data to see the format
|
|
||||||
var b bytes.Buffer
|
|
||||||
if err := tmpl.Execute(&b, map[string][]api.ToolCall{
|
|
||||||
"ToolCalls": {
|
|
||||||
{
|
|
||||||
Function: api.ToolCallFunction{
|
|
||||||
Name: "function_name",
|
|
||||||
Arguments: api.ToolCallFunctionArguments{
|
|
||||||
"argument1": "value1",
|
|
||||||
// "argument2": "value2",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}); err == nil {
|
|
||||||
// Look for special tokens in the template output
|
|
||||||
output := strings.TrimSpace(b.String())
|
|
||||||
slog.Debug("tool call template output", "output", output)
|
|
||||||
if strings.Contains(output, "<") {
|
|
||||||
// Extract the special token between < and >
|
|
||||||
start := strings.Index(output, "<")
|
|
||||||
end := strings.Index(output, ">")
|
|
||||||
if start >= 0 && end > start {
|
|
||||||
token := output[start : end+1]
|
|
||||||
return output, token, true
|
|
||||||
}
|
|
||||||
} else if strings.Contains(output, "[") {
|
|
||||||
// Check if it's a tool call token rather than JSON array
|
|
||||||
start := strings.Index(output, "[")
|
|
||||||
end := strings.Index(output, "]")
|
|
||||||
if start >= 0 && end > start {
|
|
||||||
token := output[start : end+1]
|
|
||||||
// Only consider it a token if it's not valid JSON
|
|
||||||
var jsonTest any
|
|
||||||
if err := json.Unmarshal([]byte(token), &jsonTest); err != nil {
|
|
||||||
return output, token, true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return "", "", false
|
|
||||||
}
|
|
||||||
|
|
||||||
func parsePythonFunctionCall(s string) ([]api.ToolCall, bool) {
|
|
||||||
re := regexp.MustCompile(`(\w+)\((.*?)\)`)
|
|
||||||
matches := re.FindAllStringSubmatchIndex(s, -1)
|
|
||||||
if len(matches) == 0 {
|
|
||||||
return nil, false
|
|
||||||
}
|
|
||||||
|
|
||||||
var toolCalls []api.ToolCall
|
|
||||||
for _, match := range matches {
|
|
||||||
name := s[match[2]:match[3]]
|
|
||||||
args := s[match[4]:match[5]]
|
|
||||||
|
|
||||||
arguments := make(api.ToolCallFunctionArguments)
|
|
||||||
if strings.Contains(args, "=") { // Keyword args
|
|
||||||
pairs := strings.SplitSeq(args, ",")
|
|
||||||
for pair := range pairs {
|
|
||||||
pair = strings.TrimSpace(pair)
|
|
||||||
kv := strings.Split(pair, "=")
|
|
||||||
if len(kv) == 2 {
|
|
||||||
key := strings.TrimSpace(kv[0])
|
|
||||||
value := strings.TrimSpace(kv[1])
|
|
||||||
arguments[key] = value
|
|
||||||
}
|
|
||||||
}
|
|
||||||
toolCalls = append(toolCalls, api.ToolCall{
|
|
||||||
Function: api.ToolCallFunction{
|
|
||||||
Name: name,
|
|
||||||
Arguments: arguments,
|
|
||||||
},
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(toolCalls) > 0 {
|
|
||||||
return toolCalls, true
|
|
||||||
}
|
|
||||||
return nil, false
|
|
||||||
}
|
|
||||||
|
|
||||||
// token, partial, success
|
// token, partial, success
|
||||||
func deriveToolToken(s string, prefix string) (string, bool, bool) {
|
func deriveToolToken(s string, prefix string) (string, bool, bool) {
|
||||||
// There shouldn't be spaces in a tool token
|
// There shouldn't be spaces in a tool token
|
||||||
|
@ -488,27 +384,21 @@ func deriveToolToken(s string, prefix string) (string, bool, bool) {
|
||||||
|
|
||||||
func parseJSON(s string) ([]api.ToolCall, bool) {
|
func parseJSON(s string) ([]api.ToolCall, bool) {
|
||||||
objs := parseObjects(s)
|
objs := parseObjects(s)
|
||||||
var toolCalls []api.ToolCall
|
tcs := []api.ToolCall{}
|
||||||
for _, obj := range objs {
|
for _, obj := range objs {
|
||||||
if n, nok := obj["name"].(string); nok {
|
toolCalls, ok := parseJSONToolCalls(obj)
|
||||||
if a, aok := obj["arguments"].(map[string]any); aok {
|
if ok {
|
||||||
toolCalls = append(toolCalls, api.ToolCall{
|
tcs = append(tcs, toolCalls...)
|
||||||
Function: api.ToolCallFunction{
|
|
||||||
Name: n,
|
|
||||||
Arguments: a,
|
|
||||||
},
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if len(toolCalls) > 0 {
|
if len(tcs) > 0 {
|
||||||
return toolCalls, true
|
return tcs, true
|
||||||
}
|
}
|
||||||
return nil, false
|
return nil, false
|
||||||
}
|
}
|
||||||
|
|
||||||
// returns tool calls, partial, success
|
// returns tool calls, partial, success
|
||||||
func (m *Model) ParseToolCallsNew(s string, toolToken *string) ([]api.ToolCall, bool, bool) {
|
func (m *Model) ParseToolCalls(s string, toolToken *string) ([]api.ToolCall, bool, bool) {
|
||||||
// [ case can either be JSON, Python or a Tool Token
|
// [ case can either be JSON, Python or a Tool Token
|
||||||
s = strings.TrimSpace(s)
|
s = strings.TrimSpace(s)
|
||||||
fmt.Printf("ParseToolCallsNew input: %q\n", s)
|
fmt.Printf("ParseToolCallsNew input: %q\n", s)
|
||||||
|
@ -539,7 +429,7 @@ func (m *Model) ParseToolCallsNew(s string, toolToken *string) ([]api.ToolCall,
|
||||||
}
|
}
|
||||||
// Tool Token Case - this is okay if it's a real tool token and we couldn't get from template
|
// Tool Token Case - this is okay if it's a real tool token and we couldn't get from template
|
||||||
fmt.Println("Attempting to derive tool token")
|
fmt.Println("Attempting to derive tool token")
|
||||||
if toolToken == nil || (toolToken != nil && *toolToken == "") {
|
if toolToken == nil || *toolToken == "" {
|
||||||
toolTok, partial, ok := deriveToolToken(s, "[")
|
toolTok, partial, ok := deriveToolToken(s, "[")
|
||||||
if !ok {
|
if !ok {
|
||||||
return nil, false, false
|
return nil, false, false
|
||||||
|
@ -552,21 +442,26 @@ func (m *Model) ParseToolCallsNew(s string, toolToken *string) ([]api.ToolCall,
|
||||||
fmt.Printf("Found tool token: %q\n", *toolToken)
|
fmt.Printf("Found tool token: %q\n", *toolToken)
|
||||||
s = strings.TrimSpace(s[len(*toolToken):])
|
s = strings.TrimSpace(s[len(*toolToken):])
|
||||||
fmt.Printf("Recursing with remaining string: %q\n", s)
|
fmt.Printf("Recursing with remaining string: %q\n", s)
|
||||||
if toolCalls, partial, ok := m.ParseToolCallsNew(s, toolToken); ok {
|
if toolCalls, partial, ok := m.ParseToolCalls(s, toolToken); ok {
|
||||||
return toolCalls, partial, true
|
return toolCalls, partial, true
|
||||||
}
|
}
|
||||||
return nil, true, true
|
return nil, true, true
|
||||||
} else if strings.HasPrefix(s, "{") || strings.HasPrefix(s, "```") {
|
} else if strings.HasPrefix(s, "{") || strings.HasPrefix(s, "```") {
|
||||||
fmt.Println("Found { prefix - attempting JSON parse")
|
// // TODO: temp fix
|
||||||
|
// if strings.HasPrefix(s, "```") && len(s) == 3 {
|
||||||
|
// return nil, false, false
|
||||||
|
// }
|
||||||
|
fmt.Println("Found { prefix - attempting JSON parse with ", s)
|
||||||
if calls, ok := parseJSON(s); ok {
|
if calls, ok := parseJSON(s); ok {
|
||||||
fmt.Printf("Successfully parsed JSON object, found %d calls\n", len(calls))
|
fmt.Printf("Successfully parsed JSON object, found %d calls\n", len(calls))
|
||||||
return calls, false, true
|
return calls, false, true
|
||||||
}
|
}
|
||||||
|
fmt.Println("Failed to parse JSON in JSON case")
|
||||||
// TODO: possible case where it never finishes parsing - then what?
|
// TODO: possible case where it never finishes parsing - then what?
|
||||||
return nil, true, true
|
return nil, true, true
|
||||||
} else if strings.HasPrefix(s, "<") {
|
} else if strings.HasPrefix(s, "<") {
|
||||||
fmt.Println("Found < prefix - attempting to derive tool token")
|
fmt.Println("Found < prefix - attempting to derive tool token")
|
||||||
if toolToken == nil || (toolToken != nil && *toolToken == "") {
|
if toolToken == nil || *toolToken == "" {
|
||||||
toolTok, partial, ok := deriveToolToken(s, "<")
|
toolTok, partial, ok := deriveToolToken(s, "<")
|
||||||
if !ok {
|
if !ok {
|
||||||
return nil, false, false
|
return nil, false, false
|
||||||
|
@ -575,12 +470,12 @@ func (m *Model) ParseToolCallsNew(s string, toolToken *string) ([]api.ToolCall,
|
||||||
return nil, true, true
|
return nil, true, true
|
||||||
}
|
}
|
||||||
*toolToken = toolTok
|
*toolToken = toolTok
|
||||||
fmt.Printf("Found tool token: %q\n", toolToken)
|
fmt.Printf("Found tool token: %q\n", *toolToken)
|
||||||
}
|
}
|
||||||
fmt.Printf("Found tool token: %q\n", *toolToken)
|
fmt.Printf("Found tool token: %q\n", *toolToken)
|
||||||
s = strings.TrimSpace(s[len(*toolToken):])
|
s = strings.TrimSpace(s[len(*toolToken):])
|
||||||
fmt.Printf("Recursing with remaining string: %q\n", s)
|
fmt.Printf("Recursing with remaining string: %q\n", s)
|
||||||
if toolCalls, partial, ok := m.ParseToolCallsNew(s, toolToken); ok {
|
if toolCalls, partial, ok := m.ParseToolCalls(s, toolToken); ok {
|
||||||
return toolCalls, partial, true
|
return toolCalls, partial, true
|
||||||
}
|
}
|
||||||
return nil, true, true
|
return nil, true, true
|
||||||
|
|
|
@ -1529,8 +1529,8 @@ func (s *Server) ChatHandler(c *gin.Context) {
|
||||||
var sentWithTools int = 0
|
var sentWithTools int = 0
|
||||||
// var prefix string
|
// var prefix string
|
||||||
// var templateToolToken string
|
// var templateToolToken string
|
||||||
_, templateToolToken, _ := m.GetToolCallFormat()
|
_, templateToolToken, _ := m.TemplateToolToken()
|
||||||
fmt.Println("special token", templateToolToken)
|
// fmt.Println("special token", templateToolToken)
|
||||||
|
|
||||||
var minDuration time.Duration = math.MaxInt64
|
var minDuration time.Duration = math.MaxInt64
|
||||||
var maxDuration time.Duration
|
var maxDuration time.Duration
|
||||||
|
@ -1562,9 +1562,9 @@ func (s *Server) ChatHandler(c *gin.Context) {
|
||||||
slog.Debug("total duration", "duration", totalDuration)
|
slog.Debug("total duration", "duration", totalDuration)
|
||||||
slog.Debug("check count", "count", checkCount)
|
slog.Debug("check count", "count", checkCount)
|
||||||
// slog.Debug("average duration", "duration", totalDuration/time.Duration(checkCount))
|
// slog.Debug("average duration", "duration", totalDuration/time.Duration(checkCount))
|
||||||
if sb.Len() > 0 {
|
// if sb.Len() > 0 {
|
||||||
res.Message.Content = sb.String()
|
// res.Message.Content = sb.String()
|
||||||
}
|
// }
|
||||||
res.DoneReason = r.DoneReason.String()
|
res.DoneReason = r.DoneReason.String()
|
||||||
res.TotalDuration = time.Since(checkpointStart)
|
res.TotalDuration = time.Since(checkpointStart)
|
||||||
res.LoadDuration = checkpointLoaded.Sub(checkpointStart)
|
res.LoadDuration = checkpointLoaded.Sub(checkpointStart)
|
||||||
|
@ -1582,12 +1582,10 @@ func (s *Server) ChatHandler(c *gin.Context) {
|
||||||
// If tools are recognized, use a flag to track the sending of a tool downstream
|
// If tools are recognized, use a flag to track the sending of a tool downstream
|
||||||
// This ensures that content is cleared from the message on the last chunk sent
|
// This ensures that content is cleared from the message on the last chunk sent
|
||||||
sb.WriteString(r.Content)
|
sb.WriteString(r.Content)
|
||||||
// TODO: here we want to prefix check the tool ideally or derive the tool token from the model
|
|
||||||
// TODO: if we are deriving the tool token, then a heuristic must be applied to stream eventually
|
|
||||||
// TODO: if the prefix check fails, send the content downstream and reset the builder
|
|
||||||
startTime := time.Now()
|
startTime := time.Now()
|
||||||
|
// TODO: work max tool tok logic
|
||||||
if len(req.Tools) > 0 && sentWithTools < maxToolTokens {
|
if len(req.Tools) > 0 && sentWithTools < maxToolTokens {
|
||||||
toolCalls, partial, ok := m.ParseToolCallsNew(sb.String(), &templateToolToken)
|
toolCalls, partial, ok := m.ParseToolCalls(sb.String(), &templateToolToken)
|
||||||
duration := time.Since(startTime)
|
duration := time.Since(startTime)
|
||||||
checkCount++
|
checkCount++
|
||||||
minDuration = min(minDuration, duration)
|
minDuration = min(minDuration, duration)
|
||||||
|
@ -1600,6 +1598,7 @@ func (s *Server) ChatHandler(c *gin.Context) {
|
||||||
// If the tool call is partial, we need to wait for the next chunk
|
// If the tool call is partial, we need to wait for the next chunk
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
slog.Debug("toolCalls", "toolCalls", toolCalls, "partial", partial, "ok", ok)
|
||||||
res.Message.ToolCalls = toolCalls
|
res.Message.ToolCalls = toolCalls
|
||||||
for i := range toolCalls {
|
for i := range toolCalls {
|
||||||
toolCalls[i].Function.Index = toolCallIndex
|
toolCalls[i].Function.Index = toolCallIndex
|
||||||
|
@ -1611,6 +1610,9 @@ func (s *Server) ChatHandler(c *gin.Context) {
|
||||||
res.Message.Content = ""
|
res.Message.Content = ""
|
||||||
sb.Reset()
|
sb.Reset()
|
||||||
ch <- res
|
ch <- res
|
||||||
|
// TODO: revisit this
|
||||||
|
sentWithTools++
|
||||||
|
slog.Debug("fired on tool call", "toolCalls", toolCalls, "toolCallIndex", toolCallIndex)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1634,15 +1636,16 @@ func (s *Server) ChatHandler(c *gin.Context) {
|
||||||
const MAX_TOOL_TOKENS = 1
|
const MAX_TOOL_TOKENS = 1
|
||||||
sentWithTools := 0
|
sentWithTools := 0
|
||||||
var tb strings.Builder
|
var tb strings.Builder
|
||||||
_, templateToolToken, _ := m.GetToolCallFormat()
|
_, templateToolToken, _ := m.TemplateToolToken()
|
||||||
for rr := range ch {
|
for rr := range ch {
|
||||||
switch t := rr.(type) {
|
switch t := rr.(type) {
|
||||||
case api.ChatResponse:
|
case api.ChatResponse:
|
||||||
sb.WriteString(t.Message.Content)
|
sb.WriteString(t.Message.Content)
|
||||||
resp = t
|
resp = t
|
||||||
|
// TODO: work max tool tok logic
|
||||||
if len(req.Tools) > 0 && sentWithTools < MAX_TOOL_TOKENS {
|
if len(req.Tools) > 0 && sentWithTools < MAX_TOOL_TOKENS {
|
||||||
tb.WriteString(t.Message.Content)
|
tb.WriteString(t.Message.Content)
|
||||||
if tcs, partial, ok := m.ParseToolCallsNew(tb.String(), &templateToolToken); ok {
|
if tcs, partial, ok := m.ParseToolCalls(tb.String(), &templateToolToken); ok {
|
||||||
if !partial {
|
if !partial {
|
||||||
// resp.Message.ToolCalls = toolCalls
|
// resp.Message.ToolCalls = toolCalls
|
||||||
toolCalls = append(toolCalls, tcs...)
|
toolCalls = append(toolCalls, tcs...)
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue