package git import ( "context" "crypto/tls" "net/http" "os" "path/filepath" "strings" "time" "github.com/pkg/errors" "github.com/go-git/go-git/v5" "github.com/go-git/go-git/v5/config" "github.com/go-git/go-git/v5/plumbing" "github.com/go-git/go-git/v5/plumbing/transport/client" githttp "github.com/go-git/go-git/v5/plumbing/transport/http" "github.com/go-git/go-git/v5/storage/memory" ) var ( ErrAuthenticationFailure = errors.New("Authentication failed, please ensure that the git credentials are correct.") ) type fetchOptions struct { repositoryUrl string username string password string referenceName string } type cloneOptions struct { repositoryUrl string username string password string referenceName string depth int } type downloader interface { download(ctx context.Context, dst string, opt cloneOptions) error latestCommitID(ctx context.Context, opt fetchOptions) (string, error) } type gitClient struct { preserveGitDirectory bool } func (c gitClient) download(ctx context.Context, dst string, opt cloneOptions) error { gitOptions := git.CloneOptions{ URL: opt.repositoryUrl, Depth: opt.depth, Auth: getAuth(opt.username, opt.password), } if opt.referenceName != "" { gitOptions.ReferenceName = plumbing.ReferenceName(opt.referenceName) } _, err := git.PlainCloneContext(ctx, dst, false, &gitOptions) if err != nil { if err.Error() == "authentication required" { return ErrAuthenticationFailure } return errors.Wrap(err, "failed to clone git repository") } if !c.preserveGitDirectory { os.RemoveAll(filepath.Join(dst, ".git")) } return nil } func (c gitClient) latestCommitID(ctx context.Context, opt fetchOptions) (string, error) { remote := git.NewRemote(memory.NewStorage(), &config.RemoteConfig{ Name: "origin", URLs: []string{opt.repositoryUrl}, }) listOptions := &git.ListOptions{ Auth: getAuth(opt.username, opt.password), } refs, err := remote.List(listOptions) if err != nil { if err.Error() == "authentication required" { return "", ErrAuthenticationFailure } return "", errors.Wrap(err, "failed to list repository refs") } referenceName := opt.referenceName if referenceName == "" { for _, ref := range refs { if strings.EqualFold(ref.Name().String(), "HEAD") { referenceName = ref.Target().String() } } } for _, ref := range refs { if strings.EqualFold(ref.Name().String(), referenceName) { return ref.Hash().String(), nil } } return "", errors.Errorf("could not find ref %q in the repository", opt.referenceName) } func getAuth(username, password string) *githttp.BasicAuth { if password != "" { if username == "" { username = "token" } return &githttp.BasicAuth{ Username: username, Password: password, } } return nil } // Service represents a service for managing Git. type Service struct { httpsCli *http.Client azure downloader git downloader } // NewService initializes a new service. func NewService() *Service { httpsCli := &http.Client{ Transport: &http.Transport{ TLSClientConfig: &tls.Config{InsecureSkipVerify: true}, Proxy: http.ProxyFromEnvironment, }, Timeout: 300 * time.Second, } client.InstallProtocol("https", githttp.NewClient(httpsCli)) return &Service{ httpsCli: httpsCli, azure: NewAzureDownloader(httpsCli), git: gitClient{}, } } // CloneRepository clones a git repository using the specified URL in the specified // destination folder. func (service *Service) CloneRepository(destination, repositoryURL, referenceName, username, password string) error { options := cloneOptions{ repositoryUrl: repositoryURL, username: username, password: password, referenceName: referenceName, depth: 1, } return service.cloneRepository(destination, options) } func (service *Service) cloneRepository(destination string, options cloneOptions) error { if isAzureUrl(options.repositoryUrl) { return service.azure.download(context.TODO(), destination, options) } return service.git.download(context.TODO(), destination, options) } // LatestCommitID returns SHA1 of the latest commit of the specified reference func (service *Service) LatestCommitID(repositoryURL, referenceName, username, password string) (string, error) { options := fetchOptions{ repositoryUrl: repositoryURL, username: username, password: password, referenceName: referenceName, } if isAzureUrl(options.repositoryUrl) { return service.azure.latestCommitID(context.TODO(), options) } return service.git.latestCommitID(context.TODO(), options) }