Files
metasploit-gs/lib/msf/core/module_set.rb
T

342 lines
12 KiB
Ruby
Raw Normal View History

2012-10-01 13:09:30 -05:00
# -*- coding: binary -*-
require 'msf/core'
require 'pathname'
2012-10-02 16:38:25 -05:00
#
# Define used for a place-holder module that is used to indicate that the
# module has not yet been demand-loaded. Soon to go away.
#
Msf::SymbolicModule = '__SYMBOLIC__'
###
#
# A module set contains zero or more named module classes of an arbitrary
# type.
#
###
class Msf::ModuleSet < Hash
2012-10-02 18:21:24 -05:00
include Msf::Framework::Offspring
2012-10-02 16:38:25 -05:00
# Wrapper that detects if a symbolic module is in use. If it is, it creates an instance to demand load the module
# and then returns the now-loaded class afterwords.
2012-10-01 13:09:30 -05:00
#
2012-10-02 16:38:25 -05:00
# @param [String] name the module reference name
# @return [Msf::Module] instance of the of the Msf::Module subclass with the given reference name
def [](name)
2012-10-02 18:21:24 -05:00
if (super == Msf::SymbolicModule)
2012-10-02 16:38:25 -05:00
create(name)
end
2012-10-01 13:09:30 -05:00
2012-10-02 16:38:25 -05:00
super
end
2012-10-01 13:09:30 -05:00
2013-05-21 08:19:47 -05:00
# Create an instance of the supplied module by its reference name
2012-10-02 16:38:25 -05:00
#
2013-05-21 08:19:47 -05:00
# @param reference_name [String] The module reference name.
2012-12-03 22:23:40 -06:00
# @return [Msf::Module,nil] Instance of the named module or nil if it
# could not be created.
2013-05-21 08:19:47 -05:00
def create(reference_name)
klass = fetch(reference_name, nil)
2012-10-02 16:38:25 -05:00
instance = nil
# If there is no module associated with this class, then try to demand
# load it.
2012-10-02 18:21:24 -05:00
if klass.nil? or klass == Msf::SymbolicModule
2013-05-21 08:19:47 -05:00
framework.modules.load_cached_module(module_type, reference_name)
2012-10-01 13:09:30 -05:00
2012-10-02 16:38:25 -05:00
recalculate
2012-10-01 13:09:30 -05:00
2013-05-21 08:19:47 -05:00
klass = fetch(reference_name, nil)
2012-10-02 16:38:25 -05:00
end
2012-10-01 13:09:30 -05:00
2013-05-21 08:19:47 -05:00
# If the klass is valid for this reference_name, try to create it
unless klass.nil? or klass == Msf::SymbolicModule
2012-10-02 16:38:25 -05:00
instance = klass.new
end
2012-10-01 13:09:30 -05:00
2012-10-02 16:38:25 -05:00
# Notify any general subscribers of the creation event
if instance
self.framework.events.on_module_created(instance)
2015-04-28 22:21:31 -05:00
else
self.delete(reference_name)
2012-10-02 16:38:25 -05:00
end
2012-10-01 13:09:30 -05:00
2012-10-02 16:38:25 -05:00
return instance
end
2012-10-01 13:09:30 -05:00
2012-10-02 16:38:25 -05:00
# Overrides the builtin 'each' operator to avoid the following exception on Ruby 1.9.2+
# "can't add a new key into hash during iteration"
#
# @yield [module_reference_name, module]
2013-05-21 08:19:47 -05:00
# @yieldparam [String] module_reference_name the reference_name of the module.
2012-10-02 16:38:25 -05:00
# @yieldparam [Class] module The module class: a subclass of Msf::Module.
# @return [void]
def each(&block)
list = []
self.keys.sort.each do |sidx|
list << [sidx, self[sidx]]
2012-10-01 13:09:30 -05:00
end
2012-10-02 16:38:25 -05:00
list.each(&block)
end
2012-10-01 13:09:30 -05:00
2012-10-02 16:38:25 -05:00
# Enumerates each module class in the set.
#
# @param opts (see #each_module_list)
# @yield (see #each_module_list)
# @yieldparam (see #each_module_list)
# @return (see #each_module_list)
def each_module(opts = {}, &block)
demand_load_modules
2012-10-01 13:09:30 -05:00
2012-10-02 16:38:25 -05:00
self.mod_sorted = self.sort
2012-10-01 13:09:30 -05:00
2012-10-02 16:38:25 -05:00
each_module_list(mod_sorted, opts, &block)
end
2012-10-01 13:09:30 -05:00
2012-10-02 16:38:25 -05:00
# Custom each_module filtering if an advanced set supports doing extended filtering.
#
# @param opts (see #each_module_list)
# @param [String] name the module reference name
# @param [Array<String, Class>] entry pair of the module reference name and the module class.
# @return [false] if the module should not be filtered; it should be yielded by {#each_module_list}.
# @return [true] if the module should be filtered; it should not be yielded by {#each_module_list}.
def each_module_filter(opts, name, entry)
return false
end
2012-10-01 13:09:30 -05:00
2012-10-02 16:38:25 -05:00
# Enumerates each module class in the set based on their relative ranking to one another. Modules that are ranked
# higher are shown first.
#
# @param opts (see #each_module_list)
# @yield (see #each_module_list)
# @yieldparam (see #each_module_list)
# @return (see #each_module_list)
def each_module_ranked(opts = {}, &block)
demand_load_modules
2012-10-01 13:09:30 -05:00
2015-04-28 22:27:09 -05:00
each_module_list(rank_modules, opts, &block)
2012-10-02 16:38:25 -05:00
end
2012-10-01 13:09:30 -05:00
2012-10-02 16:38:25 -05:00
# Forces all modules in this set to be loaded.
#
# @return [void]
def force_load_set
each_module { |name, mod| }
end
2012-10-01 13:09:30 -05:00
2012-10-02 16:38:25 -05:00
# Initializes a module set that will contain modules of a specific type and expose the mechanism necessary to create
# instances of them.
#
# @param [String] type The type of modules cached by this {Msf::ModuleSet}.
def initialize(type = nil)
2012-10-01 13:09:30 -05:00
#
2012-10-02 16:38:25 -05:00
# Defaults
2012-10-01 13:09:30 -05:00
#
2012-10-02 16:38:25 -05:00
self.ambiguous_module_reference_name_set = Set.new
# Hashes that convey the supported architectures and platforms for a
# given module
self.architectures_by_module = {}
self.platforms_by_module = {}
self.mod_sorted = nil
self.mod_extensions = []
2012-10-01 13:09:30 -05:00
#
2012-10-02 16:38:25 -05:00
# Arguments
2012-10-01 13:09:30 -05:00
#
2012-10-02 16:38:25 -05:00
self.module_type = type
end
2012-10-01 13:09:30 -05:00
2012-10-02 16:38:25 -05:00
# @!attribute [r] module_type
# The type of modules stored by this {Msf::ModuleSet}.
#
# @return [String] type of modules
attr_reader :module_type
2012-10-01 13:09:30 -05:00
2012-10-02 16:38:25 -05:00
# Gives the module set an opportunity to handle a module reload event
#
# @param [Class] mod the module class: a subclass of Msf::Module
# @return [void]
def on_module_reload(mod)
end
2012-10-01 13:09:30 -05:00
2012-10-02 16:38:25 -05:00
# Dummy placeholder to recalculate aliases and other fun things.
#
# @return [void]
def recalculate
end
2012-10-01 13:09:30 -05:00
2013-05-21 08:19:47 -05:00
# Checks to see if the supplied module reference name is valid.
2012-10-02 16:38:25 -05:00
#
2013-05-21 08:19:47 -05:00
# @param reference_name [String] The module reference name.
2012-10-02 16:38:25 -05:00
# @return [true] if the module can be {#create created} and cached.
# @return [false] otherwise
2013-05-21 08:19:47 -05:00
def valid?(reference_name)
create(reference_name)
(self[reference_name]) ? true : false
2012-10-02 16:38:25 -05:00
end
2012-10-01 13:09:30 -05:00
2013-05-21 08:19:47 -05:00
# Adds a module with a the supplied reference_name.
2012-10-02 16:38:25 -05:00
#
2013-05-21 08:19:47 -05:00
# @param [Class<Msf::Module>] klass The module class.
# @param [String] reference_name The module reference name.
# @param [Hash{String => Object}] info optional module information.
# @option info [Array<String>] 'files' List of paths to files that defined
# +klass+.
# @return [Class] The klass parameter modified to have
2016-04-20 11:02:15 -07:00
# Msf::Module.framework, Msf::Module#refname, Msf::Module#file_path,
# and Msf::Module#orig_cls set.
2013-05-21 08:19:47 -05:00
def add_module(klass, reference_name, info = {})
# Set the module's reference_name so that it can be referenced when
2012-10-02 16:38:25 -05:00
# instances are created.
2013-05-21 08:19:47 -05:00
klass.framework = framework
klass.refname = reference_name
klass.file_path = ((info and info['files']) ? info['files'][0] : nil)
klass.orig_cls = klass
2012-10-02 16:38:25 -05:00
2012-10-04 11:14:08 -05:00
# don't want to trigger a create, so use fetch
2013-05-21 08:19:47 -05:00
cached_module = self.fetch(reference_name, nil)
2012-10-02 16:38:25 -05:00
2012-10-02 18:21:24 -05:00
if (cached_module and cached_module != Msf::SymbolicModule)
2013-05-21 08:19:47 -05:00
ambiguous_module_reference_name_set.add(reference_name)
2012-10-02 16:38:25 -05:00
# TODO this isn't terribly helpful since the refnames will always match, that's why they are ambiguous.
2013-05-21 08:19:47 -05:00
wlog("The module #{klass.refname} is ambiguous with #{self[reference_name].refname}.")
2012-10-02 16:38:25 -05:00
end
2012-10-01 13:09:30 -05:00
2013-05-21 08:19:47 -05:00
self[reference_name] = klass
2013-05-21 08:19:47 -05:00
klass
2012-10-02 16:38:25 -05:00
end
2012-10-01 13:09:30 -05:00
2012-12-03 22:23:40 -06:00
protected
2012-10-02 16:38:25 -05:00
# Load all modules that are marked as being symbolic.
#
# @return [void]
def demand_load_modules
2012-12-03 22:23:40 -06:00
found_symbolics = false
2012-10-02 16:38:25 -05:00
# Pre-scan the module list for any symbolic modules
self.each_pair { |name, mod|
2012-10-02 18:21:24 -05:00
if (mod == Msf::SymbolicModule)
2012-12-03 22:23:40 -06:00
found_symbolics = true
2012-10-02 16:38:25 -05:00
mod = create(name)
next if (mod.nil?)
2012-10-01 13:09:30 -05:00
end
2012-10-02 16:38:25 -05:00
}
# If we found any symbolic modules, then recalculate.
2012-12-03 22:23:40 -06:00
if (found_symbolics)
2012-10-02 16:38:25 -05:00
recalculate
2012-10-01 13:09:30 -05:00
end
2012-10-02 16:38:25 -05:00
end
2012-10-01 13:09:30 -05:00
2012-10-02 16:38:25 -05:00
# Enumerates the modules in the supplied array with possible limiting factors.
#
# @param [Array<Array<String, Class>>] ary Array of module reference name and module class pairs
# @param [Hash{String => Object}] opts
# @option opts [Array<String>] 'Arch' List of 1 or more architectures that the module must support. The module need
# only support one of the architectures in the array to be included, not all architectures.
# @option opts [Array<String>] 'Platform' List of 1 or more platforms that the module must support. The module need
# only support one of the platforms in the array to be include, not all platforms.
# @yield [module_reference_name, module]
# @yieldparam [String] module_reference_name the name of module
# @yieldparam [Class] module The module class: a subclass of {Msf::Module}.
# @return [void]
def each_module_list(ary, opts, &block)
ary.each { |entry|
name, mod = entry
# Skip any lingering symbolic modules.
2012-10-02 18:21:24 -05:00
next if (mod == Msf::SymbolicModule)
2012-10-02 16:38:25 -05:00
# Filter out incompatible architectures
if (opts['Arch'])
if (!architectures_by_module[mod])
architectures_by_module[mod] = mod.new.arch
2012-10-01 13:09:30 -05:00
end
2012-10-02 16:38:25 -05:00
next if ((architectures_by_module[mod] & opts['Arch']).empty? == true)
end
2012-10-01 13:09:30 -05:00
2012-10-02 16:38:25 -05:00
# Filter out incompatible platforms
if (opts['Platform'])
if (!platforms_by_module[mod])
platforms_by_module[mod] = mod.new.platform
2012-10-01 13:09:30 -05:00
end
2012-10-02 16:38:25 -05:00
next if ((platforms_by_module[mod] & opts['Platform']).empty? == true)
end
2012-10-01 13:09:30 -05:00
2012-10-02 16:38:25 -05:00
# Custom filtering
next if (each_module_filter(opts, name, entry) == true)
2012-10-01 13:09:30 -05:00
2012-10-02 16:38:25 -05:00
block.call(name, mod)
}
end
2012-10-01 13:09:30 -05:00
2012-10-02 16:38:25 -05:00
# @!attribute [rw] ambiguous_module_reference_name_set
# Set of module reference names that are ambiguous because two or more paths have modules with the same reference
# name
#
# @return [Set<String>] set of module reference names loaded from multiple paths.
attr_accessor :ambiguous_module_reference_name_set
# @!attribute [rw] architectures_by_module
# Maps a module to the list of architectures it supports.
#
# @return [Hash{Class => Array<String>}] Maps module class to Array of architecture Strings.
attr_accessor :architectures_by_module
attr_accessor :mod_extensions
# @!attribute [rw] platforms_by_module
# Maps a module to the list of platforms it supports.
#
# @return [Hash{Class => Array<String>}] Maps module class to Array of platform Strings.
attr_accessor :platforms_by_module
# @!attribute [rw] mod_sorted
# Array of module names and module classes ordered by their names.
#
# @return [Array<Array<String, Class>>] Array of arrays where the inner array is a pair of the module reference
# name and the module class.
attr_accessor :mod_sorted
# @!attribute [w] module_type
# The type of modules stored by this {Msf::ModuleSet}.
#
# @return [String] type of modules
attr_writer :module_type
# Ranks modules based on their constant rank value, if they have one. Modules without a Rank are treated as if they
# had {Msf::NormalRanking} for Rank.
#
# @return [Array<Array<String, Class>>] Array of arrays where the inner array is a pair of the module reference name
# and the module class.
def rank_modules
2015-04-28 22:27:09 -05:00
self.sort_by { |pair| module_rank(*pair) }.reverse!
2012-10-01 13:09:30 -05:00
end
# Retrieves the rank from a loaded, not-yet-loaded, or unloadable Metasploit Module.
#
# @param reference_name [String] The reference name of the Metasploit Module
# @param metasploit_module_class [Class<Msf::Module>, Msf::SymbolicModule] The loaded `Class` for the Metasploit
# Module, or {Msf::SymbolicModule} if the Metasploit Module is not loaded yet.
# @return [Integer] an `Msf::*Ranking`. `Msf::ManualRanking` if `metasploit_module_class` is `nil` or
# {Msf::SymbolicModule} and it could not be loaded by {#create}. Otherwise, the `Rank` constant of the
# `metasploit_module_class` or {Msf::NormalRanking} if `metasploit_module_class` does not define `Rank`.
def module_rank(reference_name, metasploit_module_class)
if metasploit_module_class.nil?
Msf::ManualRanking
elsif metasploit_module_class == Msf::SymbolicModule
# TODO don't create an instance just to get the Class.
created_metasploit_module_instance = create(reference_name)
if created_metasploit_module_instance.nil?
module_rank(reference_name, nil)
else
module_rank(reference_name, created_metasploit_module_instance.class)
end
elsif metasploit_module_class.const_defined? :Rank
metasploit_module_class.const_get :Rank
else
Msf::NormalRanking
end
end
2012-10-01 13:09:30 -05:00
end