4ab879e461
Expande el paquete tools/file/ con 4 operaciones nuevas para que los agentes puedan interactuar con carpetas de trabajo (workspaces, outputs). Cambios: - Extraer validatePath() y resolveReal() a validate.go para reutilizarlos - Agregar validateWritePath() que verifica ReadOnly == false - write_file: crea/sobreescribe archivos, crea dirs padre, limite 1MB - list_directory: lista archivos con metadata, modo recursivo, limite 500 entries - append_file: agrega contenido al final, crea si no existe, limite 10MB total - delete_file: borra solo archivos (nunca directorios), previene rm -rf accidental - Registrar las 4 tools nuevas en runtime.go condicionalmente: - list_directory: siempre (no requiere escritura) - write/append/delete: solo si ReadOnly == false Seguridad: todas las tools reutilizan validatePath() con deny-by-default, resolucion de symlinks y proteccion contra path traversal.
58 lines
1.7 KiB
Go
58 lines
1.7 KiB
Go
package file
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
|
|
"github.com/enmanuel/agents/internal/config"
|
|
"github.com/enmanuel/agents/tools"
|
|
)
|
|
|
|
// NewDeleteFile creates a delete_file tool that deletes a single file.
|
|
// Deny-by-default: if AllowedPaths is empty, all operations are rejected.
|
|
// Rejects if ReadOnly is true. Only deletes files, never directories.
|
|
// Resolves symlinks before deleting to prevent escaping allowed paths.
|
|
func NewDeleteFile(cfg config.FileOpsCfg) tools.Tool {
|
|
return tools.Tool{
|
|
Def: tools.Def{
|
|
Name: "delete_file",
|
|
Description: "Delete a single file. Cannot delete directories.",
|
|
Parameters: []tools.Param{
|
|
{Name: "path", Type: "string", Description: "Absolute path to the file to delete", Required: true},
|
|
},
|
|
},
|
|
Exec: func(ctx context.Context, args map[string]any) tools.Result {
|
|
path := tools.GetString(args, "path")
|
|
if path == "" {
|
|
return tools.Result{Err: fmt.Errorf("delete_file: path is required")}
|
|
}
|
|
|
|
absPath, err := filepath.Abs(path)
|
|
if err != nil {
|
|
return tools.Result{Err: fmt.Errorf("delete_file: %w", err)}
|
|
}
|
|
|
|
if err := validateWritePath(absPath, cfg.AllowedPaths, cfg.ReadOnly); err != nil {
|
|
return tools.Result{Err: err}
|
|
}
|
|
|
|
// Stat the file to ensure it exists and is not a directory.
|
|
info, err := os.Stat(absPath)
|
|
if err != nil {
|
|
return tools.Result{Err: fmt.Errorf("delete_file: %w", err)}
|
|
}
|
|
if info.IsDir() {
|
|
return tools.Result{Err: fmt.Errorf("delete_file: %q is a directory, only files can be deleted", absPath)}
|
|
}
|
|
|
|
if err := os.Remove(absPath); err != nil {
|
|
return tools.Result{Err: fmt.Errorf("delete_file: %w", err)}
|
|
}
|
|
|
|
return tools.Result{Output: fmt.Sprintf("deleted %s", absPath)}
|
|
},
|
|
}
|
|
}
|