2020-05-26 22:44:08 -04:00
|
|
|
package common
|
|
|
|
|
|
|
|
import (
|
|
|
|
"testing"
|
|
|
|
|
2020-05-27 19:27:36 -04:00
|
|
|
"bytes"
|
2020-05-28 02:16:42 -04:00
|
|
|
"encoding/hex"
|
2020-05-26 22:44:08 -04:00
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2020-06-04 06:07:07 -04:00
|
|
|
"strings"
|
2020-05-26 22:44:08 -04:00
|
|
|
)
|
|
|
|
|
2020-05-27 15:40:06 -04:00
|
|
|
func consumeString(s string) (out chan byte) {
|
2020-05-26 22:44:08 -04:00
|
|
|
out = make(chan byte)
|
|
|
|
go func() {
|
|
|
|
for _, ch := range s {
|
|
|
|
out <- byte(ch)
|
|
|
|
}
|
|
|
|
close(out)
|
|
|
|
}()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-06-04 06:07:07 -04:00
|
|
|
func collectIntoStringList(in chan string) (result []string) {
|
|
|
|
for item := range in {
|
|
|
|
result = append(result, item)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-05-26 22:44:08 -04:00
|
|
|
func uncommentFromString(s string) string {
|
2020-05-27 15:40:06 -04:00
|
|
|
inCh := consumeString(s)
|
|
|
|
out := uncomment(inCh)
|
2020-05-26 22:44:08 -04:00
|
|
|
|
|
|
|
result := ""
|
|
|
|
for reading := true; reading; {
|
2020-05-27 15:40:06 -04:00
|
|
|
if item, ok := <-out; !ok {
|
|
|
|
break
|
|
|
|
} else {
|
|
|
|
result += string(item)
|
2020-05-26 22:44:08 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestParserUncomment(t *testing.T) {
|
|
|
|
var result string
|
|
|
|
|
|
|
|
test_0 := "this is a straight-up line"
|
|
|
|
result_0 := test_0
|
|
|
|
|
|
|
|
result = uncommentFromString(test_0)
|
|
|
|
if result != result_0 {
|
|
|
|
t.Errorf("Expected %#v, received %#v", result_0, result)
|
|
|
|
}
|
|
|
|
|
|
|
|
test_1 := "this is a straight-up line with a newline\n"
|
|
|
|
result_1 := test_1
|
|
|
|
|
|
|
|
result = uncommentFromString(test_1)
|
|
|
|
if result != result_1 {
|
|
|
|
t.Errorf("Expected %#v, received %#v", result_1, result)
|
|
|
|
}
|
|
|
|
|
|
|
|
test_2 := "this line has a comment # at its end"
|
|
|
|
result_2 := "this line has a comment "
|
|
|
|
|
|
|
|
result = uncommentFromString(test_2)
|
|
|
|
if result != result_2 {
|
|
|
|
t.Errorf("Expected %#v, received %#v", result_2, result)
|
|
|
|
}
|
|
|
|
|
|
|
|
test_3 := "# this whole line is commented"
|
|
|
|
result_3 := ""
|
|
|
|
|
|
|
|
result = uncommentFromString(test_3)
|
|
|
|
if result != result_3 {
|
|
|
|
t.Errorf("Expected %#v, received %#v", result_3, result)
|
|
|
|
}
|
|
|
|
|
|
|
|
test_4 := "this\nhas\nmultiple\nlines"
|
|
|
|
result_4 := test_4
|
|
|
|
|
|
|
|
result = uncommentFromString(test_4)
|
|
|
|
if result != result_4 {
|
|
|
|
t.Errorf("Expected %#v, received %#v", result_4, result)
|
|
|
|
}
|
|
|
|
|
|
|
|
test_5 := "this only has\n# one line"
|
|
|
|
result_5 := "this only has\n"
|
|
|
|
result = uncommentFromString(test_5)
|
|
|
|
if result != result_5 {
|
|
|
|
t.Errorf("Expected %#v, received %#v", result_5, result)
|
|
|
|
}
|
|
|
|
|
|
|
|
test_6 := "this is\npartially # commented"
|
|
|
|
result_6 := "this is\npartially "
|
|
|
|
|
|
|
|
result = uncommentFromString(test_6)
|
|
|
|
if result != result_6 {
|
|
|
|
t.Errorf("Expected %#v, received %#v", result_6, result)
|
|
|
|
}
|
|
|
|
|
|
|
|
test_7 := "this # has\nmultiple # lines\ncommented # out"
|
|
|
|
result_7 := "this \nmultiple \ncommented "
|
|
|
|
|
|
|
|
result = uncommentFromString(test_7)
|
|
|
|
if result != result_7 {
|
|
|
|
t.Errorf("Expected %#v, received %#v", result_7, result)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func tokenizeDhcpConfigFromString(s string) []string {
|
2020-05-27 15:40:06 -04:00
|
|
|
inCh := consumeString(s)
|
|
|
|
out := tokenizeDhcpConfig(inCh)
|
2020-05-26 22:44:08 -04:00
|
|
|
|
|
|
|
result := make([]string, 0)
|
2020-05-27 15:40:06 -04:00
|
|
|
for {
|
|
|
|
if item, ok := <-out; !ok {
|
|
|
|
break
|
|
|
|
} else {
|
|
|
|
result = append(result, item)
|
2020-05-26 22:44:08 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestParserTokenizeDhcp(t *testing.T) {
|
|
|
|
|
|
|
|
test_1 := `
|
|
|
|
subnet 127.0.0.0 netmask 255.255.255.252 {
|
|
|
|
item 1234 5678;
|
|
|
|
tabbed item-1 1234;
|
|
|
|
quoted item-2 "hola mundo.";
|
|
|
|
}
|
|
|
|
`
|
|
|
|
expected := []string{
|
|
|
|
"subnet", "127.0.0.0", "netmask", "255.255.255.252", "{",
|
|
|
|
"item", "1234", "5678", ";",
|
|
|
|
"tabbed", "item-1", "1234", ";",
|
|
|
|
"quoted", "item-2", "\"hola mundo.\"", ";",
|
|
|
|
"}",
|
|
|
|
}
|
|
|
|
result := tokenizeDhcpConfigFromString(test_1)
|
|
|
|
|
2020-05-26 23:56:02 -04:00
|
|
|
t.Logf("testing for: %v", expected)
|
|
|
|
t.Logf("checking out: %v", result)
|
2020-05-26 22:44:08 -04:00
|
|
|
if len(result) != len(expected) {
|
|
|
|
t.Fatalf("length of token lists do not match (%d != %d)", len(result), len(expected))
|
|
|
|
}
|
|
|
|
|
|
|
|
for index := range expected {
|
2020-05-28 14:12:42 -04:00
|
|
|
if expected[index] != result[index] {
|
2020-05-26 23:56:02 -04:00
|
|
|
t.Errorf("unexpected token at index %d: %v != %v", index, expected[index], result[index])
|
2020-05-26 22:44:08 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func consumeTokens(tokes []string) chan string {
|
|
|
|
out := make(chan string)
|
|
|
|
go func() {
|
|
|
|
for _, item := range tokes {
|
|
|
|
out <- item
|
|
|
|
}
|
|
|
|
out <- ";"
|
|
|
|
close(out)
|
|
|
|
}()
|
|
|
|
return out
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestParserDhcpParameters(t *testing.T) {
|
|
|
|
var ch chan string
|
|
|
|
|
|
|
|
test_1 := []string{"option", "whee", "whooo"}
|
|
|
|
ch = consumeTokens(test_1)
|
|
|
|
|
|
|
|
result := parseTokenParameter(ch)
|
|
|
|
if result.name != "option" {
|
|
|
|
t.Errorf("expected name %s, got %s", test_1[0], result.name)
|
|
|
|
}
|
|
|
|
if len(result.operand) == 2 {
|
|
|
|
if result.operand[0] != "whee" {
|
|
|
|
t.Errorf("expected operand[%d] as %s, got %s", 0, "whee", result.operand[0])
|
|
|
|
}
|
|
|
|
if result.operand[1] != "whooo" {
|
|
|
|
t.Errorf("expected operand[%d] as %s, got %s", 0, "whooo", result.operand[1])
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
t.Errorf("expected %d operands, got %d", 2, len(result.operand))
|
|
|
|
}
|
|
|
|
|
|
|
|
test_2 := []string{"whaaa", "whoaaa", ";", "wooops"}
|
|
|
|
ch = consumeTokens(test_2)
|
|
|
|
|
|
|
|
result = parseTokenParameter(ch)
|
|
|
|
if result.name != "whaaa" {
|
|
|
|
t.Errorf("expected name %s, got %s", test_2[0], result.name)
|
|
|
|
}
|
|
|
|
if len(result.operand) == 1 {
|
|
|
|
if result.operand[0] != "whoaaa" {
|
|
|
|
t.Errorf("expected operand[%d] as %s, got %s", 0, "whoaaa", result.operand[0])
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
t.Errorf("expected %d operands, got %d", 1, len(result.operand))
|
|
|
|
}
|
|
|
|
|
|
|
|
test_3 := []string{"optionz", "only", "{", "culled"}
|
|
|
|
ch = consumeTokens(test_3)
|
|
|
|
|
|
|
|
result = parseTokenParameter(ch)
|
|
|
|
if result.name != "optionz" {
|
|
|
|
t.Errorf("expected name %s, got %s", test_3[0], result.name)
|
|
|
|
}
|
|
|
|
if len(result.operand) == 1 {
|
|
|
|
if result.operand[0] != "only" {
|
|
|
|
t.Errorf("expected operand[%d] as %s, got %s", 0, "only", result.operand[0])
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
t.Errorf("expected %d operands, got %d", 1, len(result.operand))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func consumeDhcpConfig(items []string) (tkGroup, error) {
|
|
|
|
out := make(chan string)
|
|
|
|
tch := consumeTokens(items)
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
for item := range tch {
|
|
|
|
out <- item
|
|
|
|
}
|
|
|
|
close(out)
|
|
|
|
}()
|
|
|
|
|
2020-05-27 15:40:06 -04:00
|
|
|
return parseDhcpConfig(out)
|
2020-05-26 22:44:08 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
func compareSlice(a, b []string) bool {
|
|
|
|
if len(a) != len(b) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
for i := range a {
|
|
|
|
if a[i] != b[i] {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestParserDhcpConfigParse(t *testing.T) {
|
|
|
|
test_1 := []string{
|
|
|
|
"allow", "unused-option", ";",
|
|
|
|
"lease-option", "1234", ";",
|
|
|
|
"more", "options", "hi", ";",
|
|
|
|
}
|
|
|
|
result_1, err := consumeDhcpConfig(test_1)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("%s", err)
|
|
|
|
}
|
|
|
|
if len(result_1.params) != 4 {
|
|
|
|
t.Fatalf("expected %d params, got %d", 3, len(result_1.params))
|
|
|
|
}
|
|
|
|
if result_1.params[0].name != "allow" {
|
|
|
|
t.Errorf("expected %s, got %s", "allow", result_1.params[0].name)
|
|
|
|
}
|
|
|
|
if !compareSlice(result_1.params[0].operand, []string{"unused-option"}) {
|
|
|
|
t.Errorf("unexpected options parsed: %v", result_1.params[0].operand)
|
|
|
|
}
|
|
|
|
if result_1.params[1].name != "lease-option" {
|
|
|
|
t.Errorf("expected %s, got %s", "lease-option", result_1.params[1].name)
|
|
|
|
}
|
|
|
|
if !compareSlice(result_1.params[1].operand, []string{"1234"}) {
|
|
|
|
t.Errorf("unexpected options parsed: %v", result_1.params[1].operand)
|
|
|
|
}
|
|
|
|
if result_1.params[2].name != "more" {
|
|
|
|
t.Errorf("expected %s, got %s", "lease-option", result_1.params[2].name)
|
|
|
|
}
|
|
|
|
if !compareSlice(result_1.params[2].operand, []string{"options", "hi"}) {
|
|
|
|
t.Errorf("unexpected options parsed: %v", result_1.params[2].operand)
|
|
|
|
}
|
|
|
|
|
|
|
|
test_2 := []string{
|
|
|
|
"first-option", ";",
|
|
|
|
"child", "group", "{",
|
|
|
|
"blah", ";",
|
|
|
|
"meh", ";",
|
|
|
|
"}",
|
|
|
|
"hidden", "option", "57005", ";",
|
|
|
|
"host", "device", "two", "{",
|
|
|
|
"skipped", "option", ";",
|
|
|
|
"more", "skipped", "options", ";",
|
|
|
|
"}",
|
|
|
|
"last", "option", "but", "unterminated",
|
|
|
|
}
|
|
|
|
result_2, err := consumeDhcpConfig(test_2)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("%s", err)
|
|
|
|
}
|
|
|
|
if len(result_2.groups) != 2 {
|
|
|
|
t.Fatalf("expected %d groups, got %d", 2, len(result_2.groups))
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(result_2.params) != 3 {
|
|
|
|
t.Errorf("expected %d options, got %d", 3, len(result_2.params))
|
|
|
|
}
|
|
|
|
|
|
|
|
group0 := result_2.groups[0]
|
|
|
|
if group0.id.name != "child" {
|
|
|
|
t.Errorf("expected group %s, got %s", "child", group0.id.name)
|
|
|
|
}
|
|
|
|
if len(group0.id.operand) != 1 {
|
|
|
|
t.Errorf("expected group operand %d, got %d", 1, len(group0.params))
|
|
|
|
}
|
|
|
|
if len(group0.params) != 2 {
|
|
|
|
t.Errorf("expected group params %d, got %d", 2, len(group0.params))
|
|
|
|
}
|
|
|
|
|
|
|
|
group1 := result_2.groups[1]
|
|
|
|
if group1.id.name != "host" {
|
|
|
|
t.Errorf("expected group %s, got %s", "host", group1.id.name)
|
|
|
|
}
|
|
|
|
if len(group1.id.operand) != 2 {
|
|
|
|
t.Errorf("expected group operand %d, got %d", 2, len(group1.params))
|
|
|
|
}
|
|
|
|
if len(group1.params) != 2 {
|
|
|
|
t.Errorf("expected group params %d, got %d", 2, len(group1.params))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestParserReadDhcpConfig(t *testing.T) {
|
|
|
|
expected := []string{
|
|
|
|
`{global}
|
|
|
|
grants : map[unknown-clients:0]
|
|
|
|
parameters : map[default-lease-time:1800 max-lease-time:7200]
|
|
|
|
`,
|
|
|
|
|
|
|
|
`{subnet4 172.33.33.0/24},{global}
|
|
|
|
address : range4:172.33.33.128-172.33.33.254
|
|
|
|
options : map[broadcast-address:172.33.33.255 routers:172.33.33.2]
|
|
|
|
grants : map[unknown-clients:0]
|
|
|
|
parameters : map[default-lease-time:2400 max-lease-time:9600]
|
|
|
|
`,
|
|
|
|
|
|
|
|
`{host name:vmnet8},{global}
|
|
|
|
address : hardware-address:ethernet[00:50:56:c0:00:08],fixed-address4:172.33.33.1
|
|
|
|
options : map[domain-name:"packer.test"]
|
|
|
|
grants : map[unknown-clients:0]
|
|
|
|
parameters : map[default-lease-time:1800 max-lease-time:7200]
|
|
|
|
`,
|
|
|
|
}
|
|
|
|
|
|
|
|
f, err := os.Open(filepath.Join("testdata", "dhcpd-example.conf"))
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Unable to open dhcpd.conf sample: %s", err)
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
|
|
|
|
config, err := ReadDhcpConfiguration(f)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Unable to read dhcpd.conf samplpe: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(config) != 3 {
|
|
|
|
t.Fatalf("expected %d entries, got %d", 3, len(config))
|
|
|
|
}
|
|
|
|
|
|
|
|
for index, item := range config {
|
|
|
|
if item.repr() != expected[index] {
|
|
|
|
t.Errorf("Parsing of declaration %d did not match what was expected", index)
|
|
|
|
t.Logf("Result from parsing:\n%s", item.repr())
|
|
|
|
t.Logf("Expected to parse:\n%s", expected[index])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-05-26 23:56:02 -04:00
|
|
|
|
|
|
|
func TestParserTokenizeNetworkMap(t *testing.T) {
|
|
|
|
|
|
|
|
test_1 := "group.attribute = \"string\""
|
|
|
|
expected := []string{
|
|
|
|
"group.attribute", "=", "\"string\"",
|
|
|
|
}
|
|
|
|
result := tokenizeDhcpConfigFromString(test_1)
|
|
|
|
if len(result) != len(expected) {
|
|
|
|
t.Fatalf("length of token lists do not match (%d != %d)", len(result), len(expected))
|
|
|
|
}
|
|
|
|
|
|
|
|
for index := range expected {
|
2020-05-28 14:12:42 -04:00
|
|
|
if expected[index] != result[index] {
|
2020-05-26 23:56:02 -04:00
|
|
|
t.Errorf("unexpected token at index %d: %v != %v", index, expected[index], result[index])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
test_2 := "attribute == \""
|
|
|
|
expected = []string{
|
|
|
|
"attribute", "==", "\"",
|
|
|
|
}
|
|
|
|
result = tokenizeDhcpConfigFromString(test_2)
|
|
|
|
if len(result) != len(expected) {
|
|
|
|
t.Fatalf("length of token lists do not match (%d != %d)", len(result), len(expected))
|
|
|
|
}
|
|
|
|
|
|
|
|
test_3 := "attribute ....... ======\nnew lines should make no difference"
|
|
|
|
expected = []string{
|
|
|
|
"attribute", ".......", "======", "new", "lines", "should", "make", "no", "difference",
|
|
|
|
}
|
|
|
|
result = tokenizeDhcpConfigFromString(test_3)
|
|
|
|
if len(result) != len(expected) {
|
|
|
|
t.Fatalf("length of token lists do not match (%d != %d)", len(result), len(expected))
|
|
|
|
}
|
|
|
|
|
|
|
|
test_4 := "\t\t\t\t thishadwhitespacebeforebeingparsed\t \t \t \t\n\n"
|
|
|
|
expected = []string{
|
|
|
|
"thishadwhitespacebeforebeingparsed",
|
|
|
|
}
|
|
|
|
result = tokenizeDhcpConfigFromString(test_4)
|
|
|
|
if len(result) != len(expected) {
|
|
|
|
t.Fatalf("length of token lists do not match (%d != %d)", len(result), len(expected))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestParserReadNetworkMap(t *testing.T) {
|
|
|
|
f, err := os.Open(filepath.Join("testdata", "netmap-example.conf"))
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Unable to open netmap.conf sample: %s", err)
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
|
|
|
|
netmap, err := ReadNetworkMap(f)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Unable to read netmap.conf samplpe: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
expected_keys := []string{"device", "name"}
|
|
|
|
for _, item := range netmap {
|
|
|
|
for _, name := range expected_keys {
|
|
|
|
_, ok := item[name]
|
|
|
|
if !ok {
|
|
|
|
t.Errorf("unable to find expected key %v in map: %v", name, item)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
expected_vmnet0 := [][]string{
|
|
|
|
[]string{"device", "vmnet0"},
|
|
|
|
[]string{"name", "Bridged"},
|
|
|
|
}
|
|
|
|
for _, item := range netmap {
|
|
|
|
if item["device"] != "vmnet0" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
for _, expectpair := range expected_vmnet0 {
|
|
|
|
name := expectpair[0]
|
|
|
|
value := expectpair[1]
|
|
|
|
if item[name] != value {
|
|
|
|
t.Errorf("expected value %v for attribute %v, got %v", value, name, item[name])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
expected_vmnet1 := [][]string{
|
|
|
|
[]string{"device", "vmnet1"},
|
|
|
|
[]string{"name", "HostOnly"},
|
|
|
|
}
|
|
|
|
for _, item := range netmap {
|
|
|
|
if item["device"] != "vmnet1" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
for _, expectpair := range expected_vmnet1 {
|
|
|
|
name := expectpair[0]
|
|
|
|
value := expectpair[1]
|
|
|
|
if item[name] != value {
|
|
|
|
t.Errorf("expected value %v for attribute %v, got %v", value, name, item[name])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
expected_vmnet8 := [][]string{
|
|
|
|
[]string{"device", "vmnet8"},
|
|
|
|
[]string{"name", "NAT"},
|
|
|
|
}
|
|
|
|
for _, item := range netmap {
|
|
|
|
if item["device"] != "vmnet8" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
for _, expectpair := range expected_vmnet8 {
|
|
|
|
name := expectpair[0]
|
|
|
|
value := expectpair[1]
|
|
|
|
if item[name] != value {
|
|
|
|
t.Errorf("expected value %v for attribute %v, got %v", value, name, item[name])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-05-27 19:27:36 -04:00
|
|
|
|
|
|
|
func collectIntoString(in chan byte) string {
|
|
|
|
result := ""
|
|
|
|
for item := range in {
|
|
|
|
result += string(item)
|
|
|
|
}
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestParserConsumeUntilSentinel(t *testing.T) {
|
|
|
|
|
|
|
|
test_1 := "consume until a semicolon; yeh?"
|
|
|
|
expected_1 := "consume until a semicolon"
|
|
|
|
|
|
|
|
ch := consumeString(test_1)
|
|
|
|
resultch, _ := consumeUntilSentinel(';', ch)
|
|
|
|
result := string(resultch)
|
|
|
|
if expected_1 != result {
|
|
|
|
t.Errorf("expected %#v, got %#v", expected_1, result)
|
|
|
|
}
|
|
|
|
|
|
|
|
test_2 := "; this is only a semi"
|
|
|
|
expected_2 := ""
|
|
|
|
|
|
|
|
ch = consumeString(test_2)
|
|
|
|
resultch, _ = consumeUntilSentinel(';', ch)
|
|
|
|
result = string(resultch)
|
|
|
|
if expected_2 != result {
|
|
|
|
t.Errorf("expected %#v, got %#v", expected_2, result)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestParserFilterCharacters(t *testing.T) {
|
|
|
|
|
|
|
|
test_1 := []string{" ", "ignore all spaces"}
|
|
|
|
expected_1 := "ignoreallspaces"
|
|
|
|
|
|
|
|
ch := consumeString(test_1[1])
|
|
|
|
outch := filterOutCharacters(bytes.NewBufferString(test_1[0]).Bytes(), ch)
|
|
|
|
result := collectIntoString(outch)
|
|
|
|
if result != expected_1 {
|
|
|
|
t.Errorf("expected %#v, got %#v", expected_1, result)
|
|
|
|
}
|
|
|
|
|
|
|
|
test_2 := []string{"\n\v\t\r ", "ignore\nall\rwhite\v\v space "}
|
|
|
|
expected_2 := "ignoreallwhitespace"
|
|
|
|
|
|
|
|
ch = consumeString(test_2[1])
|
|
|
|
outch = filterOutCharacters(bytes.NewBufferString(test_2[0]).Bytes(), ch)
|
|
|
|
result = collectIntoString(outch)
|
|
|
|
if result != expected_2 {
|
|
|
|
t.Errorf("expected %#v, got %#v", expected_2, result)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestParserConsumeOpenClosePair(t *testing.T) {
|
|
|
|
test_1 := "(everything)"
|
|
|
|
expected_1 := []string{"", test_1}
|
|
|
|
|
|
|
|
testch := consumeString(test_1)
|
|
|
|
prefix, ch := consumeOpenClosePair('(', ')', testch)
|
|
|
|
if string(prefix) != expected_1[0] {
|
|
|
|
t.Errorf("expected prefix %#v, got %#v", expected_1[0], prefix)
|
|
|
|
}
|
|
|
|
result := collectIntoString(ch)
|
|
|
|
if result != expected_1[1] {
|
|
|
|
t.Errorf("expected %#v, got %#v", expected_1[1], test_1)
|
|
|
|
}
|
|
|
|
|
|
|
|
test_2 := "prefixed (everything)"
|
|
|
|
expected_2 := []string{"prefixed ", "(everything)"}
|
|
|
|
|
|
|
|
testch = consumeString(test_2)
|
|
|
|
prefix, ch = consumeOpenClosePair('(', ')', testch)
|
|
|
|
if string(prefix) != expected_2[0] {
|
|
|
|
t.Errorf("expected prefix %#v, got %#v", expected_2[0], prefix)
|
|
|
|
}
|
|
|
|
result = collectIntoString(ch)
|
|
|
|
if result != expected_2[1] {
|
|
|
|
t.Errorf("expected %#v, got %#v", expected_2[1], test_2)
|
|
|
|
}
|
|
|
|
|
|
|
|
test_3 := "this(is()suffixed"
|
|
|
|
expected_3 := []string{"this", "(is()"}
|
|
|
|
|
|
|
|
testch = consumeString(test_3)
|
|
|
|
prefix, ch = consumeOpenClosePair('(', ')', testch)
|
|
|
|
if string(prefix) != expected_3[0] {
|
|
|
|
t.Errorf("expected prefix %#v, got %#v", expected_3[0], prefix)
|
|
|
|
}
|
|
|
|
result = collectIntoString(ch)
|
|
|
|
if result != expected_3[1] {
|
|
|
|
t.Errorf("expected %#v, got %#v", expected_3[1], test_2)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestParserCombinators(t *testing.T) {
|
|
|
|
|
|
|
|
test_1 := "across # ignore\nmultiple lines;"
|
|
|
|
expected_1 := "across multiple lines"
|
|
|
|
|
|
|
|
ch := consumeString(test_1)
|
|
|
|
inch := uncomment(ch)
|
|
|
|
whch := filterOutCharacters([]byte{'\n'}, inch)
|
|
|
|
resultch, _ := consumeUntilSentinel(';', whch)
|
|
|
|
result := string(resultch)
|
|
|
|
if expected_1 != result {
|
|
|
|
t.Errorf("expected %#v, got %#v", expected_1, result)
|
|
|
|
}
|
|
|
|
|
|
|
|
test_2 := "lease blah {\n blah\r\n# skipping this line\nblahblah # ignore semicolon;\n last item;\n\n };;;;;;"
|
|
|
|
expected_2 := []string{"lease blah ", "{ blahblahblah last item; }"}
|
|
|
|
|
|
|
|
ch = consumeString(test_2)
|
|
|
|
inch = uncomment(ch)
|
|
|
|
whch = filterOutCharacters([]byte{'\n', '\v', '\r'}, inch)
|
|
|
|
prefix, pairch := consumeOpenClosePair('{', '}', whch)
|
|
|
|
|
|
|
|
result = collectIntoString(pairch)
|
|
|
|
if string(prefix) != expected_2[0] {
|
|
|
|
t.Errorf("expected prefix %#v, got %#v", expected_2[0], prefix)
|
|
|
|
}
|
|
|
|
if result != expected_2[1] {
|
|
|
|
t.Errorf("expected %#v, got %#v", expected_2[1], result)
|
|
|
|
}
|
|
|
|
|
|
|
|
test_3 := "lease blah { # comment\n item 1;\n item 2;\n } not imortant"
|
|
|
|
expected_3_prefix := "lease blah "
|
|
|
|
expected_3 := []string{"{ item 1", " item 2", " }"}
|
|
|
|
|
|
|
|
sch := consumeString(test_3)
|
|
|
|
inch = uncomment(sch)
|
|
|
|
wch := filterOutCharacters([]byte{'\n', '\v', '\r'}, inch)
|
|
|
|
lease, itemch := consumeOpenClosePair('{', '}', wch)
|
|
|
|
if string(lease) != expected_3_prefix {
|
|
|
|
t.Errorf("expected %#v, got %#v", expected_3_prefix, string(lease))
|
|
|
|
}
|
|
|
|
|
|
|
|
result_3 := []string{}
|
|
|
|
for reading := true; reading; {
|
|
|
|
item, ok := consumeUntilSentinel(';', itemch)
|
|
|
|
result_3 = append(result_3, string(item))
|
|
|
|
if !ok {
|
|
|
|
reading = false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for index := range expected_3 {
|
|
|
|
if expected_3[index] != result_3[index] {
|
|
|
|
t.Errorf("expected index %d as %#v, got %#v", index, expected_3[index], result_3[index])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-05-27 22:35:24 -04:00
|
|
|
|
|
|
|
func TestParserDhcpdLeaseBytesDecoder(t *testing.T) {
|
|
|
|
test_1 := "00:0d:0e:0a:0d:00"
|
|
|
|
expected_1 := []byte{0, 13, 14, 10, 13, 0}
|
|
|
|
|
|
|
|
result, err := decodeDhcpdLeaseBytes(test_1)
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("unable to decode address: %s", err)
|
|
|
|
}
|
2020-05-28 14:53:49 -04:00
|
|
|
if !bytes.Equal(result, expected_1) {
|
2020-05-27 22:35:24 -04:00
|
|
|
t.Errorf("expected %v, got %v", expected_1, result)
|
|
|
|
}
|
|
|
|
|
|
|
|
test_2 := "11"
|
|
|
|
expected_2 := []byte{17}
|
|
|
|
|
|
|
|
result, err = decodeDhcpdLeaseBytes(test_2)
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("unable to decode address: %s", err)
|
|
|
|
}
|
2020-05-28 14:53:49 -04:00
|
|
|
if !bytes.Equal(result, expected_2) {
|
2020-05-27 22:35:24 -04:00
|
|
|
t.Errorf("expected %v, got %v", expected_2, result)
|
|
|
|
}
|
|
|
|
|
|
|
|
failtest_1 := ""
|
2020-05-28 14:53:49 -04:00
|
|
|
_, err = decodeDhcpdLeaseBytes(failtest_1)
|
2020-05-27 22:35:24 -04:00
|
|
|
if err == nil {
|
|
|
|
t.Errorf("expected decoding error: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
failtest_2 := "000000"
|
2020-05-28 14:53:49 -04:00
|
|
|
_, err = decodeDhcpdLeaseBytes(failtest_2)
|
2020-05-27 22:35:24 -04:00
|
|
|
if err == nil {
|
|
|
|
t.Errorf("expected decoding error: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
failtest_3 := "000:00"
|
2020-05-28 14:53:49 -04:00
|
|
|
_, err = decodeDhcpdLeaseBytes(failtest_3)
|
2020-05-27 22:35:24 -04:00
|
|
|
if err == nil {
|
|
|
|
t.Errorf("expected decoding error: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
failtest_4 := "00:00:"
|
2020-05-28 14:53:49 -04:00
|
|
|
_, err = decodeDhcpdLeaseBytes(failtest_4)
|
2020-05-27 22:35:24 -04:00
|
|
|
if err == nil {
|
|
|
|
t.Errorf("expected decoding error: %s", err)
|
|
|
|
}
|
|
|
|
}
|
2020-05-28 02:16:42 -04:00
|
|
|
|
|
|
|
func consumeLeaseString(s string) chan byte {
|
|
|
|
sch := consumeString(s)
|
|
|
|
uncommentedch := uncomment(sch)
|
|
|
|
return filterOutCharacters([]byte{'\n', '\r', '\v'}, uncommentedch)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestParserReadDhcpdLeaseEntry(t *testing.T) {
|
|
|
|
test_1 := "lease 127.0.0.1 {\nhardware ethernet 00:11:22:33 ;\nuid 00:11 ;\n }"
|
|
|
|
expected_1 := map[string]string{
|
|
|
|
"address": "127.0.0.1",
|
|
|
|
"ether": "00112233",
|
|
|
|
"uid": "0011",
|
|
|
|
}
|
|
|
|
|
2020-05-28 02:51:22 -04:00
|
|
|
result, err := readDhcpdLeaseEntry(consumeLeaseString(test_1))
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("error parsing entry: %v", err)
|
|
|
|
}
|
2020-05-28 02:16:42 -04:00
|
|
|
if result.address != expected_1["address"] {
|
|
|
|
t.Errorf("expected address %v, got %v", expected_1["address"], result.address)
|
|
|
|
}
|
|
|
|
if hex.EncodeToString(result.ether) != expected_1["ether"] {
|
|
|
|
t.Errorf("expected ether %v, got %v", expected_1["ether"], hex.EncodeToString(result.ether))
|
|
|
|
}
|
|
|
|
if hex.EncodeToString(result.uid) != expected_1["uid"] {
|
|
|
|
t.Errorf("expected uid %v, got %v", expected_1["uid"], hex.EncodeToString(result.uid))
|
|
|
|
}
|
|
|
|
|
|
|
|
test_2 := " \n\t lease 192.168.21.254{ hardware\n ethernet 44:55:66:77:88:99;uid 00:1\n1:22:3\r3:44;\n starts 57005 2006/01/02 15:04:05;ends 57005 2006/01/03 15:04:05;\tunknown item1; unknown item2; } "
|
|
|
|
expected_2 := map[string]string{
|
|
|
|
"address": "192.168.21.254",
|
|
|
|
"ether": "445566778899",
|
|
|
|
"uid": "0011223344",
|
|
|
|
"starts": "2006-01-02 15:04:05 +0000 UTC",
|
|
|
|
"ends": "2006-01-03 15:04:05 +0000 UTC",
|
|
|
|
}
|
2020-05-28 02:51:22 -04:00
|
|
|
result, err = readDhcpdLeaseEntry(consumeLeaseString(test_2))
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("error parsing entry: %v", err)
|
|
|
|
}
|
2020-05-28 02:16:42 -04:00
|
|
|
if result.address != expected_2["address"] {
|
|
|
|
t.Errorf("expected address %v, got %v", expected_2["address"], result.address)
|
|
|
|
}
|
|
|
|
if hex.EncodeToString(result.ether) != expected_2["ether"] {
|
|
|
|
t.Errorf("expected ether %v, got %v", expected_2["ether"], hex.EncodeToString(result.ether))
|
|
|
|
}
|
|
|
|
if hex.EncodeToString(result.uid) != expected_2["uid"] {
|
|
|
|
t.Errorf("expected uid %v, got %v", expected_2["uid"], hex.EncodeToString(result.uid))
|
|
|
|
}
|
|
|
|
if result.starts.String() != expected_2["starts"] {
|
|
|
|
t.Errorf("expected starts %v, got %v", expected_2["starts"], result.starts)
|
|
|
|
}
|
|
|
|
if result.ends.String() != expected_2["ends"] {
|
|
|
|
t.Errorf("expected ends %v, got %v", expected_2["ends"], result.ends)
|
|
|
|
}
|
|
|
|
if result.starts_weekday != 57005 {
|
|
|
|
t.Errorf("expected starts weekday %v, got %v", 57005, result.starts_weekday)
|
|
|
|
}
|
|
|
|
if result.ends_weekday != 57005 {
|
|
|
|
t.Errorf("expected ends weekday %v, got %v", 57005, result.ends_weekday)
|
|
|
|
}
|
|
|
|
}
|
2020-05-28 04:08:50 -04:00
|
|
|
|
|
|
|
func TestParserReadDhcpdLeases(t *testing.T) {
|
|
|
|
f, err := os.Open(filepath.Join("testdata", "dhcpd-example.leases"))
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Unable to open dhcpd.leases sample: %s", err)
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
|
|
|
|
results, err := ReadDhcpdLeaseEntries(f)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Error reading lease: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// some simple utilities
|
|
|
|
filter_address := func(address string, items []dhcpLeaseEntry) (result []dhcpLeaseEntry) {
|
|
|
|
for _, item := range items {
|
|
|
|
if item.address == address {
|
|
|
|
result = append(result, item)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
find_uid := func(uid string, items []dhcpLeaseEntry) *dhcpLeaseEntry {
|
|
|
|
for _, item := range items {
|
|
|
|
if uid == hex.EncodeToString(item.uid) {
|
|
|
|
return &item
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
find_ether := func(ether string, items []dhcpLeaseEntry) *dhcpLeaseEntry {
|
|
|
|
for _, item := range items {
|
|
|
|
if ether == hex.EncodeToString(item.ether) {
|
|
|
|
return &item
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// actual unit tests
|
|
|
|
test_1 := map[string]string{
|
|
|
|
"address": "127.0.0.19",
|
|
|
|
"uid": "010dead099aabb",
|
|
|
|
"ether": "0dead099aabb",
|
|
|
|
}
|
|
|
|
test_1_findings := filter_address(test_1["address"], results)
|
|
|
|
if len(test_1_findings) != 2 {
|
|
|
|
t.Errorf("expected %d matching entries, got %d", 2, len(test_1_findings))
|
|
|
|
} else {
|
|
|
|
res := find_ether(test_1["ether"], test_1_findings)
|
|
|
|
if res == nil {
|
|
|
|
t.Errorf("unable to find item with ether %v", test_1["ether"])
|
|
|
|
} else if hex.EncodeToString(res.uid) != test_1["uid"] {
|
|
|
|
t.Errorf("expected uid %s, got %s", test_1["uid"], hex.EncodeToString(res.uid))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
test_2 := map[string]string{
|
|
|
|
"address": "127.0.0.19",
|
|
|
|
"uid": "010dead0667788",
|
|
|
|
"ether": "0dead0667788",
|
|
|
|
}
|
|
|
|
test_2_findings := filter_address(test_2["address"], results)
|
|
|
|
if len(test_2_findings) != 2 {
|
|
|
|
t.Errorf("expected %d matching entries, got %d", 2, len(test_2_findings))
|
|
|
|
} else {
|
|
|
|
res := find_ether(test_2["ether"], test_2_findings)
|
|
|
|
if res == nil {
|
|
|
|
t.Errorf("unable to find item with ether %v", test_2["ether"])
|
|
|
|
} else if hex.EncodeToString(res.uid) != test_2["uid"] {
|
|
|
|
t.Errorf("expected uid %s, got %s", test_2["uid"], hex.EncodeToString(res.uid))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
test_3 := map[string]string{
|
|
|
|
"address": "127.0.0.17",
|
|
|
|
"uid": "010dead0334455",
|
|
|
|
"ether": "0dead0667788",
|
|
|
|
}
|
|
|
|
test_3_findings := filter_address(test_3["address"], results)
|
|
|
|
if len(test_3_findings) != 2 {
|
|
|
|
t.Errorf("expected %d matching entries, got %d", 2, len(test_3_findings))
|
|
|
|
} else {
|
|
|
|
res := find_uid(test_3["uid"], test_3_findings)
|
|
|
|
if res == nil {
|
|
|
|
t.Errorf("unable to find item with uid %v", test_3["uid"])
|
|
|
|
} else if hex.EncodeToString(res.ether) != test_3["ether"] {
|
|
|
|
t.Errorf("expected ethernet hardware %s, got %s", test_3["ether"], hex.EncodeToString(res.ether))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
test_4 := map[string]string{
|
|
|
|
"address": "127.0.0.17",
|
|
|
|
"uid": "010dead0001122",
|
|
|
|
"ether": "0dead0667788",
|
|
|
|
}
|
|
|
|
test_4_findings := filter_address(test_4["address"], results)
|
|
|
|
if len(test_4_findings) != 2 {
|
|
|
|
t.Errorf("expected %d matching entries, got %d", 2, len(test_4_findings))
|
|
|
|
} else {
|
|
|
|
res := find_uid(test_4["uid"], test_4_findings)
|
|
|
|
if res == nil {
|
|
|
|
t.Errorf("unable to find item with uid %v", test_4["uid"])
|
|
|
|
} else if hex.EncodeToString(res.ether) != test_4["ether"] {
|
|
|
|
t.Errorf("expected ethernet hardware %s, got %s", test_4["ether"], hex.EncodeToString(res.ether))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-06-04 06:07:07 -04:00
|
|
|
|
2020-12-13 17:26:53 -05:00
|
|
|
func consumeAppleLeaseString(s string) chan byte {
|
|
|
|
sch := consumeString(s)
|
|
|
|
uncommentedch := uncomment(sch)
|
|
|
|
return filterOutCharacters([]byte{'\r', '\v'}, uncommentedch)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestParserReadAppleDhcpdLeaseEntry(t *testing.T) {
|
|
|
|
test_1 := `{
|
|
|
|
ip_address=192.168.111.3
|
|
|
|
hw_address=1,0:c:56:3c:e7:22
|
|
|
|
identifier=1,0:c:56:3c:e7:22
|
|
|
|
lease=0x5fd78ae2
|
|
|
|
name=vagrant-2019
|
|
|
|
fake=field
|
|
|
|
}`
|
|
|
|
expected_1 := map[string]string{
|
|
|
|
"ipAddress": "192.168.111.3",
|
|
|
|
"hwAddress": "000c563ce722",
|
|
|
|
"id": "000c563ce722",
|
|
|
|
"lease": "0x5fd78ae2",
|
|
|
|
"name": "vagrant-2019",
|
|
|
|
}
|
|
|
|
expected_extra_1 := map[string]string{
|
|
|
|
"fake": "field",
|
|
|
|
}
|
|
|
|
|
|
|
|
result, err := readAppleDhcpdLeaseEntry(consumeAppleLeaseString(test_1))
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("error parsing entry: %v", err)
|
|
|
|
}
|
|
|
|
if result.ipAddress != expected_1["ipAddress"] {
|
|
|
|
t.Errorf("expected ipAddress %v, got %v", expected_1["ipAddress"], result.ipAddress)
|
|
|
|
}
|
|
|
|
if hex.EncodeToString(result.hwAddress) != expected_1["hwAddress"] {
|
|
|
|
t.Errorf("expected hwAddress %v, got %v", expected_1["hwAddress"], hex.EncodeToString(result.hwAddress))
|
|
|
|
}
|
|
|
|
if hex.EncodeToString(result.id) != expected_1["id"] {
|
|
|
|
t.Errorf("expected id %v, got %v", expected_1["id"], hex.EncodeToString(result.id))
|
|
|
|
}
|
|
|
|
if result.lease != expected_1["lease"] {
|
|
|
|
t.Errorf("expected lease %v, got %v", expected_1["lease"], result.lease)
|
|
|
|
}
|
|
|
|
if result.name != expected_1["name"] {
|
|
|
|
t.Errorf("expected name %v, got %v", expected_1["name"], result.name)
|
|
|
|
}
|
|
|
|
if result.extra["fake"] != expected_extra_1["fake"] {
|
|
|
|
t.Errorf("expected extra %v, got %v", expected_extra_1["fake"], result.extra["fake"])
|
|
|
|
}
|
2020-12-14 13:15:34 -05:00
|
|
|
|
|
|
|
test_2 := `{
|
|
|
|
ip_address=192.168.111.4
|
|
|
|
hw_address=1,0:c:56:3c:e7:23
|
|
|
|
identifier=1,0:c:56:3c:e7:23
|
|
|
|
}`
|
|
|
|
expected_2 := map[string]string{
|
|
|
|
"ipAddress": "192.168.111.4",
|
|
|
|
"hwAddress": "000c563ce723",
|
|
|
|
"id": "000c563ce723",
|
|
|
|
}
|
|
|
|
|
|
|
|
result, err = readAppleDhcpdLeaseEntry(consumeAppleLeaseString(test_2))
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("error parsing entry: %v", err)
|
|
|
|
}
|
|
|
|
if result.ipAddress != expected_2["ipAddress"] {
|
|
|
|
t.Errorf("expected ipAddress %v, got %v", expected_2["ipAddress"], result.ipAddress)
|
|
|
|
}
|
|
|
|
if hex.EncodeToString(result.hwAddress) != expected_2["hwAddress"] {
|
|
|
|
t.Errorf("expected hwAddress %v, got %v", expected_2["hwAddress"], hex.EncodeToString(result.hwAddress))
|
|
|
|
}
|
|
|
|
if hex.EncodeToString(result.id) != expected_2["id"] {
|
|
|
|
t.Errorf("expected id %v, got %v", expected_2["id"], hex.EncodeToString(result.id))
|
|
|
|
}
|
2020-12-13 17:26:53 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestParserReadAppleDhcpdLeases(t *testing.T) {
|
|
|
|
f, err := os.Open(filepath.Join("testdata", "apple-dhcpd-example.leases"))
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Unable to open dhcpd.leases sample: %s", err)
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
|
|
|
|
results, err := ReadAppleDhcpdLeaseEntries(f)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Error reading lease: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// some simple utilities
|
|
|
|
filter_ipAddr := func(ipAddress string, items []appleDhcpLeaseEntry) (result []appleDhcpLeaseEntry) {
|
|
|
|
for _, item := range items {
|
|
|
|
if item.ipAddress == ipAddress {
|
|
|
|
result = append(result, item)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
find_id := func(id string, items []appleDhcpLeaseEntry) *appleDhcpLeaseEntry {
|
|
|
|
for _, item := range items {
|
|
|
|
if id == hex.EncodeToString(item.id) {
|
|
|
|
return &item
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
find_hwAddr := func(hwAddr string, items []appleDhcpLeaseEntry) *appleDhcpLeaseEntry {
|
|
|
|
for _, item := range items {
|
|
|
|
if hwAddr == hex.EncodeToString(item.hwAddress) {
|
|
|
|
return &item
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// actual unit tests
|
|
|
|
test_1 := map[string]string{
|
|
|
|
"ipAddress": "127.0.0.19",
|
|
|
|
"id": "0dead099aabb",
|
|
|
|
"hwAddress": "0dead099aabb",
|
|
|
|
}
|
|
|
|
test_1_findings := filter_ipAddr(test_1["ipAddress"], results)
|
|
|
|
if len(test_1_findings) != 2 {
|
|
|
|
t.Errorf("expected %d matching entries, got %d", 2, len(test_1_findings))
|
|
|
|
} else {
|
|
|
|
res := find_hwAddr(test_1["hwAddress"], test_1_findings)
|
|
|
|
if res == nil {
|
|
|
|
t.Errorf("unable to find item with hwAddress %v", test_1["hwAddress"])
|
|
|
|
} else if hex.EncodeToString(res.id) != test_1["id"] {
|
|
|
|
t.Errorf("expected id %s, got %s", test_1["id"], hex.EncodeToString(res.id))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
test_2 := map[string]string{
|
|
|
|
"ipAddress": "127.0.0.19",
|
|
|
|
"id": "0dead0667788",
|
|
|
|
"hwAddress": "0dead0667788",
|
|
|
|
}
|
|
|
|
test_2_findings := filter_ipAddr(test_2["ipAddress"], results)
|
|
|
|
if len(test_2_findings) != 2 {
|
|
|
|
t.Errorf("expected %d matching entries, got %d", 2, len(test_2_findings))
|
|
|
|
} else {
|
|
|
|
res := find_hwAddr(test_2["hwAddress"], test_2_findings)
|
|
|
|
if res == nil {
|
|
|
|
t.Errorf("unable to find item with hwAddress %v", test_2["hwAddress"])
|
|
|
|
} else if hex.EncodeToString(res.id) != test_2["id"] {
|
|
|
|
t.Errorf("expected id %s, got %s", test_2["id"], hex.EncodeToString(res.id))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
test_3 := map[string]string{
|
|
|
|
"ipAddress": "127.0.0.17",
|
|
|
|
"id": "0dead0334455",
|
|
|
|
"hwAddress": "0dead0667788",
|
|
|
|
}
|
|
|
|
test_3_findings := filter_ipAddr(test_3["ipAddress"], results)
|
|
|
|
if len(test_3_findings) != 2 {
|
|
|
|
t.Errorf("expected %d matching entries, got %d", 2, len(test_3_findings))
|
|
|
|
} else {
|
|
|
|
res := find_id(test_3["id"], test_3_findings)
|
|
|
|
if res == nil {
|
|
|
|
t.Errorf("unable to find item with id %v", test_3["id"])
|
|
|
|
} else if hex.EncodeToString(res.hwAddress) != test_3["hwAddress"] {
|
|
|
|
t.Errorf("expected hardware address %s, got %s", test_3["hwAddress"], hex.EncodeToString(res.hwAddress))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
test_4 := map[string]string{
|
|
|
|
"ipAddress": "127.0.0.17",
|
|
|
|
"id": "0dead0001122",
|
|
|
|
"hwAddress": "0dead0667788",
|
|
|
|
}
|
|
|
|
test_4_findings := filter_ipAddr(test_4["ipAddress"], results)
|
|
|
|
if len(test_4_findings) != 2 {
|
|
|
|
t.Errorf("expected %d matching entries, got %d", 2, len(test_4_findings))
|
|
|
|
} else {
|
|
|
|
res := find_id(test_4["id"], test_4_findings)
|
|
|
|
if res == nil {
|
|
|
|
t.Errorf("unable to find item with id %v", test_4["id"])
|
|
|
|
} else if hex.EncodeToString(res.hwAddress) != test_4["hwAddress"] {
|
|
|
|
t.Errorf("expected hardware address %s, got %s", test_4["hwAddress"], hex.EncodeToString(res.hwAddress))
|
|
|
|
}
|
|
|
|
}
|
2020-12-14 13:15:34 -05:00
|
|
|
|
|
|
|
test_5 := map[string]string{
|
|
|
|
"ipAddress": "127.0.0.20",
|
|
|
|
"id": "0dead099aabc",
|
|
|
|
"hwAddress": "0dead099aabc",
|
|
|
|
}
|
|
|
|
test_5_findings := filter_ipAddr(test_5["ipAddress"], results)
|
|
|
|
if len(test_5_findings) != 1 {
|
|
|
|
t.Errorf("expected %d matching entries, got %d", 1, len(test_5_findings))
|
|
|
|
} else {
|
|
|
|
res := find_id(test_5["id"], test_5_findings)
|
|
|
|
if res == nil {
|
|
|
|
t.Errorf("unable to find item with id %v", test_5["id"])
|
|
|
|
} else if hex.EncodeToString(res.hwAddress) != test_5["hwAddress"] {
|
|
|
|
t.Errorf("expected hardware address %s, got %s", test_5["hwAddress"], hex.EncodeToString(res.hwAddress))
|
|
|
|
}
|
|
|
|
}
|
2020-12-13 17:26:53 -05:00
|
|
|
}
|
|
|
|
|
2020-06-04 06:07:07 -04:00
|
|
|
func TestParserTokenizeNetworkingConfig(t *testing.T) {
|
|
|
|
tests := []string{
|
|
|
|
"words words words",
|
|
|
|
"newlines\n\n\n\n\n\n\n\nnewlines\r\r\r\r\r\r\r\rnewlines\n\n\n\n",
|
|
|
|
" newline-less",
|
|
|
|
}
|
|
|
|
expects := [][]string{
|
|
|
|
[]string{"words", "words", "words"},
|
|
|
|
[]string{"newlines", "\n", "newlines", "\n", "newlines", "\n"},
|
|
|
|
[]string{"newline-less"},
|
|
|
|
}
|
|
|
|
|
|
|
|
for testnum := 0; testnum < len(tests); testnum += 1 {
|
|
|
|
inCh := consumeString(tests[testnum])
|
|
|
|
outCh := tokenizeNetworkingConfig(inCh)
|
|
|
|
result := collectIntoStringList(outCh)
|
|
|
|
|
|
|
|
expected := expects[testnum]
|
|
|
|
if len(result) != len(expected) {
|
|
|
|
t.Errorf("test %d expected %d items, got %d instead", 1+testnum, len(expected), len(result))
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
ok := true
|
|
|
|
for index := 0; index < len(expected); index += 1 {
|
|
|
|
if result[index] != expected[index] {
|
|
|
|
ok = false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !ok {
|
|
|
|
t.Errorf("test %d expected %#v, got %#v", 1+testnum, expects[testnum], result)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestParserSplitNetworkingConfig(t *testing.T) {
|
|
|
|
tests := []string{
|
|
|
|
"this is a story\n\n\nabout some newlines",
|
|
|
|
"\n\n\nthat can begin and end with newlines\n\n\n",
|
|
|
|
" in\n\n\nsome\ncases\nit\ncan\nend\nwith\nan\nempty\nstring\n\n\n\n",
|
|
|
|
"\n\n\nand\nbegin\nwith\nan\nempty\nstring ",
|
|
|
|
}
|
|
|
|
expects := [][]string{
|
|
|
|
[]string{"this is a story", "about some newlines"},
|
|
|
|
[]string{"that can begin and end with newlines"},
|
|
|
|
[]string{"in", "some", "cases", "it", "can", "end", "with", "an", "empty", "string"},
|
|
|
|
[]string{"and", "begin", "with", "an", "empty", "string"},
|
|
|
|
}
|
|
|
|
|
|
|
|
for testnum := 0; testnum < len(tests); testnum += 1 {
|
|
|
|
inCh := consumeString(tests[testnum])
|
|
|
|
stringCh := tokenizeNetworkingConfig(inCh)
|
|
|
|
outCh := splitNetworkingConfig(stringCh)
|
|
|
|
|
|
|
|
result := make([]string, 0)
|
|
|
|
for item := range outCh {
|
|
|
|
result = append(result, strings.Join(item, " "))
|
|
|
|
}
|
|
|
|
|
|
|
|
expected := expects[testnum]
|
|
|
|
if len(result) != len(expected) {
|
|
|
|
t.Errorf("test %d expected %d items, got %d instead", 1+testnum, len(expected), len(result))
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
ok := true
|
|
|
|
for index := 0; index < len(expected); index += 1 {
|
|
|
|
if result[index] != expected[index] {
|
|
|
|
ok = false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !ok {
|
|
|
|
t.Errorf("test %d expected %#v, got %#v", 1+testnum, expects[testnum], result)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestParserParseNetworkingConfigVersion(t *testing.T) {
|
|
|
|
success_tests := []string{"VERSION=4,2"}
|
|
|
|
failure_tests := []string{
|
|
|
|
"VERSION=1=2",
|
|
|
|
"VERSION=3,4,5",
|
|
|
|
"VERSION=a,b",
|
|
|
|
}
|
|
|
|
|
|
|
|
for testnum := 0; testnum < len(success_tests); testnum += 1 {
|
|
|
|
test := []string{success_tests[testnum]}
|
|
|
|
if _, err := networkingReadVersion(test); err != nil {
|
|
|
|
t.Errorf("success-test %d parsing failed: %v", 1+testnum, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for testnum := 0; testnum < len(success_tests); testnum += 1 {
|
|
|
|
test := []string{failure_tests[testnum]}
|
|
|
|
if _, err := networkingReadVersion(test); err == nil {
|
|
|
|
t.Errorf("failure-test %d should have failed", 1+testnum)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestParserParseNetworkingConfigEntries(t *testing.T) {
|
|
|
|
tests := []string{
|
|
|
|
"answer VNET_999_ANYTHING option",
|
|
|
|
"remove_answer VNET_123_ALSOANYTHING",
|
|
|
|
"add_nat_portfwd 24 udp 42 127.0.0.1 24",
|
|
|
|
"remove_nat_portfwd 42 tcp 2502",
|
|
|
|
"add_dhcp_mac_to_ip 57005 00:0d:0e:0a:0d:00 127.0.0.2",
|
|
|
|
"remove_dhcp_mac_to_ip 57005 00:0d:0e:0a:0d:00",
|
|
|
|
"add_bridge_mapping string 51",
|
|
|
|
"remove_bridge_mapping string",
|
|
|
|
"add_nat_prefix 57005 /24",
|
|
|
|
"remove_nat_prefix 57005 /31",
|
|
|
|
}
|
|
|
|
|
|
|
|
for testnum := 0; testnum < len(tests); testnum += 1 {
|
|
|
|
test := strings.Split(tests[testnum], " ")
|
|
|
|
parser := NetworkingParserByCommand(test[0])
|
|
|
|
if parser == nil {
|
|
|
|
t.Errorf("test %d unable to parse command: %#v", 1+testnum, test)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
operand_parser := *parser
|
|
|
|
|
|
|
|
_, err := operand_parser(test[1:])
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("test %d unable to parse command parameters %#v: %v", 1+testnum, test, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestParserReadNetworingConfig(t *testing.T) {
|
|
|
|
expected_answer_vnet_1 := map[string]string{
|
|
|
|
"DHCP": "yes",
|
|
|
|
"DHCP_CFG_HASH": "01F4CE0D79A1599698B6E5814CCB68058BB0ED5E",
|
|
|
|
"HOSTONLY_NETMASK": "255.255.255.0",
|
|
|
|
"HOSTONLY_SUBNET": "192.168.70.0",
|
|
|
|
"NAT": "no",
|
|
|
|
"VIRTUAL_ADAPTER": "yes",
|
|
|
|
}
|
|
|
|
|
|
|
|
f, err := os.Open(filepath.Join("testdata", "networking-example"))
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Unable to open networking-example sample: %v", err)
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
|
|
|
|
config, err := ReadNetworkingConfig(f)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("error parsing networking-example: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if vnet, ok := config.answer[1]; ok {
|
|
|
|
for ans_key := range expected_answer_vnet_1 {
|
|
|
|
result, ok := vnet[ans_key]
|
|
|
|
if !ok {
|
|
|
|
t.Errorf("unable to find key %s in VNET_%d answer", ans_key, 1)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if result != expected_answer_vnet_1[ans_key] {
|
|
|
|
t.Errorf("expected key %s for VNET_%d to be %v, got %v", ans_key, 1, expected_answer_vnet_1[ans_key], result)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
t.Errorf("unable to find VNET_%d answer", 1)
|
|
|
|
}
|
|
|
|
|
|
|
|
expected_answer_vnet_8 := map[string]string{
|
|
|
|
"DHCP": "yes",
|
|
|
|
"DHCP_CFG_HASH": "C30F14F65A0FE4B5DCC6C67497D7A8A33E5E538C",
|
|
|
|
"HOSTONLY_NETMASK": "255.255.255.0",
|
|
|
|
"HOSTONLY_SUBNET": "172.16.41.0",
|
|
|
|
"NAT": "yes",
|
|
|
|
"VIRTUAL_ADAPTER": "yes",
|
|
|
|
}
|
|
|
|
|
|
|
|
if vnet, ok := config.answer[8]; ok {
|
|
|
|
for ans_key := range expected_answer_vnet_8 {
|
|
|
|
result, ok := vnet[ans_key]
|
|
|
|
if !ok {
|
|
|
|
t.Errorf("unable to find key %s in VNET_%d answer", ans_key, 8)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if result != expected_answer_vnet_8[ans_key] {
|
|
|
|
t.Errorf("expected key %s for VNET_%d to be %v, got %v", ans_key, 8, expected_answer_vnet_8[ans_key], result)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
t.Errorf("unable to find VNET_%d answer", 8)
|
|
|
|
}
|
|
|
|
|
|
|
|
expected_nat_portfwd_8 := map[string]string{
|
|
|
|
"tcp/2200": "172.16.41.129:3389",
|
|
|
|
"tcp/2201": "172.16.41.129:3389",
|
|
|
|
"tcp/2222": "172.16.41.129:22",
|
|
|
|
"tcp/3389": "172.16.41.131:3389",
|
|
|
|
"tcp/55985": "172.16.41.129:5985",
|
|
|
|
"tcp/55986": "172.16.41.129:5986",
|
|
|
|
}
|
|
|
|
|
|
|
|
if vnet, ok := config.nat_portfwd[8-1]; ok {
|
|
|
|
for nat_key := range expected_nat_portfwd_8 {
|
|
|
|
result, ok := vnet[nat_key]
|
|
|
|
if !ok {
|
|
|
|
t.Errorf("unable to find key %s in VNET_%d nat_portfwd", nat_key, 8)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if result != expected_nat_portfwd_8[nat_key] {
|
|
|
|
t.Errorf("expected key %s for VNET_%d to be %v, got %v", nat_key, 8, expected_nat_portfwd_8[nat_key], result)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
t.Errorf("unable to find VNET_%d answer", 8-1)
|
|
|
|
}
|
|
|
|
}
|