Skip to content
This repository has been archived by the owner on Sep 9, 2020. It is now read-only.

Commit

Permalink
check cfg filename case on case insensitive systems
Browse files Browse the repository at this point in the history
- `fs`
  - Export `IsCaseSensitiveFilesystem`. Add and run test on windows, linux.
  - Add function `ReadActualFilenames` to read actual file names of given
    string slice. Add tests to be run on windows.
- `project`
  - Add function `checkCfgFilenames` to check the filenames for manifest
    and lock have the expected case. Use `fs#IsCaseSensitiveFilesystem`
    for an early return as the check is costly. Add test to be run on windows.
- `context`
  - Call `project#checkCfgFilenames` after resolving project root. Add test
    for invalid manifest file name to be run on windows.
  • Loading branch information
sudo-suhas committed Sep 2, 2017
1 parent 876083e commit e24eef4
Show file tree
Hide file tree
Showing 6 changed files with 307 additions and 10 deletions.
5 changes: 5 additions & 0 deletions context.go
Original file line number Diff line number Diff line change
Expand Up @@ -103,6 +103,11 @@ func (c *Ctx) LoadProject() (*Project, error) {
return nil, err
}

err = checkCfgFilenames(root)
if err != nil {
return nil, err
}

p := new(Project)

if err = p.SetRoot(root); err != nil {
Expand Down
45 changes: 45 additions & 0 deletions context_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@
package dep

import (
"fmt"
"io/ioutil"
"log"
"os"
Expand Down Expand Up @@ -263,6 +264,50 @@ func TestLoadProjectNoSrcDir(t *testing.T) {
}
}

func TestLoadProjectCfgFileCase(t *testing.T) {
if runtime.GOOS != "windows" {
t.Skip("skip this test on non-Windows")
}

// Here we test that a manifest filename with incorrect case
// throws an error. Similar error will also be thrown for the
// lock file as well which has been tested in
// `project_test.go#TestCheckCfgFilenames`. So not repeating here.

h := test.NewHelper(t)
defer h.Cleanup()

invalidMfName := strings.ToLower(ManifestName)

wd := filepath.Join("src", "test")
h.TempFile(filepath.Join(wd, invalidMfName), "")

ctx := &Ctx{
Out: discardLogger,
Err: discardLogger,
}

err := ctx.SetPaths(h.Path(wd), h.Path("."))
if err != nil {
t.Fatalf("%+v", err)
}

_, err = ctx.LoadProject()

if err == nil {
t.Fatal("should have returned 'Manifest Filename' error")
}

expectedErrMsg := fmt.Sprintf(
"manifest filename '%s' does not match '%s'",
invalidMfName, ManifestName,
)

if err.Error() != expectedErrMsg {
t.Fatalf("unexpected error: %+v", err)
}
}

// TestCaseInsentitive is test for Windows. This should work even though set
// difference letter cases in GOPATH.
func TestCaseInsentitiveGOPATH(t *testing.T) {
Expand Down
89 changes: 84 additions & 5 deletions internal/fs/fs.go
Original file line number Diff line number Diff line change
Expand Up @@ -35,9 +35,9 @@ func HasFilepathPrefix(path, prefix string) bool {
// handling of volume name/drive letter on Windows. vnPath and vnPrefix
// are first compared, and then used to initialize initial values of p and
// d which will be appended to for incremental checks using
// isCaseSensitiveFilesystem and then equality.
// IsCaseSensitiveFilesystem and then equality.

// no need to check isCaseSensitiveFilesystem because VolumeName return
// no need to check IsCaseSensitiveFilesystem because VolumeName return
// empty string on all non-Windows machines
vnPath := strings.ToLower(filepath.VolumeName(path))
vnPrefix := strings.ToLower(filepath.VolumeName(prefix))
Expand Down Expand Up @@ -84,7 +84,7 @@ func HasFilepathPrefix(path, prefix string) bool {
// something like ext4 filesystem mounted on FAT
// mountpoint, mounted on ext4 filesystem, i.e. the
// problematic filesystem is not the last one.
if isCaseSensitiveFilesystem(filepath.Join(d, dirs[i])) {
if IsCaseSensitiveFilesystem(filepath.Join(d, dirs[i])) {
d = filepath.Join(d, dirs[i])
p = filepath.Join(p, prefixes[i])
} else {
Expand Down Expand Up @@ -140,7 +140,7 @@ func renameByCopy(src, dst string) error {
return errors.Wrapf(os.RemoveAll(src), "cannot delete %s", src)
}

// isCaseSensitiveFilesystem determines if the filesystem where dir
// IsCaseSensitiveFilesystem determines if the filesystem where dir
// exists is case sensitive or not.
//
// CAVEAT: this function works by taking the last component of the given
Expand All @@ -159,7 +159,7 @@ func renameByCopy(src, dst string) error {
// If the input directory is such that the last component is composed
// exclusively of case-less codepoints (e.g. numbers), this function will
// return false.
func isCaseSensitiveFilesystem(dir string) bool {
func IsCaseSensitiveFilesystem(dir string) bool {
alt := filepath.Join(filepath.Dir(dir),
genTestFilename(filepath.Base(dir)))

Expand All @@ -176,6 +176,85 @@ func isCaseSensitiveFilesystem(dir string) bool {
return !os.SameFile(dInfo, aInfo)
}

var errPathNotDir = errors.New("given path is not a directory")

// ReadActualFilenames is used to determine the actual file names in given directory.
//
// On case sensitive file systems like ext4, it will check if those files exist using
// `os#Stat` and return a map with key and value as filenames which exist in the folder.
//
// Otherwise, it reads the contents of the directory
func ReadActualFilenames(dirPath string, names []string) (map[string]string, error) {
actualFilenames := make(map[string]string, len(names))
if len(names) <= 0 {
// This isn't expected to happen for current usage.
// Adding edge case handling, maybe useful in future
return actualFilenames, nil
}
// First, check that the given path is valid and it is a directory
dirStat, err := os.Stat(dirPath)
if err != nil {
return nil, err
}

if !dirStat.IsDir() {
return nil, errPathNotDir
}

// Ideally, we would use `os#Stat` for getting the actual file names
// but that returns the name we passed in as an argument and not the actual filename.
// So we are forced to list the directory contents and check
// against that. Since this check is costly, we do it only if absolutely necessary.
if IsCaseSensitiveFilesystem(dirPath) {
// There will be no difference between actual filename and given filename
// So just check if those files exist.
for _, name := range names {
_, err := os.Stat(filepath.Join(dirPath, name))
if err == nil {
actualFilenames[name] = name
} else if !os.IsNotExist(err) {
// Some unexpected err, return it.
return nil, err
}
}
return actualFilenames, nil
}

dir, err := os.Open(dirPath)
if err != nil {
return nil, err
}
defer dir.Close()

// Pass -1 to read all files in directory
files, err := dir.Readdir(-1)
if err != nil {
return nil, err
}

// namesMap holds the mapping from lowercase name to search name.
// Using this, we can avoid repeatedly looping through names.
namesMap := make(map[string]string, len(names))
for _, name := range names {
namesMap[strings.ToLower(name)] = name
}

for _, file := range files {
if file.Mode().IsRegular() {
searchName, ok := namesMap[strings.ToLower(file.Name())]
if ok {
// We are interested in this file, case insensitive match successful
actualFilenames[searchName] = file.Name()
if len(actualFilenames) == len(names) {
// We found all that we were looking for
return actualFilenames, nil
}
}
}
}
return actualFilenames, nil
}

// genTestFilename returns a string with at most one rune case-flipped.
//
// The transformation is applied only to the first rune that can be
Expand Down
91 changes: 86 additions & 5 deletions internal/fs/fs_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ import (
"io/ioutil"
"os"
"path/filepath"
"reflect"
"runtime"
"strings"
"testing"
Expand Down Expand Up @@ -172,6 +173,85 @@ func TestRenameWithFallback(t *testing.T) {
}
}

func TestIsCaseSensitiveFilesystem(t *testing.T) {
if runtime.GOOS != "windows" && runtime.GOOS != "linux" {
t.Skip("skip this test on non-Windows, non-Linux")
}

dir, err := ioutil.TempDir("", "TestCaseSensitivity")
if err != nil {
t.Fatal(err)
}
defer os.RemoveAll(dir)

isSensitive := IsCaseSensitiveFilesystem(dir)
isWindows := runtime.GOOS == "windows"

if !isSensitive && !isWindows {
t.Fatal("expected isSensitive to be true on linux")
} else if isSensitive && isWindows {
t.Fatal("expected isSensitive to be false on windows")
}
}

func TestReadActualFilenames(t *testing.T) {
if runtime.GOOS != "windows" {
t.Skip("skip this test on non-Windows")
}

h := test.NewHelper(t)
defer h.Cleanup()

h.TempDir("")
tmpPath := h.Path(".")

_, err := ReadActualFilenames(filepath.Join(tmpPath, "does_not_exists"), []string{""})
switch {
case err == nil:
t.Fatal("expected err for non-existing folder")
case !os.IsNotExist(err):
t.Fatalf("unexpected error: %+v", err)
}
h.TempFile("tmpFile", "")
_, err = ReadActualFilenames(h.Path("tmpFile"), []string{""})
switch {
case err == nil:
t.Fatal("expected err for passing file instead of directory")
case err != errPathNotDir:
t.Fatalf("unexpected error: %+v", err)
}

cases := []struct {
createFiles []string
names []string
want map[string]string
}{
{nil, nil, map[string]string{}}, {
[]string{"test1.txt"},
[]string{"Test1.txt"},
map[string]string{"Test1.txt": "test1.txt"},
}, {
[]string{"test2.txt", "test3.TXT"},
[]string{"test2.txt", "Test3.txt", "Test4.txt"},
map[string]string{
"test2.txt": "test2.txt",
"Test3.txt": "test3.TXT",
"Test4.txt": "",
},
},
}
for _, c := range cases {
for _, file := range c.createFiles {
h.TempFile(file, "")
}
got, err := ReadActualFilenames(tmpPath, c.names)
if err != nil {
t.Fatalf("unexpected error: %+v", err)
}
reflect.DeepEqual(c.want, got)
}
}

func TestGenTestFilename(t *testing.T) {
cases := []struct {
str string
Expand All @@ -197,11 +277,11 @@ func TestGenTestFilename(t *testing.T) {

func BenchmarkGenTestFilename(b *testing.B) {
cases := []string{
"aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa",
"AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA",
"αααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααααα",
"11111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111",
"⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘⌘",
strings.Repeat("a", 128),
strings.Repeat("A", 128),
strings.Repeat("α", 128),
strings.Repeat("1", 128),
strings.Repeat("⌘", 128),
}

for i := 0; i < b.N; i++ {
Expand Down Expand Up @@ -556,6 +636,7 @@ func TestCopyFileLongFilePath(t *testing.T) {

h := test.NewHelper(t)
h.TempDir(".")
defer h.Cleanup()

tmpPath := h.Path(".")

Expand Down
31 changes: 31 additions & 0 deletions project.go
Original file line number Diff line number Diff line change
Expand Up @@ -41,6 +41,37 @@ func findProjectRoot(from string) (string, error) {
}
}

// checkCfgFilenames validates filename case for the manifest and lock files.
// This is relevant on case-insensitive systems like Windows.
func checkCfgFilenames(projectRoot string) error {
// ReadActualFilenames is actually costly. Since this check is not relevant
// to case-sensitive filesystems like ext4, try for an early return.
if fs.IsCaseSensitiveFilesystem(projectRoot) {
return nil
}

actualFilenames, err := fs.ReadActualFilenames(projectRoot, []string{ManifestName, LockName})

if err != nil {
return err
}

// Since this check is done after `findProjectRoot`, we can assume that
// manifest file will be present. Even if it is not, error will still be thrown.
// But error message will be a tad inaccurate.
actualMfName := actualFilenames[ManifestName]
if actualMfName != ManifestName {
return fmt.Errorf("manifest filename '%s' does not match '%s'", actualMfName, ManifestName)
}

actualLfName := actualFilenames[LockName]
if len(actualLfName) > 0 && actualLfName != LockName {
return fmt.Errorf("lock filename '%s' does not match '%s'", actualLfName, LockName)
}

return nil
}

// A Project holds a Manifest and optional Lock for a project.
type Project struct {
// AbsRoot is the absolute path to the root directory of the project.
Expand Down
Loading

0 comments on commit e24eef4

Please sign in to comment.