mirror of
https://github.com/ruby/ruby.git
synced 2022-11-09 12:17:21 -05:00
df731e37a1
* lib/soap/header/* * lib/soap/rpc/httpserver.rb * lib/wsdl/soap/cgiStubCreator.rb * lib/wsdl/soap/classDefCreator.rb * lib/wsdl/soap/classDefCreatorSupport.rb * lib/wsdl/soap/clientSkeltonCreator.rb * lib/wsdl/soap/driverCreator.rb * lib/wsdl/soap/mappingRegistryCreator.rb * lib/wsdl/soap/methodDefCreator.rb * lib/wsdl/soap/servantSkeltonCreator.rb * lib/wsdl/soap/standaloneServerStubCreator.rb * lib/wsdl/xmlSchema/enumeration.rb * lib/wsdl/xmlSchema/simpleRestriction.rb * lib/wsdl/xmlSchema/simpleType.rb * lib/xsd/codegen/* * lib/xsd/codegen.rb * sample/soap/authheader/* * sample/soap/raa2.4/* * sample/soap/ssl/* * sample/soap/swa/* * sample/soap/whois.rb * sample/wsdl/raa2.4/* * test/soap/header/* * test/soap/ssl/* * test/soap/struct/* * test/soap/swa/* * test/soap/wsdlDriver/* * test/wsdl/multiplefault.wsdl * test/wsdl/simpletype/* * test/wsdl/test_multiplefault.rb * modified files: * lib/soap/baseData.rb * lib/soap/element.rb * lib/soap/generator.rb * lib/soap/netHttpClient.rb * lib/soap/parser.rb * lib/soap/property.rb * lib/soap/soap.rb * lib/soap/streamHandler.rb * lib/soap/wsdlDriver.rb * lib/soap/wsdlDriver.rb * lib/soap/encodingstyle/handler.rb * lib/soap/encodingstyle/literalHandler.rb * lib/soap/encodingstyle/soapHandler.rb * lib/soap/mapping/factory.rb * lib/soap/mapping/mapping.rb * lib/soap/mapping/registry.rb * lib/soap/mapping/rubytypeFactory.rb * lib/soap/mapping/wsdlRegistry.rb * lib/soap/rpc/cgistub.rb * lib/soap/rpc/driver.rb * lib/soap/rpc/proxy.rb * lib/soap/rpc/router.rb * lib/soap/rpc/soaplet.rb * lib/soap/rpc/standaloneServer.rb * lib/wsdl/data.rb * lib/wsdl/definitions.rb * lib/wsdl/operation.rb * lib/wsdl/parser.rb * lib/wsdl/soap/definitions.rb * lib/wsdl/xmlSchema/complexContent.rb * lib/wsdl/xmlSchema/complexType.rb * lib/wsdl/xmlSchema/data.rb * lib/wsdl/xmlSchema/parser.rb * lib/wsdl/xmlSchema/schema.rb * lib/xsd/datatypes.rb * lib/xsd/qname.rb * sample/soap/sampleStruct/server.rb * sample/wsdl/amazon/AmazonSearch.rb * sample/wsdl/amazon/AmazonSearchDriver.rb * test/soap/test_property.rb * test/soap/calc/test_calc_cgi.rb * test/wsdl/test_emptycomplextype.rb * summary * add SOAP Header mustUnderstand support. * add HTTP client SSL configuration and Cookies support (works completely with http-access2). * add header handler for handling sending/receiving SOAP Header. * map Ruby's anonymous Struct to common SOAP Struct in SOAP Object Model. it caused error. * add WSDL simpleType support to restrict lexical value space. git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@6565 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
479 lines
14 KiB
Ruby
479 lines
14 KiB
Ruby
# SOAP4R - Mapping registry.
|
|
# Copyright (C) 2000, 2001, 2002, 2003 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'
|
|
require 'soap/mapping/mapping'
|
|
require 'soap/mapping/typeMap'
|
|
require 'soap/mapping/factory'
|
|
require 'soap/mapping/rubytypeFactory'
|
|
|
|
|
|
module SOAP
|
|
|
|
|
|
module Marshallable
|
|
# @@type_ns = Mapping::RubyCustomTypeNamespace
|
|
end
|
|
|
|
|
|
module Mapping
|
|
|
|
|
|
module MappedException; end
|
|
|
|
|
|
RubyTypeName = XSD::QName.new(RubyTypeInstanceNamespace, 'rubyType')
|
|
RubyExtendName = XSD::QName.new(RubyTypeInstanceNamespace, 'extends')
|
|
RubyIVarName = XSD::QName.new(RubyTypeInstanceNamespace, 'ivars')
|
|
|
|
|
|
# Inner class to pass an exception.
|
|
class SOAPException; include Marshallable
|
|
attr_reader :excn_type_name, :cause
|
|
def initialize(e)
|
|
@excn_type_name = Mapping.name2elename(e.class.to_s)
|
|
@cause = e
|
|
end
|
|
|
|
def to_e
|
|
if @cause.is_a?(::Exception)
|
|
@cause.extend(::SOAP::Mapping::MappedException)
|
|
return @cause
|
|
elsif @cause.respond_to?(:message) and @cause.respond_to?(:backtrace)
|
|
e = RuntimeError.new(@cause.message)
|
|
e.set_backtrace(@cause.backtrace)
|
|
return e
|
|
end
|
|
klass = Mapping.class_from_name(
|
|
Mapping.elename2name(@excn_type_name.to_s))
|
|
if klass.nil? or not klass <= ::Exception
|
|
return RuntimeError.new(@cause.inspect)
|
|
end
|
|
obj = klass.new(@cause.message)
|
|
obj.extend(::SOAP::Mapping::MappedException)
|
|
obj
|
|
end
|
|
end
|
|
|
|
|
|
# For anyType object: SOAP::Mapping::Object not ::Object
|
|
class Object; include Marshallable
|
|
def initialize
|
|
@__members = []
|
|
@__value_type = {}
|
|
end
|
|
|
|
def [](name)
|
|
if @__members.include?(name)
|
|
self.__send__(name)
|
|
else
|
|
self.__send__(Object.safe_name(name))
|
|
end
|
|
end
|
|
|
|
def []=(name, value)
|
|
if @__members.include?(name)
|
|
self.__send__(name + '=', value)
|
|
else
|
|
self.__send__(Object.safe_name(name) + '=', value)
|
|
end
|
|
end
|
|
|
|
def __set_property(name, value)
|
|
var_name = name
|
|
unless @__members.include?(name)
|
|
var_name = __define_attr_accessor(var_name)
|
|
end
|
|
__set_property_value(var_name, value)
|
|
var_name
|
|
end
|
|
|
|
def __members
|
|
@__members
|
|
end
|
|
|
|
private
|
|
|
|
def __set_property_value(name, value)
|
|
org = self.__send__(name)
|
|
case @__value_type[name]
|
|
when :single
|
|
self.__send__(name + '=', [org, value])
|
|
@__value_type[name] = :multi
|
|
when :multi
|
|
org << value
|
|
else
|
|
self.__send__(name + '=', value)
|
|
@__value_type[name] = :single
|
|
end
|
|
value
|
|
end
|
|
|
|
def __define_attr_accessor(name)
|
|
var_name = name
|
|
begin
|
|
instance_eval <<-EOS
|
|
def #{ var_name }
|
|
@#{ var_name }
|
|
end
|
|
|
|
def #{ var_name }=(value)
|
|
@#{ var_name } = value
|
|
end
|
|
EOS
|
|
rescue SyntaxError
|
|
var_name = Object.safe_name(var_name)
|
|
retry
|
|
end
|
|
@__members << var_name
|
|
var_name
|
|
end
|
|
|
|
def Object.safe_name(name)
|
|
require 'md5'
|
|
"var_" << MD5.new(name).hexdigest
|
|
end
|
|
end
|
|
|
|
|
|
class MappingError < Error; end
|
|
|
|
|
|
class Registry
|
|
class Map
|
|
def initialize(registry)
|
|
@map = []
|
|
@registry = registry
|
|
end
|
|
|
|
def obj2soap(klass, obj, type_qname = nil)
|
|
@map.each do |obj_class, soap_class, factory, info|
|
|
if klass == obj_class or
|
|
(info[:derived_class] and klass <= obj_class)
|
|
ret = factory.obj2soap(soap_class, obj, info, @registry)
|
|
return ret if ret
|
|
end
|
|
end
|
|
nil
|
|
end
|
|
|
|
def soap2obj(klass, node)
|
|
@map.each do |obj_class, soap_class, factory, info|
|
|
if klass == soap_class or
|
|
(info[:derived_class] and klass <= soap_class)
|
|
conv, obj = factory.soap2obj(obj_class, node, info, @registry)
|
|
return true, obj if conv
|
|
end
|
|
end
|
|
return false
|
|
end
|
|
|
|
# Give priority to former entry.
|
|
def init(init_map = [])
|
|
clear
|
|
init_map.reverse_each do |obj_class, soap_class, factory, info|
|
|
add(obj_class, soap_class, factory, info)
|
|
end
|
|
end
|
|
|
|
# Give priority to latter entry.
|
|
def add(obj_class, soap_class, factory, info)
|
|
info ||= {}
|
|
@map.unshift([obj_class, soap_class, factory, info])
|
|
end
|
|
|
|
def clear
|
|
@map.clear
|
|
end
|
|
|
|
def find_mapped_soap_class(target_obj_class)
|
|
@map.each do |obj_class, soap_class, factory, info|
|
|
if obj_class == target_obj_class
|
|
return soap_class
|
|
end
|
|
end
|
|
nil
|
|
end
|
|
|
|
def find_mapped_obj_class(target_soap_class)
|
|
@map.each do |obj_class, soap_class, factory, info|
|
|
if soap_class == target_soap_class
|
|
return obj_class
|
|
end
|
|
end
|
|
nil
|
|
end
|
|
end
|
|
|
|
StringFactory = StringFactory_.new
|
|
BasetypeFactory = BasetypeFactory_.new
|
|
DateTimeFactory = DateTimeFactory_.new
|
|
ArrayFactory = ArrayFactory_.new
|
|
Base64Factory = Base64Factory_.new
|
|
URIFactory = URIFactory_.new
|
|
TypedArrayFactory = TypedArrayFactory_.new
|
|
TypedStructFactory = TypedStructFactory_.new
|
|
|
|
HashFactory = HashFactory_.new
|
|
|
|
SOAPBaseMap = [
|
|
[::NilClass, ::SOAP::SOAPNil, BasetypeFactory],
|
|
[::TrueClass, ::SOAP::SOAPBoolean, BasetypeFactory],
|
|
[::FalseClass, ::SOAP::SOAPBoolean, BasetypeFactory],
|
|
[::String, ::SOAP::SOAPString, StringFactory],
|
|
[::DateTime, ::SOAP::SOAPDateTime, DateTimeFactory],
|
|
[::Date, ::SOAP::SOAPDateTime, DateTimeFactory],
|
|
[::Date, ::SOAP::SOAPDate, DateTimeFactory],
|
|
[::Time, ::SOAP::SOAPDateTime, DateTimeFactory],
|
|
[::Time, ::SOAP::SOAPTime, DateTimeFactory],
|
|
[::Float, ::SOAP::SOAPDouble, BasetypeFactory,
|
|
{:derived_class => true}],
|
|
[::Float, ::SOAP::SOAPFloat, BasetypeFactory,
|
|
{:derived_class => true}],
|
|
[::Integer, ::SOAP::SOAPInt, BasetypeFactory,
|
|
{:derived_class => true}],
|
|
[::Integer, ::SOAP::SOAPLong, BasetypeFactory,
|
|
{:derived_class => true}],
|
|
[::Integer, ::SOAP::SOAPInteger, BasetypeFactory,
|
|
{:derived_class => true}],
|
|
[::Integer, ::SOAP::SOAPShort, BasetypeFactory,
|
|
{:derived_class => true}],
|
|
[::URI::Generic, ::SOAP::SOAPAnyURI, URIFactory,
|
|
{:derived_class => true}],
|
|
[::String, ::SOAP::SOAPBase64, Base64Factory],
|
|
[::String, ::SOAP::SOAPHexBinary, Base64Factory],
|
|
[::String, ::SOAP::SOAPDecimal, BasetypeFactory],
|
|
[::String, ::SOAP::SOAPDuration, BasetypeFactory],
|
|
[::String, ::SOAP::SOAPGYearMonth, BasetypeFactory],
|
|
[::String, ::SOAP::SOAPGYear, BasetypeFactory],
|
|
[::String, ::SOAP::SOAPGMonthDay, BasetypeFactory],
|
|
[::String, ::SOAP::SOAPGDay, BasetypeFactory],
|
|
[::String, ::SOAP::SOAPGMonth, BasetypeFactory],
|
|
[::String, ::SOAP::SOAPQName, BasetypeFactory],
|
|
|
|
[::Hash, ::SOAP::SOAPArray, HashFactory],
|
|
[::Hash, ::SOAP::SOAPStruct, HashFactory],
|
|
|
|
[::Array, ::SOAP::SOAPArray, ArrayFactory,
|
|
{:derived_class => true}],
|
|
|
|
[::SOAP::Mapping::SOAPException,
|
|
::SOAP::SOAPStruct, TypedStructFactory,
|
|
{:type => XSD::QName.new(RubyCustomTypeNamespace, "SOAPException")}],
|
|
]
|
|
|
|
RubyOriginalMap = [
|
|
[::NilClass, ::SOAP::SOAPNil, BasetypeFactory],
|
|
[::TrueClass, ::SOAP::SOAPBoolean, BasetypeFactory],
|
|
[::FalseClass, ::SOAP::SOAPBoolean, BasetypeFactory],
|
|
[::String, ::SOAP::SOAPString, StringFactory],
|
|
[::DateTime, ::SOAP::SOAPDateTime, DateTimeFactory],
|
|
[::Date, ::SOAP::SOAPDateTime, DateTimeFactory],
|
|
[::Date, ::SOAP::SOAPDate, DateTimeFactory],
|
|
[::Time, ::SOAP::SOAPDateTime, DateTimeFactory],
|
|
[::Time, ::SOAP::SOAPTime, DateTimeFactory],
|
|
[::Float, ::SOAP::SOAPDouble, BasetypeFactory,
|
|
{:derived_class => true}],
|
|
[::Float, ::SOAP::SOAPFloat, BasetypeFactory,
|
|
{:derived_class => true}],
|
|
[::Integer, ::SOAP::SOAPInt, BasetypeFactory,
|
|
{:derived_class => true}],
|
|
[::Integer, ::SOAP::SOAPLong, BasetypeFactory,
|
|
{:derived_class => true}],
|
|
[::Integer, ::SOAP::SOAPInteger, BasetypeFactory,
|
|
{:derived_class => true}],
|
|
[::Integer, ::SOAP::SOAPShort, BasetypeFactory,
|
|
{:derived_class => true}],
|
|
[::URI::Generic, ::SOAP::SOAPAnyURI, URIFactory,
|
|
{:derived_class => true}],
|
|
[::String, ::SOAP::SOAPBase64, Base64Factory],
|
|
[::String, ::SOAP::SOAPHexBinary, Base64Factory],
|
|
[::String, ::SOAP::SOAPDecimal, BasetypeFactory],
|
|
[::String, ::SOAP::SOAPDuration, BasetypeFactory],
|
|
[::String, ::SOAP::SOAPGYearMonth, BasetypeFactory],
|
|
[::String, ::SOAP::SOAPGYear, BasetypeFactory],
|
|
[::String, ::SOAP::SOAPGMonthDay, BasetypeFactory],
|
|
[::String, ::SOAP::SOAPGDay, BasetypeFactory],
|
|
[::String, ::SOAP::SOAPGMonth, BasetypeFactory],
|
|
[::String, ::SOAP::SOAPQName, BasetypeFactory],
|
|
|
|
[::Hash, ::SOAP::SOAPArray, HashFactory],
|
|
[::Hash, ::SOAP::SOAPStruct, HashFactory],
|
|
|
|
# Does not allow Array's subclass here.
|
|
[::Array, ::SOAP::SOAPArray, ArrayFactory],
|
|
|
|
[::SOAP::Mapping::SOAPException,
|
|
::SOAP::SOAPStruct, TypedStructFactory,
|
|
{:type => XSD::QName.new(RubyCustomTypeNamespace, "SOAPException")}],
|
|
]
|
|
|
|
def initialize(config = {})
|
|
@config = config
|
|
@map = Map.new(self)
|
|
if @config[:allow_original_mapping]
|
|
@allow_original_mapping = true
|
|
@map.init(RubyOriginalMap)
|
|
else
|
|
@allow_original_mapping = false
|
|
@map.init(SOAPBaseMap)
|
|
end
|
|
|
|
@allow_untyped_struct = @config.key?(:allow_untyped_struct) ?
|
|
@config[:allow_untyped_struct] : true
|
|
@rubytype_factory = RubytypeFactory.new(
|
|
:allow_untyped_struct => @allow_untyped_struct,
|
|
:allow_original_mapping => @allow_original_mapping
|
|
)
|
|
@default_factory = @rubytype_factory
|
|
@excn_handler_obj2soap = nil
|
|
@excn_handler_soap2obj = nil
|
|
end
|
|
|
|
def add(obj_class, soap_class, factory, info = nil)
|
|
@map.add(obj_class, soap_class, factory, info)
|
|
end
|
|
alias set add
|
|
|
|
# This mapping registry ignores type hint.
|
|
def obj2soap(klass, obj, type_qname = nil)
|
|
soap = _obj2soap(klass, obj, type_qname)
|
|
if @allow_original_mapping
|
|
addextend2soap(soap, obj)
|
|
end
|
|
soap
|
|
end
|
|
|
|
def soap2obj(klass, node)
|
|
obj = _soap2obj(klass, node)
|
|
if @allow_original_mapping
|
|
addextend2obj(obj, node.extraattr[RubyExtendName])
|
|
addiv2obj(obj, node.extraattr[RubyIVarName])
|
|
end
|
|
obj
|
|
end
|
|
|
|
def default_factory=(factory)
|
|
@default_factory = factory
|
|
end
|
|
|
|
def excn_handler_obj2soap=(handler)
|
|
@excn_handler_obj2soap = handler
|
|
end
|
|
|
|
def excn_handler_soap2obj=(handler)
|
|
@excn_handler_soap2obj = handler
|
|
end
|
|
|
|
def find_mapped_soap_class(obj_class)
|
|
@map.find_mapped_soap_class(obj_class)
|
|
end
|
|
|
|
def find_mapped_obj_class(soap_class)
|
|
@map.find_mapped_obj_class(soap_class)
|
|
end
|
|
|
|
private
|
|
|
|
def _obj2soap(klass, obj, type_qname)
|
|
ret = nil
|
|
if obj.is_a?(SOAPStruct) or obj.is_a?(SOAPArray)
|
|
obj.replace do |ele|
|
|
Mapping._obj2soap(ele, self)
|
|
end
|
|
return obj
|
|
elsif obj.is_a?(SOAPBasetype)
|
|
return obj
|
|
end
|
|
begin
|
|
ret = @map.obj2soap(klass, obj, type_qname) ||
|
|
@default_factory.obj2soap(klass, obj, nil, self)
|
|
rescue MappingError
|
|
end
|
|
return ret if ret
|
|
if @excn_handler_obj2soap
|
|
ret = @excn_handler_obj2soap.call(obj) { |yield_obj|
|
|
Mapping._obj2soap(yield_obj, self)
|
|
}
|
|
end
|
|
return ret if ret
|
|
raise MappingError.new("Cannot map #{ klass.name } to SOAP/OM.")
|
|
end
|
|
|
|
# Might return nil as a mapping result.
|
|
def _soap2obj(klass, node)
|
|
if node.extraattr.key?(RubyTypeName)
|
|
conv, obj = @rubytype_factory.soap2obj(klass, node, nil, self)
|
|
return obj if conv
|
|
else
|
|
conv, obj = @map.soap2obj(klass, node)
|
|
return obj if conv
|
|
conv, obj = @default_factory.soap2obj(klass, node, nil, self)
|
|
return obj if conv
|
|
end
|
|
|
|
if @excn_handler_soap2obj
|
|
begin
|
|
return @excn_handler_soap2obj.call(node) { |yield_node|
|
|
Mapping._soap2obj(yield_node, self)
|
|
}
|
|
rescue Exception
|
|
end
|
|
end
|
|
raise MappingError.new("Cannot map #{ node.type.name } to Ruby object.")
|
|
end
|
|
|
|
def addiv2obj(obj, attr)
|
|
return unless attr
|
|
vars = {}
|
|
attr.__getobj__.each do |name, value|
|
|
vars[name] = Mapping._soap2obj(value, self)
|
|
end
|
|
Mapping.set_instance_vars(obj, vars)
|
|
end
|
|
|
|
if RUBY_VERSION >= '1.8.0'
|
|
def addextend2obj(obj, attr)
|
|
return unless attr
|
|
attr.split(/ /).reverse_each do |mstr|
|
|
obj.extend(Mapping.class_from_name(mstr))
|
|
end
|
|
end
|
|
else
|
|
# (class < false; self; end).ancestors includes "TrueClass" under 1.6...
|
|
def addextend2obj(obj, attr)
|
|
return unless attr
|
|
attr.split(/ /).reverse_each do |mstr|
|
|
m = Mapping.class_from_name(mstr)
|
|
obj.extend(m) if m.class == Module
|
|
end
|
|
end
|
|
end
|
|
|
|
def addextend2soap(node, obj)
|
|
return if obj.is_a?(Symbol) or obj.is_a?(Fixnum)
|
|
list = (class << obj; self; end).ancestors - obj.class.ancestors
|
|
unless list.empty?
|
|
node.extraattr[RubyExtendName] = list.collect { |c|
|
|
if c.name.empty?
|
|
raise TypeError.new("singleton can't be dumped #{ obj }")
|
|
end
|
|
c.name
|
|
}.join(" ")
|
|
end
|
|
end
|
|
|
|
end
|
|
|
|
|
|
DefaultRegistry = Registry.new
|
|
RubyOriginalRegistry = Registry.new(:allow_original_mapping => true)
|
|
|
|
|
|
end
|
|
end
|