From 0a2de6010146e5cf0e7da0879cca6766c21507eb Mon Sep 17 00:00:00 2001 From: Simone Vellei Date: Wed, 29 May 2024 10:00:07 +0200 Subject: [PATCH 01/10] Feat add tools (#201) --- examples/llm/openai/thread/main.go | 24 ++-- examples/llm/openai/tools/python/main.go | 32 +++++ examples/llm/openai/tools/rag/main.go | 66 +++++++++ examples/tools/duckduckgo/main.go | 15 ++ examples/tools/python/main.go | 16 +++ examples/tools/serpapi/main.go | 15 ++ examples/tools/shell/main.go | 16 +++ go.mod | 1 + go.sum | 2 + llm/openai/function.go | 19 +++ tools/dalle/dalle.go | 56 ++++++++ tools/duckduckgo/api.go | 168 +++++++++++++++++++++++ tools/duckduckgo/duckduckgo.go | 85 ++++++++++++ tools/llm/llm.go | 68 +++++++++ tools/python/python.go | 68 +++++++++ tools/rag/rag.go | 62 +++++++++ tools/serpapi/api.go | 119 ++++++++++++++++ tools/serpapi/serpapi.go | 98 +++++++++++++ tools/shell/shell.go | 91 ++++++++++++ tools/tool_router/tool_router.go | 84 ++++++++++++ 20 files changed, 1093 insertions(+), 12 deletions(-) create mode 100644 examples/llm/openai/tools/python/main.go create mode 100644 examples/llm/openai/tools/rag/main.go create mode 100644 examples/tools/duckduckgo/main.go create mode 100644 examples/tools/python/main.go create mode 100644 examples/tools/serpapi/main.go create mode 100644 examples/tools/shell/main.go create mode 100644 tools/dalle/dalle.go create mode 100644 tools/duckduckgo/api.go create mode 100644 tools/duckduckgo/duckduckgo.go create mode 100644 tools/llm/llm.go create mode 100644 tools/python/python.go create mode 100644 tools/rag/rag.go create mode 100644 tools/serpapi/api.go create mode 100644 tools/serpapi/serpapi.go create mode 100644 tools/shell/shell.go create mode 100644 tools/tool_router/tool_router.go diff --git a/examples/llm/openai/thread/main.go b/examples/llm/openai/thread/main.go index 5180349c..695f4714 100644 --- a/examples/llm/openai/thread/main.go +++ b/examples/llm/openai/thread/main.go @@ -2,11 +2,12 @@ package main import ( "context" + "encoding/json" "fmt" - "strings" "github.com/henomis/lingoose/llm/openai" "github.com/henomis/lingoose/thread" + "github.com/henomis/lingoose/tools/dalle" "github.com/henomis/lingoose/transformer" ) @@ -32,15 +33,7 @@ func newStr(str string) *string { func main() { openaillm := openai.New().WithModel(openai.GPT4o) - openaillm.WithToolChoice(newStr("auto")) - err := openaillm.BindFunction( - crateImage, - "createImage", - "use this function to create an image from a description", - ) - if err != nil { - panic(err) - } + openaillm.WithToolChoice(newStr("auto")).WithTools(dalle.New()) t := thread.New().AddMessage( thread.NewUserMessage().AddContent( @@ -48,15 +41,22 @@ func main() { ), ) - err = openaillm.Generate(context.Background(), t) + err := openaillm.Generate(context.Background(), t) if err != nil { panic(err) } if t.LastMessage().Role == thread.RoleTool { + var output dalle.Output + + err = json.Unmarshal([]byte(t.LastMessage().Contents[0].AsToolResponseData().Result), &output) + if err != nil { + panic(err) + } + t.AddMessage(thread.NewUserMessage().AddContent( thread.NewImageContentFromURL( - strings.ReplaceAll(t.LastMessage().Contents[0].AsToolResponseData().Result, `"`, ""), + output.ImageURL, ), ).AddContent( thread.NewTextContent("can you describe the image?"), diff --git a/examples/llm/openai/tools/python/main.go b/examples/llm/openai/tools/python/main.go new file mode 100644 index 00000000..1e133410 --- /dev/null +++ b/examples/llm/openai/tools/python/main.go @@ -0,0 +1,32 @@ +package main + +import ( + "context" + "fmt" + + "github.com/henomis/lingoose/llm/openai" + "github.com/henomis/lingoose/thread" + "github.com/henomis/lingoose/tools/python" +) + +func main() { + newStr := func(str string) *string { + return &str + } + llm := openai.New().WithModel(openai.GPT3Dot5Turbo0613).WithToolChoice(newStr("auto")).WithTools( + python.New(), + ) + + t := thread.New().AddMessage( + thread.NewUserMessage().AddContent( + thread.NewTextContent("calculate reverse string of 'ailatiditalia', don't try to guess, let's use appropriate tool"), + ), + ) + + llm.Generate(context.Background(), t) + if t.LastMessage().Role == thread.RoleTool { + llm.Generate(context.Background(), t) + } + + fmt.Println(t) +} diff --git a/examples/llm/openai/tools/rag/main.go b/examples/llm/openai/tools/rag/main.go new file mode 100644 index 00000000..29deadc9 --- /dev/null +++ b/examples/llm/openai/tools/rag/main.go @@ -0,0 +1,66 @@ +package main + +import ( + "context" + "fmt" + "os" + + openaiembedder "github.com/henomis/lingoose/embedder/openai" + "github.com/henomis/lingoose/index" + "github.com/henomis/lingoose/index/vectordb/jsondb" + "github.com/henomis/lingoose/llm/openai" + "github.com/henomis/lingoose/rag" + "github.com/henomis/lingoose/thread" + ragtool "github.com/henomis/lingoose/tools/rag" + "github.com/henomis/lingoose/tools/serpapi" + "github.com/henomis/lingoose/tools/shell" +) + +func main() { + + rag := rag.New( + index.New( + jsondb.New().WithPersist("index.json"), + openaiembedder.New(openaiembedder.AdaEmbeddingV2), + ), + ).WithChunkSize(1000).WithChunkOverlap(0) + + _, err := os.Stat("index.json") + if os.IsNotExist(err) { + err = rag.AddSources(context.Background(), "state_of_the_union.txt") + if err != nil { + panic(err) + } + } + + newStr := func(str string) *string { + return &str + } + llm := openai.New().WithModel(openai.GPT4o).WithToolChoice(newStr("auto")).WithTools( + ragtool.New(rag, "US covid vaccines"), + serpapi.New(), + shell.New(), + ) + + topics := []string{ + "how many covid vaccine doses US has donated to other countries.", + "who's the author of LinGoose github project.", + "which process is consuming the most memory.", + } + + for _, topic := range topics { + t := thread.New().AddMessage( + thread.NewUserMessage().AddContent( + thread.NewTextContent("Please tell me " + topic), + ), + ) + + llm.Generate(context.Background(), t) + if t.LastMessage().Role == thread.RoleTool { + llm.Generate(context.Background(), t) + } + + fmt.Println(t) + } + +} diff --git a/examples/tools/duckduckgo/main.go b/examples/tools/duckduckgo/main.go new file mode 100644 index 00000000..7be5f65c --- /dev/null +++ b/examples/tools/duckduckgo/main.go @@ -0,0 +1,15 @@ +package main + +import ( + "fmt" + + "github.com/henomis/lingoose/tools/duckduckgo" +) + +func main() { + + t := duckduckgo.New().WithMaxResults(5) + f := t.Fn().(duckduckgo.FnPrototype) + + fmt.Println(f(duckduckgo.Input{Query: "Simone Vellei"})) +} diff --git a/examples/tools/python/main.go b/examples/tools/python/main.go new file mode 100644 index 00000000..57052fb6 --- /dev/null +++ b/examples/tools/python/main.go @@ -0,0 +1,16 @@ +package main + +import ( + "fmt" + + "github.com/henomis/lingoose/tools/python" +) + +func main() { + t := python.New().WithPythonPath("python3") + + pythonScript := `print("Hello from Python!")` + f := t.Fn().(python.FnPrototype) + + fmt.Println(f(python.Input{PythonCode: pythonScript})) +} diff --git a/examples/tools/serpapi/main.go b/examples/tools/serpapi/main.go new file mode 100644 index 00000000..d0578d3a --- /dev/null +++ b/examples/tools/serpapi/main.go @@ -0,0 +1,15 @@ +package main + +import ( + "fmt" + + "github.com/henomis/lingoose/tools/serpapi" +) + +func main() { + + t := serpapi.New() + f := t.Fn().(serpapi.FnPrototype) + + fmt.Println(f(serpapi.Input{Query: "Simone Vellei"})) +} diff --git a/examples/tools/shell/main.go b/examples/tools/shell/main.go new file mode 100644 index 00000000..3ec73377 --- /dev/null +++ b/examples/tools/shell/main.go @@ -0,0 +1,16 @@ +package main + +import ( + "fmt" + + "github.com/henomis/lingoose/tools/shell" +) + +func main() { + t := shell.New() + + bashScript := `echo "Hello from $SHELL!"` + f := t.Fn().(shell.FnPrototype) + + fmt.Println(f(shell.Input{BashScript: bashScript})) +} diff --git a/go.mod b/go.mod index 25c30a82..4131f84e 100644 --- a/go.mod +++ b/go.mod @@ -15,6 +15,7 @@ require ( github.com/henomis/restclientgo v1.2.0 github.com/invopop/jsonschema v0.7.0 github.com/sashabaranov/go-openai v1.24.0 + golang.org/x/net v0.25.0 ) require ( diff --git a/go.sum b/go.sum index e2d3391c..a79137ce 100644 --- a/go.sum +++ b/go.sum @@ -38,6 +38,8 @@ github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/ github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU= github.com/stretchr/testify v1.8.2 h1:+h33VjcLVPDHtOdpUCuF+7gSuG3yGIftsP1YvFihtJ8= github.com/stretchr/testify v1.8.2/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= +golang.org/x/net v0.25.0 h1:d/OCCoBEUq33pjydKrGQhw7IlUPI2Oylr+8qLx49kac= +golang.org/x/net v0.25.0/go.mod h1:JkAGAh7GEvH74S6FOH42FLoXpXbE/aqXSrIQjXgsiwM= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= diff --git a/llm/openai/function.go b/llm/openai/function.go index 4b9c77bb..9c5953bd 100644 --- a/llm/openai/function.go +++ b/llm/openai/function.go @@ -79,6 +79,25 @@ func (o *OpenAI) BindFunction( return nil } +type Tool interface { + Description() string + Name() string + Fn() any +} + +func (o OpenAI) WithTools(tools ...Tool) OpenAI { + for _, tool := range tools { + function, err := bindFunction(tool.Fn(), tool.Name(), tool.Description()) + if err != nil { + fmt.Println(err) + } + + o.functions[tool.Name()] = *function + } + + return o +} + func (o *Legacy) getFunctions() []openai.FunctionDefinition { var functions []openai.FunctionDefinition diff --git a/tools/dalle/dalle.go b/tools/dalle/dalle.go new file mode 100644 index 00000000..03e3cb5e --- /dev/null +++ b/tools/dalle/dalle.go @@ -0,0 +1,56 @@ +package dalle + +import ( + "context" + "fmt" + "time" + + "github.com/henomis/lingoose/transformer" +) + +const ( + defaultTimeoutInSeconds = 60 +) + +type Tool struct { +} + +type Input struct { + Description string `json:"description" jsonschema:"description=the description of the image that should be created"` +} + +type Output struct { + Error string `json:"error,omitempty"` + ImageURL string `json:"imageURL,omitempty"` +} + +type FnPrototype func(Input) Output + +func New() *Tool { + return &Tool{} +} + +func (t *Tool) Name() string { + return "dalle" +} + +func (t *Tool) Description() string { + return "A tool that creates an image from a description." +} + +func (t *Tool) Fn() any { + return t.fn +} + +func (t *Tool) fn(i Input) Output { + ctx, cancel := context.WithTimeout(context.Background(), defaultTimeoutInSeconds*time.Second) + defer cancel() + + d := transformer.NewDallE().WithImageSize(transformer.DallEImageSize512x512) + imageURL, err := d.Transform(ctx, i.Description) + if err != nil { + return Output{Error: fmt.Sprintf("error creating image: %v", err)} + } + + return Output{ImageURL: imageURL.(string)} +} diff --git a/tools/duckduckgo/api.go b/tools/duckduckgo/api.go new file mode 100644 index 00000000..a301c5e7 --- /dev/null +++ b/tools/duckduckgo/api.go @@ -0,0 +1,168 @@ +package duckduckgo + +import ( + "bytes" + "io" + "regexp" + "strings" + + "github.com/henomis/restclientgo" + "golang.org/x/net/html" +) + +const ( + class = "class" +) + +type request struct { + Query string +} + +type response struct { + MaxResults uint + HTTPStatusCode int + RawBody []byte + Results []result +} + +type result struct { + Title string + Info string + URL string +} + +func (r *request) Path() (string, error) { + return "/html/?q=" + r.Query, nil +} + +func (r *request) Encode() (io.Reader, error) { + return nil, nil +} + +func (r *request) ContentType() string { + return "" +} + +func (r *response) Decode(body io.Reader) error { + results, err := r.parseBody(body) + if err != nil { + return err + } + + r.Results = results + return nil +} + +func (r *response) SetBody(body io.Reader) error { + r.RawBody, _ = io.ReadAll(body) + return nil +} + +func (r *response) AcceptContentType() string { + return "text/html" +} + +func (r *response) SetStatusCode(code int) error { + r.HTTPStatusCode = code + return nil +} + +func (r *response) SetHeaders(_ restclientgo.Headers) error { return nil } + +func (r *response) parseBody(body io.Reader) ([]result, error) { + doc, err := html.Parse(body) + if err != nil { + return nil, err + } + ch := make(chan result) + go r.findWebResults(ch, doc) + + results := []result{} + for n := range ch { + results = append(results, n) + } + + return results, nil +} + +func (r *response) findWebResults(ch chan result, doc *html.Node) { + var results uint + var f func(*html.Node) + f = func(n *html.Node) { + if results >= r.MaxResults { + return + } + if n.Type == html.ElementNode && n.Data == "div" { + for _, div := range n.Attr { + if div.Key == class && strings.Contains(div.Val, "web-result") { + info, href := r.findInfo(n) + ch <- result{ + Title: r.findTitle(n), + Info: info, + URL: href, + } + results++ + break + } + } + } + for c := n.FirstChild; c != nil; c = c.NextSibling { + f(c) + } + } + f(doc) + close(ch) +} + +func (r *response) findTitle(n *html.Node) string { + var title string + var f func(*html.Node) + f = func(n *html.Node) { + if n.Type == html.ElementNode && n.Data == "a" { + for _, a := range n.Attr { + if a.Key == class && strings.Contains(a.Val, "result__a") { + title = n.FirstChild.Data + break + } + } + } + for c := n.FirstChild; c != nil; c = c.NextSibling { + f(c) + } + } + f(n) + return title +} + +//nolint:gocognit +func (r *response) findInfo(n *html.Node) (string, string) { + var info string + var link string + var f func(*html.Node) + f = func(n *html.Node) { + if n.Type == html.ElementNode && n.Data == "a" { + for _, a := range n.Attr { + if a.Key == class && strings.Contains(a.Val, "result__snippet") { + var b bytes.Buffer + _ = html.Render(&b, n) + + re := regexp.MustCompile("<.*?>") + info = html.UnescapeString(re.ReplaceAllString(b.String(), "")) + + for _, h := range n.Attr { + if h.Key == "href" { + link = "https:" + h.Val + break + } + } + break + } + } + } + for c := n.FirstChild; c != nil; c = c.NextSibling { + f(c) + } + } + f(n) + return info, link +} diff --git a/tools/duckduckgo/duckduckgo.go b/tools/duckduckgo/duckduckgo.go new file mode 100644 index 00000000..ec374942 --- /dev/null +++ b/tools/duckduckgo/duckduckgo.go @@ -0,0 +1,85 @@ +package duckduckgo + +import ( + "context" + "fmt" + "net/http" + "time" + + "github.com/henomis/restclientgo" +) + +const ( + defaultTimeoutInSeconds = 60 +) + +type Tool struct { + maxResults uint + userAgent string + restClient *restclientgo.RestClient +} + +type Input struct { + Query string `json:"query" jsonschema:"description=the query to search for"` +} + +type Output struct { + Error string `json:"error,omitempty"` + Results []result `json:"results,omitempty"` +} + +type FnPrototype func(Input) Output + +func New() *Tool { + t := &Tool{ + maxResults: 1, + } + + restClient := restclientgo.New("https://html.duckduckgo.com"). + WithRequestModifier( + func(r *http.Request) *http.Request { + r.Header.Add("User-Agent", t.userAgent) + return r + }, + ) + + t.restClient = restClient + return t +} + +func (t *Tool) WithUserAgent(userAgent string) *Tool { + t.userAgent = userAgent + return t +} + +func (t *Tool) WithMaxResults(maxResults uint) *Tool { + t.maxResults = maxResults + return t +} + +func (t *Tool) Name() string { + return "duckduckgo" +} + +func (t *Tool) Description() string { + return "A tool that uses the DuckDuckGo internet search engine for a query." +} + +func (t *Tool) Fn() any { + return t.fn +} + +func (t *Tool) fn(i Input) Output { + ctx, cancel := context.WithTimeout(context.Background(), defaultTimeoutInSeconds*time.Second) + defer cancel() + + req := &request{Query: i.Query} + res := &response{MaxResults: t.maxResults} + + err := t.restClient.Get(ctx, req, res) + if err != nil { + return Output{Error: fmt.Sprintf("failed to search DuckDuckGo: %v", err)} + } + + return Output{Results: res.Results} +} diff --git a/tools/llm/llm.go b/tools/llm/llm.go new file mode 100644 index 00000000..4f6190cb --- /dev/null +++ b/tools/llm/llm.go @@ -0,0 +1,68 @@ +package llm + +import ( + "context" + "time" + + "github.com/henomis/lingoose/thread" +) + +const ( + defaultTimeoutInMinutes = 6 +) + +type LLM interface { + Generate(context.Context, *thread.Thread) error +} + +type Tool struct { + llm LLM +} + +func New(llm LLM) *Tool { + return &Tool{ + llm: llm, + } +} + +type Input struct { + Query string `json:"query" jsonschema:"description=user query"` +} + +type Output struct { + Error string `json:"error,omitempty"` + Result string `json:"result,omitempty"` +} + +type FnPrototype func(Input) Output + +func (t *Tool) Name() string { + return "llm" +} + +func (t *Tool) Description() string { + return "A tool that uses a language model to generate a response to a user query." +} + +func (t *Tool) Fn() any { + return t.fn +} + +//nolint:gosec +func (t *Tool) fn(i Input) Output { + ctx, cancel := context.WithTimeout(context.Background(), defaultTimeoutInMinutes*time.Minute) + defer cancel() + + th := thread.New().AddMessage( + thread.NewUserMessage().AddContent( + thread.NewTextContent(i.Query), + ), + ) + + err := t.llm.Generate(ctx, th) + if err != nil { + return Output{Error: err.Error()} + } + + return Output{Result: th.LastMessage().Contents[0].AsString()} +} diff --git a/tools/python/python.go b/tools/python/python.go new file mode 100644 index 00000000..2ac686f2 --- /dev/null +++ b/tools/python/python.go @@ -0,0 +1,68 @@ +package python + +import ( + "bytes" + "fmt" + "os/exec" +) + +type Tool struct { + pythonPath string +} + +func New() *Tool { + return &Tool{ + pythonPath: "python3", + } +} + +func (t *Tool) WithPythonPath(pythonPath string) *Tool { + t.pythonPath = pythonPath + return t +} + +type Input struct { + PythonCode string `json:"python_code" jsonschema:"description=python code that prints the final result to stdout."` +} + +type Output struct { + Error string `json:"error,omitempty"` + Result string `json:"result,omitempty"` +} + +type FnPrototype = func(Input) Output + +func (t *Tool) Name() string { + return "python" +} + +func (t *Tool) Description() string { + return "A tool that runs Python code using the Python interpreter. The code should print the final result to stdout." +} + +func (t *Tool) Fn() any { + return t.fn +} + +//nolint:gosec +func (t *Tool) fn(i Input) Output { + // Create a command to run the Python interpreter with the script. + cmd := exec.Command(t.pythonPath, "-c", i.PythonCode) + + // Create a buffer to capture the output. + var out bytes.Buffer + var stderr bytes.Buffer + cmd.Stdout = &out + cmd.Stderr = &stderr + + // Run the command. + err := cmd.Run() + if err != nil { + return Output{ + Error: fmt.Sprintf("failed to run script: %v, stderr: %v", err, stderr.String()), + } + } + + // Return the output as a string. + return Output{Result: out.String()} +} diff --git a/tools/rag/rag.go b/tools/rag/rag.go new file mode 100644 index 00000000..c46788cf --- /dev/null +++ b/tools/rag/rag.go @@ -0,0 +1,62 @@ +package rag + +import ( + "context" + "strings" + "time" + + "github.com/henomis/lingoose/rag" +) + +const ( + defaultTimeoutInMinutes = 6 +) + +type Tool struct { + rag *rag.RAG + topic string +} + +func New(rag *rag.RAG, topic string) *Tool { + return &Tool{ + rag: rag, + topic: topic, + } +} + +type Input struct { + Query string `json:"rag_query" jsonschema:"description=search query"` +} + +type Output struct { + Error string `json:"error,omitempty"` + Result string `json:"result,omitempty"` +} + +type FnPrototype = func(Input) Output + +func (t *Tool) Name() string { + return "rag" +} + +func (t *Tool) Description() string { + return "A tool that searches information ONLY for this topic: " + t.topic + ". DO NOT use this tool for other topics." +} + +func (t *Tool) Fn() any { + return t.fn +} + +//nolint:gosec +func (t *Tool) fn(i Input) Output { + ctx, cancel := context.WithTimeout(context.Background(), defaultTimeoutInMinutes*time.Minute) + defer cancel() + + results, err := t.rag.Retrieve(ctx, i.Query) + if err != nil { + return Output{Error: err.Error()} + } + + // Return the output as a string. + return Output{Result: strings.Join(results, "\n")} +} diff --git a/tools/serpapi/api.go b/tools/serpapi/api.go new file mode 100644 index 00000000..944b71fe --- /dev/null +++ b/tools/serpapi/api.go @@ -0,0 +1,119 @@ +package serpapi + +import ( + "encoding/json" + "io" + + "github.com/henomis/restclientgo" +) + +type request struct { + Query string + GoogleDomain string + CountryCode string + LanguageCode string + APIKey string +} + +type response struct { + HTTPStatusCode int + Map map[string]interface{} + RawBody []byte + apiResponse apiResponse + Results []result +} + +type apiResponse struct { + OrganicResults []OrganicResults `json:"organic_results"` +} + +type Top struct { + Extensions []string `json:"extensions"` +} + +type RichSnippet struct { + Top Top `json:"top"` +} + +type OrganicResults struct { + Position int `json:"position"` + Title string `json:"title"` + Link string `json:"link"` + RedirectLink string `json:"redirect_link"` + DisplayedLink string `json:"displayed_link"` + Thumbnail string `json:"thumbnail,omitempty"` + Favicon string `json:"favicon"` + Snippet string `json:"snippet"` + Source string `json:"source"` + RichSnippet RichSnippet `json:"rich_snippet,omitempty"` + SnippetHighlightedWords []string `json:"snippet_highlighted_words,omitempty"` +} + +type result struct { + Title string + Info string + URL string +} + +func (r *request) Path() (string, error) { + urlValues := restclientgo.NewURLValues() + urlValues.Add("q", &r.Query) + urlValues.Add("api_key", &r.APIKey) + + if r.GoogleDomain != "" { + urlValues.Add("google_domain", &r.GoogleDomain) + } + + if r.CountryCode != "" { + urlValues.Add("gl", &r.CountryCode) + } + + if r.LanguageCode != "" { + urlValues.Add("hl", &r.LanguageCode) + } + + params := urlValues.Encode() + + return "/search?" + params, nil +} + +func (r *request) Encode() (io.Reader, error) { + return nil, nil +} + +func (r *request) ContentType() string { + return "" +} + +func (r *response) Decode(body io.Reader) error { + err := json.NewDecoder(body).Decode(&r.apiResponse) + if err != nil { + return err + } + + for _, res := range r.apiResponse.OrganicResults { + r.Results = append(r.Results, result{ + Title: res.Title, + Info: res.Snippet, + URL: res.Link, + }) + } + + return nil +} + +func (r *response) SetBody(body io.Reader) error { + r.RawBody, _ = io.ReadAll(body) + return nil +} + +func (r *response) AcceptContentType() string { + return "application/json" +} + +func (r *response) SetStatusCode(code int) error { + r.HTTPStatusCode = code + return nil +} + +func (r *response) SetHeaders(_ restclientgo.Headers) error { return nil } diff --git a/tools/serpapi/serpapi.go b/tools/serpapi/serpapi.go new file mode 100644 index 00000000..9637e691 --- /dev/null +++ b/tools/serpapi/serpapi.go @@ -0,0 +1,98 @@ +package serpapi + +import ( + "context" + "fmt" + "os" + "time" + + "github.com/henomis/restclientgo" +) + +const ( + defaultTimeoutInSeconds = 60 +) + +type Tool struct { + restClient *restclientgo.RestClient + googleDomain string + countryCode string + languageCode string + apiKey string +} + +type Input struct { + Query string `json:"query" jsonschema:"description=the query to search for"` +} + +type Output struct { + Error string `json:"error,omitempty"` + Results []result `json:"results,omitempty"` +} + +type FnPrototype = func(Input) Output + +func New() *Tool { + t := &Tool{ + apiKey: os.Getenv("SERPAPI_API_KEY"), + restClient: restclientgo.New("https://serpapi.com"), + googleDomain: "google.com", + countryCode: "us", + languageCode: "en", + } + + return t +} + +func (t *Tool) WithGoogleDomain(googleDomain string) *Tool { + t.googleDomain = googleDomain + return t +} + +func (t *Tool) WithCountryCode(countryCode string) *Tool { + t.countryCode = countryCode + return t +} + +func (t *Tool) WithLanguageCode(languageCode string) *Tool { + t.languageCode = languageCode + return t +} + +func (t *Tool) WithAPIKey(apiKey string) *Tool { + t.apiKey = apiKey + return t +} + +func (t *Tool) Name() string { + return "google" +} + +func (t *Tool) Description() string { + return "A tool that uses the Google internet search engine for a query." +} + +func (t *Tool) Fn() any { + return t.fn +} + +func (t *Tool) fn(i Input) Output { + ctx, cancel := context.WithTimeout(context.Background(), defaultTimeoutInSeconds*time.Second) + defer cancel() + + req := &request{ + Query: i.Query, + GoogleDomain: t.googleDomain, + CountryCode: t.countryCode, + LanguageCode: t.languageCode, + APIKey: t.apiKey, + } + res := &response{} + + err := t.restClient.Get(ctx, req, res) + if err != nil { + return Output{Error: fmt.Sprintf("failed to search serpapi: %v", err)} + } + + return Output{Results: res.Results} +} diff --git a/tools/shell/shell.go b/tools/shell/shell.go new file mode 100644 index 00000000..c5c5e308 --- /dev/null +++ b/tools/shell/shell.go @@ -0,0 +1,91 @@ +package shell + +import ( + "bytes" + "fmt" + "os/exec" +) + +type Tool struct { + shell string + askForConfirm bool +} + +func New() *Tool { + return &Tool{ + shell: "bash", + askForConfirm: true, + } +} + +func (t *Tool) WithShell(shell string) *Tool { + t.shell = shell + return t +} + +func (t *Tool) WithAskForConfirm(askForConfirm bool) *Tool { + t.askForConfirm = askForConfirm + return t +} + +type Input struct { + BashScript string `json:"bash_code" jsonschema:"description=shell script"` +} + +type Output struct { + Error string `json:"error,omitempty"` + Result string `json:"result,omitempty"` +} + +type FnPrototype = func(Input) Output + +func (t *Tool) Name() string { + return "bash" +} + +func (t *Tool) Description() string { + return "A tool that runs a shell script using the " + t.shell + " interpreter. Use it to interact with the OS." +} + +func (t *Tool) Fn() any { + return t.fn +} + +//nolint:gosec +func (t *Tool) fn(i Input) Output { + // Ask for confirmation if the flag is set. + if t.askForConfirm { + fmt.Println("Are you sure you want to run the following script?") + fmt.Println("-------------------------------------------------") + fmt.Println(i.BashScript) + fmt.Println("-------------------------------------------------") + fmt.Print("Type 'yes' to confirm > ") + var confirm string + fmt.Scanln(&confirm) + if confirm != "yes" { + return Output{ + Error: "script execution aborted", + } + } + } + + // Create a command to run the Bash interpreter with the script. + cmd := exec.Command(t.shell, "-c", i.BashScript) + + // Create a buffer to capture the output. + var out bytes.Buffer + var stderr bytes.Buffer + cmd.Stdout = &out + cmd.Stderr = &stderr + + // Run the command. + err := cmd.Run() + if err != nil { + return Output{ + Error: fmt.Sprintf("failed to run script: %v, stderr: %v", err, stderr.String()), + } + } + + // Return the output as a string. + return Output{Result: out.String()} +} diff --git a/tools/tool_router/tool_router.go b/tools/tool_router/tool_router.go new file mode 100644 index 00000000..17d35bcd --- /dev/null +++ b/tools/tool_router/tool_router.go @@ -0,0 +1,84 @@ +package toolrouter + +import ( + "context" + "time" + + "github.com/henomis/lingoose/thread" +) + +const ( + defaultTimeoutInMinutes = 6 +) + +type TTool interface { + Description() string + Name() string + Fn() any +} + +type Tool struct { + llm LLM + tools []TTool +} + +type LLM interface { + Generate(context.Context, *thread.Thread) error +} + +func New(llm LLM, tools ...TTool) *Tool { + return &Tool{ + tools: tools, + llm: llm, + } +} + +type Input struct { + Query string `json:"query" jsonschema:"description=user query"` +} + +type Output struct { + Error string `json:"error,omitempty"` + Result any `json:"result,omitempty"` +} + +type FnPrototype func(Input) Output + +func (t *Tool) Name() string { + return "query_router" +} + +func (t *Tool) Description() string { + return "A tool that select the right tool to answer to user queries." +} + +func (t *Tool) Fn() any { + return t.fn +} + +//nolint:gosec +func (t *Tool) fn(i Input) Output { + ctx, cancel := context.WithTimeout(context.Background(), defaultTimeoutInMinutes*time.Minute) + defer cancel() + + query := "Here's a list of available tools:\n\n" + for _, tool := range t.tools { + query += "Name: " + tool.Name() + "\nDescription: " + tool.Description() + "\n\n" + } + + query += "\nPlease select the right tool that can better answer the query '" + i.Query + + "'. Give me only the name of the tool, nothing else." + + th := thread.New().AddMessage( + thread.NewUserMessage().AddContent( + thread.NewTextContent(query), + ), + ) + + err := t.llm.Generate(ctx, th) + if err != nil { + return Output{Error: err.Error()} + } + + return Output{Result: th.LastMessage().Contents[0].AsString()} +} From 3474837c555aee009cd0079bd2d69a467e22a13b Mon Sep 17 00:00:00 2001 From: Simone Vellei Date: Thu, 30 May 2024 21:18:52 +0200 Subject: [PATCH 02/10] fix: openai WithTools pointer receiver (#204) * fix: use pointer receiver * chore: add assistant execute --- assistant/assistant.go | 76 +++++++++++++++++++++++++++++++++++++++--- assistant/prompt.go | 2 +- llm/openai/function.go | 2 +- 3 files changed, 73 insertions(+), 7 deletions(-) diff --git a/assistant/assistant.go b/assistant/assistant.go index 87adb93d..92645ceb 100644 --- a/assistant/assistant.go +++ b/assistant/assistant.go @@ -22,11 +22,16 @@ type observer interface { SpanEnd(s *obs.Span) (*obs.Span, error) } +const ( + DefaultMaxIterations = 3 +) + type Assistant struct { - llm LLM - rag RAG - thread *thread.Thread - parameters Parameters + llm LLM + rag RAG + thread *thread.Thread + parameters Parameters + maxIterations uint } type LLM interface { @@ -48,6 +53,7 @@ func New(llm LLM) *Assistant { CompanyName: defaultCompanyName, CompanyDescription: defaultCompanyDescription, }, + maxIterations: DefaultMaxIterations, } return assistant @@ -123,7 +129,7 @@ func (a *Assistant) generateRAGMessage(ctx context.Context) error { a.thread.AddMessage(thread.NewSystemMessage().AddContent( thread.NewTextContent( - systemRAGPrompt, + systemPrompt, ).Format( types.M{ "assistantName": a.parameters.AssistantName, @@ -147,6 +153,42 @@ func (a *Assistant) generateRAGMessage(ctx context.Context) error { return nil } +func (a *Assistant) WithMaxIterations(maxIterations uint) *Assistant { + a.maxIterations = maxIterations + return a +} + +func (a *Assistant) Execute(ctx context.Context) error { + if a.thread == nil { + return nil + } + + ctx, spanAssistant, err := a.startObserveSpan(ctx, "assistant") + if err != nil { + return err + } + + a.injectSystemMessage() + + for i := 0; i < int(a.maxIterations); i++ { + err = a.llm.Generate(ctx, a.thread) + if err != nil { + return err + } + + if a.thread.LastMessage().Role != thread.RoleTool { + break + } + } + + err = a.stopObserveSpan(ctx, spanAssistant) + if err != nil { + return err + } + + return nil +} + func (a *Assistant) startObserveSpan(ctx context.Context, name string) (context.Context, *obs.Span, error) { o, ok := obs.ContextValueObserverInstance(ctx).(observer) if o == nil || !ok { @@ -183,3 +225,27 @@ func (a *Assistant) stopObserveSpan(ctx context.Context, span *obs.Span) error { _, err := o.SpanEnd(span) return err } + +func (a *Assistant) injectSystemMessage() { + for _, message := range a.thread.Messages { + if message.Role == thread.RoleSystem { + return + } + } + + systemMessage := thread.NewSystemMessage().AddContent( + thread.NewTextContent( + systemPrompt, + ).Format( + types.M{ + "assistantName": a.parameters.AssistantName, + "assistantIdentity": a.parameters.AssistantIdentity, + "assistantScope": a.parameters.AssistantScope, + "companyName": a.parameters.CompanyName, + "companyDescription": a.parameters.CompanyDescription, + }, + ), + ) + + a.thread.Messages = append([]*thread.Message{systemMessage}, a.thread.Messages...) +} diff --git a/assistant/prompt.go b/assistant/prompt.go index 18a4f8fe..29f04df8 100644 --- a/assistant/prompt.go +++ b/assistant/prompt.go @@ -4,7 +4,7 @@ const ( //nolint:lll baseRAGPrompt = "Use the following pieces of retrieved context to answer the question.\n\nQuestion: {{.question}}\nContext:\n{{range .results}}{{.}}\n\n{{end}}" //nolint:lll - systemRAGPrompt = "You name is {{.assistantName}}, and you are {{.assistantIdentity}} {{if ne .companyName \"\" }}at {{.companyName}}{{end}}{{if ne .companyDescription \"\" }}, {{.companyDescription}}{{end}}. Your task is to assist humans {{.assistantScope}}." + systemPrompt = "You name is {{.assistantName}}, and you are {{.assistantIdentity}} {{if ne .companyName \"\" }}at {{.companyName}}{{end}}{{if ne .companyDescription \"\" }}, {{.companyDescription}}{{end}}. Your task is to assist humans {{.assistantScope}}." defaultAssistantName = "AI assistant" defaultAssistantIdentity = "a helpful and polite assistant" diff --git a/llm/openai/function.go b/llm/openai/function.go index 9c5953bd..deaa12a5 100644 --- a/llm/openai/function.go +++ b/llm/openai/function.go @@ -85,7 +85,7 @@ type Tool interface { Fn() any } -func (o OpenAI) WithTools(tools ...Tool) OpenAI { +func (o *OpenAI) WithTools(tools ...Tool) *OpenAI { for _, tool := range tools { function, err := bindFunction(tool.Fn(), tool.Name(), tool.Description()) if err != nil { From e201fa40d5289f7145aaad630c282202fc9e5cc4 Mon Sep 17 00:00:00 2001 From: Simone Vellei Date: Fri, 31 May 2024 15:44:45 +0200 Subject: [PATCH 03/10] chore: change the way assistant actually works --- assistant/assistant.go | 48 ++++++++-------------------- examples/assistant/agent/main.go | 47 +++++++++++++++++++++++++++ examples/assistant/{ => rag}/main.go | 0 3 files changed, 61 insertions(+), 34 deletions(-) create mode 100644 examples/assistant/agent/main.go rename examples/assistant/{ => rag}/main.go (100%) diff --git a/assistant/assistant.go b/assistant/assistant.go index 92645ceb..6d41d28e 100644 --- a/assistant/assistant.go +++ b/assistant/assistant.go @@ -2,6 +2,7 @@ package assistant import ( "context" + "fmt" "strings" obs "github.com/henomis/lingoose/observer" @@ -89,11 +90,21 @@ func (a *Assistant) Run(ctx context.Context) error { if errGenerate != nil { return errGenerate } + } else { + a.injectSystemMessage() } - err = a.llm.Generate(ctx, a.thread) - if err != nil { - return err + for i := 0; i < int(a.maxIterations); i++ { + err = a.llm.Generate(ctx, a.thread) + if err != nil { + return err + } + + if a.thread.LastMessage().Role != thread.RoleTool { + break + } + + fmt.Println(a.Thread()) } err = a.stopObserveSpan(ctx, spanAssistant) @@ -158,37 +169,6 @@ func (a *Assistant) WithMaxIterations(maxIterations uint) *Assistant { return a } -func (a *Assistant) Execute(ctx context.Context) error { - if a.thread == nil { - return nil - } - - ctx, spanAssistant, err := a.startObserveSpan(ctx, "assistant") - if err != nil { - return err - } - - a.injectSystemMessage() - - for i := 0; i < int(a.maxIterations); i++ { - err = a.llm.Generate(ctx, a.thread) - if err != nil { - return err - } - - if a.thread.LastMessage().Role != thread.RoleTool { - break - } - } - - err = a.stopObserveSpan(ctx, spanAssistant) - if err != nil { - return err - } - - return nil -} - func (a *Assistant) startObserveSpan(ctx context.Context, name string) (context.Context, *obs.Span, error) { o, ok := obs.ContextValueObserverInstance(ctx).(observer) if o == nil || !ok { diff --git a/examples/assistant/agent/main.go b/examples/assistant/agent/main.go new file mode 100644 index 00000000..58b570fd --- /dev/null +++ b/examples/assistant/agent/main.go @@ -0,0 +1,47 @@ +package main + +import ( + "context" + "fmt" + + "github.com/henomis/lingoose/assistant" + "github.com/henomis/lingoose/llm/openai" + "github.com/henomis/lingoose/thread" + + pythontool "github.com/henomis/lingoose/tools/python" + serpapitool "github.com/henomis/lingoose/tools/serpapi" +) + +func main() { + + auto := "auto" + a := assistant.New( + openai.New().WithModel(openai.GPT4o).WithToolChoice(&auto).WithTools( + pythontool.New(), + serpapitool.New(), + ), + ).WithParameters( + assistant.Parameters{ + AssistantName: "AI Assistant", + AssistantIdentity: "an helpful assistant", + AssistantScope: "with their questions.", + CompanyName: "", + CompanyDescription: "", + }, + ).WithThread( + thread.New().AddMessages( + thread.NewUserMessage().AddContent( + thread.NewTextContent("calculate the average temperature in celsius degrees of New York, Rome, and Tokyo."), + ), + ), + ).WithMaxIterations(10) + + err := a.Run(context.Background()) + if err != nil { + panic(err) + } + + fmt.Println("----") + fmt.Println(a.Thread()) + fmt.Println("----") +} diff --git a/examples/assistant/main.go b/examples/assistant/rag/main.go similarity index 100% rename from examples/assistant/main.go rename to examples/assistant/rag/main.go From 0e086ffa18683fcbc401f22a5473c22a692ee6a2 Mon Sep 17 00:00:00 2001 From: Simone Vellei Date: Fri, 31 May 2024 15:45:41 +0200 Subject: [PATCH 04/10] chore: rename tools in tool --- examples/assistant/agent/main.go | 4 ++-- examples/llm/openai/thread/main.go | 2 +- examples/llm/openai/tools/python/main.go | 2 +- examples/llm/openai/tools/rag/main.go | 6 +++--- examples/tools/duckduckgo/main.go | 2 +- examples/tools/python/main.go | 2 +- examples/tools/serpapi/main.go | 2 +- examples/tools/shell/main.go | 2 +- {tools => tool}/dalle/dalle.go | 0 {tools => tool}/duckduckgo/api.go | 0 {tools => tool}/duckduckgo/duckduckgo.go | 0 {tools => tool}/llm/llm.go | 0 {tools => tool}/python/python.go | 0 {tools => tool}/rag/rag.go | 0 {tools => tool}/serpapi/api.go | 0 {tools => tool}/serpapi/serpapi.go | 0 {tools => tool}/shell/shell.go | 0 {tools => tool}/tool_router/tool_router.go | 0 18 files changed, 11 insertions(+), 11 deletions(-) rename {tools => tool}/dalle/dalle.go (100%) rename {tools => tool}/duckduckgo/api.go (100%) rename {tools => tool}/duckduckgo/duckduckgo.go (100%) rename {tools => tool}/llm/llm.go (100%) rename {tools => tool}/python/python.go (100%) rename {tools => tool}/rag/rag.go (100%) rename {tools => tool}/serpapi/api.go (100%) rename {tools => tool}/serpapi/serpapi.go (100%) rename {tools => tool}/shell/shell.go (100%) rename {tools => tool}/tool_router/tool_router.go (100%) diff --git a/examples/assistant/agent/main.go b/examples/assistant/agent/main.go index 58b570fd..53b106bc 100644 --- a/examples/assistant/agent/main.go +++ b/examples/assistant/agent/main.go @@ -8,8 +8,8 @@ import ( "github.com/henomis/lingoose/llm/openai" "github.com/henomis/lingoose/thread" - pythontool "github.com/henomis/lingoose/tools/python" - serpapitool "github.com/henomis/lingoose/tools/serpapi" + pythontool "github.com/henomis/lingoose/tool/python" + serpapitool "github.com/henomis/lingoose/tool/serpapi" ) func main() { diff --git a/examples/llm/openai/thread/main.go b/examples/llm/openai/thread/main.go index 695f4714..7bc462f0 100644 --- a/examples/llm/openai/thread/main.go +++ b/examples/llm/openai/thread/main.go @@ -7,7 +7,7 @@ import ( "github.com/henomis/lingoose/llm/openai" "github.com/henomis/lingoose/thread" - "github.com/henomis/lingoose/tools/dalle" + "github.com/henomis/lingoose/tool/dalle" "github.com/henomis/lingoose/transformer" ) diff --git a/examples/llm/openai/tools/python/main.go b/examples/llm/openai/tools/python/main.go index 1e133410..71a8d580 100644 --- a/examples/llm/openai/tools/python/main.go +++ b/examples/llm/openai/tools/python/main.go @@ -6,7 +6,7 @@ import ( "github.com/henomis/lingoose/llm/openai" "github.com/henomis/lingoose/thread" - "github.com/henomis/lingoose/tools/python" + "github.com/henomis/lingoose/tool/python" ) func main() { diff --git a/examples/llm/openai/tools/rag/main.go b/examples/llm/openai/tools/rag/main.go index 29deadc9..30010b5b 100644 --- a/examples/llm/openai/tools/rag/main.go +++ b/examples/llm/openai/tools/rag/main.go @@ -11,9 +11,9 @@ import ( "github.com/henomis/lingoose/llm/openai" "github.com/henomis/lingoose/rag" "github.com/henomis/lingoose/thread" - ragtool "github.com/henomis/lingoose/tools/rag" - "github.com/henomis/lingoose/tools/serpapi" - "github.com/henomis/lingoose/tools/shell" + ragtool "github.com/henomis/lingoose/tool/rag" + "github.com/henomis/lingoose/tool/serpapi" + "github.com/henomis/lingoose/tool/shell" ) func main() { diff --git a/examples/tools/duckduckgo/main.go b/examples/tools/duckduckgo/main.go index 7be5f65c..fbcfd0a8 100644 --- a/examples/tools/duckduckgo/main.go +++ b/examples/tools/duckduckgo/main.go @@ -3,7 +3,7 @@ package main import ( "fmt" - "github.com/henomis/lingoose/tools/duckduckgo" + "github.com/henomis/lingoose/tool/duckduckgo" ) func main() { diff --git a/examples/tools/python/main.go b/examples/tools/python/main.go index 57052fb6..3eee0b30 100644 --- a/examples/tools/python/main.go +++ b/examples/tools/python/main.go @@ -3,7 +3,7 @@ package main import ( "fmt" - "github.com/henomis/lingoose/tools/python" + "github.com/henomis/lingoose/tool/python" ) func main() { diff --git a/examples/tools/serpapi/main.go b/examples/tools/serpapi/main.go index d0578d3a..55ca5163 100644 --- a/examples/tools/serpapi/main.go +++ b/examples/tools/serpapi/main.go @@ -3,7 +3,7 @@ package main import ( "fmt" - "github.com/henomis/lingoose/tools/serpapi" + "github.com/henomis/lingoose/tool/serpapi" ) func main() { diff --git a/examples/tools/shell/main.go b/examples/tools/shell/main.go index 3ec73377..86cb39a9 100644 --- a/examples/tools/shell/main.go +++ b/examples/tools/shell/main.go @@ -3,7 +3,7 @@ package main import ( "fmt" - "github.com/henomis/lingoose/tools/shell" + "github.com/henomis/lingoose/tool/shell" ) func main() { diff --git a/tools/dalle/dalle.go b/tool/dalle/dalle.go similarity index 100% rename from tools/dalle/dalle.go rename to tool/dalle/dalle.go diff --git a/tools/duckduckgo/api.go b/tool/duckduckgo/api.go similarity index 100% rename from tools/duckduckgo/api.go rename to tool/duckduckgo/api.go diff --git a/tools/duckduckgo/duckduckgo.go b/tool/duckduckgo/duckduckgo.go similarity index 100% rename from tools/duckduckgo/duckduckgo.go rename to tool/duckduckgo/duckduckgo.go diff --git a/tools/llm/llm.go b/tool/llm/llm.go similarity index 100% rename from tools/llm/llm.go rename to tool/llm/llm.go diff --git a/tools/python/python.go b/tool/python/python.go similarity index 100% rename from tools/python/python.go rename to tool/python/python.go diff --git a/tools/rag/rag.go b/tool/rag/rag.go similarity index 100% rename from tools/rag/rag.go rename to tool/rag/rag.go diff --git a/tools/serpapi/api.go b/tool/serpapi/api.go similarity index 100% rename from tools/serpapi/api.go rename to tool/serpapi/api.go diff --git a/tools/serpapi/serpapi.go b/tool/serpapi/serpapi.go similarity index 100% rename from tools/serpapi/serpapi.go rename to tool/serpapi/serpapi.go diff --git a/tools/shell/shell.go b/tool/shell/shell.go similarity index 100% rename from tools/shell/shell.go rename to tool/shell/shell.go diff --git a/tools/tool_router/tool_router.go b/tool/tool_router/tool_router.go similarity index 100% rename from tools/tool_router/tool_router.go rename to tool/tool_router/tool_router.go From 40f3055417e0d286d72e9cfdea902290ffa294e3 Mon Sep 17 00:00:00 2001 From: Simone Vellei Date: Fri, 31 May 2024 15:45:41 +0200 Subject: [PATCH 05/10] docs: update docs --- .gitignore | 3 +- docs/content/reference/assistant.md | 35 ++++++++++++++++++- docs/content/reference/examples.md | 2 +- docs/content/reference/linglet.md | 2 +- docs/content/reference/observer.md | 2 +- docs/content/reference/tool.md | 52 +++++++++++++++++++++++++++++ 6 files changed, 91 insertions(+), 5 deletions(-) create mode 100644 docs/content/reference/tool.md diff --git a/.gitignore b/.gitignore index 663d0bcb..e0123285 100644 --- a/.gitignore +++ b/.gitignore @@ -19,4 +19,5 @@ bin/ llama.cpp/ whisper.cpp/ -*/.hugo_build.lock \ No newline at end of file +*/.hugo_build.lock +docs/public/ \ No newline at end of file diff --git a/docs/content/reference/assistant.md b/docs/content/reference/assistant.md index a7f95357..f66b1c31 100644 --- a/docs/content/reference/assistant.md +++ b/docs/content/reference/assistant.md @@ -31,4 +31,37 @@ if err != nil { fmt.Println(myAssistant.Thread()) ``` -We can define the LinGoose `Assistant` as a `Thread` runner with an optional `RAG` component that will help to produce the response. \ No newline at end of file +We can define the LinGoose `Assistant` as a `Thread` runner with an optional `RAG` component that will help to produce the response. + +## Assistant as Agent + +The `Assistant` can be used as an agent in a conversation. It can be used to automate tasks, answer questions, and provide information. + +```go +auto := "auto" +myAgent := assistant.New( + openai.New().WithModel(openai.GPT4o).WithToolChoice(&auto).WithTools( + pythontool.New(), + serpapitool.New(), + ), +).WithParameters( + assistant.Parameters{ + AssistantName: "AI Assistant", + AssistantIdentity: "an helpful assistant", + AssistantScope: "with their questions.", + CompanyName: "", + CompanyDescription: "", + }, +).WithThread( + thread.New().AddMessages( + thread.NewUserMessage().AddContent( + thread.NewTextContent("calculate the average temperature in celsius degrees of New York, Rome, and Tokyo."), + ), + ), +).WithMaxIterations(10) + +err := myAgent.Run(context.Background()) +if err != nil { + panic(err) +} +``` \ No newline at end of file diff --git a/docs/content/reference/examples.md b/docs/content/reference/examples.md index 7a48602c..18a600cc 100644 --- a/docs/content/reference/examples.md +++ b/docs/content/reference/examples.md @@ -2,7 +2,7 @@ title: "LinGoose Examples" description: linkTitle: "Examples" -menu: { main: { parent: 'reference', weight: -88 } } +menu: { main: { parent: 'reference', weight: -87 } } --- LinGoose provides a number of examples to help you get started with building your own AI app. You can use these examples as a reference to understand how to build your own assistant. diff --git a/docs/content/reference/linglet.md b/docs/content/reference/linglet.md index f113e0be..16f7c90f 100644 --- a/docs/content/reference/linglet.md +++ b/docs/content/reference/linglet.md @@ -2,7 +2,7 @@ title: "LinGoose Linglets" description: linkTitle: "Linglets" -menu: { main: { parent: 'reference', weight: -89 } } +menu: { main: { parent: 'reference', weight: -88 } } --- Linglets are pre-built LinGoose Assistants with a specific purpose. They are designed to be used as a starting point for building your own AI app. You can use them as a reference to understand how to build your own assistant. diff --git a/docs/content/reference/observer.md b/docs/content/reference/observer.md index 2202f3a4..0c6d937e 100644 --- a/docs/content/reference/observer.md +++ b/docs/content/reference/observer.md @@ -1,5 +1,5 @@ --- -title: "Observer" +title: "Observe and Analyze LLM Applications" description: linkTitle: "Observer" menu: { main: { parent: 'reference', weight: -92 } } diff --git a/docs/content/reference/tool.md b/docs/content/reference/tool.md new file mode 100644 index 00000000..279968dc --- /dev/null +++ b/docs/content/reference/tool.md @@ -0,0 +1,52 @@ +--- +title: "Performing tasks with Tools" +description: +linkTitle: "Tool" +menu: { main: { parent: 'reference', weight: -89 } } +--- + +Tools are components that can be used to perform specific tasks. They can be used to automate, answer questions, and provide information. LinGoose offers a variety of tools that can be used to perform different actions. + +## Available Tools + +- *Python*: It can be used to run Python code and get the output. +- *SerpApi*: It can be used to get search results from Google and other search engines. +- *Dall-e*: It can be used to generate images based on text descriptions. +- *DuckDuckGo*: It can be used to get search results from DuckDuckGo. +- *RAG*: It can be used to retrieve relevant documents based on a query. +- *LLM*: It can be used to generate text based on a prompt. +- *Shell*: It can be used to run shell commands and get the output. + + +## Using Tools + +LinGoose tools can be used to perform specific tasks. Here is an example of using the `Python` and `serpapi` tools to get information and run Python code and get the output. + +```go +auto := "auto" +myAgent := assistant.New( + openai.New().WithModel(openai.GPT4o).WithToolChoice(&auto).WithTools( + pythontool.New(), + serpapitool.New(), + ), +).WithParameters( + assistant.Parameters{ + AssistantName: "AI Assistant", + AssistantIdentity: "an helpful assistant", + AssistantScope: "with their questions.", + CompanyName: "", + CompanyDescription: "", + }, +).WithThread( + thread.New().AddMessages( + thread.NewUserMessage().AddContent( + thread.NewTextContent("calculate the average temperature in celsius degrees of New York, Rome, and Tokyo."), + ), + ), +).WithMaxIterations(10) + +err := myAgent.Run(context.Background()) +if err != nil { + panic(err) +} +``` \ No newline at end of file From a86c0e366a3f09e1dddfd3d267dd5462cbe72e97 Mon Sep 17 00:00:00 2001 From: Simone Vellei Date: Sun, 2 Jun 2024 14:52:03 +0200 Subject: [PATCH 06/10] Refactor ssistant observer (#206) * refactor assistant observer * fix: linting --- assistant/assistant.go | 3 -- examples/assistant/agent/main.go | 16 +++++++- examples/observer/assistant/main.go | 2 + examples/observer/langfuse/main.go | 14 ++++--- llm/antropic/antropic.go | 6 +-- llm/cohere/cohere.go | 6 +-- llm/observer/observer.go | 4 +- llm/ollama/ollama.go | 6 +-- llm/openai/openai.go | 8 ++-- observer/langfuse/formatter.go | 61 +++++++++++++++++++++++++++-- observer/observer.go | 2 +- 11 files changed, 100 insertions(+), 28 deletions(-) diff --git a/assistant/assistant.go b/assistant/assistant.go index 6d41d28e..6b4764b9 100644 --- a/assistant/assistant.go +++ b/assistant/assistant.go @@ -2,7 +2,6 @@ package assistant import ( "context" - "fmt" "strings" obs "github.com/henomis/lingoose/observer" @@ -103,8 +102,6 @@ func (a *Assistant) Run(ctx context.Context) error { if a.thread.LastMessage().Role != thread.RoleTool { break } - - fmt.Println(a.Thread()) } err = a.stopObserveSpan(ctx, spanAssistant) diff --git a/examples/assistant/agent/main.go b/examples/assistant/agent/main.go index 53b106bc..dac59c26 100644 --- a/examples/assistant/agent/main.go +++ b/examples/assistant/agent/main.go @@ -6,6 +6,8 @@ import ( "github.com/henomis/lingoose/assistant" "github.com/henomis/lingoose/llm/openai" + "github.com/henomis/lingoose/observer" + "github.com/henomis/lingoose/observer/langfuse" "github.com/henomis/lingoose/thread" pythontool "github.com/henomis/lingoose/tool/python" @@ -13,6 +15,16 @@ import ( ) func main() { + ctx := context.Background() + + o := langfuse.New(ctx) + trace, err := o.Trace(&observer.Trace{Name: "state of the union"}) + if err != nil { + panic(err) + } + + ctx = observer.ContextWithObserverInstance(ctx, o) + ctx = observer.ContextWithTraceID(ctx, trace.ID) auto := "auto" a := assistant.New( @@ -36,7 +48,7 @@ func main() { ), ).WithMaxIterations(10) - err := a.Run(context.Background()) + err = a.Run(ctx) if err != nil { panic(err) } @@ -44,4 +56,6 @@ func main() { fmt.Println("----") fmt.Println(a.Thread()) fmt.Println("----") + + o.Flush(ctx) } diff --git a/examples/observer/assistant/main.go b/examples/observer/assistant/main.go index 9e744f9d..f5880c09 100644 --- a/examples/observer/assistant/main.go +++ b/examples/observer/assistant/main.go @@ -71,4 +71,6 @@ func main() { fmt.Println("----") fmt.Println(a.Thread()) fmt.Println("----") + + o.Flush(ctx) } diff --git a/examples/observer/langfuse/main.go b/examples/observer/langfuse/main.go index 6de4022c..80c46d49 100644 --- a/examples/observer/langfuse/main.go +++ b/examples/observer/langfuse/main.go @@ -84,12 +84,14 @@ func main() { panic(err) } - generation.Output = &thread.Message{ - Role: thread.RoleAssistant, - Contents: []*thread.Content{ - { - Type: thread.ContentTypeText, - Data: "The Q3 OKRs contain goals for multiple teams...", + generation.Output = []*thread.Message{ + { + Role: thread.RoleAssistant, + Contents: []*thread.Content{ + { + Type: thread.ContentTypeText, + Data: "The Q3 OKRs contain goals for multiple teams...", + }, }, }, } diff --git a/llm/antropic/antropic.go b/llm/antropic/antropic.go index 648bf521..a4a81e70 100644 --- a/llm/antropic/antropic.go +++ b/llm/antropic/antropic.go @@ -171,7 +171,7 @@ func (o *Antropic) Generate(ctx context.Context, t *thread.Thread) error { return err } - err = o.stopObserveGeneration(ctx, generation, t) + err = o.stopObserveGeneration(ctx, generation, []*thread.Message{t.LastMessage()}) if err != nil { return fmt.Errorf("%w: %w", ErrAnthropicChat, err) } @@ -281,11 +281,11 @@ func (o *Antropic) startObserveGeneration(ctx context.Context, t *thread.Thread) func (o *Antropic) stopObserveGeneration( ctx context.Context, generation *observer.Generation, - t *thread.Thread, + messagges []*thread.Message, ) error { return llmobserver.StopObserveGeneration( ctx, generation, - t, + messagges, ) } diff --git a/llm/cohere/cohere.go b/llm/cohere/cohere.go index 4966f9af..1c53a04e 100644 --- a/llm/cohere/cohere.go +++ b/llm/cohere/cohere.go @@ -233,7 +233,7 @@ func (c *Cohere) Generate(ctx context.Context, t *thread.Thread) error { return err } - err = c.stopObserveGeneration(ctx, generation, t) + err = c.stopObserveGeneration(ctx, generation, []*thread.Message{t.LastMessage()}) if err != nil { return fmt.Errorf("%w: %w", ErrCohereChat, err) } @@ -309,11 +309,11 @@ func (c *Cohere) startObserveGeneration(ctx context.Context, t *thread.Thread) ( func (c *Cohere) stopObserveGeneration( ctx context.Context, generation *observer.Generation, - t *thread.Thread, + messages []*thread.Message, ) error { return llmobserver.StopObserveGeneration( ctx, generation, - t, + messages, ) } diff --git a/llm/observer/observer.go b/llm/observer/observer.go index b5ddfbcb..b54ad7a9 100644 --- a/llm/observer/observer.go +++ b/llm/observer/observer.go @@ -47,7 +47,7 @@ func StartObserveGeneration( func StopObserveGeneration( ctx context.Context, generation *observer.Generation, - t *thread.Thread, + messages []*thread.Message, ) error { o, ok := observer.ContextValueObserverInstance(ctx).(LLMObserver) if o == nil || !ok { @@ -55,7 +55,7 @@ func StopObserveGeneration( return nil } - generation.Output = t.LastMessage() + generation.Output = messages _, err := o.GenerationEnd(generation) return err } diff --git a/llm/ollama/ollama.go b/llm/ollama/ollama.go index 51e9c61f..329446cb 100644 --- a/llm/ollama/ollama.go +++ b/llm/ollama/ollama.go @@ -150,7 +150,7 @@ func (o *Ollama) Generate(ctx context.Context, t *thread.Thread) error { return err } - err = o.stopObserveGeneration(ctx, generation, t) + err = o.stopObserveGeneration(ctx, generation, []*thread.Message{t.LastMessage()}) if err != nil { return fmt.Errorf("%w: %w", ErrOllamaChat, err) } @@ -248,11 +248,11 @@ func (o *Ollama) startObserveGeneration(ctx context.Context, t *thread.Thread) ( func (o *Ollama) stopObserveGeneration( ctx context.Context, generation *observer.Generation, - t *thread.Thread, + messages []*thread.Message, ) error { return llmobserver.StopObserveGeneration( ctx, generation, - t, + messages, ) } diff --git a/llm/openai/openai.go b/llm/openai/openai.go index bf6deba0..f0da97b8 100644 --- a/llm/openai/openai.go +++ b/llm/openai/openai.go @@ -202,6 +202,8 @@ func (o *OpenAI) Generate(ctx context.Context, t *thread.Thread) error { return fmt.Errorf("%w: %w", ErrOpenAIChat, err) } + nMessageBeforeGeneration := len(t.Messages) + if o.streamCallbackFn != nil { err = o.stream(ctx, t, chatCompletionRequest) } else { @@ -211,7 +213,7 @@ func (o *OpenAI) Generate(ctx context.Context, t *thread.Thread) error { return err } - err = o.stopObserveGeneration(ctx, generation, t) + err = o.stopObserveGeneration(ctx, generation, t.Messages[nMessageBeforeGeneration:]) if err != nil { return fmt.Errorf("%w: %w", ErrOpenAIChat, err) } @@ -454,11 +456,11 @@ func (o *OpenAI) startObserveGeneration(ctx context.Context, t *thread.Thread) ( func (o *OpenAI) stopObserveGeneration( ctx context.Context, generation *observer.Generation, - t *thread.Thread, + messages []*thread.Message, ) error { return llmobserver.StopObserveGeneration( ctx, generation, - t, + messages, ) } diff --git a/observer/langfuse/formatter.go b/observer/langfuse/formatter.go index 9409178a..8d77a5e0 100644 --- a/observer/langfuse/formatter.go +++ b/observer/langfuse/formatter.go @@ -54,22 +54,77 @@ func threadMessagesToLangfuseMSlice(messages []*thread.Message) []model.M { return mSlice } +func threadOutputMessagesToLangfuseOutput(messages []*thread.Message) any { + if len(messages) == 1 && + messages[0].Role == thread.RoleAssistant && + len(messages[0].Contents) == 1 && + messages[0].Contents[0].Type == thread.ContentTypeText { + return threadMessageToLangfuseM(messages[0]) + } + + toolCalls := model.M{} + toolMessages := []*thread.Message{} + + for _, message := range messages { + if message.Role == thread.RoleAssistant && + message.Contents[0].Type == thread.ContentTypeToolCall { + toolCalls = threadMessageToLangfuseM(message) + } else if message.Role == thread.RoleTool && + message.Contents[0].Type == thread.ContentTypeToolResponse { + toolMessages = append(toolMessages, message) + } + } + + return append([]model.M{toolCalls}, threadMessagesToLangfuseMSlice(toolMessages)...) +} + func threadMessageToLangfuseM(message *thread.Message) model.M { if message == nil { return nil } - messageContent := "" role := message.Role + if message.Role == thread.RoleTool { + data := message.Contents[0].AsToolResponseData() + m := model.M{ + "type": message.Contents[0].Type, + "id": data.ID, + "name": data.Name, + "results": data.Result, + } + + return model.M{ + "role": role, + "content": m, + } + } + + messageContent := "" + m := make([]model.M, 0) for _, content := range message.Contents { if content.Type == thread.ContentTypeText { messageContent += content.AsString() + } else if content.Type == thread.ContentTypeToolCall { + for _, data := range content.AsToolCallData() { + m = append(m, model.M{ + "type": content.Type, + "id": data.ID, + "name": data.Name, + "arguments": data.Arguments, + }) + } } } - return model.M{ + output := model.M{ "role": role, "content": messageContent, } + + if len(m) > 0 { + output["content"] = m + } + + return output } func observerGenerationToLangfuseGeneration(g *observer.Generation) *model.Generation { @@ -81,7 +136,7 @@ func observerGenerationToLangfuseGeneration(g *observer.Generation) *model.Gener Model: g.Model, ModelParameters: g.ModelParameters, Input: threadMessagesToLangfuseMSlice(g.Input), - Output: threadMessageToLangfuseM(g.Output), + Output: threadOutputMessagesToLangfuseOutput(g.Output), Metadata: g.Metadata, } } diff --git a/observer/observer.go b/observer/observer.go index 89fb8680..b3912350 100644 --- a/observer/observer.go +++ b/observer/observer.go @@ -38,7 +38,7 @@ type Generation struct { Model string ModelParameters types.M Input []*thread.Message - Output *thread.Message + Output []*thread.Message Metadata types.M } From 168974278758f38ebf3c121b31b4c63a77d207cf Mon Sep 17 00:00:00 2001 From: Simone Vellei Date: Sun, 2 Jun 2024 20:04:40 +0200 Subject: [PATCH 07/10] fix: assistant observer --- assistant/assistant.go | 11 +++++++++++ assistant/prompt.go | 2 +- examples/assistant/agent/main.go | 26 +++++++++++--------------- tool/python/python.go | 8 +++++++- 4 files changed, 30 insertions(+), 17 deletions(-) diff --git a/assistant/assistant.go b/assistant/assistant.go index 6b4764b9..fb3b9dce 100644 --- a/assistant/assistant.go +++ b/assistant/assistant.go @@ -2,6 +2,7 @@ package assistant import ( "context" + "fmt" "strings" obs "github.com/henomis/lingoose/observer" @@ -94,11 +95,21 @@ func (a *Assistant) Run(ctx context.Context) error { } for i := 0; i < int(a.maxIterations); i++ { + ctx, spanIteration, err := a.startObserveSpan(ctx, fmt.Sprintf("iteration-%d", i+1)) + if err != nil { + return err + } + err = a.llm.Generate(ctx, a.thread) if err != nil { return err } + err = a.stopObserveSpan(ctx, spanIteration) + if err != nil { + return err + } + if a.thread.LastMessage().Role != thread.RoleTool { break } diff --git a/assistant/prompt.go b/assistant/prompt.go index 29f04df8..c6f14b08 100644 --- a/assistant/prompt.go +++ b/assistant/prompt.go @@ -4,7 +4,7 @@ const ( //nolint:lll baseRAGPrompt = "Use the following pieces of retrieved context to answer the question.\n\nQuestion: {{.question}}\nContext:\n{{range .results}}{{.}}\n\n{{end}}" //nolint:lll - systemPrompt = "You name is {{.assistantName}}, and you are {{.assistantIdentity}} {{if ne .companyName \"\" }}at {{.companyName}}{{end}}{{if ne .companyDescription \"\" }}, {{.companyDescription}}{{end}}. Your task is to assist humans {{.assistantScope}}." + systemPrompt = "{{if ne .assistantName \"\"}}You name is {{.assistantName}}, {{end}}{{if ne .assistantIdentity \"\"}}you are {{.assistantIdentity}}.{{end}} {{if ne .companyName \"\" }}at {{.companyName}}{{end}}{{if ne .companyDescription \"\" }}, {{.companyDescription}}.{{end}} Your task is to assist humans {{.assistantScope}}." defaultAssistantName = "AI assistant" defaultAssistantIdentity = "a helpful and polite assistant" diff --git a/examples/assistant/agent/main.go b/examples/assistant/agent/main.go index dac59c26..5c0e723b 100644 --- a/examples/assistant/agent/main.go +++ b/examples/assistant/agent/main.go @@ -17,45 +17,41 @@ import ( func main() { ctx := context.Background() - o := langfuse.New(ctx) - trace, err := o.Trace(&observer.Trace{Name: "state of the union"}) + langfuseObserver := langfuse.New(ctx) + trace, err := langfuseObserver.Trace(&observer.Trace{Name: "Average Temperature calculator"}) if err != nil { panic(err) } - ctx = observer.ContextWithObserverInstance(ctx, o) + ctx = observer.ContextWithObserverInstance(ctx, langfuseObserver) ctx = observer.ContextWithTraceID(ctx, trace.ID) auto := "auto" - a := assistant.New( + myAssistant := assistant.New( openai.New().WithModel(openai.GPT4o).WithToolChoice(&auto).WithTools( pythontool.New(), serpapitool.New(), ), ).WithParameters( assistant.Parameters{ - AssistantName: "AI Assistant", - AssistantIdentity: "an helpful assistant", - AssistantScope: "with their questions.", - CompanyName: "", - CompanyDescription: "", + AssistantName: "AI Assistant", + AssistantIdentity: "a helpful assistant", + AssistantScope: "answering questions", }, ).WithThread( thread.New().AddMessages( thread.NewUserMessage().AddContent( - thread.NewTextContent("calculate the average temperature in celsius degrees of New York, Rome, and Tokyo."), + thread.NewTextContent("Search the current temperature of New York, Rome, and Tokyo, then calculate the average temperature in Celsius."), ), ), ).WithMaxIterations(10) - err = a.Run(ctx) + err = myAssistant.Run(ctx) if err != nil { panic(err) } - fmt.Println("----") - fmt.Println(a.Thread()) - fmt.Println("----") + fmt.Println(myAssistant.Thread()) - o.Flush(ctx) + langfuseObserver.Flush(ctx) } diff --git a/tool/python/python.go b/tool/python/python.go index 2ac686f2..ff677735 100644 --- a/tool/python/python.go +++ b/tool/python/python.go @@ -37,7 +37,7 @@ func (t *Tool) Name() string { } func (t *Tool) Description() string { - return "A tool that runs Python code using the Python interpreter. The code should print the final result to stdout." + return "A tool that runs Python code using the Python interpreter. Use this tool to solve calculations, manipulate data, or perform any other Python-related tasks. The code should print the final result to stdout." } func (t *Tool) Fn() any { @@ -63,6 +63,12 @@ func (t *Tool) fn(i Input) Output { } } + if out.String() == "" { + return Output{ + Error: "no output from script, script must print the final result to stdout", + } + } + // Return the output as a string. return Output{Result: out.String()} } From 9d6ebccb2a662c23e9a2e7b6728f45d99f723b98 Mon Sep 17 00:00:00 2001 From: Simone Vellei Date: Sun, 2 Jun 2024 20:10:17 +0200 Subject: [PATCH 08/10] fix: linting --- assistant/assistant.go | 31 ++++++++++++++++++++----------- tool/python/python.go | 1 + 2 files changed, 21 insertions(+), 11 deletions(-) diff --git a/assistant/assistant.go b/assistant/assistant.go index fb3b9dce..e591ebd1 100644 --- a/assistant/assistant.go +++ b/assistant/assistant.go @@ -95,17 +95,7 @@ func (a *Assistant) Run(ctx context.Context) error { } for i := 0; i < int(a.maxIterations); i++ { - ctx, spanIteration, err := a.startObserveSpan(ctx, fmt.Sprintf("iteration-%d", i+1)) - if err != nil { - return err - } - - err = a.llm.Generate(ctx, a.thread) - if err != nil { - return err - } - - err = a.stopObserveSpan(ctx, spanIteration) + err = a.runIteration(ctx, i) if err != nil { return err } @@ -123,6 +113,25 @@ func (a *Assistant) Run(ctx context.Context) error { return nil } +func (a *Assistant) runIteration(ctx context.Context, iteration int) error { + ctx, spanIteration, err := a.startObserveSpan(ctx, fmt.Sprintf("iteration-%d", iteration+1)) + if err != nil { + return err + } + + err = a.llm.Generate(ctx, a.thread) + if err != nil { + return err + } + + err = a.stopObserveSpan(ctx, spanIteration) + if err != nil { + return err + } + + return nil +} + func (a *Assistant) RunWithThread(ctx context.Context, thread *thread.Thread) error { a.thread = thread return a.Run(ctx) diff --git a/tool/python/python.go b/tool/python/python.go index ff677735..261e2080 100644 --- a/tool/python/python.go +++ b/tool/python/python.go @@ -36,6 +36,7 @@ func (t *Tool) Name() string { return "python" } +//nolint:lll func (t *Tool) Description() string { return "A tool that runs Python code using the Python interpreter. Use this tool to solve calculations, manipulate data, or perform any other Python-related tasks. The code should print the final result to stdout." } From 7d37e2dc27b4170c5a7573b17dc2f50dfb2742c9 Mon Sep 17 00:00:00 2001 From: Simone Vellei Date: Mon, 3 Jun 2024 10:38:04 +0200 Subject: [PATCH 09/10] fix: serpapi response parsing --- tool/serpapi/api.go | 47 ++++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 46 insertions(+), 1 deletion(-) diff --git a/tool/serpapi/api.go b/tool/serpapi/api.go index 944b71fe..15491ec3 100644 --- a/tool/serpapi/api.go +++ b/tool/serpapi/api.go @@ -24,7 +24,10 @@ type response struct { } type apiResponse struct { - OrganicResults []OrganicResults `json:"organic_results"` + AnswerBox map[string]interface{} `json:"answer_box,omitempty"` + SportsResults map[string]interface{} `json:"sports_results,omitempty"` + KnowledgeGraph map[string]interface{} `json:"knowledge_graph,omitempty"` + OrganicResults []OrganicResults `json:"organic_results"` } type Top struct { @@ -91,6 +94,48 @@ func (r *response) Decode(body io.Reader) error { return err } + if r.apiResponse.AnswerBox != nil { + answerBox, errMarshall := json.Marshal(r.apiResponse.AnswerBox) + if errMarshall != nil { + return errMarshall + } + + r.Results = append(r.Results, result{ + Title: "Answer Box", + Info: string(answerBox), + }) + + return nil + } + + if r.apiResponse.SportsResults != nil { + sportsResults, errMarshall := json.Marshal(r.apiResponse.SportsResults) + if errMarshall != nil { + return errMarshall + } + + r.Results = append(r.Results, result{ + Title: "Sports Results", + Info: string(sportsResults), + }) + + return nil + } + + if r.apiResponse.KnowledgeGraph != nil { + knowledgeGraph, errMarshall := json.Marshal(r.apiResponse.KnowledgeGraph) + if errMarshall != nil { + return errMarshall + } + + r.Results = append(r.Results, result{ + Title: "Knowledge Graph", + Info: string(knowledgeGraph), + }) + + return nil + } + for _, res := range r.apiResponse.OrganicResults { r.Results = append(r.Results, result{ Title: res.Title, From 5edf00c69a68f276b85d2fa69d6aabfe0fc2aec5 Mon Sep 17 00:00:00 2001 From: Simone Vellei Date: Wed, 5 Jun 2024 10:53:37 +0200 Subject: [PATCH 10/10] fix: tools --- examples/assistant/agent/main.go | 6 +++-- tool/human/human.go | 44 ++++++++++++++++++++++++++++++++ tool/python/python.go | 4 +-- tool/serpapi/api.go | 42 ------------------------------ 4 files changed, 50 insertions(+), 46 deletions(-) create mode 100644 tool/human/human.go diff --git a/examples/assistant/agent/main.go b/examples/assistant/agent/main.go index 5c0e723b..96441d56 100644 --- a/examples/assistant/agent/main.go +++ b/examples/assistant/agent/main.go @@ -10,6 +10,7 @@ import ( "github.com/henomis/lingoose/observer/langfuse" "github.com/henomis/lingoose/thread" + humantool "github.com/henomis/lingoose/tool/human" pythontool "github.com/henomis/lingoose/tool/python" serpapitool "github.com/henomis/lingoose/tool/serpapi" ) @@ -18,7 +19,7 @@ func main() { ctx := context.Background() langfuseObserver := langfuse.New(ctx) - trace, err := langfuseObserver.Trace(&observer.Trace{Name: "Average Temperature calculator"}) + trace, err := langfuseObserver.Trace(&observer.Trace{Name: "Italian guests calculator"}) if err != nil { panic(err) } @@ -31,6 +32,7 @@ func main() { openai.New().WithModel(openai.GPT4o).WithToolChoice(&auto).WithTools( pythontool.New(), serpapitool.New(), + humantool.New(), ), ).WithParameters( assistant.Parameters{ @@ -41,7 +43,7 @@ func main() { ).WithThread( thread.New().AddMessages( thread.NewUserMessage().AddContent( - thread.NewTextContent("Search the current temperature of New York, Rome, and Tokyo, then calculate the average temperature in Celsius."), + thread.NewTextContent("search the top 3 italian dishes and then their costs, then ask the user's budget in euros and calculate how many guests can be invited for each dish"), ), ), ).WithMaxIterations(10) diff --git a/tool/human/human.go b/tool/human/human.go new file mode 100644 index 00000000..9e6f6e55 --- /dev/null +++ b/tool/human/human.go @@ -0,0 +1,44 @@ +package human + +import ( + "fmt" +) + +type Tool struct { +} + +func New() *Tool { + return &Tool{} +} + +type Input struct { + Question string `json:"question" jsonschema:"description=the question to ask the human"` +} + +type Output struct { + Error string `json:"error,omitempty"` + Result string `json:"result,omitempty"` +} + +type FnPrototype = func(Input) Output + +func (t *Tool) Name() string { + return "human" +} + +func (t *Tool) Description() string { + return "A tool that asks a question to a human and returns the answer. Use it to interact with a human." +} + +func (t *Tool) Fn() any { + return t.fn +} + +func (t *Tool) fn(i Input) Output { + var answer string + + fmt.Printf("\n\n%s > ", i.Question) + fmt.Scanln(&answer) + + return Output{Result: answer} +} diff --git a/tool/python/python.go b/tool/python/python.go index 261e2080..3fd6b7ad 100644 --- a/tool/python/python.go +++ b/tool/python/python.go @@ -22,7 +22,7 @@ func (t *Tool) WithPythonPath(pythonPath string) *Tool { } type Input struct { - PythonCode string `json:"python_code" jsonschema:"description=python code that prints the final result to stdout."` + PythonCode string `json:"python_code" jsonschema:"description=python code that uses print() to print the final result to stdout."` } type Output struct { @@ -38,7 +38,7 @@ func (t *Tool) Name() string { //nolint:lll func (t *Tool) Description() string { - return "A tool that runs Python code using the Python interpreter. Use this tool to solve calculations, manipulate data, or perform any other Python-related tasks. The code should print the final result to stdout." + return "Use this tool to solve calculations, manipulate data, or perform any other Python-related tasks. The code should use print() to print the final result to stdout." } func (t *Tool) Fn() any { diff --git a/tool/serpapi/api.go b/tool/serpapi/api.go index 15491ec3..aa3f4c6f 100644 --- a/tool/serpapi/api.go +++ b/tool/serpapi/api.go @@ -94,48 +94,6 @@ func (r *response) Decode(body io.Reader) error { return err } - if r.apiResponse.AnswerBox != nil { - answerBox, errMarshall := json.Marshal(r.apiResponse.AnswerBox) - if errMarshall != nil { - return errMarshall - } - - r.Results = append(r.Results, result{ - Title: "Answer Box", - Info: string(answerBox), - }) - - return nil - } - - if r.apiResponse.SportsResults != nil { - sportsResults, errMarshall := json.Marshal(r.apiResponse.SportsResults) - if errMarshall != nil { - return errMarshall - } - - r.Results = append(r.Results, result{ - Title: "Sports Results", - Info: string(sportsResults), - }) - - return nil - } - - if r.apiResponse.KnowledgeGraph != nil { - knowledgeGraph, errMarshall := json.Marshal(r.apiResponse.KnowledgeGraph) - if errMarshall != nil { - return errMarshall - } - - r.Results = append(r.Results, result{ - Title: "Knowledge Graph", - Info: string(knowledgeGraph), - }) - - return nil - } - for _, res := range r.apiResponse.OrganicResults { r.Results = append(r.Results, result{ Title: res.Title,