mirror of https://github.com/portainer/portainer
190 lines
4.5 KiB
Go
190 lines
4.5 KiB
Go
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)
|
|
}
|