Blame SOURCES/chrony.helper

4ed8e1
#!/bin/bash
4ed8e1
# This script configures running chronyd to use NTP servers obtained from
4ed8e1
# DHCP and _ntp._udp DNS SRV records. Files with servers from DHCP are managed
4ed8e1
# externally (e.g. by a dhclient script). Files with servers from DNS SRV
4ed8e1
# records are updated here using the dig utility. The script can also list
4ed8e1
# and set static sources in the chronyd configuration file.
4ed8e1
4ed8e1
chronyc=/usr/bin/chronyc
4ed8e1
chrony_conf=/etc/chrony.conf
4ed8e1
chrony_service=chronyd.service
bdfa49
helper_dir=/run/chrony-helper
4ed8e1
added_servers_file=$helper_dir/added_servers
4ed8e1
4ed8e1
network_sysconfig_file=/etc/sysconfig/network
bdfa49
nm_servers_files="$helper_dir/nm-dhcp.*"
4ed8e1
dhclient_servers_files="/var/lib/dhclient/chrony.servers.*"
4ed8e1
dnssrv_servers_files="$helper_dir/dnssrv@*"
4ed8e1
dnssrv_timer_prefix=chrony-dnssrv@
4ed8e1
4ed8e1
. $network_sysconfig_file &> /dev/null
4ed8e1
4ed8e1
chrony_command() {
8d00b6
    $chronyc -n -m "$@"
4ed8e1
}
4ed8e1
4ed8e1
is_running() {
4ed8e1
    chrony_command "tracking" &> /dev/null
4ed8e1
}
4ed8e1
4ed8e1
get_servers_files() {
bdfa49
    [ "$PEERNTP" != "no" ] && echo "$nm_servers_files"
4ed8e1
    [ "$PEERNTP" != "no" ] && echo "$dhclient_servers_files"
4ed8e1
    echo "$dnssrv_servers_files"
4ed8e1
}
4ed8e1
4ed8e1
is_update_needed() {
4ed8e1
    for file in $(get_servers_files) $added_servers_file; do
4ed8e1
        [ -e "$file" ] && return 0
4ed8e1
    done
4ed8e1
    return 1
4ed8e1
}
4ed8e1
bdfa49
remove_daemon_state() {
bdfa49
    rm -f $added_servers_file
bdfa49
}
bdfa49
4ed8e1
update_daemon() {
4ed8e1
    local all_servers_with_args all_servers added_servers
4ed8e1
4ed8e1
    if ! is_running; then
bdfa49
        remove_daemon_state
4ed8e1
        return 0
4ed8e1
    fi
4ed8e1
4ed8e1
    all_servers_with_args=$(cat $(get_servers_files) 2> /dev/null)
4ed8e1
4ed8e1
    all_servers=$(
4ed8e1
        echo "$all_servers_with_args" |
4ed8e1
            while read -r server serverargs; do
4ed8e1
                echo "$server"
4ed8e1
            done | sort -u)
4ed8e1
    added_servers=$( (
4ed8e1
        cat $added_servers_file 2> /dev/null
4ed8e1
        echo "$all_servers_with_args" |
4ed8e1
            while read -r server serverargs; do
4ed8e1
                [ -z "$server" ] && continue
4ed8e1
                chrony_command "add server $server $serverargs" &> /dev/null &&
4ed8e1
                    echo "$server"
4ed8e1
            done) | sort -u)
4ed8e1
4ed8e1
    comm -23 <(echo -n "$added_servers") <(echo -n "$all_servers") |
4ed8e1
        while read -r server; do
8d00b6
            chrony_command -c sources -a 2>/dev/null |
8d00b6
                    while IFS=, read -r type _ address _; do
8d00b6
                [ "$type" = "^" ] || continue
8d00b6
                [ "$(chrony_command "sourcename $address")" = "$server" ] || continue
8d00b6
                chrony_command "delete $address" &> /dev/null
8d00b6
                break
8d00b6
            done
4ed8e1
        done
4ed8e1
4ed8e1
    added_servers=$(comm -12 <(echo -n "$added_servers") <(echo -n "$all_servers"))
4ed8e1
4ed8e1
    if [ -n "$added_servers" ]; then
4ed8e1
        echo "$added_servers" > $added_servers_file
4ed8e1
    else
4ed8e1
        rm -f $added_servers_file
4ed8e1
    fi
4ed8e1
}
4ed8e1
4ed8e1
get_dnssrv_servers() {
4ed8e1
    local name=$1 output
4ed8e1
4ed8e1
    if ! command -v dig &> /dev/null; then
4ed8e1
        echo "Missing dig (DNS lookup utility)" >&2
4ed8e1
        return 1
4ed8e1
    fi
4ed8e1
4ed8e1
    output=$(dig "$name" srv +short +ndots=2 +search 2> /dev/null) || return 0
4ed8e1
4ed8e1
    echo "$output" | while read -r _ _ port target; do
4ed8e1
        server=${target%.}
4ed8e1
        [ -z "$server" ] && continue
4ed8e1
        echo "$server port $port ${NTPSERVERARGS:-iburst}"
4ed8e1
    done
4ed8e1
}
4ed8e1
4ed8e1
check_dnssrv_name() {
4ed8e1
    local name=$1
4ed8e1
4ed8e1
    if [ -z "$name" ]; then
4ed8e1
        echo "No DNS SRV name specified" >&2
4ed8e1
        return 1
4ed8e1
    fi
4ed8e1
4ed8e1
    if [ "${name:0:9}" != _ntp._udp ]; then
4ed8e1
        echo "DNS SRV name $name doesn't start with _ntp._udp" >&2
4ed8e1
        return 1
4ed8e1
    fi
4ed8e1
}
4ed8e1
4ed8e1
update_dnssrv_servers() {
4ed8e1
    local name=$1
4ed8e1
    local srv_file=$helper_dir/dnssrv@$name servers
4ed8e1
4ed8e1
    check_dnssrv_name "$name" || return 1
4ed8e1
4ed8e1
    servers=$(get_dnssrv_servers "$name")
4ed8e1
    if [ -n "$servers" ]; then
4ed8e1
        echo "$servers" > "$srv_file"
4ed8e1
    else
4ed8e1
        rm -f "$srv_file"
4ed8e1
    fi
4ed8e1
}
4ed8e1
4ed8e1
set_dnssrv_timer() {
4ed8e1
    local state=$1 name=$2
4ed8e1
    local srv_file=$helper_dir/dnssrv@$name servers
4ed8e1
    local timer
4ed8e1
4ed8e1
    timer=$dnssrv_timer_prefix$(systemd-escape "$name").timer || return 1
4ed8e1
4ed8e1
    check_dnssrv_name "$name" || return 1
4ed8e1
4ed8e1
    if [ "$state" = enable ]; then
4ed8e1
        systemctl enable "$timer"
4ed8e1
        systemctl start "$timer"
4ed8e1
    elif [ "$state" = disable ]; then
4ed8e1
        systemctl stop "$timer"
4ed8e1
        systemctl disable "$timer"
4ed8e1
        rm -f "$srv_file"
4ed8e1
    fi
4ed8e1
}
4ed8e1
4ed8e1
list_dnssrv_timers() {
4ed8e1
    systemctl --all --full -t timer list-units | grep "^$dnssrv_timer_prefix" | \
4ed8e1
            sed "s|^$dnssrv_timer_prefix\(.*\)\.timer.*|\1|" |
4ed8e1
        while read -r name; do
4ed8e1
            systemd-escape --unescape "$name"
4ed8e1
        done
4ed8e1
}
4ed8e1
4ed8e1
prepare_helper_dir() {
4ed8e1
    mkdir -p $helper_dir
4ed8e1
    exec 100> $helper_dir/lock
4ed8e1
    if ! flock -w 20 100; then
4ed8e1
        echo "Failed to lock $helper_dir" >&2
4ed8e1
        return 1
4ed8e1
    fi
4ed8e1
}
4ed8e1
4ed8e1
is_source_line() {
4ed8e1
    local pattern="^[ \t]*(server|pool|peer|refclock)[ \t]+[^ \t]+"
4ed8e1
    [[ "$1" =~ $pattern ]]
4ed8e1
}
4ed8e1
4ed8e1
list_static_sources() {
4ed8e1
    while read -r line; do
4ed8e1
        if is_source_line "$line"; then
4ed8e1
            echo "$line"
4ed8e1
        fi
4ed8e1
    done < $chrony_conf
4ed8e1
}
4ed8e1
4ed8e1
set_static_sources() {
4ed8e1
    local new_config tmp_conf
4ed8e1
4ed8e1
    new_config=$(
4ed8e1
        sources=$(
4ed8e1
            while read -r line; do
4ed8e1
                is_source_line "$line" && echo "$line"
4ed8e1
            done)
4ed8e1
4ed8e1
        while read -r line; do
4ed8e1
            if ! is_source_line "$line"; then
4ed8e1
                echo "$line"
4ed8e1
                continue
4ed8e1
            fi
4ed8e1
4ed8e1
            tmp_sources=$(
4ed8e1
                local removed=0
4ed8e1
4ed8e1
                echo "$sources" | while read -r line2; do
4ed8e1
                    if [ "$removed" -ne 0 ] || [ "$line" != "$line2" ]; then
4ed8e1
                        echo "$line2"
4ed8e1
                    else
4ed8e1
                        removed=1
4ed8e1
                    fi
4ed8e1
                done)
4ed8e1
4ed8e1
            [ "$sources" == "$tmp_sources" ] && continue
4ed8e1
            sources=$tmp_sources
4ed8e1
            echo "$line"
4ed8e1
        done < $chrony_conf
4ed8e1
4ed8e1
        echo "$sources"
4ed8e1
    )
4ed8e1
4ed8e1
    tmp_conf=${chrony_conf}.tmp
4ed8e1
4ed8e1
    cp -a $chrony_conf $tmp_conf &&
4ed8e1
        echo "$new_config" > $tmp_conf &&
4ed8e1
        mv $tmp_conf $chrony_conf || return 1
4ed8e1
4ed8e1
    systemctl try-restart $chrony_service
4ed8e1
}
4ed8e1
4ed8e1
print_help() {
4ed8e1
    echo "Usage: $0 COMMAND"
4ed8e1
    echo
4ed8e1
    echo "Commands:"
bdfa49
    echo "	create-helper-directory"
4ed8e1
    echo "	update-daemon"
bdfa49
    echo "	remove-daemon-state"
4ed8e1
    echo "	update-dnssrv-servers NAME"
4ed8e1
    echo "	enable-dnssrv NAME"
4ed8e1
    echo "	disable-dnssrv NAME"
4ed8e1
    echo "	list-dnssrv"
4ed8e1
    echo "	list-static-sources"
4ed8e1
    echo "	set-static-sources < sources.list"
4ed8e1
    echo "	is-running"
4ed8e1
    echo "	command CHRONYC-COMMAND"
4ed8e1
}
4ed8e1
4ed8e1
case "$1" in
bdfa49
    create-helper-directory)
bdfa49
        prepare_helper_dir
bdfa49
        ;;
4ed8e1
    update-daemon|add-dhclient-servers|remove-dhclient-servers)
4ed8e1
        is_update_needed || exit 0
4ed8e1
        prepare_helper_dir && update_daemon
4ed8e1
        ;;
bdfa49
    remove-daemon-state)
bdfa49
        remove_daemon_state
bdfa49
        ;;
4ed8e1
    update-dnssrv-servers)
4ed8e1
        prepare_helper_dir && update_dnssrv_servers "$2" && update_daemon
4ed8e1
        ;;
4ed8e1
    enable-dnssrv)
4ed8e1
        set_dnssrv_timer enable "$2"
4ed8e1
        ;;
4ed8e1
    disable-dnssrv)
4ed8e1
        set_dnssrv_timer disable "$2" && prepare_helper_dir && update_daemon
4ed8e1
        ;;
4ed8e1
    list-dnssrv)
4ed8e1
        list_dnssrv_timers
4ed8e1
        ;;
4ed8e1
    list-static-sources)
4ed8e1
        list_static_sources
4ed8e1
        ;;
4ed8e1
    set-static-sources)
4ed8e1
        set_static_sources
4ed8e1
        ;;
4ed8e1
    is-running)
4ed8e1
        is_running
4ed8e1
        ;;
4ed8e1
    command|forced-command)
4ed8e1
        chrony_command "$2"
4ed8e1
        ;;
4ed8e1
    *)
4ed8e1
        print_help
4ed8e1
        exit 2
4ed8e1
esac
4ed8e1
4ed8e1
exit $?