mirror of
https://github.com/nmap/nmap.git
synced 2025-12-06 04:31:29 +00:00
again. Here are the items which were merged: ------------------------------------------------------------------------ r13971 | jah | 2009-06-29 14:30:27 -0700 (Mon, 29 Jun 2009) | 2 lines Improved a pattern for matching HTTP status-line, tidy away some variables and fix a typo. ------------------------------------------------------------------------ r13967 | daniel | 2009-06-29 13:47:04 -0700 (Mon, 29 Jun 2009) | 5 lines o Added a convenience top-level BSD makefile redirecting BSD make to GNU make on BSD systems. This should help prevent bogus error reports when users run "make" instead of "gmake" on BSD systems. [Daniel Roethlisberger] ------------------------------------------------------------------------ r13965 | batrick | 2009-06-29 06:50:11 -0700 (Mon, 29 Jun 2009) | 14 lines [NSE] The NSE Nsock Library binding no longer relies on garbage collection to monitor the use of socket "slots". A thread (script) attempting to connect must first obtain one of a limited number of available socket locks (usually 10 or --max-parallelism). The binding would use garbage collection of sockets to determine when a thread has finished using its allocated sockets. This is unfortunately slow and requires us to constantly run the garbage collector to cause timely reallocation. I have changed the binding to now regularly inspect allocated sockets in the nsock_loop function. Available sockets slots are now immediately reallocated and done with far less execution time. See [1] for benchmarks and further explanation. [1] http://seclists.org/nmap-dev/2009/q2/0624.html ------------------------------------------------------------------------ r13964 | batrick | 2009-06-29 06:37:49 -0700 (Mon, 29 Jun 2009) | 10 lines [NSE] Fixed a rare (and usually undetectable) bug that can cause a SEGFAULT. The NSE nsock library binding may attempt to push values on the stack of a thread that ended due to an error. It is possible that the internal Lua stack was completely full and any further pushed values would result in a segmentation memory violation. This bug is very hard to reproduce with a SEGFAULT but is usually visible when Lua assertion checks are turned on. A socket handler routine must be called AFTER a thread has ended in error. ------------------------------------------------------------------------ r13963 | batrick | 2009-06-29 05:51:20 -0700 (Mon, 29 Jun 2009) | 3 lines Fixed some global scoped variables to be local. This caused a many scripts to overwrite each others' sockets, options, etc. ------------------------------------------------------------------------ r13939 | joao | 2009-06-27 16:07:35 -0700 (Sat, 27 Jun 2009) | 2 lines Fixed port rule to include ssl pop3 port, now that pop3.lua supports SSL connections in function capabilities ------------------------------------------------------------------------ r13938 | joao | 2009-06-27 16:06:28 -0700 (Sat, 27 Jun 2009) | 2 lines Added transparent SSL support using comm.tryssl ------------------------------------------------------------------------ r13937 | joao | 2009-06-27 16:05:19 -0700 (Sat, 27 Jun 2009) | 2 lines Added transparent SSL support using comm.tryssl ------------------------------------------------------------------------ r13936 | joao | 2009-06-27 16:03:50 -0700 (Sat, 27 Jun 2009) | 2 lines Added SSL transparent support using comm.tryssl ------------------------------------------------------------------------ r13935 | joao | 2009-06-27 16:02:39 -0700 (Sat, 27 Jun 2009) | 2 lines Added SSL transparent support using comm.tryssl ------------------------------------------------------------------------ r13934 | joao | 2009-06-27 16:01:38 -0700 (Sat, 27 Jun 2009) | 2 lines Added SSL transparent support using comm.tryssl ------------------------------------------------------------------------ r13933 | joao | 2009-06-27 16:00:27 -0700 (Sat, 27 Jun 2009) | 2 lines SSL transparent support using comm.tryssl ------------------------------------------------------------------------ r13932 | joao | 2009-06-27 15:19:58 -0700 (Sat, 27 Jun 2009) | 2 lines Included transparent ssl support to function pop3.capabilities using comm.tryssl ------------------------------------------------------------------------ r13931 | joao | 2009-06-27 15:19:06 -0700 (Sat, 27 Jun 2009) | 3 lines New version of comm.lua with function tryssl, that transparently adds support to ssl connections ------------------------------------------------------------------------ r13930 | joao | 2009-06-27 14:50:38 -0700 (Sat, 27 Jun 2009) | 6 lines Fixed buffering problem exposed by david on nmap-dev list. The problem was solved using a buffer to receive the data, making the script work fine in cases where the ssh packets are fragmented. A very similar solution was applied to ssh1.lua. ------------------------------------------------------------------------ r13928 | batrick | 2009-06-27 04:43:12 -0700 (Sat, 27 Jun 2009) | 18 lines [NSE] We now propogate a NSE initiated yield on a script through all user coroutines so that NSE may resume control. Previously, scripts that would yield in a child coroutine (e.g. a script's child coroutine generated by Lua's coroutine.create function) would give control back to the script. A script would yield in this way by making a blocking socket operation. NSE would be unable to correctly resume child coroutine when the socket operation is finished processing. By yielding the chain of coroutines a script has operating, we allow to NSE to handle the socket operation properly. NSE would then resume the entire chain so execution may correctly resume at the coroutine which initiated the socket operation. This restores the "illusion" that a script executes without interruption. See [1] for more information, further explanation, and some use cases. [1] http://seclists.org/nmap-dev/2009/q2/0586.html ------------------------------------------------------------------------ r13817 | david | 2009-06-18 15:57:29 -0700 (Thu, 18 Jun 2009) | 3 lines Improve an OS fingerprint with a model number and broader matching. Based on a follow-up report from a submitter. ------------------------------------------------------------------------ r13814 | josh | 2009-06-17 21:34:15 -0700 (Wed, 17 Jun 2009) | 3 lines [zenmap] Added support to zenmap for the new SCTP options: -PY, -sY and -sZ ------------------------------------------------------------------------ r13797 | ron | 2009-06-17 11:02:18 -0700 (Wed, 17 Jun 2009) | 1 line Applied a patch from Mak Kolibabi that enhances the output of smb-enum-processes. The output is now modeled after the output of the 'ps' tool for higher verbosity levels. ------------------------------------------------------------------------ r13795 | david | 2009-06-17 09:05:21 -0700 (Wed, 17 Jun 2009) | 6 lines The configure script now allows cross-compiling by assuming that libpcap is recent enough. Previously it would quit because a test program could not be run. libpcap will always be recent enough when the included copy is used. The patch was contributed by Mike Frysinger.
713 lines
25 KiB
Lua
713 lines
25 KiB
Lua
-- Arguments when this file (function) is called, accessible via ...
|
|
-- [1] The NSE C library. This is saved in the local variable cnse for
|
|
-- access throughout the file.
|
|
-- [2] The list of categories/files/directories passed via --script.
|
|
-- The actual arguments passed to the anonymous main function:
|
|
-- [1] The list of hosts we run against.
|
|
--
|
|
-- When making changes to this code, please ensure you do not add any
|
|
-- code relying global indexing. Instead, create a local below for the
|
|
-- global you need access to. This protects the engine from possible
|
|
-- replacements made to the global environment, speeds up access, and
|
|
-- documents dependencies.
|
|
--
|
|
-- A few notes about the safety of the engine, that is, the ability for
|
|
-- a script developer to crash or otherwise stall NSE. The purpose of noting
|
|
-- these attack vectors is more to show the difficulty in accidently
|
|
-- breaking the system than to indicate a user may wish to break the
|
|
-- system through these means.
|
|
-- - A script writer can use the undocumented Lua function newproxy
|
|
-- to inject __gc code that could run (and error) at any location.
|
|
-- - A script writer can use the debug library to break out of
|
|
-- the "sandbox" we give it. This is made a little more difficult by
|
|
-- our use of locals to all Lua functions we use and the exclusion
|
|
-- of the main thread and subsequent user threads.
|
|
-- - A simple while true do end loop can stall the system. This can be
|
|
-- avoided by debug hooks to yield the thread at periodic intervals
|
|
-- (and perhaps kill the thread) but a C function like string.find and
|
|
-- a malicious pattern can stall the system from C just as easily.
|
|
-- - The garbage collector function is available to users and they may
|
|
-- cause the system to stall through improper use.
|
|
-- - Of course the os and io library can cause the system to also break.
|
|
|
|
local NAME = "NSE";
|
|
|
|
local YIELD = "NSE_YIELD";
|
|
local BASE = "NSE_BASE";
|
|
local WAITING_TO_RUNNING = "NSE_WAITING_TO_RUNNING";
|
|
local DESTRUCTOR = "NSE_DESTRUCTOR";
|
|
|
|
local _R = debug.getregistry(); -- The registry
|
|
local _G = _G;
|
|
|
|
local assert = assert;
|
|
local collectgarbage = collectgarbage;
|
|
local error = error;
|
|
local ipairs = ipairs;
|
|
local loadfile = loadfile;
|
|
local loadstring = loadstring;
|
|
local next = next;
|
|
local pairs = pairs;
|
|
local rawget = rawget;
|
|
local select = select;
|
|
local setfenv = setfenv;
|
|
local setmetatable = setmetatable;
|
|
local tonumber = tonumber;
|
|
local tostring = tostring;
|
|
local type = type;
|
|
local unpack = unpack;
|
|
|
|
local create = coroutine.create;
|
|
local resume = coroutine.resume;
|
|
local status = coroutine.status;
|
|
local yield = coroutine.yield;
|
|
|
|
local traceback = debug.traceback;
|
|
|
|
local byte = string.byte;
|
|
local find = string.find;
|
|
local format = string.format;
|
|
local gsub = string.gsub;
|
|
local lower = string.lower;
|
|
local match = string.match;
|
|
local sub = string.sub;
|
|
|
|
local insert = table.insert;
|
|
local remove = table.remove;
|
|
local sort = table.sort;
|
|
|
|
local nmap = require "nmap";
|
|
|
|
local cnse, rules = ...; -- The NSE C library and Script Rules
|
|
|
|
do -- Append the nselib directory to the Lua search path
|
|
local t, path = assert(cnse.fetchfile_absolute("nselib/"));
|
|
assert(t == "directory", "could not locate nselib directory!");
|
|
package.path = package.path..";"..path.."?.lua";
|
|
end
|
|
|
|
-- NSE_YIELD_VALUE
|
|
-- This is the table C uses to yield a thread with a unique value to
|
|
-- differentiate between yields initiated by NSE or regular coroutine yields.
|
|
local NSE_YIELD_VALUE = {};
|
|
|
|
do
|
|
-- This is the method by which we allow a script to have nested
|
|
-- coroutines. If a sub-thread yields in an NSE function such as
|
|
-- nsock.connect, then we propogate the yield up. These replacements
|
|
-- to the coroutine library are used only by Script Threads, not the engine.
|
|
|
|
local function handle (co, status, ...)
|
|
if status and NSE_YIELD_VALUE == ... then -- NSE has yielded the thread
|
|
return handle(co, resume(co, yield(NSE_YIELD_VALUE)));
|
|
else
|
|
return status, ...;
|
|
end
|
|
end
|
|
|
|
function coroutine.resume (co, ...)
|
|
return handle(co, resume(co, ...));
|
|
end
|
|
|
|
local resume = coroutine.resume; -- local reference to new coroutine.resume
|
|
local function aux_wrap (status, ...)
|
|
if not status then
|
|
return error(..., 2);
|
|
else
|
|
return ...;
|
|
end
|
|
end
|
|
function coroutine.wrap (f)
|
|
local co = create(f);
|
|
return function (...)
|
|
return aux_wrap(resume(co, ...));
|
|
end
|
|
end
|
|
end
|
|
|
|
-- Some local helper functions --
|
|
|
|
local log_write, verbosity, debugging =
|
|
nmap.log_write, nmap.verbosity, nmap.debugging;
|
|
|
|
local function print_verbose (level, fmt, ...)
|
|
if verbosity() >= assert(tonumber(level)) or debugging() > 0 then
|
|
log_write("stdout", format(fmt, ...));
|
|
end
|
|
end
|
|
|
|
local function print_debug (level, fmt, ...)
|
|
if debugging() >= assert(tonumber(level)) then
|
|
log_write("stdout", format(fmt, ...));
|
|
end
|
|
end
|
|
|
|
local function log_error (fmt, ...)
|
|
log_write("stderr", format(fmt, ...));
|
|
end
|
|
|
|
local function table_size (t)
|
|
local n = 0; for _ in pairs(t) do n = n + 1; end return n;
|
|
end
|
|
|
|
-- recursively copy a table, for host/port tables
|
|
-- not very rigorous, but it doesn't need to be
|
|
local function tcopy (t)
|
|
local tc = {};
|
|
for k,v in pairs(t) do
|
|
if type(v) == "table" then
|
|
tc[k] = tcopy(v);
|
|
else
|
|
tc[k] = v;
|
|
end
|
|
end
|
|
return tc;
|
|
end
|
|
|
|
local Script = {}; -- The Script Class, its constructor is Script.new.
|
|
local Thread = {}; -- The Thread Class, its constructor is Script:new_thread.
|
|
do
|
|
-- Thread:d()
|
|
-- Outputs debug information at level 1 or higher.
|
|
-- Changes "%THREAD" with an appropriate identifier for the debug level
|
|
function Thread:d (fmt, ...)
|
|
if debugging() > 1 then
|
|
print_debug(1, gsub(fmt, "%%THREAD", self.info), ...);
|
|
else
|
|
print_debug(1, gsub(fmt, "%%THREAD", self.short_basename), ...);
|
|
end
|
|
end
|
|
|
|
function Thread:close ()
|
|
local ch = self.close_handlers;
|
|
for key, destructor_t in pairs(ch) do
|
|
destructor_t.destructor(destructor_t.thread, key);
|
|
ch[key] = nil;
|
|
end
|
|
end
|
|
|
|
-- thread = Script:new_thread(rule, ...)
|
|
-- Creates a new thread for the script Script.
|
|
-- Arguments:
|
|
-- rule The rule argument the rule, hostrule or portrule, tested.
|
|
-- ... The arguments passed to the rule function (host[, port]).
|
|
-- Returns:
|
|
-- thread The thread (class) is returned, or nil.
|
|
function Script:new_thread (rule, ...)
|
|
assert(rule == "hostrule" or rule == "portrule");
|
|
if not self[rule] then return nil end -- No rule for this script?
|
|
local file_closure = self.file_closure;
|
|
local env = setmetatable({
|
|
runlevel = 1,
|
|
filename = self.filename,
|
|
}, {__index = _G});
|
|
setfenv(file_closure, env);
|
|
local unique_value = {}; -- to test valid yield
|
|
local function main (...)
|
|
file_closure(); -- loads script globals
|
|
return env.action(yield(unique_value, env[rule](...)));
|
|
end
|
|
setfenv(main, env);
|
|
-- This thread allows us to load the script's globals in the
|
|
-- same Lua thread the action and rule functions will execute in.
|
|
local co = create(main);
|
|
local s, value, rule_return = resume(co, ...);
|
|
if s and value ~= unique_value then
|
|
print_debug(1,
|
|
"A thread for %s yielded unexpectedly in the file or %s function:\n%s\n",
|
|
self.filename, rule, traceback(co));
|
|
elseif s and rule_return then
|
|
local thread = setmetatable({
|
|
co = co,
|
|
env = env,
|
|
runlevel = tonumber(rawget(env, "runlevel")) or 1,
|
|
identifier = tostring(co),
|
|
info = format("'%s' (%s)", self.short_basename, tostring(co));
|
|
type = rule == "hostrule" and "host" or "port",
|
|
close_handlers = {},
|
|
}, {
|
|
__metatable = Thread,
|
|
__index = function (thread, k) return Thread[k] or self[k] end
|
|
}); -- Access to the parent Script
|
|
thread.parent = thread; -- itself
|
|
return thread;
|
|
elseif not s then
|
|
print_debug(1, "a thread for %s failed to load:\n%s\n", self.filename,
|
|
traceback(co, tostring(rule_return)));
|
|
end
|
|
return nil;
|
|
end
|
|
|
|
local required_fields = {
|
|
description = "string",
|
|
action = "function",
|
|
categories = "table",
|
|
};
|
|
-- script = Script.new(filename)
|
|
-- Creates a new Script Class for the script.
|
|
-- Arguments:
|
|
-- filename The filename (path) of the script to load.
|
|
-- Returns:
|
|
-- script The script (class) created.
|
|
function Script.new (filename)
|
|
assert(type(filename) == "string", "string expected");
|
|
if not find(filename, "%.nse$") then
|
|
log_error(
|
|
"Warning: Loading '%s' -- the recommended file extension is '.nse'.",
|
|
filename);
|
|
end
|
|
local file_closure = assert(loadfile(filename));
|
|
-- Give the closure its own environment, with global access
|
|
local env = setmetatable({}, {__index = _G});
|
|
setfenv(file_closure, env);
|
|
local co = create(file_closure); -- Create a garbage thread
|
|
assert(resume(co)); -- Get the globals it loads in env
|
|
-- Check that all the required fields were set
|
|
for f, t in pairs(required_fields) do
|
|
local field = rawget(env, f);
|
|
if field == nil then
|
|
error(filename.." is missing required field: '"..f.."'");
|
|
elseif type(field) ~= t then
|
|
error(filename.." field '"..f.."' is of improper type '"..
|
|
type(field).."', expected type '"..t.."'");
|
|
end
|
|
end
|
|
-- Check one of two required rule functions exists
|
|
local hostrule, portrule = rawget(env, "hostrule"), rawget(env, "portrule");
|
|
assert(type(hostrule) == "function" or type(portrule) == "function",
|
|
filename.." is missing a required function: 'hostrule' or 'portrule'");
|
|
-- Assert that categories is an array of strings
|
|
for i, category in ipairs(rawget(env, "categories")) do
|
|
assert(type(category) == "string",
|
|
filename.." has non-string entries in the 'categories' array");
|
|
end
|
|
-- Return the script
|
|
return setmetatable({
|
|
filename = filename,
|
|
basename = match(filename, "[/\\]([^/\\]-)$") or filename,
|
|
short_basename = match(filename, "[/\\]([^/\\]-)%.nse$") or
|
|
match(filename, "[/\\]([^/\\]-)%.[^.]*$") or
|
|
filename,
|
|
id = match(filename, "^.-[/\\]([^\\/]-)%.nse$") or filename,
|
|
file_closure = file_closure,
|
|
hostrule = type(hostrule) == "function" and hostrule or nil,
|
|
portrule = type(portrule) == "function" and portrule or nil,
|
|
args = {n = 0};
|
|
categories = rawget(env, "categories"),
|
|
author = rawget(env, "author"),
|
|
license = rawget(env, "license"),
|
|
runlevel = tonumber(rawget(env, "runlevel")) or 1,
|
|
threads = {},
|
|
}, {__index = Script, __metatable = Script});
|
|
end
|
|
end
|
|
|
|
-- check_rules(rules)
|
|
-- Adds the "default" category if no rules were specified.
|
|
-- Adds other implicitly specified rules (e.g. "version")
|
|
--
|
|
-- Arguments:
|
|
-- rules The array of rules to check.
|
|
local function check_rules (rules)
|
|
if cnse.default and #rules == 0 then rules[1] = "default" end
|
|
if cnse.scriptversion then rules[#rules+1] = "version" end
|
|
end
|
|
|
|
-- chosen_scripts = get_chosen_scripts(rules)
|
|
-- Loads all the scripts for the given rules using the Script Database.
|
|
-- Arguments:
|
|
-- rules The array of rules to use for loading scripts.
|
|
-- Returns:
|
|
-- chosen_scripts The array of scripts loaded for the given rules.
|
|
local function get_chosen_scripts (rules)
|
|
check_rules(rules);
|
|
|
|
local script_dbpath = cnse.script_dbpath;
|
|
local t, path = cnse.fetchfile_absolute(script_dbpath);
|
|
if not t then
|
|
print_verbose(1, "Creating non-existent script database.");
|
|
assert(cnse.updatedb(), "could not update script database!");
|
|
t, path = assert(cnse.fetchfile_absolute(script_dbpath));
|
|
end
|
|
local db_closure = assert(loadfile(path),
|
|
"database appears to be corrupt or out of date;\n"..
|
|
"\tplease update using: nmap --script-updatedb");
|
|
|
|
local chosen_scripts, entry_rules, used_rules, files_loaded = {}, {}, {}, {};
|
|
|
|
-- Tokens that are allowed in script rules (--script)
|
|
local protected_lua_tokens = {
|
|
["and"] = true,
|
|
["or"] = true,
|
|
["not"] = true,
|
|
};
|
|
-- Globalize all names in str that are not protected_lua_tokens
|
|
local function globalize (str)
|
|
local lstr = lower(str);
|
|
if protected_lua_tokens[lstr] then
|
|
return lstr;
|
|
else
|
|
return 'm("'..str..'")';
|
|
end
|
|
end
|
|
|
|
for i, rule in ipairs(rules) do
|
|
rule = match(rule, "^%s*(.-)%s*$"); -- strip surrounding whitespace
|
|
used_rules[rule] = false; -- has not been used yet
|
|
-- Globalize all `names`, all visible characters not ',', '(', ')', and ';'
|
|
local globalized_rule =
|
|
gsub(rule, "[\033-\039\042-\043\045-\058\060-\126]+", globalize);
|
|
-- Precompile the globalized rule
|
|
local compiled_rule, err = loadstring("return "..globalized_rule, "rule");
|
|
if not compiled_rule then
|
|
err = err:match("rule\"]:%d+:(.+)$"); -- remove (luaL_)where in code
|
|
error("Bad script rule:\n\t"..rule.." -> "..err);
|
|
end
|
|
entry_rules[globalized_rule] = {
|
|
original_rule = rule,
|
|
compiled_rule = compiled_rule,
|
|
};
|
|
end
|
|
|
|
-- Checks if a given script, script_entry, should be loaded. A script_entry
|
|
-- should be in the form: { filename = "name.nse", categories = { ... } }
|
|
local function entry (script_entry)
|
|
local categories, filename = script_entry.categories, script_entry.filename;
|
|
assert(type(categories) == "table" and type(filename) == "string",
|
|
"script database appears corrupt, try `nmap --script-updatedb`");
|
|
local escaped_basename = match(filename, "([^/\\]-)%.nse$") or
|
|
match(filename, "([^/\\]-)$");
|
|
|
|
local r_categories = {all = true}; -- A reverse table of categories
|
|
for i, category in ipairs(categories) do
|
|
assert(type(category) == "string", "bad entry in script database");
|
|
r_categories[lower(category)] = true; -- Lowercase the entry
|
|
end
|
|
-- A matching function for each script rule.
|
|
-- If the pattern directly matches a category (e.g. "all"), then
|
|
-- we return true. Otherwise we test if it is a filename or if
|
|
-- the script_entry.filename matches the pattern.
|
|
local function m (pattern)
|
|
-- Check categories
|
|
if r_categories[lower(pattern)] then return true end
|
|
-- Check filename with wildcards
|
|
pattern = gsub(pattern, "%.nse$", ""); -- remove optional extension
|
|
pattern = gsub(pattern, "[%^%$%(%)%%%.%[%]%+%-%?]", "%%%1"); -- esc magic
|
|
pattern = gsub(pattern, "%*", ".*"); -- change to Lua wildcard
|
|
pattern = "^"..pattern.."$"; -- anchor to beginning and end
|
|
return not not find(escaped_basename, pattern);
|
|
end
|
|
local env = {m = m};
|
|
|
|
for globalized_rule, rule_table in pairs(entry_rules) do
|
|
if setfenv(rule_table.compiled_rule, env)() then -- run the compiled rule
|
|
used_rules[rule_table.original_rule] = true;
|
|
local t, path = cnse.fetchfile_absolute(filename);
|
|
if t == "file" then
|
|
if not files_loaded[path] then
|
|
chosen_scripts[#chosen_scripts+1] = Script.new(path);
|
|
files_loaded[path] = true;
|
|
-- do not break so other rules can be marked as used
|
|
end
|
|
else
|
|
log_error("Warning: Could not load '%s': %s", filename, path);
|
|
break;
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
setfenv(db_closure, {Entry = entry});
|
|
db_closure(); -- Load the scripts
|
|
|
|
-- Now load any scripts listed by name rather than by category.
|
|
for rule, loaded in pairs(used_rules) do
|
|
if not loaded then -- attempt to load the file/directory
|
|
local t, path = cnse.fetchfile_absolute(rule);
|
|
if t == nil then -- perhaps omitted the extension?
|
|
t, path = cnse.fetchfile_absolute(rule..".nse");
|
|
end
|
|
if t == nil then
|
|
error("'"..rule.."' did not match a category, filename, or directory");
|
|
elseif t == "file" and not files_loaded[path] then
|
|
chosen_scripts[#chosen_scripts+1] = Script.new(path);
|
|
files_loaded[path] = true;
|
|
elseif t == "directory" then
|
|
for i, file in ipairs(cnse.dump_dir(path)) do
|
|
if not files_loaded[file] then
|
|
chosen_scripts[#chosen_scripts+1] = Script.new(file);
|
|
files_loaded[file] = true;
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|
|
return chosen_scripts;
|
|
end
|
|
|
|
-- run(threads)
|
|
-- The main loop function for NSE. It handles running all the script threads.
|
|
-- Arguments:
|
|
-- threads An array of threads (a runlevel) to run.
|
|
local function run (threads)
|
|
-- running scripts may be resumed at any time. waiting scripts are
|
|
-- yielded until Nsock wakes them. After being awakened with
|
|
-- nse_restore, waiting threads become pending and later are moved all
|
|
-- at once back to running.
|
|
local running, waiting, pending = {}, {}, {};
|
|
local all = setmetatable({}, {__mode = "kv"}); -- base coroutine to Thread
|
|
-- hosts maps a host to a list of threads for that host.
|
|
local hosts, total = {}, 0;
|
|
local current;
|
|
local progress = cnse.scan_progress_meter(NAME);
|
|
|
|
print_debug(1, "NSE Script Threads (%d) running:", #threads);
|
|
while #threads > 0 do
|
|
local thread = remove(threads);
|
|
thread:d("Starting %THREAD against %s%s.", thread.host.ip,
|
|
thread.port and ":"..thread.port.number or "");
|
|
all[thread.co], running[thread.co], total = thread, thread, total+1;
|
|
hosts[thread.host] = hosts[thread.host] or {};
|
|
hosts[thread.host][thread.co] = true;
|
|
end
|
|
|
|
-- Map of yielded threads to the base Thread
|
|
local yielded_base = setmetatable({}, {__mode = "kv"});
|
|
-- _R[YIELD] is called by nse_yield in nse_main.cc
|
|
_R[YIELD] = function (co)
|
|
yielded_base[co] = current; -- set base
|
|
return NSE_YIELD_VALUE; -- return NSE_YIELD_VALUE
|
|
end
|
|
_R[BASE] = function ()
|
|
return current.co;
|
|
end
|
|
-- _R[WAITING_TO_RUNNING] is called by nse_restore in nse_main.cc
|
|
_R[WAITING_TO_RUNNING] = function (co, ...)
|
|
local base = yielded_base[co] or all[co]; -- translate to base thread
|
|
if base then
|
|
co = base.co;
|
|
if waiting[co] then -- ignore a thread not waiting
|
|
pending[co], waiting[co] = waiting[co], nil;
|
|
pending[co].args = {n = select("#", ...), ...};
|
|
end
|
|
end
|
|
end
|
|
-- _R[DESTRUCTOR] is called by nse_destructor in nse_main.cc
|
|
_R[DESTRUCTOR] = function (what, co, key, destructor)
|
|
local thread = yielded_base[co] or all[co] or current;
|
|
if thread then
|
|
local ch = thread.close_handlers;
|
|
if what == "add" then
|
|
ch[key] = {
|
|
thread = co,
|
|
destructor = destructor
|
|
};
|
|
elseif what == "remove" then
|
|
ch[key] = nil;
|
|
end
|
|
end
|
|
end
|
|
|
|
-- Loop while any thread is running or waiting.
|
|
while next(running) or next(waiting) do
|
|
local nr, nw = table_size(running), table_size(waiting);
|
|
if cnse.key_was_pressed() then
|
|
print_verbose(1, "Active NSE Script Threads: %d (%d waiting)\n",
|
|
nr+nw, nw);
|
|
progress("printStats", 1-(nr+nw)/total);
|
|
elseif progress "mayBePrinted" then
|
|
if verbosity() > 1 or debugging() > 0 then
|
|
progress("printStats", 1-(nr+nw)/total);
|
|
else
|
|
progress("printStatsIfNecessary", 1-(nr+nw)/total);
|
|
end
|
|
end
|
|
|
|
-- Checked for timed-out hosts.
|
|
for co, thread in pairs(waiting) do
|
|
if cnse.timedOut(thread.host) then
|
|
waiting[co] = nil;
|
|
thread:d("%THREAD target timed out");
|
|
thread:close();
|
|
end
|
|
end
|
|
|
|
for co, thread in pairs(running) do
|
|
current, running[co] = thread, nil;
|
|
cnse.startTimeOutClock(thread.host);
|
|
|
|
local s, result = resume(co, unpack(thread.args, 1, thread.args.n));
|
|
if not s then -- script error...
|
|
hosts[thread.host][co] = nil;
|
|
thread:d("%THREAD threw an error!\n%s\n",
|
|
traceback(co, tostring(result)));
|
|
thread:close();
|
|
elseif status(co) == "suspended" then
|
|
if result == NSE_YIELD_VALUE then
|
|
waiting[co] = thread;
|
|
else
|
|
thread:d("%THREAD yielded unexpectedly and cannot be rerun.");
|
|
thread:close();
|
|
end
|
|
elseif status(co) == "dead" then
|
|
hosts[thread.host][co] = nil;
|
|
if type(result) == "string" then
|
|
-- Escape any character outside the range 32-126 except for tab,
|
|
-- carriage return, and line feed. This makes the string safe for
|
|
-- screen display as well as XML (see section 2.2 of the XML spec).
|
|
result = gsub(result, "[^\t\r\n\032-\126]", function(a)
|
|
return format("\\x%02X", byte(a));
|
|
end);
|
|
if thread.type == "host" then
|
|
cnse.host_set_output(thread.host, thread.id, result);
|
|
else
|
|
cnse.port_set_output(thread.host, thread.port, thread.id, result);
|
|
end
|
|
end
|
|
thread:d("Finished %THREAD against %s%s.", thread.host.ip,
|
|
thread.port and ":"..thread.port.number or "");
|
|
thread:close();
|
|
end
|
|
|
|
-- Any more threads running for this host?
|
|
if not next(hosts[thread.host]) then
|
|
cnse.stopTimeOutClock(thread.host);
|
|
end
|
|
end
|
|
|
|
cnse.nsock_loop(50); -- Allow nsock to perform any pending callbacks
|
|
-- Move pending threads back to running.
|
|
for co, thread in pairs(pending) do
|
|
pending[co], running[co] = nil, thread;
|
|
end
|
|
|
|
collectgarbage "step";
|
|
end
|
|
|
|
progress "endTask";
|
|
end
|
|
|
|
do -- Load script arguments (--script-args)
|
|
local args = cnse.scriptargs or "";
|
|
|
|
-- Parse a string in 'str' at 'start'.
|
|
local function parse_string (str, start)
|
|
-- Unquoted
|
|
local uqi, uqj, uqm = find(str,
|
|
"^%s*([^'\"%s{},=][^%s{},=]*)%s*[},=]", start);
|
|
-- Quoted
|
|
local qi, qj, q, qm = find(str, "^%s*(['\"])(.-[^\\])%1%s*[},=]", start);
|
|
-- Empty Quote
|
|
local eqi, eqj = find(str, "^%s*(['\"])%1%s*[},=]", start);
|
|
if uqi then
|
|
return uqm, uqj-1;
|
|
elseif qi then
|
|
return gsub(qm, "\\"..q, q), qj-1;
|
|
elseif eqi then
|
|
return "", eqj-1;
|
|
else
|
|
error("Value around '"..sub(str, start, start+10)..
|
|
"' is invalid or is unterminated by a valid seperator");
|
|
end
|
|
end
|
|
-- Takes 'str' at index 'start' and parses a table.
|
|
-- Returns the table and the place in the string it finished reading.
|
|
local function parse_table (str, start)
|
|
local _, j = find(str, "^%s*{", start);
|
|
local t = {}; -- table we return
|
|
local tmp, nc; -- temporary and next character inspected
|
|
|
|
while true do
|
|
j = j+1; -- move past last token
|
|
|
|
_, j, nc = find(str, "^%s*(%S)", j);
|
|
|
|
if nc == "}" then -- end of table
|
|
return t, j;
|
|
else -- try to read key/value pair, or array value
|
|
local av = false; -- this is an array value?
|
|
if nc == "{" then -- array value
|
|
av, tmp, j = true, parse_table(str, j);
|
|
else
|
|
tmp, j = parse_string(str, j);
|
|
end
|
|
nc = sub(str, j+1, j+1); -- next token
|
|
if not av and nc == "=" then -- key/value?
|
|
_, j, nc = find(str, "^%s*(%S)", j+2);
|
|
if nc == "{" then
|
|
t[tmp], j = parse_table(str, j);
|
|
else -- regular string
|
|
t[tmp], j = parse_string(str, j);
|
|
end
|
|
nc = sub(str, j+1, j+1); -- next token
|
|
else -- not key/value pair, save array value
|
|
t[#t+1] = tmp;
|
|
end
|
|
if nc == "," then j = j+1 end -- skip "," token
|
|
end
|
|
end
|
|
end
|
|
nmap.registry.args = parse_table("{"..args.."}", 1);
|
|
end
|
|
|
|
-- Load all user chosen scripts
|
|
local chosen_scripts = get_chosen_scripts(rules);
|
|
print_verbose(1, "Loaded %d scripts for scanning.", #chosen_scripts);
|
|
for i, script in ipairs(chosen_scripts) do
|
|
print_debug(2, "Loaded '%s'.", script.basename);
|
|
end
|
|
|
|
-- main(hosts)
|
|
-- This is the main function we return to NSE (on the C side) which actually
|
|
-- runs a scan against an array of hosts. nse_main.cc gets this function
|
|
-- by loading and executing nse_main.lua.
|
|
-- Arguments:
|
|
-- hosts An array of hosts to scan.
|
|
return function (hosts)
|
|
if #hosts > 1 then
|
|
print_verbose(1, "Script scanning %d hosts.", #hosts);
|
|
elseif #hosts == 1 then
|
|
print_verbose(1, "Script scanning %s.", hosts[1].ip);
|
|
end
|
|
|
|
-- Set up the runlevels.
|
|
local threads, runlevels = {}, {};
|
|
for j, host in ipairs(hosts) do
|
|
-- Check hostrules for this host.
|
|
for i, script in ipairs(chosen_scripts) do
|
|
local thread = script:new_thread("hostrule", tcopy(host));
|
|
if thread then
|
|
local runlevel = thread.runlevel;
|
|
if threads[runlevel] == nil then insert(runlevels, runlevel); end
|
|
threads[runlevel] = threads[runlevel] or {};
|
|
insert(threads[runlevel], thread);
|
|
thread.args, thread.host = {n = 1, tcopy(host)}, host;
|
|
end
|
|
end
|
|
-- Check portrules for this host.
|
|
for port in cnse.ports(host) do
|
|
for i, script in ipairs(chosen_scripts) do
|
|
local thread = script:new_thread("portrule", tcopy(host),
|
|
tcopy(port));
|
|
if thread then
|
|
local runlevel = thread.runlevel;
|
|
if threads[runlevel] == nil then insert(runlevels, runlevel); end
|
|
threads[runlevel] = threads[runlevel] or {};
|
|
insert(threads[runlevel], thread);
|
|
thread.args, thread.host, thread.port =
|
|
{n = 2, tcopy(host), tcopy(port)}, host, port;
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
sort(runlevels);
|
|
for i, runlevel in ipairs(runlevels) do
|
|
print_verbose(1, "Starting runlevel %g scan", runlevel);
|
|
run(threads[runlevel]);
|
|
end
|
|
|
|
collectgarbage "collect";
|
|
print_verbose(1, "Script Scanning completed.");
|
|
end
|