mirror of
https://github.com/rails/rails.git
synced 2022-11-09 12:12:34 -05:00
884fd4792d
consistency. action pack filters don't necessarily include enough information about the request since they occur before AWS actually sees the request and unpacks it. git-svn-id: http://svn-commit.rubyonrails.org/rails/trunk@1294 5ecf4fe2-1ee6-0310-87b1-e25e094e27de
139 lines
5.7 KiB
Ruby
139 lines
5.7 KiB
Ruby
require 'benchmark'
|
|
|
|
module ActionWebService # :nodoc:
|
|
module Dispatcher # :nodoc:
|
|
class DispatcherError < ActionWebService::ActionWebServiceError # :nodoc:
|
|
end
|
|
|
|
def self.append_features(base) # :nodoc:
|
|
super
|
|
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
|
|
def invoke_web_service_request(protocol_request)
|
|
invocation = web_service_invocation(protocol_request)
|
|
case web_service_dispatching_mode
|
|
when :direct
|
|
web_service_direct_invoke(invocation)
|
|
when :delegated, :layered
|
|
web_service_delegated_invoke(invocation)
|
|
end
|
|
end
|
|
|
|
def web_service_direct_invoke(invocation)
|
|
@method_params = invocation.method_ordered_params
|
|
arity = method(invocation.api_method.name).arity rescue 0
|
|
if arity < 0 || arity > 0
|
|
params = @method_params
|
|
else
|
|
params = []
|
|
end
|
|
web_service_filtered_invoke(invocation, params)
|
|
end
|
|
|
|
def web_service_delegated_invoke(invocation)
|
|
web_service_filtered_invoke(invocation, invocation.method_ordered_params)
|
|
end
|
|
|
|
def web_service_filtered_invoke(invocation, params)
|
|
cancellation_reason = nil
|
|
return_value = invocation.service.perform_invocation(invocation.api_method.name, params) do |x|
|
|
cancellation_reason = x
|
|
end
|
|
if cancellation_reason
|
|
raise(DispatcherError, "request canceled: #{cancellation_reason}")
|
|
end
|
|
web_service_create_response(invocation.protocol, invocation.protocol_options, invocation.api, invocation.api_method, return_value)
|
|
end
|
|
|
|
def web_service_invocation(request)
|
|
public_method_name = request.method_name
|
|
invocation = Invocation.new
|
|
invocation.protocol = request.protocol
|
|
invocation.protocol_options = request.protocol_options
|
|
invocation.service_name = request.service_name
|
|
if web_service_dispatching_mode == :layered
|
|
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
|
|
end
|
|
end
|
|
case web_service_dispatching_mode
|
|
when :direct
|
|
invocation.api = self.class.web_service_api
|
|
invocation.service = self
|
|
when :delegated, :layered
|
|
invocation.service = web_service_object(invocation.service_name)
|
|
invocation.api = invocation.service.class.web_service_api
|
|
end
|
|
if invocation.api.nil?
|
|
raise(DispatcherError, "no API attached to #{invocation.service.class}")
|
|
end
|
|
invocation.protocol.register_api(invocation.api)
|
|
request.api = invocation.api
|
|
if invocation.api.has_public_api_method?(public_method_name)
|
|
invocation.api_method = invocation.api.public_api_method_instance(public_method_name)
|
|
else
|
|
if invocation.api.default_api_method.nil?
|
|
raise(DispatcherError, "no such method '#{public_method_name}' on API #{invocation.api}")
|
|
else
|
|
invocation.api_method = invocation.api.default_api_method_instance
|
|
end
|
|
end
|
|
if invocation.service.nil?
|
|
raise(DispatcherError, "no service available for service name #{invocation.service_name}")
|
|
end
|
|
unless invocation.service.respond_to?(invocation.api_method.name)
|
|
raise(DispatcherError, "no such method '#{public_method_name}' on API #{invocation.api} (#{invocation.api_method.name})")
|
|
end
|
|
request.api_method = invocation.api_method
|
|
begin
|
|
invocation.method_ordered_params = invocation.api_method.cast_expects(request.method_params.dup)
|
|
rescue
|
|
logger.warn "Casting of method parameters failed" unless logger.nil?
|
|
invocation.method_ordered_params = request.method_params
|
|
end
|
|
request.method_params = invocation.method_ordered_params
|
|
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
|
|
end
|
|
invocation
|
|
end
|
|
|
|
def web_service_create_response(protocol, protocol_options, api, api_method, return_value)
|
|
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
|
|
protocol.encode_response(api_method.public_name + 'Response', return_value, return_type, protocol_options)
|
|
end
|
|
|
|
class Invocation # :nodoc:
|
|
attr_accessor :protocol
|
|
attr_accessor :protocol_options
|
|
attr_accessor :service_name
|
|
attr_accessor :api
|
|
attr_accessor :api_method
|
|
attr_accessor :method_ordered_params
|
|
attr_accessor :method_named_params
|
|
attr_accessor :service
|
|
end
|
|
end
|
|
end
|
|
end
|