Module: Msf::DBManager::Import::MetasploitFramework::Zip
- Included in:
- Msf::DBManager::Import::MetasploitFramework
- Defined in:
- lib/msf/core/db_manager/import/metasploit_framework/zip.rb
Instance Method Summary collapse
-
#import_msf_collateral(args = {}, &block) ⇒ Object
Imports loot, tasks, and reports from an MSF ZIP report.
-
#import_msf_zip(args = {}, &block) ⇒ Object
Import a Metasploit Express ZIP file.
- #is_child_of?(target_dir, target) ⇒ Boolean
-
#parse_zip_host(host, wspace, bl, allow_yaml, btag, args, basedir, host_info, &block) ⇒ Object
Parses host Nokogiri::XML::Element.
-
#parse_zip_loot(loot, wspace, bl, allow_yaml, btag, args, basedir, host_info, &block) ⇒ Object
Parses loot Nokogiri::XML::Element.
-
#parse_zip_report(report, wspace, bl, allow_yaml, btag, args, basedir, host_info, &block) ⇒ Object
Parses report Nokogiri::XML::Element.
-
#parse_zip_task(task, wspace, bl, allow_yaml, btag, args, basedir, host_info, &block) ⇒ Object
Parses task Nokogiri::XML::Element.
Instance Method Details
#import_msf_collateral(args = {}, &block) ⇒ Object
Imports loot, tasks, and reports from an MSF ZIP report. XXX: This function is stupidly long. It needs to be refactored.
4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 |
# File 'lib/msf/core/db_manager/import/metasploit_framework/zip.rb', line 4 def import_msf_collateral(args={}, &block) data = ::File.open(args[:filename], "rb") {|f| f.read(f.stat.size)} wspace = Msf::Util::DBManager.process_opts_workspace(args, framework) args = args.clone() args.delete(:workspace) bl = validate_ips(args[:blacklist]) ? args[:blacklist].split : [] basedir = args[:basedir] || args['basedir'] || ::File.join(Msf::Config.data_directory, "msf") allow_yaml = false btag = nil doc = Nokogiri::XML::Reader.from_memory(data) case doc.first.name when "MetasploitExpressV1" m_ver = 1 allow_yaml = true btag = "MetasploitExpressV1" when "MetasploitExpressV2" m_ver = 2 allow_yaml = true btag = "MetasploitExpressV2" when "MetasploitExpressV3" m_ver = 3 btag = "MetasploitExpressV3" when "MetasploitExpressV4" m_ver = 4 btag = "MetasploitExpressV4" when "MetasploitV4" m_ver = 4 btag = "MetasploitV4" else m_ver = nil end unless m_ver and btag raise Msf::DBImportError.new("Unsupported Metasploit XML document format") end host_info = {} doc.each do |node| if ['host', 'loot', 'task', 'report'].include?(node.name) unless node.inner_xml.empty? send("parse_zip_#{node.name}", Nokogiri::XML(node.outer_xml).at("./#{node.name}"), wspace, bl, allow_yaml, btag, args, basedir, host_info, &block) end end end end |
#import_msf_zip(args = {}, &block) ⇒ Object
Import a Metasploit Express ZIP file. Note that this requires a fair bit of filesystem manipulation, and is very much tied up with the Metasploit Express ZIP file format export (for obvious reasons). In the event directories exist, they will be reused. If target files exist, they will be overwritten.
XXX: Refactor so it’s not quite as sanity-blasting.
166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 |
# File 'lib/msf/core/db_manager/import/metasploit_framework/zip.rb', line 166 def import_msf_zip(args={}, &block) data = args[:data] wspace = Msf::Util::DBManager.process_opts_workspace(args, framework) bl = validate_ips(args[:blacklist]) ? args[:blacklist].split : [] new_tmp = ::File.join(Dir::tmpdir,"msf","imp_#{Rex::Text::rand_text_alphanumeric(4)}",@import_filedata[:zip_basename]) if ::File.exist? new_tmp unless (::File.directory?(new_tmp) && ::File.writable?(new_tmp)) raise Msf::DBImportError.new("Could not extract zip file to #{new_tmp}") end else FileUtils.mkdir_p(new_tmp) end @import_filedata[:zip_tmp] = new_tmp # Grab the list of unique basedirs over all entries. @import_filedata[:zip_tmp_subdirs] = @import_filedata[:zip_entry_names].map {|x| ::File.split(x)}.map {|x| x[0]}.uniq.reject {|x| x == "."} # mkdir all of the base directories we just pulled out, if they don't # already exist @import_filedata[:zip_tmp_subdirs].each {|sub| tmp_subdirs = ::File.join(@import_filedata[:zip_tmp],sub) if File.exist? tmp_subdirs unless (::File.directory?(tmp_subdirs) && File.writable?(tmp_subdirs)) # if it exists but we can't write to it, give up raise Msf::DBImportError.new("Could not extract zip file to #{tmp_subdirs}") end else ::FileUtils.mkdir(tmp_subdirs) end } data.entries.each do |e| # normalize entry name to an absolute path target = File.(File.join(@import_filedata[:zip_tmp], e.name), '/').to_s # skip if the target would be extracted outside of the zip # tmp dir to mitigate any directory traversal attacks next unless is_child_of?(@import_filedata[:zip_tmp], target) e.extract(target) if target =~ /\.xml\z/ target_data = ::File.open(target, "rb") {|f| f.read 1024} if import_filetype_detect(target_data) == :msf_xml @import_filedata[:zip_extracted_xml] = target end end end # Import any creds if there are some in the import file Dir.entries(@import_filedata[:zip_tmp]).each do |entry| if entry =~ /^.*#{Regexp.quote(Metasploit::Credential::Exporter::Core::CREDS_DUMP_FILE_IDENTIFIER)}.*/ manifest_file_path = File.join(@import_filedata[:zip_tmp], entry, Metasploit::Credential::Importer::Zip::MANIFEST_FILE_NAME) if File.exist? manifest_file_path import_msf_cred_dump(manifest_file_path, wspace) end end end # This will kick the newly-extracted XML file through # the import_file process all over again. if @import_filedata[:zip_extracted_xml] new_args = args.dup new_args[:filename] = @import_filedata[:zip_extracted_xml] new_args[:data] = nil new_args[:ifd] = @import_filedata.dup if block import_file(new_args, &block) else import_file(new_args) end end # Kick down to all the MSFX ZIP specific items if block import_msf_collateral(new_args, &block) else import_msf_collateral(new_args) end end |
#is_child_of?(target_dir, target) ⇒ Boolean
248 249 250 |
# File 'lib/msf/core/db_manager/import/metasploit_framework/zip.rb', line 248 def is_child_of?(target_dir, target) target.downcase.start_with?(target_dir.downcase) end |
#parse_zip_host(host, wspace, bl, allow_yaml, btag, args, basedir, host_info, &block) ⇒ Object
Parses host Nokogiri::XML::Element
53 54 55 |
# File 'lib/msf/core/db_manager/import/metasploit_framework/zip.rb', line 53 def parse_zip_host(host, wspace, bl, allow_yaml, btag, args, basedir, host_info, &block) host_info[host.at("id").text.to_s.strip] = nils_for_nulls(host.at("address").text.to_s.strip) unless host.at('address').nil? end |
#parse_zip_loot(loot, wspace, bl, allow_yaml, btag, args, basedir, host_info, &block) ⇒ Object
Parses loot Nokogiri::XML::Element
58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 |
# File 'lib/msf/core/db_manager/import/metasploit_framework/zip.rb', line 58 def parse_zip_loot(loot, wspace, bl, allow_yaml, btag, args, basedir, host_info, &block) return 0 if bl.include? host_info[loot.at("host-id").text.to_s.strip] loot_info = {} loot_info[:host] = host_info[loot.at("host-id").text.to_s.strip] loot_info[:workspace] = wspace loot_info[:ctype] = nils_for_nulls(loot.at("content-type").text.to_s.strip) loot_info[:info] = nils_for_nulls(unserialize_object(loot.at("info"), allow_yaml)) loot_info[:ltype] = nils_for_nulls(loot.at("ltype").text.to_s.strip) loot_info[:name] = nils_for_nulls(loot.at("name").text.to_s.strip) loot_info[:created_at] = nils_for_nulls(loot.at("created-at").text.to_s.strip) loot_info[:updated_at] = nils_for_nulls(loot.at("updated-at").text.to_s.strip) loot_info[:name] = nils_for_nulls(loot.at("name").text.to_s.strip) loot_info[:orig_path] = nils_for_nulls(loot.at("path").text.to_s.strip) loot_info[:task] = args[:task] tmp = args[:ifd][:zip_tmp] loot_info[:orig_path].gsub!(/^\./,tmp) if loot_info[:orig_path] if !loot.at("service-id").text.to_s.strip.empty? unless loot.at("service-id").text.to_s.strip == "NULL" loot_info[:service] = loot.at("service-id").text.to_s.strip end end # Only report loot if we actually have it. # TODO: Copypasta. Separate this out. if ::File.exist? loot_info[:orig_path] loot_dir = ::File.join(basedir,"loot") loot_file = ::File.split(loot_info[:orig_path]).last if ::File.exist? loot_dir unless (::File.directory?(loot_dir) && ::File.writable?(loot_dir)) raise Msf::DBImportError.new("Could not move files to #{loot_dir}") end else ::FileUtils.mkdir_p(loot_dir) end new_loot = ::File.join(loot_dir,loot_file) loot_info[:path] = new_loot if ::File.exist?(new_loot) ::File.unlink new_loot # Delete it, and don't report it. else report_loot(loot_info) # It's new, so report it. end ::FileUtils.copy(loot_info[:orig_path], new_loot) yield(:msf_loot, new_loot) if block end end |
#parse_zip_report(report, wspace, bl, allow_yaml, btag, args, basedir, host_info, &block) ⇒ Object
Parses report Nokogiri::XML::Element
155 156 157 |
# File 'lib/msf/core/db_manager/import/metasploit_framework/zip.rb', line 155 def parse_zip_report(report, wspace, bl, allow_yaml, btag, args, basedir, host_info, &block) import_report(report, args, basedir) end |
#parse_zip_task(task, wspace, bl, allow_yaml, btag, args, basedir, host_info, &block) ⇒ Object
Parses task Nokogiri::XML::Element
105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 |
# File 'lib/msf/core/db_manager/import/metasploit_framework/zip.rb', line 105 def parse_zip_task(task, wspace, bl, allow_yaml, btag, args, basedir, host_info, &block) task_info = {} task_info[:workspace] = wspace # Should user be imported (original) or declared (the importing user)? task_info[:user] = nils_for_nulls(task.at("created-by").text.to_s.strip) task_info[:desc] = nils_for_nulls(task.at("description").text.to_s.strip) task_info[:info] = nils_for_nulls(unserialize_object(task.at("info"), allow_yaml)) task_info[:mod] = nils_for_nulls(task.at("module").text.to_s.strip) task_info[:options] = nils_for_nulls(task.at("options").text.to_s.strip) task_info[:prog] = nils_for_nulls(task.at("progress").text.to_s.strip).to_i task_info[:created_at] = nils_for_nulls(task.at("created-at").text.to_s.strip) task_info[:updated_at] = nils_for_nulls(task.at("updated-at").text.to_s.strip) if !task.at("completed-at").text.to_s.empty? task_info[:completed_at] = nils_for_nulls(task.at("completed-at").text.to_s.strip) end if !task.at("error").text.to_s.empty? task_info[:error] = nils_for_nulls(task.at("error").text.to_s.strip) end if !task.at("result").text.to_s.empty? task_info[:result] = nils_for_nulls(task.at("result").text.to_s.strip) end task_info[:orig_path] = nils_for_nulls(task.at("path").text.to_s.strip) tmp = args[:ifd][:zip_tmp] task_info[:orig_path].gsub!(/^\./,tmp) if task_info[:orig_path] # Only report a task if we actually have it. # TODO: Copypasta. Separate this out. if ::File.exist? task_info[:orig_path] tasks_dir = ::File.join(basedir,"tasks") task_file = ::File.split(task_info[:orig_path]).last if ::File.exist? tasks_dir unless (::File.directory?(tasks_dir) && ::File.writable?(tasks_dir)) raise Msf::DBImportError.new("Could not move files to #{tasks_dir}") end else ::FileUtils.mkdir_p(tasks_dir) end new_task = ::File.join(tasks_dir,task_file) task_info[:path] = new_task if ::File.exist?(new_task) ::File.unlink new_task # Delete it, and don't report it. else msf_import_task(task_info) # It's new, so report it. end ::FileUtils.copy(task_info[:orig_path], new_task) yield(:msf_task, new_task) if block end end |