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.

185 lines
4.7 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 gitclone
import (
"context"
"fmt"
"os"
"os/exec"
"path/filepath"
"strings"
"github.com/cloudwego/eino/components/tool"
"github.com/cloudwego/eino/components/tool/utils"
)
type GitCloneFileImpl struct {
config *GitCloneFileConfig
}
type GitCloneFileConfig struct {
BaseDir string
}
func defaultGitCloneFileConfig(ctx context.Context) (*GitCloneFileConfig, error) {
config := &GitCloneFileConfig{
BaseDir: "./data/repos",
}
return config, nil
}
func NewGitCloneFile(ctx context.Context, config *GitCloneFileConfig) (tn tool.BaseTool, err error) {
if config == nil {
config, err = defaultGitCloneFileConfig(ctx)
if err != nil {
return nil, err
}
}
if config.BaseDir == "" {
return nil, fmt.Errorf("base dir cannot be empty")
}
t := &GitCloneFileImpl{config: config}
tn, err = t.ToEinoTool()
if err != nil {
return nil, err
}
return tn, nil
}
func (g *GitCloneFileImpl) ToEinoTool() (tool.BaseTool, error) {
return utils.InferTool("gitclone", "git clone or pull a repository", g.Invoke)
}
func (g *GitCloneFileImpl) Invoke(ctx context.Context, req *GitCloneRequest) (res *GitCloneResponse, err error) {
res = &GitCloneResponse{}
if req.Url == "" {
res.Error = "URL cannot be empty"
return res, nil
}
valid, cloneURL := isValidGitURL(req.Url)
if !valid {
res.Error = fmt.Sprintf("Invalid Git URL format: %s", req.Url)
return res, nil
}
repoDir, repoName := extractRepoDir(cloneURL)
repoDir = filepath.Join(g.config.BaseDir, repoDir)
repoPath := filepath.Join(repoDir, repoName)
if err := os.MkdirAll(g.config.BaseDir, 0755); err != nil {
res.Error = fmt.Sprintf("Failed to create directory: %v", err)
return res, nil
}
if req.Action == GitCloneActionClone {
if _, err := os.Stat(repoPath); err == nil {
res.Error = "Repository already exists"
return res, nil
}
cmd := exec.CommandContext(ctx, "git", "clone", cloneURL, repoPath)
if output, err := cmd.CombinedOutput(); err != nil {
res.Error = fmt.Sprintf("Clone failed: %v, output: %s", err, output)
return res, nil
}
} else if req.Action == GitCloneActionPull {
if _, err := os.Stat(repoPath); os.IsNotExist(err) {
res.Error = fmt.Sprintf("repo does not exist: %s", repoPath)
return res, nil
}
cmd := exec.CommandContext(ctx, "git", "-C", repoPath, "pull")
if output, err := cmd.CombinedOutput(); err != nil {
res.Error = fmt.Sprintf("Pull failed: %v, output: %s", err, output)
return res, nil
}
}
absPath, err := filepath.Abs(repoPath)
if err != nil {
res.Error = fmt.Sprintf("failed to get absolute [%s] path: %v", repoPath, err)
return res, nil
}
res.Message = fmt.Sprintf("success, repo path: %s", absPath)
return res, nil
}
// 辅助函数:验证 Git URL 格式
func isValidGitURL(url string) (bool, string) {
cleanURL := strings.TrimSuffix(url, ".git")
parts := strings.Split(cleanURL, "/")
if len(parts) < 2 {
return false, ""
}
var standardURL string
switch {
// SSH 格式: git@domain:group/repo
case strings.HasPrefix(url, "git@"):
if strings.Contains(url, ":") {
return true, withGit(url) // 已经是标准 SSH 格式
}
return false, ""
// 完整 HTTPS 格式: https://domain/group/repo
case strings.HasPrefix(url, "http://"), strings.HasPrefix(url, "https://"):
return true, withGit(url) // 已经是标准 HTTPS 格式
default:
standardURL = "https://" + withGit(url)
}
return true, standardURL
}
func withGit(url string) string {
if !strings.HasSuffix(url, ".git") {
url += ".git"
}
return url
}
// 辅助函数:从 URL 提取 group 和 repo
func extractRepoDir(url string) (string, string) {
parts := strings.Split(url, "/")
repoDir := parts[len(parts)-2]
repoName := strings.TrimSuffix(parts[len(parts)-1], ".git")
return repoDir, repoName
}
type GitCloneAction string
const (
GitCloneActionClone GitCloneAction = "clone"
GitCloneActionPull GitCloneAction = "pull"
)
type GitCloneRequest struct {
Url string `json:"url" jsonschema_description:"The URL of the repository to clone"`
Action GitCloneAction `json:"action" jsonschema_description:"The action to perform, 'clone' or 'pull'"`
}
type GitCloneResponse struct {
Message string `json:"message"`
Error string `json:"error"`
}