672 lines
25 KiB
Ruby
672 lines
25 KiB
Ruby
#
|
|
# Copyright (c) 2006-2017 Wade Alcorn - wade@bindshell.net
|
|
# Browser Exploitation Framework (BeEF) - http://beefproject.com
|
|
# See the file 'doc/COPYING' for copying permission
|
|
#
|
|
module BeEF
|
|
module Extension
|
|
module AdminUI
|
|
module Controllers
|
|
|
|
#
|
|
#
|
|
#
|
|
class Modules < BeEF::Extension::AdminUI::HttpController
|
|
|
|
BD = BeEF::Core::Models::BrowserDetails
|
|
|
|
def initialize
|
|
super({
|
|
'paths' => {
|
|
'/getRestfulApiToken.json' => method(:get_restful_api_token),
|
|
'/select/commandmodules/all.json' => method(:select_all_command_modules),
|
|
'/select/commandmodules/tree.json' => method(:select_command_modules_tree),
|
|
'/select/commandmodule.json' => method(:select_command_module),
|
|
'/select/command.json' => method(:select_command),
|
|
'/select/command_results.json' => method(:select_command_results),
|
|
'/select/zombie_summary.json' => method(:select_zombie_summary),
|
|
'/commandmodule/commands.json' => method(:select_command_module_commands),
|
|
'/commandmodule/new' => method(:attach_command_module),
|
|
'/commandmodule/dynamicnew' => method(:attach_dynamic_command_module),
|
|
'/commandmodule/reexecute' => method(:reexecute_command_module)
|
|
}
|
|
})
|
|
|
|
@session = BeEF::Extension::AdminUI::Session.instance
|
|
end
|
|
|
|
# @note Returns the RESTful api key. Authenticated call, so callable only
|
|
# from the admin UI after successful authentication (cookie).
|
|
# -> http://127.0.0.1:3000/ui/modules/getRestfulApiToken.json
|
|
# response
|
|
# <- {"token":"800679edbb59976935d7673924caaa9e99f55c32"}
|
|
def get_restful_api_token
|
|
@body = {
|
|
'token' => BeEF::Core::Configuration.instance.get("beef.api_token")
|
|
}.to_json
|
|
end
|
|
|
|
# Returns a JSON array containing the summary for a selected zombie.
|
|
def select_zombie_summary
|
|
|
|
# get the zombie
|
|
zombie_session = @params['zombie_session'] || nil
|
|
(print_error "Zombie session is nil";return) if zombie_session.nil?
|
|
zombie = BeEF::Core::Models::HookedBrowser.first(:session => zombie_session)
|
|
(print_error "Zombie is nil";return) if zombie.nil?
|
|
|
|
# init the summary grid
|
|
summary_grid_hash = {
|
|
'success' => 'true',
|
|
'results' => []
|
|
}
|
|
|
|
# zombie properties
|
|
# in the form of: category, UI label, value
|
|
zombie_properties = [
|
|
|
|
# Browser
|
|
['Browser', 'Browser Name', 'BrowserName'],
|
|
['Browser', 'Browser Version', 'BrowserVersion'],
|
|
['Browser', 'Browser UA String', 'BrowserReportedName'],
|
|
['Browser', 'Browser Language', 'BrowserLanguage'],
|
|
['Browser', 'Browser Platform', 'BrowserPlatform'],
|
|
['Browser', 'Browser Plugins', 'BrowserPlugins'],
|
|
['Browser', 'Using Proxy', 'UsingProxy'],
|
|
['Browser', 'Proxy Client', 'ProxyClient'],
|
|
['Browser', 'Proxy Server', 'ProxyServer'],
|
|
['Browser', 'Window Size', 'WindowSize'],
|
|
|
|
# Browser Components
|
|
['Browser Components', 'Flash', 'HasFlash'],
|
|
['Browser Components', 'Java', 'JavaEnabled'],
|
|
['Browser Components', 'VBScript', 'VBScriptEnabled'],
|
|
['Browser Components', 'PhoneGap', 'HasPhonegap'],
|
|
['Browser Components', 'Google Gears', 'HasGoogleGears'],
|
|
['Browser Components', 'Web Sockets', 'HasWebSocket'],
|
|
['Browser Components', 'QuickTime', 'HasQuickTime'],
|
|
['Browser Components', 'RealPlayer', 'HasRealPlayer'],
|
|
['Browser Components', 'Windows Media Player','HasWMP'],
|
|
['Browser Components', 'VLC', 'HasVLC'],
|
|
['Browser Components', 'WebRTC', 'HasWebRTC'],
|
|
['Browser Components', 'ActiveX', 'HasActiveX'],
|
|
['Browser Components', 'Session Cookies', 'hasSessionCookies'],
|
|
['Browser Components', 'Persistent Cookies', 'hasPersistentCookies'],
|
|
['Browser Components', 'Unity', 'HasUnity'],
|
|
['Browser Components', 'Foxit', 'HasFoxit'],
|
|
|
|
# Geolocation
|
|
['Location', 'City', 'LocationCity'],
|
|
['Location', 'Country', 'LocationCountry'],
|
|
['Location', 'CountryCode2', 'LocationCountryCode2'],
|
|
['Location', 'CountryCode3', 'LocationCountryCode3'],
|
|
['Location', 'Continent', 'LocationContinentCode'],
|
|
['Location', 'Post Code', 'LocationPostCode'],
|
|
['Location', 'Latitude', 'LocationLatitude'],
|
|
['Location', 'Longitude', 'LocationLongitude'],
|
|
['Location', 'DMA Code', 'LocationDMACode'],
|
|
['Location', 'Area Code', 'LocationAreaCode'],
|
|
['Location', 'Timezone', 'LocationTimezone'],
|
|
['Location', 'Region', 'LocationRegionName'],
|
|
|
|
# Hooked Page
|
|
['Hooked Page', 'Page Title', 'PageTitle'],
|
|
['Hooked Page', 'Page URI', 'PageURI'],
|
|
['Hooked Page', 'Page Referrer', 'PageReferrer'],
|
|
['Hooked Page', 'Host Name/IP', 'HostName'],
|
|
['Hooked Page', 'Cookies', 'Cookies'],
|
|
|
|
# Host
|
|
['Host', 'Host Name/IP', 'IP'],
|
|
['Host', 'Date', 'DateStamp'],
|
|
['Host', 'Operating System', 'OsName'],
|
|
['Host', 'Hardware', 'Hardware'],
|
|
['Host', 'CPU', 'CPU'],
|
|
['Host', 'Default Browser', 'DefaultBrowser'],
|
|
['Host', 'Screen Size', 'ScreenSize'],
|
|
['Host', 'Touch Screen', 'TouchEnabled']
|
|
]
|
|
|
|
# set and add the return values for each browser property
|
|
# in the form of: category, UI label, value
|
|
zombie_properties.each do |p|
|
|
|
|
case p[2]
|
|
when "BrowserName"
|
|
data = BeEF::Core::Constants::Browsers.friendly_name(BD.get(zombie_session, p[2]))
|
|
|
|
when "ScreenSize"
|
|
screen_size = BD.get(zombie_session, "ScreenSize")
|
|
if screen_size.nil?
|
|
data = "Unknown"
|
|
else
|
|
screen_size_hash = JSON.parse(screen_size.gsub(/\"\=\>/, '":')) # tidy up the string for JSON
|
|
width = screen_size_hash['width']
|
|
height = screen_size_hash['height']
|
|
cdepth = screen_size_hash['colordepth']
|
|
data = "Width: #{width}, Height: #{height}, Colour Depth: #{cdepth}"
|
|
end
|
|
when "WindowSize"
|
|
window_size = BD.get(zombie_session, "WindowSize")
|
|
if window_size.nil?
|
|
data = "Unknown"
|
|
else
|
|
window_size_hash = JSON.parse(window_size.gsub(/\"\=\>/, '":')) # tidy up the string for JSON
|
|
width = window_size_hash['width']
|
|
height = window_size_hash['height']
|
|
data = "Width: #{width}, Height: #{height}"
|
|
end
|
|
else
|
|
data = BD.get(zombie_session, p[2])
|
|
end
|
|
|
|
# add property to summary hash
|
|
if not data.nil?
|
|
summary_grid_hash['results'].push({
|
|
'category' => p[0],
|
|
'data' => { p[1] => CGI.escapeHTML("#{data}") },
|
|
'from' => 'Initialization'
|
|
})
|
|
end
|
|
|
|
end
|
|
|
|
@body = summary_grid_hash.to_json
|
|
end
|
|
|
|
# Returns the list of all command_modules in a JSON format
|
|
def select_all_command_modules
|
|
@body = command_modules2json(BeEF::Modules.get_enabled.keys)
|
|
end
|
|
|
|
# Set the correct icon for the command module
|
|
def set_command_module_icon(status)
|
|
path = BeEF::Extension::AdminUI::Constants::Icons::MODULE_TARGET_IMG_PATH # add icon path
|
|
case status
|
|
when BeEF::Core::Constants::CommandModule::VERIFIED_NOT_WORKING
|
|
path += BeEF::Extension::AdminUI::Constants::Icons::VERIFIED_NOT_WORKING_IMG
|
|
when BeEF::Core::Constants::CommandModule::VERIFIED_USER_NOTIFY
|
|
path += BeEF::Extension::AdminUI::Constants::Icons::VERIFIED_USER_NOTIFY_IMG
|
|
when BeEF::Core::Constants::CommandModule::VERIFIED_WORKING
|
|
path += BeEF::Extension::AdminUI::Constants::Icons::VERIFIED_WORKING_IMG
|
|
when BeEF::Core::Constants::CommandModule::VERIFIED_UNKNOWN
|
|
path += BeEF::Extension::AdminUI::Constants::Icons::VERIFIED_UNKNOWN_IMG
|
|
else
|
|
path += BeEF::Extension::AdminUI::Constants::Icons::VERIFIED_UNKNOWN_IMG
|
|
end
|
|
#return path
|
|
path
|
|
end
|
|
|
|
# Set the correct working status for the command module
|
|
def set_command_module_status(mod)
|
|
hook_session_id = @params['zombie_session'] || nil
|
|
if hook_session_id == nil
|
|
return BeEF::Core::Constants::CommandModule::VERIFIED_UNKNOWN
|
|
end
|
|
return BeEF::Module.support(mod, {'browser' => BD.get(hook_session_id, 'BrowserName'), 'ver' => BD.get(hook_session_id, 'BrowserVersion'), 'os' => [BD.get(hook_session_id, 'OsName')]})
|
|
end
|
|
|
|
# If we're adding a leaf to the command tree, and it's in a subfolder, we need to recurse
|
|
# into the tree to find where it goes
|
|
def update_command_module_tree_recurse(tree,category,leaf)
|
|
working_category = category.shift
|
|
|
|
tree.each {|t|
|
|
if t['text'].eql? working_category and category.count > 0
|
|
#We have deeper to go
|
|
update_command_module_tree_recurse(t['children'],category,leaf)
|
|
elsif t['text'].eql? working_category
|
|
#Bingo
|
|
t['children'].push(leaf)
|
|
break
|
|
end
|
|
}
|
|
|
|
#return tree
|
|
|
|
end
|
|
|
|
#Add the command to the tree
|
|
def update_command_module_tree(tree, cmd_category, cmd_icon_path, cmd_status, cmd_name, cmd_id)
|
|
# construct leaf node for the command module tree
|
|
leaf_node = {
|
|
'text' => cmd_name,
|
|
'leaf' => true,
|
|
'icon' => cmd_icon_path,
|
|
'status' => cmd_status,
|
|
'id' => cmd_id
|
|
}
|
|
|
|
# add the node to the branch in the command module tree
|
|
if cmd_category.is_a?(Array)
|
|
#The category is an array, therefore it's a sub-folderised category
|
|
cat_copy = cmd_category.dup #Don't work with the original array, because, then it breaks shit
|
|
update_command_module_tree_recurse(tree,cat_copy,leaf_node)
|
|
else
|
|
#original logic here, simply add the command to the tree.
|
|
tree.each {|x|
|
|
if x['text'].eql? cmd_category
|
|
x['children'].push( leaf_node )
|
|
break
|
|
end
|
|
}
|
|
end
|
|
end
|
|
|
|
#Recursive function to build the tree now with sub-folders
|
|
def build_recursive_tree(parent,input)
|
|
cinput = input.shift.chomp('/')
|
|
if cinput.split('/').count == 1 #then we have a single folder now
|
|
if parent.detect {|p| p['text'] == cinput}.nil?
|
|
parent << {'text' => cinput, 'cls' => 'folder', 'children' => []}
|
|
else
|
|
if input.count > 0
|
|
parent.each {|p|
|
|
if p['text'] == cinput
|
|
p['children'] = build_recursive_tree(p['children'],input)
|
|
end
|
|
}
|
|
end
|
|
end
|
|
else
|
|
#we have multiple folders
|
|
newinput = cinput.split('/')
|
|
newcinput = newinput.shift
|
|
if parent.detect {|p| p['text'] == newcinput }.nil?
|
|
fldr = {'text' => newcinput, 'cls' => 'folder', 'children' => []}
|
|
parent << build_recursive_tree(fldr['children'],newinput)
|
|
else
|
|
parent.each {|p|
|
|
if p['text'] == newcinput
|
|
p['children'] = build_recursive_tree(p['children'],newinput)
|
|
end
|
|
}
|
|
end
|
|
end
|
|
|
|
if input.count > 0
|
|
return build_recursive_tree(parent,input)
|
|
else
|
|
return parent
|
|
end
|
|
end
|
|
|
|
#Recursive function to sort all the parent's children
|
|
def sort_recursive_tree(parent)
|
|
# sort the children nodes by status and name
|
|
parent.each {|x|
|
|
#print_info "Sorting: " + x['children'].to_s
|
|
if x.is_a?(Hash) and x.has_key?('children')
|
|
x['children'] = x['children'].sort_by {|a|
|
|
fldr = a['cls'] ? a['cls'] : 'zzzzz'
|
|
"#{fldr}#{a['status']}#{a['text']}"
|
|
}
|
|
x['children'].each {|c|
|
|
sort_recursive_tree([c]) if c.has_key?('cls') and c['cls'] == 'folder'
|
|
}
|
|
end
|
|
}
|
|
end
|
|
|
|
#Recursive function to retitle folders with the number of children
|
|
def retitle_recursive_tree(parent)
|
|
# append the number of command modules so the branch name results in: "<category name> (num)"
|
|
parent.each {|command_module_branch|
|
|
if command_module_branch.is_a?(Hash) and command_module_branch.has_key?('children')
|
|
num_of_subs = 0
|
|
command_module_branch['children'].each {|c|
|
|
#add in the submodules and subtract 1 for the folder node
|
|
num_of_subs+=c['children'].length-1 if c.has_key?('children')
|
|
retitle_recursive_tree([c]) if c.has_key?('cls') and c['cls'] == 'folder'
|
|
}
|
|
num_of_command_modules = command_module_branch['children'].length + num_of_subs
|
|
command_module_branch['text'] = command_module_branch['text'] + " (" + num_of_command_modules.to_s() + ")"
|
|
|
|
end
|
|
}
|
|
end
|
|
|
|
# Returns the list of all command_modules for a TreePanel in the interface.
|
|
def select_command_modules_tree
|
|
blanktree = []
|
|
tree = []
|
|
|
|
#Due to the sub-folder nesting, we use some really badly hacked together recursion
|
|
#Note to the bored - if someone (anyone please) wants to refactor, I'll buy you cookies. -x
|
|
tree = build_recursive_tree(blanktree,BeEF::Modules.get_categories)
|
|
|
|
BeEF::Modules.get_enabled.each{|k, mod|
|
|
# get the hooked browser session id and set it in the command module
|
|
hook_session_id = @params['zombie_session'] || nil
|
|
(print_error "hook_session_id is nil";return) if hook_session_id.nil?
|
|
|
|
# create url path and file for the command module icon
|
|
command_module_status = set_command_module_status(k)
|
|
command_module_icon_path = set_command_module_icon(command_module_status)
|
|
|
|
update_command_module_tree(tree, mod['category'], command_module_icon_path, command_module_status, mod['name'],mod['db']['id'])
|
|
}
|
|
|
|
# if dynamic modules are found in the DB, then we don't have yaml config for them
|
|
# and loading must proceed in a different way.
|
|
dynamic_modules = BeEF::Core::Models::CommandModule.all(:path.like => "Dynamic/")
|
|
|
|
if(dynamic_modules != nil)
|
|
all_modules = BeEF::Core::Models::CommandModule.all(:order => [:id.asc])
|
|
all_modules.each{|dyn_mod|
|
|
next if !dyn_mod.path.split('/')[1].match(/^metasploit/)
|
|
command_mod_name = dyn_mod["name"]
|
|
dyn_mod_category = "Metasploit"
|
|
command_module_status = set_command_module_status(command_mod_name)
|
|
command_module_icon_path = set_command_module_icon(command_module_status)
|
|
|
|
update_command_module_tree(tree, dyn_mod_category, command_module_icon_path, command_module_status, command_mod_name,dyn_mod.id)
|
|
}
|
|
end
|
|
|
|
# sort the parent array nodes
|
|
tree.sort! {|a,b| a['text'] <=> b['text']}
|
|
|
|
sort_recursive_tree(tree)
|
|
|
|
retitle_recursive_tree(tree)
|
|
|
|
|
|
|
|
# return a JSON array of hashes
|
|
@body = tree.to_json
|
|
end
|
|
|
|
# Returns the inputs definition of an command_module.
|
|
def select_command_module
|
|
command_module_id = @params['command_module_id'] || nil
|
|
(print_error "command_module_id is nil";return) if command_module_id.nil?
|
|
command_module = BeEF::Core::Models::CommandModule.get(command_module_id)
|
|
key = BeEF::Module.get_key_by_database_id(command_module_id)
|
|
|
|
payload_name = @params['payload_name'] || nil
|
|
if not payload_name.nil?
|
|
@body = dynamic_payload2json(command_module_id, payload_name)
|
|
else
|
|
@body = command_modules2json([key])
|
|
end
|
|
end
|
|
|
|
# Returns the list of commands for an command_module
|
|
def select_command_module_commands
|
|
commands = []
|
|
i=0
|
|
|
|
# get params
|
|
zombie_session = @params['zombie_session'] || nil
|
|
(print_error "Zombie session is nil";return) if zombie_session.nil?
|
|
command_module_id = @params['command_module_id'] || nil
|
|
(print_error "command_module id is nil";return) if command_module_id.nil?
|
|
# validate nonce
|
|
nonce = @params['nonce'] || nil
|
|
(print_error "nonce is nil";return) if nonce.nil?
|
|
(print_error "nonce incorrect";return) if @session.get_nonce != nonce
|
|
|
|
# get the browser id
|
|
zombie = Z.first(:session => zombie_session)
|
|
(print_error "Zombie is nil";return) if zombie.nil?
|
|
zombie_id = zombie.id
|
|
(print_error "Zombie id is nil";return) if zombie_id.nil?
|
|
|
|
C.all(:command_module_id => command_module_id, :hooked_browser_id => zombie_id).each do |command|
|
|
commands.push({
|
|
'id' => i,
|
|
'object_id' => command.id,
|
|
'creationdate' => Time.at(command.creationdate.to_i).strftime("%Y-%m-%d %H:%M").to_s,
|
|
'label' => command.label
|
|
})
|
|
i+=1
|
|
end
|
|
|
|
@body = {
|
|
'success' => 'true',
|
|
'commands' => commands}.to_json
|
|
|
|
end
|
|
|
|
# Attaches an command_module to a zombie.
|
|
def attach_command_module
|
|
|
|
definition = {}
|
|
|
|
# get params
|
|
zombie_session = @params['zombie_session'] || nil
|
|
(print_error "Zombie id is nil";return) if zombie_session.nil?
|
|
command_module_id = @params['command_module_id'] || nil
|
|
(print_error "command_module id is nil";return) if command_module_id.nil?
|
|
# validate nonce
|
|
nonce = @params['nonce'] || nil
|
|
(print_error "nonce is nil";return) if nonce.nil?
|
|
(print_error "nonce incorrect";return) if @session.get_nonce != nonce
|
|
|
|
@params.keys.each {|param|
|
|
(print_error "invalid key param string";return) if not BeEF::Filters.has_valid_param_chars?(param)
|
|
(print_error "first char is num";return) if BeEF::Filters.first_char_is_num?(param)
|
|
definition[param[4..-1]] = params[param]
|
|
oc = BeEF::Core::Models::OptionCache.first_or_create(:name => param[4..-1])
|
|
oc.value = params[param]
|
|
oc.save
|
|
}
|
|
|
|
mod_key = BeEF::Module.get_key_by_database_id(command_module_id)
|
|
# Hack to rework the old option system into the new option system
|
|
def2 = []
|
|
definition.each{|k,v|
|
|
def2.push({'name' => k, 'value' => v})
|
|
}
|
|
# End hack
|
|
exec_results = BeEF::Module.execute(mod_key, zombie_session, def2)
|
|
@body = (exec_results != nil) ? '{success: true}' : '{success: false}'
|
|
end
|
|
|
|
# Re-execute an command_module to a zombie.
|
|
def reexecute_command_module
|
|
|
|
# get params
|
|
command_id = @params['command_id'] || nil
|
|
(print_error "Command id is nil";return) if command_id.nil?
|
|
command = BeEF::Core::Models::Command.first(:id => command_id.to_i) || nil
|
|
(print_error "Command is nil";return) if command.nil?
|
|
# validate nonce
|
|
nonce = @params['nonce'] || nil
|
|
(print_error "nonce is nil";return) if nonce.nil?
|
|
(print_error "nonce incorrect";return) if @session.get_nonce != nonce
|
|
|
|
command.instructions_sent = false
|
|
command.save
|
|
|
|
@body = '{success : true}'
|
|
end
|
|
|
|
def attach_dynamic_command_module
|
|
|
|
definition = {}
|
|
|
|
# get params
|
|
zombie_session = @params['zombie_session'] || nil
|
|
(print_error "Zombie id is nil";return) if zombie_session.nil?
|
|
command_module_id = @params['command_module_id'] || nil
|
|
(print_error "command_module id is nil";return) if command_module_id.nil?
|
|
# validate nonce
|
|
nonce = @params['nonce'] || nil
|
|
(print_error "nonce is nil";return) if nonce.nil?
|
|
(print_error "nonce incorrect";return) if @session.get_nonce != nonce
|
|
|
|
@params.keys.each {|param|
|
|
(print_error "invalid key param string";return) if not BeEF::Filters.has_valid_param_chars?(param)
|
|
(print_error "first char is num";return) if BeEF::Filters.first_char_is_num?(param)
|
|
definition[param[4..-1]] = params[param]
|
|
oc = BeEF::Core::Models::OptionCache.first_or_create(:name => param[4..-1])
|
|
oc.value = params[param]
|
|
oc.save
|
|
}
|
|
|
|
zombie = Z.first(:session => zombie_session)
|
|
(print_error "Zombie is nil";return) if zombie.nil?
|
|
zombie_id = zombie.id
|
|
(print_error "Zombie id is nil";return) if zombie_id.nil?
|
|
command_module = BeEF::Core::Models::CommandModule.get(command_module_id)
|
|
|
|
if(command_module != nil && command_module.path.match(/^Dynamic/))
|
|
dyn_mod_name = command_module.path.split('/').last
|
|
e = BeEF::Modules::Commands.const_get(dyn_mod_name.capitalize).new
|
|
e.update_info(command_module_id)
|
|
e.update_data()
|
|
ret = e.launch_exploit(definition)
|
|
|
|
return {'success' => 'false'}.to_json if ret['result'] != 'success'
|
|
|
|
basedef = {}
|
|
basedef['sploit_url'] = ret['uri']
|
|
|
|
C.new( :data => basedef.to_json,
|
|
:hooked_browser_id => zombie_id,
|
|
:command_module_id => command_module_id,
|
|
:creationdate => Time.new.to_i
|
|
).save
|
|
|
|
@body = '{success : true}'
|
|
else
|
|
# return {'success' => 'false'}.to_json
|
|
{'success' => 'false'}.to_json
|
|
end
|
|
|
|
|
|
|
|
end
|
|
|
|
# Returns the results of a command
|
|
def select_command_results
|
|
results = []
|
|
|
|
# get params
|
|
command_id = @params['command_id']|| nil
|
|
(print_error "Command id is nil";return) if command_id.nil?
|
|
command = BeEF::Core::Models::Command.first(:id => command_id.to_i) || nil
|
|
(print_error "Command is nil";return) if command.nil?
|
|
|
|
# get command_module
|
|
command_module = BeEF::Core::Models::CommandModule.first(:id => command.command_module_id)
|
|
(print_error "command_module is nil";return) if command_module.nil?
|
|
|
|
resultsdb = BeEF::Core::Models::Result.all(:command_id => command_id)
|
|
(print_error "Command id result is nil";return) if resultsdb.nil?
|
|
|
|
resultsdb.each{ |result| results.push({'date' => result.date, 'data' => JSON.parse(result.data)}) }
|
|
|
|
@body = {
|
|
'success' => 'true',
|
|
'command_module_name' => command_module.name,
|
|
'command_module_id' => command_module.id,
|
|
'results' => results}.to_json
|
|
|
|
end
|
|
|
|
# Returns the definition of a command.
|
|
# In other words it returns the command that was used to command_module a zombie.
|
|
def select_command
|
|
|
|
# get params
|
|
command_id = @params['command_id'] || nil
|
|
(print_error "Command id is nil";return) if command_id.nil?
|
|
command = BeEF::Core::Models::Command.first(:id => command_id.to_i) || nil
|
|
(print_error "Command is nil";return) if command.nil?
|
|
|
|
command_module = BeEF::Core::Models::CommandModule.get(command.command_module_id)
|
|
(print_error "command_module is nil";return) if command_module.nil?
|
|
|
|
if(command_module.path.split('/').first.match(/^Dynamic/))
|
|
dyn_mod_name = command_module.path.split('/').last
|
|
e = BeEF::Modules::Commands.const_get(dyn_mod_name.capitalize).new
|
|
else
|
|
command_module_name = command_module.name
|
|
e = BeEF::Core::Command.const_get(command_module_name.capitalize).new(command_module_name)
|
|
end
|
|
|
|
@body = {
|
|
'success' => 'true',
|
|
'command_module_name' => command_module_name,
|
|
'command_module_id' => command_module.id,
|
|
'data' => BeEF::Module.get_options(command_module_name),
|
|
'definition' => JSON.parse(e.to_json)
|
|
}.to_json
|
|
|
|
end
|
|
|
|
private
|
|
|
|
# Takes a list of command_modules and returns them as a JSON array
|
|
def command_modules2json(command_modules)
|
|
command_modules_json = {}
|
|
i = 1
|
|
config = BeEF::Core::Configuration.instance
|
|
command_modules.each do |command_module|
|
|
h = {
|
|
'Name'=> config.get("beef.module.#{command_module}.name"),
|
|
'Description'=> config.get("beef.module.#{command_module}.description"),
|
|
'Category'=> config.get("beef.module.#{command_module}.category"),
|
|
'Data'=> BeEF::Module.get_options(command_module)
|
|
}
|
|
command_modules_json[i] = h
|
|
i += 1
|
|
end
|
|
|
|
if not command_modules_json.empty?
|
|
return {'success' => 'true', 'command_modules' => command_modules_json}.to_json
|
|
else
|
|
return {'success' => 'false'}.to_json
|
|
end
|
|
end
|
|
|
|
# return the input requred for the module in JSON format
|
|
def dynamic_modules2json(id)
|
|
command_modules_json = {}
|
|
|
|
mod = BeEF::Core::Models::CommandModule.first(:id => id)
|
|
|
|
# if the module id is not in the database return false
|
|
return {'success' => 'false'}.to_json if(not mod)
|
|
|
|
# the path will equal Dynamic/<type> and this will get just the type
|
|
dynamic_type = mod.path.split("/").last
|
|
|
|
e = BeEF::Modules::Commands.const_get(dynamic_type.capitalize).new
|
|
e.update_info(mod.id)
|
|
e.update_data()
|
|
command_modules_json[1] = JSON.parse(e.to_json)
|
|
if not command_modules_json.empty?
|
|
return {'success' => 'true', 'dynamic' => 'true', 'command_modules' => command_modules_json}.to_json
|
|
else
|
|
return {'success' => 'false'}.to_json
|
|
end
|
|
end
|
|
|
|
def dynamic_payload2json(id, payload_name)
|
|
command_modules_json = {}
|
|
|
|
command_module = BeEF::Core::Models::CommandModule.get(id)
|
|
(print_error "Module does not exists";return 'success' => 'false') if command_module.nil?
|
|
|
|
payload_options = BeEF::Module.get_payload_options(command_module.name,payload_name)
|
|
# get payload options in JSON
|
|
#e = BeEF::Modules::Commands.const_get(dynamic_type.capitalize).new
|
|
payload_options_json = []
|
|
payload_options_json[1] = payload_options
|
|
#payload_options_json[1] = e.get_payload_options(payload_name)
|
|
return {'success' => 'true', 'command_modules' => payload_options_json}.to_json
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
end
|
|
end
|
|
end
|