2014-10-18 02:36:22 +04:00
|
|
|
/*
|
|
|
|
Copyright 2014 CoreOS, Inc.
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2014-03-18 20:00:41 +04:00
|
|
|
package initialize
|
2014-03-05 04:36:05 +04:00
|
|
|
|
|
|
|
import (
|
2014-11-04 01:43:29 +03:00
|
|
|
"fmt"
|
|
|
|
"strings"
|
2014-03-05 04:36:05 +04:00
|
|
|
"testing"
|
2014-06-06 04:40:53 +04:00
|
|
|
|
|
|
|
"github.com/coreos/coreos-cloudinit/system"
|
2014-03-05 04:36:05 +04:00
|
|
|
)
|
|
|
|
|
2014-11-04 01:43:29 +03:00
|
|
|
func TestCloudConfigInvalidKeys(t *testing.T) {
|
|
|
|
defer func() {
|
|
|
|
if r := recover(); r != nil {
|
|
|
|
t.Fatalf("panic while instantiating CloudConfig with nil keys: %v", r)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
for _, tt := range []struct {
|
|
|
|
contents string
|
|
|
|
}{
|
|
|
|
{"coreos:"},
|
|
|
|
{"ssh_authorized_keys:"},
|
|
|
|
{"ssh_authorized_keys:\n -"},
|
|
|
|
{"ssh_authorized_keys:\n - 0:"},
|
|
|
|
{"write_files:"},
|
|
|
|
{"write_files:\n -"},
|
|
|
|
{"write_files:\n - 0:"},
|
|
|
|
{"users:"},
|
|
|
|
{"users:\n -"},
|
|
|
|
{"users:\n - 0:"},
|
|
|
|
} {
|
|
|
|
_, err := NewCloudConfig(tt.contents)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("error instantiating CloudConfig with invalid keys: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCloudConfigUnknownKeys(t *testing.T) {
|
|
|
|
contents := `
|
|
|
|
coreos:
|
|
|
|
etcd:
|
|
|
|
discovery: "https://discovery.etcd.io/827c73219eeb2fa5530027c37bf18877"
|
|
|
|
coreos_unknown:
|
|
|
|
foo: "bar"
|
|
|
|
section_unknown:
|
|
|
|
dunno:
|
|
|
|
something
|
|
|
|
bare_unknown:
|
|
|
|
bar
|
|
|
|
write_files:
|
|
|
|
- content: fun
|
|
|
|
path: /var/party
|
|
|
|
file_unknown: nofun
|
|
|
|
users:
|
|
|
|
- name: fry
|
|
|
|
passwd: somehash
|
|
|
|
user_unknown: philip
|
|
|
|
hostname:
|
|
|
|
foo
|
|
|
|
`
|
|
|
|
cfg, err := NewCloudConfig(contents)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("error instantiating CloudConfig with unknown keys: %v", err)
|
|
|
|
}
|
|
|
|
if cfg.Hostname != "foo" {
|
|
|
|
t.Fatalf("hostname not correctly set when invalid keys are present")
|
|
|
|
}
|
|
|
|
if len(cfg.Coreos.Etcd) < 1 {
|
|
|
|
t.Fatalf("etcd section not correctly set when invalid keys are present")
|
|
|
|
}
|
|
|
|
if len(cfg.WriteFiles) < 1 || cfg.WriteFiles[0].Content != "fun" || cfg.WriteFiles[0].Path != "/var/party" {
|
|
|
|
t.Fatalf("write_files section not correctly set when invalid keys are present")
|
|
|
|
}
|
|
|
|
if len(cfg.Users) < 1 || cfg.Users[0].Name != "fry" || cfg.Users[0].PasswordHash != "somehash" {
|
|
|
|
t.Fatalf("users section not correctly set when invalid keys are present")
|
|
|
|
}
|
|
|
|
|
|
|
|
var warnings string
|
|
|
|
catchWarn := func(f string, v ...interface{}) {
|
|
|
|
warnings += fmt.Sprintf(f, v...)
|
|
|
|
}
|
|
|
|
|
|
|
|
warnOnUnrecognizedKeys(contents, catchWarn)
|
|
|
|
|
|
|
|
if !strings.Contains(warnings, "coreos_unknown") {
|
|
|
|
t.Errorf("warnings did not catch unrecognized coreos option coreos_unknown")
|
|
|
|
}
|
|
|
|
if !strings.Contains(warnings, "bare_unknown") {
|
|
|
|
t.Errorf("warnings did not catch unrecognized key bare_unknown")
|
|
|
|
}
|
|
|
|
if !strings.Contains(warnings, "section_unknown") {
|
|
|
|
t.Errorf("warnings did not catch unrecognized key section_unknown")
|
|
|
|
}
|
|
|
|
if !strings.Contains(warnings, "user_unknown") {
|
|
|
|
t.Errorf("warnings did not catch unrecognized user key user_unknown")
|
|
|
|
}
|
|
|
|
if !strings.Contains(warnings, "file_unknown") {
|
|
|
|
t.Errorf("warnings did not catch unrecognized file key file_unknown")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Assert that the parsing of a cloud config file "generally works"
|
|
|
|
func TestCloudConfigEmpty(t *testing.T) {
|
|
|
|
cfg, err := NewCloudConfig("")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Encountered unexpected error :%v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
keys := cfg.SSHAuthorizedKeys
|
|
|
|
if len(keys) != 0 {
|
|
|
|
t.Error("Parsed incorrect number of SSH keys")
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(cfg.WriteFiles) != 0 {
|
|
|
|
t.Error("Expected zero WriteFiles")
|
|
|
|
}
|
|
|
|
|
|
|
|
if cfg.Hostname != "" {
|
|
|
|
t.Errorf("Expected hostname to be empty, got '%s'", cfg.Hostname)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Assert that the parsing of a cloud config file "generally works"
|
|
|
|
func TestCloudConfig(t *testing.T) {
|
|
|
|
contents := `
|
|
|
|
coreos:
|
|
|
|
etcd:
|
|
|
|
discovery: "https://discovery.etcd.io/827c73219eeb2fa5530027c37bf18877"
|
|
|
|
update:
|
|
|
|
reboot-strategy: reboot
|
|
|
|
units:
|
|
|
|
- name: 50-eth0.network
|
|
|
|
runtime: yes
|
|
|
|
content: '[Match]
|
|
|
|
|
|
|
|
Name=eth47
|
|
|
|
|
|
|
|
|
|
|
|
[Network]
|
|
|
|
|
|
|
|
Address=10.209.171.177/19
|
|
|
|
|
|
|
|
'
|
|
|
|
oem:
|
|
|
|
id: rackspace
|
|
|
|
name: Rackspace Cloud Servers
|
|
|
|
version-id: 168.0.0
|
|
|
|
home-url: https://www.rackspace.com/cloud/servers/
|
|
|
|
bug-report-url: https://github.com/coreos/coreos-overlay
|
|
|
|
ssh_authorized_keys:
|
|
|
|
- foobar
|
|
|
|
- foobaz
|
|
|
|
write_files:
|
|
|
|
- content: |
|
|
|
|
penny
|
|
|
|
elroy
|
|
|
|
path: /etc/dogepack.conf
|
|
|
|
permissions: '0644'
|
|
|
|
owner: root:dogepack
|
|
|
|
hostname: trontastic
|
|
|
|
`
|
|
|
|
cfg, err := NewCloudConfig(contents)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Encountered unexpected error :%v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
keys := cfg.SSHAuthorizedKeys
|
|
|
|
if len(keys) != 2 {
|
|
|
|
t.Error("Parsed incorrect number of SSH keys")
|
|
|
|
} else if keys[0] != "foobar" {
|
|
|
|
t.Error("Expected first SSH key to be 'foobar'")
|
|
|
|
} else if keys[1] != "foobaz" {
|
|
|
|
t.Error("Expected first SSH key to be 'foobaz'")
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(cfg.WriteFiles) != 1 {
|
|
|
|
t.Error("Failed to parse correct number of write_files")
|
|
|
|
} else {
|
|
|
|
wf := cfg.WriteFiles[0]
|
|
|
|
if wf.Content != "penny\nelroy\n" {
|
|
|
|
t.Errorf("WriteFile has incorrect contents '%s'", wf.Content)
|
|
|
|
}
|
|
|
|
if wf.Encoding != "" {
|
|
|
|
t.Errorf("WriteFile has incorrect encoding %s", wf.Encoding)
|
|
|
|
}
|
|
|
|
if perm, _ := wf.Permissions(); perm != 0644 {
|
|
|
|
t.Errorf("WriteFile has incorrect permissions %s", perm)
|
|
|
|
}
|
|
|
|
if wf.Path != "/etc/dogepack.conf" {
|
|
|
|
t.Errorf("WriteFile has incorrect path %s", wf.Path)
|
|
|
|
}
|
|
|
|
if wf.Owner != "root:dogepack" {
|
|
|
|
t.Errorf("WriteFile has incorrect owner %s", wf.Owner)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(cfg.Coreos.Units) != 1 {
|
|
|
|
t.Error("Failed to parse correct number of units")
|
|
|
|
} else {
|
|
|
|
u := cfg.Coreos.Units[0]
|
|
|
|
expect := `[Match]
|
|
|
|
Name=eth47
|
|
|
|
|
|
|
|
[Network]
|
|
|
|
Address=10.209.171.177/19
|
|
|
|
`
|
|
|
|
if u.Content != expect {
|
|
|
|
t.Errorf("Unit has incorrect contents '%s'.\nExpected '%s'.", u.Content, expect)
|
|
|
|
}
|
|
|
|
if u.Runtime != true {
|
|
|
|
t.Errorf("Unit has incorrect runtime value")
|
|
|
|
}
|
|
|
|
if u.Name != "50-eth0.network" {
|
|
|
|
t.Errorf("Unit has incorrect name %s", u.Name)
|
|
|
|
}
|
|
|
|
if u.Type() != "network" {
|
|
|
|
t.Errorf("Unit has incorrect type '%s'", u.Type())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if cfg.Coreos.OEM.ID != "rackspace" {
|
|
|
|
t.Errorf("Failed parsing coreos.oem. Expected ID 'rackspace', got %q.", cfg.Coreos.OEM.ID)
|
|
|
|
}
|
|
|
|
|
|
|
|
if cfg.Hostname != "trontastic" {
|
|
|
|
t.Errorf("Failed to parse hostname")
|
|
|
|
}
|
|
|
|
if cfg.Coreos.Update["reboot-strategy"] != "reboot" {
|
|
|
|
t.Errorf("Failed to parse locksmith strategy")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Assert that our interface conversion doesn't panic
|
|
|
|
func TestCloudConfigKeysNotList(t *testing.T) {
|
|
|
|
contents := `
|
|
|
|
ssh_authorized_keys:
|
|
|
|
- foo: bar
|
|
|
|
`
|
|
|
|
cfg, err := NewCloudConfig(contents)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Encountered unexpected error: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
keys := cfg.SSHAuthorizedKeys
|
|
|
|
if len(keys) != 0 {
|
|
|
|
t.Error("Parsed incorrect number of SSH keys")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCloudConfigSerializationHeader(t *testing.T) {
|
|
|
|
cfg, _ := NewCloudConfig("")
|
|
|
|
contents := cfg.String()
|
|
|
|
header := strings.SplitN(contents, "\n", 2)[0]
|
|
|
|
if header != "#cloud-config" {
|
|
|
|
t.Fatalf("Serialized config did not have expected header")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TestDropInIgnored asserts that users are unable to set DropIn=True on units
|
|
|
|
func TestDropInIgnored(t *testing.T) {
|
|
|
|
contents := `
|
|
|
|
coreos:
|
|
|
|
units:
|
|
|
|
- name: test
|
|
|
|
dropin: true
|
|
|
|
`
|
|
|
|
cfg, err := NewCloudConfig(contents)
|
|
|
|
if err != nil || len(cfg.Coreos.Units) != 1 {
|
|
|
|
t.Fatalf("Encountered unexpected error: %v", err)
|
|
|
|
}
|
|
|
|
if len(cfg.Coreos.Units) != 1 || cfg.Coreos.Units[0].Name != "test" {
|
|
|
|
t.Fatalf("Expected 1 unit, but got %d: %v", len(cfg.Coreos.Units), cfg.Coreos.Units)
|
|
|
|
}
|
|
|
|
if cfg.Coreos.Units[0].DropIn {
|
|
|
|
t.Errorf("dropin option on unit in cloud-config was not ignored!")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCloudConfigUsers(t *testing.T) {
|
|
|
|
contents := `
|
|
|
|
users:
|
|
|
|
- name: elroy
|
|
|
|
passwd: somehash
|
|
|
|
ssh-authorized-keys:
|
|
|
|
- somekey
|
|
|
|
gecos: arbitrary comment
|
|
|
|
homedir: /home/place
|
|
|
|
no-create-home: yes
|
|
|
|
primary-group: things
|
|
|
|
groups:
|
|
|
|
- ping
|
|
|
|
- pong
|
|
|
|
no-user-group: true
|
|
|
|
system: y
|
|
|
|
no-log-init: True
|
|
|
|
`
|
|
|
|
cfg, err := NewCloudConfig(contents)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Encountered unexpected error: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(cfg.Users) != 1 {
|
|
|
|
t.Fatalf("Parsed %d users, expected 1", len(cfg.Users))
|
|
|
|
}
|
|
|
|
|
|
|
|
user := cfg.Users[0]
|
|
|
|
|
|
|
|
if user.Name != "elroy" {
|
|
|
|
t.Errorf("User name is %q, expected 'elroy'", user.Name)
|
|
|
|
}
|
|
|
|
|
|
|
|
if user.PasswordHash != "somehash" {
|
|
|
|
t.Errorf("User passwd is %q, expected 'somehash'", user.PasswordHash)
|
|
|
|
}
|
|
|
|
|
|
|
|
if keys := user.SSHAuthorizedKeys; len(keys) != 1 {
|
|
|
|
t.Errorf("Parsed %d ssh keys, expected 1", len(keys))
|
|
|
|
} else {
|
|
|
|
key := user.SSHAuthorizedKeys[0]
|
|
|
|
if key != "somekey" {
|
|
|
|
t.Errorf("User SSH key is %q, expected 'somekey'", key)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if user.GECOS != "arbitrary comment" {
|
|
|
|
t.Errorf("Failed to parse gecos field, got %q", user.GECOS)
|
|
|
|
}
|
|
|
|
|
|
|
|
if user.Homedir != "/home/place" {
|
|
|
|
t.Errorf("Failed to parse homedir field, got %q", user.Homedir)
|
|
|
|
}
|
|
|
|
|
|
|
|
if !user.NoCreateHome {
|
|
|
|
t.Errorf("Failed to parse no-create-home field")
|
|
|
|
}
|
|
|
|
|
|
|
|
if user.PrimaryGroup != "things" {
|
|
|
|
t.Errorf("Failed to parse primary-group field, got %q", user.PrimaryGroup)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(user.Groups) != 2 {
|
|
|
|
t.Errorf("Failed to parse 2 goups, got %d", len(user.Groups))
|
|
|
|
} else {
|
|
|
|
if user.Groups[0] != "ping" {
|
|
|
|
t.Errorf("First group was %q, not expected value 'ping'", user.Groups[0])
|
|
|
|
}
|
|
|
|
if user.Groups[1] != "pong" {
|
|
|
|
t.Errorf("First group was %q, not expected value 'pong'", user.Groups[1])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if !user.NoUserGroup {
|
|
|
|
t.Errorf("Failed to parse no-user-group field")
|
|
|
|
}
|
|
|
|
|
|
|
|
if !user.System {
|
|
|
|
t.Errorf("Failed to parse system field")
|
|
|
|
}
|
|
|
|
|
|
|
|
if !user.NoLogInit {
|
|
|
|
t.Errorf("Failed to parse no-log-init field")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-06 04:40:53 +04:00
|
|
|
type TestUnitManager struct {
|
|
|
|
placed []string
|
|
|
|
enabled []string
|
|
|
|
masked []string
|
|
|
|
unmasked []string
|
2014-12-02 23:46:35 +03:00
|
|
|
commands []UnitAction
|
2014-06-06 04:40:53 +04:00
|
|
|
reload bool
|
|
|
|
}
|
|
|
|
|
2014-12-02 23:46:35 +03:00
|
|
|
type UnitAction struct {
|
|
|
|
unit string
|
|
|
|
command string
|
|
|
|
}
|
|
|
|
|
2014-06-06 04:40:53 +04:00
|
|
|
func (tum *TestUnitManager) PlaceUnit(unit *system.Unit, dst string) error {
|
|
|
|
tum.placed = append(tum.placed, unit.Name)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
func (tum *TestUnitManager) EnableUnitFile(unit string, runtime bool) error {
|
|
|
|
tum.enabled = append(tum.enabled, unit)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
func (tum *TestUnitManager) RunUnitCommand(command, unit string) (string, error) {
|
2014-12-02 23:46:35 +03:00
|
|
|
tum.commands = append(tum.commands, UnitAction{unit, command})
|
2014-06-06 04:40:53 +04:00
|
|
|
return "", nil
|
|
|
|
}
|
|
|
|
func (tum *TestUnitManager) DaemonReload() error {
|
|
|
|
tum.reload = true
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
func (tum *TestUnitManager) MaskUnit(unit *system.Unit) error {
|
|
|
|
tum.masked = append(tum.masked, unit.Name)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
func (tum *TestUnitManager) UnmaskUnit(unit *system.Unit) error {
|
|
|
|
tum.unmasked = append(tum.unmasked, unit.Name)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestProcessUnits(t *testing.T) {
|
|
|
|
tum := &TestUnitManager{}
|
|
|
|
units := []system.Unit{
|
2014-11-04 01:43:29 +03:00
|
|
|
system.Unit{
|
2014-06-06 04:40:53 +04:00
|
|
|
Name: "foo",
|
|
|
|
Mask: true,
|
2014-11-04 01:43:29 +03:00
|
|
|
},
|
2014-06-06 04:40:53 +04:00
|
|
|
}
|
|
|
|
if err := processUnits(units, "", tum); err != nil {
|
|
|
|
t.Fatalf("unexpected error calling processUnits: %v", err)
|
|
|
|
}
|
|
|
|
if len(tum.masked) != 1 || tum.masked[0] != "foo" {
|
|
|
|
t.Errorf("expected foo to be masked, but found %v", tum.masked)
|
|
|
|
}
|
|
|
|
|
|
|
|
tum = &TestUnitManager{}
|
|
|
|
units = []system.Unit{
|
2014-11-04 01:43:29 +03:00
|
|
|
system.Unit{
|
2014-06-06 04:40:53 +04:00
|
|
|
Name: "bar.network",
|
2014-11-04 01:43:29 +03:00
|
|
|
},
|
2014-06-06 04:40:53 +04:00
|
|
|
}
|
|
|
|
if err := processUnits(units, "", tum); err != nil {
|
|
|
|
t.Fatalf("unexpected error calling processUnits: %v", err)
|
|
|
|
}
|
2014-12-02 23:46:35 +03:00
|
|
|
if len(tum.commands) != 1 || (tum.commands[0] != UnitAction{"systemd-networkd.service", "restart"}) {
|
2014-06-06 04:40:53 +04:00
|
|
|
t.Errorf("expected systemd-networkd.service to be reloaded!")
|
|
|
|
}
|
|
|
|
|
|
|
|
tum = &TestUnitManager{}
|
|
|
|
units = []system.Unit{
|
2014-11-04 01:43:29 +03:00
|
|
|
system.Unit{
|
2014-06-06 04:40:53 +04:00
|
|
|
Name: "baz.service",
|
|
|
|
Content: "[Service]\nExecStart=/bin/true",
|
2014-11-04 01:43:29 +03:00
|
|
|
},
|
2014-06-06 04:40:53 +04:00
|
|
|
}
|
|
|
|
if err := processUnits(units, "", tum); err != nil {
|
|
|
|
t.Fatalf("unexpected error calling processUnits: %v", err)
|
|
|
|
}
|
|
|
|
if len(tum.placed) != 1 || tum.placed[0] != "baz.service" {
|
|
|
|
t.Fatalf("expected baz.service to be written, but got %v", tum.placed)
|
|
|
|
}
|
|
|
|
|
|
|
|
tum = &TestUnitManager{}
|
|
|
|
units = []system.Unit{
|
2014-11-04 01:43:29 +03:00
|
|
|
system.Unit{
|
2014-06-06 04:40:53 +04:00
|
|
|
Name: "locksmithd.service",
|
|
|
|
Runtime: true,
|
2014-11-04 01:43:29 +03:00
|
|
|
},
|
2014-06-06 04:40:53 +04:00
|
|
|
}
|
|
|
|
if err := processUnits(units, "", tum); err != nil {
|
|
|
|
t.Fatalf("unexpected error calling processUnits: %v", err)
|
|
|
|
}
|
|
|
|
if len(tum.unmasked) != 1 || tum.unmasked[0] != "locksmithd.service" {
|
|
|
|
t.Fatalf("expected locksmithd.service to be unmasked, but got %v", tum.unmasked)
|
|
|
|
}
|
|
|
|
|
|
|
|
tum = &TestUnitManager{}
|
|
|
|
units = []system.Unit{
|
2014-11-04 01:43:29 +03:00
|
|
|
system.Unit{
|
2014-06-06 04:40:53 +04:00
|
|
|
Name: "woof",
|
|
|
|
Enable: true,
|
2014-11-04 01:43:29 +03:00
|
|
|
},
|
2014-06-06 04:40:53 +04:00
|
|
|
}
|
|
|
|
if err := processUnits(units, "", tum); err != nil {
|
|
|
|
t.Fatalf("unexpected error calling processUnits: %v", err)
|
|
|
|
}
|
|
|
|
if len(tum.enabled) != 1 || tum.enabled[0] != "woof" {
|
|
|
|
t.Fatalf("expected woof to be enabled, but got %v", tum.enabled)
|
|
|
|
}
|
|
|
|
}
|