bigcommerce/gruf

View on GitHub
lib/gruf/controllers/request.rb

Summary

Maintainability
A
0 mins
Test Coverage
A
100%
# frozen_string_literal: true

# Copyright (c) 2017-present, BigCommerce Pty. Ltd. All rights reserved
#
# Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated
# documentation files (the "Software"), to deal in the Software without restriction, including without limitation the
# rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit
# persons to whom the Software is furnished to do so, subject to the following conditions:
#
# The above copyright notice and this permission notice shall be included in all copies or substantial portions of the
# Software.
#
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
# WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
# COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
# OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
#
module Gruf
  module Controllers
    ##
    # Encapsulates a request for a controller
    #
    class Request
      # @!attribute [r] message
      #   @return [Object] The protobuf message in the request
      attr_reader :message
      # @!attribute [r] active_call
      #   @return [GRPC::ActiveCall] The active call object used for this request
      attr_reader :active_call
      # @!attribute [r] method_key
      #   @return [Symbol] The method name being requested
      attr_reader :method_key
      # @!attribute [r] type
      #   @return [Gruf::Controllers::Request::Type] The type of request
      attr_reader :type
      # @!attribute [r] service
      #   @return [Class] The GRPC service class for this request
      attr_reader :service
      # @!attribute [r] context
      #   @return [::ActiveSupport::HashWithIndifferentAccess] An arbitrary hash of key/value entries that are
      #     accessible for interceptors, that can be used to shared information between interceptors and pass down into
      #     the controller.
      attr_reader :context

      delegate :metadata, to: :active_call
      delegate :messages, :client_streamer?, :server_streamer?, :bidi_streamer?, :request_response?, to: :type

      ##
      # Abstract representation of a gRPC request type
      #
      class Type
        delegate :client_streamer?, :server_streamer?, :bidi_streamer?, :request_response?, to: :@rpc_desc

        ##
        # Initialize a new request type object
        #
        # @param [GRPC::RpcDesc] rpc_desc The RPC descriptor for the request type
        #
        def initialize(rpc_desc)
          @rpc_desc = rpc_desc
        end
      end

      ##
      # Initialize an inbound controller request object
      #
      # @param [Symbol] method_key The method symbol of the RPC method being executed
      # @param [Class] service The class of the service being executed against
      # @param [GRPC::RpcDesc] rpc_desc The RPC descriptor of the call
      # @param [GRPC::ActiveCall] active_call The restricted view of the call
      # @param [Object|Google::Protobuf::MessageExts] message The protobuf message (or messages) of the request
      #
      def initialize(method_key:, service:, rpc_desc:, active_call:, message:)
        @method_key = method_key
        @service = service
        @active_call = active_call
        @message = message
        @rpc_desc = rpc_desc
        @type = Type.new(rpc_desc)
        @context = ::ActiveSupport::HashWithIndifferentAccess.new
      end

      ##
      # Returns the service name as a translated name separated by periods. Strips
      # the superfluous "Service" suffix from the name
      #
      # @return [String] The mapped service key
      #
      def service_key
        @service.name.to_s.underscore.tr('/', '.').gsub('.service', '')
      end

      ##
      # @return [Class] The class of the response message
      #
      def response_class
        @rpc_desc.output
      end

      ##
      # @return [Class] The class of the request message
      #
      def request_class
        @rpc_desc.input
      end

      ##
      # Parse the method signature into a service.method name format
      #
      # @return [String] The parsed service method name
      #
      def method_name
        "#{service_key}.#{@method_key}"
      end

      ##
      # Return all messages for this request, properly handling different request types
      #
      # @return [Enumerable<Object>] All messages for this request
      # @return [Object] If a bidi streamed request, will return the message object
      #
      def messages
        if client_streamer?
          # rubocop:disable Style/ExplicitBlockArgument
          @message.call { |msg| yield msg }
          # rubocop:enable Style/ExplicitBlockArgument
        elsif bidi_streamer?
          @message
        else
          [@message]
        end
      end
    end
  end
end