Skip to content

Commit dc7f6bc

Browse files
authored
Backport fix from #497 to v2 (#498)
* Recreate git config during update to prevent git config alteration Backport fix from #497 to v2 * Bump version of Golang container to match module version The Go module for v2 is 1.19, bumping the base docker image to match the minimum version ensures go-getter can be compiled and executed on the container. This change resolves the failing acceptance test for Samba. ``` Run docker exec -i gogetter bash -c "env ACC_SMB_TEST=1 go test -v ./... -run=TestSmb_" docker exec -i gogetter bash -c "env ACC_SMB_TEST=1 go test -v ./... -run=TestSmb_" shell: /usr/bin/bash -e {0} env: TEST_RESULTS_PATH: /tmp/test-results Error: ./get_git.go:366:16: undefined: os.ReadDir Error: ./get_git_test.go:886:9: undefined: os.WriteFile Error: ./get_git_test.go:904:22: undefined: os.ReadFile note: module requires Go 1.19 FAIL github.com/hashicorp/go-getter/v2 [build failed] ? github.com/hashicorp/go-getter/v2/helper/testing [no test files] testing: warning: no tests to run PASS ok github.com/hashicorp/go-getter/v2/helper/url 0.006s [no tests to run] FAIL Error: Process completed with exit code 2. ```
1 parent e310de3 commit dc7f6bc

File tree

3 files changed

+220
-14
lines changed

3 files changed

+220
-14
lines changed

Dockerfile

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
# Dockerfile to create a go-getter container with smbclient dependency that is used by the get_smb.go tests
2-
FROM golang:1.15
2+
FROM golang:1.19.13
33

44
COPY . /go-getter
55
WORKDIR /go-getter

get_git.go

+70-13
Original file line numberDiff line numberDiff line change
@@ -122,7 +122,7 @@ func (g *GitGetter) Get(ctx context.Context, req *Request) error {
122122
return err
123123
}
124124
if err == nil {
125-
err = g.update(ctx, req.Dst, sshKeyFile, ref, depth)
125+
err = g.update(ctx, req.Dst, sshKeyFile, ref, req.URL(), depth)
126126
} else {
127127
err = g.clone(ctx, sshKeyFile, depth, req)
128128
}
@@ -193,28 +193,63 @@ func (g *GitGetter) clone(ctx context.Context, sshKeyFile string, depth int, req
193193
return getRunCommand(cmd)
194194
}
195195

196-
func (g *GitGetter) update(ctx context.Context, dst, sshKeyFile, ref string, depth int) error {
197-
// Determine if we're a branch. If we're NOT a branch, then we just
198-
// switch to master prior to checking out
199-
cmd := exec.CommandContext(ctx, "git", "show-ref", "-q", "--verify", "refs/heads/"+ref)
196+
func (g *GitGetter) update(ctx context.Context, dst, sshKeyFile, ref string, u *url.URL, depth int) error {
197+
// Remove all variations of .git directories
198+
err := removeCaseInsensitiveGitDirectory(dst)
199+
if err != nil {
200+
return err
201+
202+
}
203+
204+
// Initialize the git repository
205+
cmd := exec.CommandContext(ctx, "git", "init")
206+
cmd.Dir = dst
207+
err = getRunCommand(cmd)
208+
if err != nil {
209+
return err
210+
}
211+
212+
// Add the git remote
213+
cmd = exec.CommandContext(ctx, "git", "remote", "add", "origin", "--", u.String())
214+
cmd.Dir = dst
215+
err = getRunCommand(cmd)
216+
if err != nil {
217+
return err
218+
}
219+
220+
// Fetch the remote ref
221+
cmd = exec.CommandContext(ctx, "git", "fetch", "--tags")
222+
cmd.Dir = dst
223+
err = getRunCommand(cmd)
224+
if err != nil {
225+
return err
226+
}
227+
228+
// Fetch the remote ref
229+
cmd = exec.CommandContext(ctx, "git", "fetch", "origin", "--", ref)
200230
cmd.Dir = dst
231+
err = getRunCommand(cmd)
232+
if err != nil {
233+
return err
234+
}
201235

202-
if getRunCommand(cmd) != nil {
203-
// Not a branch, switch to default branch. This will also catch
204-
// non-existent branches, in which case we want to switch to default
205-
// and then checkout the proper branch later.
206-
ref = findDefaultBranch(ctx, dst)
236+
// Reset the branch to the fetched ref
237+
cmd = exec.CommandContext(ctx, "git", "reset", "--hard", "FETCH_HEAD")
238+
cmd.Dir = dst
239+
err = getRunCommand(cmd)
240+
if err != nil {
241+
return err
207242
}
208243

209-
// We have to be on a branch to pull
244+
// Checkout ref branch
210245
if err := g.checkout(ctx, dst, ref); err != nil {
211246
return err
212247
}
213248

214249
if depth > 0 {
215-
cmd = exec.CommandContext(ctx, "git", "pull", "--depth", strconv.Itoa(depth), "--ff-only")
250+
cmd = exec.CommandContext(ctx, "git", "pull", "origin", "--depth", strconv.Itoa(depth), "--ff-only", "--", ref)
216251
} else {
217-
cmd = exec.CommandContext(ctx, "git", "pull", "--ff-only")
252+
cmd = exec.CommandContext(ctx, "git", "pull", "origin", "--ff-only", "--", ref)
218253
}
219254

220255
cmd.Dir = dst
@@ -326,6 +361,28 @@ func checkGitVersion(ctx context.Context, min string) error {
326361
return nil
327362
}
328363

364+
// removeCaseInsensitiveGitDirectory removes all .git directory variations
365+
func removeCaseInsensitiveGitDirectory(dst string) error {
366+
files, err := os.ReadDir(dst)
367+
if err != nil {
368+
return fmt.Errorf("Failed to read the destination directory %s during git update", dst)
369+
370+
}
371+
for _, f := range files {
372+
if strings.EqualFold(f.Name(), ".git") && f.IsDir() {
373+
err := os.RemoveAll(filepath.Join(dst, f.Name()))
374+
if err != nil {
375+
return fmt.Errorf("Failed to remove the .git directory in the destination directory %s during git update", dst)
376+
377+
}
378+
379+
}
380+
381+
}
382+
return nil
383+
384+
}
385+
329386
func (g *GitGetter) Detect(req *Request) (bool, error) {
330387
src := req.Src
331388
if len(src) == 0 {

get_git_test.go

+149
Original file line numberDiff line numberDiff line change
@@ -138,6 +138,7 @@ func TestGitGetter_remoteWithoutMaster(t *testing.T) {
138138
}
139139

140140
// Get again should work
141+
req.Dst = testing_helper.TempDir(t)
141142
if err := g.Get(ctx, req); err != nil {
142143
t.Fatalf("err: %s", err)
143144
}
@@ -841,6 +842,154 @@ func TestGitGetter_subdirectory_traversal(t *testing.T) {
841842
}
842843
}
843844

845+
func TestGitGetter_BadGitConfig(t *testing.T) {
846+
if !testHasGit {
847+
t.Log("git not found, skipping")
848+
t.Skip()
849+
850+
}
851+
852+
ctx := context.Background()
853+
g := new(GitGetter)
854+
dst := testing_helper.TempDir(t)
855+
856+
url, err := url.Parse("https://github.com/hashicorp/go-getter")
857+
if err != nil {
858+
t.Fatal(err)
859+
860+
}
861+
862+
_, err = os.Stat(dst)
863+
if err != nil && !os.IsNotExist(err) {
864+
t.Fatalf(err.Error())
865+
}
866+
867+
if err == nil {
868+
// Update the repository containing the bad git config.
869+
// This should remove the bad git config file and initialize a new one.
870+
err = g.update(ctx, dst, testGitToken, "main", url, 1)
871+
872+
} else {
873+
// Clone a repository with a git config file
874+
req := &Request{
875+
Dst: dst,
876+
u: url,
877+
}
878+
err = g.clone(ctx, testGitToken, 1, req)
879+
if err != nil {
880+
t.Fatalf(err.Error())
881+
882+
}
883+
884+
// Edit the git config file to simulate a bad git config
885+
gitConfigPath := filepath.Join(dst, ".git", "config")
886+
err = os.WriteFile(gitConfigPath, []byte("bad config"), 0600)
887+
if err != nil {
888+
t.Fatalf(err.Error())
889+
890+
}
891+
892+
// Update the repository containing the bad git config.
893+
// This should remove the bad git config file and initialize a new one.
894+
err = g.update(ctx, dst, testGitToken, "main", url, 1)
895+
896+
}
897+
if err != nil {
898+
t.Fatalf(err.Error())
899+
900+
}
901+
902+
// Check if the .git/config file contains "bad config"
903+
gitConfigPath := filepath.Join(dst, ".git", "config")
904+
configBytes, err := os.ReadFile(gitConfigPath)
905+
if err != nil {
906+
t.Fatalf(err.Error())
907+
908+
}
909+
if strings.Contains(string(configBytes), "bad config") {
910+
t.Fatalf("The .git/config file contains 'bad config'")
911+
912+
}
913+
914+
}
915+
916+
func TestGitGetter_BadGitDirName(t *testing.T) {
917+
if !testHasGit {
918+
t.Log("git not found, skipping")
919+
t.Skip()
920+
921+
}
922+
923+
ctx := context.Background()
924+
g := new(GitGetter)
925+
dst := testing_helper.TempDir(t)
926+
927+
url, err := url.Parse("https://github.com/hashicorp/go-getter")
928+
if err != nil {
929+
t.Fatal(err)
930+
931+
}
932+
933+
_, err = os.Stat(dst)
934+
if err != nil && !os.IsNotExist(err) {
935+
t.Fatalf(err.Error())
936+
}
937+
if err == nil {
938+
// Remove all variations of .git directories
939+
err = removeCaseInsensitiveGitDirectory(dst)
940+
if err != nil {
941+
t.Fatalf(err.Error())
942+
943+
}
944+
945+
} else {
946+
// Clone a repository with a git directory
947+
req := &Request{
948+
Dst: dst,
949+
u: url,
950+
}
951+
err = g.clone(ctx, testGitToken, 1, req)
952+
if err != nil {
953+
t.Fatalf(err.Error())
954+
955+
}
956+
957+
// Rename the .git directory to .GIT
958+
oldPath := filepath.Join(dst, ".git")
959+
newPath := filepath.Join(dst, ".GIT")
960+
err = os.Rename(oldPath, newPath)
961+
if err != nil {
962+
t.Fatalf(err.Error())
963+
964+
}
965+
966+
// Remove all variations of .git directories
967+
err = removeCaseInsensitiveGitDirectory(dst)
968+
if err != nil {
969+
t.Fatalf(err.Error())
970+
971+
}
972+
973+
}
974+
if err != nil {
975+
t.Fatalf(err.Error())
976+
977+
}
978+
979+
// Check if the .GIT directory exists
980+
if _, err := os.Stat(filepath.Join(dst, ".GIT")); !os.IsNotExist(err) {
981+
t.Fatalf(".GIT directory still exists")
982+
983+
}
984+
985+
// Check if the .git directory exists
986+
if _, err := os.Stat(filepath.Join(dst, ".git")); !os.IsNotExist(err) {
987+
t.Fatalf(".git directory still exists")
988+
989+
}
990+
991+
}
992+
844993
// gitRepo is a helper struct which controls a single temp git repo.
845994
type gitRepo struct {
846995
t *testing.T

0 commit comments

Comments
 (0)