@@ -2,78 +2,83 @@ package git
2
2
3
3
import (
4
4
"context"
5
+ "encoding/hex"
5
6
"fmt"
7
+ "log"
8
+ "strings"
6
9
7
10
"github.com/go-git/go-git/v5"
8
11
"github.com/go-git/go-git/v5/plumbing"
12
+ "github.com/go-git/go-git/v5/plumbing/transport/http"
13
+ "helm.sh/helm/v3/pkg/repo"
9
14
)
10
15
11
- func gitCheckoutImpl (ctx context.Context , repoURL , ref , destDir string ) error {
12
- /*err := runCmds(destDir, [][]string{
13
- {"git", "init", "--quiet"},
14
- {"git", "remote", "add", "origin", repoURL},
15
-
16
- {"git", "config", "core.sparseCheckout", "true"},
17
- {"git", "sparse-checkout", "set", path},
18
- {"git", "pull", "--quiet", "--depth", "1", "origin", ref},
19
-
20
- //{"git", "fetch", "--quiet", "--tags", "origin"},
21
- //{"git", "checkout", "--quiet", ref},
22
- })*/
23
- r , err := git .PlainCloneContext (ctx , destDir , false , & git.CloneOptions {
24
- URL : repoURL ,
25
- // TODO: Auth: ...
26
- RemoteName : "origin" ,
27
- SingleBranch : true ,
28
- Depth : 1 ,
29
- NoCheckout : true ,
30
- })
16
+ func gitCheckoutImpl (ctx context.Context , repoURL , ref string , repo * repo.Entry , destDir string ) error {
17
+ cloneOpts := git.CloneOptions {
18
+ URL : repoURL ,
19
+ RemoteName : "origin" ,
20
+ NoCheckout : true ,
21
+ }
22
+ isCommitRef := isCommitSHA (ref )
23
+ if ! isCommitRef { // cannot find commit in other branch when checking out single branch
24
+ cloneOpts .SingleBranch = true
25
+ cloneOpts .Depth = 1
26
+ }
27
+ scheme := strings .SplitN (repoURL , ":" , 2 )[0 ]
28
+ switch scheme {
29
+ case "https" :
30
+ cloneOpts .Auth = & http.BasicAuth {
31
+ Username : repo .Username ,
32
+ Password : repo .Password ,
33
+ }
34
+ default :
35
+ if repo .Username != "" || repo .Password != "" {
36
+ log .Printf ("WARNING: ignoring auth config for %s since authentication is not supported for url scheme %q" , repoURL , scheme )
37
+ }
38
+ }
39
+ r , err := git .PlainCloneContext (ctx , destDir , false , & cloneOpts )
31
40
if err != nil {
32
- return err
41
+ return fmt . Errorf ( "git clone: %w" , err )
33
42
}
34
43
tree , err := r .Worktree ()
35
44
if err != nil {
36
45
return fmt .Errorf ("git worktree: %w" , err )
37
46
}
38
47
// TODO: support sparse checkout, see https://github.com/go-git/go-git/issues/90
48
+ refType := "without ref"
39
49
opts := git.CheckoutOptions {}
40
50
if ref != "" {
41
- opts .Branch = plumbing .ReferenceName ("refs/tags/" + ref )
51
+ if isCommitRef {
52
+ opts .Hash = plumbing .NewHash (ref )
53
+ refType = fmt .Sprintf ("commit %s" , ref )
54
+ } else {
55
+ opts .Branch = plumbing .ReferenceName (fmt .Sprintf ("refs/tags/%s" , ref ))
56
+ refType = fmt .Sprintf ("tag %s" , ref )
57
+ }
42
58
}
43
59
err = tree .Checkout (& opts )
44
60
if err != nil {
45
- return err
61
+ return fmt . Errorf ( "git checkout %s: %w" , refType , err )
46
62
}
47
- return nil
48
- }
63
+ /*err := runCmds(destDir, [][]string{
64
+ {"git", "init", "--quiet"},
65
+ {"git", "remote", "add", "origin", repoURL},
49
66
50
- /*
51
- func runCmds(dir string, cmds [][]string) error {
52
- for _, c := range cmds {
53
- err := runCmd(dir, c[0], c[1:]...)
54
- if err != nil {
55
- return err
56
- }
57
- }
67
+ {"git", "config", "core.sparseCheckout", "true"},
68
+ {"git", "sparse-checkout", "set", path},
69
+ {"git", "pull", "--quiet", "--depth", "1", "origin", ref},
70
+
71
+ //{"git", "fetch", "--quiet", "--tags", "origin"},
72
+ //{"git", "checkout", "--quiet", ref},
73
+ })*/
58
74
return nil
59
75
}
60
76
61
- func runCmd(dir, cmd string, args ...string) error {
62
- ctx, cancel := context.WithTimeout(context.Background(), 30*time.Second)
63
- defer cancel()
64
- c := exec.CommandContext(ctx, cmd, args...)
65
- var stderr bytes.Buffer
66
- c.Stderr = &stderr
67
- c.Dir = dir
68
- err := c.Run()
69
- if err != nil {
70
- msg := strings.TrimSpace(stderr.String())
71
- if msg == "" {
72
- msg = err.Error()
77
+ func isCommitSHA (s string ) bool {
78
+ if len (s ) == 40 {
79
+ if _ , err := hex .DecodeString (s ); err == nil {
80
+ return true
73
81
}
74
- cmds := append([]string{cmd}, args...)
75
- return fmt.Errorf("%s: %s", strings.Join(cmds, " "), msg)
76
82
}
77
- return err
83
+ return false
78
84
}
79
- */
0 commit comments