abstract.rb 8.5 KB
Newer Older
L
Leon Breedt 已提交
1 2 3 4 5 6 7
require 'benchmark'

module ActionWebService # :nodoc:
  module Dispatcher # :nodoc:
    class DispatcherError < ActionWebService::ActionWebServiceError # :nodoc:
    end

8
    def self.included(base) # :nodoc:
L
Leon Breedt 已提交
9 10 11 12 13 14 15
      base.class_inheritable_option(:web_service_dispatching_mode, :direct)
      base.class_inheritable_option(:web_service_exception_reporting, true)
      base.send(:include, ActionWebService::Dispatcher::InstanceMethods)
    end

    module InstanceMethods # :nodoc:
      private
16 17
        def invoke_web_service_request(protocol_request)
          invocation = web_service_invocation(protocol_request)
18 19 20 21
          if invocation.is_a?(Array) && protocol_request.protocol.is_a?(Protocol::XmlRpc::XmlRpcProtocol)
            xmlrpc_multicall_invoke(invocation)
          else
            web_service_invoke(invocation)
L
Leon Breedt 已提交
22 23
          end
        end
24 25 26
      
        def web_service_direct_invoke(invocation)
          @method_params = invocation.method_ordered_params
27
          arity = method(invocation.api_method.name).arity rescue 0
28
          if arity < 0 || arity > 0
29
            params = @method_params
30
          else
31
            params = []
32
          end
33
          web_service_filtered_invoke(invocation, params)
L
Leon Breedt 已提交
34 35
        end

36
        def web_service_delegated_invoke(invocation)
37 38 39 40
          web_service_filtered_invoke(invocation, invocation.method_ordered_params)
        end

        def web_service_filtered_invoke(invocation, params)
L
Leon Breedt 已提交
41
          cancellation_reason = nil
42
          return_value = invocation.service.perform_invocation(invocation.api_method.name, params) do |x|
L
Leon Breedt 已提交
43 44 45 46 47
            cancellation_reason = x
          end
          if cancellation_reason
            raise(DispatcherError, "request canceled: #{cancellation_reason}")
          end
48 49 50 51 52 53 54 55 56 57
          return_value
        end
        
        def web_service_invoke(invocation)
          case web_service_dispatching_mode
          when :direct
            return_value = web_service_direct_invoke(invocation)
          when :delegated, :layered
            return_value = web_service_delegated_invoke(invocation)
          end
58
          web_service_create_response(invocation.protocol, invocation.protocol_options, invocation.api, invocation.api_method, return_value)
L
Leon Breedt 已提交
59
        end
60 61 62 63
        
        def xmlrpc_multicall_invoke(invocations)
          responses = []
          invocations.each do |invocation|
64 65 66 67
            if invocation.is_a?(Hash)
              responses << invocation
              next
            end
68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86
            begin
              case web_service_dispatching_mode
              when :direct
                return_value = web_service_direct_invoke(invocation)
              when :delegated, :layered
                return_value = web_service_delegated_invoke(invocation)
              end
              api_method = invocation.api_method
              if invocation.api.has_api_method?(api_method.name)
                return_value = api_method.cast_returns(return_value)
              end
              responses << [return_value]
            rescue Exception => e
              responses << { 'faultCode' => 3, 'faultString' => e.message }
            end
          end
          invocation = invocations[0]
          invocation.protocol.encode_response('system.multicall', responses, nil, invocation.protocol_options)
        end
L
Leon Breedt 已提交
87

88
        def web_service_invocation(request, level = 0)
89
          public_method_name = request.method_name
90 91
          invocation = Invocation.new
          invocation.protocol = request.protocol
92
          invocation.protocol_options = request.protocol_options
93
          invocation.service_name = request.service_name
94
          if web_service_dispatching_mode == :layered
95 96 97 98 99 100 101 102 103 104 105
            case invocation.protocol
            when Protocol::Soap::SoapProtocol
              soap_action = request.protocol_options[:soap_action]
              if soap_action && soap_action =~ /^\/\w+\/(\w+)\//
                invocation.service_name = $1
              end
            when Protocol::XmlRpc::XmlRpcProtocol
              if request.method_name =~ /^([^\.]+)\.(.*)$/
                public_method_name = $2
                invocation.service_name = $1
              end
106
            end
107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125
          end
          if invocation.protocol.is_a? Protocol::XmlRpc::XmlRpcProtocol
            if public_method_name == 'multicall' && invocation.service_name == 'system'
              if level > 0
                raise(DispatcherError, "Recursive system.multicall invocations not allowed")
              end
              multicall = request.method_params.dup
              unless multicall.is_a?(Array) && multicall[0].is_a?(Array)
                raise(DispatcherError, "Malformed multicall (expected array of Hash elements)")
              end
              multicall = multicall[0]
              return multicall.map do |item|
                raise(DispatcherError, "Multicall elements must be Hash") unless item.is_a?(Hash)
                raise(DispatcherError, "Multicall elements must contain a 'methodName' key") unless item.has_key?('methodName')
                method_name = item['methodName']
                params = item.has_key?('params') ? item['params'] : []
                multicall_request = request.dup
                multicall_request.method_name = method_name
                multicall_request.method_params = params
126 127 128 129 130
                begin
                  web_service_invocation(multicall_request, level + 1)
                rescue Exception => e
                  {'faultCode' => 4, 'faultMessage' => e.message}
                end
131 132
              end
            end
133
          end
L
Leon Breedt 已提交
134 135
          case web_service_dispatching_mode
          when :direct
136 137
            invocation.api = self.class.web_service_api
            invocation.service = self
138
          when :delegated, :layered
139
            invocation.service = web_service_object(invocation.service_name)
140
            invocation.api = invocation.service.class.web_service_api
L
Leon Breedt 已提交
141
          end
142 143 144
          if invocation.api.nil?
            raise(DispatcherError, "no API attached to #{invocation.service.class}")
          end
145
          invocation.protocol.register_api(invocation.api)
146
          request.api = invocation.api
147
          if invocation.api.has_public_api_method?(public_method_name)
148
            invocation.api_method = invocation.api.public_api_method_instance(public_method_name)
149 150 151 152
          else
            if invocation.api.default_api_method.nil?
              raise(DispatcherError, "no such method '#{public_method_name}' on API #{invocation.api}")
            else
153
              invocation.api_method = invocation.api.default_api_method_instance
154 155
            end
          end
156 157 158
          if invocation.service.nil?
            raise(DispatcherError, "no service available for service name #{invocation.service_name}")
          end
159
          unless invocation.service.respond_to?(invocation.api_method.name)
160
            raise(DispatcherError, "no such method '#{public_method_name}' on API #{invocation.api} (#{invocation.api_method.name})")
161
          end
162 163
          request.api_method = invocation.api_method
          begin
164
            invocation.method_ordered_params = invocation.api_method.cast_expects(request.method_params.dup)
165
          rescue
166 167
            logger.warn "Casting of method parameters failed" unless logger.nil?
            invocation.method_ordered_params = request.method_params
L
Leon Breedt 已提交
168
          end
169
          request.method_params = invocation.method_ordered_params
170 171 172 173
          invocation.method_named_params = {}
          invocation.api_method.param_names.inject(0) do |m, n|
            invocation.method_named_params[n] = invocation.method_ordered_params[m]
            m + 1
174
          end
175
          invocation
L
Leon Breedt 已提交
176 177
        end

178
        def web_service_create_response(protocol, protocol_options, api, api_method, return_value)
179 180 181 182 183 184
          if api.has_api_method?(api_method.name)
            return_type = api_method.returns ? api_method.returns[0] : nil
            return_value = api_method.cast_returns(return_value)
          else
            return_type = ActionWebService::SignatureTypes.canonical_signature_entry(return_value.class, 0)
          end
185
          protocol.encode_response(api_method.public_name + 'Response', return_value, return_type, protocol_options)
186 187
        end

188
        class Invocation # :nodoc:
189
          attr_accessor :protocol
190
          attr_accessor :protocol_options
191 192
          attr_accessor :service_name
          attr_accessor :api
193
          attr_accessor :api_method
194 195 196
          attr_accessor :method_ordered_params
          attr_accessor :method_named_params
          attr_accessor :service
L
Leon Breedt 已提交
197 198 199 200
        end
    end
  end
end