2005-07-09 21:26:44 +00:00
|
|
|
module Msf
|
|
|
|
module Serializer
|
|
|
|
|
|
|
|
###
|
|
|
|
#
|
|
|
|
# This class formats information in a plain-text format that
|
|
|
|
# is meant to be displayed on a console or some other non-GUI
|
|
|
|
# medium.
|
|
|
|
#
|
|
|
|
###
|
|
|
|
class ReadableText
|
|
|
|
|
2006-10-29 17:31:53 +00:00
|
|
|
DefaultColumnWrap = 70
|
|
|
|
DefaultIndent = 2
|
2005-07-17 02:04:39 +00:00
|
|
|
|
2005-07-09 21:26:44 +00:00
|
|
|
#
|
|
|
|
# Returns a formatted string that contains information about
|
|
|
|
# the supplied module instance.
|
|
|
|
#
|
2006-10-29 17:31:53 +00:00
|
|
|
def self.dump_module(mod, indent = " ")
|
2005-07-09 21:26:44 +00:00
|
|
|
case mod.type
|
|
|
|
when MODULE_PAYLOAD
|
2005-07-10 00:49:12 +00:00
|
|
|
return dump_payload_module(mod, indent)
|
|
|
|
when MODULE_NOP
|
2005-07-10 21:01:05 +00:00
|
|
|
return dump_basic_module(mod, indent)
|
|
|
|
when MODULE_ENCODER
|
|
|
|
return dump_basic_module(mod, indent)
|
2005-07-09 21:26:44 +00:00
|
|
|
when MODULE_EXPLOIT
|
2005-07-10 00:49:12 +00:00
|
|
|
return dump_exploit_module(mod, indent)
|
2006-01-24 03:59:44 +00:00
|
|
|
when MODULE_AUX
|
|
|
|
return dump_auxiliary_module(mod, indent)
|
2005-07-09 21:26:44 +00:00
|
|
|
else
|
2005-07-10 00:49:12 +00:00
|
|
|
return dump_generic_module(mod, indent)
|
2005-07-09 21:26:44 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2005-10-02 05:48:05 +00:00
|
|
|
#
|
|
|
|
# Dumps an exploit's targets.
|
|
|
|
#
|
|
|
|
def self.dump_exploit_targets(mod, indent = '', h = nil)
|
|
|
|
tbl = Rex::Ui::Text::Table.new(
|
|
|
|
'Indent' => indent.length,
|
|
|
|
'Header' => h,
|
|
|
|
'Columns' =>
|
|
|
|
[
|
|
|
|
'Id',
|
|
|
|
'Name',
|
|
|
|
])
|
|
|
|
|
|
|
|
mod.targets.each_with_index { |target, idx|
|
|
|
|
tbl << [ idx.to_s, target.name || 'All' ]
|
|
|
|
}
|
|
|
|
|
|
|
|
tbl.to_s + "\n"
|
|
|
|
end
|
|
|
|
|
2006-04-02 22:33:34 +00:00
|
|
|
#
|
|
|
|
# Dumps an auxiliary's actions
|
|
|
|
#
|
|
|
|
def self.dump_auxiliary_actions(mod, indent = '', h = nil)
|
|
|
|
tbl = Rex::Ui::Text::Table.new(
|
|
|
|
'Indent' => indent.length,
|
|
|
|
'Header' => h,
|
|
|
|
'Columns' =>
|
|
|
|
[
|
|
|
|
'Name',
|
|
|
|
'Description'
|
|
|
|
])
|
|
|
|
|
|
|
|
mod.actions.each_with_index { |target, idx|
|
|
|
|
tbl << [ target.name || 'All' ]
|
|
|
|
}
|
|
|
|
|
|
|
|
tbl.to_s + "\n"
|
|
|
|
end
|
|
|
|
|
2005-11-15 15:11:43 +00:00
|
|
|
#
|
|
|
|
# Dumps the table of payloads that are compatible with the supplied
|
|
|
|
# exploit.
|
|
|
|
#
|
2005-10-02 05:48:05 +00:00
|
|
|
def self.dump_compatible_payloads(exploit, indent = '', h = nil)
|
|
|
|
tbl = Rex::Ui::Text::Table.new(
|
|
|
|
'Indent' => indent.length,
|
|
|
|
'Header' => h,
|
|
|
|
'Columns' =>
|
|
|
|
[
|
|
|
|
'Name',
|
|
|
|
'Description',
|
|
|
|
])
|
|
|
|
|
|
|
|
exploit.compatible_payloads.each { |entry|
|
|
|
|
tbl << [ entry[0], entry[1].new.description ]
|
|
|
|
}
|
|
|
|
|
|
|
|
tbl.to_s + "\n"
|
|
|
|
end
|
|
|
|
|
2005-07-11 15:34:31 +00:00
|
|
|
#
|
|
|
|
# Dumps information about an exploit module.
|
|
|
|
#
|
|
|
|
def self.dump_exploit_module(mod, indent = '')
|
|
|
|
output = "\n"
|
|
|
|
output += " Name: #{mod.name}\n"
|
|
|
|
output += " Version: #{mod.version}\n"
|
|
|
|
output += " Platform: #{mod.platform_to_s}\n"
|
|
|
|
output += " Privileged: " + (mod.privileged? ? "Yes" : "No") + "\n"
|
2006-01-22 19:25:55 +00:00
|
|
|
output += " License: #{mod.license}\n"
|
2005-07-11 15:34:31 +00:00
|
|
|
output += "\n"
|
|
|
|
|
|
|
|
# Authors
|
|
|
|
output += "Provided by:\n"
|
|
|
|
mod.each_author { |author|
|
|
|
|
output += indent + author.to_s + "\n"
|
|
|
|
}
|
|
|
|
output += "\n"
|
|
|
|
|
|
|
|
# Targets
|
|
|
|
output += "Available targets:\n"
|
2005-10-02 05:48:05 +00:00
|
|
|
output += dump_exploit_targets(mod, indent)
|
|
|
|
|
2005-07-11 15:34:31 +00:00
|
|
|
# Options
|
|
|
|
if (mod.options.has_options?)
|
2006-10-29 17:31:53 +00:00
|
|
|
output += "Basic options:\n"
|
2005-10-02 05:48:05 +00:00
|
|
|
output += dump_options(mod, indent)
|
2005-07-11 15:34:31 +00:00
|
|
|
output += "\n"
|
|
|
|
end
|
2006-01-05 03:57:12 +00:00
|
|
|
|
2005-07-11 15:34:31 +00:00
|
|
|
# Payload information
|
2005-07-16 16:06:44 +00:00
|
|
|
if (mod.payload_info.length)
|
2005-07-11 15:34:31 +00:00
|
|
|
output += "Payload information:\n"
|
2005-07-14 07:32:11 +00:00
|
|
|
if (mod.payload_space)
|
|
|
|
output += indent + "Space: " + mod.payload_space.to_s + "\n"
|
|
|
|
end
|
|
|
|
if (mod.payload_badchars)
|
|
|
|
output += indent + "Avoid: " + mod.payload_badchars.length.to_s + " characters\n"
|
|
|
|
end
|
2005-07-11 15:34:31 +00:00
|
|
|
output += "\n"
|
|
|
|
end
|
|
|
|
|
|
|
|
# Description
|
|
|
|
output += "Description:\n"
|
2005-10-01 06:15:39 +00:00
|
|
|
output += word_wrap(Rex::Text.compress(mod.description))
|
2005-07-11 16:46:47 +00:00
|
|
|
output += "\n"
|
|
|
|
|
|
|
|
# References
|
2005-07-14 07:32:11 +00:00
|
|
|
if (mod.references.length > 0)
|
|
|
|
output += "References:\n"
|
|
|
|
mod.references.each { |ref|
|
|
|
|
output += indent + ref.to_s + "\n"
|
|
|
|
}
|
|
|
|
output += "\n"
|
|
|
|
end
|
2005-07-11 15:34:31 +00:00
|
|
|
|
|
|
|
return output
|
|
|
|
|
2005-07-09 21:26:44 +00:00
|
|
|
end
|
|
|
|
|
2005-10-29 13:47:07 +00:00
|
|
|
#
|
2006-01-24 03:59:44 +00:00
|
|
|
# Dumps information about an auxiliary module.
|
2005-10-29 13:47:07 +00:00
|
|
|
#
|
2006-01-24 03:59:44 +00:00
|
|
|
def self.dump_auxiliary_module(mod, indent = '')
|
2005-10-29 13:47:07 +00:00
|
|
|
output = "\n"
|
|
|
|
output += " Name: #{mod.name}\n"
|
|
|
|
output += " Version: #{mod.version}\n"
|
|
|
|
output += "\n"
|
|
|
|
|
|
|
|
# Authors
|
|
|
|
output += "Provided by:\n"
|
|
|
|
mod.each_author { |author|
|
|
|
|
output += indent + author.to_s + "\n"
|
|
|
|
}
|
|
|
|
output += "\n"
|
|
|
|
|
|
|
|
# Options
|
|
|
|
if (mod.options.has_options?)
|
2006-10-29 17:31:53 +00:00
|
|
|
output += "Basic options:\n"
|
2005-10-29 13:47:07 +00:00
|
|
|
output += dump_options(mod, indent)
|
|
|
|
output += "\n"
|
|
|
|
end
|
2006-01-05 03:57:12 +00:00
|
|
|
|
2005-10-29 13:47:07 +00:00
|
|
|
# Description
|
|
|
|
output += "Description:\n"
|
|
|
|
output += word_wrap(Rex::Text.compress(mod.description))
|
|
|
|
output += "\n"
|
|
|
|
|
|
|
|
return output
|
|
|
|
end
|
|
|
|
|
2005-07-09 21:26:44 +00:00
|
|
|
#
|
|
|
|
# Dumps information about a payload module.
|
|
|
|
#
|
2005-07-11 15:34:31 +00:00
|
|
|
def self.dump_payload_module(mod, indent = '')
|
2005-07-09 21:26:44 +00:00
|
|
|
# General
|
|
|
|
output = "\n"
|
|
|
|
output += " Name: #{mod.name}\n"
|
|
|
|
output += " Version: #{mod.version}\n"
|
2005-07-11 04:09:16 +00:00
|
|
|
output += " Platform: #{mod.platform_to_s}\n"
|
2005-12-14 15:09:54 +00:00
|
|
|
output += " Arch: #{mod.arch_to_s}\n"
|
2005-07-09 21:26:44 +00:00
|
|
|
output += "Needs Admin: " + (mod.privileged? ? "Yes" : "No") + "\n"
|
|
|
|
output += " Total size: #{mod.size}\n"
|
|
|
|
output += "\n"
|
|
|
|
|
|
|
|
# Authors
|
|
|
|
output += "Provided by:\n"
|
|
|
|
mod.each_author { |author|
|
|
|
|
output += indent + author.to_s + "\n"
|
|
|
|
}
|
|
|
|
output += "\n"
|
|
|
|
|
|
|
|
# Options
|
2005-07-10 00:49:12 +00:00
|
|
|
if (mod.options.has_options?)
|
2006-10-29 17:31:53 +00:00
|
|
|
output += "Basic options:\n"
|
2005-07-10 00:49:12 +00:00
|
|
|
output += dump_options(mod)
|
|
|
|
output += "\n"
|
|
|
|
end
|
2005-07-09 21:26:44 +00:00
|
|
|
|
2005-07-10 00:49:12 +00:00
|
|
|
# Description
|
|
|
|
output += "Description:\n"
|
2005-10-01 06:15:39 +00:00
|
|
|
output += word_wrap(Rex::Text.compress(mod.description))
|
2005-07-10 00:49:12 +00:00
|
|
|
output += "\n\n"
|
|
|
|
|
|
|
|
return output
|
|
|
|
end
|
|
|
|
|
|
|
|
#
|
2005-07-10 21:01:05 +00:00
|
|
|
# Dumps information about a module, just the basics.
|
2005-07-10 00:49:12 +00:00
|
|
|
#
|
2005-07-11 15:34:31 +00:00
|
|
|
def self.dump_basic_module(mod, indent = '')
|
2005-07-10 00:49:12 +00:00
|
|
|
# General
|
|
|
|
output = "\n"
|
|
|
|
output += " Name: #{mod.name}\n"
|
|
|
|
output += " Version: #{mod.version}\n"
|
2005-07-10 21:01:05 +00:00
|
|
|
output += " Platform: #{mod.platform_to_s}\n"
|
2005-12-14 15:09:54 +00:00
|
|
|
output += " Arch: #{mod.arch_to_s}\n"
|
2005-07-10 00:49:12 +00:00
|
|
|
output += "\n"
|
|
|
|
|
|
|
|
# Authors
|
|
|
|
output += "Provided by:\n"
|
|
|
|
mod.each_author { |author|
|
|
|
|
output += indent + author.to_s + "\n"
|
|
|
|
}
|
2005-07-09 21:26:44 +00:00
|
|
|
output += "\n"
|
2006-01-05 03:57:12 +00:00
|
|
|
|
2005-07-09 21:26:44 +00:00
|
|
|
# Description
|
|
|
|
output += "Description:\n"
|
2005-10-01 06:15:39 +00:00
|
|
|
output += word_wrap(Rex::Text.compress(mod.description))
|
2005-07-09 21:26:44 +00:00
|
|
|
output += "\n\n"
|
|
|
|
|
|
|
|
return output
|
2005-07-10 00:49:12 +00:00
|
|
|
|
2005-07-09 21:26:44 +00:00
|
|
|
end
|
|
|
|
|
2005-07-11 15:34:31 +00:00
|
|
|
def self.dump_generic_module(mod, indent = '')
|
2005-07-09 21:26:44 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
#
|
|
|
|
# Dumps the list of options associated with the
|
|
|
|
# supplied module.
|
|
|
|
#
|
2005-10-02 05:48:05 +00:00
|
|
|
def self.dump_options(mod, indent = '')
|
2005-07-09 21:26:44 +00:00
|
|
|
tbl = Rex::Ui::Text::Table.new(
|
2005-10-02 05:48:05 +00:00
|
|
|
'Indent' => indent.length,
|
2005-07-09 21:26:44 +00:00
|
|
|
'Columns' =>
|
|
|
|
[
|
|
|
|
'Name',
|
2006-01-20 19:15:27 +00:00
|
|
|
'Current Setting',
|
2005-07-11 05:44:36 +00:00
|
|
|
'Required',
|
2005-07-09 21:26:44 +00:00
|
|
|
'Description'
|
|
|
|
])
|
|
|
|
|
2005-07-11 15:34:31 +00:00
|
|
|
mod.options.sorted.each { |entry|
|
|
|
|
name, opt = entry
|
|
|
|
|
2005-07-09 21:26:44 +00:00
|
|
|
next if (opt.advanced?)
|
2006-01-05 03:57:12 +00:00
|
|
|
next if (opt.evasion?)
|
|
|
|
|
2005-07-09 21:26:44 +00:00
|
|
|
val = mod.datastore[name] || opt.default || ''
|
|
|
|
|
2005-07-11 05:44:36 +00:00
|
|
|
tbl << [ name, val.to_s, opt.required? ? "yes" : "no", opt.desc ]
|
2005-07-09 21:26:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return tbl.to_s
|
|
|
|
end
|
|
|
|
|
2005-11-15 15:11:43 +00:00
|
|
|
#
|
|
|
|
# Dumps the advanced options associated with the supplied module.
|
|
|
|
#
|
2005-10-02 05:48:05 +00:00
|
|
|
def self.dump_advanced_options(mod, indent = '')
|
2005-07-11 05:49:04 +00:00
|
|
|
output = ''
|
2005-10-02 05:48:05 +00:00
|
|
|
pad = indent
|
2005-07-09 21:26:44 +00:00
|
|
|
|
2005-07-11 15:34:31 +00:00
|
|
|
mod.options.sorted.each { |entry|
|
|
|
|
name, opt = entry
|
|
|
|
|
2005-07-09 21:26:44 +00:00
|
|
|
next if (!opt.advanced?)
|
|
|
|
|
|
|
|
val = mod.datastore[name] || opt.default || ''
|
2006-01-11 22:15:04 +00:00
|
|
|
desc = word_wrap(opt.desc, indent.length + 3)
|
|
|
|
desc = desc.slice(indent.length + 3, desc.length)
|
2005-07-09 21:26:44 +00:00
|
|
|
|
2006-01-20 19:15:27 +00:00
|
|
|
output += pad + "Name : #{name}\n"
|
|
|
|
output += pad + "Current Setting: #{val}\n"
|
|
|
|
output += pad + "Description : #{desc}\n"
|
2005-07-09 21:26:44 +00:00
|
|
|
}
|
|
|
|
|
2005-07-11 05:49:04 +00:00
|
|
|
return output
|
2005-07-09 21:26:44 +00:00
|
|
|
end
|
|
|
|
|
2006-01-05 03:57:12 +00:00
|
|
|
#
|
|
|
|
# Dumps the evasion options associated with the supplied module.
|
|
|
|
#
|
|
|
|
def self.dump_evasion_options(mod, indent = '')
|
|
|
|
output = ''
|
|
|
|
pad = indent
|
|
|
|
|
|
|
|
mod.options.sorted.each { |entry|
|
|
|
|
name, opt = entry
|
|
|
|
|
|
|
|
next if (!opt.evasion?)
|
|
|
|
|
|
|
|
val = mod.datastore[name] || opt.default || ''
|
2006-01-11 22:15:04 +00:00
|
|
|
|
|
|
|
desc = word_wrap(opt.desc, indent.length + 3)
|
|
|
|
desc = desc.slice(indent.length + 3, desc.length)
|
2006-01-05 03:57:12 +00:00
|
|
|
|
2006-01-20 19:15:27 +00:00
|
|
|
output += pad + "Name : #{name}\n"
|
|
|
|
output += pad + "Current Setting: #{val}\n"
|
|
|
|
output += pad + "Description : #{desc}\n"
|
2006-01-05 03:57:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return output
|
|
|
|
end
|
|
|
|
|
2005-07-14 06:34:58 +00:00
|
|
|
#
|
2005-11-15 15:11:43 +00:00
|
|
|
# Dumps the contents of a datastore.
|
2005-07-14 06:34:58 +00:00
|
|
|
#
|
2005-07-17 02:04:39 +00:00
|
|
|
def self.dump_datastore(name, ds, indent = DefaultIndent, col = DefaultColumnWrap)
|
2005-07-14 06:34:58 +00:00
|
|
|
tbl = Rex::Ui::Text::Table.new(
|
|
|
|
'Indent' => indent,
|
|
|
|
'Header' => name,
|
|
|
|
'Columns' =>
|
|
|
|
[
|
|
|
|
'Name',
|
|
|
|
'Value'
|
|
|
|
])
|
|
|
|
|
2005-07-15 22:30:04 +00:00
|
|
|
ds.keys.sort.each { |k|
|
2005-11-15 21:50:10 +00:00
|
|
|
tbl << [ k, (ds[k] != nil) ? ds[k].to_s : '' ]
|
2005-07-14 06:34:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return ds.length > 0 ? tbl.to_s : "#{tbl.header_to_s}No entries in data store.\n"
|
|
|
|
end
|
|
|
|
|
2005-07-17 02:04:39 +00:00
|
|
|
#
|
2005-11-15 15:11:43 +00:00
|
|
|
# Dumps the list of active sessions.
|
2005-07-17 02:04:39 +00:00
|
|
|
#
|
2006-10-28 01:50:04 +00:00
|
|
|
def self.dump_sessions(framework, verbose = false, indent = DefaultIndent, col = DefaultColumnWrap)
|
|
|
|
columns =
|
|
|
|
[
|
|
|
|
'Id',
|
|
|
|
'Description',
|
|
|
|
'Tunnel'
|
|
|
|
]
|
|
|
|
|
|
|
|
columns << 'Via' if verbose
|
|
|
|
|
2005-07-17 02:04:39 +00:00
|
|
|
tbl = Rex::Ui::Text::Table.new(
|
|
|
|
'Indent' => indent,
|
|
|
|
'Header' => "Active sessions",
|
2006-10-28 01:50:04 +00:00
|
|
|
'Columns' => columns)
|
2005-07-17 02:04:39 +00:00
|
|
|
|
|
|
|
framework.sessions.each_sorted { |k|
|
|
|
|
session = framework.sessions[k]
|
|
|
|
|
2006-10-28 01:50:04 +00:00
|
|
|
row = [ session.sid.to_s, session.desc, session.tunnel_to_s ]
|
|
|
|
row << session.via_exploit if verbose and session.via_exploit
|
|
|
|
|
|
|
|
tbl << row
|
2005-07-17 02:04:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return framework.sessions.length > 0 ? tbl.to_s : "#{tbl.header_to_s}No active sessions.\n"
|
|
|
|
end
|
|
|
|
|
2005-09-22 04:53:46 +00:00
|
|
|
#
|
2005-11-15 15:11:43 +00:00
|
|
|
# Dumps the list of running jobs.
|
2005-09-22 04:53:46 +00:00
|
|
|
#
|
|
|
|
def self.dump_jobs(framework, indent = DefaultIndent, col = DefaultColumnWrap)
|
|
|
|
tbl = Rex::Ui::Text::Table.new(
|
|
|
|
'Indent' => indent,
|
|
|
|
'Header' => "Jobs",
|
|
|
|
'Columns' =>
|
|
|
|
[
|
2005-09-23 14:41:26 +00:00
|
|
|
'Id',
|
2005-09-22 04:53:46 +00:00
|
|
|
'Name'
|
|
|
|
])
|
|
|
|
|
|
|
|
framework.jobs.keys.sort.each { |k|
|
2005-09-23 14:41:26 +00:00
|
|
|
tbl << [ k, framework.jobs[k].name ]
|
2005-09-22 04:53:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return framework.jobs.keys.length > 0 ? tbl.to_s : "#{tbl.header_to_s}No active jobs.\n"
|
|
|
|
end
|
|
|
|
|
2005-07-09 21:26:44 +00:00
|
|
|
#
|
2005-07-10 00:49:12 +00:00
|
|
|
# Jacked from Ernest Ellingson <erne [at] powernav.com>, modified
|
|
|
|
# a bit to add indention
|
2005-07-09 21:26:44 +00:00
|
|
|
#
|
2006-10-29 17:31:53 +00:00
|
|
|
def self.word_wrap(str, indent = DefaultIndent, col = DefaultColumnWrap)
|
2005-07-10 07:15:20 +00:00
|
|
|
return Rex::Text.wordwrap(str, indent, col)
|
2005-07-09 21:26:44 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
end
|
|
|
|
|
|
|
|
end end
|