2014-11-03 22:20:21 +00:00
|
|
|
# -*- coding: binary -*-
|
|
|
|
require 'resolv'
|
|
|
|
require 'msf/core'
|
|
|
|
require 'rex/socket'
|
|
|
|
|
|
|
|
module Msf
|
|
|
|
|
|
|
|
###
|
|
|
|
#
|
|
|
|
# The base class for all options.
|
|
|
|
#
|
|
|
|
###
|
|
|
|
class OptBase
|
|
|
|
|
|
|
|
#
|
|
|
|
# Initializes a named option with the supplied attribute array.
|
|
|
|
# The array is composed of three values.
|
|
|
|
#
|
|
|
|
# attrs[0] = required (boolean type)
|
|
|
|
# attrs[1] = description (string)
|
|
|
|
# attrs[2] = default value
|
|
|
|
# attrs[3] = possible enum values
|
|
|
|
# attrs[4] = Regex to validate the option
|
|
|
|
#
|
usability improvements ith how base options are registered
This adds named parameters for all of the current array-index based
options. It also allows specifying the description as the 2nd parameter,
allowing the 'required' parameter to be implicitly false (the most
common value).
A simple parameter like:
OptAddress.new('ReverseListenerBindAddress',
[false, 'The specific IP address to bind to on the local system']),
Can now be rewritten as:
OptAddress.new('ReverseListenerBindAddress',
'The specific IP address to bind to on the local system'),
More complex options are also now easier to read:
OptString.new(
'HttpUserAgent',
'The user-agent that the payload should use',
default: Rex::UserAgent.shortest,
aliases: ['MeterpreterUserAgent']
),
This also makes dealing with enums easier because default is implicit
unless specified. This:
OptEnum.new('PayloadProxyType',
[true, 'The proxy type, HTTP or SOCKS', 'HTTP', ['HTTP', 'SOCKS']]),
Becomes:
OptEnum.new('HttpProxyType',
'The proxy type, HTTP or SOCKS', required: true, enums: ['HTTP', 'SOCKS'])
This maintains full backward compatibility with existing code as well.
2017-09-21 00:02:15 +00:00
|
|
|
# Attrs can also be specified explicitly via named parameters, or attrs can
|
|
|
|
# also be a string as standin for the required description field.
|
|
|
|
#
|
|
|
|
def initialize(in_name, attrs = [],
|
|
|
|
required: false, desc: nil, default: nil, enums: [], regex: nil, aliases: [])
|
2014-11-03 22:20:21 +00:00
|
|
|
self.name = in_name
|
|
|
|
self.advanced = false
|
|
|
|
self.evasion = false
|
usability improvements ith how base options are registered
This adds named parameters for all of the current array-index based
options. It also allows specifying the description as the 2nd parameter,
allowing the 'required' parameter to be implicitly false (the most
common value).
A simple parameter like:
OptAddress.new('ReverseListenerBindAddress',
[false, 'The specific IP address to bind to on the local system']),
Can now be rewritten as:
OptAddress.new('ReverseListenerBindAddress',
'The specific IP address to bind to on the local system'),
More complex options are also now easier to read:
OptString.new(
'HttpUserAgent',
'The user-agent that the payload should use',
default: Rex::UserAgent.shortest,
aliases: ['MeterpreterUserAgent']
),
This also makes dealing with enums easier because default is implicit
unless specified. This:
OptEnum.new('PayloadProxyType',
[true, 'The proxy type, HTTP or SOCKS', 'HTTP', ['HTTP', 'SOCKS']]),
Becomes:
OptEnum.new('HttpProxyType',
'The proxy type, HTTP or SOCKS', required: true, enums: ['HTTP', 'SOCKS'])
This maintains full backward compatibility with existing code as well.
2017-09-21 00:02:15 +00:00
|
|
|
self.aliases = aliases
|
|
|
|
|
|
|
|
if attrs.is_a?(String) || attrs.length == 0
|
|
|
|
self.required = required
|
|
|
|
self.desc = attrs.is_a?(String) ? attrs : desc
|
|
|
|
self.enums = [ *(enums) ].map { |x| x.to_s }
|
2017-09-21 14:20:12 +00:00
|
|
|
if default.nil? && enums.length > 0
|
|
|
|
self.default = enums[0]
|
|
|
|
else
|
|
|
|
self.default = default
|
|
|
|
end
|
|
|
|
regex_temp = regex
|
usability improvements ith how base options are registered
This adds named parameters for all of the current array-index based
options. It also allows specifying the description as the 2nd parameter,
allowing the 'required' parameter to be implicitly false (the most
common value).
A simple parameter like:
OptAddress.new('ReverseListenerBindAddress',
[false, 'The specific IP address to bind to on the local system']),
Can now be rewritten as:
OptAddress.new('ReverseListenerBindAddress',
'The specific IP address to bind to on the local system'),
More complex options are also now easier to read:
OptString.new(
'HttpUserAgent',
'The user-agent that the payload should use',
default: Rex::UserAgent.shortest,
aliases: ['MeterpreterUserAgent']
),
This also makes dealing with enums easier because default is implicit
unless specified. This:
OptEnum.new('PayloadProxyType',
[true, 'The proxy type, HTTP or SOCKS', 'HTTP', ['HTTP', 'SOCKS']]),
Becomes:
OptEnum.new('HttpProxyType',
'The proxy type, HTTP or SOCKS', required: true, enums: ['HTTP', 'SOCKS'])
This maintains full backward compatibility with existing code as well.
2017-09-21 00:02:15 +00:00
|
|
|
else
|
2017-09-21 13:05:33 +00:00
|
|
|
if attrs[0].nil?
|
|
|
|
self.required = required
|
|
|
|
else
|
|
|
|
self.required = attrs[0]
|
|
|
|
end
|
usability improvements ith how base options are registered
This adds named parameters for all of the current array-index based
options. It also allows specifying the description as the 2nd parameter,
allowing the 'required' parameter to be implicitly false (the most
common value).
A simple parameter like:
OptAddress.new('ReverseListenerBindAddress',
[false, 'The specific IP address to bind to on the local system']),
Can now be rewritten as:
OptAddress.new('ReverseListenerBindAddress',
'The specific IP address to bind to on the local system'),
More complex options are also now easier to read:
OptString.new(
'HttpUserAgent',
'The user-agent that the payload should use',
default: Rex::UserAgent.shortest,
aliases: ['MeterpreterUserAgent']
),
This also makes dealing with enums easier because default is implicit
unless specified. This:
OptEnum.new('PayloadProxyType',
[true, 'The proxy type, HTTP or SOCKS', 'HTTP', ['HTTP', 'SOCKS']]),
Becomes:
OptEnum.new('HttpProxyType',
'The proxy type, HTTP or SOCKS', required: true, enums: ['HTTP', 'SOCKS'])
This maintains full backward compatibility with existing code as well.
2017-09-21 00:02:15 +00:00
|
|
|
self.desc = attrs[1] || desc
|
|
|
|
self.default = attrs[2] || default
|
|
|
|
self.enums = attrs[3] || enums
|
|
|
|
self.enums = [ *(self.enums) ].map { |x| x.to_s }
|
|
|
|
regex_temp = attrs[4] || regex
|
|
|
|
end
|
|
|
|
|
2014-11-03 22:20:21 +00:00
|
|
|
if regex_temp
|
|
|
|
# convert to string
|
|
|
|
regex_temp = regex_temp.to_s if regex_temp.is_a? Regexp
|
|
|
|
# remove start and end character, they will be added later
|
|
|
|
regex_temp = regex_temp.sub(/^\^/, '').sub(/\$$/, '')
|
|
|
|
# Add start and end marker to match the whole regex
|
|
|
|
regex_temp = "^#{regex_temp}$"
|
|
|
|
begin
|
|
|
|
Regexp.compile(regex_temp)
|
|
|
|
self.regex = regex_temp
|
|
|
|
rescue RegexpError, TypeError => e
|
|
|
|
raise("Invalid Regex #{regex_temp}: #{e}")
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
#
|
|
|
|
# Returns true if this is a required option.
|
|
|
|
#
|
|
|
|
def required?
|
usability improvements ith how base options are registered
This adds named parameters for all of the current array-index based
options. It also allows specifying the description as the 2nd parameter,
allowing the 'required' parameter to be implicitly false (the most
common value).
A simple parameter like:
OptAddress.new('ReverseListenerBindAddress',
[false, 'The specific IP address to bind to on the local system']),
Can now be rewritten as:
OptAddress.new('ReverseListenerBindAddress',
'The specific IP address to bind to on the local system'),
More complex options are also now easier to read:
OptString.new(
'HttpUserAgent',
'The user-agent that the payload should use',
default: Rex::UserAgent.shortest,
aliases: ['MeterpreterUserAgent']
),
This also makes dealing with enums easier because default is implicit
unless specified. This:
OptEnum.new('PayloadProxyType',
[true, 'The proxy type, HTTP or SOCKS', 'HTTP', ['HTTP', 'SOCKS']]),
Becomes:
OptEnum.new('HttpProxyType',
'The proxy type, HTTP or SOCKS', required: true, enums: ['HTTP', 'SOCKS'])
This maintains full backward compatibility with existing code as well.
2017-09-21 00:02:15 +00:00
|
|
|
required
|
2014-11-03 22:20:21 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
#
|
|
|
|
# Returns true if this is an advanced option.
|
|
|
|
#
|
|
|
|
def advanced?
|
usability improvements ith how base options are registered
This adds named parameters for all of the current array-index based
options. It also allows specifying the description as the 2nd parameter,
allowing the 'required' parameter to be implicitly false (the most
common value).
A simple parameter like:
OptAddress.new('ReverseListenerBindAddress',
[false, 'The specific IP address to bind to on the local system']),
Can now be rewritten as:
OptAddress.new('ReverseListenerBindAddress',
'The specific IP address to bind to on the local system'),
More complex options are also now easier to read:
OptString.new(
'HttpUserAgent',
'The user-agent that the payload should use',
default: Rex::UserAgent.shortest,
aliases: ['MeterpreterUserAgent']
),
This also makes dealing with enums easier because default is implicit
unless specified. This:
OptEnum.new('PayloadProxyType',
[true, 'The proxy type, HTTP or SOCKS', 'HTTP', ['HTTP', 'SOCKS']]),
Becomes:
OptEnum.new('HttpProxyType',
'The proxy type, HTTP or SOCKS', required: true, enums: ['HTTP', 'SOCKS'])
This maintains full backward compatibility with existing code as well.
2017-09-21 00:02:15 +00:00
|
|
|
advanced
|
2014-11-03 22:20:21 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
#
|
|
|
|
# Returns true if this is an evasion option.
|
|
|
|
#
|
|
|
|
def evasion?
|
usability improvements ith how base options are registered
This adds named parameters for all of the current array-index based
options. It also allows specifying the description as the 2nd parameter,
allowing the 'required' parameter to be implicitly false (the most
common value).
A simple parameter like:
OptAddress.new('ReverseListenerBindAddress',
[false, 'The specific IP address to bind to on the local system']),
Can now be rewritten as:
OptAddress.new('ReverseListenerBindAddress',
'The specific IP address to bind to on the local system'),
More complex options are also now easier to read:
OptString.new(
'HttpUserAgent',
'The user-agent that the payload should use',
default: Rex::UserAgent.shortest,
aliases: ['MeterpreterUserAgent']
),
This also makes dealing with enums easier because default is implicit
unless specified. This:
OptEnum.new('PayloadProxyType',
[true, 'The proxy type, HTTP or SOCKS', 'HTTP', ['HTTP', 'SOCKS']]),
Becomes:
OptEnum.new('HttpProxyType',
'The proxy type, HTTP or SOCKS', required: true, enums: ['HTTP', 'SOCKS'])
This maintains full backward compatibility with existing code as well.
2017-09-21 00:02:15 +00:00
|
|
|
evasion
|
2014-11-03 22:20:21 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
#
|
|
|
|
# Returns true if the supplied type is equivalent to this option's type.
|
|
|
|
#
|
|
|
|
def type?(in_type)
|
usability improvements ith how base options are registered
This adds named parameters for all of the current array-index based
options. It also allows specifying the description as the 2nd parameter,
allowing the 'required' parameter to be implicitly false (the most
common value).
A simple parameter like:
OptAddress.new('ReverseListenerBindAddress',
[false, 'The specific IP address to bind to on the local system']),
Can now be rewritten as:
OptAddress.new('ReverseListenerBindAddress',
'The specific IP address to bind to on the local system'),
More complex options are also now easier to read:
OptString.new(
'HttpUserAgent',
'The user-agent that the payload should use',
default: Rex::UserAgent.shortest,
aliases: ['MeterpreterUserAgent']
),
This also makes dealing with enums easier because default is implicit
unless specified. This:
OptEnum.new('PayloadProxyType',
[true, 'The proxy type, HTTP or SOCKS', 'HTTP', ['HTTP', 'SOCKS']]),
Becomes:
OptEnum.new('HttpProxyType',
'The proxy type, HTTP or SOCKS', required: true, enums: ['HTTP', 'SOCKS'])
This maintains full backward compatibility with existing code as well.
2017-09-21 00:02:15 +00:00
|
|
|
type == in_type
|
2014-11-03 22:20:21 +00:00
|
|
|
end
|
|
|
|
|
2016-03-29 04:03:17 +00:00
|
|
|
#
|
|
|
|
# Returns true if this option can be validated on assignment
|
|
|
|
#
|
|
|
|
def validate_on_assignment?
|
|
|
|
true
|
|
|
|
end
|
|
|
|
|
2014-11-03 22:20:21 +00:00
|
|
|
#
|
|
|
|
# If it's required and the value is nil or empty, then it's not valid.
|
|
|
|
#
|
2016-05-23 19:56:19 +00:00
|
|
|
def valid?(value, check_empty: true)
|
|
|
|
if check_empty && required?
|
2014-11-03 22:20:21 +00:00
|
|
|
# required variable not set
|
2016-05-23 22:33:06 +00:00
|
|
|
return false if (value.nil? || value.to_s.empty?)
|
2014-11-03 22:20:21 +00:00
|
|
|
end
|
|
|
|
if regex
|
2016-05-23 22:33:06 +00:00
|
|
|
return !!value.match(regex)
|
2014-11-03 22:20:21 +00:00
|
|
|
end
|
usability improvements ith how base options are registered
This adds named parameters for all of the current array-index based
options. It also allows specifying the description as the 2nd parameter,
allowing the 'required' parameter to be implicitly false (the most
common value).
A simple parameter like:
OptAddress.new('ReverseListenerBindAddress',
[false, 'The specific IP address to bind to on the local system']),
Can now be rewritten as:
OptAddress.new('ReverseListenerBindAddress',
'The specific IP address to bind to on the local system'),
More complex options are also now easier to read:
OptString.new(
'HttpUserAgent',
'The user-agent that the payload should use',
default: Rex::UserAgent.shortest,
aliases: ['MeterpreterUserAgent']
),
This also makes dealing with enums easier because default is implicit
unless specified. This:
OptEnum.new('PayloadProxyType',
[true, 'The proxy type, HTTP or SOCKS', 'HTTP', ['HTTP', 'SOCKS']]),
Becomes:
OptEnum.new('HttpProxyType',
'The proxy type, HTTP or SOCKS', required: true, enums: ['HTTP', 'SOCKS'])
This maintains full backward compatibility with existing code as well.
2017-09-21 00:02:15 +00:00
|
|
|
true
|
2014-11-03 22:20:21 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
#
|
|
|
|
# Returns true if the value supplied is nil and it's required to be
|
|
|
|
# a valid value
|
|
|
|
#
|
|
|
|
def empty_required_value?(value)
|
usability improvements ith how base options are registered
This adds named parameters for all of the current array-index based
options. It also allows specifying the description as the 2nd parameter,
allowing the 'required' parameter to be implicitly false (the most
common value).
A simple parameter like:
OptAddress.new('ReverseListenerBindAddress',
[false, 'The specific IP address to bind to on the local system']),
Can now be rewritten as:
OptAddress.new('ReverseListenerBindAddress',
'The specific IP address to bind to on the local system'),
More complex options are also now easier to read:
OptString.new(
'HttpUserAgent',
'The user-agent that the payload should use',
default: Rex::UserAgent.shortest,
aliases: ['MeterpreterUserAgent']
),
This also makes dealing with enums easier because default is implicit
unless specified. This:
OptEnum.new('PayloadProxyType',
[true, 'The proxy type, HTTP or SOCKS', 'HTTP', ['HTTP', 'SOCKS']]),
Becomes:
OptEnum.new('HttpProxyType',
'The proxy type, HTTP or SOCKS', required: true, enums: ['HTTP', 'SOCKS'])
This maintains full backward compatibility with existing code as well.
2017-09-21 00:02:15 +00:00
|
|
|
required? && value.nil?
|
2014-11-03 22:20:21 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
#
|
|
|
|
# Normalizes the supplied value to conform with the type that the option is
|
|
|
|
# conveying.
|
|
|
|
#
|
|
|
|
def normalize(value)
|
|
|
|
value
|
|
|
|
end
|
|
|
|
|
|
|
|
#
|
|
|
|
# Returns a string representing a user-friendly display of the chosen value
|
|
|
|
#
|
|
|
|
def display_value(value)
|
|
|
|
value.to_s
|
|
|
|
end
|
|
|
|
|
|
|
|
#
|
|
|
|
# The name of the option.
|
|
|
|
#
|
|
|
|
attr_reader :name
|
|
|
|
#
|
|
|
|
# Whether or not the option is required.
|
|
|
|
#
|
|
|
|
attr_reader :required
|
|
|
|
#
|
|
|
|
# The description of the option.
|
|
|
|
#
|
|
|
|
attr_reader :desc
|
|
|
|
#
|
|
|
|
# The default value of the option.
|
|
|
|
#
|
|
|
|
attr_reader :default
|
|
|
|
#
|
|
|
|
# Storing the name of the option.
|
|
|
|
#
|
|
|
|
attr_writer :name
|
|
|
|
#
|
|
|
|
# Whether or not this is an advanced option.
|
|
|
|
#
|
|
|
|
attr_accessor :advanced
|
|
|
|
#
|
|
|
|
# Whether or not this is an evasion option.
|
|
|
|
#
|
|
|
|
attr_accessor :evasion
|
|
|
|
#
|
|
|
|
# The module or entity that owns this option.
|
|
|
|
#
|
|
|
|
attr_accessor :owner
|
|
|
|
#
|
|
|
|
# The list of potential valid values
|
|
|
|
#
|
|
|
|
attr_accessor :enums
|
|
|
|
#
|
|
|
|
# A optional regex to validate the option value
|
|
|
|
#
|
|
|
|
attr_accessor :regex
|
2017-08-05 07:22:55 +00:00
|
|
|
#
|
|
|
|
# Aliases for this option for backward compatibility
|
|
|
|
#
|
|
|
|
attr_accessor :aliases
|
2014-11-03 22:20:21 +00:00
|
|
|
|
|
|
|
protected
|
|
|
|
|
|
|
|
attr_writer :required, :desc, :default # :nodoc:
|
|
|
|
end
|
|
|
|
end
|