1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
|
#!/bin/sh
# Copyright (C) 2014-2021 Internet Systems Consortium, Inc. ("ISC")
#
# This Source Code Form is subject to the terms of the Mozilla Public
# License, v. 2.0. If a copy of the MPL was not distributed with this
# file, You can obtain one at http://mozilla.org/MPL/2.0/.
# shellcheck disable=SC1091
# SC1091: Not following: ... was not specified as input (see shellcheck -x).
# shellcheck disable=SC2039
# SC2039: In POSIX sh, 'local' is undefined.
# Exit with error if commands exit with non-zero and if undefined variables are
# used.
set -eu
# Path to the temporary configuration file.
CFG_FILE="@abs_top_builddir@/src/bin/d2/tests/test_config.json"
# Path to the D2 log file.
LOG_FILE="@abs_top_builddir@/src/bin/d2/tests/test.log"
# D2 configuration to be stored in the configuration file.
CONFIG="{
\"DhcpDdns\":
{
\"ip-address\": \"127.0.0.1\",
\"port\": 53001,
\"tsig-keys\": [],
\"forward-ddns\" : {},
\"reverse-ddns\" : {},
\"loggers\": [
{
\"name\": \"kea-dhcp-ddns\",
\"output_options\": [
{
\"output\": \"$LOG_FILE\"
}
],
\"severity\": \"DEBUG\"
}
]
}
}"
# Invalid configuration (syntax error) to check that Kea can check syntax.
CONFIG_BAD_SYNTAX="{
\"DhcpDdns\":
{
\"ip-address\": \"127.0.0.1\",
\"port\": BOGUS,
\"tsig-keys\": [],
\"forward-ddns\" : {},
\"reverse-ddns\" : {},
\"loggers\": [
{
\"name\": \"kea-dhcp-ddns\",
\"output_options\": [
{
\"output\": \"$LOG_FILE\"
}
],
\"severity\": \"INFO\"
}
]
}
}"
# Invalid configuration (out of range port) to check that Kea can check syntax.
CONFIG_BAD_VALUE="{
\"DhcpDdns\":
{
\"ip-address\": \"127.0.0.1\",
\"port\": 80000,
\"tsig-keys\": [],
\"forward-ddns\" : {},
\"reverse-ddns\" : {},
\"loggers\": [
{
\"name\": \"kea-dhcp-ddns\",
\"output_options\": [
{
\"output\": \"$LOG_FILE\"
}
],
\"severity\": \"INFO\"
}
]
}
}"
# Invalid value configuration (invalid port) to check that D2
# gracefully handles reconfiguration errors.
CONFIG_INVALID="{
\"DhcpDdns\":
{
\"ip-address\": \"127.0.0.1\",
\"port\": BOGUS,
\"tsig-keys\": [],
\"forward-ddns\" : {},
\"reverse-ddns\" : {},
\"loggers\": [
{
\"name\": \"kea-dhcp-ddns\",
\"output_options\": [
{
\"output\": \"$LOG_FILE\"
}
],
\"severity\": \"INFO\"
}
]
}
}"
CONFIG_WITH_SECRET='
{
"DhcpDdns": {
"tsig-keys": [
{
"algorithm": "HMAC-MD5",
"name": "d2.md5.key",
"secret": "sensitivejdPJI5QxlpnfQ=="
}
],
"user-context": {
"password": "superadmin",
"secret": "superadmin",
"shared-info": {
"password": "superadmin",
"secret": "superadmin"
}
}
}
}
'
# Set the location of the executable.
bin="kea-dhcp-ddns"
bin_path="@abs_top_builddir@/src/bin/d2"
# Import common test library.
. "@abs_top_builddir@/src/lib/testutils/dhcp_test_lib.sh"
# This test verifies that syntax checking works properly. This function
# requires 3 parameters:
# test_name
# config - string with a content of the config (will be written to a file)
# expected_code - expected exit code returned by kea (0 - success, 1 - failure)
syntax_check_test() {
local test_name="${1}"
local config="${2}"
local expected_code="${3}"
# Log the start of the test and print test name.
test_start "${test_name}"
# Create correct configuration file.
create_config "${config}"
# Check it
printf "Running command %s.\n" "\"${bin_path}/${bin} -t ${CFG_FILE}\""
run_command \
"${bin_path}/${bin}" -t "${CFG_FILE}"
if [ "${EXIT_CODE}" -ne "${expected_code}" ]; then
printf 'ERROR: expected exit code %s, got %s\n' "${expected_code}" "${EXIT_CODE}"
clean_exit 1
fi
test_finish 0
}
# This test verifies that D2 can be reconfigured with a SIGHUP signal.
dynamic_reconfiguration_test() {
# Log the start of the test and print test name.
test_start "dhcp_ddns.dynamic_reconfiguration"
# Create new configuration file.
create_config "${CONFIG}"
# Instruct D2 to log to the specific file.
set_logger
# Start D2.
start_kea ${bin_path}/${bin}
# Wait up to 20s for D2 to start.
wait_for_kea 20
if [ "${_WAIT_FOR_KEA}" -eq 0 ]; then
printf "ERROR: timeout waiting for D2 to start.\n"
clean_exit 1
fi
# Check if it is still running. It could have terminated (e.g. as a result
# of configuration failure).
get_pid ${bin}
if [ "${_GET_PIDS_NUM}" -ne 1 ]; then
printf "ERROR: expected one D2 process to be started. Found %d processes\
started.\n" "${_GET_PIDS_NUM}"
clean_exit 1
fi
# Check in the log file, how many times server has been configured.
# It should be just once on startup.
get_reconfigs
if [ "${_GET_RECONFIGS}" -ne 1 ]; then
printf "ERROR: D2 hasn't been configured.\n"
clean_exit 1
else
printf "D2 successfully configured.\n"
fi
# Now use invalid configuration.
create_config "${CONFIG_INVALID}"
# Try to reconfigure by sending SIGHUP
send_signal 1 ${bin}
# Wait up to 10s for the D2Controller to log reload signal received.
wait_for_message 10 "DCTL_CFG_FILE_RELOAD_SIGNAL_RECVD" 1
if [ "${_WAIT_FOR_MESSAGE}" -eq 0 ]; then
printf "ERROR: D2 did report the reload signal receipt.\n"
clean_exit 1
fi
# After receiving SIGHUP the server should try to reconfigure itself.
# The configuration provided is invalid so it should result in
# reconfiguration failure but the server should still be running.
wait_for_message 10 "DCTL_CFG_FILE_RELOAD_ERROR" 1
if [ "${_WAIT_FOR_MESSAGE}" -eq 0 ]; then
printf "ERROR: D2 did not report reload error.\n"
clean_exit 1
fi
# Make sure the server is still operational.
get_pid ${bin}
if [ "${_GET_PIDS_NUM}" -ne 1 ]; then
printf "ERROR: D2 was killed when attempting reconfiguration.\n"
clean_exit 1
fi
# Restore the good configuration.
create_config "${CONFIG}"
# Reconfigure the server with SIGHUP.
send_signal 1 ${bin}
# There should be two occurrences of the DHCP4_CONFIG_COMPLETE messages.
# Wait for it up to 10s.
wait_for_message 10 "DCTL_CONFIG_COMPLETE" 2
# After receiving SIGHUP the server should get reconfigured and the
# reconfiguration should be noted in the log file. We should now
# have two configurations logged in the log file.
if [ "${_WAIT_FOR_MESSAGE}" -eq 0 ]; then
printf "ERROR: D2 hasn't been reconfigured.\n"
clean_exit 1
else
printf "D2 successfully reconfigured.\n"
fi
# Make sure the server is still operational.
get_pid ${bin}
if [ "${_GET_PIDS_NUM}" -ne 1 ]; then
printf "ERROR: D2 was killed when attempting reconfiguration.\n"
clean_exit 1
fi
# All ok. Shut down D2 and exit.
test_finish 0
}
# This test verifies that DHCPv4 server is shut down gracefully when it
# receives a SIGINT or SIGTERM signal.
shutdown_test() {
local test_name="${1}" # Test name
local signum="${2}" # Signal number
# Log the start of the test and print test name.
test_start "${test_name}"
# Create new configuration file.
create_config "${CONFIG}"
# Instruct D2 to log to the specific file.
set_logger
# Start D2.
start_kea ${bin_path}/${bin}
# Wait up to 20s for D2 to start.
wait_for_kea 20
if [ "${_WAIT_FOR_KEA}" -eq 0 ]; then
printf "ERROR: timeout waiting for D2 to start.\n"
clean_exit 1
fi
# Check if it is still running. It could have terminated (e.g. as a result
# of configuration failure).
get_pid ${bin}
if [ "${_GET_PIDS_NUM}" -ne 1 ]; then
printf "ERROR: expected one D2 process to be started. Found %d processes\
started.\n" "${_GET_PIDS_NUM}"
clean_exit 1
fi
# Check in the log file, how many times server has been configured.
# It should be just once on startup.
get_reconfigs
if [ "${_GET_RECONFIGS}" -ne 1 ]; then
printf "ERROR: server hasn't been configured.\n"
clean_exit 1
else
printf "Server successfully configured.\n"
fi
# Send signal to D2 (SIGTERM, SIGINT etc.)
send_signal "${signum}" "${bin}"
# Now wait for process to log that it is exiting.
wait_for_message 10 "DCTL_SHUTDOWN" 1
if [ "${_WAIT_FOR_MESSAGE}" -eq 0 ]; then
printf "ERROR: DHCP-DDNS did not log shutdown.\n"
clean_exit 1
fi
# Make sure the server is down.
wait_for_server_down 5 ${bin}
assert_eq 1 "${_WAIT_FOR_SERVER_DOWN}" \
"Expected wait_for_server_down return %d, returned %d"
test_finish 0
}
server_pid_file_test "${CONFIG}" DCTL_ALREADY_RUNNING
dynamic_reconfiguration_test
shutdown_test "dhcp-ddns.sigterm_test" 15
shutdown_test "dhcp-ddns.sigint_test" 2
version_test "dhcp-ddns.version"
logger_vars_test "dhcp-ddns.variables"
syntax_check_test "dhcp-ddns.syntax_check_success" "${CONFIG}" 0
syntax_check_test "dhcp-ddns.syntax_check_bad_syntax" "${CONFIG_BAD_SYNTAX}" 1
syntax_check_test "dhcp-ddns.syntax_check_bad_values" "${CONFIG_BAD_VALUE}" 1
password_redact_test "dhcp-ddns.password_redact_test" "${CONFIG_WITH_SECRET}" 0
|