466 lines
12 KiB
Go
466 lines
12 KiB
Go
package filetree
|
|
|
|
import (
|
|
"fmt"
|
|
"testing"
|
|
)
|
|
|
|
func stringInSlice(a string, list []string) bool {
|
|
for _, b := range list {
|
|
if b == a {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
func AssertDiffType(node *FileNode, expectedDiffType DiffType) error {
|
|
if node.Data.DiffType != expectedDiffType {
|
|
return fmt.Errorf("Expecting node at %s to have DiffType %v, but had %v", node.Path(), expectedDiffType, node.Data.DiffType)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func TestPrintTree(t *testing.T) {
|
|
tree := NewFileTree()
|
|
tree.Root.AddChild("first node!", FileInfo{})
|
|
two := tree.Root.AddChild("second node!", FileInfo{})
|
|
tree.Root.AddChild("third node!", FileInfo{})
|
|
two.AddChild("forth, one level down...", FileInfo{})
|
|
|
|
expected :=
|
|
`├── first node!
|
|
├── second node!
|
|
│ └── forth, one level down...
|
|
└── third node!
|
|
`
|
|
actual := tree.String(false)
|
|
|
|
if expected != actual {
|
|
t.Errorf("Expected tree string:\n--->%s<---\nGot:\n--->%s<---", expected, actual)
|
|
}
|
|
|
|
}
|
|
|
|
func TestAddPath(t *testing.T) {
|
|
tree := NewFileTree()
|
|
tree.AddPath("/etc/nginx/nginx.conf", FileInfo{})
|
|
tree.AddPath("/etc/nginx/public", FileInfo{})
|
|
tree.AddPath("/var/run/systemd", FileInfo{})
|
|
tree.AddPath("/var/run/bashful", FileInfo{})
|
|
tree.AddPath("/tmp", FileInfo{})
|
|
tree.AddPath("/tmp/nonsense", FileInfo{})
|
|
|
|
expected :=
|
|
`├── etc
|
|
│ └── nginx
|
|
│ ├── nginx.conf
|
|
│ └── public
|
|
├── tmp
|
|
│ └── nonsense
|
|
└── var
|
|
└── run
|
|
├── bashful
|
|
└── systemd
|
|
`
|
|
actual := tree.String(false)
|
|
|
|
if expected != actual {
|
|
t.Errorf("Expected tree string:\n--->%s<---\nGot:\n--->%s<---", expected, actual)
|
|
}
|
|
|
|
}
|
|
|
|
func TestRemovePath(t *testing.T) {
|
|
tree := NewFileTree()
|
|
tree.AddPath("/etc/nginx/nginx.conf", FileInfo{})
|
|
tree.AddPath("/etc/nginx/public", FileInfo{})
|
|
tree.AddPath("/var/run/systemd", FileInfo{})
|
|
tree.AddPath("/var/run/bashful", FileInfo{})
|
|
tree.AddPath("/tmp", FileInfo{})
|
|
tree.AddPath("/tmp/nonsense", FileInfo{})
|
|
|
|
tree.RemovePath("/var/run/bashful")
|
|
tree.RemovePath("/tmp")
|
|
|
|
expected :=
|
|
`├── etc
|
|
│ └── nginx
|
|
│ ├── nginx.conf
|
|
│ └── public
|
|
└── var
|
|
└── run
|
|
└── systemd
|
|
`
|
|
actual := tree.String(false)
|
|
|
|
if expected != actual {
|
|
t.Errorf("Expected tree string:\n--->%s<---\nGot:\n--->%s<---", expected, actual)
|
|
}
|
|
|
|
}
|
|
|
|
func TestStack(t *testing.T) {
|
|
payloadKey := "/var/run/systemd"
|
|
payloadValue := FileInfo{
|
|
Path: "yup",
|
|
}
|
|
|
|
tree1 := NewFileTree()
|
|
|
|
tree1.AddPath("/etc/nginx/public", FileInfo{})
|
|
tree1.AddPath(payloadKey, FileInfo{})
|
|
tree1.AddPath("/var/run/bashful", FileInfo{})
|
|
tree1.AddPath("/tmp", FileInfo{})
|
|
tree1.AddPath("/tmp/nonsense", FileInfo{})
|
|
|
|
tree2 := NewFileTree()
|
|
// add new files
|
|
tree2.AddPath("/etc/nginx/nginx.conf", FileInfo{})
|
|
// modify current files
|
|
tree2.AddPath(payloadKey, payloadValue)
|
|
// whiteout the following files
|
|
tree2.AddPath("/var/run/.wh.bashful", FileInfo{})
|
|
tree2.AddPath("/.wh.tmp", FileInfo{})
|
|
|
|
err := tree1.Stack(tree2)
|
|
|
|
if err != nil {
|
|
t.Errorf("Could not stack refTrees: %v", err)
|
|
}
|
|
|
|
expected :=
|
|
`├── etc
|
|
│ └── nginx
|
|
│ ├── nginx.conf
|
|
│ └── public
|
|
└── var
|
|
└── run
|
|
└── systemd
|
|
`
|
|
|
|
node, err := tree1.GetNode(payloadKey)
|
|
if err != nil {
|
|
t.Errorf("Expected '%s' to still exist, but it doesn't", payloadKey)
|
|
}
|
|
|
|
if node.Data.FileInfo.Path != payloadValue.Path {
|
|
t.Errorf("Expected '%s' value to be %+v but got %+v", payloadKey, payloadValue, node.Data.FileInfo)
|
|
}
|
|
|
|
actual := tree1.String(false)
|
|
|
|
if expected != actual {
|
|
t.Errorf("Expected tree string:\n--->%s<---\nGot:\n--->%s<---", expected, actual)
|
|
}
|
|
|
|
}
|
|
|
|
func TestCopy(t *testing.T) {
|
|
tree := NewFileTree()
|
|
tree.AddPath("/etc/nginx/nginx.conf", FileInfo{})
|
|
tree.AddPath("/etc/nginx/public", FileInfo{})
|
|
tree.AddPath("/var/run/systemd", FileInfo{})
|
|
tree.AddPath("/var/run/bashful", FileInfo{})
|
|
tree.AddPath("/tmp", FileInfo{})
|
|
tree.AddPath("/tmp/nonsense", FileInfo{})
|
|
|
|
tree.RemovePath("/var/run/bashful")
|
|
tree.RemovePath("/tmp")
|
|
|
|
expected :=
|
|
`├── etc
|
|
│ └── nginx
|
|
│ ├── nginx.conf
|
|
│ └── public
|
|
└── var
|
|
└── run
|
|
└── systemd
|
|
`
|
|
|
|
NewFileTree := tree.Copy()
|
|
actual := NewFileTree.String(false)
|
|
|
|
if expected != actual {
|
|
t.Errorf("Expected tree string:\n--->%s<---\nGot:\n--->%s<---", expected, actual)
|
|
}
|
|
|
|
}
|
|
|
|
func TestCompareWithNoChanges(t *testing.T) {
|
|
lowerTree := NewFileTree()
|
|
upperTree := NewFileTree()
|
|
paths := [...]string{"/etc", "/etc/sudoers", "/etc/hosts", "/usr/bin", "/usr/bin/bash", "/usr"}
|
|
|
|
for _, value := range paths {
|
|
fakeData := FileInfo{
|
|
Path: value,
|
|
Typeflag: 1,
|
|
MD5sum: [16]byte{0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
|
|
}
|
|
lowerTree.AddPath(value, fakeData)
|
|
upperTree.AddPath(value, fakeData)
|
|
}
|
|
lowerTree.Compare(upperTree)
|
|
asserter := func(n *FileNode) error {
|
|
if n.Path() == "/" {
|
|
return nil
|
|
}
|
|
if (n.Data.DiffType) != Unchanged {
|
|
t.Errorf("Expecting node at %s to have DiffType unchanged, but had %v", n.Path(), n.Data.DiffType)
|
|
}
|
|
return nil
|
|
}
|
|
err := lowerTree.VisitDepthChildFirst(asserter, nil)
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
}
|
|
|
|
func TestCompareWithAdds(t *testing.T) {
|
|
lowerTree := NewFileTree()
|
|
upperTree := NewFileTree()
|
|
lowerPaths := [...]string{"/etc", "/etc/sudoers", "/usr", "/etc/hosts", "/usr/bin"}
|
|
upperPaths := [...]string{"/etc", "/etc/sudoers", "/usr", "/etc/hosts", "/usr/bin", "/usr/bin/bash"}
|
|
|
|
for _, value := range lowerPaths {
|
|
lowerTree.AddPath(value, FileInfo{
|
|
Path: value,
|
|
Typeflag: 1,
|
|
MD5sum: [16]byte{0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
|
|
})
|
|
}
|
|
|
|
for _, value := range upperPaths {
|
|
upperTree.AddPath(value, FileInfo{
|
|
Path: value,
|
|
Typeflag: 1,
|
|
MD5sum: [16]byte{0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
|
|
})
|
|
}
|
|
|
|
failedAssertions := []error{}
|
|
err := lowerTree.Compare(upperTree)
|
|
if err != nil {
|
|
t.Errorf("Expected tree compare to have no errors, got: %v", err)
|
|
}
|
|
asserter := func(n *FileNode) error {
|
|
|
|
p := n.Path()
|
|
if p == "/" {
|
|
return nil
|
|
} else if stringInSlice(p,[]string{"/usr/bin/bash"}) {
|
|
if err := AssertDiffType(n, Added); err != nil {
|
|
failedAssertions = append(failedAssertions, err)
|
|
}
|
|
} else if stringInSlice(p,[]string{"/usr/bin", "/usr"}) {
|
|
if err := AssertDiffType(n, Changed); err != nil {
|
|
failedAssertions = append(failedAssertions, err)
|
|
}
|
|
} else {
|
|
if err := AssertDiffType(n, Unchanged); err != nil {
|
|
failedAssertions = append(failedAssertions, err)
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
err = lowerTree.VisitDepthChildFirst(asserter, nil)
|
|
if err != nil {
|
|
t.Errorf("Expected no errors when visiting nodes, got: %+v", err)
|
|
}
|
|
|
|
if len(failedAssertions) > 0 {
|
|
str := "\n"
|
|
for _, value := range failedAssertions {
|
|
str += fmt.Sprintf(" - %s\n", value.Error())
|
|
}
|
|
t.Errorf("Expected no errors when evaluating nodes, got: %s", str)
|
|
}
|
|
}
|
|
|
|
func TestCompareWithChanges(t *testing.T) {
|
|
lowerTree := NewFileTree()
|
|
upperTree := NewFileTree()
|
|
paths := [...]string{"/etc", "/usr", "/etc/hosts", "/etc/sudoers", "/usr/bin"}
|
|
|
|
for _, value := range paths {
|
|
lowerTree.AddPath(value, FileInfo{
|
|
Path: value,
|
|
Typeflag: 1,
|
|
MD5sum: [16]byte{0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
|
|
})
|
|
upperTree.AddPath(value, FileInfo{
|
|
Path: value,
|
|
Typeflag: 1,
|
|
MD5sum: [16]byte{1, 1, 1, 0, 1, 0, 0, 0, 0, 0, 0, 0},
|
|
})
|
|
}
|
|
|
|
|
|
lowerTree.Compare(upperTree)
|
|
failedAssertions := []error{}
|
|
asserter := func(n *FileNode) error {
|
|
p := n.Path()
|
|
if p == "/" {
|
|
return nil
|
|
} else if stringInSlice(p, []string{"/etc", "/usr", "/etc/hosts", "/etc/sudoers", "/usr/bin"}) {
|
|
if err := AssertDiffType(n, Changed); err != nil {
|
|
failedAssertions = append(failedAssertions, err)
|
|
}
|
|
} else {
|
|
if err := AssertDiffType(n, Unchanged); err != nil {
|
|
failedAssertions = append(failedAssertions, err)
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
err := lowerTree.VisitDepthChildFirst(asserter, nil)
|
|
if err != nil {
|
|
t.Errorf("Expected no errors when visiting nodes, got: %+v", err)
|
|
}
|
|
|
|
if len(failedAssertions) > 0 {
|
|
str := "\n"
|
|
for _, value := range failedAssertions {
|
|
str += fmt.Sprintf(" - %s\n", value.Error())
|
|
}
|
|
t.Errorf("Expected no errors when evaluating nodes, got: %s", str)
|
|
}
|
|
}
|
|
|
|
|
|
func TestCompareWithRemoves(t *testing.T) {
|
|
lowerTree := NewFileTree()
|
|
upperTree := NewFileTree()
|
|
lowerPaths := [...]string{"/etc", "/usr", "/etc/hosts", "/etc/sudoers", "/usr/bin", "/root", "/root/example", "/root/example/some1", "/root/example/some2"}
|
|
upperPaths := [...]string{"/.wh.etc", "/usr", "/usr/.wh.bin", "/root/.wh.example"}
|
|
|
|
for _, value := range lowerPaths {
|
|
fakeData := FileInfo{
|
|
Path: value,
|
|
Typeflag: 1,
|
|
MD5sum: [16]byte{0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
|
|
}
|
|
lowerTree.AddPath(value, fakeData)
|
|
}
|
|
|
|
for _, value := range upperPaths {
|
|
fakeData := FileInfo{
|
|
Path: value,
|
|
Typeflag: 1,
|
|
MD5sum: [16]byte{0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
|
|
}
|
|
upperTree.AddPath(value, fakeData)
|
|
}
|
|
|
|
lowerTree.Compare(upperTree)
|
|
failedAssertions := []error{}
|
|
asserter := func(n *FileNode) error {
|
|
p := n.Path()
|
|
if p == "/" {
|
|
return nil
|
|
} else if stringInSlice(p,[]string{"/etc", "/usr/bin", "/etc/hosts", "/etc/sudoers", "/root/example/some1", "/root/example/some2", "/root/example"}) {
|
|
if err := AssertDiffType(n, Removed); err != nil {
|
|
failedAssertions = append(failedAssertions, err)
|
|
}
|
|
} else if stringInSlice(p,[]string{"/usr", "/root"}) {
|
|
if err := AssertDiffType(n, Changed); err != nil {
|
|
failedAssertions = append(failedAssertions, err)
|
|
}
|
|
} else {
|
|
if err := AssertDiffType(n, Unchanged); err != nil {
|
|
failedAssertions = append(failedAssertions, err)
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
err := lowerTree.VisitDepthChildFirst(asserter, nil)
|
|
if err != nil {
|
|
t.Errorf("Expected no errors when visiting nodes, got: %+v", err)
|
|
}
|
|
|
|
if len(failedAssertions) > 0 {
|
|
str := "\n"
|
|
for _, value := range failedAssertions {
|
|
str += fmt.Sprintf(" - %s\n", value.Error())
|
|
}
|
|
t.Errorf("Expected no errors when evaluating nodes, got: %s", str)
|
|
}
|
|
}
|
|
|
|
func TestStackRange(t *testing.T) {
|
|
tree := NewFileTree()
|
|
tree.AddPath("/etc/nginx/nginx.conf", FileInfo{})
|
|
tree.AddPath("/etc/nginx/public", FileInfo{})
|
|
tree.AddPath("/var/run/systemd", FileInfo{})
|
|
tree.AddPath("/var/run/bashful", FileInfo{})
|
|
tree.AddPath("/tmp", FileInfo{})
|
|
tree.AddPath("/tmp/nonsense", FileInfo{})
|
|
|
|
tree.RemovePath("/var/run/bashful")
|
|
tree.RemovePath("/tmp")
|
|
|
|
lowerTree := NewFileTree()
|
|
upperTree := NewFileTree()
|
|
lowerPaths := [...]string{"/etc", "/usr", "/etc/hosts", "/etc/sudoers", "/usr/bin"}
|
|
upperPaths := [...]string{"/etc", "/usr", "/etc/hosts", "/etc/sudoers", "/usr/bin"}
|
|
|
|
for _, value := range lowerPaths {
|
|
fakeData := FileInfo{
|
|
Path: value,
|
|
Typeflag: 1,
|
|
MD5sum: [16]byte{0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
|
|
}
|
|
lowerTree.AddPath(value, fakeData)
|
|
}
|
|
|
|
for _, value := range upperPaths {
|
|
fakeData := FileInfo{
|
|
Path: value,
|
|
Typeflag: 1,
|
|
MD5sum: [16]byte{1, 1, 1, 0, 1, 0, 0, 0, 0, 0, 0, 0},
|
|
}
|
|
upperTree.AddPath(value, fakeData)
|
|
}
|
|
trees := []*FileTree{lowerTree, upperTree, tree}
|
|
StackRange(trees, 0, 2)
|
|
}
|
|
|
|
|
|
func TestRemoveOnIterate(t *testing.T) {
|
|
|
|
tree := NewFileTree()
|
|
paths := [...]string{"/etc", "/usr", "/etc/hosts", "/etc/sudoers", "/usr/bin", "/usr/something"}
|
|
|
|
for _, value := range paths {
|
|
fakeData := FileInfo{
|
|
Path: value,
|
|
Typeflag: 1,
|
|
MD5sum: [16]byte{0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
|
|
}
|
|
node, err := tree.AddPath(value, fakeData)
|
|
if err == nil && stringInSlice(node.Path(), []string{"/etc"}) {
|
|
node.Data.ViewInfo.Hidden = true
|
|
}
|
|
}
|
|
|
|
tree.VisitDepthChildFirst(func(node *FileNode) error {
|
|
if node.Data.ViewInfo.Hidden {
|
|
tree.RemovePath(node.Path())
|
|
}
|
|
return nil
|
|
}, nil)
|
|
|
|
expected :=
|
|
`└── usr
|
|
├── bin
|
|
└── something
|
|
`
|
|
actual := tree.String(false)
|
|
if expected != actual {
|
|
t.Errorf("Expected tree string:\n--->%s<---\nGot:\n--->%s<---", expected, actual)
|
|
}
|
|
|
|
}
|
|
|