summaryrefslogtreecommitdiffstats
path: root/scripts/citrix-enum-servers.nse
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-17 07:42:04 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-17 07:42:04 +0000
commit0d47952611198ef6b1163f366dc03922d20b1475 (patch)
tree3d840a3b8c0daef0754707bfb9f5e873b6b1ac13 /scripts/citrix-enum-servers.nse
parentInitial commit. (diff)
downloadnmap-0d47952611198ef6b1163f366dc03922d20b1475.tar.xz
nmap-0d47952611198ef6b1163f366dc03922d20b1475.zip
Adding upstream version 7.94+git20230807.3be01efb1+dfsg.upstream/7.94+git20230807.3be01efb1+dfsgupstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to '')
-rw-r--r--scripts/citrix-enum-servers.nse145
1 files changed, 145 insertions, 0 deletions
diff --git a/scripts/citrix-enum-servers.nse b/scripts/citrix-enum-servers.nse
new file mode 100644
index 0000000..b94c51e
--- /dev/null
+++ b/scripts/citrix-enum-servers.nse
@@ -0,0 +1,145 @@
+local nmap = require "nmap"
+local shortport = require "shortport"
+local stdnse = require "stdnse"
+local string = require "string"
+local table = require "table"
+
+description = [[
+Extracts a list of Citrix servers from the ICA Browser service.
+]]
+
+---
+-- @usage sudo ./nmap -sU --script=citrix-enum-servers -p 1604
+--
+-- @output
+-- PORT STATE SERVICE
+-- 1604/udp open unknown
+-- | citrix-enum-servers:
+-- | CITRIXSRV01
+-- |_ CITRIXSRV02
+--
+
+-- Version 0.2
+
+-- Created 11/26/2009 - v0.1 - created by Patrik Karlsson <patrik@cqure.net>
+-- Revised 11/26/2009 - v0.2 - minor packet documentation
+
+
+author = "Patrik Karlsson"
+license = "Same as Nmap--See https://nmap.org/book/man-legal.html"
+categories = {"discovery", "safe"}
+
+
+portrule = shortport.portnumber(1604, "udp")
+
+--
+-- process the response from the server
+-- @param response string, complete server response
+-- @return string row delimited with \n containing all published applications
+--
+function process_server_response(response)
+
+ local packet_len, pos = string.unpack("<I2", response)
+ local server_name
+ local server_list = {}
+
+ if packet_len < 40 then
+ return
+ end
+
+ -- the list of published applications starts at offset 40
+ local offset = 41
+
+ while offset < packet_len do
+ server_name, pos = string.unpack("z", response:sub(offset))
+ offset = offset + pos - 1
+ table.insert(server_list, server_name)
+ end
+
+ return server_list
+
+end
+
+
+action = function(host, port)
+
+ local packet, counter, socket
+ local query = {}
+ local server_list = {}
+
+ --
+ -- Packets were intercepted from the Citrix Program Neighborhood client
+ -- They are used to query a server for its list of published applications
+ --
+ -- We're really not interested in the responses to the first two packets
+ -- The third response contains the list of published applications
+ -- I couldn't find any documentation on this protocol so I'm providing
+ -- some brief information for the bits and bytes this script uses.
+ --
+ -- Spec. of response to query[2] that contains a list of published apps
+ --
+ -- offset size content
+ -- -------------------------
+ -- 0 16-bit Length
+ -- 12 32-bit Server IP (not used here)
+ -- 30 8-bit Last packet(1), More packets(0)
+ -- 40 - null-separated list of applications
+ --
+ query[0] = string.char(
+ 0x1e, 0x00, -- Length: 30
+ 0x01, 0x30, 0x02, 0xfd, 0xa8, 0xe3, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00
+ )
+
+ query[1] = string.char(
+ 0x2a, 0x00, -- Length: 42
+ 0x01, 0x32, 0x02, 0xfd, 0xa8, 0xe3, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x01, 0x00, 0x02, 0x00, 0x00, 0x00,
+ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00
+ )
+
+ counter = 0
+
+ socket = nmap.new_socket()
+ socket:set_timeout(5000)
+
+ local try = nmap.new_try(function() socket:close() end)
+ try(socket:connect(host, port))
+
+ -- send the two first packets and never look back
+ repeat
+ try(socket:send(query[counter]))
+ packet = try(socket:receive())
+ counter = counter + 1
+ until (counter>#query)
+
+ -- process the first response
+ server_list = process_server_response( packet )
+
+ --
+ -- the byte at offset 31 in the response has a really magic function
+ -- if it is set to zero (0) we have more response packets to process
+ -- if it is set to one (1) we have arrived at the last packet of our journey
+ --
+ while packet:sub(31,31) ~= "\x01" do
+ packet = try( socket:receive() )
+ local tmp_table = process_server_response( packet )
+
+ for _, v in ipairs(tmp_table) do
+ table.insert(server_list, v)
+ end
+ end
+
+ if #server_list>0 then
+ nmap.set_port_state(host, port, "open")
+ end
+
+ socket:close()
+
+ return stdnse.format_output(true, server_list)
+
+end