ntpsweep.in revision 290001
1#! @PATH_PERL@ -w
2#
3# $Id$
4#
5# DISCLAIMER
6# 
7# Copyright (C) 1999,2000 Hans Lambermont and Origin B.V.
8# 
9# Permission to use, copy, modify and distribute this software and its
10# documentation for any purpose and without fee is hereby granted,
11# provided that the above copyright notice appears in all copies and
12# that both the copyright notice and this permission notice appear in
13# supporting documentation. This software is supported as is and without
14# any express or implied warranties, including, without limitation, the
15# implied warranties of merchantability and fitness for a particular
16# purpose. The name Origin B.V. must not be used to endorse or promote
17# products derived from this software without prior written permission.
18#
19# Hans Lambermont <ntpsweep@lambermont.dyndns.org>
20
21package ntpsweep;
22use 5.006_000;
23use strict;
24use lib "@PERLLIBDIR@";
25use NTP::Util qw(do_dns ntp_read_vars ntp_peers ntp_sntp_line);
26
27(my $program = $0) =~ s%.*/(.+?)(.pl)?$%$1%;
28my ($showpeers, $maxlevel, $strip);
29my (%known_host_info, %known_host_peers);
30
31exit run(@ARGV) unless caller;
32
33sub run {
34    my $opts;
35    if (!processOptions(\@_, $opts) || 
36        (((@_ != 1) && !$opts->{host} && !@{$opts->{'host-list'}}))) {
37        usage(1);
38    };
39
40    # no STDOUT buffering
41    $| = 1;
42    ($showpeers, $maxlevel, $strip) = 
43        ($opts->{peers}, $opts->{maxlevel}, $opts->{strip});
44
45    my $hostsfile = shift;
46
47    # Main program
48
49    my @hosts;
50
51    if ($opts->{host}) {
52        push @hosts, $opts->{host};
53    }
54    else {
55        @hosts = read_hosts($hostsfile) if $hostsfile;
56        push @hosts, @{$opts->{'host-list'}};
57    }
58
59    # Print header
60    print <<EOF;
61Host                             st offset(s) version     system       processor
62--------------------------------+--+---------+-----------+------------+---------
63EOF
64
65    %known_host_info = ();
66    %known_host_peers = ();
67    scan_hosts(@hosts);
68
69    return 0;
70}
71
72sub scan_hosts {
73    my (@hosts) = @_;
74
75    my $host;
76    for $host (@hosts) {
77        scan_host($host, 0, $host => 1);
78    }
79}
80
81sub read_hosts {
82    my ($hostsfile) = @_;
83    my @hosts;
84
85    open my $hosts, $hostsfile 
86        or die "$program: FATAL: unable to read $hostsfile: $!\n";
87
88    while (<$hosts>) {
89        next if /^\s*(#|$)/; # comment/empty
90        chomp;
91        push @hosts, $_;
92    }
93
94    close $hosts;
95    return @hosts;
96}
97
98sub scan_host {
99    my ($host, $level, %trace) = @_;
100    my $stratum = 0;
101    my $offset = 0;
102    my $daemonversion = "";
103    my $system = "";
104    my $processor = "";
105    my @peers;
106    my $known_host = 0;
107
108    if (exists $known_host_info{$host}) {
109        $known_host = 1;
110    }
111    else {
112        ($offset, $stratum) = ntp_sntp_line($host);
113
114        # got answers ? If so, go on.
115        if ($stratum) {
116            my $vars = ntp_read_vars(0, [qw(processor system daemon_version)], $host) || {};
117            $daemonversion = $vars->{daemon_version};
118            $system        = $vars->{system};
119            $processor     = $vars->{processor};
120
121            # Shorten daemon_version string.
122            $daemonversion =~ s/(;|Mon|Tue|Wed|Thu|Fri|Sat|Sun).*$//;
123            $daemonversion =~ s/version=//;
124            $daemonversion =~ s/(x|)ntpd //;
125            $daemonversion =~ s/(\(|\))//g;
126            $daemonversion =~ s/beta/b/;
127            $daemonversion =~ s/multicast/mc/;
128
129            # Shorten system string
130            $system =~ s/UNIX\///;
131            $system =~ s/RELEASE/r/;
132            $system =~ s/CURRENT/c/;
133
134            # Shorten processor string
135            $processor =~ s/unknown//;
136        }
137
138        # got answers ? If so, go on.
139        if ($daemonversion) {
140            if ($showpeers) {
141                my $peers_ref = ntp_peers($host);
142                my @peers_tmp = @$peers_ref;
143                for (@peers_tmp) {
144                    $_->{remote} =~ s/^(?: |x|\.|-|\+|#|\*|o)([^ ]+)/$1/;
145                    push @peers, $_->{remote};
146                }
147            }
148        }
149
150        # Add scanned host to known_hosts array
151        #push @known_hosts, $host;
152        if ($stratum) {
153            $known_host_info{$host} = sprintf "%2d %9.3f %-11s %-12s %s",
154                $stratum, $offset, (substr $daemonversion, 0, 11),
155                (substr $system, 0, 12), (substr $processor, 0, 9);
156        }
157        else {
158            # Stratum level 0 is consider invalid
159            $known_host_info{$host} = " ?";
160        }
161        $known_host_peers{$host} = [@peers];
162    }
163
164    if ($stratum || $known_host) { # Valid or known host
165        my $printhost = ' ' x $level . (do_dns($host) || $host);
166        # Shorten host string
167        if ($strip) {
168            $printhost =~ s/$strip//;
169        }
170        # append number of peers in brackets if requested and valid
171        if ($showpeers && ($known_host_info{$host} ne " ?")) {
172            $printhost .= " (" . @{$known_host_peers{$host}} . ")";
173        }
174        # Finally print complete host line
175        printf "%-32s %s\n",
176            (substr $printhost, 0, 32), $known_host_info{$host};
177        if ($showpeers && ($maxlevel ? $level < $maxlevel : 1)) {
178            $trace{$host} = 1;
179            # Loop through peers
180            foreach my $peer (@{$known_host_peers{$host}}) {
181                if (exists $trace{$peer}) {
182                    # we've detected a loop !
183                    $printhost = ' ' x ($level + 1) . "= " . $peer;
184                    # Shorten host string
185                    $printhost =~ s/$strip// if $strip;
186                    printf "%-32s\n", substr $printhost, 0, 32;
187                } else {
188                    if ((substr $peer, 0, 3) ne "127") {
189                        scan_host($peer, $level + 1, %trace);
190                    }
191                }
192            }
193        }
194    }
195    else { # We did not get answers from this host
196        my $printhost = ' ' x $level . (do_dns($host) || $host);
197        $printhost =~ s/$strip// if $strip;
198        printf "%-32s  ?\n", substr $printhost, 0, 32;
199    }
200}
201
202@ntpsweep_opts@
203
2041;
205__END__
206