diff options
Diffstat (limited to '')
-rw-r--r-- | nselib/ncp.lua | 1116 |
1 files changed, 1116 insertions, 0 deletions
diff --git a/nselib/ncp.lua b/nselib/ncp.lua new file mode 100644 index 0000000..2c03e74 --- /dev/null +++ b/nselib/ncp.lua @@ -0,0 +1,1116 @@ +--- A tiny implementation of the Netware Core Protocol (NCP). +-- While NCP was originally a Netware only protocol it's now present on +-- both Linux and Windows platforms running Novell eDirectory. +-- +-- The library implements a small amount of NCP functions based on various +-- packet dumps generated by Novell software, such as the Novell Client and +-- Console One. The functions are mainly used for enumeration and discovery +-- +-- The library implements a number of different classes where the Helper is +-- the one that should be the easiest to use from scripts. +-- +-- The following classes exist: +-- +-- * Packet +-- - Implements functions for creating and serializing a NCP packet +-- +-- * ResponseParser +-- - A static class containing a bunch of functions to decode server +-- responses +-- +-- * Response +-- - Class responsible for decoding NCP responses +-- +-- * NCP +-- - Contains the "native" NCP functions sending the actual request to the +-- server. +-- +-- * Helper +-- - The preferred script interface to the library containing functions +-- that wrap functions from the NCP class using more descriptive names +-- and easier interface. +-- +-- * Util +-- - A class containing mostly decoding and helper functions +-- +-- The following example code illustrates how to use the Helper class from a +-- script. The example queries the server for all User objects from the root. +-- +-- <code> +-- local helper = ncp.Helper:new(host,port) +-- local status, resp = helper:connect() +-- status, resp = helper:search("[Root]", "User", "*") +-- status = helper:close() +-- </code> +-- + +--@author Patrik Karlsson <patrik@cqure.net> +--@copyright Same as Nmap--See https://nmap.org/book/man-legal.html + +-- Version 0.1 +-- Created 24/04/2011 - v0.1 - created by Patrik Karlsson <patrik@cqure.net> + +local ipOps = require "ipOps" +local match = require "match" +local nmap = require "nmap" +local stdnse = require "stdnse" +local string = require "string" +local table = require "table" +local unicode = require "unicode" +_ENV = stdnse.module("ncp", stdnse.seeall) + + +NCPType = { + CreateConnection = 0x1111, + ServiceRequest = 0x2222, + ServiceReply = 0x3333, + DestroyConnection = 0x5555, +} + +Status = { + CONNECTION_OK = 0, + COMPLETION_OK = 0, +} + +NCPFunction = { + GetMountVolumeList = 0x16, + GetFileServerInfo = 0x17, + Ping = 0x68, + EnumerateNetworkAddress = 0x7b, + SendFragmentedRequest = 0x68, +} + +NCPVerb = { + Resolve = 1, + List = 5, + Search = 6, +} + +-- The NCP Packet +Packet = { + + --- Creates a new instance of Packet + -- @return o instance of Packet + new = function(self) + local o = {} + setmetatable(o, self) + self.__index = self + o.ncp_ip = { signature = "DmdT", replybuf = 0, version = 1 } + o.task = 1 + o.func = 0 + return o + end, + + --- Sets the NCP Reply buffer size + -- @param n number containing the buffer size + setNCPReplyBuf = function(self, n) self.ncp_ip.replybuf = n end, + + --- Sets the NCP packet length + -- @param n number containing the length + setNCPLength = function(self, n) self.ncp_ip.length = n end, + + --- Gets the NCP packet length + -- @return n number containing the NCP length + getNCPLength = function(self) return self.ncp_ip.length end, + + --- Sets the NCP packet type + -- @param t number containing the NCP packet type + setType = function(self, t) self.type = t end, + + --- Gets the NCP packet type + -- @return type number containing the NCP packet type + getType = function(self) return self.type end, + + --- Sets the NCP packet function + -- @param t number containing the NCP function + setFunc = function(self, f) self.func = f end, + + --- Gets the NCP packet function + -- @return func number containing the NCP packet function + getFunc = function(self) return self.func end, + + --- Sets the NCP sequence number + -- @param seqno number containing the sequence number + setSeqNo = function(self, n) self.seqno = n end, + + --- Sets the NCP connection number + -- @param conn number containing the connection number + setConnNo = function(self, n) self.conn = n end, + + --- Gets the NCP connection number + -- @return conn number containing the connection number + getConnNo = function(self) return self.conn end, + + --- Sets the NCP sub function + -- @param subfunc number containing the subfunction + setSubFunc = function(self, n) self.subfunc = n end, + + --- Gets the NCP sub function + -- @return subfunc number containing the subfunction + getSubFunc = function(self) return self.subfunc end, + + --- Gets the Sequence number + -- @return seqno number containing the sequence number + getSeqNo = function(self) return self.seqno end, + + --- Sets the packet length + -- @param len number containing the packet length + setLength = function(self, n) self.length = n end, + + --- Sets the packet data + -- @param data string containing the packet data + setData = function(self, data) self.data = data end, + + --- Gets the packet data + -- @return data string containing the packet data + getData = function(self) return self.data end, + + --- Sets the packet task + -- @param task number containing the packet number + setTask = function(self, task) self.task = task end, + + --- "Serializes" the packet to a string + __tostring = function(self) + local UNKNOWN = 0 + local data = self.ncp_ip.signature + .. string.pack(">I4I4I4I2BBBBB", + self.ncp_ip.length or 0, self.ncp_ip.version, + self.ncp_ip.replybuf, self.type, self.seqno, + self.conn, self.task, UNKNOWN, self.func ) + .. (self.length and string.pack(">I2", self.length) or "") + .. (self.subfunc and string.pack("B", self.subfunc) or "") + .. (self.data or "") + + return data + end, + +} + +-- Parses different responses into suitable tables +ResponseParser = { + + --- Determines what parser to call based on the contents of the client + -- request and server response. + -- @param req instance of Packet containing the request to the server + -- @param resp instance of Response containing the server response + -- @return status true on success, false on failure + -- @return resp table (on success) containing the decoded response + -- @return err string (on failure) containing the error message + parse = function(req, resp) + local func, subfunc, typ = req:getFunc(), req:getSubFunc(), req:getType() + + if ( ResponseParser[func] ) then + return ResponseParser[func](resp) + elseif ( NCPFunction.SendFragmentedRequest == func ) then + if ( 1 == subfunc ) then + return ResponseParser.Ping(resp) + elseif ( 2 == subfunc ) then + local data = req:getData() + if ( #data < 21 ) then + return false, "Invalid NCP request, could not parse" + end + local verb, pos = string.unpack("<I4", data, 17) + + if ( NCPVerb.Resolve == verb ) then + return ResponseParser.Resolve(resp) + elseif ( NCPVerb.List == verb ) then + return ResponseParser.List(resp) + elseif ( NCPVerb.Search == verb ) then + return ResponseParser.Search(resp) + end + return false, "ResponseParser: Failed to parse response" + end + end + + return false, "ResponseParser: Failed to parse response" + end, + + --- Decodes a GetFileServerInfo response + -- @param resp string containing the response as received from the server + -- @return status true on success, false on failure + -- @return response table (if status is true) containing: + -- <code>srvname</code> + -- <code>os_major</code> + -- <code>os_minor</code> + -- <code>conns_supported</code> + -- <code>conns_inuse</code> + -- <code>vols_supported</code> + -- <code>os_rev</code> + -- <code>sft_support</code> + -- <code>tts_level</code> + -- <code>conns_max_use</code> + -- <code>acct_version</code> + -- <code>vap_version</code> + -- <code>qms_version</code> + -- <code>print_version</code> + -- <code>internet_bridge_ver</code> + -- <code>mixed_mode_path</code> + -- <code>local_login_info</code> + -- <code>product_major</code> + -- <code>product_minor</code> + -- <code>product_rev</code> + -- <code>os_lang_id</code> + -- <code>support_64_bit</code> + -- @return error message (if status is false) + [NCPFunction.GetFileServerInfo] = function(resp) + local data = resp:getData() + local len = #data + + if ( len < 78 ) then + return false, "Failed to decode GetFileServerInfo" + end + + local result = {} + local pos + + result.srvname, result.os_major, result.os_minor, + result.conns_supported, result.conns_inuse, + result.vols_supported, result.os_rev, result.sft_support, + result.tts_level, result.conns_max_use, result.acct_version, + result.vap_version, result.qms_version, result.print_version, + result.virt_console_ver, result.sec_restrict_ver, + result.internet_bridge_ver, result.mixed_mode_path, + result.local_login_info, result.product_major, + result.product_minor, result.product_rev, result.os_lang_id, + result.support_64_bit, pos = string.unpack(">c48BBI2I2I2BBBI2BBBBBBBBBI2I2I2BB", data) + + return true, result + end, + + --- Decodes a GetMountVolumeList response + -- @param resp string containing the response as received from the server + -- @return status true on success, false on failure + -- @return response table of vol entries (if status is true) + -- Each vol entry is a table containing the following fields: + -- <code>vol_no</code> and <code>vol_name</code> + -- @return error message (if status is false) + [NCPFunction.GetMountVolumeList] = function(resp) + local data = resp:getData() + local len = #data + + local items, next_vol_no, pos = string.unpack("<I4I4", data) + local vols = {} + for i=1, items do + local vol = {} + vol.vol_no, vol.vol_name, pos = string.unpack("<I4s1", data, pos) + table.insert(vols, vol) + end + return true, vols + end, + + --- Decodes a Ping response + -- @param resp string containing the response as received from the server + -- @return status true on success, false on failure + -- @return response table (if status is true) containing: + -- <code>tree_name</code> + -- @return error message (if status is false) + Ping = function(resp) + local data = resp:getData() + local len = #data + local pos + local result = {} + + if ( len < 40 ) then return false, "NCP Ping result too short" end + + result.nds_version, pos = string.unpack("B", data) + -- move to the offset of the + pos = pos + 7 + result.tree_name, pos = string.unpack("c32", data, pos) + + result.tree_name = (result.tree_name:match("^([^_]*)_*$")) + + return true, result + end, + + --- Decodes a EnumerateNetworkAddress response + -- @param resp string containing the response as received from the server + -- @return status true on success, false on failure + -- @return response table (if status is true) containing: + -- <code>ip</code>, <code>port</code> and <code>proto</code> + -- @return error message (if status is false) + [NCPFunction.EnumerateNetworkAddress] = function(resp) + local pos, result = 1, {} + local items + local data = resp:getData() + local len = #data + + result.time_since_boot, result.console_version, result.console_revision, + result.srvinfo_flags, result.guid, result.next_search, + items, pos = string.unpack("<I4 BBI2 c16 I4I4", data) + + local function DecodeAddress(data, pos) + local COMM_TYPES = { [5] = "udp", [6] = "tcp" } + local comm_type, port, ip, _ + comm_type, _, _, _, port, ip, pos = string.unpack(">BBI4I2I2I4", data, pos) + + return pos, { port = port, ip = ipOps.fromdword(ip), + proto = COMM_TYPES[comm_type] or "unknown" } + end + + if ( ( pos - 1 ) + (items * 14 ) > len ) then + return false, "EnumerateNetworkAddress packet too short" + end + + result.addr = {} + for i=1, items do + local addr = {} + pos, addr = DecodeAddress(data, pos) + table.insert(result.addr, addr ) + end + return true, result + end, + + + --- Decodes a Resolve response + -- @param resp string containing the response as received from the server + -- @return status true on success, false on failure + -- @return response table (if status is true) containing: + -- <code>tag</code> and <code>id</code> + -- @return error message (if status is false) + Resolve = function(resp) + local data = resp:getData() + local len = #data + + if ( len < 12 ) then + return false, "ResponseParser: NCP Resolve, packet too short" + end + + local frag_size, frag_handle, comp_code, pos = string.unpack("<I4I4I4", data) + + if ( len < 38 ) then + return false, "ResponseParser: message too short" + end + + if ( comp_code ~= 0 ) then + return false, ("ResponseParser: Completion code returned" .. + " non-zero value (%d)"):format(comp_code) + end + + local tag, entry, pos = string.unpack("<I4I4", data, pos) + + return true, { tag = tag, id = entry } + end, + + + --- Decodes a Search response + -- @param resp string containing the response as received from the server + -- @return status true on success, false on failure + -- @return entries table (if status is true) as return by: + -- <code>EntryDecoder</code> + -- @return error message (if status is false) + Search = function(resp) + local data = resp:getData() + local len = #data + local entries = {} + + if ( len < 12 ) then + return false, "ResponseParser: NCP Resolve, packet too short" + end + + local frag_size, frag_handle, comp_code, iter_handle, pos = string.unpack("<I4I4I4I4", data) + + if ( comp_code ~= 0 ) then + return false, ("ResponseParser: Completion code returned" .. + " non-zero value (%d)"):format(comp_code) + end + + pos = pos + 12 + local entry_count + entry_count, pos = string.unpack("<I4", data, pos) + + for i=1, entry_count do + local entry + pos, entry = ResponseParser.EntryDecoder(data, pos) + -- pad for unknown trailing data in searches + pos = pos + 8 + table.insert(entries, entry) + end + return true, entries + end, + + --- The EntryDecoder is used by the Search and List function, for decoding + -- the returned entries. + -- @param data containing the response as returned by the server + -- @param pos number containing the offset into data to start decoding + -- @return pos number containing the new offset after decoding + -- @return entry table containing the decoded entry, currently it contains + -- one or more of the following fields: + -- <code>flags</code> + -- <code>mod_time</code> + -- <code>sub_count</code> + -- <code>baseclass</code> + -- <code>rdn</code> + -- <code>name</code> + EntryDecoder = function(data, pos) + + -- The InfoFlags class takes a numeric value and facilitates + -- bit decoding into InfoFlag fields, the current supported fields + -- are: + -- <code>Output</code> + -- <code>Entry</code> + -- <code>Count</code> + -- <code>ModTime</code> + -- <code>BaseClass</code> + -- <code>RelDN</code> + -- <code>DN</code> + local InfoFlags = { + -- Creates a new instance + -- @param val number containing the numeric representation of flags + -- @return a new instance of InfoFlags + new = function(self, val) + local o = {} + setmetatable(o, self) + self.__index = self + o.val = val + o:parse() + return o + end, + + -- Parses the numeric value and creates a number of class fields + parse = function(self) + local fields = { "Output", "_u1", "Entry", "Count", "ModTime", + "_u2", "_u3", "_u4", "_u5", "_u6", "_u7", "BaseClass", + "RelDN", "DN" } + local bits = 1 + for _, field in ipairs(fields) do + self[field] = ((self.val & bits) == bits) + bits = bits * 2 + end + end + } + + local entry = {} + local f, len + f, pos = string.unpack("<I4", data, pos) + local iflags = InfoFlags:new(f) + + if ( iflags.Entry ) then + entry.flags, entry.sub_count, pos = string.unpack("<I4I4", data, pos) + end + + if ( iflags.ModTime ) then + entry.mod_time, pos = string.unpack("<I4", data, pos) + end + + if ( iflags.BaseClass ) then + len, pos = string.unpack("<I4", data, pos) + entry.baseclass, pos = string.unpack("c" .. len, data, pos) + entry.baseclass = unicode.utf16to8(entry.baseclass) + entry.baseclass = Util.CToLuaString(entry.baseclass) + pos = ( len % 4 == 0 ) and pos or pos + ( 4 - ( len % 4 ) ) + end + + if ( iflags.RelDN ) then + len, pos = string.unpack("<I4", data, pos) + entry.rdn, pos = string.unpack("c" .. len, data, pos) + entry.rdn = unicode.utf16to8(entry.rdn) + entry.rdn = Util.CToLuaString(entry.rdn) + pos = ( len % 4 == 0 ) and pos or pos + ( 4 - ( len % 4 ) ) + end + + if ( iflags.DN ) then + len, pos = string.unpack("<I4", data, pos) + entry.name, pos = string.unpack("c" .. len, data, pos) + entry.name = unicode.utf16to8(entry.name) + entry.name = Util.CToLuaString(entry.name) + pos = ( len % 4 == 0 ) and pos or pos + ( 4 - ( len % 4 ) ) + end + + return pos, entry + end, + + + --- Decodes a List response + -- @param resp string containing the response as received from the server + -- @return status true on success, false on failure + -- @return entries table (if status is true) as return by: + -- <code>EntryDecoder</code> + -- @return error message (if status is false) + List = function(resp) + local data = resp:getData() + local len = #data + + if ( len < 12 ) then + return false, "ResponseParser: NCP Resolve, packet too short" + end + + local frag_size, frag_handle, comp_code, iter_handle, pos = string.unpack("<I4I4I4I4", data) + + if ( comp_code ~= 0 ) then + return false, ("ResponseParser: Completion code returned" .. + " non-zero value (%d)"):format(comp_code) + end + + local entry_count + entry_count, pos = string.unpack("<I4", data, pos) + + local entries = {} + + for i=1, entry_count do + local entry = {} + pos, entry = ResponseParser.EntryDecoder(data, pos) + table.insert(entries, entry) + end + + return true, entries + end, +} + +-- The response class holds the NCP data. An instance is usually created +-- using the fromSocket static function that reads a NCP packet of the +-- the socket and makes necessary parsing. +Response = { + + --- Creates a new Response instance + -- @param header string containing the header part of the response + -- @param data string containing the data part of the response + -- @return o new instance of Response + new = function(self, header, data) + local o = {} + setmetatable(o, self) + self.__index = self + o.header = header + o.data = data + o:parse() + return o + end, + + --- Parses the Response + parse = function(self) + local pos, _ + + self.signature, self.length, self.type, + self.seqno, self.conn, _, self.compl_code, + self.status_code, pos = string.unpack(">I4 I4 I2BBI2BB", self.header) + + if ( self.data ) then + local len = #self.data - pos + if ( ( #self.data - pos ) ~= ( self.length - 33 ) ) then + stdnse.debug1("NCP packet length mismatched") + return + end + end + end, + + --- Gets the sequence number + -- @return seqno number + getSeqNo = function(self) return self.seqno end, + + --- Gets the connection number + -- @return conn number + getConnNo = function(self) return self.conn end, + + --- Gets the data portion of the response + -- @return data string + getData = function(self) return self.data end, + + --- Gets the header portion of the response + getHeader = function(self) return self.header end, + + --- Returns true if there are any errors + -- @return error true if the response error code is anything else than OK + hasErrors = function(self) + return not( ( self.compl_code == Status.COMPLETION_OK ) and + ( self.status_code == Status.CONNECTION_OK ) ) + + end, + + --- Creates a Response instance from the data read of the socket + -- @param socket socket connected to server and ready to receive data + -- @return Response containing a new Response instance + fromSocket = function(socket) + local status, header = socket:receive_buf(match.numbytes(16), true) + if ( not(status) ) then return false, "Failed to receive data" end + + local sig, len, pos = string.unpack(">I4I4", header) + if ( len < 8 ) then return false, "NCP packet too short" end + + local data + + if ( 0 < len - 16 ) then + status, data = socket:receive_buf(match.numbytes(len - 16), true) + if ( not(status) ) then return false, "Failed to receive data" end + end + return true, Response:new(header, data) + end, + + --- "Serializes" the Response instance to a string + __tostring = function(self) + return self.header .. self.data + end, + +} + +-- The NCP class +NCP = { + + --- Creates a new NCP instance + -- @param socket containing a socket connected to the NCP server + -- @return o instance of NCP + new = function(self, socket) + local o = {} + setmetatable(o, self) + self.__index = self + o.socket = socket + o.seqno = -1 + o.conn = 0 + return o + end, + + --- Handles sending and receiving a NCP message + -- @param p Packet containing the request to send to the server + -- @return status true on success false on failure + -- @return response table (if status is true) containing the parsed + -- response + -- @return error string (if status is false) containing the error + Exch = function(self, p) + local status, err = self:SendPacket(p) + if ( not(status) ) then return status, err end + + local status, resp = Response.fromSocket(self.socket) + if ( not(status) or resp:hasErrors() ) then return false, resp end + + self.seqno = resp:getSeqNo() + self.conn = resp:getConnNo() + + return ResponseParser.parse(p, resp) + end, + + --- Sends a packet to the server + -- @param p Packet to be sent to the server + -- @return status true on success, false on failure + -- @return err string containing the error message on failure + SendPacket = function(self, p) + if ( not(p:getSeqNo() ) ) then p:setSeqNo(self.seqno + 1) end + if ( not(p:getConnNo() ) ) then p:setConnNo(self.conn) end + + if ( not(p:getNCPLength()) ) then + local len = #(tostring(p)) + p:setNCPLength(len) + end + + local status, err = self.socket:send(tostring(p)) + if ( not(status) ) then return status, "Failed to send data" end + + return true + end, + + --- Creates a connection to the NCP server + -- @return status true on success, false on failure + CreateConnect = function(self) + local p = Packet:new() + p:setType(NCPType.CreateConnection) + + local resp = self:Exch( p ) + return true + end, + + --- Destroys a connection established with the NCP server + -- @return status true on success, false on failure + DestroyConnect = function(self) + local p = Packet:new() + p:setType(NCPType.DestroyConnection) + + local resp = self:Exch( p ) + return true + end, + + --- Gets file server information + -- @return status true on success, false on failure + -- @return response table (if status is true) containing: + -- <code>srvname</code> + -- <code>os_major</code> + -- <code>os_minor</code> + -- <code>conns_supported</code> + -- <code>conns_inuse</code> + -- <code>vols_supported</code> + -- <code>os_rev</code> + -- <code>sft_support</code> + -- <code>tts_level</code> + -- <code>conns_max_use</code> + -- <code>acct_version</code> + -- <code>vap_version</code> + -- <code>qms_version</code> + -- <code>print_version</code> + -- <code>internet_bridge_ver</code> + -- <code>mixed_mode_path</code> + -- <code>local_login_info</code> + -- <code>product_major</code> + -- <code>product_minor</code> + -- <code>product_rev</code> + -- <code>os_lang_id</code> + -- <code>support_64_bit</code> + -- @return error message (if status is false) + GetFileServerInfo = function(self) + local p = Packet:new() + p:setType(NCPType.ServiceRequest) + p:setFunc(NCPFunction.GetFileServerInfo) + p:setNCPReplyBuf(128) + p:setLength(1) + p:setSubFunc(17) + return self:Exch( p ) + end, + + + -- NEEDS authentication, disabled for now + -- + -- Get the logged on user for the specified connection + -- @param conn_no number containing the connection number + -- GetStationLoggedInfo = function(self, conn_no) + -- local p = Packet:new() + -- p:setType(NCPType.ServiceRequest) + -- p:setFunc(NCPFunction.GetFileServerInfo) + -- p:setNCPReplyBuf(62) + -- p:setLength(5) + -- p:setSubFunc(28) + -- p:setTask(4) + -- + -- local data = string.pack("<I4", conn_no) + -- p:setData(data) + -- return self:Exch( p ) + -- end, + + --- Sends a PING to the server which responds with the tree name + -- @return status true on success, false on failure + -- @return response table (if status is true) containing: + -- <code>tree_name</code> + -- @return error message (if status is false) + Ping = function(self) + local p = Packet:new() + p:setType(NCPType.ServiceRequest) + p:setFunc(NCPFunction.Ping) + p:setSubFunc(1) + p:setNCPReplyBuf(45) + p:setData("\0\0\0") + + return self:Exch( p ) + end, + + --- Enumerates the IP addresses associated with the server + -- @return status true on success, false on failure + -- @return response table (if status is true) containing: + -- <code>ip</code>, <code>port</code> and <code>proto</code> + -- @return error message (if status is false) + EnumerateNetworkAddress = function(self) + local p = Packet:new() + p:setType(NCPType.ServiceRequest) + p:setFunc(NCPFunction.EnumerateNetworkAddress) + p:setSubFunc(17) + p:setNCPReplyBuf(4096) + p:setData("\0\0\0\0") + p:setLength(5) + return self:Exch( p ) + end, + + --- Resolves an directory entry id from a name + -- @param name string containing the name to resolve + -- @return status true on success, false on failure + -- @return response table (if status is true) containing: + -- <code>tag</code> and <code>id</code> + -- @return error message (if status is false) + ResolveName = function(self, name) + local p = Packet:new() + p:setType(NCPType.ServiceRequest) + p:setFunc(NCPFunction.SendFragmentedRequest) + p:setSubFunc(2) + p:setNCPReplyBuf(4108) + + local pad = (4 - ( #name % 4 ) ) + name = Util.ZeroPad(name, #name + pad) + + local w_name = unicode.utf8to16(name) + local frag_handle, frag_size = 0xffffffff, 64176 + local msg_size, unknown, proto_flags, nds_verb = 44 + #w_name, 0, 0, 1 + local nds_reply_buf, version, flags, scope = 4096, 1, 0x2062, 0 + -- TODO: unknown2 is not used. Should it be? + local unknown2 = 0x0e + + local data = { + string.pack("<I4I4I4 I2I2I4I4I4I2 I2I4 s4", frag_handle, frag_size, msg_size, + unknown, proto_flags, nds_verb, nds_reply_buf, version, flags, + unknown, scope, w_name) + } + + local comms = { { transport = "TCP" } } + local walkers= { { transport = "TCP" } } + local PROTOCOLS = { ["TCP"] = 9 } + + data[#data+1] = string.pack("<I4", #comms) + for _, comm in ipairs(comms) do + data[#data+1] = string.pack("<I4", PROTOCOLS[comm.transport]) + end + + data[#data+1] = string.pack("<I4", #walkers) + for _, walker in ipairs(walkers) do + data[#data+1] = string.pack("<I4", PROTOCOLS[walker.transport]) + end + + p:setData(table.concat(data)) + return self:Exch( p ) + end, + + --- Gets a list of volumes from the server + -- @return status true on success, false on failure + -- @return response table of vol entries (if status is true) + -- Each vol entry is a table containing the following fields: + -- <code>vol_no</code> and <code>vol_name</code> + -- @return error message (if status is false) + GetMountVolumeList = function(self) + local p = Packet:new() + p:setType(NCPType.ServiceRequest) + p:setFunc(NCPFunction.GetMountVolumeList) + p:setSubFunc(52) + p:setNCPReplyBuf(538) + p:setTask(4) + p:setLength(12) + + local start_vol = 0 + local vol_req_flags = 1 + local src_name_space = 0 + + local data = string.pack("<I4I4I4", start_vol, vol_req_flags, src_name_space ) + p:setData(data) + return self:Exch( p ) + end, + + --- Searches the directory + -- @param base entry as resolved by <code>Resolve</code> + -- @param class string containing a class name (or * wildcard) + -- @param name string containing a entry name (or * wildcard) + -- @param options table containing one or more of the following + -- <code>numobjs</code> + -- @return status true on success false on failure + -- @return entries table (if status is true) as return by: + -- <code>ResponseDecoder.EntryDecoder</code> + -- @return error string (if status is false) containing the error + Search = function(self, base, class, name, options) + assert( ( base and base.id ), "No base entry was specified") + + local class = class and class .. '\0' or '*\0' + local name = name and name .. '\0' or '*\0' + local w_name = unicode.utf8to16(name) + local w_class = unicode.utf8to16(class) + local options = options or {} + local p = Packet:new() + p:setType(NCPType.ServiceRequest) + p:setFunc(NCPFunction.SendFragmentedRequest) + p:setSubFunc(2) + p:setNCPReplyBuf(64520) + p:setTask(5) + + local frag_handle, frag_size, msg_size = 0xffffffff, 64176, 98 + local unknown, proto_flags, nds_verb, version, flags = 0, 0, 6, 3, 0 + local nds_reply_buf = 64520 + local iter_handle = 0xffffffff + local repl_type = 2 -- base and all subordinates + local numobjs = options.numobjs or 0 + local info_types = 1 -- Names + local info_flags = 0x0000381d + -- a bunch of unknowns + local u2, u3, u4, u5, u6, u7, u8, u9 = 0, 0, 2, 2, 0, 0x10, 0, 0x11 + + local data = string.pack("<I4I4I4 I2I2I4 I4 I4 I4 I4 I4 I4 I4 I4 I4 I4 I4 I4 I4 I4 I4 s4 I4 I4 s4", + frag_handle, frag_size, msg_size, unknown, proto_flags, + nds_verb, nds_reply_buf, version, flags, iter_handle, + base.id, repl_type, numobjs, info_types, info_flags, u2, u3, u4, + u5, u6, u7, w_name, u8, u9, w_class ) + p:setData(data) + return self:Exch( p ) + end, + + --- Lists the contents of entry + -- @param entry entry as resolved by <code>Resolve</code> + -- @return status true on success false on failure + -- @return entries table (if status is true) as return by: + -- <code>ResponseDecoder.EntryDecoder</code> + -- @return error string (if status is false) containing the error + List = function(self, entry) + local p = Packet:new() + p:setType(NCPType.ServiceRequest) + p:setFunc(NCPFunction.SendFragmentedRequest) + p:setSubFunc(2) + p:setNCPReplyBuf(4112) + p:setTask(2) + + local frag_handle, frag_size = 0xffffffff, 64176 + local msg_size, unknown, proto_flags, nds_verb = 40, 0, 0, 5 + local nds_reply_buf, version, flags = 4100, 1, 0x0001 + local iter_handle = 0xffffffff + -- TODO: unknown2 is not used. Should it be? + local unknown2 = 0x0e + local info_flags = 0x0000381d + + local data = string.pack("<I4I4I4I2I2I4I4I4I2I2I4I4I4", frag_handle, frag_size, msg_size, + unknown, proto_flags, nds_verb, nds_reply_buf, version, flags, + unknown, iter_handle, entry.id, info_flags ) + + -- no name filter + .. "\0\0\0\0" + + -- no class filter + .. "\0\0\0\0" + + p:setData(data) + local status, entries = self:Exch( p ) + if ( not(status) ) then return false, entries end + + return true, entries + end, + +} + + +Helper = { + + --- Creates a new Helper instance + -- @return a new Helper instance + new = function(self, host, port) + local o = {} + setmetatable(o, self) + self.__index = self + o.host = host + o.port = port + return o + end, + + --- Connect the socket and creates a NCP connection + -- @return true on success false on failure + connect = function(self) + self.socket = nmap.new_socket() + self.socket:set_timeout(5000) + local status, err = self.socket:connect(self.host, self.port) + if ( not(status) ) then return status, err end + + self.ncp = NCP:new(self.socket) + return self.ncp:CreateConnect() + end, + + --- Closes the helper connection + close = function(self) + self.ncp:DestroyConnect() + self.socket:close() + end, + + --- Performs a directory search + -- @param base string containing the name of the base to search + -- @param class string containing the type of class to search + -- @param name string containing the name of the object to find + -- @param options table containing on or more of the following + -- <code>numobjs</code> - number of objects to limit the search to + search = function(self, base, class, name, options) + local base = base or "[Root]" + local status, entry = self.ncp:ResolveName(base) + + if ( not(status) ) then + return false, "Search failed, base could not be resolved" + end + + local status, result = self.ncp:Search(entry, class, name, options) + if (not(status)) then return false, result end + + return status, result + end, + + --- Retrieves some information from the server using the following NCP + -- functions: + -- + -- * <code>GetFileServerInfo</code> + -- * <code>Ping</code> + -- * <code>EnumerateNetworkAddress</code> + -- * <code>GetMountVolumeList</code> + -- + -- The result contains the Tree name, product versions and mounts + getServerInfo = function(self) + local status, srv_info = self.ncp:GetFileServerInfo() + if ( not(status) ) then return false, srv_info end + + local status, ping_info = self.ncp:Ping() + if ( not(status) ) then return false, ping_info end + + local status, net_info = self.ncp:EnumerateNetworkAddress() + if ( not(status) ) then return false, net_info end + + local status, mnt_list = self.ncp:GetMountVolumeList() + if ( not(status) ) then return false, mnt_list end + + local output = {} + table.insert(output, ("Server name: %s"):format(srv_info.srvname)) + table.insert(output, ("Tree Name: %s"):format(ping_info.tree_name)) + table.insert(output, + ("OS Version: %d.%d (rev %d)"):format(srv_info.os_major, + srv_info.os_minor, srv_info.os_rev)) + table.insert(output, + ("Product version: %d.%d (rev %d)"):format(srv_info.product_major, + srv_info.product_minor, srv_info.product_rev)) + table.insert(output, ("OS Language ID: %d"):format(srv_info.os_lang_id)) + + local niceaddr = {} + for _, addr in ipairs(net_info.addr) do + table.insert(niceaddr, ("%s %d/%s"):format(addr.ip,addr.port, + addr.proto)) + end + + niceaddr.name = "Addresses" + table.insert(output, niceaddr) + + local mounts = {} + for _, mount in ipairs(mnt_list) do + table.insert(mounts, mount.vol_name) + end + + mounts.name = "Mounts" + table.insert(output, mounts) + + if ( nmap.debugging() > 0 ) then + table.insert(output, ("Acct version: %d"):format(srv_info.acct_version)) + table.insert(output, ("VAP version: %d"):format(srv_info.vap_version)) + table.insert(output, ("QMS version: %d"):format(srv_info.qms_version)) + table.insert(output, + ("Print server version: %d"):format(srv_info.print_version)) + table.insert(output, + ("Virtual console version: %d"):format(srv_info.virt_console_ver)) + table.insert(output, + ("Security Restriction Version: %d"):format(srv_info.sec_restrict_ver)) + table.insert(output, + ("Internet Bridge Version: %d"):format(srv_info.internet_bridge_ver)) + end + + return true, output + end, +} + +--- "static" Utility class containing mostly conversion functions +Util = +{ + --- Pads a string with zeroes + -- + -- @param str string containing the string to be padded + -- @param len number containing the length of the new string + -- @return str string containing the new string + ZeroPad = function( str, len ) + return str .. string.rep('\0', len - #str) + end, + + -- Removes trailing nulls + -- + -- @param str containing the string + -- @return ret the string with any trailing nulls removed + CToLuaString = function( str ) + local ret + + if ( not(str) ) then return "" end + if ( str:sub(-1, -1 ) ~= "\0" ) then return str end + + for i=1, #str do + if ( str:sub(-i,-i) == "\0" ) then + ret = str:sub(1, -i - 1) + else + break + end + end + return ret + end, + +} + +return _ENV; |