summaryrefslogtreecommitdiffstats
path: root/scripts/ntp-monlist.nse
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--scripts/ntp-monlist.nse1036
1 files changed, 1036 insertions, 0 deletions
diff --git a/scripts/ntp-monlist.nse b/scripts/ntp-monlist.nse
new file mode 100644
index 0000000..3ebe193
--- /dev/null
+++ b/scripts/ntp-monlist.nse
@@ -0,0 +1,1036 @@
+local ipOps = require "ipOps"
+local math = require "math"
+local nmap = require "nmap"
+local packet = require "packet"
+local shortport = require "shortport"
+local stdnse = require "stdnse"
+local string = require "string"
+local table = require "table"
+
+author = "jah"
+license = "Same as Nmap--See https://nmap.org/book/man-legal.html"
+categories = {"discovery", "intrusive"}
+description = [[
+Obtains and prints an NTP server's monitor data.
+
+Monitor data is a list of the most recently used (MRU) having NTP associations
+with the target. Each record contains information about the most recent NTP
+packet sent by a host to the target including the source and destination
+addresses and the NTP version and mode of the packet. With this information it
+is possible to classify associated hosts as Servers, Peers, and Clients.
+
+A Peers command is also sent to the target and the peers list in the response
+allows differentiation between configured Mode 1 Peers and clients which act
+like Peers (such as the Windows W32Time service).
+
+Associated hosts are further classified as either public or private.
+Private hosts are those
+having IP addresses which are not routable on the public Internet and thus can
+help to form a picture about the topology of the private network on which the
+target resides.
+
+Other information revealed by the monlist and peers commands are the host with
+which the target clock is synchronized and hosts which send Control Mode (6)
+and Private Mode (7) commands to the target and which may be used by admins for
+the NTP service.
+
+It should be noted that the very nature of the NTP monitor data means that the
+Mode 7 commands sent by this script are recorded by the target (and will often
+appear in these results). Since the monitor data is a MRU list, it is probable
+that you can overwrite the record of the Mode 7 command by sending an innocuous
+looking Client Mode request. This can be achieved easily using Nmap:
+<code>nmap -sU -pU:123 -Pn -n --max-retries=0 <target></code>
+
+Notes:
+* The monitor list in response to the monlist command is limited to 600 associations.
+* The monitor capability may not be enabled on the target in which case you may receive an error number 4 (No Data Available).
+* There may be a restriction on who can perform Mode 7 commands (e.g. "restrict noquery" in <code>ntp.conf</code>) in which case you may not receive a reply.
+* This script does not handle authenticating and targets expecting auth info may respond with error number 3 (Format Error).
+]]
+
+---
+-- @usage
+-- nmap -sU -pU:123 -Pn -n --script=ntp-monlist <target>
+--
+-- @output
+-- PORT STATE SERVICE REASON
+-- 123/udp open ntp udp-response
+-- | ntp-monlist:
+-- | Target is synchronised with 127.127.38.0 (reference clock)
+-- | Alternative Target Interfaces:
+-- | 10.17.4.20
+-- | Private Servers (0)
+-- | Public Servers (0)
+-- | Private Peers (0)
+-- | Public Peers (0)
+-- | Private Clients (2)
+-- | 10.20.8.69 169.254.138.63
+-- | Public Clients (597)
+-- | 4.79.17.248 68.70.72.194 74.247.37.194 99.190.119.152
+-- | ...
+-- | 12.10.160.20 68.80.36.133 75.1.39.42 108.7.58.118
+-- | 68.56.205.98
+-- | 2001:1400:0:0:0:0:0:1 2001:16d8:dd00:38:0:0:0:2
+-- | 2002:db5a:bccd:1:21d:e0ff:feb7:b96f 2002:b6ef:81c4:0:0:1145:59c5:3682
+-- | Other Associations (1)
+-- |_ 127.0.0.1 seen 1949869 times. last tx was unicast v2 mode 7
+
+-- This script uses the NTP sequence numbers and the 'more' bit found in
+-- response packets in order to determine when to stop the reception loop. It
+-- would be possible for a malicious target to tie-up this script by sending
+-- a continuous stream of UDP datagrams.
+-- Therefore MAXIMUM_EVIL has been defined to limit the number of malformed or
+-- duplicate packets that will be processed before a target is rejected and
+-- MAX_RECORDS simply limits the storage of valid looking NTP data to a sane
+-- level.
+local MAXIMUM_EVIL = 25
+local MAX_RECORDS = 1200
+
+local TIMEOUT = 5000 -- ms
+
+
+---
+-- ntp-monlist will run against the ntp service which only runs on UDP 123
+--
+portrule = shortport.port_or_service(123, 'ntp', {'udp'})
+
+---
+-- Send an NTPv2 Mode 7 'monlist' command to the target, receive any responses
+-- and parse records from those responses. If the target responds favourably
+-- then send a 'peers' command and parse the responses. Finally, categorise the
+-- discovered NTP associations (hosts) and output the interpreted results.
+--
+action = function(host, port)
+
+ -- Define the request code and implementation numbers of the NTP request to
+ -- send to the target.
+ local REQ_MON_GETLIST_1 = 42
+ local REQ_PEER_LIST = 0
+ local IMPL_XNTPD = 3
+
+ -- parsed records will be stored in this table.
+ local records = {['peerlist'] = {}}
+
+ -- send monlist command and fill the records table with the responses.
+ local inum, rcode = IMPL_XNTPD, REQ_MON_GETLIST_1
+ local sock = doquery(nil, host, port, inum, rcode, records)
+
+ -- end here if there are zero records.
+ if #records == 0 then
+ if sock then sock:close() end
+ return nil
+ end
+
+ -- send peers command and add the responses to records.peerlist.
+ rcode = REQ_PEER_LIST
+ sock = doquery(sock, host, port, inum, rcode, records)
+ if sock then sock:close() end
+
+ -- now we can interpret the collected records.
+ local interpreted = interpret(records, host.ip)
+
+ -- output.
+ return summary(interpreted)
+
+end
+
+
+---
+-- Sends NTPv2 Mode 7 requests to the target, receives any responses and parses
+-- records from those responses.
+--
+-- @param sock Socket object which must be supplied in a connected state.
+-- nil may be supplied instead and a socket will be created.
+-- @param host Nmap host table for the target.
+-- @param port Nmap port table for the target.
+-- @param inum NTP implementation number (i.e. 0, 2 or 3).
+-- @param rcode NTP Mode 7 request code (e.g. 42 for 'monlist').
+-- @param records Table in which to store records parsed from responses.
+-- @return sock Socket object connected to the target.
+--
+function doquery(sock, host, port, inum, rcode, records)
+
+ local target = ('%s%s%d'):format(
+ host.ip, host.ip:match(':') and '.' or ':', port.number
+ )
+ records.badpkts = records.badpkts or 0
+ records.peerlist = records.peerlist or {}
+
+ if #records + #records.peerlist >= MAX_RECORDS then
+ stdnse.debug1('MAX_RECORDS has been reached for target %s - only processing what we have already!', target)
+ if sock then sock:close() end
+ return nil
+ end
+
+ -- connect a new socket if one wasn't supplied
+ if not sock then
+ sock = nmap.new_socket()
+ sock:set_timeout(TIMEOUT)
+ local constatus, conerr = sock:connect(host, port)
+ if not constatus then
+ stdnse.debug1('Error establishing a UDP connection for %s - %s', target, conerr)
+ return nil
+ end
+ end
+
+ -- send
+ stdnse.debug2('Sending NTPv2 Mode 7 Request %d Implementation %d to %s.', rcode, inum, target)
+ local ntpData = getPrivateMode(inum, rcode)
+ local sendstatus, senderr = sock:send(ntpData)
+ if not sendstatus then
+ stdnse.debug1('Error sending NTP request to %s:%d - %s', host.ip, port.number, senderr)
+ sock:close()
+ return nil
+ end
+
+ local track = {
+ ['evil_pkts'] = records.badpkts, -- a count of bad packets
+ ['hseq'] = -1, -- highest received seq number
+ ['mseq'] = '|', -- missing seq numbers
+ ['errcond'] = false, -- true if sock, ntp or sane response error exists
+ ['rcv_again'] = false, -- true if we should receive_bytes again (more bit is set or missing seq).
+ ['target'] = target, -- target ip and port
+ ['v'] = 2, -- ntp version
+ ['m'] = 7, -- ntp mode
+ ['c'] = rcode, -- ntp request code
+ ['i'] = inum -- ntp request implementation number
+ }
+
+ -- receive and parse
+ repeat
+ -- receive any response
+ local rcvstatus, response = sock:receive_bytes(1)
+ -- check the response
+ local packet_to_parse = check(rcvstatus, response, track)
+ -- parse the response
+ if not track.errcond then
+ local remain = parse_v2m7(packet_to_parse, records)
+ if remain > 0 then
+ stdnse.debug1('MAX_RECORDS has been reached while parsing NTPv2 Mode 7 Code %d responses from the target %s.', rcode, target)
+ track.rcv_again = false
+ elseif remain == -1 then
+ stdnse.debug1('Parsing of NTPv2 Mode 7 implementation number %d request code %d response from %s has not been implemented.', inum, rcode, target)
+ track.rcv_again = false
+ end
+ end
+ records.badpkts = records.badpkts + track.evil_pkts
+ if records.badpkts >= MAXIMUM_EVIL then
+ stdnse.debug1('Had %d bad packets from %s - Not continuing with this host!', target, records.badpkts)
+ sock:close()
+ return nil
+ end
+
+ until not track.rcv_again
+
+ return sock
+
+end
+
+
+---
+-- Generates an NTP Private Mode (7) request with the supplied implementation
+-- number and request code.
+--
+-- @param impl number - valid values are 0, 2 and 3 - defaults to 3
+-- @param requestCode number - defaults to 42
+-- @return String request.
+--
+function getPrivateMode(impl, requestCode)
+ local pay
+ -- udp payload is 48 bytes.
+ -- For a description of Mode 7 packets see NTP source file:
+ -- include/ntp_request.h
+ --
+ -- Flags 8bits: 0x17
+ -- (Response Bit: 0, More Bit: 0, Version Number 3bits: 2, Mode 3bits: 7)
+ -- Authenticated Bit: 0, Sequence Number 7bits: 0
+ -- Implementation Number 8bits: e.g. 0x03 (IMPL_XNTPD)
+ -- Request Code 8bits: e.g. 0x2a (MON_GETLIST_1)
+ -- Err 4bits: 0, Number of Data Items 12bits: 0
+ -- MBZ 4bits: 0, Size of Data Items 12bits: 0
+ return string.char(
+ 0x17, 0x00, impl or 0x03, requestCode or 0x2a,
+ 0x00, 0x00, 0x00, 0x00
+ )
+ -- Data 40 Octets: 0
+ .. ("\x00"):rep(40)
+ -- The following are optional if the Authenticated bit is set:
+ -- Encryption Keyid 4 Octets: 0
+ -- Message Authentication Code 16 Octets (MD5): 0
+end
+
+
+---
+-- Checks that the response from the target is a valid NTP response.
+--
+-- Starts by checking that the socket read was successful and then creates a
+-- packet object from the response (with dummy IP and UDP headers). Then
+-- perform checks that ensure that the response is of the expected type and
+-- length, that the records in the response are of the correct size and that
+-- the response is part of a sequence of 1 or more responses and is not a
+-- duplicate.
+--
+-- @param status boolean returned from a socket read operation.
+-- @param response string response returned from a socket operation.
+-- @param track table used for tracking a sequence of NTP responses.
+-- @return A Packet object ready for parsing or
+-- nil if the response does not pass all checks.
+--
+function check(status, response, track)
+
+ -- check for socket error
+ if not status then
+ track.errcond = true
+ track.rcv_again = false
+ if track.rcv_again then -- we were expecting more responses
+ stdnse.debug1('Socket error while reading from %s - %s', track.target, response)
+ end
+ return nil
+ end
+
+ -- reset flags
+ track.errcond = false
+ track.rcv_again = false
+
+ -- create a packet object
+ local pkt = make_udp_packet(response)
+ if pkt == nil then
+ track.errcond = true
+ track.evil_pkts = track.evil_pkts+1
+ stdnse.debug1('Failed to create a Packet object with response from %s', track.target)
+ return nil
+ end
+
+ -- off is the start of udp payload i.e. NTP
+ local off = 28
+
+ -- NTP sanity checks
+
+ local val
+
+ -- NTP data must be at least 8 bytes
+ val = response:len()
+ if val < 8 then
+ track.errcond = true
+ track.evil_pkts = track.evil_pkts+1
+ stdnse.debug1('Expected a response of at least 8 bytes from %s, got %d bytes.', track.target, val)
+ return nil
+ end
+
+ -- response bit set
+ if (pkt:u8(off) >> 7) ~= 1 then
+ track.errcond = true
+ track.evil_pkts = track.evil_pkts+1
+ stdnse.debug1('Bad response from %s - did not have response bit set.', track.target)
+ return nil
+ end
+ -- version is as expected
+ val = (pkt:u8(off) >> 3) & 0x07
+ if val ~= track.v then
+ track.errcond = true
+ track.evil_pkts = track.evil_pkts+1
+ stdnse.debug1('Bad response from %s - expected NTP version %d, got %d', track.target, track.v, val)
+ return nil
+ end
+ -- mode is as expected
+ val = pkt:u8(off) & 0x07
+ if val ~= track.m then
+ track.errcond = true
+ track.evil_pkts = track.evil_pkts+1
+ stdnse.debug1('Bad response from %s - expected NTP mode %d, got %d', track.target, track.m, val)
+ return nil
+ end
+ -- implementation number is as expected
+ val = pkt:u8(off+2)
+ if val ~= track.i then
+ track.errcond = true
+ track.evil_pkts = track.evil_pkts+1
+ stdnse.debug1('Bad response from %s - expected NTP implementation number %d, got %d', track.target, track.i, val)
+ return nil
+ end
+ -- request code is as expected
+ val = pkt:u8(off+3)
+ if val ~= track.c then
+ track.errcond = true
+ track.evil_pkts = track.evil_pkts+1
+ stdnse.debug1('Bad response from %s - expected NTP request code %d got %d.', track.target, track.c, val)
+ return nil
+ end
+ -- NTP error conditions - defined codes are not evil (bogus codes are).
+ local fail, msg = false
+ local err = (pkt:u8(off+4) >> 4) & 0x0f
+ if err == 0 then
+ -- NoOp
+ elseif err == 1 then
+ fail = true
+ msg = 'Incompatible Implementation Number'
+ elseif err == 2 then
+ fail = true
+ msg = 'Unimplemented Request Code'
+ elseif err == 3 then
+ fail = true
+ msg = 'Format Error' -- could be that auth is required - we didn't provide it.
+ elseif err == 4 then
+ fail = true
+ msg = 'No Data Available' -- monitor not enabled or nothing in mru list.
+ elseif err == 5 or err == 6 then
+ fail = true
+ msg = 'I don\'t know'
+ elseif err == 7 then
+ fail = true
+ msg = 'Authentication Failure'
+ elseif err > 7 then
+ fail = true
+ track.evil_pkts = track.evil_pkts+1
+ msg = 'Bogus Error Code!' -- should not happen...
+ end
+ if fail then
+ track.errcond = true
+ stdnse.debug1('Response from %s was NTP Error Code %d - "%s"', track.target, err, msg)
+ return nil
+ end
+
+ -- length checks - the data (number of items * size of an item) should be
+ -- 8 <= data <= 500 and each data item should be of correct length for the
+ -- implementation and request type.
+
+ -- Err 4 bits, Number of Data Items 12 bits
+ local icount = pkt:u16(off+4) & 0xFFF
+ -- MBZ 4 bits, Size of Data Items: 12 bits
+ local isize = pkt:u16(off+6) & 0xFFF
+ if icount < 1 then
+ track.errcond = true
+ track.evil_pkts = track.evil_pkts+1
+ stdnse.debug1('Expected at least one record from %s.', track.target)
+ return nil
+ elseif icount*isize + 8 > response:len() then
+ track.errcond = true
+ track.evil_pkts = track.evil_pkts+1
+ stdnse.debug1('NTP Mode 7 response from %s has invalid count (%d) and/or size (%d) values.', track.target, icount, isize)
+ return nil
+ elseif icount*isize > 500 then
+ track.errcond = true
+ track.evil_pkts = track.evil_pkts+1
+ stdnse.debug1('NTP Mode 7 data section is larger than 500 bytes (%d) in response from %s.', icount*isize, track.target)
+ return nil
+ end
+
+ if track.c == 42 and track.i == 3 and isize ~= 72 then
+ track.errcond = true
+ track.evil_pkts = track.evil_pkts+1
+ stdnse.debug1(
+ 'Expected item size of 72 bytes (got %d) for request code 42 implementation number 3 in response from %s.',
+ isize, track.target
+ )
+ return nil
+ elseif track.c == 0 and track.i == 3 and isize ~= 32 then
+ track.errcond = true
+ track.evil_pkts = track.evil_pkts+1
+ stdnse.debug1(
+ 'Expected item size of 32 bytes (got %d) for request code 0 implementation number 3 in response from %s.',
+ isize, track.target
+ )
+ return nil
+ end
+
+ -- is the response out of sequence, a duplicate or is it peachy
+ local seq = pkt:u8(off+1) & 0x7f
+ if seq == track.hseq+1 then -- all good
+ track.hseq = track.hseq+1
+ elseif track.mseq:match(('|%d|'):format(seq)) then -- one of our missing seq#
+ track.mseq:gsub(('|%d|'):format(seq), '|', 1)
+ stdnse.debug3('Response from %s with sequence number %s was previously missing.', -- this never seems to happen!
+ track.target, seq
+ )
+ elseif seq > track.hseq then -- some seq# have gone missing
+ for i=track.hseq+1, seq-1 do
+ track.mseq = ('%s%d|'):format(track.mseq, i)
+ end
+ stdnse.debug3(
+ 'Response from %s was out of sequence - expected #%d but got #%d (missing:%s)',
+ track.target, track.hseq+1, seq, track.mseq
+ )
+ track.hseq = seq
+ else -- seq <= hseq !duplicate!
+ track.evil_pkts = track.evil_pkts+1
+ stdnse.debug1(
+ 'Response from %s had a duplicate sequence number - dropping it.',
+ track.target
+ )
+ return nil
+ end
+
+ -- if the more bit is set or if we have missing sequence numbers then we'll
+ -- want to receive more packets after parsing this one.
+ local more = (pkt:u8(off) >> 6) & 0x01
+ if more == 1 then
+ track.rcv_again = true
+ elseif track.mseq:len() > 1 then
+ track.rcv_again = true
+ end
+
+ return pkt
+
+end
+
+
+---
+-- Returns a Packet Object generated with dummy IP and UDP headers and the
+-- supplied UDP payload so that the payload may be conveniently parsed using
+-- packet library methods. The dummy headers contain the barest information
+-- needed to appear valid to packet.lua
+--
+-- @param response String UDP payload.
+-- @return Packet object or nil in case of an error.
+--
+function make_udp_packet(response)
+
+ -- udp len
+ local udplen = 8 + response:len()
+ -- ip len
+ local iplen = 20 + udplen
+
+ -- dummy headers
+ -- ip
+ local dh = "\x45\x00" -- IPv4, 20-byte header, no DSCP, no ECN
+ .. string.pack('>I2', iplen) -- total length
+ .. "\x00\x00" -- IPID 0
+ .. "\x40\x00" -- DF
+ .. "\x40\x11" -- TTL 0x40, UDP (proto 17)
+ .. "\x00\x00" -- checksum 0
+ .. "\x00\x00\x00\x00\x00\x00\x00\x00" -- Source, destination 0.0.0.0
+ .. "\x00\x00\x00\x00" -- UDP source, dest port 0
+ .. string.pack('>I2', udplen) -- UDP length
+ .. "\x00\x00" -- UDP checksum 0
+
+ return packet.Packet:new(dh .. response, iplen)
+
+end
+
+
+---
+-- Invokes parsing routines for NTPv2 Mode 7 response packets based on the
+-- implementation number and request code defined in the response.
+--
+-- @param pkt Packet Object to be parsed.
+-- @param recs Table to hold the accumulated records parsed from supplied
+-- packet objects.
+-- @return Number of records not parsed from the packet (usually zero) or
+-- -1 if the response does not have an associated parsing routine.
+--
+function parse_v2m7(pkt, recs)
+ local off = pkt.udp_offset + 8
+ local impl = pkt:u8(off+2)
+ local code = pkt:u8(off+3)
+ if (impl == 3 or impl == 2) and code == 42 then
+ return parse_monlist_1(pkt, recs)
+ elseif (impl == 3 or impl == 2) and code == 0 then
+ return parse_peerlist(pkt, recs)
+ else
+ return -1
+ end
+end
+
+
+---
+-- Parsed records from the supplied monitor list packet into the supplied table
+-- of accumulated records.
+--
+-- The supplied response packets should be NTPv2 Mode 7 implementation number 2
+-- or 3 and request code 42.
+-- The fields parsed are the source and destination IP addresses, the count of
+-- times the target has seen the host, the method of transmission (uni|broad|
+-- multicast), NTP Version and Mode of the last packet received by the target
+-- from the host.
+--
+-- @param pkt Packet object to extract monitor records from.
+-- @param recs A table of accumulated monitor records for storage of parsed
+-- records.
+-- @return Number of records not parsed from the packet which will be zero
+-- except when MAX_RECORDS is reached.
+--
+function parse_monlist_1(pkt, recs)
+
+ local off = pkt.udp_offset + 8 -- beginning of NTP
+ local icount = pkt:u16(off+4) & 0xFFF
+ local isize = pkt:u16(off+6) & 0xFFF
+ local remaining = icount
+
+ off = off+8 -- beginning of data section
+
+ for i=1, icount, 1 do
+ if #recs + #recs.peerlist >= MAX_RECORDS then
+ return remaining
+ end
+ local pos = off + isize * (i-1) -- beginning of item
+ local t = {}
+
+ -- src and dst addresses
+ -- IPv4 if impl == 2 or v6 flag is not set
+ if isize == 32 or pkt:u8(pos+32) ~= 1 then -- IPv4
+ local saddr = ipOps.str_to_ip(pkt:raw(pos+16, 4))
+ local daddr = ipOps.str_to_ip(pkt:raw(pos+20, 4))
+ t.saddr = saddr
+ t.daddr = daddr
+ else -- IPv6
+ local saddr = {}
+ for j=40, 54, 2 do
+ saddr[#saddr+1] = stdnse.tohex(pkt:u16(pos+j))
+ end
+ t.saddr = table.concat(saddr, ':')
+ local daddr = {}
+ for j=56, 70, 2 do
+ daddr[#daddr+1] = stdnse.tohex(pkt:u16(pos+j))
+ end
+ t.daddr = table.concat(daddr, ':')
+ end
+
+ t.count = pkt:u32(pos+12)
+ t.flags = pkt:u32(pos+24)
+ -- I've seen flags be wrong-endian just once. why? I really don't know.
+ -- Some implementations are not doing htonl for this field?
+ if t.flags > 0xFFFFFF then
+ -- only concerned with the high order byte
+ t.flags = t.flags >> 24
+ end
+ t.mode = pkt:u8(pos+30)
+ t.version = pkt:u8(pos+31)
+ recs[#recs+1] = t
+ remaining = remaining -1
+ end
+
+ return remaining
+end
+
+
+---
+-- Parsed records from the supplied peer list packet into the supplied table of
+-- accumulated records.
+--
+-- The supplied response packets should be NTPv2 Mode 7 implementation number 2
+-- or 3 and request code 0.
+-- The fields parsed are the source IP address and the peer information flag.
+--
+-- @param pkt Packet object to extract peer records from.
+-- @param recs A table of accumulated monitor and peer records for storage of
+-- parsed records.
+-- @return Number of records not parsed from the packet which will be zero
+-- except when MAX_RECORDS is reached.
+--
+function parse_peerlist(pkt, recs)
+
+ local off = pkt.udp_offset + 8 -- beginning of NTP
+ local icount = pkt:u16(off+4) & 0xFFF
+ local isize = pkt:u16(off+6) & 0xFFF
+ local remaining = icount
+
+ off = off+8 -- beginning of data section
+
+ for i=0, icount-1, 1 do
+ if #recs + #recs.peerlist >= MAX_RECORDS then
+ return remaining
+ end
+ local pos = off + (i * isize) -- beginning of item
+ local t = {}
+
+ -- src address
+ -- IPv4 if impl == 2 or v6 flag is not set
+ if isize == 8 or pkt:u8(pos+8) ~= 1 then
+ local saddr = ipOps.str_to_ip(pkt:raw(pos, 4))
+ t.saddr = saddr
+ else -- IPv6
+ local saddr = {}
+ for j=16, 30, 2 do
+ saddr[#saddr+1] = stdnse.tohex(pkt:u16(pos+j))
+ end
+ t.saddr = table.concat(saddr, ':')
+ end
+
+ t.flags = pkt:u8(pos+7)
+ table.insert(recs.peerlist, t)
+ remaining = remaining -1
+ end
+
+ return remaining
+end
+
+
+---
+-- Interprets the supplied records to discover information about the target
+-- NTP associations.
+--
+-- Associations are categorised as NTP Servers, Peers and Clients based on the
+-- Mode of packets sent to the target. Alternative target interfaces are
+-- recorded as well as the transmission mode of packets sent to the target (i.e.
+-- unicast, broadcast or multicast).
+--
+-- @param recs A table of accumulated monitor and peer records for storage
+-- of parsed records.
+-- @param targetip String target IP address (e.g. host.ip)
+-- @return Table of interpreted results with fields such as servs, clients,
+-- peers, ifaces etc.
+--
+function interpret(recs, targetip)
+ local txtyp = {
+ ['1'] = 'unicast',
+ ['2'] = 'broadcast',
+ ['4'] = 'multicast'
+ }
+ local t = {}
+ t.servs = {['pub']={['4']={},['6']={}}, ['prv']={['4']={},['6']={}}}
+ t.peers = {['pub']={['4']={},['6']={}}, ['prv']={['4']={},['6']={}}}
+ t.porc = {['pub']={['4']={},['6']={}}, ['prv']={['4']={},['6']={}}}
+ t.clients = {['pub']={['4']={},['6']={}}, ['prv']={['4']={},['6']={}}}
+ t.casts = {['b']={['4']={},['6']={}}, ['m']={['4']={},['6']={}}}
+ t.ifaces = {['4']={},['6']={}}
+ t.other = {}
+ t.sync = ''
+ if #recs.peerlist > 0 then
+ t.have_peerlist = true
+ recs.peerhash = {}
+ for _, peer in ipairs(recs.peerlist) do
+ recs.peerhash[peer.saddr] = peer
+ end
+ else
+ t.have_peerlist = false
+ end
+
+ for _, r in ipairs(recs) do
+ local vis = ipOps.isPrivate(r.saddr) and 'prv' or 'pub'
+ local af = r.saddr:match(':') and '6' or '4'
+
+ -- is the host a client, peer, server or other?
+ if r.mode == 3 then
+ table.insert(t.clients[vis][af], r.saddr)
+ elseif r.mode == 4 then
+ table.insert(t.servs[vis][af], r.saddr)
+ elseif r.mode == 2 then
+ table.insert(t.peers[vis][af], r.saddr)
+ elseif r.mode == 1 then
+
+ -- if we have a list of peers we can distinguish between mode 1 peers and
+ -- mode 1 peers that are really clients (i.e. not configured as peers).
+ if t.have_peerlist then
+ if recs.peerhash[r.saddr] then
+ table.insert(t.peers[vis][af], r.saddr)
+ else
+ table.insert(t.clients[vis][af], r.saddr)
+ end
+ else
+ table.insert(t.porc[vis][af], r.saddr)
+ end
+
+ elseif r.mode == 5 then
+ table.insert(t.servs[vis][af], r.saddr)
+ else
+ local tx = tostring(r.flags)
+ table.insert(
+ t.other,
+ ('%s%s seen %d time%s. last tx was %s v%d mode %d'):format(
+ r.saddr, _ == 1 and ' (You?)' or '', r.count,
+ r.count > 1 and 's' or '',
+ txtyp[tx] or tx, r.version, r.mode
+ )
+ )
+ end
+
+ local function isLoopback(addr)
+ if addr:match(':') then
+ if ipOps.compare_ip(addr, 'eq', '::1') then return true end
+ elseif addr:match('^127') then
+ return true
+ end
+ return false
+ end
+
+ -- destination addresses are target interfaces or broad/multicast addresses.
+ if not isLoopback(r.daddr) then
+ if r.flags == 1 then
+ t.ifaces[af][r.daddr] = r.daddr
+ elseif r.flags == 2 then
+ t.casts.b[af][r.daddr] = r.daddr
+ elseif r.flags == 4 then
+ t.casts.m[af][r.daddr] = r.daddr
+ else -- shouldn't happen
+ stdnse.debug1(
+ 'Host associated with %s had transmission flag value %d - Strange!',
+ targetip, r.flags
+ )
+ end
+ end
+
+ end -- for
+
+ local function isTarget(addr, target)
+ local targ_af = target:match(':') and 6 or 4
+ local test_af = addr:match(':') and 6 or 4
+ if test_af ~= targ_af then return false end
+ if targ_af == 4 and addr == target then return true end
+ if targ_af == 6
+ and (ipOps.compare_ip(addr, 'eq', target)) then return true end
+ return false
+ end
+
+ -- reorganise ifaces and casts tables
+ local _ = {}
+ for k, v in pairs(t.ifaces['4']) do
+ if not isTarget(v, targetip) then
+ _[#_+1] = v
+ end
+ end
+ t.ifaces['4'] = _
+ _ = {}
+ for k, v in pairs(t.ifaces['6']) do
+ if not isTarget(v, targetip) then
+ _[#_+1] = v
+ end
+ end
+ t.ifaces['6'] = _
+ _ = {}
+ for k, v in pairs(t.casts.b['4']) do
+ _[#_+1] = v
+ end
+ t.casts.b['4'] = _
+ _ = {}
+ for k, v in pairs(t.casts.b['6']) do
+ _[#_+1] = v
+ end
+ t.casts.b['6'] = _
+ _ = {}
+ for k, v in pairs(t.casts.m['4']) do
+ _[#_+1] = v
+ end
+ t.casts.m['4'] = _
+ _ = {}
+ for k, v in pairs(t.casts.m['6']) do
+ _[#_+1] = v
+ end
+ t.casts.m['6'] = _
+
+ -- Single out the server to which the target is synched.
+ -- Note that this server may not even appear in the monlist - it depends how
+ -- busy the server is.
+ if t.have_peerlist then
+ for _, peer in ipairs(recs.peerlist) do
+ if (peer.flags & 0x2) == 0x2 then
+ t.sync = peer.saddr
+ if peer.saddr:match('^127') then -- always IPv4, never IPv6!
+ t.sync = t.sync .. ' (reference clock)'
+ end
+ break
+ end
+ end
+ end
+
+ return t
+
+end
+
+
+---
+-- Outputs the supplied table of interpreted records.
+--
+-- @param t Table of interpreted records as returned from interpret().
+-- @return String script output.
+--
+function summary(t)
+
+ local o = {}
+ local count = 0
+ local vbs = nmap.verbosity()
+
+ -- Target is Synchronised with:
+ if t.sync ~= '' then
+ table.insert(o, ('Target is synchronised with %s'):format(t.sync))
+ end
+
+ -- Alternative Target Interfaces
+ if #t.ifaces['4'] > 0 or #t.ifaces['6'] > 0 then
+ table.insert(o,
+ {
+ ['name'] = 'Alternative Target Interfaces:',
+ output_ips(t.ifaces)
+ }
+ )
+ end
+
+ -- Target listens to Broadcast Addresses
+ if #t.casts.b['4'] > 0 or #t.casts.b['6'] > 0 then
+ table.insert(o,
+ {
+ ['name'] = 'Target listens to Broadcast Addresses:',
+ output_ips(t.casts.b)
+ }
+ )
+ end
+
+ -- Target listens to Multicast Addresses
+ if #t.casts.m['4'] > 0 or #t.casts.m['6'] > 0 then
+ table.insert(o,
+ {
+ ['name'] = 'Target listens to Multicast Addresses:',
+ output_ips(t.casts.m)
+ }
+ )
+ end
+
+ -- Private Servers
+ count = #t.servs.prv['4']+#t.servs.prv['6']
+ if count > 0 or vbs > 1 then
+ table.insert(o,
+ {
+ ['name'] = ('Private Servers (%d)'):format(count),
+ output_ips(t.servs.prv)
+ }
+ )
+ end
+ -- Public Servers
+ count = #t.servs.pub['4']+#t.servs.pub['6']
+ if count > 0 or vbs > 1 then
+ table.insert(o,
+ {
+ ['name'] = ('Public Servers (%d)'):format(count),
+ output_ips(t.servs.pub)
+ }
+ )
+ end
+
+ -- Private Peers
+ count = #t.peers.prv['4']+#t.peers.prv['6']
+ if count > 0 or vbs > 1 then
+ table.insert(o,
+ {
+ ['name'] = ('Private Peers (%d)'):format(count),
+ output_ips(t.peers.prv)
+ }
+ )
+ end
+ -- Public Peers
+ count = #t.peers.pub['4']+#t.peers.pub['6']
+ if count > 0 or vbs > 1 then
+ table.insert(o,
+ {
+ ['name'] = ('Public Peers (%d)'):format(count),
+ output_ips(t.peers.pub)
+ }
+ )
+ end
+
+ -- Private Peers or Clients
+ count = #t.porc.prv['4']+#t.porc.prv['6']
+ if not t.have_peerlist and (count > 0 or vbs > 1) then
+ table.insert(o,
+ {
+ ['name'] = ('Private Peers or Clients (%d)'):format(count),
+ output_ips(t.porc.prv)
+ }
+ )
+ end
+ -- Public Peers or Clients
+ count = #t.porc.pub['4']+#t.porc.pub['6']
+ if not t.have_peerlist and (count > 0 or vbs > 1) then
+ table.insert(o,
+ {
+ ['name'] = ('Public Peers or Clients (%d)'):format(count),
+ output_ips(t.porc.pub)
+ }
+ )
+ end
+
+ -- Private Clients
+ count = #t.clients.prv['4']+#t.clients.prv['6']
+ if count > 0 or vbs > 1 then
+ table.insert(o,
+ {
+ ['name'] = ('Private Clients (%d)'):format(count),
+ output_ips(t.clients.prv)
+ }
+ )
+ end
+ -- Public Clients
+ count = #t.clients.pub['4']+#t.clients.pub['6']
+ if count > 0 or vbs > 1 then
+ table.insert(o,
+ {
+ ['name'] = ('Public Clients (%d)'):format(count),
+ output_ips(t.clients.pub)
+ }
+ )
+ end
+
+ -- Other
+ count = #t.other
+ if count > 0 then
+ table.insert(o,
+ {
+ ['name'] = ('Other Associations (%d)'):format(count),
+ t.other
+ }
+ )
+ end
+
+ return stdnse.format_output(true, o)
+
+end
+
+
+---
+-- Sorts and combines a set of IPv4 and IPv6 addresses into a table of rows.
+-- IPv4 addresses are ascending-sorted numerically and arranged in four columns
+-- and IPv6 appear in subsequent rows, sorted and arranged to fit as many
+-- addresses into a row as possible without the need for wrapping.
+--
+-- @param t Table containing two subtables indexed as '4' and '6' containing
+-- a list of IPv4 and IPv6 addresses respectively.
+-- @return Table where each entry is a row of sorted and arranged IP addresses.
+--
+function output_ips(t)
+
+ if #t['4'] < 1 and #t['6'] < 1 then return nil end
+
+ local o = {}
+
+ -- sort and tabulate IPv4 addresses
+ table.sort(t['4'], function(a,b) return ipOps.compare_ip(a, "lt", b) end)
+ local limit = #t['4']
+ local cols = 4
+ local rows = math.ceil(limit/cols)
+ local numlast = limit - cols*rows + cols
+ local pad4 = (' '):rep(15)
+ local index = 0
+ for c=1, cols, 1 do
+ for r=1, rows, 1 do
+ if r == rows and c > numlast then break end
+ index = index+1
+ o[r] = o[r] or ''
+ local padlen = pad4:len() - t['4'][index]:len()
+ o[r] = ('%s%s%s '):format(o[r], t['4'][index], pad4:sub(1, padlen))
+ end
+ end
+
+ -- IPv6
+ -- Rows are allowed to be 71 chars wide
+ table.sort(t['6'], function(a,b) return ipOps.compare_ip(a, "lt", b) end)
+ local i = 1
+ local limit = #t['6']
+ while i <= limit do
+ local work = {}
+ local len = 0
+ local j = i
+ repeat
+ if not t['6'][j] then j = j-1; break end
+ len = len + t['6'][j]:len() + 1
+ if len > 71 then
+ j = j-1
+ else
+ j = j+1
+ end
+ until len > 71
+ for n = i, j, 1 do
+ work[#work+1] = t['6'][n]
+ end
+ o[#o+1] = table.concat(work, ' ')
+ i = j+1
+ end
+ return o
+end