2016-02-18 16:25:40 +00:00
|
|
|
require 'redcarpet'
|
|
|
|
require 'erb'
|
|
|
|
|
|
|
|
module Redcarpet
|
|
|
|
module Render
|
|
|
|
class MsfMdHTML < Redcarpet::Render::HTML
|
2016-02-20 02:46:39 +00:00
|
|
|
|
2016-02-18 16:25:40 +00:00
|
|
|
def block_code(code, language)
|
2016-07-25 20:38:59 +00:00
|
|
|
code = $1 if code =~ /^<ruby>(.+)<\/ruby>/m
|
|
|
|
|
2016-02-18 16:25:40 +00:00
|
|
|
"<pre>" \
|
|
|
|
"<code>#{code}</code>" \
|
|
|
|
"</pre>"
|
|
|
|
end
|
2016-02-20 02:46:39 +00:00
|
|
|
|
|
|
|
|
|
|
|
def list(content, list_type)
|
|
|
|
if list_type == :unordered && content.scan(/<li>/).flatten.length > 15
|
|
|
|
%Q|<p><div id=\"long_list\"><ul>#{content}<ul></div></p>|
|
2016-03-02 00:52:47 +00:00
|
|
|
elsif list_type == :unordered
|
2016-02-20 02:46:39 +00:00
|
|
|
%Q|<ul>#{content}</ul>|
|
2016-03-02 00:52:47 +00:00
|
|
|
elsif list_type == :ordered
|
|
|
|
%Q|<ol>#{content}</ol>|
|
|
|
|
else
|
|
|
|
content
|
2016-02-20 02:46:39 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-03-08 17:25:15 +00:00
|
|
|
def header(text, header_level)
|
|
|
|
%Q|<h#{header_level}>#{text}</h#{header_level}><hr>|
|
|
|
|
end
|
|
|
|
|
2016-03-08 18:19:27 +00:00
|
|
|
def table(header, body)
|
|
|
|
%Q|<table class="kb_table" cellpadding="5" cellspacing="2" border="1">#{header}#{body}</table><br>|
|
|
|
|
end
|
|
|
|
|
2016-02-18 16:25:40 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-03-08 18:19:27 +00:00
|
|
|
|
2016-02-18 16:25:40 +00:00
|
|
|
module Msf
|
|
|
|
module Util
|
|
|
|
module DocumentGenerator
|
|
|
|
class DocumentNormalizer
|
|
|
|
|
2016-03-03 01:52:11 +00:00
|
|
|
#
|
|
|
|
# Markdown templates
|
|
|
|
#
|
|
|
|
|
2016-03-08 16:10:10 +00:00
|
|
|
CSS_BASE_PATH = 'markdown.css'
|
|
|
|
HTML_TEMPLATE = 'html_template.erb'
|
|
|
|
TEMPLATE_PATH = 'default_template.erb'
|
2016-03-03 01:52:11 +00:00
|
|
|
|
|
|
|
#
|
|
|
|
# Demo templates
|
|
|
|
#
|
|
|
|
|
2016-03-08 16:10:10 +00:00
|
|
|
REMOTE_EXPLOIT_DEMO_TEMPLATE = 'remote_exploit_demo_template.erb'
|
|
|
|
BES_DEMO_TEMPLATE = 'bes_demo_template.erb'
|
|
|
|
HTTPSERVER_DEMO_TEMPLATE = 'httpserver_demo_template.erb'
|
|
|
|
GENERIC_DEMO_TEMPLATE = 'generic_demo_template.erb'
|
|
|
|
LOCALEXPLOIT_DEMO_TEMPLATE = 'localexploit_demo_template.erb'
|
|
|
|
POST_DEMO_TEMPLATE = 'post_demo_template.erb'
|
|
|
|
AUXILIARY_SCANNER_DEMO_TEMPLATE = 'auxiliary_scanner_template.erb'
|
|
|
|
PAYLOAD_DEMO_TEMPLATE = 'payload_demo_template.erb'
|
2016-02-18 16:25:40 +00:00
|
|
|
|
2016-02-18 17:44:14 +00:00
|
|
|
|
|
|
|
# Returns the module document in HTML form.
|
|
|
|
#
|
|
|
|
# @param items [Hash] Items to be documented.
|
2016-02-18 21:02:24 +00:00
|
|
|
# @param kb [String] Additional information to be added in the doc.
|
2016-02-18 17:44:14 +00:00
|
|
|
# @return [String] HTML.
|
2016-02-18 21:02:24 +00:00
|
|
|
def get_md_content(items, kb)
|
2016-02-18 16:25:40 +00:00
|
|
|
@md_template ||= lambda {
|
|
|
|
template = ''
|
2016-03-08 16:10:10 +00:00
|
|
|
path = File.expand_path(File.join(Msf::Config.data_directory, 'markdown_doc', TEMPLATE_PATH))
|
|
|
|
File.open(path, 'rb') { |f| template = f.read }
|
2016-02-18 16:25:40 +00:00
|
|
|
return template
|
|
|
|
}.call
|
2016-03-08 16:25:29 +00:00
|
|
|
md_to_html(ERB.new(@md_template).result(binding()), kb.gsub(/</, '<'))
|
2016-02-18 16:25:40 +00:00
|
|
|
end
|
|
|
|
|
2016-02-18 17:44:14 +00:00
|
|
|
|
2016-02-18 16:25:40 +00:00
|
|
|
private
|
|
|
|
|
2016-02-18 17:44:14 +00:00
|
|
|
|
|
|
|
# Returns the CSS code for the HTML document.
|
|
|
|
#
|
|
|
|
# @return [String]
|
2016-02-18 16:25:40 +00:00
|
|
|
def load_css
|
|
|
|
@css ||= lambda {
|
|
|
|
data = ''
|
2016-03-08 16:10:10 +00:00
|
|
|
path = File.expand_path(File.join(Msf::Config.data_directory, 'markdown_doc', CSS_BASE_PATH))
|
|
|
|
File.open(path, 'rb') { |f| data = f.read }
|
2016-02-18 16:25:40 +00:00
|
|
|
return data
|
|
|
|
}.call
|
|
|
|
end
|
|
|
|
|
2016-02-18 17:44:14 +00:00
|
|
|
|
|
|
|
# Returns the HTML document.
|
|
|
|
#
|
|
|
|
# @param md [String] Markdown document.
|
2016-02-18 21:02:24 +00:00
|
|
|
# @param kb [String] Additional information to add.
|
2016-02-18 17:44:14 +00:00
|
|
|
# @return [String] HTML document.
|
2016-02-18 21:02:24 +00:00
|
|
|
def md_to_html(md, kb)
|
2016-03-08 18:19:27 +00:00
|
|
|
opts = {
|
|
|
|
fenced_code_blocks: true,
|
|
|
|
no_intra_emphasis: true,
|
|
|
|
escape_html: true,
|
|
|
|
tables: true
|
|
|
|
}
|
|
|
|
|
|
|
|
r = Redcarpet::Markdown.new(Redcarpet::Render::MsfMdHTML, opts)
|
2016-02-18 21:02:24 +00:00
|
|
|
ERB.new(@html_template ||= lambda {
|
|
|
|
html_template = ''
|
2016-03-08 16:10:10 +00:00
|
|
|
path = File.expand_path(File.join(Msf::Config.data_directory, 'markdown_doc', HTML_TEMPLATE))
|
|
|
|
File.open(path, 'rb') { |f| html_template = f.read }
|
2016-02-18 21:02:24 +00:00
|
|
|
return html_template
|
|
|
|
}.call).result(binding())
|
2016-02-18 16:25:40 +00:00
|
|
|
end
|
|
|
|
|
2016-02-18 17:44:14 +00:00
|
|
|
|
|
|
|
# Returns the markdown format for pull requests.
|
|
|
|
#
|
|
|
|
# @param pull_requests [Hash] Pull requests
|
|
|
|
# @return [String]
|
2016-02-18 16:25:40 +00:00
|
|
|
def normalize_pull_requests(pull_requests)
|
|
|
|
if pull_requests.kind_of?(PullRequestFinder::Exception)
|
2016-02-18 22:11:34 +00:00
|
|
|
error = pull_requests.message
|
2016-02-20 02:46:39 +00:00
|
|
|
case error
|
|
|
|
when /GITHUB_OAUTH_TOKEN/i
|
|
|
|
error << " [See how]("
|
|
|
|
error << "https://help.github.com/articles/creating-an-access-token-for-command-line-use/"
|
|
|
|
error << ")"
|
|
|
|
end
|
2016-02-18 16:25:40 +00:00
|
|
|
return error
|
|
|
|
end
|
|
|
|
|
|
|
|
formatted_pr = []
|
|
|
|
|
|
|
|
pull_requests.each_pair do |number, pr|
|
|
|
|
formatted_pr << "* <a href=\"https://github.com/rapid7/metasploit-framework/pull/#{number}\">##{number}</a> - #{pr[:title]}"
|
|
|
|
end
|
|
|
|
|
|
|
|
formatted_pr * "\n"
|
|
|
|
end
|
|
|
|
|
2016-02-18 17:44:14 +00:00
|
|
|
|
|
|
|
# Returns the markdown format for module datastore options.
|
|
|
|
#
|
|
|
|
# @param mod_options [Hash] Datastore options
|
|
|
|
# @return [String]
|
2016-02-18 16:25:40 +00:00
|
|
|
def normalize_options(mod_options)
|
|
|
|
required_options = []
|
|
|
|
|
|
|
|
mod_options.each_pair do |name, props|
|
|
|
|
if props.required && props.default.nil?
|
|
|
|
required_options << "* #{name} - #{props.desc}"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
required_options * "\n"
|
|
|
|
end
|
|
|
|
|
2016-02-18 17:44:14 +00:00
|
|
|
|
|
|
|
# Returns the markdown format for module description.
|
|
|
|
#
|
|
|
|
# @param description [String] Module description.
|
|
|
|
# @return [String]
|
2016-02-18 16:25:40 +00:00
|
|
|
def normalize_description(description)
|
|
|
|
Rex::Text.wordwrap(Rex::Text.compress(description))
|
|
|
|
end
|
|
|
|
|
2016-02-18 17:44:14 +00:00
|
|
|
|
|
|
|
# Returns the markdown format for module authors.
|
|
|
|
#
|
2016-04-18 19:00:21 +00:00
|
|
|
# @param authors [Array, String] Module Authors
|
2016-02-18 17:44:14 +00:00
|
|
|
# @return [String]
|
2016-02-18 16:25:40 +00:00
|
|
|
def normalize_authors(authors)
|
|
|
|
if authors.kind_of?(Array)
|
|
|
|
authors.collect { |a| "* #{Rex::Text.html_encode(a)}" } * "\n"
|
|
|
|
else
|
2016-02-18 22:11:34 +00:00
|
|
|
authors
|
2016-02-18 16:25:40 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-02-18 17:44:14 +00:00
|
|
|
|
|
|
|
# Returns the markdown format for module targets.
|
|
|
|
#
|
|
|
|
# @param targets [Array] Module targets.
|
|
|
|
# @return [String]
|
2016-02-18 16:25:40 +00:00
|
|
|
def normalize_targets(targets)
|
|
|
|
targets.collect { |c| "* #{c.name}" } * "\n"
|
|
|
|
end
|
|
|
|
|
2016-02-18 17:44:14 +00:00
|
|
|
|
|
|
|
# Returns the markdown format for module references.
|
|
|
|
#
|
|
|
|
# @param refs [Array] Module references.
|
|
|
|
# @return [String]
|
2016-02-18 16:25:40 +00:00
|
|
|
def normalize_references(refs)
|
|
|
|
refs.collect { |r| "* <a href=\"#{r}\">#{r}</a>" } * "\n"
|
|
|
|
end
|
|
|
|
|
2016-02-18 17:44:14 +00:00
|
|
|
|
|
|
|
# Returns the markdown format for module platforms.
|
|
|
|
#
|
2016-04-18 19:00:21 +00:00
|
|
|
# @param platforms [Array, String] Module platforms.
|
2016-02-18 17:44:14 +00:00
|
|
|
# @return [String]
|
2016-02-18 16:25:40 +00:00
|
|
|
def normalize_platforms(platforms)
|
|
|
|
if platforms.kind_of?(Array)
|
|
|
|
platforms.collect { |p| "* #{p}" } * "\n"
|
|
|
|
else
|
|
|
|
platforms
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-02-18 17:44:14 +00:00
|
|
|
|
|
|
|
# Returns the markdown format for module rank.
|
|
|
|
#
|
|
|
|
# @param rank [String] Module rank.
|
|
|
|
# @return [String]
|
2016-02-18 16:25:40 +00:00
|
|
|
def normalize_rank(rank)
|
|
|
|
"[#{Msf::RankingName[rank].capitalize}](https://github.com/rapid7/metasploit-framework/wiki/Exploit-Ranking)"
|
|
|
|
end
|
|
|
|
|
2016-02-18 17:44:14 +00:00
|
|
|
|
2016-03-08 16:10:10 +00:00
|
|
|
# Returns a parsed demo ERB template.
|
2016-02-18 17:44:14 +00:00
|
|
|
#
|
|
|
|
# @param mod [Msf::Module] Metasploit module.
|
|
|
|
# @param path [String] Template path.
|
|
|
|
# @return [String]
|
2016-03-08 16:10:10 +00:00
|
|
|
def load_demo_template(mod, path)
|
2016-02-18 16:25:40 +00:00
|
|
|
data = ''
|
2016-03-08 16:10:10 +00:00
|
|
|
path = File.expand_path(File.join(Msf::Config.data_directory, 'markdown_doc', path))
|
2016-02-18 16:25:40 +00:00
|
|
|
File.open(path, 'rb') { |f| data = f.read }
|
|
|
|
ERB.new(data).result(binding())
|
|
|
|
end
|
|
|
|
|
2016-02-18 17:44:14 +00:00
|
|
|
|
2016-03-03 06:53:00 +00:00
|
|
|
# Returns whether the module is a remote exploit or not.
|
|
|
|
#
|
|
|
|
# @param mod [Msf::Module] Metasploit module.
|
|
|
|
# @return [TrueClass] Module is a remote exploit.
|
|
|
|
# @return [FalseClass] Module is not really a remote exploit.
|
|
|
|
def is_remote_exploit?(mod)
|
|
|
|
# It's actually a little tricky to determine this, so we'll try to be as
|
|
|
|
# specific as possible. Rather have false negatives than false positives,
|
|
|
|
# because the worst case would be using the generic demo template.
|
|
|
|
mod.type == 'exploit' && # Must be an exploit
|
|
|
|
mod.kind_of?(Msf::Exploit::Remote) && # Should always have this
|
|
|
|
!mod.kind_of?(Msf::Exploit::FILEFORMAT) && # Definitely not a file format
|
|
|
|
!mod.kind_of?(Msf::Exploit::Remote::TcpServer) && # If there is a server mixin, things might get complicated
|
|
|
|
mod.options['DisablePayloadHandler'] # Must allow this option
|
|
|
|
end
|
|
|
|
|
|
|
|
|
2016-02-18 17:44:14 +00:00
|
|
|
# Returns a demo template suitable for the module. Currently supported templates:
|
|
|
|
# BrowserExploitServer modules, HttpServer modules, local exploit modules, post
|
|
|
|
# modules, payloads, auxiliary scanner modules.
|
|
|
|
#
|
|
|
|
# @param mod [Msf::Module] Metasploit module.
|
|
|
|
# @return [String]
|
2016-02-18 16:25:40 +00:00
|
|
|
def normalize_demo_output(mod)
|
2016-02-18 21:19:39 +00:00
|
|
|
if mod.kind_of?(Msf::Exploit::Remote::BrowserExploitServer) && mod.shortname != 'browser_autopwn2'
|
2016-03-08 16:10:10 +00:00
|
|
|
load_demo_template(mod, BES_DEMO_TEMPLATE)
|
2016-02-18 16:25:40 +00:00
|
|
|
elsif mod.kind_of?(Msf::Exploit::Remote::HttpServer)
|
2016-03-08 16:10:10 +00:00
|
|
|
load_demo_template(mod, HTTPSERVER_DEMO_TEMPLATE)
|
2016-02-18 16:25:40 +00:00
|
|
|
elsif mod.kind_of?(Msf::Exploit::Local)
|
2016-03-08 16:10:10 +00:00
|
|
|
load_demo_template(mod, LOCALEXPLOIT_DEMO_TEMPLATE)
|
2016-02-18 16:25:40 +00:00
|
|
|
elsif mod.kind_of?(Msf::Post)
|
2016-03-08 16:10:10 +00:00
|
|
|
load_demo_template(mod, POST_DEMO_TEMPLATE)
|
2016-02-18 16:25:40 +00:00
|
|
|
elsif mod.kind_of?(Msf::Payload)
|
2016-03-08 16:10:10 +00:00
|
|
|
load_demo_template(mod, PAYLOAD_DEMO_TEMPLATE)
|
2016-02-18 16:25:40 +00:00
|
|
|
elsif mod.kind_of?(Msf::Auxiliary::Scanner)
|
2016-03-08 16:10:10 +00:00
|
|
|
load_demo_template(mod, AUXILIARY_SCANNER_DEMO_TEMPLATE)
|
2016-03-03 06:53:00 +00:00
|
|
|
elsif is_remote_exploit?(mod)
|
2016-03-08 16:10:10 +00:00
|
|
|
load_demo_template(mod, REMOTE_EXPLOIT_DEMO_TEMPLATE)
|
2016-02-18 16:25:40 +00:00
|
|
|
else
|
2016-03-08 16:10:10 +00:00
|
|
|
load_demo_template(mod, GENERIC_DEMO_TEMPLATE)
|
2016-02-18 16:25:40 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2016-02-18 17:44:14 +00:00
|
|
|
end
|