You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
189 lines
5.2 KiB
Go
189 lines
5.2 KiB
Go
/*
|
|
* Copyright 2025 CloudWeGo Authors
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
package tools
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"log"
|
|
"path/filepath"
|
|
"strings"
|
|
|
|
"github.com/cloudwego/eino/components/tool"
|
|
"github.com/cloudwego/eino/schema"
|
|
jsoniter "github.com/json-iterator/go"
|
|
|
|
"github.com/cloudwego/eino-examples/adk/multiagent/deep/utils"
|
|
)
|
|
|
|
type ToolRequestPreprocess func(ctx context.Context, baseTool tool.InvokableTool, toolArguments string) (string, error)
|
|
|
|
type ToolResponsePostprocess func(ctx context.Context, baseTool tool.InvokableTool, toolResponse, toolArguments string) (string, error)
|
|
|
|
func NewWrapTool(t tool.InvokableTool, preprocess []ToolRequestPreprocess, postprocess []ToolResponsePostprocess) tool.InvokableTool {
|
|
return &wrapTool{
|
|
baseTool: t,
|
|
preprocess: preprocess,
|
|
postprocess: postprocess,
|
|
}
|
|
}
|
|
|
|
type wrapTool struct {
|
|
baseTool tool.InvokableTool
|
|
preprocess []ToolRequestPreprocess
|
|
postprocess []ToolResponsePostprocess
|
|
}
|
|
|
|
func (w *wrapTool) Info(ctx context.Context) (*schema.ToolInfo, error) {
|
|
return w.baseTool.Info(ctx)
|
|
}
|
|
|
|
func (w *wrapTool) InvokableRun(ctx context.Context, argumentsInJSON string, opts ...tool.Option) (string, error) {
|
|
for _, pre := range w.preprocess {
|
|
var err error
|
|
argumentsInJSON, err = pre(ctx, w.baseTool, argumentsInJSON)
|
|
if err != nil {
|
|
log.Printf("[WrapTool.PreProcess] failed to process response: %v", err)
|
|
}
|
|
}
|
|
|
|
resp, err := w.baseTool.InvokableRun(ctx, argumentsInJSON, opts...)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
for _, post := range w.postprocess {
|
|
resp, err = post(ctx, w.baseTool, resp, argumentsInJSON)
|
|
if err != nil {
|
|
log.Printf("[WrapTool.PostProcess] failed to process response: %v", err)
|
|
return resp, err
|
|
}
|
|
}
|
|
|
|
return resp, nil
|
|
}
|
|
|
|
func ToolRequestRepairJSON(ctx context.Context, baseTool tool.InvokableTool, toolArguments string) (string, error) {
|
|
return utils.RepairJSON(toolArguments), nil
|
|
}
|
|
|
|
type runResult struct {
|
|
Command string `json:"command"`
|
|
StdOut []*stdoutData `json:"stdout"`
|
|
StdErr []*stderrData `json:"stderr"`
|
|
FileChange []*fileChangeData `json:"file_change"`
|
|
ErrData []*errData `json:"err_data"`
|
|
}
|
|
|
|
type stdoutData struct {
|
|
Stdout string `json:"stdout"`
|
|
}
|
|
|
|
type stderrData struct {
|
|
Stderr string `json:"stderr"`
|
|
}
|
|
|
|
type errData struct {
|
|
Err string `json:"err"`
|
|
}
|
|
|
|
type fileChangeData struct {
|
|
FileType string `json:"file_type"`
|
|
Path string `json:"path"`
|
|
Type string `json:"type"`
|
|
Uri string `json:"uri"`
|
|
Url string `json:"url"`
|
|
MultiMediaInfo *multiMediaInfo `json:"multi_media_info,omitempty"`
|
|
}
|
|
|
|
type multiMediaInfo struct {
|
|
MediaType string `json:"media_type"`
|
|
AdditionalType string `json:"additional_type"`
|
|
AdditionalInfo string `json:"additional_info"`
|
|
}
|
|
|
|
func FilePostProcess(ctx context.Context, baseTool tool.InvokableTool, toolResponse, toolArguments string) (string, error) {
|
|
rawResult := runResult{}
|
|
if err := json.Unmarshal([]byte(toolResponse), &rawResult); err != nil {
|
|
return toolResponse, nil
|
|
}
|
|
|
|
type fileOutputFormat struct {
|
|
FileType string `json:"Change subject (file/directory)"`
|
|
Path string `json:"File/directory relative path"`
|
|
Type string `json:"Change type (create/delete/update)"`
|
|
}
|
|
var (
|
|
stdOut []string
|
|
fileChange []fileOutputFormat
|
|
stdErr []string
|
|
)
|
|
for _, item := range rawResult.StdOut {
|
|
if item != nil {
|
|
stdOut = append(stdOut, item.Stdout)
|
|
}
|
|
}
|
|
for _, item := range rawResult.FileChange {
|
|
if item != nil {
|
|
fileChange = append(fileChange, fileOutputFormat{
|
|
FileType: item.FileType,
|
|
Path: item.Path,
|
|
Type: item.Type,
|
|
})
|
|
}
|
|
}
|
|
for _, item := range rawResult.StdErr {
|
|
if item != nil {
|
|
stdErr = append(stdErr, item.Stderr)
|
|
}
|
|
}
|
|
for _, item := range rawResult.ErrData {
|
|
if item != nil {
|
|
stdErr = append(stdErr, item.Err)
|
|
}
|
|
}
|
|
|
|
var output string
|
|
if len(fileChange) > 0 {
|
|
fcText, _ := jsoniter.MarshalToString(fileChange)
|
|
output += "fileChange: \n" + fcText + "\n"
|
|
}
|
|
if len(stdErr) > 0 {
|
|
output += "shell command stderr and warnings:" + strings.Join(stdErr, "\n") + "\n"
|
|
}
|
|
if len(rawResult.StdOut) > 0 {
|
|
output += "shell command stdout: " + strings.Join(stdOut, "\n") + "\n"
|
|
}
|
|
|
|
return output, nil
|
|
}
|
|
|
|
func EditFilePostProcess(ctx context.Context, baseTool tool.InvokableTool, toolResponse, toolArguments string) (string, error) {
|
|
return fmt.Sprintf("Write file: %s success!", toolResponse), nil
|
|
}
|
|
|
|
func isImage(uri string) bool {
|
|
ext := filepath.Ext(uri)
|
|
for _, e := range []string{".jpg", ".jpeg", ".png", ".gif", ".bmp", ".webp", ".tiff", ".heic"} {
|
|
if ext == e {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|