Skip to content

Commit

Permalink
utils/copy: skip symlink files and improve tests assertivity
Browse files Browse the repository at this point in the history
Previously if some symlink exists on project path during the copy we was
creating a directory instead evaluating the sysmlink.

Evaluating the symlink makes us need to deal with several scenarios,
such as, how should we copy a file to the .horuse folder when it is not
in the project path? Another scenario would be how do we handle symlinks
from files that don't exist on the user's machine?

With that in mind, this commit changes the behavior of the `Copy` function
to ignore symlinks by default and only copy directories and files. Since
previously we were no longer analyzing sysmlinks, this change will not be
noticed by the user.

This commit also change the assertiveness of tests to check if all files
and directories was copied correctly.

Updates #718

Signed-off-by: Matheus Alcantara <matheus.alcantara@zup.com.br>
  • Loading branch information
matheusalcantarazup committed Nov 17, 2021
1 parent de514e9 commit 5c51e9e
Show file tree
Hide file tree
Showing 2 changed files with 50 additions and 52 deletions.
57 changes: 17 additions & 40 deletions internal/utils/copy/copy.go
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@
package copy

import (
"fmt"
"io"
"os"
"path/filepath"
Expand All @@ -23,43 +24,35 @@ import (
"github.com/ZupIT/horusec-devkit/pkg/utils/logger"
)

// Copy copy src directory to dst ignoring files that make skip function return true.
//
// Note that symlink files will be ignored by default.
//
// nolint:gocyclo
func Copy(src, dst string, skip func(src string) bool) error {
if err := os.MkdirAll(dst, os.ModePerm); err != nil {
return err
}
return filepath.Walk(src, func(path string, info os.FileInfo, err error) error {
if err != nil {
if err != nil || skip(path) || info.Mode()&os.ModeSymlink != 0 {
return err
}
if isToSkip := skip(path); !isToSkip {
return copyByType(src, dst, path, info)
logger.LogTraceWithLevel(fmt.Sprintf("Copying src: %s dst: %s path: %s", src, dst, path))
if info.IsDir() {
return copyDir(src, dst, path)
}
return nil
})
}

func copyByType(src, dst, path string, info os.FileInfo) error {
logger.LogTraceWithLevel("Copying ", "src: "+src, "dst: "+dst, "path: "+path)
switch {
case info.IsDir():
return copyDir(src, dst, path)
case info.Mode()&os.ModeSymlink != 0:
return copyLink(src, dst, path)
default:
return copyFile(src, dst, path)
}
})
}

func copyFile(src, dst, path string) error {
file, err := os.Create(replacePathSrcToDst(path, src, dst))
if file != nil {
defer func() {
logger.LogError("Error defer file close", file.Close())
}()
}
if err != nil {
return err
}
defer func() {
logger.LogError("Error defer file close", file.Close())
}()
return copyContentSrcFileToDstFile(path, file)
}

Expand All @@ -69,14 +62,12 @@ func replacePathSrcToDst(path, src, dst string) string {

func copyContentSrcFileToDstFile(srcPath string, dstFile *os.File) error {
srcFile, err := os.Open(srcPath)
if srcFile != nil {
defer func() {
logger.LogError("Error defer file close", srcFile.Close())
}()
}
if err != nil {
return err
}
defer func() {
logger.LogError("Error defer file close", srcFile.Close())
}()

_, err = io.Copy(dstFile, srcFile)
return err
Expand All @@ -86,17 +77,3 @@ func copyDir(src, dst, path string) error {
newPath := replacePathSrcToDst(path, src, dst)
return os.MkdirAll(newPath, os.ModePerm)
}

func copyLink(src, dst, path string) error {
orig, err := filepath.EvalSymlinks(src)
if err != nil {
return err
}

info, err := os.Lstat(orig)
if err != nil {
return err
}

return copyByType(orig, dst, path, info)
}
45 changes: 33 additions & 12 deletions internal/utils/copy/copy_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -12,31 +12,52 @@
// See the License for the specific language governing permissions and
// limitations under the License.

package copy
package copy_test

import (
"fmt"
"os"
"path/filepath"
"testing"

"github.com/ZupIT/horusec/internal/utils/copy"
"github.com/ZupIT/horusec/internal/utils/testutil"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
)

func TestCopy(t *testing.T) {
t.Run("Should success copy dir", func(t *testing.T) {
srcPath, err := filepath.Abs("../../../config")
assert.NoError(t, err)
src := testutil.GoExample1
dst := filepath.Join(t.TempDir(), t.Name())

dstPath, err := filepath.Abs(".")
assert.NoError(t, err)
tmpFile, err := os.CreateTemp(os.TempDir(), "test-symlink")
require.Nil(t, err, "Expected nil error to create temp file")

dstPath = fmt.Sprintf(dstPath+"%s", "/tmp-test")
symlinkFile := filepath.Join(src, "symlink")
err = os.Symlink(tmpFile.Name(), symlinkFile)
require.NoError(t, err, "Expected nil error to create symlink file: %v", err)

err = Copy(srcPath, dstPath, func(src string) bool { return false })
assert.NoError(t, err)
t.Cleanup(func() {
err := os.Remove(symlinkFile)
assert.NoError(t, err, "Expected nil error to clean up symlink file: %v", err)
})

err = os.RemoveAll(dstPath)
assert.NoError(t, err)
err = copy.Copy(src, dst, func(src string) bool {
ext := filepath.Ext(src)
return ext == ".mod" || ext == ".sum"
})

assert.NoError(t, err)

assert.DirExists(t, dst)
assert.DirExists(t, filepath.Join(dst, "api", "routes"))
assert.DirExists(t, filepath.Join(dst, "api", "util"))

assert.NoFileExists(t, filepath.Join(dst, "symlink"))
assert.FileExists(t, filepath.Join(dst, "api", "server.go"))
assert.FileExists(t, filepath.Join(dst, "api", "routes", "healthcheck.go"))
assert.FileExists(t, filepath.Join(dst, "api", "util", "util.go"))

assert.NoFileExists(t, filepath.Join(dst, "go.mod"))
assert.NoFileExists(t, filepath.Join(dst, "go.sum"))

}

0 comments on commit 5c51e9e

Please sign in to comment.