| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212 | 
							- package nebula
 
- import (
 
- 	"io/ioutil"
 
- 	"os"
 
- 	"path/filepath"
 
- 	"testing"
 
- 	"time"
 
- 	"github.com/stretchr/testify/assert"
 
- )
 
- func TestConfig_Load(t *testing.T) {
 
- 	dir, err := ioutil.TempDir("", "config-test")
 
- 	// invalid yaml
 
- 	c := NewConfig()
 
- 	ioutil.WriteFile(filepath.Join(dir, "01.yaml"), []byte(" invalid yaml"), 0644)
 
- 	assert.EqualError(t, c.Load(dir), "yaml: unmarshal errors:\n  line 1: cannot unmarshal !!str `invalid...` into map[interface {}]interface {}")
 
- 	// simple multi config merge
 
- 	c = NewConfig()
 
- 	os.RemoveAll(dir)
 
- 	os.Mkdir(dir, 0755)
 
- 	assert.Nil(t, err)
 
- 	ioutil.WriteFile(filepath.Join(dir, "01.yaml"), []byte("outer:\n  inner: hi"), 0644)
 
- 	ioutil.WriteFile(filepath.Join(dir, "02.yml"), []byte("outer:\n  inner: override\nnew: hi"), 0644)
 
- 	assert.Nil(t, c.Load(dir))
 
- 	expected := map[interface{}]interface{}{
 
- 		"outer": map[interface{}]interface{}{
 
- 			"inner": "override",
 
- 		},
 
- 		"new": "hi",
 
- 	}
 
- 	assert.Equal(t, expected, c.Settings)
 
- 	//TODO: test symlinked file
 
- 	//TODO: test symlinked directory
 
- }
 
- func TestConfig_Get(t *testing.T) {
 
- 	// test simple type
 
- 	c := NewConfig()
 
- 	c.Settings["firewall"] = map[interface{}]interface{}{"outbound": "hi"}
 
- 	assert.Equal(t, "hi", c.Get("firewall.outbound"))
 
- 	// test complex type
 
- 	inner := []map[interface{}]interface{}{{"port": "1", "code": "2"}}
 
- 	c.Settings["firewall"] = map[interface{}]interface{}{"outbound": inner}
 
- 	assert.EqualValues(t, inner, c.Get("firewall.outbound"))
 
- 	// test missing
 
- 	assert.Nil(t, c.Get("firewall.nope"))
 
- }
 
- func TestConfig_GetStringSlice(t *testing.T) {
 
- 	c := NewConfig()
 
- 	c.Settings["slice"] = []interface{}{"one", "two"}
 
- 	assert.Equal(t, []string{"one", "two"}, c.GetStringSlice("slice", []string{}))
 
- }
 
- func TestConfig_GetBool(t *testing.T) {
 
- 	c := NewConfig()
 
- 	c.Settings["bool"] = true
 
- 	assert.Equal(t, true, c.GetBool("bool", false))
 
- 	c.Settings["bool"] = "true"
 
- 	assert.Equal(t, true, c.GetBool("bool", false))
 
- 	c.Settings["bool"] = false
 
- 	assert.Equal(t, false, c.GetBool("bool", true))
 
- 	c.Settings["bool"] = "false"
 
- 	assert.Equal(t, false, c.GetBool("bool", true))
 
- 	c.Settings["bool"] = "Y"
 
- 	assert.Equal(t, true, c.GetBool("bool", false))
 
- 	c.Settings["bool"] = "yEs"
 
- 	assert.Equal(t, true, c.GetBool("bool", false))
 
- 	c.Settings["bool"] = "N"
 
- 	assert.Equal(t, false, c.GetBool("bool", true))
 
- 	c.Settings["bool"] = "nO"
 
- 	assert.Equal(t, false, c.GetBool("bool", true))
 
- }
 
- func TestConfig_GetAllowList(t *testing.T) {
 
- 	c := NewConfig()
 
- 	c.Settings["allowlist"] = map[interface{}]interface{}{
 
- 		"192.168.0.0": true,
 
- 	}
 
- 	r, err := c.GetAllowList("allowlist", false)
 
- 	assert.EqualError(t, err, "config `allowlist` has invalid CIDR: 192.168.0.0")
 
- 	assert.Nil(t, r)
 
- 	c.Settings["allowlist"] = map[interface{}]interface{}{
 
- 		"192.168.0.0/16": "abc",
 
- 	}
 
- 	r, err = c.GetAllowList("allowlist", false)
 
- 	assert.EqualError(t, err, "config `allowlist` has invalid value (type string): abc")
 
- 	c.Settings["allowlist"] = map[interface{}]interface{}{
 
- 		"192.168.0.0/16": true,
 
- 		"10.0.0.0/8":     false,
 
- 	}
 
- 	r, err = c.GetAllowList("allowlist", false)
 
- 	assert.EqualError(t, err, "config `allowlist` contains both true and false rules, but no default set for 0.0.0.0/0")
 
- 	c.Settings["allowlist"] = map[interface{}]interface{}{
 
- 		"0.0.0.0/0":     true,
 
- 		"10.0.0.0/8":    false,
 
- 		"10.42.42.0/24": true,
 
- 	}
 
- 	r, err = c.GetAllowList("allowlist", false)
 
- 	if assert.NoError(t, err) {
 
- 		assert.NotNil(t, r)
 
- 	}
 
- 	// Test interface names
 
- 	c.Settings["allowlist"] = map[interface{}]interface{}{
 
- 		"interfaces": map[interface{}]interface{}{
 
- 			`docker.*`: false,
 
- 		},
 
- 	}
 
- 	r, err = c.GetAllowList("allowlist", false)
 
- 	assert.EqualError(t, err, "config `allowlist` does not support `interfaces`")
 
- 	c.Settings["allowlist"] = map[interface{}]interface{}{
 
- 		"interfaces": map[interface{}]interface{}{
 
- 			`docker.*`: "foo",
 
- 		},
 
- 	}
 
- 	r, err = c.GetAllowList("allowlist", true)
 
- 	assert.EqualError(t, err, "config `allowlist.interfaces` has invalid value (type string): foo")
 
- 	c.Settings["allowlist"] = map[interface{}]interface{}{
 
- 		"interfaces": map[interface{}]interface{}{
 
- 			`docker.*`: false,
 
- 			`eth.*`:    true,
 
- 		},
 
- 	}
 
- 	r, err = c.GetAllowList("allowlist", true)
 
- 	assert.EqualError(t, err, "config `allowlist.interfaces` values must all be the same true/false value")
 
- 	c.Settings["allowlist"] = map[interface{}]interface{}{
 
- 		"interfaces": map[interface{}]interface{}{
 
- 			`docker.*`: false,
 
- 		},
 
- 	}
 
- 	r, err = c.GetAllowList("allowlist", true)
 
- 	if assert.NoError(t, err) {
 
- 		assert.NotNil(t, r)
 
- 	}
 
- }
 
- func TestConfig_HasChanged(t *testing.T) {
 
- 	// No reload has occurred, return false
 
- 	c := NewConfig()
 
- 	c.Settings["test"] = "hi"
 
- 	assert.False(t, c.HasChanged(""))
 
- 	// Test key change
 
- 	c = NewConfig()
 
- 	c.Settings["test"] = "hi"
 
- 	c.oldSettings = map[interface{}]interface{}{"test": "no"}
 
- 	assert.True(t, c.HasChanged("test"))
 
- 	assert.True(t, c.HasChanged(""))
 
- 	// No key change
 
- 	c = NewConfig()
 
- 	c.Settings["test"] = "hi"
 
- 	c.oldSettings = map[interface{}]interface{}{"test": "hi"}
 
- 	assert.False(t, c.HasChanged("test"))
 
- 	assert.False(t, c.HasChanged(""))
 
- }
 
- func TestConfig_ReloadConfig(t *testing.T) {
 
- 	done := make(chan bool, 1)
 
- 	dir, err := ioutil.TempDir("", "config-test")
 
- 	assert.Nil(t, err)
 
- 	ioutil.WriteFile(filepath.Join(dir, "01.yaml"), []byte("outer:\n  inner: hi"), 0644)
 
- 	c := NewConfig()
 
- 	assert.Nil(t, c.Load(dir))
 
- 	assert.False(t, c.HasChanged("outer.inner"))
 
- 	assert.False(t, c.HasChanged("outer"))
 
- 	assert.False(t, c.HasChanged(""))
 
- 	ioutil.WriteFile(filepath.Join(dir, "01.yaml"), []byte("outer:\n  inner: ho"), 0644)
 
- 	c.RegisterReloadCallback(func(c *Config) {
 
- 		done <- true
 
- 	})
 
- 	c.ReloadConfig()
 
- 	assert.True(t, c.HasChanged("outer.inner"))
 
- 	assert.True(t, c.HasChanged("outer"))
 
- 	assert.True(t, c.HasChanged(""))
 
- 	// Make sure we call the callbacks
 
- 	select {
 
- 	case <-done:
 
- 	case <-time.After(1 * time.Second):
 
- 		panic("timeout")
 
- 	}
 
- }
 
 
  |