reset/ridley

View on GitHub
lib/ridley/middleware/chef_auth.rb

Summary

Maintainability
A
0 mins
Test Coverage
require 'mixlib/authentication/signedheaderauth'

module Ridley
  module Middleware
    class ChefAuth < Faraday::Middleware
      class << self
        include Mixlib::Authentication

        # Generate authentication headers for a request to a Chef Server
        #
        # @param [String] client_name
        # @param [String] client_key
        #   the path OR actual client key
        #
        # @option options [String] :host
        #
        # @see {#signing_object} for options
        def authentication_headers(client_name, client_key, options = {})
          contents = File.exists?(client_key) ? File.read(client_key) : client_key.to_s
          rsa_key = OpenSSL::PKey::RSA.new(contents)

          headers = signing_object(client_name, options).sign(rsa_key).merge(host: options[:host])
          headers.inject({}) { |memo, kv| memo["#{kv[0].to_s.upcase}"] = kv[1];memo }
        end

        # Create a signing object for a Request to a Chef Server
        #
        # @param [String] client_name
        #
        # @option options [String] :http_method
        # @option options [String] :path
        # @option options [String] :body
        # @option options [Time] :timestamp
        #
        # @return [SigningObject]
        def signing_object(client_name, options = {})
          options = options.reverse_merge(
            body: String.new,
            timestamp: Time.now.utc.iso8601
          )
          options[:user_id]       = client_name
          options[:proto_version] = "1.0"

          SignedHeaderAuth.signing_object(options)
        end
      end

      include Ridley::Logging

      attr_reader :client_name
      attr_reader :client_key

      def initialize(app, client_name, client_key)
        super(app)
        @client_name = client_name
        @client_key  = client_key
      end

      def call(env)
        signing_options = {
          http_method: env[:method],
          host: "#{env[:url].host}:#{env[:url].port}",
          path: env[:url].path,
          body: env[:body] || ''
        }
        authentication_headers = self.class.authentication_headers(client_name, client_key, signing_options)

        env[:request_headers] = default_headers.merge(env[:request_headers]).merge(authentication_headers)
        env[:request_headers] = env[:request_headers].merge('Content-Length' => env[:body].bytesize.to_s) if env[:body]

        log.debug { "==> performing authenticated Chef request as '#{client_name}'"}
        log.debug { "request env: #{env}"}

        @app.call(env)
      end

      private

        def default_headers
          {
            'Accept' => 'application/json',
            'Content-Type' => 'application/json',
            'X-Chef-Version' => Ridley::CHEF_VERSION
          }
        end
    end
  end
end

Faraday::Request.register_middleware chef_auth: Ridley::Middleware::ChefAuth