mirror of https://github.com/databricks/cli.git
386 lines
11 KiB
Go
386 lines
11 KiB
Go
package fs_test
|
|
|
|
import (
|
|
"context"
|
|
"io"
|
|
"path"
|
|
"path/filepath"
|
|
"regexp"
|
|
"runtime"
|
|
"strings"
|
|
"testing"
|
|
|
|
"github.com/databricks/cli/internal/testcli"
|
|
"github.com/databricks/cli/internal/testutil"
|
|
"github.com/databricks/cli/libs/filer"
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func setupSourceDir(t *testing.T, ctx context.Context, f filer.Filer) {
|
|
var err error
|
|
|
|
err = f.Write(ctx, "pyNb.py", strings.NewReader("# Databricks notebook source\nprint(123)"))
|
|
require.NoError(t, err)
|
|
|
|
err = f.Write(ctx, "query.sql", strings.NewReader("SELECT 1"))
|
|
require.NoError(t, err)
|
|
|
|
err = f.Write(ctx, "a/b/c/hello.txt", strings.NewReader("hello, world\n"), filer.CreateParentDirectories)
|
|
require.NoError(t, err)
|
|
}
|
|
|
|
func setupSourceFile(t *testing.T, ctx context.Context, f filer.Filer) {
|
|
err := f.Write(ctx, "foo.txt", strings.NewReader("abc"))
|
|
require.NoError(t, err)
|
|
}
|
|
|
|
func assertTargetFile(t *testing.T, ctx context.Context, f filer.Filer, relPath string) {
|
|
var err error
|
|
|
|
r, err := f.Read(ctx, relPath)
|
|
assert.NoError(t, err)
|
|
defer r.Close()
|
|
b, err := io.ReadAll(r)
|
|
require.NoError(t, err)
|
|
assert.Equal(t, "abc", string(b))
|
|
}
|
|
|
|
func assertFileContent(t *testing.T, ctx context.Context, f filer.Filer, path, expectedContent string) {
|
|
r, err := f.Read(ctx, path)
|
|
require.NoError(t, err)
|
|
defer r.Close()
|
|
b, err := io.ReadAll(r)
|
|
require.NoError(t, err)
|
|
assert.Equal(t, expectedContent, string(b))
|
|
}
|
|
|
|
func assertTargetDir(t *testing.T, ctx context.Context, f filer.Filer) {
|
|
assertFileContent(t, ctx, f, "pyNb.py", "# Databricks notebook source\nprint(123)")
|
|
assertFileContent(t, ctx, f, "query.sql", "SELECT 1")
|
|
assertFileContent(t, ctx, f, "a/b/c/hello.txt", "hello, world\n")
|
|
}
|
|
|
|
type cpTest struct {
|
|
name string
|
|
setupSource func(testutil.TestingT) (filer.Filer, string)
|
|
setupTarget func(testutil.TestingT) (filer.Filer, string)
|
|
}
|
|
|
|
func copyTests() []cpTest {
|
|
return []cpTest{
|
|
// source: local file system
|
|
{
|
|
name: "local to local",
|
|
setupSource: setupLocalFiler,
|
|
setupTarget: setupLocalFiler,
|
|
},
|
|
{
|
|
name: "local to dbfs",
|
|
setupSource: setupLocalFiler,
|
|
setupTarget: setupDbfsFiler,
|
|
},
|
|
{
|
|
name: "local to uc-volumes",
|
|
setupSource: setupLocalFiler,
|
|
setupTarget: setupUcVolumesFiler,
|
|
},
|
|
|
|
// source: dbfs
|
|
{
|
|
name: "dbfs to local",
|
|
setupSource: setupDbfsFiler,
|
|
setupTarget: setupLocalFiler,
|
|
},
|
|
{
|
|
name: "dbfs to dbfs",
|
|
setupSource: setupDbfsFiler,
|
|
setupTarget: setupDbfsFiler,
|
|
},
|
|
{
|
|
name: "dbfs to uc-volumes",
|
|
setupSource: setupDbfsFiler,
|
|
setupTarget: setupUcVolumesFiler,
|
|
},
|
|
|
|
// source: uc-volumes
|
|
{
|
|
name: "uc-volumes to local",
|
|
setupSource: setupUcVolumesFiler,
|
|
setupTarget: setupLocalFiler,
|
|
},
|
|
{
|
|
name: "uc-volumes to dbfs",
|
|
setupSource: setupUcVolumesFiler,
|
|
setupTarget: setupDbfsFiler,
|
|
},
|
|
{
|
|
name: "uc-volumes to uc-volumes",
|
|
setupSource: setupUcVolumesFiler,
|
|
setupTarget: setupUcVolumesFiler,
|
|
},
|
|
}
|
|
}
|
|
|
|
func TestFsCpDir(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
for _, testCase := range copyTests() {
|
|
tc := testCase
|
|
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
sourceFiler, sourceDir := tc.setupSource(t)
|
|
targetFiler, targetDir := tc.setupTarget(t)
|
|
setupSourceDir(t, context.Background(), sourceFiler)
|
|
|
|
testcli.RequireSuccessfulRun(t, "fs", "cp", sourceDir, targetDir, "--recursive")
|
|
|
|
assertTargetDir(t, context.Background(), targetFiler)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestFsCpFileToFile(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
for _, testCase := range copyTests() {
|
|
tc := testCase
|
|
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
sourceFiler, sourceDir := tc.setupSource(t)
|
|
targetFiler, targetDir := tc.setupTarget(t)
|
|
setupSourceFile(t, context.Background(), sourceFiler)
|
|
|
|
testcli.RequireSuccessfulRun(t, "fs", "cp", path.Join(sourceDir, "foo.txt"), path.Join(targetDir, "bar.txt"))
|
|
|
|
assertTargetFile(t, context.Background(), targetFiler, "bar.txt")
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestFsCpFileToDir(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
for _, testCase := range copyTests() {
|
|
tc := testCase
|
|
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
sourceFiler, sourceDir := tc.setupSource(t)
|
|
targetFiler, targetDir := tc.setupTarget(t)
|
|
setupSourceFile(t, context.Background(), sourceFiler)
|
|
|
|
testcli.RequireSuccessfulRun(t, "fs", "cp", path.Join(sourceDir, "foo.txt"), targetDir)
|
|
|
|
assertTargetFile(t, context.Background(), targetFiler, "foo.txt")
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestFsCpFileToDirForWindowsPaths(t *testing.T) {
|
|
if runtime.GOOS != "windows" {
|
|
t.Skip("Skipping test on non-windows OS")
|
|
}
|
|
|
|
ctx := context.Background()
|
|
sourceFiler, sourceDir := setupLocalFiler(t)
|
|
targetFiler, targetDir := setupDbfsFiler(t)
|
|
setupSourceFile(t, ctx, sourceFiler)
|
|
|
|
windowsPath := filepath.Join(filepath.FromSlash(sourceDir), "foo.txt")
|
|
|
|
testcli.RequireSuccessfulRun(t, "fs", "cp", windowsPath, targetDir)
|
|
assertTargetFile(t, ctx, targetFiler, "foo.txt")
|
|
}
|
|
|
|
func TestFsCpDirToDirFileNotOverwritten(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
for _, testCase := range copyTests() {
|
|
tc := testCase
|
|
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
sourceFiler, sourceDir := tc.setupSource(t)
|
|
targetFiler, targetDir := tc.setupTarget(t)
|
|
setupSourceDir(t, context.Background(), sourceFiler)
|
|
|
|
// Write a conflicting file to target
|
|
err := targetFiler.Write(context.Background(), "a/b/c/hello.txt", strings.NewReader("this should not be overwritten"), filer.CreateParentDirectories)
|
|
require.NoError(t, err)
|
|
|
|
testcli.RequireSuccessfulRun(t, "fs", "cp", sourceDir, targetDir, "--recursive")
|
|
assertFileContent(t, context.Background(), targetFiler, "a/b/c/hello.txt", "this should not be overwritten")
|
|
assertFileContent(t, context.Background(), targetFiler, "query.sql", "SELECT 1")
|
|
assertFileContent(t, context.Background(), targetFiler, "pyNb.py", "# Databricks notebook source\nprint(123)")
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestFsCpFileToDirFileNotOverwritten(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
for _, testCase := range copyTests() {
|
|
tc := testCase
|
|
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
sourceFiler, sourceDir := tc.setupSource(t)
|
|
targetFiler, targetDir := tc.setupTarget(t)
|
|
setupSourceDir(t, context.Background(), sourceFiler)
|
|
|
|
// Write a conflicting file to target
|
|
err := targetFiler.Write(context.Background(), "a/b/c/hello.txt", strings.NewReader("this should not be overwritten"), filer.CreateParentDirectories)
|
|
require.NoError(t, err)
|
|
|
|
testcli.RequireSuccessfulRun(t, "fs", "cp", path.Join(sourceDir, "a/b/c/hello.txt"), path.Join(targetDir, "a/b/c"))
|
|
assertFileContent(t, context.Background(), targetFiler, "a/b/c/hello.txt", "this should not be overwritten")
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestFsCpFileToFileFileNotOverwritten(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
for _, testCase := range copyTests() {
|
|
tc := testCase
|
|
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
sourceFiler, sourceDir := tc.setupSource(t)
|
|
targetFiler, targetDir := tc.setupTarget(t)
|
|
setupSourceDir(t, context.Background(), sourceFiler)
|
|
|
|
// Write a conflicting file to target
|
|
err := targetFiler.Write(context.Background(), "a/b/c/dontoverwrite.txt", strings.NewReader("this should not be overwritten"), filer.CreateParentDirectories)
|
|
require.NoError(t, err)
|
|
|
|
testcli.RequireSuccessfulRun(t, "fs", "cp", path.Join(sourceDir, "a/b/c/hello.txt"), path.Join(targetDir, "a/b/c/dontoverwrite.txt"))
|
|
assertFileContent(t, context.Background(), targetFiler, "a/b/c/dontoverwrite.txt", "this should not be overwritten")
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestFsCpDirToDirWithOverwriteFlag(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
for _, testCase := range copyTests() {
|
|
tc := testCase
|
|
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
sourceFiler, sourceDir := tc.setupSource(t)
|
|
targetFiler, targetDir := tc.setupTarget(t)
|
|
setupSourceDir(t, context.Background(), sourceFiler)
|
|
|
|
// Write a conflicting file to target
|
|
err := targetFiler.Write(context.Background(), "a/b/c/hello.txt", strings.NewReader("this should be overwritten"), filer.CreateParentDirectories)
|
|
require.NoError(t, err)
|
|
|
|
testcli.RequireSuccessfulRun(t, "fs", "cp", sourceDir, targetDir, "--recursive", "--overwrite")
|
|
assertTargetDir(t, context.Background(), targetFiler)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestFsCpFileToFileWithOverwriteFlag(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
for _, testCase := range copyTests() {
|
|
tc := testCase
|
|
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
sourceFiler, sourceDir := tc.setupSource(t)
|
|
targetFiler, targetDir := tc.setupTarget(t)
|
|
setupSourceDir(t, context.Background(), sourceFiler)
|
|
|
|
// Write a conflicting file to target
|
|
err := targetFiler.Write(context.Background(), "a/b/c/overwritten.txt", strings.NewReader("this should be overwritten"), filer.CreateParentDirectories)
|
|
require.NoError(t, err)
|
|
|
|
testcli.RequireSuccessfulRun(t, "fs", "cp", path.Join(sourceDir, "a/b/c/hello.txt"), path.Join(targetDir, "a/b/c/overwritten.txt"), "--overwrite")
|
|
assertFileContent(t, context.Background(), targetFiler, "a/b/c/overwritten.txt", "hello, world\n")
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestFsCpFileToDirWithOverwriteFlag(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
for _, testCase := range copyTests() {
|
|
tc := testCase
|
|
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
sourceFiler, sourceDir := tc.setupSource(t)
|
|
targetFiler, targetDir := tc.setupTarget(t)
|
|
setupSourceDir(t, context.Background(), sourceFiler)
|
|
|
|
// Write a conflicting file to target
|
|
err := targetFiler.Write(context.Background(), "a/b/c/hello.txt", strings.NewReader("this should be overwritten"), filer.CreateParentDirectories)
|
|
require.NoError(t, err)
|
|
|
|
testcli.RequireSuccessfulRun(t, "fs", "cp", path.Join(sourceDir, "a/b/c/hello.txt"), path.Join(targetDir, "a/b/c"), "--overwrite")
|
|
assertFileContent(t, context.Background(), targetFiler, "a/b/c/hello.txt", "hello, world\n")
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestFsCpErrorsWhenSourceIsDirWithoutRecursiveFlag(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
for _, testCase := range fsTests {
|
|
tc := testCase
|
|
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
_, tmpDir := tc.setupFiler(t)
|
|
|
|
_, _, err := testcli.RequireErrorRun(t, "fs", "cp", path.Join(tmpDir), path.Join(tmpDir, "foobar"))
|
|
r := regexp.MustCompile("source path .* is a directory. Please specify the --recursive flag")
|
|
assert.Regexp(t, r, err.Error())
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestFsCpErrorsOnInvalidScheme(t *testing.T) {
|
|
_, _, err := testcli.RequireErrorRun(t, "fs", "cp", "dbfs:/a", "https:/b")
|
|
assert.Equal(t, "invalid scheme: https", err.Error())
|
|
}
|
|
|
|
func TestFsCpSourceIsDirectoryButTargetIsFile(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
for _, testCase := range copyTests() {
|
|
tc := testCase
|
|
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
sourceFiler, sourceDir := tc.setupSource(t)
|
|
targetFiler, targetDir := tc.setupTarget(t)
|
|
setupSourceDir(t, context.Background(), sourceFiler)
|
|
|
|
// Write a conflicting file to target
|
|
err := targetFiler.Write(context.Background(), "my_target", strings.NewReader("I'll block any attempts to recursively copy"), filer.CreateParentDirectories)
|
|
require.NoError(t, err)
|
|
|
|
_, _, err = testcli.RequireErrorRun(t, "fs", "cp", sourceDir, path.Join(targetDir, "my_target"), "--recursive")
|
|
assert.Error(t, err)
|
|
})
|
|
}
|
|
}
|