mirror of
https://github.com/mudler/LocalAI.git
synced 2024-06-07 19:40:48 +00:00
beb598e4f9
feat(functions): support mixed JSON BNF grammar This PR provides new options to control how functions are extracted from the LLM, and also provides more control on how JSON grammars can be used (also in conjunction). New YAML settings introduced: - `grammar_message`: when enabled, the generated grammar can also decide to push strings and not only JSON objects. This allows the LLM to pick to either respond freely or using JSON. - `grammar_prefix`: Allows to prefix a string to the JSON grammar definition. - `replace_results`: Is a map that allows to replace strings in the LLM result. As an example, consider the following settings for Hermes-2-Pro-Mistral, which allow extracting both JSON results coming from the model, and the ones coming from the grammar: ```yaml function: # disable injecting the "answer" tool disable_no_action: true # This allows the grammar to also return messages grammar_message: true # Suffix to add to the grammar grammar_prefix: '<tool_call>\n' return_name_in_function_response: true # Without grammar uncomment the lines below # Warning: this is relying only on the capability of the # LLM model to generate the correct function call. # no_grammar: true # json_regex_match: "(?s)<tool_call>(.*?)</tool_call>" replace_results: "<tool_call>": "" "\'": "\"" ``` Note: To disable entirely grammars usage in the example above, uncomment the `no_grammar` and `json_regex_match`. Signed-off-by: Ettore Di Giacinto <mudler@localai.io>
203 lines
6.9 KiB
Go
203 lines
6.9 KiB
Go
package functions
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"regexp"
|
|
"strings"
|
|
|
|
"github.com/go-skynet/LocalAI/pkg/utils"
|
|
"github.com/rs/zerolog/log"
|
|
)
|
|
|
|
// FunctionsConfig is the configuration for the tool/function call.
|
|
// It includes setting to map the function name and arguments from the response
|
|
// and, for instance, also if processing the requests with BNF grammars.
|
|
type FunctionsConfig struct {
|
|
// DisableNoAction disables the "no action" tool
|
|
// By default we inject a tool that does nothing and is used to return an answer from the LLM
|
|
DisableNoAction bool `yaml:"disable_no_action"`
|
|
|
|
// NoActionFunctionName is the name of the function that does nothing. It defaults to "answer"
|
|
NoActionFunctionName string `yaml:"no_action_function_name"`
|
|
|
|
// NoActionDescriptionName is the name of the function that returns the description of the no action function
|
|
NoActionDescriptionName string `yaml:"no_action_description_name"`
|
|
|
|
// ParallelCalls enables the LLM to return multiple function calls in the same response
|
|
ParallelCalls bool `yaml:"parallel_calls"`
|
|
|
|
// GrammarMessage enables the LLM to return strings and not only JSON objects
|
|
// This is useful for models to not constraing returning only JSON and also messages back to the user
|
|
GrammarMessage bool `yaml:"grammar_message"`
|
|
|
|
// NoGrammar disables the grammar parsing and parses the responses directly from the LLM
|
|
NoGrammar bool `yaml:"no_grammar"`
|
|
|
|
// ResponseRegex is a named regex to extract the function name and arguments from the response
|
|
ResponseRegex string `yaml:"response_regex"`
|
|
|
|
// JSONRegexMatch is a regex to extract the JSON object from the response
|
|
JSONRegexMatch string `yaml:"json_regex_match"`
|
|
|
|
// GrammarPrefix is the suffix to append to the grammar when being generated
|
|
// This is useful when models prepend a tag before returning JSON
|
|
GrammarPrefix string `yaml:"grammar_prefix"`
|
|
|
|
// ReplaceResults allow to replace strings in the results before parsing them
|
|
ReplaceResults map[string]string `yaml:"replace_results"`
|
|
|
|
// FunctionName enable the LLM to return { "name": "function_name", "arguments": { "arg1": "value1", "arg2": "value2" } }
|
|
// instead of { "function": "function_name", "arguments": { "arg1": "value1", "arg2": "value2" } }.
|
|
// This might be useful for certain models trained with the function name as the first token.
|
|
FunctionName bool `yaml:"return_name_in_function_response"`
|
|
}
|
|
|
|
type FuncCallResults struct {
|
|
Name string
|
|
Arguments string
|
|
}
|
|
|
|
func ParseFunctionCall(llmresult string, functionConfig FunctionsConfig) []FuncCallResults {
|
|
log.Debug().Msgf("LLM result: %s", llmresult)
|
|
|
|
for k, v := range functionConfig.ReplaceResults {
|
|
log.Debug().Msgf("Replacing %s with %s", k, v)
|
|
llmresult = strings.ReplaceAll(llmresult, k, v)
|
|
}
|
|
|
|
log.Debug().Msgf("LLM result(processed): %s", llmresult)
|
|
|
|
multipleResults := functionConfig.ParallelCalls
|
|
useGrammars := !functionConfig.NoGrammar
|
|
|
|
functionNameKey := "function"
|
|
if functionConfig.FunctionName {
|
|
functionNameKey = "name"
|
|
}
|
|
|
|
results := []FuncCallResults{}
|
|
|
|
returnResult := func(s string) (name, arguments string, e error) {
|
|
// As we have to change the result before processing, we can't stream the answer token-by-token (yet?)
|
|
ss := map[string]interface{}{}
|
|
// This prevent newlines to break JSON parsing for clients
|
|
s = utils.EscapeNewLines(s)
|
|
err := json.Unmarshal([]byte(s), &ss)
|
|
if err != nil {
|
|
log.Warn().Err(err).Str("escapedLLMResult", s).Msg("unable to unmarshal llm result")
|
|
}
|
|
log.Debug().Msgf("Function return: %s %+v", s, ss)
|
|
|
|
// The grammar defines the function name as "function", while OpenAI returns "name"
|
|
func_name, ok := ss[functionNameKey]
|
|
if !ok {
|
|
return "", "", fmt.Errorf("unable to find function name in result")
|
|
}
|
|
// Similarly, while here arguments is a map[string]interface{}, OpenAI actually want a stringified object
|
|
args, ok := ss["arguments"] // arguments needs to be a string, but we return an object from the grammar result (TODO: fix)
|
|
if !ok {
|
|
return "", "", fmt.Errorf("unable to find arguments in result")
|
|
}
|
|
d, _ := json.Marshal(args)
|
|
funcName, ok := func_name.(string)
|
|
if !ok {
|
|
return "", "", fmt.Errorf("unable to cast function name to string")
|
|
}
|
|
|
|
return funcName, string(d), nil
|
|
}
|
|
|
|
// if no grammar is used, we have to extract function and arguments from the result
|
|
if !useGrammars {
|
|
// the response is a string that we have to parse
|
|
result := make(map[string]string)
|
|
|
|
if functionConfig.ResponseRegex != "" {
|
|
// We use named regexes here to extract the function name and arguments
|
|
// obviously, this expects the LLM to be stable and return correctly formatted JSON
|
|
// TODO: optimize this and pre-compile it
|
|
var respRegex = regexp.MustCompile(functionConfig.ResponseRegex)
|
|
match := respRegex.FindStringSubmatch(llmresult)
|
|
for i, name := range respRegex.SubexpNames() {
|
|
if i != 0 && name != "" && len(match) > i {
|
|
result[name] = match[i]
|
|
}
|
|
}
|
|
|
|
// TODO: open point about multiple results and/or mixed with chat messages
|
|
// This is not handled as for now, we only expect one function call per response
|
|
functionName := result[functionNameKey]
|
|
if functionName == "" {
|
|
return results
|
|
}
|
|
} else if functionConfig.JSONRegexMatch != "" {
|
|
//re := regexp.MustCompile(`(?s)<tool_call>(.*?)</tool_call>`)
|
|
//m:= re.FindStringSubmatch(`<tool_call>{ foo barr }</tool_call>`)
|
|
|
|
// We use a regex to extract the JSON object from the response
|
|
var respRegex = regexp.MustCompile(functionConfig.JSONRegexMatch)
|
|
match := respRegex.FindStringSubmatch(llmresult)
|
|
if len(match) < 2 {
|
|
return results
|
|
}
|
|
|
|
funcName, args, err := returnResult(match[1])
|
|
if err != nil {
|
|
return results
|
|
}
|
|
|
|
return append(results, FuncCallResults{Name: funcName, Arguments: args})
|
|
|
|
} else {
|
|
|
|
funcName, args, err := returnResult(llmresult)
|
|
if err != nil {
|
|
return results
|
|
}
|
|
|
|
return append(results, FuncCallResults{Name: funcName, Arguments: args})
|
|
}
|
|
|
|
return append(results, FuncCallResults{Name: result[functionNameKey], Arguments: result["arguments"]})
|
|
}
|
|
|
|
// with grammars
|
|
// TODO: use generics to avoid this code duplication
|
|
if multipleResults {
|
|
ss := []map[string]interface{}{}
|
|
s := utils.EscapeNewLines(llmresult)
|
|
err := json.Unmarshal([]byte(s), &ss)
|
|
if err != nil {
|
|
log.Warn().Err(err).Str("escapedLLMResult", s).Msg("multiple results: unable to unmarshal llm result")
|
|
}
|
|
log.Debug().Msgf("Function return: %s %+v", s, ss)
|
|
|
|
for _, s := range ss {
|
|
func_name, ok := s[functionNameKey]
|
|
if !ok {
|
|
continue
|
|
}
|
|
args, ok := s["arguments"]
|
|
if !ok {
|
|
continue
|
|
}
|
|
d, _ := json.Marshal(args)
|
|
funcName, ok := func_name.(string)
|
|
if !ok {
|
|
continue
|
|
}
|
|
results = append(results, FuncCallResults{Name: funcName, Arguments: string(d)})
|
|
}
|
|
} else {
|
|
funcName, args, err := returnResult(llmresult)
|
|
if err != nil {
|
|
return results
|
|
}
|
|
|
|
results = append(results, FuncCallResults{Name: funcName, Arguments: args})
|
|
}
|
|
|
|
return results
|
|
}
|