diff --git a/core/http/endpoints/openai/chat.go b/core/http/endpoints/openai/chat.go index 2b0b10a89b4e..ccbf094667d4 100644 --- a/core/http/endpoints/openai/chat.go +++ b/core/http/endpoints/openai/chat.go @@ -81,7 +81,7 @@ func ChatEndpoint(cl *config.BackendConfigLoader, ml *model.ModelLoader, startup } responses <- initialMessage - result, err := handleQuestion(config, req, ml, startupOptions, results, prompt) + result, err := handleQuestion(config, req, ml, startupOptions, results, result, prompt) if err != nil { log.Error().Err(err).Msg("error handling question") return @@ -470,7 +470,7 @@ func ChatEndpoint(cl *config.BackendConfigLoader, ml *model.ModelLoader, startup switch { case noActionsToRun: - result, err := handleQuestion(config, input, ml, startupOptions, results, predInput) + result, err := handleQuestion(config, input, ml, startupOptions, results, s, predInput) if err != nil { log.Error().Err(err).Msg("error handling question") return @@ -550,7 +550,14 @@ func ChatEndpoint(cl *config.BackendConfigLoader, ml *model.ModelLoader, startup } } -func handleQuestion(config *config.BackendConfig, input *schema.OpenAIRequest, ml *model.ModelLoader, o *config.ApplicationConfig, funcResults []functions.FuncCallResults, prompt string) (string, error) { +func handleQuestion(config *config.BackendConfig, input *schema.OpenAIRequest, ml *model.ModelLoader, o *config.ApplicationConfig, funcResults []functions.FuncCallResults, result, prompt string) (string, error) { + + if len(funcResults) == 0 && result != "" { + log.Debug().Msgf("nothing function results but we had a message from the LLM") + + return result, nil + } + log.Debug().Msgf("nothing to do, computing a reply") arg := "" if len(funcResults) > 0 { diff --git a/pkg/functions/parse.go b/pkg/functions/parse.go index 304f336f2fd5..c6941ff690b0 100644 --- a/pkg/functions/parse.go +++ b/pkg/functions/parse.go @@ -2,6 +2,7 @@ package functions import ( "encoding/json" + "fmt" "regexp" "github.com/go-skynet/LocalAI/pkg/utils" @@ -16,6 +17,8 @@ type FunctionsConfig struct { NoGrammar bool `yaml:"no_grammar"` ResponseRegex string `yaml:"response_regex"` + JSONRegexMatch string `yaml:"json_regex_match"` + // 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. @@ -38,6 +41,36 @@ func ParseFunctionCall(llmresult string, functionConfig FunctionsConfig) []FuncC 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.Error().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 @@ -61,13 +94,32 @@ func ParseFunctionCall(llmresult string, functionConfig FunctionsConfig) []FuncC if functionName == "" { return results } + } else if functionConfig.JSONRegexMatch != "" { + //re := regexp.MustCompile(`(?s)(.*?)`) + //m:= re.FindStringSubmatch(`{ foo barr }`) + + // 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 { - // We expect the result to be a JSON object with a function name and arguments - err := json.Unmarshal([]byte(llmresult), &result) + + funcName, args, err := returnResult(llmresult) if err != nil { - log.Error().Err(err).Str("llmresult", llmresult).Msg("unable to unmarshal llm result") return results } + + return append(results, FuncCallResults{Name: funcName, Arguments: args}) } return append(results, FuncCallResults{Name: result[functionNameKey], Arguments: result["arguments"]}) @@ -101,32 +153,12 @@ func ParseFunctionCall(llmresult string, functionConfig FunctionsConfig) []FuncC results = append(results, FuncCallResults{Name: funcName, Arguments: string(d)}) } } else { - // 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(llmresult) - err := json.Unmarshal([]byte(s), &ss) + funcName, args, err := returnResult(llmresult) if err != nil { - log.Error().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 results } - // 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 results - } - d, _ := json.Marshal(args) - funcName, ok := func_name.(string) - if !ok { - return results - } - results = append(results, FuncCallResults{Name: funcName, Arguments: string(d)}) + + results = append(results, FuncCallResults{Name: funcName, Arguments: args}) } return results diff --git a/pkg/functions/parse_test.go b/pkg/functions/parse_test.go index 4c2208ed31f7..7aedc0978796 100644 --- a/pkg/functions/parse_test.go +++ b/pkg/functions/parse_test.go @@ -87,7 +87,7 @@ var _ = Describe("LocalAI function parse tests", func() { It("should parse the function name and arguments correctly with the name key", func() { input := `{"name": "add", "arguments": {"x": 5, "y": 3}}` functionConfig.ParallelCalls = false - functionConfig.NoGrammar = false + functionConfig.NoGrammar = true functionConfig.ResponseRegex = "" functionConfig.FunctionName = true @@ -100,7 +100,40 @@ var _ = Describe("LocalAI function parse tests", func() { It("should parse the function name and arguments correctly with the function key", func() { input := `{"function": "add", "arguments": {"x": 5, "y": 3}}` functionConfig.ParallelCalls = false - functionConfig.NoGrammar = false + functionConfig.NoGrammar = true + functionConfig.ResponseRegex = "" + functionConfig.FunctionName = false + + results := ParseFunctionCall(input, functionConfig) + Expect(results).To(HaveLen(1)) + Expect(results[0].Name).To(Equal("add")) + Expect(results[0].Arguments).To(Equal(`{"x":5,"y":3}`)) + }) + + It("Should parse the result by matching the JSONRegexMatch", func() { + input := ` + +{"function": "add", "arguments": {"x": 5, "y": 3}} +` + functionConfig.ParallelCalls = false + functionConfig.NoGrammar = true + functionConfig.JSONRegexMatch = `(?s)(.*?)` + functionConfig.ResponseRegex = "" + functionConfig.FunctionName = false + + results := ParseFunctionCall(input, functionConfig) + Expect(results).To(HaveLen(1)) + Expect(results[0].Name).To(Equal("add")) + Expect(results[0].Arguments).To(Equal(`{"x":5,"y":3}`)) + }) + + It("Should parse the result by matching the JSONRegexMatch", func() { + input := ` +{"function": "add", "arguments": {"x": 5, "y": 3}} +` + functionConfig.ParallelCalls = false + functionConfig.NoGrammar = true + functionConfig.JSONRegexMatch = `(?s)(.*?)` functionConfig.ResponseRegex = "" functionConfig.FunctionName = false