lib/plugins/inspec-compliance/lib/inspec-compliance/http.rb



require "net/http" unless defined?(Net::HTTP)
require "net/http/post/multipart"
require "uri" unless defined?(URI)

module InspecPlugins
  module Compliance
    # implements a simple http abstraction on top of Net::HTTP
    class HTTP
      # generic get requires
      def self.get(url, headers = nil, insecure)
        uri = _parse_url(url)
        req = Net::HTTP::Get.new(uri.path)
        headers&.each do |key, value|
          req.add_field(key, value)
        end
        send_request(uri, req, insecure)
      end

      # generic post request
      def self.post(url, token, insecure, basic_auth = false)
        # form request
        uri = _parse_url(url)
        req = Net::HTTP::Post.new(uri.path)
        if basic_auth
          req.basic_auth token, ""
        else
          req["Authorization"] = "Bearer #{token}"
        end
        req.form_data = {}

        send_request(uri, req, insecure)
      end

      def self.post_with_headers(url, headers, body, insecure)
        uri = _parse_url(url)
        req = Net::HTTP::Post.new(uri.path)
        req.body = body unless body.nil?
        headers&.each do |key, value|
          req.add_field(key, value)
        end
        send_request(uri, req, insecure)
      end

      # post a file
      def self.post_file(url, headers, file_path, insecure)
        uri = _parse_url(url)
        raise "Unable to parse URL: #{url}" if uri.nil? || uri.host.nil?

        http = Net::HTTP.new(uri.host, uri.port)

        # set connection flags
        http.use_ssl = (uri.scheme == "https")
        http.verify_mode = OpenSSL::SSL::VERIFY_NONE if insecure

        req = Net::HTTP::Post.new(uri.path)
        headers.each do |key, value|
          req.add_field(key, value)
        end

        req.body_stream = File.open(file_path, "rb")
        req.add_field("Content-Length", File.size(file_path))
        req.add_field("Content-Type", "application/x-gzip")

        boundary = "INSPEC-PROFILE-UPLOAD"
        req.add_field("session", boundary)
        res = http.request(req)
        res
      end

      def self.post_multipart_file(url, headers, file_path, insecure)
        uri = _parse_url(url)
        raise "Unable to parse URL: #{url}" if uri.nil? || uri.host.nil?

        http = Net::HTTP.new(uri.host, uri.port)

        # set connection flags
        http.use_ssl = (uri.scheme == "https")
        http.verify_mode = OpenSSL::SSL::VERIFY_NONE if insecure

        File.open(file_path) do |tar|
          req = Net::HTTP::Post::Multipart.new(uri, "file" => UploadIO.new(tar, "application/x-gzip", File.basename(file_path)))
          headers.each do |key, value|
            req.add_field(key, value)
          end
          res = http.request(req)
          return res
        end
      end

      # sends a http requests
      def self.send_request(uri, req, insecure)
        opts = {
          use_ssl: uri.scheme == "https",
        }
        opts[:verify_mode] = OpenSSL::SSL::VERIFY_NONE if insecure

        raise "Unable to parse URI: #{uri}" if uri.nil? || uri.host.nil?

        res = Net::HTTP.start(uri.host, uri.port, opts) do |http|
          http.request(req)
        end
        res
      rescue OpenSSL::SSL::SSLError => e
        raise e unless e.message.include? "certificate verify failed"

        puts "Error: Failed to connect to #{uri}."
        puts "If the server uses a self-signed certificate, please re-run the login command with the --insecure option."
        exit 1
      end

      def self._parse_url(url)
        url = "https://#{url}" if URI.parse(url).scheme.nil?
        URI.parse(url)
      end
    end
  end
end