rapid7/metasploit-framework

View on GitHub
lib/msf/ui/console/command_dispatcher/payload.rb

Summary

Maintainability
C
1 day
Test Coverage
# -*- coding: binary -*-

module Msf
  module Ui
    module Console
      module CommandDispatcher
        ###
        # Payload module command dispatcher.
        ###
        class Payload
          include Msf::Ui::Console::ModuleCommandDispatcher
          include Msf::Ui::Console::ModuleOptionTabCompletion

          # Load supported formats
          @@supported_formats = \
            Msf::Simple::Buffer.transform_formats + \
            Msf::Util::EXE.to_executable_fmt_formats

          @@to_handler_opts = Rex::Parser::Arguments.new(
            '-h' => [ false, 'Show this message' ]
          )

          @@generate_opts = Rex::Parser::Arguments.new(
            '-p' => [ true, 'The platform of the payload' ],
            '-n' => [ true, 'Prepend a nopsled of [length] size on to the payload' ],
            '-f' => [ true, "Output format: #{@@supported_formats.join(',')}" ],
            '-E' => [ false, 'Force encoding' ],
            '-e' => [ true, 'The encoder to use' ],
            '-P' => [ true, 'Total desired payload size, auto-produce appropriate NOP sled length'],
            '-S' => [ true, 'The new section name to use when generating (large) Windows binaries'],
            '-b' => [ true, "The list of characters to avoid example: '\\x00\\xff'" ],
            '-i' => [ true, 'The number of times to encode the payload' ],
            '-x' => [ true, 'Specify a custom executable file to use as a template' ],
            '-k' => [ false, 'Preserve the template behavior and inject the payload as a new thread' ],
            '-o' => [ true, 'The output file name (otherwise stdout)' ],
            '-O' => [ true, "Deprecated: alias for the '-o' option" ],
            '-v' => [ false, 'Verbose output (display stage in addition to stager)' ],
            '-h' => [ false, 'Show this message' ]
          )

          #
          # Returns the hash of commands specific to payload modules.
          #
          def commands
            super.update(
              'generate' => 'Generates a payload',
              'to_handler' => 'Creates a handler with the specified payload',
              'exploit' => 'Creates a handler with the specified payload'
            )
          end

          def cmd_to_handler_help
            print_line 'Usage: to_handler [options]'
            print_line
            print_line 'Creates a handler a payload. Datastore options may be supplied after normal options.'
            print_line 'This is convenient way of using multi/handler, setting the payload, and then setting datastore options.'
            print_line
            print_line 'Example: to_handler'
            print_line 'Example: to_handler LHOST=192.168.123.1'
            print @@to_handler_opts.usage
          end

          def cmd_to_handler(*args)
            if args.include?('-r') || args.include?('--reload-libs')
              driver.run_single('reload_lib -a')
            end

            mod_with_opts = mod.replicant
            handler = framework.modules.create('exploit/multi/handler')
            handler.share_datastore(mod_with_opts.datastore)

            @@to_handler_opts.parse(args) do |opt, _idx, val|
              case opt
              when '-h'
                cmd_to_handler_help
                return false
              else
                unless val.include?('=')
                  cmd_to_handler_help
                  return false
                end

                handler.datastore.import_options_from_s(val)
              end
            end

            handler_opts = {
              'Payload' => mod.refname,
              'LocalInput' => driver.input,
              'LocalOutput' => driver.output,
              'RunAsJob' => true,
              'Options' => {
                'ExitOnSession' => false
              }
            }

            replicant_handler = nil
            handler.exploit_simple(handler_opts) do |yielded_replicant_handler|
              replicant_handler = yielded_replicant_handler
            end

            if replicant_handler.nil?
              print_error('Failed to run module')
              return
            end

            if replicant_handler.error.nil?
              job_id = handler.job_id

              print_status "Payload Handler Started as Job #{job_id}"
            end
          end

          alias cmd_exploit cmd_to_handler

          #
          # Tab completion for the generate command
          #
          def cmd_to_handler_tabs(str, words)
            fmt = {
              '-h' => [ nil ],
            }
            flags = tab_complete_generic(fmt, str, words)
            options = tab_complete_option(active_module, str, words)
            flags + options
          end

          #
          # Returns the command dispatcher name.
          #
          def name
            'Payload'
          end

          def cmd_generate_help
            print_line 'Usage: generate [options]'
            print_line
            print_line 'Generates a payload. Datastore options may be supplied after normal options.'
            print_line
            print_line 'Example: generate -f python LHOST=127.0.0.1'
            print @@generate_opts.usage
          end

          #
          # Generates a payload.
          #
          def cmd_generate(*args)
            # Parse the arguments
            encoder_name = nil
            sled_size = nil
            pad_nops = nil
            sec_name = nil
            option_str = nil
            badchars = nil
            format = 'ruby'
            ofile = nil
            iter = 1
            force = nil
            template = nil
            plat = nil
            keep = false
            verbose = false
            mod_with_opts = mod.replicant

            @@generate_opts.parse(args) do |opt, _idx, val|
              case opt
              when '-b'
                badchars = Rex::Text.dehex(val)
              when '-e'
                encoder_name = val
              when '-E'
                force = true
              when '-n'
                sled_size = val.to_i
              when '-P'
                pad_nops = val.to_i
              when '-S'
                sec_name = val
              when '-f'
                format = val
              when '-o'
                if val.include?('=')
                  print_error("The -o parameter of 'generate' is now preferred to indicate the output file, like with msfvenom\n")
                  option_str = val
                else
                  ofile = val
                end
              when '-O'
                print("Usage of the '-O' parameter is deprecated, prefer '-o' to indicate the output file")
                ofile = val
              when '-i'
                iter = val
              when '-k'
                keep = true
              when '-p'
                plat = val
              when '-x'
                template = val
              when '-v'
                verbose = true
              when '-h'
                cmd_generate_help
                return false
              else
                unless val.include?('=')
                  cmd_generate_help
                  return false
                end

                mod_with_opts.datastore.import_options_from_s(val)
              end
            end
            if encoder_name.nil? && mod_with_opts.datastore['ENCODER']
              encoder_name = mod_with_opts.datastore['ENCODER']
            end

            # Generate the payload
            begin
              buf = mod_with_opts.generate_simple(
                'BadChars' => badchars,
                'Encoder' => encoder_name,
                'Format' => format,
                'NopSledSize' => sled_size,
                'PadNops' => pad_nops,
                'SecName' => sec_name,
                'OptionStr' => option_str,
                'ForceEncode' => force,
                'Template' => template,
                'Platform' => plat,
                'KeepTemplateWorking' => keep,
                'Iterations' => iter,
                'Verbose' => verbose
              )
            rescue StandardError
              log_error("Payload generation failed: #{$ERROR_INFO}")
              return false
            end

            if !ofile
              # Display generated payload
              puts(buf)
            else
              print_status("Writing #{buf.length} bytes to #{ofile}...")
              f = File.expand_path(ofile)
              fd = File.open(f, 'wb')
              fd.write(buf)
              fd.close
            end
            true
          end

          #
          # Tab completion for the generate command
          #
          def cmd_generate_tabs(str, words)
            fmt = {
              '-b' => [ true ],
              '-E' => [ nil ],
              '-e' => [ framework.encoders.module_refnames ],
              '-h' => [ nil ],
              '-o' => [ :file ],
              '-P' => [ true ],
              '-S' => [ true ],
              '-f' => [ @@supported_formats ],
              '-p' => [ true ],
              '-k' => [ nil ],
              '-x' => [ :file ],
              '-i' => [ true ],
              '-v' => [ nil ]
            }
            flags = tab_complete_generic(fmt, str, words)
            options = tab_complete_option(active_module, str, words)
            flags + options
          end
        end
      end
    end
  end
end