mirror of
https://github.com/ruby/ruby.git
synced 2022-11-09 12:17:21 -05:00
eb3f829be9
== SOAP client and server == === for both client side and server side === * improved document/literal service support. style(rpc,document)/use(encoding, literal) combination are all supported. for the detail about combination, see test/soap/test_style.rb. * let WSDLEncodedRegistry#soap2obj map SOAP/OM to Ruby according to WSDL as well as obj2soap. closes #70. * let SOAP::Mapping::Object handle XML attribute for doc/lit service. you can set/get XML attribute via accessor methods which as a name 'xmlattr_' prefixed (<foo name="bar"/> -> Foo#xmlattr_name). === client side === * WSDLDriver capitalized name operation bug fixed. from 1.5.3-ruby1.8.2, operation which has capitalized name (such as KeywordSearchRequest in AWS) is defined as a method having uncapitalized name. (converted with GenSupport.safemethodname to handle operation name 'foo-bar'). it introduced serious incompatibility; in the past, it was defined as a capitalized. define capitalized method as well under that circumstance. * added new factory interface 'WSDLDriverFactory#create_rpc_driver' to create RPC::Driver, not WSDLDriver (RPC::Driver and WSDLDriver are merged). 'WSDLDriverFactory#create_driver' still creates WSDLDriver for compatibility but it warns that the method is deprecated. please use create_rpc_driver instead of create_driver. * allow to use an URI object as an endpoint_url even with net/http, not http-access2. === server side === * added mod_ruby support to SOAP::CGIStub. rename a CGI script server.cgi to server.rb and let mod_ruby's RubyHandler handles the script. CGIStub detects if it's running under mod_ruby environment or not. * added fcgi support to SOAP::CGIStub. see the sample at sample/soap/calc/server.fcgi. (almost same as server.cgi but has fcgi handler at the bottom.) * allow to return a SOAPFault object to respond customized SOAP fault. * added the interface 'generate_explicit_type' for server side (CGIStub, HTTPServer). call 'self.generate_explicit_type = true' if you want to return simplified XML even if it's rpc/encoded service. == WSDL == === WSDL definition === * improved XML Schema support such as extension, restriction, simpleType, complexType + simpleContent, ref, length, import, include. * reduced "unknown element/attribute" warnings (warn only 1 time for each QName). * importing XSD file at schemaLocation with xsd:import. === code generation from WSDL === * generator crashed when there's '-' in defined element/attribute name. * added ApacheMap WSDL definition. * sample/{soap,wsdl}: removed. git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@8500 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
325 lines
6.7 KiB
Ruby
325 lines
6.7 KiB
Ruby
# SOAP4R - RPC element definition.
|
|
# Copyright (C) 2000, 2001, 2003, 2005 NAKAMURA, Hiroshi <nahi@ruby-lang.org>.
|
|
|
|
# This program is copyrighted free software by NAKAMURA, Hiroshi. You can
|
|
# redistribute it and/or modify it under the same terms of Ruby's license;
|
|
# either the dual license version in 2003, or any later version.
|
|
|
|
|
|
require 'soap/baseData'
|
|
|
|
|
|
module SOAP
|
|
|
|
# Add method definitions for RPC to common definition in element.rb
|
|
class SOAPBody < SOAPStruct
|
|
public
|
|
|
|
def request
|
|
root_node
|
|
end
|
|
|
|
def response
|
|
if !@is_fault
|
|
if void?
|
|
nil
|
|
else
|
|
# Initial element is [retval].
|
|
root_node[0]
|
|
end
|
|
else
|
|
root_node
|
|
end
|
|
end
|
|
|
|
def outparams
|
|
if !@is_fault and !void?
|
|
op = root_node[1..-1]
|
|
op = nil if op && op.empty?
|
|
op
|
|
else
|
|
nil
|
|
end
|
|
end
|
|
|
|
def void?
|
|
root_node.nil?
|
|
end
|
|
|
|
def fault
|
|
if @is_fault
|
|
self['fault']
|
|
else
|
|
nil
|
|
end
|
|
end
|
|
|
|
def fault=(fault)
|
|
@is_fault = true
|
|
add_member('fault', fault)
|
|
end
|
|
end
|
|
|
|
|
|
module RPC
|
|
|
|
|
|
class RPCError < Error; end
|
|
class MethodDefinitionError < RPCError; end
|
|
class ParameterError < RPCError; end
|
|
|
|
class SOAPMethod < SOAPStruct
|
|
RETVAL = 'retval'
|
|
IN = 'in'
|
|
OUT = 'out'
|
|
INOUT = 'inout'
|
|
|
|
attr_reader :param_def
|
|
attr_reader :inparam
|
|
attr_reader :outparam
|
|
attr_reader :retval_name
|
|
attr_reader :retval_class_name
|
|
|
|
def initialize(qname, param_def = nil)
|
|
super(nil)
|
|
@elename = qname
|
|
@encodingstyle = nil
|
|
|
|
@param_def = param_def
|
|
|
|
@signature = []
|
|
@inparam_names = []
|
|
@inoutparam_names = []
|
|
@outparam_names = []
|
|
|
|
@inparam = {}
|
|
@outparam = {}
|
|
@retval_name = nil
|
|
@retval_class_name = nil
|
|
|
|
init_param(@param_def) if @param_def
|
|
end
|
|
|
|
def have_outparam?
|
|
@outparam_names.size > 0
|
|
end
|
|
|
|
def input_params
|
|
collect_params(IN, INOUT)
|
|
end
|
|
|
|
def output_params
|
|
collect_params(OUT, INOUT)
|
|
end
|
|
|
|
def set_param(params)
|
|
params.each do |param, data|
|
|
@inparam[param] = data
|
|
data.elename.name = param
|
|
data.parent = self
|
|
end
|
|
end
|
|
|
|
def set_outparam(params)
|
|
params.each do |param, data|
|
|
@outparam[param] = data
|
|
data.elename.name = param
|
|
end
|
|
end
|
|
|
|
def SOAPMethod.param_count(param_def, *type)
|
|
count = 0
|
|
param_def.each do |io_type, name, param_type|
|
|
if type.include?(io_type)
|
|
count += 1
|
|
end
|
|
end
|
|
count
|
|
end
|
|
|
|
def SOAPMethod.derive_rpc_param_def(obj, name, *param)
|
|
if param.size == 1 and param[0].is_a?(Array)
|
|
return param[0]
|
|
end
|
|
if param.empty?
|
|
method = obj.method(name)
|
|
param_names = (1..method.arity.abs).collect { |i| "p#{i}" }
|
|
else
|
|
param_names = param
|
|
end
|
|
create_rpc_param_def(param_names)
|
|
end
|
|
|
|
def SOAPMethod.create_rpc_param_def(param_names)
|
|
param_def = []
|
|
param_names.each do |param_name|
|
|
param_def.push([IN, param_name, nil])
|
|
end
|
|
param_def.push([RETVAL, 'return', nil])
|
|
param_def
|
|
end
|
|
|
|
def SOAPMethod.create_doc_param_def(req_qnames, res_qnames)
|
|
req_qnames = [req_qnames] if req_qnames.is_a?(XSD::QName)
|
|
res_qnames = [res_qnames] if res_qnames.is_a?(XSD::QName)
|
|
param_def = []
|
|
req_qnames.each do |qname|
|
|
param_def << [IN, qname.name, [nil, qname.namespace, qname.name]]
|
|
end
|
|
res_qnames.each do |qname|
|
|
param_def << [OUT, qname.name, [nil, qname.namespace, qname.name]]
|
|
end
|
|
param_def
|
|
end
|
|
|
|
private
|
|
|
|
def collect_params(*type)
|
|
names = []
|
|
@signature.each do |io_type, name, param_type|
|
|
names << name if type.include?(io_type)
|
|
end
|
|
names
|
|
end
|
|
|
|
def init_param(param_def)
|
|
param_def.each do |io_type, name, param_type|
|
|
case io_type
|
|
when IN
|
|
@signature.push([IN, name, param_type])
|
|
@inparam_names.push(name)
|
|
when OUT
|
|
@signature.push([OUT, name, param_type])
|
|
@outparam_names.push(name)
|
|
when INOUT
|
|
@signature.push([INOUT, name, param_type])
|
|
@inoutparam_names.push(name)
|
|
when RETVAL
|
|
if @retval_name
|
|
raise MethodDefinitionError.new('duplicated retval')
|
|
end
|
|
@retval_name = name
|
|
@retval_class_name = nil
|
|
if param_type
|
|
if param_type[0].is_a?(String)
|
|
@retval_class_name = Mapping.class_from_name(param_type[0])
|
|
else
|
|
@retval_class_name = param_type[0]
|
|
end
|
|
end
|
|
else
|
|
raise MethodDefinitionError.new("unknown type: #{io_type}")
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
|
|
class SOAPMethodRequest < SOAPMethod
|
|
attr_accessor :soapaction
|
|
|
|
def SOAPMethodRequest.create_request(qname, *params)
|
|
param_def = []
|
|
param_value = []
|
|
i = 0
|
|
params.each do |param|
|
|
param_name = "p#{i}"
|
|
i += 1
|
|
param_def << [IN, param_name, nil]
|
|
param_value << [param_name, param]
|
|
end
|
|
param_def << [RETVAL, 'return', nil]
|
|
o = new(qname, param_def)
|
|
o.set_param(param_value)
|
|
o
|
|
end
|
|
|
|
def initialize(qname, param_def = nil, soapaction = nil)
|
|
check_elename(qname)
|
|
super(qname, param_def)
|
|
@soapaction = soapaction
|
|
end
|
|
|
|
def each
|
|
input_params.each do |name|
|
|
unless @inparam[name]
|
|
raise ParameterError.new("parameter: #{name} was not given")
|
|
end
|
|
yield(name, @inparam[name])
|
|
end
|
|
end
|
|
|
|
def dup
|
|
req = self.class.new(@elename.dup, @param_def, @soapaction)
|
|
req.encodingstyle = @encodingstyle
|
|
req
|
|
end
|
|
|
|
def create_method_response(response_name = nil)
|
|
response_name ||=
|
|
XSD::QName.new(@elename.namespace, @elename.name + 'Response')
|
|
SOAPMethodResponse.new(response_name, @param_def)
|
|
end
|
|
|
|
private
|
|
|
|
def check_elename(qname)
|
|
# NCName & ruby's method name
|
|
unless /\A[\w_][\w\d_\-]*\z/ =~ qname.name
|
|
raise MethodDefinitionError.new("element name '#{qname.name}' not allowed")
|
|
end
|
|
end
|
|
end
|
|
|
|
|
|
class SOAPMethodResponse < SOAPMethod
|
|
|
|
def initialize(qname, param_def = nil)
|
|
super(qname, param_def)
|
|
@retval = nil
|
|
end
|
|
|
|
def retval=(retval)
|
|
@retval = retval
|
|
@retval.elename = @retval.elename.dup_name(@retval_name || 'return')
|
|
retval.parent = self
|
|
retval
|
|
end
|
|
|
|
def each
|
|
if @retval_name and !@retval.is_a?(SOAPVoid)
|
|
yield(@retval_name, @retval)
|
|
end
|
|
|
|
output_params.each do |name|
|
|
unless @outparam[name]
|
|
raise ParameterError.new("parameter: #{name} was not given")
|
|
end
|
|
yield(name, @outparam[name])
|
|
end
|
|
end
|
|
end
|
|
|
|
|
|
# To return(?) void explicitly.
|
|
# def foo(input_var)
|
|
# ...
|
|
# return SOAP::RPC::SOAPVoid.new
|
|
# end
|
|
class SOAPVoid < XSD::XSDAnySimpleType
|
|
include SOAPBasetype
|
|
extend SOAPModuleUtils
|
|
Name = XSD::QName.new(Mapping::RubyCustomTypeNamespace, nil)
|
|
|
|
public
|
|
def initialize()
|
|
@elename = Name
|
|
@id = nil
|
|
@precedents = []
|
|
@parent = nil
|
|
end
|
|
end
|
|
|
|
|
|
end
|
|
end
|