util: Add some useful path parsing functions

These two are useful for looking at path prefixes and rebasing the paths
onto other paths.
This commit is contained in:
James Shubin
2018-11-29 15:58:52 -05:00
parent aae714db6b
commit dc13d5d26b
2 changed files with 159 additions and 0 deletions

View File

@@ -349,6 +349,34 @@ func FlattenListWithSplit(input []string, split []string) []string {
return out
}
// RemoveBasePath removes an absolute base path (directory prefix) from an
// absolute path that is any file or directory.
// Eg: RemoveBasePath("/usr/bin/foo", "/usr/") -> "bin/foo"
// Eg: RemoveBasePath("/usr/bin/project/", "/usr/") -> "bin/project/".
func RemoveBasePath(path, base string) (string, error) {
if !strings.HasSuffix(base, "/") { // should end with a slash
return "", fmt.Errorf("base is not a directory")
}
if !strings.HasPrefix(path, base) {
return "", fmt.Errorf("path does not have base prefix")
}
return strings.TrimPrefix(path, base), nil
}
// Rebase takes an absolute base path (directory prefix) and removes it from an
// absolute path and then returns that path with a new root as an absolute path.
// Eg: Rebase("/usr/bin/foo", "/usr/", "/usr/local/") -> "/usr/local/bin/foo"
func Rebase(path, base, root string) (string, error) {
if !strings.HasSuffix(root, "/") { // should end with a slash
return "", fmt.Errorf("root is not a directory")
}
s, err := RemoveBasePath(path, base)
if err != nil {
return "", err
}
return root + s, nil
}
// TimeAfterOrBlock is aspecial version of time.After that blocks when given a
// negative integer. When used in a case statement, the timer restarts on each
// select call to it.

View File

@@ -816,6 +816,137 @@ func TestUtilFlattenListWithSplit1(t *testing.T) {
}
}
func TestRemoveBasePath0(t *testing.T) {
// expected successes...
if s, err := RemoveBasePath("/usr/bin/foo", "/usr/"); err != nil {
t.Errorf("unexpected error: %v", err)
} else if s != "bin/foo" {
t.Errorf("unexpected string, got: %s", s)
}
if s, err := RemoveBasePath("/usr/bin/project/", "/usr/"); err != nil {
t.Errorf("unexpected error: %v", err)
} else if s != "bin/project/" {
t.Errorf("unexpected string, got: %s", s)
}
if s, err := RemoveBasePath("/", "/"); err != nil {
t.Errorf("unexpected error: %v", err)
} else if s != "" { // TODO: is this correct?
t.Errorf("unexpected string, got: %s", s)
}
if s, err := RemoveBasePath("/usr/bin/project/", "/"); err != nil {
t.Errorf("unexpected error: %v", err)
} else if s != "usr/bin/project/" {
t.Errorf("unexpected string, got: %s", s)
}
if s, err := RemoveBasePath("/usr/bin/project/", "/usr/bin/"); err != nil {
t.Errorf("unexpected error: %v", err)
} else if s != "project/" {
t.Errorf("unexpected string, got: %s", s)
}
if s, err := RemoveBasePath("/usr/bin/foo", "/usr/bin/"); err != nil {
t.Errorf("unexpected error: %v", err)
} else if s != "foo" {
t.Errorf("unexpected string, got: %s", s)
}
// allow this one, even though it's relative paths
if s, err := RemoveBasePath("usr/bin/project/", "usr/"); err != nil {
t.Errorf("unexpected error: %v", err)
} else if s != "bin/project/" {
t.Errorf("unexpected string, got: %s", s)
}
// expected errors...
if s, err := RemoveBasePath("", ""); err == nil {
t.Errorf("expected error, got: %s", s)
}
if s, err := RemoveBasePath("", "/usr/"); err == nil {
t.Errorf("expected error, got: %s", s)
}
if s, err := RemoveBasePath("usr/bin/project/", ""); err == nil {
t.Errorf("expected error, got: %s", s)
}
if s, err := RemoveBasePath("usr/bin/project/", "/usr/"); err == nil {
t.Errorf("expected error, got: %s", s)
}
if s, err := RemoveBasePath("/usr/bin/project/", "usr/"); err == nil {
t.Errorf("expected error, got: %s", s)
}
// allow this one, even though it's relative paths
//if s, err := RemoveBasePath("usr/bin/project/", "usr/"); err == nil {
// t.Errorf("expected error, got: %s", s)
//}
if s, err := RemoveBasePath("/usr/bin/project/", "/bin/"); err == nil {
t.Errorf("expected error, got: %s", s)
}
}
func TestRebasePath0(t *testing.T) {
// expected successes...
if s, err := Rebase("/usr/bin/foo", "/usr/", "/usr/local/"); err != nil {
t.Errorf("unexpected error: %v", err)
} else if s != "/usr/local/bin/foo" {
t.Errorf("unexpected string, got: %s", s)
}
if s, err := Rebase("/usr/bin/project/", "/usr/", "/usr/local/"); err != nil {
t.Errorf("unexpected error: %v", err)
} else if s != "/usr/local/bin/project/" {
t.Errorf("unexpected string, got: %s", s)
}
if s, err := Rebase("/", "/", "/opt/"); err != nil {
t.Errorf("unexpected error: %v", err)
} else if s != "/opt/" { // TODO: is this correct?
t.Errorf("unexpected string, got: %s", s)
}
if s, err := Rebase("/usr/bin/project/", "/", "/opt/"); err != nil {
t.Errorf("unexpected error: %v", err)
} else if s != "/opt/usr/bin/project/" {
t.Errorf("unexpected string, got: %s", s)
}
if s, err := Rebase("/usr/bin/project/", "/usr/bin/", "/opt/"); err != nil {
t.Errorf("unexpected error: %v", err)
} else if s != "/opt/project/" {
t.Errorf("unexpected string, got: %s", s)
}
if s, err := Rebase("/usr/bin/foo", "/usr/bin/", "/opt/"); err != nil {
t.Errorf("unexpected error: %v", err)
} else if s != "/opt/foo" {
t.Errorf("unexpected string, got: %s", s)
}
// allow this one, even though it's relative paths
if s, err := Rebase("usr/bin/project/", "usr/", "/opt/"); err != nil {
t.Errorf("unexpected error: %v", err)
} else if s != "/opt/bin/project/" {
t.Errorf("unexpected string, got: %s", s)
}
// expected errors...
if s, err := Rebase("", "", "/opt/"); err == nil {
t.Errorf("expected error, got: %s", s)
}
if s, err := Rebase("", "/usr/", "/opt/"); err == nil {
t.Errorf("expected error, got: %s", s)
}
if s, err := Rebase("usr/bin/project/", "", "/opt/"); err == nil {
t.Errorf("expected error, got: %s", s)
}
if s, err := Rebase("usr/bin/project/", "/usr/", "/opt/"); err == nil {
t.Errorf("expected error, got: %s", s)
}
if s, err := Rebase("/usr/bin/project/", "usr/", "/opt/"); err == nil {
t.Errorf("expected error, got: %s", s)
}
// allow this one, even though it's relative paths
//if s, err := Rebase("usr/bin/project/", "usr/", "/opt/"); err == nil {
// t.Errorf("expected error, got: %s", s)
//}
if s, err := Rebase("/usr/bin/project/", "/bin/", "/opt/"); err == nil {
t.Errorf("expected error, got: %s", s)
}
if s, err := Rebase("/usr/bin/project", "/usr/", ""); err == nil {
t.Errorf("expected error, got: %s", s)
}
}
func TestSortedStrSliceCompare0(t *testing.T) {
slice0 := []string{"foo", "bar", "baz"}
slice1 := []string{"bar", "foo", "baz"}