2023-01-27 13:54:28 +00:00
|
|
|
package git
|
|
|
|
|
|
|
|
import (
|
|
|
|
"io"
|
|
|
|
"io/fs"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
)
|
|
|
|
|
2023-04-04 13:44:57 +00:00
|
|
|
func copyTestdata(t *testing.T, name string) string {
|
2023-01-27 13:54:28 +00:00
|
|
|
tempDir := t.TempDir()
|
|
|
|
|
2023-04-04 13:44:57 +00:00
|
|
|
// Copy everything under testdata ${name} to temporary directory.
|
|
|
|
err := filepath.WalkDir(name, func(path string, d fs.DirEntry, err error) error {
|
2023-01-27 13:54:28 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
if d.IsDir() {
|
|
|
|
err := os.MkdirAll(filepath.Join(tempDir, path), 0755)
|
|
|
|
require.NoError(t, err)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
fin, err := os.Open(path)
|
|
|
|
require.NoError(t, err)
|
|
|
|
defer fin.Close()
|
|
|
|
|
|
|
|
fout, err := os.Create(filepath.Join(tempDir, path))
|
|
|
|
require.NoError(t, err)
|
|
|
|
defer fout.Close()
|
|
|
|
|
|
|
|
_, err = io.Copy(fout, fin)
|
|
|
|
require.NoError(t, err)
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
|
|
|
require.NoError(t, err)
|
2023-04-04 13:44:57 +00:00
|
|
|
return filepath.Join(tempDir, name)
|
2023-01-27 13:54:28 +00:00
|
|
|
}
|
|
|
|
|
2023-04-04 13:44:57 +00:00
|
|
|
func createFakeRepo(t *testing.T, testdataName string) string {
|
|
|
|
absPath := copyTestdata(t, testdataName)
|
2023-01-27 13:54:28 +00:00
|
|
|
|
|
|
|
// Add .git directory to make it look like a Git repository.
|
|
|
|
err := os.Mkdir(filepath.Join(absPath, ".git"), 0755)
|
|
|
|
require.NoError(t, err)
|
|
|
|
return absPath
|
|
|
|
}
|
|
|
|
|
2023-01-31 17:34:36 +00:00
|
|
|
// Wrap a View and expose a panicking version of [View.Ignore].
|
|
|
|
type testView struct {
|
|
|
|
t *testing.T
|
|
|
|
v *View
|
|
|
|
}
|
|
|
|
|
|
|
|
func (v *testView) Ignore(relPath string) bool {
|
|
|
|
ign, err := v.v.Ignore(relPath)
|
|
|
|
require.NoError(v.t, err)
|
|
|
|
return ign
|
|
|
|
}
|
|
|
|
|
|
|
|
func testViewAtRoot(t *testing.T, tv testView) {
|
2023-01-27 13:54:28 +00:00
|
|
|
// Check .gitignore at root.
|
2023-01-31 17:34:36 +00:00
|
|
|
assert.True(t, tv.Ignore("root.sh"))
|
|
|
|
assert.True(t, tv.Ignore("root/foo"))
|
|
|
|
assert.True(t, tv.Ignore("root_double"))
|
|
|
|
assert.False(t, tv.Ignore("newfile"))
|
2023-08-21 07:35:02 +00:00
|
|
|
assert.True(t, tv.Ignore(".gitignore"))
|
|
|
|
assert.False(t, tv.Ignore("newfile.py"))
|
2023-01-31 17:34:36 +00:00
|
|
|
assert.True(t, tv.Ignore("ignoredirectory/"))
|
2023-01-27 13:54:28 +00:00
|
|
|
|
2023-08-21 07:35:02 +00:00
|
|
|
// Never ignore the root directory.
|
|
|
|
// This is the only path that may be checked as `.`,
|
|
|
|
// and would match the `.*` ignore pattern if specified.
|
|
|
|
assert.False(t, tv.Ignore("."))
|
|
|
|
|
2023-01-27 13:54:28 +00:00
|
|
|
// Nested .gitignores should not affect root.
|
2023-01-31 17:34:36 +00:00
|
|
|
assert.False(t, tv.Ignore("a.sh"))
|
2023-01-27 13:54:28 +00:00
|
|
|
|
|
|
|
// Nested .gitignores should apply in their path.
|
2023-01-31 17:34:36 +00:00
|
|
|
assert.True(t, tv.Ignore("a/a.sh"))
|
|
|
|
assert.True(t, tv.Ignore("a/whatever/a.sh"))
|
2023-01-27 13:54:28 +00:00
|
|
|
|
|
|
|
// .git must always be ignored.
|
2023-01-31 17:34:36 +00:00
|
|
|
assert.True(t, tv.Ignore(".git"))
|
2023-01-27 13:54:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestViewRootInBricksRepo(t *testing.T) {
|
|
|
|
v, err := NewView("./testdata")
|
|
|
|
require.NoError(t, err)
|
2023-01-31 17:34:36 +00:00
|
|
|
testViewAtRoot(t, testView{t, v})
|
2023-01-27 13:54:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestViewRootInTempRepo(t *testing.T) {
|
2023-04-04 13:44:57 +00:00
|
|
|
v, err := NewView(createFakeRepo(t, "testdata"))
|
2023-01-27 13:54:28 +00:00
|
|
|
require.NoError(t, err)
|
2023-01-31 17:34:36 +00:00
|
|
|
testViewAtRoot(t, testView{t, v})
|
2023-01-27 13:54:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestViewRootInTempDir(t *testing.T) {
|
2023-04-04 13:44:57 +00:00
|
|
|
v, err := NewView(copyTestdata(t, "testdata"))
|
2023-01-27 13:54:28 +00:00
|
|
|
require.NoError(t, err)
|
2023-01-31 17:34:36 +00:00
|
|
|
testViewAtRoot(t, testView{t, v})
|
2023-01-27 13:54:28 +00:00
|
|
|
}
|
|
|
|
|
2023-01-31 17:34:36 +00:00
|
|
|
func testViewAtA(t *testing.T, tv testView) {
|
2023-01-27 13:54:28 +00:00
|
|
|
// Inherit .gitignore from root.
|
2023-01-31 17:34:36 +00:00
|
|
|
assert.True(t, tv.Ignore("root.sh"))
|
|
|
|
assert.False(t, tv.Ignore("root/foo"))
|
|
|
|
assert.True(t, tv.Ignore("root_double"))
|
|
|
|
assert.True(t, tv.Ignore("ignoredirectory/"))
|
2023-01-27 13:54:28 +00:00
|
|
|
|
|
|
|
// Check current .gitignore
|
2023-01-31 17:34:36 +00:00
|
|
|
assert.True(t, tv.Ignore("a.sh"))
|
|
|
|
assert.True(t, tv.Ignore("a_double"))
|
|
|
|
assert.False(t, tv.Ignore("newfile"))
|
2023-01-27 13:54:28 +00:00
|
|
|
|
|
|
|
// Nested .gitignores should apply in their path.
|
2023-01-31 17:34:36 +00:00
|
|
|
assert.True(t, tv.Ignore("b/b.sh"))
|
|
|
|
assert.True(t, tv.Ignore("b/whatever/b.sh"))
|
2023-01-27 13:54:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestViewAInBricksRepo(t *testing.T) {
|
|
|
|
v, err := NewView("./testdata/a")
|
|
|
|
require.NoError(t, err)
|
2023-01-31 17:34:36 +00:00
|
|
|
testViewAtA(t, testView{t, v})
|
2023-01-27 13:54:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestViewAInTempRepo(t *testing.T) {
|
2023-04-04 13:44:57 +00:00
|
|
|
v, err := NewView(filepath.Join(createFakeRepo(t, "testdata"), "a"))
|
2023-01-27 13:54:28 +00:00
|
|
|
require.NoError(t, err)
|
2023-01-31 17:34:36 +00:00
|
|
|
testViewAtA(t, testView{t, v})
|
2023-01-27 13:54:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestViewAInTempDir(t *testing.T) {
|
|
|
|
// Since this is not a fake repo it should not traverse up the tree.
|
2023-04-04 13:44:57 +00:00
|
|
|
v, err := NewView(filepath.Join(copyTestdata(t, "testdata"), "a"))
|
2023-01-27 13:54:28 +00:00
|
|
|
require.NoError(t, err)
|
2023-01-31 17:34:36 +00:00
|
|
|
tv := testView{t, v}
|
2023-01-27 13:54:28 +00:00
|
|
|
|
|
|
|
// Check that this doesn't inherit .gitignore from root.
|
2023-01-31 17:34:36 +00:00
|
|
|
assert.False(t, tv.Ignore("root.sh"))
|
|
|
|
assert.False(t, tv.Ignore("root/foo"))
|
|
|
|
assert.False(t, tv.Ignore("root_double"))
|
2023-01-27 13:54:28 +00:00
|
|
|
|
|
|
|
// Check current .gitignore
|
2023-01-31 17:34:36 +00:00
|
|
|
assert.True(t, tv.Ignore("a.sh"))
|
|
|
|
assert.True(t, tv.Ignore("a_double"))
|
|
|
|
assert.False(t, tv.Ignore("newfile"))
|
2023-01-27 13:54:28 +00:00
|
|
|
|
|
|
|
// Nested .gitignores should apply in their path.
|
2023-01-31 17:34:36 +00:00
|
|
|
assert.True(t, tv.Ignore("b/b.sh"))
|
|
|
|
assert.True(t, tv.Ignore("b/whatever/b.sh"))
|
2023-01-27 13:54:28 +00:00
|
|
|
}
|
|
|
|
|
2023-01-31 17:34:36 +00:00
|
|
|
func testViewAtAB(t *testing.T, tv testView) {
|
2023-01-27 13:54:28 +00:00
|
|
|
// Inherit .gitignore from root.
|
2023-01-31 17:34:36 +00:00
|
|
|
assert.True(t, tv.Ignore("root.sh"))
|
|
|
|
assert.False(t, tv.Ignore("root/foo"))
|
|
|
|
assert.True(t, tv.Ignore("root_double"))
|
|
|
|
assert.True(t, tv.Ignore("ignoredirectory/"))
|
2023-01-27 13:54:28 +00:00
|
|
|
|
|
|
|
// Inherit .gitignore from root/a.
|
2023-01-31 17:34:36 +00:00
|
|
|
assert.True(t, tv.Ignore("a.sh"))
|
|
|
|
assert.True(t, tv.Ignore("a_double"))
|
2023-01-27 13:54:28 +00:00
|
|
|
|
|
|
|
// Check current .gitignore
|
2023-01-31 17:34:36 +00:00
|
|
|
assert.True(t, tv.Ignore("b.sh"))
|
|
|
|
assert.True(t, tv.Ignore("b_double"))
|
|
|
|
assert.False(t, tv.Ignore("newfile"))
|
2023-01-27 13:54:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestViewABInBricksRepo(t *testing.T) {
|
|
|
|
v, err := NewView("./testdata/a/b")
|
|
|
|
require.NoError(t, err)
|
2023-01-31 17:34:36 +00:00
|
|
|
testViewAtAB(t, testView{t, v})
|
2023-01-27 13:54:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestViewABInTempRepo(t *testing.T) {
|
2023-04-04 13:44:57 +00:00
|
|
|
v, err := NewView(filepath.Join(createFakeRepo(t, "testdata"), "a", "b"))
|
2023-01-27 13:54:28 +00:00
|
|
|
require.NoError(t, err)
|
2023-01-31 17:34:36 +00:00
|
|
|
testViewAtAB(t, testView{t, v})
|
2023-01-27 13:54:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestViewABInTempDir(t *testing.T) {
|
|
|
|
// Since this is not a fake repo it should not traverse up the tree.
|
2023-04-04 13:44:57 +00:00
|
|
|
v, err := NewView(filepath.Join(copyTestdata(t, "testdata"), "a", "b"))
|
2023-01-31 17:34:36 +00:00
|
|
|
tv := testView{t, v}
|
2023-01-27 13:54:28 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// Check that this doesn't inherit .gitignore from root.
|
2023-01-31 17:34:36 +00:00
|
|
|
assert.False(t, tv.Ignore("root.sh"))
|
|
|
|
assert.False(t, tv.Ignore("root/foo"))
|
|
|
|
assert.False(t, tv.Ignore("root_double"))
|
2023-01-27 13:54:28 +00:00
|
|
|
|
|
|
|
// Check that this doesn't inherit .gitignore from root/a.
|
2023-01-31 17:34:36 +00:00
|
|
|
assert.False(t, tv.Ignore("a.sh"))
|
|
|
|
assert.False(t, tv.Ignore("a_double"))
|
2023-01-27 13:54:28 +00:00
|
|
|
|
|
|
|
// Check current .gitignore
|
2023-01-31 17:34:36 +00:00
|
|
|
assert.True(t, tv.Ignore("b.sh"))
|
|
|
|
assert.True(t, tv.Ignore("b_double"))
|
|
|
|
assert.False(t, tv.Ignore("newfile"))
|
2023-01-27 13:54:28 +00:00
|
|
|
}
|
2023-04-04 13:44:57 +00:00
|
|
|
|
|
|
|
func TestViewDoesNotChangeGitignoreIfCacheDirAlreadyIgnoredAtRoot(t *testing.T) {
|
|
|
|
expected, err := os.ReadFile("./testdata_view_ignore/.gitignore")
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
repoPath := createFakeRepo(t, "testdata_view_ignore")
|
|
|
|
|
|
|
|
// Since root .gitignore already has .databricks, there should be no edits
|
|
|
|
// to root .gitignore
|
|
|
|
v, err := NewView(repoPath)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
err = v.EnsureValidGitIgnoreExists()
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
actual, err := os.ReadFile(filepath.Join(repoPath, ".gitignore"))
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
assert.Equal(t, string(expected), string(actual))
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestViewDoesNotChangeGitignoreIfCacheDirAlreadyIgnoredInSubdir(t *testing.T) {
|
|
|
|
expected, err := os.ReadFile("./testdata_view_ignore/a/.gitignore")
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
repoPath := createFakeRepo(t, "testdata_view_ignore")
|
|
|
|
|
|
|
|
// Since root .gitignore already has .databricks, there should be no edits
|
|
|
|
// to a/.gitignore
|
|
|
|
v, err := NewView(filepath.Join(repoPath, "a"))
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
err = v.EnsureValidGitIgnoreExists()
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
actual, err := os.ReadFile(filepath.Join(repoPath, v.targetPath, ".gitignore"))
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
assert.Equal(t, string(expected), string(actual))
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestViewAddsGitignoreWithCacheDir(t *testing.T) {
|
|
|
|
repoPath := createFakeRepo(t, "testdata")
|
|
|
|
err := os.Remove(filepath.Join(repoPath, ".gitignore"))
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
// Since root .gitignore was deleted, new view adds .databricks to root .gitignore
|
|
|
|
v, err := NewView(repoPath)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
err = v.EnsureValidGitIgnoreExists()
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
actual, err := os.ReadFile(filepath.Join(repoPath, ".gitignore"))
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
assert.Contains(t, string(actual), "\n.databricks\n")
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestViewAddsGitignoreWithCacheDirAtSubdir(t *testing.T) {
|
|
|
|
repoPath := createFakeRepo(t, "testdata")
|
|
|
|
err := os.Remove(filepath.Join(repoPath, ".gitignore"))
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// Since root .gitignore was deleted, new view adds .databricks to a/.gitignore
|
|
|
|
v, err := NewView(filepath.Join(repoPath, "a"))
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
err = v.EnsureValidGitIgnoreExists()
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
actual, err := os.ReadFile(filepath.Join(repoPath, v.targetPath, ".gitignore"))
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// created .gitignore has cache dir listed
|
|
|
|
assert.Contains(t, string(actual), "\n.databricks\n")
|
|
|
|
assert.NoFileExists(t, filepath.Join(repoPath, ".gitignore"))
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestViewAlwaysIgnoresCacheDir(t *testing.T) {
|
|
|
|
repoPath := createFakeRepo(t, "testdata")
|
|
|
|
|
|
|
|
v, err := NewView(repoPath)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
err = v.EnsureValidGitIgnoreExists()
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// Delete root .gitignore which contains .databricks entry
|
|
|
|
err = os.Remove(filepath.Join(repoPath, ".gitignore"))
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// taint rules to reload .gitignore
|
|
|
|
v.repo.taintIgnoreRules()
|
|
|
|
|
|
|
|
// assert .databricks is still being ignored
|
|
|
|
ign1, err := v.IgnoreDirectory(".databricks")
|
|
|
|
require.NoError(t, err)
|
|
|
|
assert.True(t, ign1)
|
|
|
|
|
|
|
|
ign2, err := v.IgnoreDirectory("a/.databricks")
|
|
|
|
require.NoError(t, err)
|
|
|
|
assert.True(t, ign2)
|
|
|
|
}
|