Compare commits
No commits in common. 'i8c' and 'c9' have entirely different histories.
@ -1,2 +1,3 @@
|
|||||||
4661e5df181a9761b73caeaef2f2ab755bbe086a SOURCES/chrony-4.5.tar.gz
|
bc7884eb4fde69478a00faee3d42092d426d57c1 SOURCES/chrony-4.3.tar.gz
|
||||||
e021461c23fe4e5c46fd53c449587d8f6cc217ae SOURCES/clknetsim-5d1dc0.tar.gz
|
9c453ae65e5c1a6983cd1121410faf1ffd2d9092 SOURCES/clknetsim-f00531.tar.gz
|
||||||
|
1395afa521d2e3302a31083edcf568bbc036aafc SOURCES/gpgkey-8F375C7E8D0EE125A3D3BD51537E2B76F7680DAC.asc
|
||||||
|
@ -1,2 +1,3 @@
|
|||||||
SOURCES/chrony-4.5.tar.gz
|
SOURCES/chrony-4.3.tar.gz
|
||||||
SOURCES/clknetsim-5d1dc0.tar.gz
|
SOURCES/clknetsim-f00531.tar.gz
|
||||||
|
SOURCES/gpgkey-8F375C7E8D0EE125A3D3BD51537E2B76F7680DAC.asc
|
||||||
|
@ -0,0 +1,16 @@
|
|||||||
|
-----BEGIN PGP SIGNATURE-----
|
||||||
|
|
||||||
|
iQIzBAABCAAdFiEEjzdcfo0O4SWj071RU34rdvdoDawFAmMPLJAACgkQU34rdvdo
|
||||||
|
DaxDKRAAh5wfl990Q6sTPxXI92GegZYIGUxJDlCkJtemoI98g+DQbuCJ46AXsAn/
|
||||||
|
CIBTbPU3Brvq2KR1nDze/G/YOXkaqoFyaJD00H73qBI7MOMiSS4KbMQ26xLNrnHL
|
||||||
|
MCHrgZs+MHhyo6IEpesvr7F/+qyGHZifFlHT+HtCM+SBU1qooYUyQAdnhyK0rb16
|
||||||
|
j7/Jc5A28jROZB4lcRQyvB085whPj299FsB/0wJW5RjwA5tcpPH0sTozain3vvlo
|
||||||
|
64BAJXcQsyRsilcaPFlkY5zPgFiAuaEJnfTe/uMdfDO/V/g6wADt64+HhaxNPO+z
|
||||||
|
p3vzEGpio4Oi1HyYiXpDx9bMM1RLTpmKt9p1V5Y98Fn5Ymx6I7yAe1qwvA7T8eoC
|
||||||
|
hK8C27jPytiOgaWSYqPYb0WaHY3JZZpFzdtr0bAPSkEzL4EwrxVmbgTnkuzk2hxk
|
||||||
|
6MiIuDLUd9Zl1oroqv+rTd0XA8lXUcoyFhqtsMXHWdAC3yzteaPcJKzv7l9DT6xV
|
||||||
|
YadKrSBkzob9jRWRngY3FMKjTvcwnxLE8dfsNlsDNGyLNtTEOJ/QYgh6muOHh80L
|
||||||
|
MAayI8hSWPTR/3IXKlathjLIeilsrFthIZcrPq520FoS4A7E3A80vR3uKOqAIDwh
|
||||||
|
Y+6ASvEkCHAUneJqlLihqglYTNJlFnVhGw9/LV85JsmRsCZ0+j8=
|
||||||
|
=2xMP
|
||||||
|
-----END PGP SIGNATURE-----
|
@ -1,56 +0,0 @@
|
|||||||
commit 8eb5dd54efd13aa0209aea38dbad2a7904377f75
|
|
||||||
Author: Miroslav Lichvar <mlichvar@redhat.com>
|
|
||||||
Date: Tue Sep 17 13:00:43 2024 +0200
|
|
||||||
|
|
||||||
configure: enable AES-CMAC using gnutls
|
|
||||||
|
|
||||||
Allow gnutls to be used for AES-CMAC when nettle doesn't support it
|
|
||||||
without switching also hashing.
|
|
||||||
|
|
||||||
diff --git a/configure b/configure
|
|
||||||
index eefe5de8..0fb3aa38 100755
|
|
||||||
--- a/configure
|
|
||||||
+++ b/configure
|
|
||||||
@@ -937,14 +937,26 @@ if [ $feat_sechash = "1" ] && [ "x$HASH_LINK" = "x" ] && [ $try_gnutls = "1" ];
|
|
||||||
HASH_LINK="$test_link"
|
|
||||||
MYCPPFLAGS="$MYCPPFLAGS $test_cflags"
|
|
||||||
add_def FEAT_SECHASH
|
|
||||||
+ fi
|
|
||||||
+fi
|
|
||||||
|
|
||||||
- if test_code 'CMAC in gnutls' 'gnutls/crypto.h' "$test_cflags" "$test_link" \
|
|
||||||
- 'return gnutls_hmac_init((void *)1, GNUTLS_MAC_AES_CMAC_128, (void *)2, 0);'
|
|
||||||
- then
|
|
||||||
- add_def HAVE_CMAC
|
|
||||||
- EXTRA_OBJECTS="$EXTRA_OBJECTS cmac_gnutls.o"
|
|
||||||
- EXTRA_CLI_OBJECTS="$EXTRA_CLI_OBJECTS cmac_gnutls.o"
|
|
||||||
- fi
|
|
||||||
+if [ $feat_sechash = "1" ] && [ $try_gnutls = "1" ] &&
|
|
||||||
+ ! grep '#define HAVE_CMAC' config.h > /dev/null; then
|
|
||||||
+ if [ "$HASH_OBJ" = "hash_gnutls.o" ]; then
|
|
||||||
+ test_cflags=""
|
|
||||||
+ test_link=""
|
|
||||||
+ else
|
|
||||||
+ test_cflags="`pkg_config --cflags gnutls`"
|
|
||||||
+ test_link="`pkg_config --libs gnutls`"
|
|
||||||
+ fi
|
|
||||||
+ if test_code 'CMAC in gnutls' 'gnutls/crypto.h' "$test_cflags" "$test_link" \
|
|
||||||
+ 'return gnutls_hmac_init((void *)1, GNUTLS_MAC_AES_CMAC_128, (void *)2, 0);'
|
|
||||||
+ then
|
|
||||||
+ add_def HAVE_CMAC
|
|
||||||
+ EXTRA_OBJECTS="$EXTRA_OBJECTS cmac_gnutls.o"
|
|
||||||
+ EXTRA_CLI_OBJECTS="$EXTRA_CLI_OBJECTS cmac_gnutls.o"
|
|
||||||
+ LIBS="$LIBS $test_link"
|
|
||||||
+ MYCPPFLAGS="$MYCPPFLAGS $test_cflags"
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
@@ -978,7 +990,7 @@ EXTRA_CLI_OBJECTS="$EXTRA_CLI_OBJECTS $HASH_OBJ"
|
|
||||||
LIBS="$LIBS $HASH_LINK"
|
|
||||||
|
|
||||||
if [ $feat_ntp = "1" ] && [ $feat_nts = "1" ] && [ $try_gnutls = "1" ]; then
|
|
||||||
- if [ "$HASH_OBJ" = "hash_gnutls.o" ]; then
|
|
||||||
+ if echo "$HASH_OBJ $EXTRA_OBJECTS" | grep "_gnutls\.o" > /dev/null; then
|
|
||||||
test_cflags=""
|
|
||||||
test_link=""
|
|
||||||
else
|
|
@ -1,38 +0,0 @@
|
|||||||
diff -up chrony-4.1/examples/chrony.conf.example2.defconfig chrony-4.1/examples/chrony.conf.example2
|
|
||||||
--- chrony-4.1/examples/chrony.conf.example2.defconfig 2021-05-12 13:06:15.000000000 +0200
|
|
||||||
+++ chrony-4.1/examples/chrony.conf.example2 2019-05-10 12:22:57.000000000 +0200
|
|
||||||
@@ -1,5 +1,5 @@
|
|
||||||
# Use public servers from the pool.ntp.org project.
|
|
||||||
-# Please consider joining the pool (https://www.pool.ntp.org/join.html).
|
|
||||||
+# Please consider joining the pool (http://www.pool.ntp.org/join.html).
|
|
||||||
pool pool.ntp.org iburst
|
|
||||||
|
|
||||||
# Record the rate at which the system clock gains/losses time.
|
|
||||||
@@ -25,18 +25,9 @@ rtcsync
|
|
||||||
# Serve time even if not synchronized to a time source.
|
|
||||||
#local stratum 10
|
|
||||||
|
|
||||||
-# Require authentication (nts or key option) for all NTP sources.
|
|
||||||
-#authselectmode require
|
|
||||||
-
|
|
||||||
# Specify file containing keys for NTP authentication.
|
|
||||||
#keyfile /etc/chrony.keys
|
|
||||||
|
|
||||||
-# Save NTS keys and cookies.
|
|
||||||
-ntsdumpdir /var/lib/chrony
|
|
||||||
-
|
|
||||||
-# Insert/delete leap seconds by slewing instead of stepping.
|
|
||||||
-#leapsecmode slew
|
|
||||||
-
|
|
||||||
# Get TAI-UTC offset and leap seconds from the system tz database.
|
|
||||||
#leapsectz right/UTC
|
|
||||||
|
|
||||||
diff -up chrony-4.5/examples/chrony.keys.example.keys chrony-4.5/examples/chrony.keys.example
|
|
||||||
--- chrony-4.5/examples/chrony.keys.example.keys 2023-12-05 14:22:10.000000000 +0100
|
|
||||||
+++ chrony-4.5/examples/chrony.keys.example 2023-12-06 09:59:26.089508934 +0100
|
|
||||||
@@ -11,5 +11,3 @@
|
|
||||||
#1 MD5 AVeryLongAndRandomPassword
|
|
||||||
#2 MD5 HEX:12114855C7931009B4049EF3EFC48A139C3F989F
|
|
||||||
#3 SHA1 HEX:B2159C05D6A219673A3B7E896B6DE07F6A440995
|
|
||||||
-#4 AES128 HEX:2DA837C4B6573748CA692B8C828E4891
|
|
||||||
-#5 AES256 HEX:2666B8099BFF2D5BA20876121788ED24D2BE59111B8FFB562F0F56AE6EC7246E
|
|
@ -1,8 +0,0 @@
|
|||||||
[Unit]
|
|
||||||
Description=DNS SRV lookup of %I for chrony
|
|
||||||
After=chronyd.service network-online.target
|
|
||||||
Wants=network-online.target
|
|
||||||
|
|
||||||
[Service]
|
|
||||||
Type=oneshot
|
|
||||||
ExecStart=/usr/libexec/chrony-helper update-dnssrv-servers %I
|
|
@ -1,9 +0,0 @@
|
|||||||
[Unit]
|
|
||||||
Description=Periodic DNS SRV lookup of %I for chrony
|
|
||||||
|
|
||||||
[Timer]
|
|
||||||
OnActiveSec=0
|
|
||||||
OnUnitInactiveSec=1h
|
|
||||||
|
|
||||||
[Install]
|
|
||||||
WantedBy=timers.target
|
|
@ -1,86 +0,0 @@
|
|||||||
commit f49be7f06343ee27fff2950937d7f6742f53976f
|
|
||||||
Author: Miroslav Lichvar <mlichvar@redhat.com>
|
|
||||||
Date: Tue Mar 12 14:30:27 2024 +0100
|
|
||||||
|
|
||||||
conf: don't load sourcedir during initstepslew and RTC init
|
|
||||||
|
|
||||||
If the reload sources command was received in the chronyd start-up
|
|
||||||
sequence with initstepslew and/or RTC init (-s option), the sources
|
|
||||||
loaded from sourcedirs caused a crash due to failed assertion after
|
|
||||||
adding sources specified in the config.
|
|
||||||
|
|
||||||
Ignore the reload sources command until chronyd enters the normal
|
|
||||||
operation mode.
|
|
||||||
|
|
||||||
Fixes: 519796de3756 ("conf: add sourcedirs directive")
|
|
||||||
|
|
||||||
diff --git a/conf.c b/conf.c
|
|
||||||
index 6eae11c9..8849bdce 100644
|
|
||||||
--- a/conf.c
|
|
||||||
+++ b/conf.c
|
|
||||||
@@ -298,6 +298,8 @@ static ARR_Instance ntp_sources;
|
|
||||||
static ARR_Instance ntp_source_dirs;
|
|
||||||
/* Array of uint32_t corresponding to ntp_sources (for sourcedirs reload) */
|
|
||||||
static ARR_Instance ntp_source_ids;
|
|
||||||
+/* Flag indicating ntp_sources and ntp_source_ids are used for sourcedirs */
|
|
||||||
+static int conf_ntp_sources_added = 0;
|
|
||||||
|
|
||||||
/* Array of RefclockParameters */
|
|
||||||
static ARR_Instance refclock_sources;
|
|
||||||
@@ -1689,8 +1691,12 @@ reload_source_dirs(void)
|
|
||||||
NSR_Status s;
|
|
||||||
int d, pass;
|
|
||||||
|
|
||||||
+ /* Ignore reload command before adding configured sources */
|
|
||||||
+ if (!conf_ntp_sources_added)
|
|
||||||
+ return;
|
|
||||||
+
|
|
||||||
prev_size = ARR_GetSize(ntp_source_ids);
|
|
||||||
- if (prev_size > 0 && ARR_GetSize(ntp_sources) != prev_size)
|
|
||||||
+ if (ARR_GetSize(ntp_sources) != prev_size)
|
|
||||||
assert(0);
|
|
||||||
|
|
||||||
/* Save the current sources and their configuration IDs */
|
|
||||||
@@ -1859,7 +1865,10 @@ CNF_AddSources(void)
|
|
||||||
Free(source->params.name);
|
|
||||||
}
|
|
||||||
|
|
||||||
+ /* The arrays will be used for sourcedir (re)loading */
|
|
||||||
ARR_SetSize(ntp_sources, 0);
|
|
||||||
+ ARR_SetSize(ntp_source_ids, 0);
|
|
||||||
+ conf_ntp_sources_added = 1;
|
|
||||||
|
|
||||||
reload_source_dirs();
|
|
||||||
}
|
|
||||||
diff --git a/test/simulation/203-initreload b/test/simulation/203-initreload
|
|
||||||
new file mode 100755
|
|
||||||
index 00000000..cf7924b8
|
|
||||||
--- /dev/null
|
|
||||||
+++ b/test/simulation/203-initreload
|
|
||||||
@@ -0,0 +1,26 @@
|
|
||||||
+#!/usr/bin/env bash
|
|
||||||
+
|
|
||||||
+. ./test.common
|
|
||||||
+
|
|
||||||
+check_config_h 'FEAT_CMDMON 1' || test_skip
|
|
||||||
+
|
|
||||||
+# Test fix "conf: don't load sourcedir during initstepslew and RTC init"
|
|
||||||
+
|
|
||||||
+test_start "reload during initstepslew"
|
|
||||||
+
|
|
||||||
+client_conf="initstepslew 5 192.168.123.1
|
|
||||||
+sourcedir tmp"
|
|
||||||
+client_server_conf="#"
|
|
||||||
+chronyc_conf="reload sources"
|
|
||||||
+chronyc_start=4
|
|
||||||
+
|
|
||||||
+echo 'server 192.168.123.1' > tmp/sources.sources
|
|
||||||
+
|
|
||||||
+run_test || test_fail
|
|
||||||
+check_chronyd_exit || test_fail
|
|
||||||
+check_source_selection || test_fail
|
|
||||||
+check_sync || test_fail
|
|
||||||
+
|
|
||||||
+check_log_messages "Added source 192\.168\.123\.1" 1 1 || test_fail
|
|
||||||
+
|
|
||||||
+test_pass
|
|
@ -1,39 +0,0 @@
|
|||||||
commit e11b518a1ffa704986fb1f1835c425844ba248ef
|
|
||||||
Author: Miroslav Lichvar <mlichvar@redhat.com>
|
|
||||||
Date: Mon Jan 8 11:35:56 2024 +0100
|
|
||||||
|
|
||||||
ntp: fix authenticated requests in serverstats
|
|
||||||
|
|
||||||
Fix the CLG_UpdateNtpStats() call to count requests passing the
|
|
||||||
authentication check instead of requests triggering a KoD response
|
|
||||||
(i.e. NTS NAK).
|
|
||||||
|
|
||||||
diff --git a/ntp_core.c b/ntp_core.c
|
|
||||||
index 023e60b2..35801744 100644
|
|
||||||
--- a/ntp_core.c
|
|
||||||
+++ b/ntp_core.c
|
|
||||||
@@ -2736,7 +2736,7 @@ NCR_ProcessRxUnknown(NTP_Remote_Address *remote_addr, NTP_Local_Address *local_a
|
|
||||||
CLG_DisableNtpTimestamps(&ntp_rx);
|
|
||||||
}
|
|
||||||
|
|
||||||
- CLG_UpdateNtpStats(kod != 0 && info.auth.mode != NTP_AUTH_NONE &&
|
|
||||||
+ CLG_UpdateNtpStats(kod == 0 && info.auth.mode != NTP_AUTH_NONE &&
|
|
||||||
info.auth.mode != NTP_AUTH_MSSNTP,
|
|
||||||
rx_ts->source, interleaved ? tx_ts->source : NTP_TS_DAEMON);
|
|
||||||
|
|
||||||
diff --git a/test/system/010-nts b/test/system/010-nts
|
|
||||||
index 8d92bbc8..b215efa3 100755
|
|
||||||
--- a/test/system/010-nts
|
|
||||||
+++ b/test/system/010-nts
|
|
||||||
@@ -45,6 +45,11 @@ check_chronyc_output "^Name/IP address Mode KeyID Type KLen Last Atm
|
|
||||||
=========================================================================
|
|
||||||
127\.0\.0\.1 NTS 1 (30|15) (128|256) [0-9] 0 0 [78] ( 64|100)$" || test_fail
|
|
||||||
|
|
||||||
+run_chronyc "serverstats" || test_fail
|
|
||||||
+check_chronyc_output "NTS-KE connections accepted: 1
|
|
||||||
+NTS-KE connections dropped : 0
|
|
||||||
+Authenticated NTP packets : [1-9][0-9]*" || test_fail
|
|
||||||
+
|
|
||||||
stop_chronyd || test_fail
|
|
||||||
check_chronyd_messages || test_fail
|
|
||||||
check_chronyd_files || test_fail
|
|
@ -1,12 +0,0 @@
|
|||||||
diff -up chrony-4.1/examples/chronyd.service.service-helper chrony-4.1/examples/chronyd.service
|
|
||||||
--- chrony-4.1/examples/chronyd.service.service-helper 2021-05-12 13:06:15.000000000 +0200
|
|
||||||
+++ chrony-4.1/examples/chronyd.service 2021-06-15 09:01:56.948968576 +0200
|
|
||||||
@@ -10,6 +10,8 @@ Type=forking
|
|
||||||
PIDFile=/run/chrony/chronyd.pid
|
|
||||||
EnvironmentFile=-/etc/sysconfig/chronyd
|
|
||||||
ExecStart=/usr/sbin/chronyd $OPTIONS
|
|
||||||
+ExecStartPost=/usr/libexec/chrony-helper update-daemon
|
|
||||||
+ExecStopPost=/usr/libexec/chrony-helper remove-daemon-state
|
|
||||||
PrivateTmp=yes
|
|
||||||
ProtectHome=yes
|
|
||||||
ProtectSystem=full
|
|
@ -1,285 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
# This script configures running chronyd to use NTP servers obtained from
|
|
||||||
# DHCP and _ntp._udp DNS SRV records. Files with servers from DHCP are managed
|
|
||||||
# externally (e.g. by a dhclient script). Files with servers from DNS SRV
|
|
||||||
# records are updated here using the dig utility. The script can also list
|
|
||||||
# and set static sources in the chronyd configuration file.
|
|
||||||
|
|
||||||
chronyc=/usr/bin/chronyc
|
|
||||||
chrony_conf=/etc/chrony.conf
|
|
||||||
chrony_service=chronyd.service
|
|
||||||
helper_dir=/run/chrony-helper
|
|
||||||
added_servers_file=$helper_dir/added_servers
|
|
||||||
|
|
||||||
network_sysconfig_file=/etc/sysconfig/network
|
|
||||||
nm_servers_files="$helper_dir/nm-dhcp.*"
|
|
||||||
dhclient_servers_files="/var/lib/dhclient/chrony.servers.*"
|
|
||||||
dnssrv_servers_files="$helper_dir/dnssrv@*"
|
|
||||||
dnssrv_timer_prefix=chrony-dnssrv@
|
|
||||||
|
|
||||||
. $network_sysconfig_file &> /dev/null
|
|
||||||
|
|
||||||
chrony_command() {
|
|
||||||
$chronyc -n -m "$@"
|
|
||||||
}
|
|
||||||
|
|
||||||
is_running() {
|
|
||||||
chrony_command "tracking" &> /dev/null
|
|
||||||
}
|
|
||||||
|
|
||||||
get_servers_files() {
|
|
||||||
[ "$PEERNTP" != "no" ] && echo "$nm_servers_files"
|
|
||||||
[ "$PEERNTP" != "no" ] && echo "$dhclient_servers_files"
|
|
||||||
echo "$dnssrv_servers_files"
|
|
||||||
}
|
|
||||||
|
|
||||||
is_update_needed() {
|
|
||||||
for file in $(get_servers_files) $added_servers_file; do
|
|
||||||
[ -e "$file" ] && return 0
|
|
||||||
done
|
|
||||||
return 1
|
|
||||||
}
|
|
||||||
|
|
||||||
remove_daemon_state() {
|
|
||||||
rm -f $added_servers_file
|
|
||||||
}
|
|
||||||
|
|
||||||
update_daemon() {
|
|
||||||
local all_servers_with_args all_servers added_servers
|
|
||||||
|
|
||||||
if ! is_running; then
|
|
||||||
remove_daemon_state
|
|
||||||
return 0
|
|
||||||
fi
|
|
||||||
|
|
||||||
all_servers_with_args=$(cat $(get_servers_files) 2> /dev/null)
|
|
||||||
|
|
||||||
all_servers=$(
|
|
||||||
echo "$all_servers_with_args" |
|
|
||||||
while read -r server serverargs; do
|
|
||||||
echo "$server"
|
|
||||||
done | sort -u)
|
|
||||||
added_servers=$( (
|
|
||||||
cat $added_servers_file 2> /dev/null
|
|
||||||
echo "$all_servers_with_args" |
|
|
||||||
while read -r server serverargs; do
|
|
||||||
[ -z "$server" ] && continue
|
|
||||||
chrony_command "add server $server $serverargs" &> /dev/null &&
|
|
||||||
echo "$server"
|
|
||||||
done) | sort -u)
|
|
||||||
|
|
||||||
comm -23 <(echo -n "$added_servers") <(echo -n "$all_servers") |
|
|
||||||
while read -r server; do
|
|
||||||
chrony_command -c sources -a 2>/dev/null |
|
|
||||||
while IFS=, read -r type _ address _; do
|
|
||||||
[ "$type" = "^" ] || continue
|
|
||||||
[ "$(chrony_command "sourcename $address")" = "$server" ] || continue
|
|
||||||
chrony_command "delete $address" &> /dev/null
|
|
||||||
break
|
|
||||||
done
|
|
||||||
done
|
|
||||||
|
|
||||||
added_servers=$(comm -12 <(echo -n "$added_servers") <(echo -n "$all_servers"))
|
|
||||||
|
|
||||||
if [ -n "$added_servers" ]; then
|
|
||||||
echo "$added_servers" > $added_servers_file
|
|
||||||
else
|
|
||||||
rm -f $added_servers_file
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
get_dnssrv_servers() {
|
|
||||||
local name=$1 output
|
|
||||||
|
|
||||||
if ! command -v dig &> /dev/null; then
|
|
||||||
echo "Missing dig (DNS lookup utility)" >&2
|
|
||||||
return 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
output=$(dig "$name" srv +short +ndots=2 +search 2> /dev/null) || return 0
|
|
||||||
|
|
||||||
echo "$output" | while read -r _ _ port target; do
|
|
||||||
server=${target%.}
|
|
||||||
[ -z "$server" ] && continue
|
|
||||||
echo "$server port $port ${NTPSERVERARGS:-iburst}"
|
|
||||||
done
|
|
||||||
}
|
|
||||||
|
|
||||||
check_dnssrv_name() {
|
|
||||||
local name=$1
|
|
||||||
|
|
||||||
if [ -z "$name" ]; then
|
|
||||||
echo "No DNS SRV name specified" >&2
|
|
||||||
return 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ "${name:0:9}" != _ntp._udp ]; then
|
|
||||||
echo "DNS SRV name $name doesn't start with _ntp._udp" >&2
|
|
||||||
return 1
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
update_dnssrv_servers() {
|
|
||||||
local name=$1
|
|
||||||
local srv_file=$helper_dir/dnssrv@$name servers
|
|
||||||
|
|
||||||
check_dnssrv_name "$name" || return 1
|
|
||||||
|
|
||||||
servers=$(get_dnssrv_servers "$name")
|
|
||||||
if [ -n "$servers" ]; then
|
|
||||||
echo "$servers" > "$srv_file"
|
|
||||||
else
|
|
||||||
rm -f "$srv_file"
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
set_dnssrv_timer() {
|
|
||||||
local state=$1 name=$2
|
|
||||||
local srv_file=$helper_dir/dnssrv@$name servers
|
|
||||||
local timer
|
|
||||||
|
|
||||||
timer=$dnssrv_timer_prefix$(systemd-escape "$name").timer || return 1
|
|
||||||
|
|
||||||
check_dnssrv_name "$name" || return 1
|
|
||||||
|
|
||||||
if [ "$state" = enable ]; then
|
|
||||||
systemctl enable "$timer"
|
|
||||||
systemctl start "$timer"
|
|
||||||
elif [ "$state" = disable ]; then
|
|
||||||
systemctl stop "$timer"
|
|
||||||
systemctl disable "$timer"
|
|
||||||
rm -f "$srv_file"
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
list_dnssrv_timers() {
|
|
||||||
systemctl --all --full -t timer list-units | grep "^$dnssrv_timer_prefix" | \
|
|
||||||
sed "s|^$dnssrv_timer_prefix\(.*\)\.timer.*|\1|" |
|
|
||||||
while read -r name; do
|
|
||||||
systemd-escape --unescape "$name"
|
|
||||||
done
|
|
||||||
}
|
|
||||||
|
|
||||||
prepare_helper_dir() {
|
|
||||||
mkdir -p $helper_dir
|
|
||||||
exec 100> $helper_dir/lock
|
|
||||||
if ! flock -w 20 100; then
|
|
||||||
echo "Failed to lock $helper_dir" >&2
|
|
||||||
return 1
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
is_source_line() {
|
|
||||||
local pattern="^[ \t]*(server|pool|peer|refclock)[ \t]+[^ \t]+"
|
|
||||||
[[ "$1" =~ $pattern ]]
|
|
||||||
}
|
|
||||||
|
|
||||||
list_static_sources() {
|
|
||||||
while read -r line; do
|
|
||||||
if is_source_line "$line"; then
|
|
||||||
echo "$line"
|
|
||||||
fi
|
|
||||||
done < $chrony_conf
|
|
||||||
}
|
|
||||||
|
|
||||||
set_static_sources() {
|
|
||||||
local new_config tmp_conf
|
|
||||||
|
|
||||||
new_config=$(
|
|
||||||
sources=$(
|
|
||||||
while read -r line; do
|
|
||||||
is_source_line "$line" && echo "$line"
|
|
||||||
done)
|
|
||||||
|
|
||||||
while read -r line; do
|
|
||||||
if ! is_source_line "$line"; then
|
|
||||||
echo "$line"
|
|
||||||
continue
|
|
||||||
fi
|
|
||||||
|
|
||||||
tmp_sources=$(
|
|
||||||
local removed=0
|
|
||||||
|
|
||||||
echo "$sources" | while read -r line2; do
|
|
||||||
if [ "$removed" -ne 0 ] || [ "$line" != "$line2" ]; then
|
|
||||||
echo "$line2"
|
|
||||||
else
|
|
||||||
removed=1
|
|
||||||
fi
|
|
||||||
done)
|
|
||||||
|
|
||||||
[ "$sources" == "$tmp_sources" ] && continue
|
|
||||||
sources=$tmp_sources
|
|
||||||
echo "$line"
|
|
||||||
done < $chrony_conf
|
|
||||||
|
|
||||||
echo "$sources"
|
|
||||||
)
|
|
||||||
|
|
||||||
tmp_conf=${chrony_conf}.tmp
|
|
||||||
|
|
||||||
cp -a $chrony_conf $tmp_conf &&
|
|
||||||
echo "$new_config" > $tmp_conf &&
|
|
||||||
mv $tmp_conf $chrony_conf || return 1
|
|
||||||
|
|
||||||
systemctl try-restart $chrony_service
|
|
||||||
}
|
|
||||||
|
|
||||||
print_help() {
|
|
||||||
echo "Usage: $0 COMMAND"
|
|
||||||
echo
|
|
||||||
echo "Commands:"
|
|
||||||
echo " create-helper-directory"
|
|
||||||
echo " update-daemon"
|
|
||||||
echo " remove-daemon-state"
|
|
||||||
echo " update-dnssrv-servers NAME"
|
|
||||||
echo " enable-dnssrv NAME"
|
|
||||||
echo " disable-dnssrv NAME"
|
|
||||||
echo " list-dnssrv"
|
|
||||||
echo " list-static-sources"
|
|
||||||
echo " set-static-sources < sources.list"
|
|
||||||
echo " is-running"
|
|
||||||
echo " command CHRONYC-COMMAND"
|
|
||||||
}
|
|
||||||
|
|
||||||
case "$1" in
|
|
||||||
create-helper-directory)
|
|
||||||
prepare_helper_dir
|
|
||||||
;;
|
|
||||||
update-daemon|add-dhclient-servers|remove-dhclient-servers)
|
|
||||||
is_update_needed || exit 0
|
|
||||||
prepare_helper_dir && update_daemon
|
|
||||||
;;
|
|
||||||
remove-daemon-state)
|
|
||||||
remove_daemon_state
|
|
||||||
;;
|
|
||||||
update-dnssrv-servers)
|
|
||||||
prepare_helper_dir && update_dnssrv_servers "$2" && update_daemon
|
|
||||||
;;
|
|
||||||
enable-dnssrv)
|
|
||||||
set_dnssrv_timer enable "$2"
|
|
||||||
;;
|
|
||||||
disable-dnssrv)
|
|
||||||
set_dnssrv_timer disable "$2" && prepare_helper_dir && update_daemon
|
|
||||||
;;
|
|
||||||
list-dnssrv)
|
|
||||||
list_dnssrv_timers
|
|
||||||
;;
|
|
||||||
list-static-sources)
|
|
||||||
list_static_sources
|
|
||||||
;;
|
|
||||||
set-static-sources)
|
|
||||||
set_static_sources
|
|
||||||
;;
|
|
||||||
is-running)
|
|
||||||
is_running
|
|
||||||
;;
|
|
||||||
command|forced-command)
|
|
||||||
chrony_command "$2"
|
|
||||||
;;
|
|
||||||
*)
|
|
||||||
print_help
|
|
||||||
exit 2
|
|
||||||
esac
|
|
||||||
|
|
||||||
exit $?
|
|
@ -0,0 +1,2 @@
|
|||||||
|
#Type Name ID GECOS Home directory Shell
|
||||||
|
u chrony - "chrony system user" /var/lib/chrony /sbin/nologin
|
@ -1,680 +0,0 @@
|
|||||||
#!/usr/bin/python3
|
|
||||||
#
|
|
||||||
# Convert ntp configuration to chrony
|
|
||||||
#
|
|
||||||
# Copyright (C) 2018-2019 Miroslav Lichvar <mlichvar@redhat.com>
|
|
||||||
#
|
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining
|
|
||||||
# a copy of this software and associated documentation files (the
|
|
||||||
# "Software"), to deal in the Software without restriction, including
|
|
||||||
# without limitation the rights to use, copy, modify, merge, publish,
|
|
||||||
# distribute, sublicense, and/or sell copies of the Software, and to
|
|
||||||
# permit persons to whom the Software is furnished to do so, subject to
|
|
||||||
# the following conditions:
|
|
||||||
#
|
|
||||||
# The above copyright notice and this permission notice shall be included
|
|
||||||
# in all copies or substantial portions of the Software.
|
|
||||||
#
|
|
||||||
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
||||||
# EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
||||||
# MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
|
|
||||||
# IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
|
|
||||||
# CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
|
|
||||||
# TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
|
|
||||||
# SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
||||||
|
|
||||||
|
|
||||||
import argparse
|
|
||||||
import ipaddress
|
|
||||||
import logging
|
|
||||||
import os
|
|
||||||
import re
|
|
||||||
import subprocess
|
|
||||||
import sys
|
|
||||||
|
|
||||||
# python2 compatibility hacks
|
|
||||||
if sys.version_info[0] < 3:
|
|
||||||
from io import open
|
|
||||||
reload(sys)
|
|
||||||
sys.setdefaultencoding("utf-8")
|
|
||||||
|
|
||||||
|
|
||||||
class NtpConfiguration(object):
|
|
||||||
def __init__(self, root_dir, ntp_conf, step_tickers):
|
|
||||||
self.root_dir = root_dir if root_dir != "/" else ""
|
|
||||||
self.ntp_conf_path = ntp_conf
|
|
||||||
self.step_tickers_path = step_tickers
|
|
||||||
|
|
||||||
# Read and write files using an 8-bit transparent encoding
|
|
||||||
self.file_encoding = "latin-1"
|
|
||||||
self.enabled_services = set()
|
|
||||||
self.step_tickers = []
|
|
||||||
self.time_sources = []
|
|
||||||
self.fudges = {}
|
|
||||||
self.restrictions = {
|
|
||||||
# Built-in defaults
|
|
||||||
ipaddress.ip_network(u"0.0.0.0/0"): set(),
|
|
||||||
ipaddress.ip_network(u"::/0"): set(),
|
|
||||||
}
|
|
||||||
self.keyfile = ""
|
|
||||||
self.keys = []
|
|
||||||
self.trusted_keys = []
|
|
||||||
self.driftfile = ""
|
|
||||||
self.statistics = []
|
|
||||||
self.leapfile = ""
|
|
||||||
self.tos_options = {}
|
|
||||||
self.ignored_directives = set()
|
|
||||||
self.ignored_lines = []
|
|
||||||
|
|
||||||
# self.detect_enabled_services()
|
|
||||||
self.parse_step_tickers()
|
|
||||||
self.parse_ntp_conf()
|
|
||||||
|
|
||||||
def detect_enabled_services(self):
|
|
||||||
for service in ["ntpdate", "ntpd", "ntp-wait"]:
|
|
||||||
service_path = os.path.join(self.root_dir,
|
|
||||||
"etc/systemd/system/multi-user.target.wants/{}.service".format(service))
|
|
||||||
if os.path.islink(service_path):
|
|
||||||
self.enabled_services.add(service)
|
|
||||||
logging.info("Enabled services found in /etc/systemd/system: %s",
|
|
||||||
" ".join(self.enabled_services))
|
|
||||||
|
|
||||||
def parse_step_tickers(self):
|
|
||||||
if not self.step_tickers_path:
|
|
||||||
return
|
|
||||||
|
|
||||||
path = os.path.join(self.root_dir, self.step_tickers_path)
|
|
||||||
if not os.path.isfile(path):
|
|
||||||
logging.info("Missing %s", path)
|
|
||||||
return
|
|
||||||
|
|
||||||
with open(path, encoding=self.file_encoding) as f:
|
|
||||||
for line in f:
|
|
||||||
line = line[:line.find('#')]
|
|
||||||
|
|
||||||
words = line.split()
|
|
||||||
|
|
||||||
if not words:
|
|
||||||
continue
|
|
||||||
|
|
||||||
self.step_tickers.extend(words)
|
|
||||||
|
|
||||||
def parse_ntp_conf(self, path=None):
|
|
||||||
if path is None:
|
|
||||||
path = os.path.join(self.root_dir, self.ntp_conf_path)
|
|
||||||
|
|
||||||
with open(path, encoding=self.file_encoding) as f:
|
|
||||||
logging.info("Reading %s", path)
|
|
||||||
|
|
||||||
for line in f:
|
|
||||||
line = line[:line.find('#')]
|
|
||||||
|
|
||||||
words = line.split()
|
|
||||||
|
|
||||||
if not words:
|
|
||||||
continue
|
|
||||||
|
|
||||||
if not self.parse_directive(words):
|
|
||||||
self.ignored_lines.append(line)
|
|
||||||
|
|
||||||
def parse_directive(self, words):
|
|
||||||
name = words.pop(0)
|
|
||||||
if name.startswith("logconfig"):
|
|
||||||
name = "logconfig"
|
|
||||||
|
|
||||||
if words:
|
|
||||||
if name in ["server", "peer", "pool"]:
|
|
||||||
return self.parse_source(name, words)
|
|
||||||
elif name == "fudge":
|
|
||||||
return self.parse_fudge(words)
|
|
||||||
elif name == "restrict":
|
|
||||||
return self.parse_restrict(words)
|
|
||||||
elif name == "tos":
|
|
||||||
return self.parse_tos(words)
|
|
||||||
elif name == "includefile":
|
|
||||||
return self.parse_includefile(words)
|
|
||||||
elif name == "keys":
|
|
||||||
return self.parse_keys(words)
|
|
||||||
elif name == "trustedkey":
|
|
||||||
return self.parse_trustedkey(words)
|
|
||||||
elif name == "driftfile":
|
|
||||||
self.driftfile = words[0]
|
|
||||||
elif name == "statistics":
|
|
||||||
self.statistics = words
|
|
||||||
elif name == "leapfile":
|
|
||||||
self.leapfile = words[0]
|
|
||||||
else:
|
|
||||||
self.ignored_directives.add(name)
|
|
||||||
return False
|
|
||||||
else:
|
|
||||||
self.ignored_directives.add(name)
|
|
||||||
return False
|
|
||||||
|
|
||||||
return True
|
|
||||||
|
|
||||||
def parse_source(self, source_type, words):
|
|
||||||
ipv4_only = False
|
|
||||||
ipv6_only = False
|
|
||||||
source = {
|
|
||||||
"type": source_type,
|
|
||||||
"options": []
|
|
||||||
}
|
|
||||||
|
|
||||||
if words[0] == "-4":
|
|
||||||
ipv4_only = True
|
|
||||||
words.pop(0)
|
|
||||||
elif words[0] == "-6":
|
|
||||||
ipv6_only = True
|
|
||||||
words.pop(0)
|
|
||||||
|
|
||||||
if not words:
|
|
||||||
return False
|
|
||||||
|
|
||||||
source["address"] = words.pop(0)
|
|
||||||
|
|
||||||
# Check if -4/-6 corresponds to the address and ignore hostnames
|
|
||||||
if ipv4_only or ipv6_only:
|
|
||||||
try:
|
|
||||||
version = ipaddress.ip_address(source["address"]).version
|
|
||||||
if (ipv4_only and version != 4) or (ipv6_only and version != 6):
|
|
||||||
return False
|
|
||||||
except ValueError:
|
|
||||||
return False
|
|
||||||
|
|
||||||
if source["address"].startswith("127.127."):
|
|
||||||
if not source["address"].startswith("127.127.1."):
|
|
||||||
# Ignore non-LOCAL refclocks
|
|
||||||
return False
|
|
||||||
|
|
||||||
while words:
|
|
||||||
if len(words) >= 2 and words[0] in ["minpoll", "maxpoll", "version", "key"]:
|
|
||||||
source["options"].append((words[0], words[1]))
|
|
||||||
words = words[2:]
|
|
||||||
elif words[0] in ["burst", "iburst", "noselect", "prefer", "true", "xleave"]:
|
|
||||||
source["options"].append((words[0],))
|
|
||||||
words.pop(0)
|
|
||||||
else:
|
|
||||||
return False
|
|
||||||
|
|
||||||
self.time_sources.append(source)
|
|
||||||
return True
|
|
||||||
|
|
||||||
def parse_fudge(self, words):
|
|
||||||
address = words.pop(0)
|
|
||||||
options = {}
|
|
||||||
|
|
||||||
while words:
|
|
||||||
if len(words) >= 2 and words[0] in ["stratum"]:
|
|
||||||
if not words[1].isdigit():
|
|
||||||
return False
|
|
||||||
options[words[0]] = int(words[1])
|
|
||||||
words = words[2:]
|
|
||||||
elif len(words) >= 2:
|
|
||||||
words = words[2:]
|
|
||||||
else:
|
|
||||||
return False
|
|
||||||
|
|
||||||
self.fudges[address] = options
|
|
||||||
return True
|
|
||||||
|
|
||||||
def parse_restrict(self, words):
|
|
||||||
ipv4_only = False
|
|
||||||
ipv6_only = False
|
|
||||||
flags = set()
|
|
||||||
mask = ""
|
|
||||||
|
|
||||||
if words[0] == "-4":
|
|
||||||
ipv4_only = True
|
|
||||||
words.pop(0)
|
|
||||||
elif words[0] == "-6":
|
|
||||||
ipv6_only = True
|
|
||||||
words.pop(0)
|
|
||||||
|
|
||||||
if not words:
|
|
||||||
return False
|
|
||||||
|
|
||||||
address = words.pop(0)
|
|
||||||
|
|
||||||
while words:
|
|
||||||
if len(words) >= 2 and words[0] == "mask":
|
|
||||||
mask = words[1]
|
|
||||||
words = words[2:]
|
|
||||||
else:
|
|
||||||
if words[0] not in ["kod", "nomodify", "notrap", "nopeer", "noquery",
|
|
||||||
"limited", "ignore", "noserve"]:
|
|
||||||
return False
|
|
||||||
flags.add(words[0])
|
|
||||||
words.pop(0)
|
|
||||||
|
|
||||||
# Convert to IP network(s), ignoring restrictions with hostnames
|
|
||||||
networks = []
|
|
||||||
if address == "default" and not mask:
|
|
||||||
if not ipv6_only:
|
|
||||||
networks.append(ipaddress.ip_network(u"0.0.0.0/0"))
|
|
||||||
if not ipv4_only:
|
|
||||||
networks.append(ipaddress.ip_network(u"::/0"))
|
|
||||||
else:
|
|
||||||
try:
|
|
||||||
if mask:
|
|
||||||
# Count bits in the mask (ipaddress does not support
|
|
||||||
# expanded IPv6 netmasks)
|
|
||||||
mask_ip = ipaddress.ip_address(mask)
|
|
||||||
mask_str = "{0:0{1}b}".format(int(mask_ip), mask_ip.max_prefixlen)
|
|
||||||
networks.append(ipaddress.ip_network(
|
|
||||||
u"{}/{}".format(address, len(mask_str.rstrip('0')))))
|
|
||||||
else:
|
|
||||||
networks.append(ipaddress.ip_network(address))
|
|
||||||
except ValueError:
|
|
||||||
return False
|
|
||||||
|
|
||||||
if (ipv4_only and networks[-1].version != 4) or \
|
|
||||||
(ipv6_only and networks[-1].version != 6):
|
|
||||||
return False
|
|
||||||
|
|
||||||
for network in networks:
|
|
||||||
self.restrictions[network] = flags
|
|
||||||
|
|
||||||
return True
|
|
||||||
|
|
||||||
def parse_tos(self, words):
|
|
||||||
options = {}
|
|
||||||
while words:
|
|
||||||
if len(words) >= 2 and words[0] in ["minsane", "orphan"]:
|
|
||||||
if not words[1].isdigit():
|
|
||||||
return False
|
|
||||||
options[words[0]] = int(words[1])
|
|
||||||
words = words[2:]
|
|
||||||
elif len(words) >= 2 and words[0] in ["maxdist"]:
|
|
||||||
# Check if it is a float value
|
|
||||||
if not words[1].replace('.', '', 1).isdigit():
|
|
||||||
return False
|
|
||||||
options[words[0]] = float(words[1])
|
|
||||||
words = words[2:]
|
|
||||||
else:
|
|
||||||
return False
|
|
||||||
|
|
||||||
self.tos_options.update(options)
|
|
||||||
|
|
||||||
return True
|
|
||||||
|
|
||||||
def parse_includefile(self, words):
|
|
||||||
path = os.path.join(self.root_dir, words[0])
|
|
||||||
if not os.path.isfile(path):
|
|
||||||
return False
|
|
||||||
|
|
||||||
self.parse_ntp_conf(path)
|
|
||||||
return True
|
|
||||||
|
|
||||||
def parse_keys(self, words):
|
|
||||||
keyfile = words[0]
|
|
||||||
path = os.path.join(self.root_dir, keyfile)
|
|
||||||
if not os.path.isfile(path):
|
|
||||||
logging.info("Missing %s", path)
|
|
||||||
return False
|
|
||||||
|
|
||||||
with open(path, encoding=self.file_encoding) as f:
|
|
||||||
logging.info("Reading %s", path)
|
|
||||||
keys = []
|
|
||||||
for line in f:
|
|
||||||
words = line.split()
|
|
||||||
if len(words) < 3 or not words[0].isdigit():
|
|
||||||
continue
|
|
||||||
keys.append((int(words[0]), words[1], words[2]))
|
|
||||||
|
|
||||||
self.keyfile = keyfile
|
|
||||||
self.keys = keys
|
|
||||||
|
|
||||||
return True
|
|
||||||
|
|
||||||
def parse_trustedkey(self, words):
|
|
||||||
key_ranges = []
|
|
||||||
for word in words:
|
|
||||||
if word.isdigit():
|
|
||||||
key_ranges.append((int(word), int(word)))
|
|
||||||
elif re.match("^[0-9]+-[0-9]+$", word):
|
|
||||||
first, last = word.split("-")
|
|
||||||
key_ranges.append((int(first), int(last)))
|
|
||||||
else:
|
|
||||||
return False
|
|
||||||
|
|
||||||
self.trusted_keys = key_ranges
|
|
||||||
return True
|
|
||||||
|
|
||||||
def write_chrony_configuration(self, chrony_conf_path, chrony_keys_path,
|
|
||||||
dry_run=False, backup=False):
|
|
||||||
chrony_conf = self.get_chrony_conf(chrony_keys_path)
|
|
||||||
logging.debug("Generated %s:\n%s", chrony_conf_path, chrony_conf)
|
|
||||||
|
|
||||||
if not dry_run:
|
|
||||||
self.write_file(chrony_conf_path, 0o644, chrony_conf, backup)
|
|
||||||
|
|
||||||
chrony_keys = self.get_chrony_keys()
|
|
||||||
if chrony_keys:
|
|
||||||
logging.debug("Generated %s:\n%s", chrony_keys_path, chrony_keys)
|
|
||||||
|
|
||||||
if not dry_run:
|
|
||||||
self.write_file(chrony_keys_path, 0o640, chrony_keys, backup)
|
|
||||||
|
|
||||||
def get_processed_time_sources(self):
|
|
||||||
# Convert {0,1,2,3}.*pool.ntp.org servers to 2.*pool.ntp.org pools
|
|
||||||
|
|
||||||
# Make shallow copies of all sources (only type will be modified)
|
|
||||||
time_sources = [s.copy() for s in self.time_sources]
|
|
||||||
|
|
||||||
pools = {}
|
|
||||||
for source in time_sources:
|
|
||||||
if source["type"] != "server":
|
|
||||||
continue
|
|
||||||
m = re.match("^([0123])(\\.\\w+)?\\.pool\\.ntp\\.org$", source["address"])
|
|
||||||
if m is None:
|
|
||||||
continue
|
|
||||||
number = m.group(1)
|
|
||||||
zone = m.group(2)
|
|
||||||
if zone not in pools:
|
|
||||||
pools[zone] = []
|
|
||||||
pools[zone].append((int(number), source))
|
|
||||||
|
|
||||||
remove_servers = set()
|
|
||||||
for zone, pool in pools.items():
|
|
||||||
# sort and skip all pools not in [0, 3] range
|
|
||||||
pool.sort()
|
|
||||||
if [number for number, source in pool] != [0, 1, 2, 3]:
|
|
||||||
# only exact group of 4 servers can be converted, nothing to do here
|
|
||||||
continue
|
|
||||||
# verify that parameters are the same for all servers in the pool
|
|
||||||
if not all([p[1]["options"] == pool[0][1]["options"] for p in pool]):
|
|
||||||
break
|
|
||||||
remove_servers.update([pool[i][1]["address"] for i in [0, 1, 3]])
|
|
||||||
pool[2][1]["type"] = "pool"
|
|
||||||
|
|
||||||
processed_sources = []
|
|
||||||
for source in time_sources:
|
|
||||||
if source["type"] == "server" and source["address"] in remove_servers:
|
|
||||||
continue
|
|
||||||
processed_sources.append(source)
|
|
||||||
return processed_sources
|
|
||||||
|
|
||||||
def get_chrony_conf_sources(self):
|
|
||||||
conf = ""
|
|
||||||
|
|
||||||
if self.step_tickers:
|
|
||||||
conf += "# Specify NTP servers used for initial correction.\n"
|
|
||||||
conf += "initstepslew 0.1 {}\n".format(" ".join(self.step_tickers))
|
|
||||||
conf += "\n"
|
|
||||||
|
|
||||||
conf += "# Specify time sources.\n"
|
|
||||||
|
|
||||||
for source in self.get_processed_time_sources():
|
|
||||||
address = source["address"]
|
|
||||||
if address.startswith("127.127."):
|
|
||||||
if address.startswith("127.127.1."):
|
|
||||||
continue
|
|
||||||
# No other refclocks are expected from the parser
|
|
||||||
assert False
|
|
||||||
else:
|
|
||||||
conf += "{} {}".format(source["type"], address)
|
|
||||||
for option in source["options"]:
|
|
||||||
if option[0] in ["minpoll", "maxpoll", "version", "key",
|
|
||||||
"iburst", "noselect", "prefer", "xleave"]:
|
|
||||||
conf += " {}".format(" ".join(option))
|
|
||||||
elif option[0] == "burst":
|
|
||||||
conf += " presend 6"
|
|
||||||
elif option[0] == "true":
|
|
||||||
conf += " trust"
|
|
||||||
else:
|
|
||||||
# No other options are expected from the parser
|
|
||||||
assert False
|
|
||||||
conf += "\n"
|
|
||||||
conf += "\n"
|
|
||||||
|
|
||||||
return conf
|
|
||||||
|
|
||||||
def get_chrony_conf_allows(self):
|
|
||||||
allowed_networks = filter(lambda n: "ignore" not in self.restrictions[n] and
|
|
||||||
"noserve" not in self.restrictions[n],
|
|
||||||
self.restrictions.keys())
|
|
||||||
|
|
||||||
conf = ""
|
|
||||||
for network in sorted(allowed_networks, key=lambda n: (n.version, n)):
|
|
||||||
if network.num_addresses > 1:
|
|
||||||
conf += "allow {}\n".format(network)
|
|
||||||
else:
|
|
||||||
conf += "allow {}\n".format(network.network_address)
|
|
||||||
|
|
||||||
if conf:
|
|
||||||
conf = "# Allow NTP client access.\n" + conf
|
|
||||||
conf += "\n"
|
|
||||||
|
|
||||||
return conf
|
|
||||||
|
|
||||||
def get_chrony_conf_cmdallows(self):
|
|
||||||
allowed_networks = filter(lambda n: "ignore" not in self.restrictions[n] and
|
|
||||||
"noquery" not in self.restrictions[n] and
|
|
||||||
n != ipaddress.ip_network(u"127.0.0.1/32") and
|
|
||||||
n != ipaddress.ip_network(u"::1/128"),
|
|
||||||
self.restrictions.keys())
|
|
||||||
|
|
||||||
ip_versions = set()
|
|
||||||
conf = ""
|
|
||||||
for network in sorted(allowed_networks, key=lambda n: (n.version, n)):
|
|
||||||
ip_versions.add(network.version)
|
|
||||||
if network.num_addresses > 1:
|
|
||||||
conf += "cmdallow {}\n".format(network)
|
|
||||||
else:
|
|
||||||
conf += "cmdallow {}\n".format(network.network_address)
|
|
||||||
|
|
||||||
if conf:
|
|
||||||
conf = "# Allow remote monitoring.\n" + conf
|
|
||||||
if 4 in ip_versions:
|
|
||||||
conf += "bindcmdaddress 0.0.0.0\n"
|
|
||||||
if 6 in ip_versions:
|
|
||||||
conf += "bindcmdaddress ::\n"
|
|
||||||
conf += "\n"
|
|
||||||
|
|
||||||
return conf
|
|
||||||
|
|
||||||
def get_chrony_conf(self, chrony_keys_path):
|
|
||||||
local_stratum = 0
|
|
||||||
maxdistance = 0.0
|
|
||||||
minsources = 1
|
|
||||||
orphan_stratum = 0
|
|
||||||
logs = []
|
|
||||||
|
|
||||||
for source in self.time_sources:
|
|
||||||
address = source["address"]
|
|
||||||
if address.startswith("127.127.1."):
|
|
||||||
if address in self.fudges and "stratum" in self.fudges[address]:
|
|
||||||
local_stratum = self.fudges[address]["stratum"]
|
|
||||||
else:
|
|
||||||
local_stratum = 5
|
|
||||||
|
|
||||||
if "maxdist" in self.tos_options:
|
|
||||||
maxdistance = self.tos_options["maxdist"]
|
|
||||||
if "minsane" in self.tos_options:
|
|
||||||
minsources = self.tos_options["minsane"]
|
|
||||||
if "orphan" in self.tos_options:
|
|
||||||
orphan_stratum = self.tos_options["orphan"]
|
|
||||||
|
|
||||||
if "clockstats" in self.statistics:
|
|
||||||
logs.append("refclocks")
|
|
||||||
if "loopstats" in self.statistics:
|
|
||||||
logs.append("tracking")
|
|
||||||
if "peerstats" in self.statistics:
|
|
||||||
logs.append("statistics")
|
|
||||||
if "rawstats" in self.statistics:
|
|
||||||
logs.append("measurements")
|
|
||||||
|
|
||||||
conf = "# This file was converted from {}{}.\n".format(
|
|
||||||
self.ntp_conf_path,
|
|
||||||
" and " + self.step_tickers_path if self.step_tickers_path else "")
|
|
||||||
conf += "\n"
|
|
||||||
|
|
||||||
if self.ignored_lines:
|
|
||||||
conf += "# The following directives were ignored in the conversion:\n"
|
|
||||||
|
|
||||||
for line in self.ignored_lines:
|
|
||||||
# Remove sensitive information
|
|
||||||
line = re.sub(r"\s+pw\s+\S+", " pw XXX", line.rstrip())
|
|
||||||
conf += "# " + line + "\n"
|
|
||||||
conf += "\n"
|
|
||||||
|
|
||||||
conf += self.get_chrony_conf_sources()
|
|
||||||
|
|
||||||
conf += "# Record the rate at which the system clock gains/losses time.\n"
|
|
||||||
if not self.driftfile:
|
|
||||||
conf += "#"
|
|
||||||
conf += "driftfile /var/lib/chrony/drift\n"
|
|
||||||
conf += "\n"
|
|
||||||
|
|
||||||
conf += "# Allow the system clock to be stepped in the first three updates\n"
|
|
||||||
conf += "# if its offset is larger than 1 second.\n"
|
|
||||||
conf += "makestep 1.0 3\n"
|
|
||||||
conf += "\n"
|
|
||||||
|
|
||||||
conf += "# Enable kernel synchronization of the real-time clock (RTC).\n"
|
|
||||||
conf += "rtcsync\n"
|
|
||||||
conf += "\n"
|
|
||||||
|
|
||||||
conf += "# Enable hardware timestamping on all interfaces that support it.\n"
|
|
||||||
conf += "#hwtimestamp *\n"
|
|
||||||
conf += "\n"
|
|
||||||
|
|
||||||
if maxdistance > 0.0:
|
|
||||||
conf += "# Specify the maximum distance of sources to be selectable.\n"
|
|
||||||
conf += "maxdistance {}\n".format(maxdistance)
|
|
||||||
conf += "\n"
|
|
||||||
|
|
||||||
conf += "# Increase the minimum number of selectable sources required to adjust\n"
|
|
||||||
conf += "# the system clock.\n"
|
|
||||||
if minsources > 1:
|
|
||||||
conf += "minsources {}\n".format(minsources)
|
|
||||||
else:
|
|
||||||
conf += "#minsources 2\n"
|
|
||||||
conf += "\n"
|
|
||||||
|
|
||||||
conf += self.get_chrony_conf_allows()
|
|
||||||
|
|
||||||
conf += self.get_chrony_conf_cmdallows()
|
|
||||||
|
|
||||||
conf += "# Serve time even if not synchronized to a time source.\n"
|
|
||||||
if orphan_stratum > 0 and orphan_stratum < 16:
|
|
||||||
conf += "local stratum {} orphan\n".format(orphan_stratum)
|
|
||||||
elif local_stratum > 0 and local_stratum < 16:
|
|
||||||
conf += "local stratum {}\n".format(local_stratum)
|
|
||||||
else:
|
|
||||||
conf += "#local stratum 10\n"
|
|
||||||
conf += "\n"
|
|
||||||
|
|
||||||
conf += "# Specify file containing keys for NTP authentication.\n"
|
|
||||||
conf += "keyfile {}\n".format(chrony_keys_path)
|
|
||||||
conf += "\n"
|
|
||||||
|
|
||||||
conf += "# Get TAI-UTC offset and leap seconds from the system tz database.\n"
|
|
||||||
conf += "leapsectz right/UTC\n"
|
|
||||||
conf += "\n"
|
|
||||||
|
|
||||||
conf += "# Specify directory for log files.\n"
|
|
||||||
conf += "logdir /var/log/chrony\n"
|
|
||||||
conf += "\n"
|
|
||||||
|
|
||||||
conf += "# Select which information is logged.\n"
|
|
||||||
if logs:
|
|
||||||
conf += "log {}\n".format(" ".join(logs))
|
|
||||||
else:
|
|
||||||
conf += "#log measurements statistics tracking\n"
|
|
||||||
|
|
||||||
return conf
|
|
||||||
|
|
||||||
def get_chrony_keys(self):
|
|
||||||
if not self.keyfile:
|
|
||||||
return ""
|
|
||||||
|
|
||||||
keys = "# This file was converted from {}.\n".format(self.keyfile)
|
|
||||||
keys += "\n"
|
|
||||||
|
|
||||||
for key in self.keys:
|
|
||||||
key_id = key[0]
|
|
||||||
key_type = key[1]
|
|
||||||
password = key[2]
|
|
||||||
|
|
||||||
if key_type in ["m", "M"]:
|
|
||||||
key_type = "MD5"
|
|
||||||
elif key_type == "AES128CMAC":
|
|
||||||
key_type = "AES128"
|
|
||||||
elif key_type not in ["MD5", "SHA1", "SHA256", "SHA384", "SHA512"]:
|
|
||||||
continue
|
|
||||||
|
|
||||||
prefix = "ASCII" if len(password) <= 20 else "HEX"
|
|
||||||
|
|
||||||
for first, last in self.trusted_keys:
|
|
||||||
if first <= key_id <= last:
|
|
||||||
trusted = True
|
|
||||||
break
|
|
||||||
else:
|
|
||||||
trusted = False
|
|
||||||
|
|
||||||
# Disable keys that were not marked as trusted
|
|
||||||
if not trusted:
|
|
||||||
keys += "#"
|
|
||||||
|
|
||||||
keys += "{} {} {}:{}\n".format(key_id, key_type, prefix, password)
|
|
||||||
|
|
||||||
return keys
|
|
||||||
|
|
||||||
def write_file(self, path, mode, content, backup):
|
|
||||||
path = self.root_dir + path
|
|
||||||
if backup and os.path.isfile(path):
|
|
||||||
os.rename(path, path + ".old")
|
|
||||||
|
|
||||||
with open(os.open(path, os.O_CREAT | os.O_WRONLY | os.O_EXCL, mode), "w",
|
|
||||||
encoding=self.file_encoding) as f:
|
|
||||||
logging.info("Writing %s", path)
|
|
||||||
f.write(u"" + content)
|
|
||||||
|
|
||||||
# Fix SELinux context if restorecon is installed
|
|
||||||
try:
|
|
||||||
subprocess.call(["restorecon", path])
|
|
||||||
except OSError:
|
|
||||||
pass
|
|
||||||
|
|
||||||
|
|
||||||
def main():
|
|
||||||
parser = argparse.ArgumentParser(description="Convert ntp configuration to chrony.")
|
|
||||||
parser.add_argument("-r", "--root", dest="roots", default=["/"], nargs="+",
|
|
||||||
metavar="DIR", help="specify root directory (default /)")
|
|
||||||
parser.add_argument("--ntp-conf", action="store", default="/etc/ntp.conf",
|
|
||||||
metavar="FILE", help="specify ntp config (default /etc/ntp.conf)")
|
|
||||||
parser.add_argument("--step-tickers", action="store", default="",
|
|
||||||
metavar="FILE", help="specify ntpdate step-tickers config (no default)")
|
|
||||||
parser.add_argument("--chrony-conf", action="store", default="/etc/chrony.conf",
|
|
||||||
metavar="FILE", help="specify chrony config (default /etc/chrony.conf)")
|
|
||||||
parser.add_argument("--chrony-keys", action="store", default="/etc/chrony.keys",
|
|
||||||
metavar="FILE", help="specify chrony keyfile (default /etc/chrony.keys)")
|
|
||||||
parser.add_argument("-b", "--backup", action="store_true", help="backup existing configs before writing")
|
|
||||||
parser.add_argument("-L", "--ignored-lines", action="store_true", help="print ignored lines")
|
|
||||||
parser.add_argument("-D", "--ignored-directives", action="store_true",
|
|
||||||
help="print names of ignored directives")
|
|
||||||
parser.add_argument("-n", "--dry-run", action="store_true", help="don't make any changes")
|
|
||||||
parser.add_argument("-v", "--verbose", action="count", default=0, help="increase verbosity")
|
|
||||||
|
|
||||||
args = parser.parse_args()
|
|
||||||
|
|
||||||
logging.basicConfig(format="%(message)s",
|
|
||||||
level=[logging.ERROR, logging.INFO, logging.DEBUG][min(args.verbose, 2)])
|
|
||||||
|
|
||||||
for root in args.roots:
|
|
||||||
conf = NtpConfiguration(root, args.ntp_conf, args.step_tickers)
|
|
||||||
|
|
||||||
if args.ignored_lines:
|
|
||||||
for line in conf.ignored_lines:
|
|
||||||
print(line)
|
|
||||||
|
|
||||||
if args.ignored_directives:
|
|
||||||
for directive in conf.ignored_directives:
|
|
||||||
print(directive)
|
|
||||||
|
|
||||||
conf.write_chrony_configuration(args.chrony_conf, args.chrony_keys, args.dry_run, args.backup)
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
main()
|
|
Loading…
Reference in new issue