# frozen_string_literal: true
#
# ronin-exploits - A Ruby library for ronin-rb that provides exploitation and
# payload crafting functionality.
#
# Copyright (c) 2007-2023 Hal Brodigan (postmodern.mod3 at gmail.com)
#
# ronin-exploits is free software: you can redistribute it and/or modify
# it under the terms of the GNU Lesser General Public License as published
# by the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# ronin-exploits is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public License
# along with ronin-exploits. If not, see .
#
require 'ronin/support/network/http'
require 'ronin/exploits/params/base_url'
module Ronin
module Exploits
module Mixins
#
# Adds HTTP helper methods for communicating with a web server.
#
# @api public
#
# @since 1.0.0
#
module HTTP
# Possible values for the `user_agent` param.
#
# @api private
HTTP_USER_AGENT_ALIASES = [
:random,
:chrome,
:firefox,
:safari,
:linux,
:macos,
:windows,
:iphone,
:ipad,
:android
] + Support::Network::HTTP::UserAgents::ALIASES.keys
#
# Adds the required `base_url` params to the exploit class.
#
# @param [Class] exploit
# The exploit class that included {Mixins::HTTP}.
#
# @api private
#
def self.included(exploit)
exploit.include Params::BaseURL
exploit.param :http_proxy, desc: 'The HTTP proxy to use'
exploit.param :http_user, desc: 'The HTTP Basic-Auth user'
exploit.param :http_password, desc: 'The HTTP Basic-Auth password'
exploit.param :user_agent, Core::Params::Types::Enum.new(HTTP_USER_AGENT_ALIASES), desc: 'The HTTP User-Agent to select'
exploit.param :raw_user_agent, desc: 'The raw HTTP User-Agent string to use'
exploit.param :http_cookie, desc: 'The raw HTTP Cookie to use'
end
#
# The HTTP proxy to use for all HTTP requests.
#
# @return [String, nil]
#
# @api private
#
def http_proxy
params[:http_proxy]
end
#
# The HTTP `Basic-Auth` user to use for all HTTP requests.
#
# @return [String, nil]
#
# @api private
#
def http_user
params[:http_user]
end
#
# The HTTP `Basic-Auth` password to use for all HTTP requests.
#
# @return [String, nil]
#
# @api private
#
def http_password
params[:http_password]
end
#
# Additional default headers to send with every HTTP request.
#
# @return [Hash{Symbol,String => String,Array}]
#
# @api private
#
def http_headers
{}
end
#
# The `User-Agent` to use for HTTP requests.
#
# @return [String, Symbol, :random, nil]
# The `User-Agent` string or an alias name (ex: `:chrome_linux`).
#
# @api private
#
def http_user_agent
params[:raw_user_agent] || params[:user_agent]
end
#
# The `Cookie` to use for HTTP requests.
#
# @return [String, nil]
#
# @api private
#
def http_cookie
params[:http_cookie]
end
#
# @!macro request_kwargs
# @option kwargs [String, nil] :user (http_user)
# The `Basic-Auth` user to authenticate as.
#
# @option kwargs [String, nil] :password (http_password)
# The `Basic-Auth` password to authenticate with.
#
# @option kwargs [String, nil] :query
# The query-string to append to the request path.
#
# @option kwargs [Hash, nil] :query_params
# The query-params to append to the request path.
#
# @option kwargs [String, nil] :body
# The body of the request.
#
# @option kwargs [Hash, String, nil] :form_data
# The form data that may be sent in the body of the request.
#
# @option kwargs [String, nil] :user (http_user)
# The user to authenticate as.
#
# @option kwargs [String, nil] :password (http_password)
# The password to authenticate with.
#
# @option kwargs [Hash{Symbol,String => String}, nil] :headers
# Additional HTTP headers to use for the request.
#
#
# A persistent HTTP connection to the target host and port.
#
# @return [Ronin::Support::Network::HTTP]
# The HTTP connection.
#
# @note
# This method will lazily initialize the HTTP connection on the first
# time this method is called and memoize the HTTP object.
#
def http
@http ||= Support::Network::HTTP.connect_uri(
params[:base_url], proxy: http_proxy,
headers: http_headers,
user_agent: http_user_agent,
cookie: http_cookie,
user: http_user,
password: http_password
)
end
#
# Performs and arbitrary HTTP request.
#
# @param [Symbol, String] method
# The HTTP method to use for the request.
#
# @param [String] path
# The path to create the HTTP request for.
#
# @!macro request_kwargs
#
# @yield [response]
# If a block is given it will be passed the received HTTP response.
#
# @yieldparam [Net::HTTPRresponse] response
# The received HTTP response object.
#
# @return [Net::HTTPResponse]
# The new HTTP Request object.
#
# @raise [ArgumentError]
# The `:method` option did not match a known `Net::HTTP` request
# class.
#
def http_request(method,path,**kwargs,&block)
if debug?
print_debug "Sending #{method.upcase} request to #{url_for(path)} ..."
end
http.request(method,path,**kwargs,&block)
end
#
# Sends an arbitrary HTTP request and returns the response status.
#
# @param [Symbol, String] method
# The HTTP method to use for the request.
#
# @param [String] path
# The path to create the HTTP request for.
#
# @!macro request_kwargs
#
# @return [Integer]
# The status code of the response.
#
def http_response_status(method=:head,path,**kwargs)
if debug?
print_debug "Checking response status for #{method.upcase} #{url_for(path)} ..."
end
http.response_status(method,path,**kwargs)
end
#
# Sends a HTTP request and determines if the response status was 200.
#
# @param [Symbol, String] method
# The HTTP method to use for the request.
#
# @param [String] path
# The path to create the HTTP request for.
#
# @!macro request_kwargs
#
# @return [Boolean]
# Indicates that the response status was 200.
#
def http_ok?(method=:head,path,**kwargs)
if debug?
print_debug "Checking if response status for #{method.upcase} #{url_for(path)} is 200 ..."
end
http.ok?(method,path,**kwargs)
end
#
# Sends an arbitrary HTTP request and returns the response headers.
#
# @param [Symbol, String] method
# The HTTP method to use for the request.
#
# @param [String] path
# The path to create the HTTP request for.
#
# @!macro request_kwargs
#
# @return [Hash{String => String}]
# The response headers.
#
def http_response_headers(method=:head,path,**kwargs)
if debug?
print_debug "Requesting response headers for #{method.upcase} #{url_for(path)} ..."
end
http.response_headers(method,path,**kwargs)
end
#
# Sends an HTTP request and returns the `Server` header.
#
# @param [String] path
# The path to create the HTTP request for.
#
# @param [:copy, :delete, :get, :head, :lock, :mkcol, :move,
# :options, :patch, :post, :propfind, :proppatch, :put,
# :trace, :unlock] method
# The HTTP method to use for the request.
#
# @!macro request_kwargs
#
# @return [String, nil]
# The `Server` header.
#
def http_server_header(path, method: :head, **kwargs)
if debug?
print_debug "Requesting the 'Server' header for #{method.upcase} #{url_for(path)} ..."
end
http.server_header(path, method: method, **kwargs)
end
#
# Sends an HTTP request and returns the `X-Powered-By` header.
#
# @param [String] path
# The path to create the HTTP request for.
#
# @param [:copy, :delete, :get, :head, :lock, :mkcol, :move,
# :options, :patch, :post, :propfind, :proppatch, :put,
# :trace, :unlock] method
# The HTTP method to use for the request.
#
# @!macro request_kwargs
#
# @return [String, nil]
# The `X-Powered-By` header.
#
def http_powered_by_header(path, method: :head, **kwargs)
if debug?
print_debug "Requesting the 'X-Powered-By' header for #{method.upcase} #{url_for(path)} ..."
end
http.powered_by_header(path, method: method, **kwargs)
end
#
# Sends an arbitrary HTTP request and returns the response body.
#
# @param [Symbol, String] method
# The HTTP method to use for the request.
#
# @param [String] path
# The path to create the HTTP request for.
#
# @!macro request_kwargs
#
# @return [String]
# The response body.
#
def http_response_body(method=:get, path,**kwargs)
if debug?
print_debug "Requesting response body for #{method.upcase} #{url_for(path)} ..."
end
http.response_body(method, path,**kwargs)
end
#
# Performs a `COPY` request for the given URI.
#
# @param [String] path
# The path to create the HTTP request for.
#
# @!macro request_kwargs
#
# @yield [response]
# If a block is given it will be passed the received HTTP response.
#
# @yieldparam [Net::HTTPRresponse] response
# The received HTTP response object.
#
# @return [Net::HTTPResponse]
# The new HTTP Request object.
#
def http_copy(path,**kwargs)
if debug?
print_debug "Requesting COPY #{url_for(path)} ..."
end
http.copy(path,**kwargs)
end
#
# Performs a `DELETE` request for the given URI.
#
# @param [String] path
# The path to create the HTTP request for.
#
# @!macro request_kwargs
#
# @yield [response]
# If a block is given it will be passed the received HTTP response.
#
# @yieldparam [Net::HTTPRresponse] response
# The received HTTP response object.
#
# @return [Net::HTTPResponse]
# The new HTTP Request object.
#
def http_delete(path,**kwargs,&block)
if debug?
print_debug "Requesting DELETE #{url_for(path)} ..."
end
http.delete(path,**kwargs)
end
#
# Performs a `GET` request for the given URI.
#
# @param [String] path
# The path to create the HTTP request for.
#
# @!macro request_kwargs
#
# @yield [response]
# If a block is given it will be passed the received HTTP response.
#
# @yieldparam [Net::HTTPRresponse] response
# The received HTTP response object.
#
# @return [Net::HTTPResponse]
# The new HTTP Request object.
#
def http_get(path,**kwargs,&block)
if debug?
print_debug "Requesting GET #{url_for(path)} ..."
end
http.get(path,**kwargs)
end
#
# Performs a `GET` request for the given URI and returns the response
# headers.
#
# @param [String] path
# The path to create the HTTP request for.
#
# @!macro request_kwargs
#
# @return [Hash{String => String}]
# The response headers.
#
def http_get_headers(path,**kwargs)
if debug?
print_debug "Requesting headers for GET #{url_for(path)} ..."
end
http.get_headers(path,**kwargs)
end
#
# Sends an HTTP request and returns the parsed `Set-Cookie`
# header(s).
#
# @param [String] path
# The path to create the HTTP request for.
#
# @!macro request_kwargs
#
# @return [Array, nil]
# The parsed `SetCookie` header(s).
#
def http_get_cookies(path,**kwargs)
if debug?
print_debug "Getting cookies for #{url_for(path)} ..."
end
http.get_cookies(path,**kwargs)
end
#
# Performs a `GET` request for the given URI and returns the response
# body.
#
# @param [String] path
# The path to create the HTTP request for.
#
# @!macro request_kwargs
#
# @return [String]
# The response body.
#
def http_get_body(path,**kwargs)
if debug?
print_debug "Requesting body for GET #{url_for(path)} ..."
end
http.get_body(path,**kwargs)
end
#
# Performs a `HEAD` request for the given URI.
#
# @param [String] path
# The path to create the HTTP request for.
#
# @!macro request_kwargs
#
# @yield [response]
# If a block is given it will be passed the received HTTP response.
#
# @yieldparam [Net::HTTPRresponse] response
# The received HTTP response object.
#
# @return [Net::HTTPResponse]
# The new HTTP Request object.
#
def http_head(path,**kwargs,&block)
if debug?
print_debug "Requesting HEAD #{url_for(path)} ..."
end
http.head(path,**kwargs,&block)
end
#
# Performs a `LOCK` request for the given URI.
#
# @param [String] path
# The path to create the HTTP request for.
#
# @!macro request_kwargs
#
# @yield [response]
# If a block is given it will be passed the received HTTP response.
#
# @yieldparam [Net::HTTPRresponse] response
# The received HTTP response object.
#
# @return [Net::HTTPResponse]
# The new HTTP Request object.
#
def http_lock(path,**kwargs,&block)
if debug?
print_debug "Requesting LOCK #{url_for(path)} ..."
end
http.lock(path,**kwargs,&block)
end
#
# Performs a `MKCOL` request for the given URI.
#
# @param [String] path
# The path to create the HTTP request for.
#
# @!macro request_kwargs
#
# @yield [response]
# If a block is given it will be passed the received HTTP response.
#
# @yieldparam [Net::HTTPRresponse] response
# The received HTTP response object.
#
# @return [Net::HTTPResponse]
# The new HTTP Request object.
#
def http_mkcol(path,**kwargs,&block)
if debug?
print_debug "Requesting MKCOL #{url_for(path)} ..."
end
http.mkcol(path,**kwargs,&block)
end
#
# Performs a `MOVE` request for the given URI.
#
# @param [String] path
# The path to create the HTTP request for.
#
# @!macro request_kwargs
#
# @yield [response]
# If a block is given it will be passed the received HTTP response.
#
# @yieldparam [Net::HTTPRresponse] response
# The received HTTP response object.
#
# @return [Net::HTTPResponse]
# The new HTTP Request object.
#
def http_move(path,**kwargs,&block)
if debug?
print_debug "Requesting MOVE #{url_for(path)} ..."
end
http.move(path,**kwargs,&block)
end
#
# Performs a `OPTIONS` request for the given URI.
#
# @param [String] path
# The path to create the HTTP request for.
#
# @!macro request_kwargs
#
# @yield [response]
# If a block is given it will be passed the received HTTP response.
#
# @yieldparam [Net::HTTPRresponse] response
# The received HTTP response object.
#
# @return [Net::HTTPResponse]
# The new HTTP Request object.
#
def http_options(path,**kwargs,&block)
if debug?
print_debug "Requesting OPTIONS #{url_for(path)} ..."
end
http.options(path,**kwargs,&block)
end
#
# Performs a `OPTIONS` HTTP request for the given URI and parses the
# `Allow` response header.
#
# @param [String] path
# The path to create the HTTP request for.
#
# @!macro request_kwargs
#
# @yield [response]
# If a block is given it will be passed the received HTTP response.
#
# @yieldparam [Net::HTTPRresponse] response
# The received HTTP response object.
#
# @return [Array]
# The allowed HTTP request methods for the given URL.
#
def http_allowed_methods(path,**kwargs,&block)
if debug?
print_debug "Checking allowed HTTP methods for #{url_for(path)} ..."
end
http.allowed_methods(path,**kwargs)
end
#
# Performs a `PATCH` request for the given URI.
#
# @param [String] path
# The path to create the HTTP request for.
#
# @!macro request_kwargs
#
# @yield [response]
# If a block is given it will be passed the received HTTP response.
#
# @yieldparam [Net::HTTPRresponse] response
# The received HTTP response object.
#
# @return [Net::HTTPResponse]
# The new HTTP Request object.
#
# @see Network::HTTP.patch
#
def http_patch(path,**kwargs,&block)
if debug?
print_debug "Requesting PATCH #{url_for(path)} ..."
end
http.patch(path,**kwargs,&block)
end
#
# Performs a `POST` request for the given URI.
#
# @param [String] path
# The path to create the HTTP request for.
#
# @!macro request_kwargs
#
# @yield [response]
# If a block is given it will be passed the received HTTP response.
#
# @yieldparam [Net::HTTPRresponse] response
# The received HTTP response object.
#
# @return [Net::HTTPResponse]
# The new HTTP Request object.
#
def http_post(path,**kwargs,&block)
if debug?
print_debug "Requesting POST #{url_for(path)} ..."
end
http.post(path,**kwargs,&block)
end
#
# Performs a `POST` request on the given URI and returns the response
# headers.
#
# @param [String] path
# The path to create the HTTP request for.
#
# @!macro request_kwargs
#
# @return [Hash{String => String}]
# The response headers.
#
def http_post_headers(path,**kwargs)
if debug?
print_debug "Requesting response headers for POST #{url_for(path)} ..."
end
http.post_headers(path,**kwargs)
end
#
# Performs a `POST` request for the given URI and returns the
# response body.
#
# @param [String] path
# The path to create the HTTP request for.
#
# @!macro request_kwargs
#
# @return [String]
# The response body.
#
def http_post_body(path,**kwargs)
if debug?
print_debug "Requesting response body for POST #{url_for(path)} ..."
end
http.post_body(path,**kwargs)
end
#
# Performs a `PROPFIND` request for the given URI.
#
# @param [String] path
# The path to create the HTTP request for.
#
# @!macro request_kwargs
#
# @yield [response]
# If a block is given it will be passed the received HTTP response.
#
# @yieldparam [Net::HTTPRresponse] response
# The received HTTP response object.
#
# @return [Net::HTTPResponse]
# The new HTTP Request object.
#
def http_propfind(path,**kwargs,&block)
if debug?
print_debug "Requesting PROPFIND #{url_for(path)} ..."
end
http.propfind(path,**kwargs,&block)
end
alias http_prop_find http_propfind
#
# Performs a `PROPPATCH` request for the given URI.
#
# @param [String] path
# The path to create the HTTP request for.
#
# @!macro request_kwargs
#
# @yield [response]
# If a block is given it will be passed the received HTTP response.
#
# @yieldparam [Net::HTTPRresponse] response
# The received HTTP response object.
#
# @return [Net::HTTPResponse]
# The new HTTP Request object.
#
def http_proppatch(path,**kwargs,&block)
if debug?
print_debug "Requesting PROPPATCH #{url_for(path)} ..."
end
http.proppatch(path,**kwargs,&block)
end
alias http_prop_patch http_proppatch
#
# Performs a `PUT` request for the given URI.
#
# @param [String] path
# The path to create the HTTP request for.
#
# @!macro request_kwargs
#
# @yield [response]
# If a block is given it will be passed the received HTTP response.
#
# @yieldparam [Net::HTTPRresponse] response
# The received HTTP response object.
#
# @return [Net::HTTPResponse]
# The new HTTP Request object.
#
def http_put(path,**kwargs,&block)
if debug?
print_debug "Requesting PUT #{url_for(path)} ..."
end
http.put(path,**kwargs,&block)
end
#
# Performs a `TRACE` request for the given URI.
#
# @param [String] path
# The path to create the HTTP request for.
#
# @!macro request_kwargs
#
# @yield [response]
# If a block is given it will be passed the received HTTP response.
#
# @yieldparam [Net::HTTPRresponse] response
# The received HTTP response object.
#
# @return [Net::HTTPResponse]
# The new HTTP Request object.
#
def http_trace(path,**kwargs,&block)
if debug?
print_debug "Requesting TRACE #{url_for(path)} ..."
end
http.trace(path,**kwargs,&block)
end
#
# Performs a `UNLOCK` request for the given URI.
#
# @param [String] path
# The path to create the HTTP request for.
#
# @!macro request_kwargs
#
# @yield [response]
# If a block is given it will be passed the received HTTP response.
#
# @yieldparam [Net::HTTPRresponse] response
# The received HTTP response object.
#
# @return [Net::HTTPResponse]
# The new HTTP Request object.
#
def http_unlock(path,**kwargs,&block)
if debug?
print_debug "Requesting UNLOCK #{url_for(path)} ..."
end
http.unlock(path,**kwargs,&block)
end
end
end
end
end