2024-12-13 14:38:58 +00:00
package fs_test
2023-06-16 15:09:08 +00:00
import (
"context"
"io"
"path"
"path/filepath"
2024-02-20 16:14:37 +00:00
"regexp"
2024-01-11 18:49:42 +00:00
"runtime"
2023-06-16 15:09:08 +00:00
"strings"
"testing"
2024-12-12 16:48:51 +00:00
"github.com/databricks/cli/internal/testcli"
2024-12-12 12:35:38 +00:00
"github.com/databricks/cli/internal/testutil"
2023-06-16 15:09:08 +00:00
"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 {
2024-02-20 16:14:37 +00:00
name string
2024-12-12 14:42:15 +00:00
setupSource func ( testutil . TestingT ) ( filer . Filer , string )
setupTarget func ( testutil . TestingT ) ( filer . Filer , string )
2023-06-16 15:09:08 +00:00
}
2024-02-20 16:14:37 +00:00
func copyTests ( ) [ ] cpTest {
2023-06-16 15:09:08 +00:00
return [ ] cpTest {
2024-02-20 16:14:37 +00:00
// 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 ,
} ,
2023-06-16 15:09:08 +00:00
}
}
2024-12-13 14:47:50 +00:00
func TestFsCpDir ( t * testing . T ) {
2024-02-20 16:14:37 +00:00
t . Parallel ( )
for _ , testCase := range copyTests ( ) {
tc := testCase
2023-06-16 15:09:08 +00:00
2024-02-20 16:14:37 +00:00
t . Run ( tc . name , func ( t * testing . T ) {
t . Parallel ( )
2023-06-16 15:09:08 +00:00
2024-02-20 16:14:37 +00:00
sourceFiler , sourceDir := tc . setupSource ( t )
targetFiler , targetDir := tc . setupTarget ( t )
setupSourceDir ( t , context . Background ( ) , sourceFiler )
2023-06-16 15:09:08 +00:00
2024-12-12 16:48:51 +00:00
testcli . RequireSuccessfulRun ( t , "fs" , "cp" , sourceDir , targetDir , "--recursive" )
2024-02-20 16:14:37 +00:00
assertTargetDir ( t , context . Background ( ) , targetFiler )
} )
2023-06-16 15:09:08 +00:00
}
}
2024-12-13 14:47:50 +00:00
func TestFsCpFileToFile ( t * testing . T ) {
2024-02-20 16:14:37 +00:00
t . Parallel ( )
for _ , testCase := range copyTests ( ) {
tc := testCase
2023-06-16 15:09:08 +00:00
2024-02-20 16:14:37 +00:00
t . Run ( tc . name , func ( t * testing . T ) {
t . Parallel ( )
2023-06-16 15:09:08 +00:00
2024-02-20 16:14:37 +00:00
sourceFiler , sourceDir := tc . setupSource ( t )
targetFiler , targetDir := tc . setupTarget ( t )
setupSourceFile ( t , context . Background ( ) , sourceFiler )
2023-06-16 15:09:08 +00:00
2024-12-12 16:48:51 +00:00
testcli . RequireSuccessfulRun ( t , "fs" , "cp" , path . Join ( sourceDir , "foo.txt" ) , path . Join ( targetDir , "bar.txt" ) )
2024-02-20 16:14:37 +00:00
assertTargetFile ( t , context . Background ( ) , targetFiler , "bar.txt" )
} )
2023-06-16 15:09:08 +00:00
}
}
2024-12-13 14:47:50 +00:00
func TestFsCpFileToDir ( t * testing . T ) {
2024-02-20 16:14:37 +00:00
t . Parallel ( )
for _ , testCase := range copyTests ( ) {
tc := testCase
2023-06-16 15:09:08 +00:00
2024-02-20 16:14:37 +00:00
t . Run ( tc . name , func ( t * testing . T ) {
t . Parallel ( )
2023-06-16 15:09:08 +00:00
2024-02-20 16:14:37 +00:00
sourceFiler , sourceDir := tc . setupSource ( t )
targetFiler , targetDir := tc . setupTarget ( t )
setupSourceFile ( t , context . Background ( ) , sourceFiler )
2024-12-12 16:48:51 +00:00
testcli . RequireSuccessfulRun ( t , "fs" , "cp" , path . Join ( sourceDir , "foo.txt" ) , targetDir )
2024-02-20 16:14:37 +00:00
assertTargetFile ( t , context . Background ( ) , targetFiler , "foo.txt" )
} )
2023-06-16 15:09:08 +00:00
}
}
2024-12-13 14:47:50 +00:00
func TestFsCpFileToDirForWindowsPaths ( t * testing . T ) {
2024-01-11 18:49:42 +00:00
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" )
2024-12-12 16:48:51 +00:00
testcli . RequireSuccessfulRun ( t , "fs" , "cp" , windowsPath , targetDir )
2024-01-11 18:49:42 +00:00
assertTargetFile ( t , ctx , targetFiler , "foo.txt" )
}
2024-12-13 14:47:50 +00:00
func TestFsCpDirToDirFileNotOverwritten ( t * testing . T ) {
2024-02-20 16:14:37 +00:00
t . Parallel ( )
2023-06-16 15:09:08 +00:00
2024-02-20 16:14:37 +00:00
for _ , testCase := range copyTests ( ) {
tc := testCase
2023-06-16 15:09:08 +00:00
2024-02-20 16:14:37 +00:00
t . Run ( tc . name , func ( t * testing . T ) {
t . Parallel ( )
2023-06-16 15:09:08 +00:00
2024-02-20 16:14:37 +00:00
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 )
2024-12-12 16:48:51 +00:00
testcli . RequireSuccessfulRun ( t , "fs" , "cp" , sourceDir , targetDir , "--recursive" )
2024-02-20 16:14:37 +00:00
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)" )
} )
2023-06-16 15:09:08 +00:00
}
}
2024-12-13 14:47:50 +00:00
func TestFsCpFileToDirFileNotOverwritten ( t * testing . T ) {
2024-02-20 16:14:37 +00:00
t . Parallel ( )
for _ , testCase := range copyTests ( ) {
tc := testCase
2023-06-16 15:09:08 +00:00
2024-02-20 16:14:37 +00:00
t . Run ( tc . name , func ( t * testing . T ) {
t . Parallel ( )
2023-06-16 15:09:08 +00:00
2024-02-20 16:14:37 +00:00
sourceFiler , sourceDir := tc . setupSource ( t )
targetFiler , targetDir := tc . setupTarget ( t )
setupSourceDir ( t , context . Background ( ) , sourceFiler )
2023-06-16 15:09:08 +00:00
2024-02-20 16:14:37 +00:00
// 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 )
2024-12-12 16:48:51 +00:00
testcli . RequireSuccessfulRun ( t , "fs" , "cp" , path . Join ( sourceDir , "a/b/c/hello.txt" ) , path . Join ( targetDir , "a/b/c" ) )
2024-02-20 16:14:37 +00:00
assertFileContent ( t , context . Background ( ) , targetFiler , "a/b/c/hello.txt" , "this should not be overwritten" )
} )
2023-06-16 15:09:08 +00:00
}
}
2024-12-13 14:47:50 +00:00
func TestFsCpFileToFileFileNotOverwritten ( t * testing . T ) {
2024-02-20 16:14:37 +00:00
t . Parallel ( )
for _ , testCase := range copyTests ( ) {
tc := testCase
2023-06-16 15:09:08 +00:00
2024-02-20 16:14:37 +00:00
t . Run ( tc . name , func ( t * testing . T ) {
t . Parallel ( )
2023-06-16 15:09:08 +00:00
2024-02-20 16:14:37 +00:00
sourceFiler , sourceDir := tc . setupSource ( t )
targetFiler , targetDir := tc . setupTarget ( t )
setupSourceDir ( t , context . Background ( ) , sourceFiler )
2023-06-16 15:09:08 +00:00
2024-02-20 16:14:37 +00:00
// 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 )
2024-12-12 16:48:51 +00:00
testcli . RequireSuccessfulRun ( t , "fs" , "cp" , path . Join ( sourceDir , "a/b/c/hello.txt" ) , path . Join ( targetDir , "a/b/c/dontoverwrite.txt" ) )
2024-02-20 16:14:37 +00:00
assertFileContent ( t , context . Background ( ) , targetFiler , "a/b/c/dontoverwrite.txt" , "this should not be overwritten" )
} )
2023-06-16 15:09:08 +00:00
}
}
2024-12-13 14:47:50 +00:00
func TestFsCpDirToDirWithOverwriteFlag ( t * testing . T ) {
2024-02-20 16:14:37 +00:00
t . Parallel ( )
for _ , testCase := range copyTests ( ) {
tc := testCase
2023-06-16 15:09:08 +00:00
2024-02-20 16:14:37 +00:00
t . Run ( tc . name , func ( t * testing . T ) {
t . Parallel ( )
2023-06-16 15:09:08 +00:00
2024-02-20 16:14:37 +00:00
sourceFiler , sourceDir := tc . setupSource ( t )
targetFiler , targetDir := tc . setupTarget ( t )
setupSourceDir ( t , context . Background ( ) , sourceFiler )
2023-06-16 15:09:08 +00:00
2024-02-20 16:14:37 +00:00
// 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 )
2024-12-12 16:48:51 +00:00
testcli . RequireSuccessfulRun ( t , "fs" , "cp" , sourceDir , targetDir , "--recursive" , "--overwrite" )
2024-02-20 16:14:37 +00:00
assertTargetDir ( t , context . Background ( ) , targetFiler )
} )
2023-06-16 15:09:08 +00:00
}
}
2024-12-13 14:47:50 +00:00
func TestFsCpFileToFileWithOverwriteFlag ( t * testing . T ) {
2024-02-20 16:14:37 +00:00
t . Parallel ( )
for _ , testCase := range copyTests ( ) {
tc := testCase
2023-06-16 15:09:08 +00:00
2024-02-20 16:14:37 +00:00
t . Run ( tc . name , func ( t * testing . T ) {
t . Parallel ( )
2023-06-16 15:09:08 +00:00
2024-02-20 16:14:37 +00:00
sourceFiler , sourceDir := tc . setupSource ( t )
targetFiler , targetDir := tc . setupTarget ( t )
setupSourceDir ( t , context . Background ( ) , sourceFiler )
2023-06-16 15:09:08 +00:00
2024-02-20 16:14:37 +00:00
// 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 )
2024-12-12 16:48:51 +00:00
testcli . RequireSuccessfulRun ( t , "fs" , "cp" , path . Join ( sourceDir , "a/b/c/hello.txt" ) , path . Join ( targetDir , "a/b/c/overwritten.txt" ) , "--overwrite" )
2024-02-20 16:14:37 +00:00
assertFileContent ( t , context . Background ( ) , targetFiler , "a/b/c/overwritten.txt" , "hello, world\n" )
} )
2023-06-16 15:09:08 +00:00
}
}
2024-12-13 14:47:50 +00:00
func TestFsCpFileToDirWithOverwriteFlag ( t * testing . T ) {
2024-02-20 16:14:37 +00:00
t . Parallel ( )
for _ , testCase := range copyTests ( ) {
tc := testCase
2023-06-16 15:09:08 +00:00
2024-02-20 16:14:37 +00:00
t . Run ( tc . name , func ( t * testing . T ) {
t . Parallel ( )
2023-06-16 15:09:08 +00:00
2024-02-20 16:14:37 +00:00
sourceFiler , sourceDir := tc . setupSource ( t )
targetFiler , targetDir := tc . setupTarget ( t )
setupSourceDir ( t , context . Background ( ) , sourceFiler )
2023-06-16 15:09:08 +00:00
2024-02-20 16:14:37 +00:00
// 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 )
2024-12-12 16:48:51 +00:00
testcli . RequireSuccessfulRun ( t , "fs" , "cp" , path . Join ( sourceDir , "a/b/c/hello.txt" ) , path . Join ( targetDir , "a/b/c" ) , "--overwrite" )
2024-02-20 16:14:37 +00:00
assertFileContent ( t , context . Background ( ) , targetFiler , "a/b/c/hello.txt" , "hello, world\n" )
} )
2023-06-16 15:09:08 +00:00
}
}
2024-12-13 14:47:50 +00:00
func TestFsCpErrorsWhenSourceIsDirWithoutRecursiveFlag ( t * testing . T ) {
2024-02-20 16:14:37 +00:00
t . Parallel ( )
2023-06-16 15:09:08 +00:00
2024-02-20 16:14:37 +00:00
for _ , testCase := range fsTests {
tc := testCase
2023-06-16 15:09:08 +00:00
2024-02-20 16:14:37 +00:00
t . Run ( tc . name , func ( t * testing . T ) {
t . Parallel ( )
2023-06-16 15:09:08 +00:00
2024-02-20 16:14:37 +00:00
_ , tmpDir := tc . setupFiler ( t )
2024-12-12 16:48:51 +00:00
_ , _ , err := testcli . RequireErrorRun ( t , "fs" , "cp" , path . Join ( tmpDir ) , path . Join ( tmpDir , "foobar" ) )
2024-02-20 16:14:37 +00:00
r := regexp . MustCompile ( "source path .* is a directory. Please specify the --recursive flag" )
assert . Regexp ( t , r , err . Error ( ) )
} )
}
2023-06-16 15:09:08 +00:00
}
2024-12-13 14:47:50 +00:00
func TestFsCpErrorsOnInvalidScheme ( t * testing . T ) {
2024-12-12 12:35:38 +00:00
t . Log ( testutil . GetEnvOrSkipTest ( t , "CLOUD_ENV" ) )
2023-06-16 15:09:08 +00:00
2024-12-12 16:48:51 +00:00
_ , _ , err := testcli . RequireErrorRun ( t , "fs" , "cp" , "dbfs:/a" , "https:/b" )
2023-06-23 14:07:09 +00:00
assert . Equal ( t , "invalid scheme: https" , err . Error ( ) )
2023-06-16 15:09:08 +00:00
}
2024-12-13 14:47:50 +00:00
func TestFsCpSourceIsDirectoryButTargetIsFile ( t * testing . T ) {
2024-02-20 16:14:37 +00:00
t . Parallel ( )
2023-06-16 15:09:08 +00:00
2024-02-20 16:14:37 +00:00
for _ , testCase := range copyTests ( ) {
tc := testCase
2023-06-16 15:09:08 +00:00
2024-02-20 16:14:37 +00:00
t . Run ( tc . name , func ( t * testing . T ) {
t . Parallel ( )
2023-06-16 15:09:08 +00:00
2024-02-20 16:14:37 +00:00
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 )
2023-06-16 15:09:08 +00:00
2024-12-12 16:48:51 +00:00
_ , _ , err = testcli . RequireErrorRun ( t , "fs" , "cp" , sourceDir , path . Join ( targetDir , "my_target" ) , "--recursive" )
2024-02-20 16:14:37 +00:00
assert . Error ( t , err )
} )
}
2023-06-16 15:09:08 +00:00
}