2022-09-14 15:50:29 +00:00
|
|
|
package sync
|
|
|
|
|
|
|
|
import (
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"testing"
|
2022-09-15 13:40:47 +00:00
|
|
|
"time"
|
2022-09-14 15:50:29 +00:00
|
|
|
|
|
|
|
"github.com/databricks/bricks/git"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
)
|
|
|
|
|
2022-11-21 22:42:09 +00:00
|
|
|
type testFile struct {
|
|
|
|
mtime time.Time
|
|
|
|
fd *os.File
|
|
|
|
path string
|
2022-11-28 10:34:25 +00:00
|
|
|
// to make close idempotent
|
|
|
|
isOpen bool
|
2022-11-21 22:42:09 +00:00
|
|
|
}
|
2022-09-14 15:50:29 +00:00
|
|
|
|
2022-11-21 22:42:09 +00:00
|
|
|
func createFile(t *testing.T, path string) *testFile {
|
|
|
|
f, err := os.Create(path)
|
2022-09-14 15:50:29 +00:00
|
|
|
assert.NoError(t, err)
|
2022-11-21 22:42:09 +00:00
|
|
|
|
|
|
|
fileInfo, err := os.Stat(path)
|
2022-09-14 15:50:29 +00:00
|
|
|
assert.NoError(t, err)
|
|
|
|
|
2022-11-21 22:42:09 +00:00
|
|
|
return &testFile{
|
2022-11-28 10:34:25 +00:00
|
|
|
path: path,
|
|
|
|
fd: f,
|
|
|
|
mtime: fileInfo.ModTime(),
|
|
|
|
isOpen: true,
|
2022-11-21 22:42:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *testFile) close(t *testing.T) {
|
2022-11-28 10:34:25 +00:00
|
|
|
if f.isOpen {
|
|
|
|
err := f.fd.Close()
|
|
|
|
assert.NoError(t, err)
|
|
|
|
f.isOpen = false
|
|
|
|
}
|
2022-11-21 22:42:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (f *testFile) overwrite(t *testing.T, s string) {
|
|
|
|
err := os.Truncate(f.path, 0)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
_, err = f.fd.Seek(0, 0)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
_, err = f.fd.WriteString(s)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
// We manually update mtime after write because github actions file
|
|
|
|
// system does not :')
|
|
|
|
err = os.Chtimes(f.path, f.mtime.Add(time.Minute), f.mtime.Add(time.Minute))
|
|
|
|
assert.NoError(t, err)
|
|
|
|
f.mtime = f.mtime.Add(time.Minute)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *testFile) remove(t *testing.T) {
|
2022-11-28 10:34:25 +00:00
|
|
|
f.close(t)
|
2022-11-21 22:42:09 +00:00
|
|
|
err := os.Remove(f.path)
|
2022-09-14 15:50:29 +00:00
|
|
|
assert.NoError(t, err)
|
2022-11-21 22:42:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func assertKeysOfMap(t *testing.T, m map[string]time.Time, expectedKeys []string) {
|
|
|
|
keys := make([]string, len(m))
|
|
|
|
i := 0
|
|
|
|
for k := range m {
|
|
|
|
keys[i] = k
|
|
|
|
i++
|
|
|
|
}
|
|
|
|
assert.ElementsMatch(t, expectedKeys, keys)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestDiff(t *testing.T) {
|
|
|
|
// Create temp project dir
|
|
|
|
projectDir := t.TempDir()
|
|
|
|
fileSet := git.NewFileSet(projectDir)
|
2022-10-19 14:22:55 +00:00
|
|
|
state := Snapshot{
|
2022-11-21 22:42:09 +00:00
|
|
|
LastUpdatedTimes: make(map[string]time.Time),
|
|
|
|
LocalToRemoteNames: make(map[string]string),
|
|
|
|
RemoteToLocalNames: make(map[string]string),
|
2022-10-19 14:22:55 +00:00
|
|
|
}
|
2022-09-14 15:50:29 +00:00
|
|
|
|
2022-11-21 22:42:09 +00:00
|
|
|
f1 := createFile(t, filepath.Join(projectDir, "hello.txt"))
|
|
|
|
defer f1.close(t)
|
|
|
|
worldFilePath := filepath.Join(projectDir, "world.txt")
|
|
|
|
f2 := createFile(t, worldFilePath)
|
|
|
|
defer f2.close(t)
|
|
|
|
|
|
|
|
// New files are put
|
|
|
|
files, err := fileSet.All()
|
|
|
|
assert.NoError(t, err)
|
|
|
|
change, err := state.diff(files)
|
|
|
|
assert.NoError(t, err)
|
2022-09-14 15:50:29 +00:00
|
|
|
assert.Len(t, change.delete, 0)
|
|
|
|
assert.Len(t, change.put, 2)
|
|
|
|
assert.Contains(t, change.put, "hello.txt")
|
|
|
|
assert.Contains(t, change.put, "world.txt")
|
2022-11-21 22:42:09 +00:00
|
|
|
assertKeysOfMap(t, state.LastUpdatedTimes, []string{"hello.txt", "world.txt"})
|
|
|
|
assert.Equal(t, map[string]string{"hello.txt": "hello.txt", "world.txt": "world.txt"}, state.LocalToRemoteNames)
|
|
|
|
assert.Equal(t, map[string]string{"hello.txt": "hello.txt", "world.txt": "world.txt"}, state.RemoteToLocalNames)
|
2022-09-14 15:50:29 +00:00
|
|
|
|
2022-11-21 22:42:09 +00:00
|
|
|
// world.txt is editted
|
|
|
|
f2.overwrite(t, "bunnies are cute.")
|
2022-09-14 15:50:29 +00:00
|
|
|
assert.NoError(t, err)
|
|
|
|
files, err = fileSet.All()
|
|
|
|
assert.NoError(t, err)
|
2022-11-21 22:42:09 +00:00
|
|
|
change, err = state.diff(files)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
2022-09-14 15:50:29 +00:00
|
|
|
assert.Len(t, change.delete, 0)
|
|
|
|
assert.Len(t, change.put, 1)
|
|
|
|
assert.Contains(t, change.put, "world.txt")
|
2022-11-21 22:42:09 +00:00
|
|
|
assertKeysOfMap(t, state.LastUpdatedTimes, []string{"hello.txt", "world.txt"})
|
|
|
|
assert.Equal(t, map[string]string{"hello.txt": "hello.txt", "world.txt": "world.txt"}, state.LocalToRemoteNames)
|
|
|
|
assert.Equal(t, map[string]string{"hello.txt": "hello.txt", "world.txt": "world.txt"}, state.RemoteToLocalNames)
|
2022-09-14 15:50:29 +00:00
|
|
|
|
2022-11-21 22:42:09 +00:00
|
|
|
// hello.txt is deleted
|
|
|
|
f1.remove(t)
|
2022-09-14 15:50:29 +00:00
|
|
|
assert.NoError(t, err)
|
|
|
|
files, err = fileSet.All()
|
|
|
|
assert.NoError(t, err)
|
2022-11-21 22:42:09 +00:00
|
|
|
change, err = state.diff(files)
|
|
|
|
assert.NoError(t, err)
|
2022-09-14 15:50:29 +00:00
|
|
|
assert.Len(t, change.delete, 1)
|
|
|
|
assert.Len(t, change.put, 0)
|
|
|
|
assert.Contains(t, change.delete, "hello.txt")
|
2022-11-21 22:42:09 +00:00
|
|
|
assertKeysOfMap(t, state.LastUpdatedTimes, []string{"world.txt"})
|
|
|
|
assert.Equal(t, map[string]string{"world.txt": "world.txt"}, state.LocalToRemoteNames)
|
|
|
|
assert.Equal(t, map[string]string{"world.txt": "world.txt"}, state.RemoteToLocalNames)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestPythonNotebookDiff(t *testing.T) {
|
|
|
|
// Create temp project dir
|
|
|
|
projectDir := t.TempDir()
|
|
|
|
fileSet := git.NewFileSet(projectDir)
|
|
|
|
state := Snapshot{
|
|
|
|
LastUpdatedTimes: make(map[string]time.Time),
|
|
|
|
LocalToRemoteNames: make(map[string]string),
|
|
|
|
RemoteToLocalNames: make(map[string]string),
|
|
|
|
}
|
|
|
|
|
|
|
|
foo := createFile(t, filepath.Join(projectDir, "foo.py"))
|
|
|
|
defer foo.close(t)
|
|
|
|
|
|
|
|
// Case 1: notebook foo.py is uploaded
|
|
|
|
files, err := fileSet.All()
|
|
|
|
assert.NoError(t, err)
|
|
|
|
foo.overwrite(t, "# Databricks notebook source\nprint(\"abc\")")
|
|
|
|
change, err := state.diff(files)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Len(t, change.delete, 0)
|
|
|
|
assert.Len(t, change.put, 1)
|
|
|
|
assert.Contains(t, change.put, "foo.py")
|
|
|
|
assertKeysOfMap(t, state.LastUpdatedTimes, []string{"foo.py"})
|
|
|
|
assert.Equal(t, map[string]string{"foo.py": "foo"}, state.LocalToRemoteNames)
|
|
|
|
assert.Equal(t, map[string]string{"foo": "foo.py"}, state.RemoteToLocalNames)
|
|
|
|
|
|
|
|
// Case 2: notebook foo.py is converted to python script by removing
|
|
|
|
// magic keyword
|
|
|
|
foo.overwrite(t, "print(\"abc\")")
|
|
|
|
files, err = fileSet.All()
|
|
|
|
assert.NoError(t, err)
|
|
|
|
change, err = state.diff(files)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Len(t, change.delete, 1)
|
|
|
|
assert.Len(t, change.put, 1)
|
|
|
|
assert.Contains(t, change.put, "foo.py")
|
|
|
|
assert.Contains(t, change.delete, "foo")
|
|
|
|
assertKeysOfMap(t, state.LastUpdatedTimes, []string{"foo.py"})
|
|
|
|
assert.Equal(t, map[string]string{"foo.py": "foo.py"}, state.LocalToRemoteNames)
|
|
|
|
assert.Equal(t, map[string]string{"foo.py": "foo.py"}, state.RemoteToLocalNames)
|
|
|
|
|
|
|
|
// Case 3: Python script foo.py is converted to a databricks notebook
|
|
|
|
foo.overwrite(t, "# Databricks notebook source\nprint(\"def\")")
|
|
|
|
files, err = fileSet.All()
|
|
|
|
assert.NoError(t, err)
|
|
|
|
change, err = state.diff(files)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Len(t, change.delete, 1)
|
|
|
|
assert.Len(t, change.put, 1)
|
|
|
|
assert.Contains(t, change.put, "foo.py")
|
|
|
|
assert.Contains(t, change.delete, "foo.py")
|
|
|
|
assertKeysOfMap(t, state.LastUpdatedTimes, []string{"foo.py"})
|
|
|
|
assert.Equal(t, map[string]string{"foo.py": "foo"}, state.LocalToRemoteNames)
|
|
|
|
assert.Equal(t, map[string]string{"foo": "foo.py"}, state.RemoteToLocalNames)
|
|
|
|
|
|
|
|
// Case 4: Python notebook foo.py is deleted, and its remote name is used in change.delete
|
|
|
|
foo.remove(t)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
files, err = fileSet.All()
|
|
|
|
assert.NoError(t, err)
|
|
|
|
change, err = state.diff(files)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Len(t, change.delete, 1)
|
|
|
|
assert.Len(t, change.put, 0)
|
|
|
|
assert.Contains(t, change.delete, "foo")
|
|
|
|
assert.Len(t, state.LastUpdatedTimes, 0)
|
|
|
|
assert.Equal(t, map[string]string{}, state.LocalToRemoteNames)
|
|
|
|
assert.Equal(t, map[string]string{}, state.RemoteToLocalNames)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestErrorWhenIdenticalRemoteName(t *testing.T) {
|
|
|
|
// Create temp project dir
|
|
|
|
projectDir := t.TempDir()
|
|
|
|
fileSet := git.NewFileSet(projectDir)
|
|
|
|
state := Snapshot{
|
|
|
|
LastUpdatedTimes: make(map[string]time.Time),
|
|
|
|
LocalToRemoteNames: make(map[string]string),
|
|
|
|
RemoteToLocalNames: make(map[string]string),
|
|
|
|
}
|
|
|
|
|
|
|
|
// upload should work since they point to different destinations
|
|
|
|
pythonFoo := createFile(t, filepath.Join(projectDir, "foo.py"))
|
|
|
|
defer pythonFoo.close(t)
|
|
|
|
vanillaFoo := createFile(t, filepath.Join(projectDir, "foo"))
|
|
|
|
defer vanillaFoo.close(t)
|
|
|
|
files, err := fileSet.All()
|
|
|
|
assert.NoError(t, err)
|
|
|
|
change, err := state.diff(files)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Len(t, change.delete, 0)
|
|
|
|
assert.Len(t, change.put, 2)
|
|
|
|
assert.Contains(t, change.put, "foo.py")
|
|
|
|
assert.Contains(t, change.put, "foo")
|
|
|
|
|
|
|
|
// errors out because they point to the same destination
|
|
|
|
pythonFoo.overwrite(t, "# Databricks notebook source\nprint(\"def\")")
|
|
|
|
files, err = fileSet.All()
|
|
|
|
assert.NoError(t, err)
|
|
|
|
change, err = state.diff(files)
|
|
|
|
assert.ErrorContains(t, err, "both foo and foo.py point to the same remote file location foo. Please remove one of them from your local project")
|
2022-09-14 15:50:29 +00:00
|
|
|
}
|