railsware-soap4r 1.5.8.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (174) hide show
  1. data/README +5 -0
  2. data/TODO +0 -0
  3. data/bin/wsdl2ruby.rb +137 -0
  4. data/bin/xsd2ruby.rb +90 -0
  5. data/lib/soap/XMLSchemaDatatypes.rb +9 -0
  6. data/lib/soap/XMLSchemaDatatypes1999.rb +10 -0
  7. data/lib/soap/attachment.rb +108 -0
  8. data/lib/soap/baseData.rb +1092 -0
  9. data/lib/soap/cgistub.rb +9 -0
  10. data/lib/soap/charset.rb +9 -0
  11. data/lib/soap/compat.rb +182 -0
  12. data/lib/soap/driver.rb +9 -0
  13. data/lib/soap/element.rb +277 -0
  14. data/lib/soap/encodingstyle/aspDotNetHandler.rb +207 -0
  15. data/lib/soap/encodingstyle/handler.rb +120 -0
  16. data/lib/soap/encodingstyle/literalHandler.rb +192 -0
  17. data/lib/soap/encodingstyle/soapHandler.rb +559 -0
  18. data/lib/soap/filter.rb +13 -0
  19. data/lib/soap/filter/filterchain.rb +51 -0
  20. data/lib/soap/filter/handler.rb +31 -0
  21. data/lib/soap/filter/streamhandler.rb +29 -0
  22. data/lib/soap/generator.rb +304 -0
  23. data/lib/soap/header/handler.rb +61 -0
  24. data/lib/soap/header/handlerset.rb +70 -0
  25. data/lib/soap/header/mappinghandler.rb +47 -0
  26. data/lib/soap/header/simplehandler.rb +44 -0
  27. data/lib/soap/httpconfigloader.rb +139 -0
  28. data/lib/soap/mapping.rb +12 -0
  29. data/lib/soap/mapping/encodedregistry.rb +548 -0
  30. data/lib/soap/mapping/factory.rb +388 -0
  31. data/lib/soap/mapping/literalregistry.rb +388 -0
  32. data/lib/soap/mapping/mapping.rb +609 -0
  33. data/lib/soap/mapping/registry.rb +295 -0
  34. data/lib/soap/mapping/rubytypeFactory.rb +478 -0
  35. data/lib/soap/mapping/schemadefinition.rb +170 -0
  36. data/lib/soap/mapping/typeMap.rb +106 -0
  37. data/lib/soap/mapping/wsdlencodedregistry.rb +211 -0
  38. data/lib/soap/mapping/wsdlliteralregistry.rb +243 -0
  39. data/lib/soap/mappingRegistry.rb +9 -0
  40. data/lib/soap/marshal.rb +59 -0
  41. data/lib/soap/mimemessage.rb +241 -0
  42. data/lib/soap/namespace.rb +9 -0
  43. data/lib/soap/nestedexception.rb +42 -0
  44. data/lib/soap/netHttpClient.rb +241 -0
  45. data/lib/soap/ns.rb +34 -0
  46. data/lib/soap/parser.rb +252 -0
  47. data/lib/soap/processor.rb +66 -0
  48. data/lib/soap/property.rb +333 -0
  49. data/lib/soap/proxy.rb +14 -0
  50. data/lib/soap/qname.rb +9 -0
  51. data/lib/soap/rpc/cgistub.rb +247 -0
  52. data/lib/soap/rpc/driver.rb +247 -0
  53. data/lib/soap/rpc/element.rb +369 -0
  54. data/lib/soap/rpc/httpserver.rb +169 -0
  55. data/lib/soap/rpc/proxy.rb +576 -0
  56. data/lib/soap/rpc/router.rb +669 -0
  57. data/lib/soap/rpc/rpc.rb +25 -0
  58. data/lib/soap/rpc/soaplet.rb +210 -0
  59. data/lib/soap/rpc/standaloneServer.rb +43 -0
  60. data/lib/soap/rpcRouter.rb +9 -0
  61. data/lib/soap/rpcUtils.rb +9 -0
  62. data/lib/soap/server.rb +9 -0
  63. data/lib/soap/soap.rb +151 -0
  64. data/lib/soap/standaloneServer.rb +9 -0
  65. data/lib/soap/streamHandler.rb +302 -0
  66. data/lib/soap/wsdlDriver.rb +597 -0
  67. data/lib/tags +5690 -0
  68. data/lib/wsdl/binding.rb +65 -0
  69. data/lib/wsdl/data.rb +64 -0
  70. data/lib/wsdl/definitions.rb +236 -0
  71. data/lib/wsdl/documentation.rb +32 -0
  72. data/lib/wsdl/import.rb +80 -0
  73. data/lib/wsdl/importer.rb +38 -0
  74. data/lib/wsdl/info.rb +50 -0
  75. data/lib/wsdl/message.rb +54 -0
  76. data/lib/wsdl/operation.rb +178 -0
  77. data/lib/wsdl/operationBinding.rb +130 -0
  78. data/lib/wsdl/param.rb +93 -0
  79. data/lib/wsdl/parser.rb +164 -0
  80. data/lib/wsdl/part.rb +52 -0
  81. data/lib/wsdl/port.rb +84 -0
  82. data/lib/wsdl/portType.rb +75 -0
  83. data/lib/wsdl/service.rb +61 -0
  84. data/lib/wsdl/soap/address.rb +40 -0
  85. data/lib/wsdl/soap/binding.rb +49 -0
  86. data/lib/wsdl/soap/body.rb +58 -0
  87. data/lib/wsdl/soap/cgiStubCreator.rb +93 -0
  88. data/lib/wsdl/soap/classDefCreator.rb +433 -0
  89. data/lib/wsdl/soap/classDefCreatorSupport.rb +230 -0
  90. data/lib/wsdl/soap/classNameCreator.rb +54 -0
  91. data/lib/wsdl/soap/clientSkeltonCreator.rb +95 -0
  92. data/lib/wsdl/soap/complexType.rb +172 -0
  93. data/lib/wsdl/soap/data.rb +42 -0
  94. data/lib/wsdl/soap/definitions.rb +200 -0
  95. data/lib/wsdl/soap/driverCreator.rb +118 -0
  96. data/lib/wsdl/soap/element.rb +33 -0
  97. data/lib/wsdl/soap/encodedMappingRegistryCreator.rb +73 -0
  98. data/lib/wsdl/soap/fault.rb +56 -0
  99. data/lib/wsdl/soap/header.rb +86 -0
  100. data/lib/wsdl/soap/headerfault.rb +56 -0
  101. data/lib/wsdl/soap/literalMappingRegistryCreator.rb +115 -0
  102. data/lib/wsdl/soap/mappingRegistryCreator.rb +58 -0
  103. data/lib/wsdl/soap/mappingRegistryCreatorSupport.rb +376 -0
  104. data/lib/wsdl/soap/methodDefCreator.rb +263 -0
  105. data/lib/wsdl/soap/operation.rb +122 -0
  106. data/lib/wsdl/soap/servantSkeltonCreator.rb +80 -0
  107. data/lib/wsdl/soap/servletStubCreator.rb +107 -0
  108. data/lib/wsdl/soap/standaloneServerStubCreator.rb +103 -0
  109. data/lib/wsdl/soap/wsdl2ruby.rb +217 -0
  110. data/lib/wsdl/types.rb +44 -0
  111. data/lib/wsdl/wsdl.rb +24 -0
  112. data/lib/wsdl/xmlSchema/all.rb +24 -0
  113. data/lib/wsdl/xmlSchema/annotation.rb +34 -0
  114. data/lib/wsdl/xmlSchema/any.rb +61 -0
  115. data/lib/wsdl/xmlSchema/anyAttribute.rb +48 -0
  116. data/lib/wsdl/xmlSchema/attribute.rb +123 -0
  117. data/lib/wsdl/xmlSchema/attributeGroup.rb +86 -0
  118. data/lib/wsdl/xmlSchema/choice.rb +58 -0
  119. data/lib/wsdl/xmlSchema/complexContent.rb +97 -0
  120. data/lib/wsdl/xmlSchema/complexExtension.rb +119 -0
  121. data/lib/wsdl/xmlSchema/complexRestriction.rb +104 -0
  122. data/lib/wsdl/xmlSchema/complexType.rb +193 -0
  123. data/lib/wsdl/xmlSchema/content.rb +95 -0
  124. data/lib/wsdl/xmlSchema/data.rb +116 -0
  125. data/lib/wsdl/xmlSchema/element.rb +173 -0
  126. data/lib/wsdl/xmlSchema/enumeration.rb +36 -0
  127. data/lib/wsdl/xmlSchema/fractiondigits.rb +37 -0
  128. data/lib/wsdl/xmlSchema/group.rb +118 -0
  129. data/lib/wsdl/xmlSchema/import.rb +53 -0
  130. data/lib/wsdl/xmlSchema/importHandler.rb +45 -0
  131. data/lib/wsdl/xmlSchema/importer.rb +102 -0
  132. data/lib/wsdl/xmlSchema/include.rb +48 -0
  133. data/lib/wsdl/xmlSchema/length.rb +37 -0
  134. data/lib/wsdl/xmlSchema/list.rb +48 -0
  135. data/lib/wsdl/xmlSchema/maxexclusive.rb +37 -0
  136. data/lib/wsdl/xmlSchema/maxinclusive.rb +37 -0
  137. data/lib/wsdl/xmlSchema/maxlength.rb +37 -0
  138. data/lib/wsdl/xmlSchema/minexclusive.rb +37 -0
  139. data/lib/wsdl/xmlSchema/mininclusive.rb +37 -0
  140. data/lib/wsdl/xmlSchema/minlength.rb +37 -0
  141. data/lib/wsdl/xmlSchema/parser.rb +167 -0
  142. data/lib/wsdl/xmlSchema/pattern.rb +36 -0
  143. data/lib/wsdl/xmlSchema/schema.rb +178 -0
  144. data/lib/wsdl/xmlSchema/sequence.rb +54 -0
  145. data/lib/wsdl/xmlSchema/simpleContent.rb +69 -0
  146. data/lib/wsdl/xmlSchema/simpleExtension.rb +62 -0
  147. data/lib/wsdl/xmlSchema/simpleRestriction.rb +132 -0
  148. data/lib/wsdl/xmlSchema/simpleType.rb +87 -0
  149. data/lib/wsdl/xmlSchema/totaldigits.rb +37 -0
  150. data/lib/wsdl/xmlSchema/union.rb +35 -0
  151. data/lib/wsdl/xmlSchema/unique.rb +34 -0
  152. data/lib/wsdl/xmlSchema/whitespace.rb +37 -0
  153. data/lib/wsdl/xmlSchema/xsd2ruby.rb +174 -0
  154. data/lib/xsd/charset.rb +189 -0
  155. data/lib/xsd/codegen.rb +12 -0
  156. data/lib/xsd/codegen/classdef.rb +208 -0
  157. data/lib/xsd/codegen/commentdef.rb +34 -0
  158. data/lib/xsd/codegen/gensupport.rb +273 -0
  159. data/lib/xsd/codegen/methoddef.rb +70 -0
  160. data/lib/xsd/codegen/moduledef.rb +208 -0
  161. data/lib/xsd/datatypes.rb +1466 -0
  162. data/lib/xsd/datatypes1999.rb +20 -0
  163. data/lib/xsd/iconvcharset.rb +33 -0
  164. data/lib/xsd/mapping.rb +67 -0
  165. data/lib/xsd/namedelements.rb +132 -0
  166. data/lib/xsd/ns.rb +182 -0
  167. data/lib/xsd/qname.rb +79 -0
  168. data/lib/xsd/xmlparser.rb +75 -0
  169. data/lib/xsd/xmlparser/parser.rb +100 -0
  170. data/lib/xsd/xmlparser/rexmlparser.rb +58 -0
  171. data/lib/xsd/xmlparser/xmlparser.rb +50 -0
  172. data/lib/xsd/xmlparser/xmlscanner.rb +149 -0
  173. data/soap4r.gemspec +218 -0
  174. metadata +233 -0
@@ -0,0 +1,669 @@
1
+ # SOAP4R - RPC Routing library
2
+ # Copyright (C) 2000-2007 NAKAMURA, Hiroshi <nahi@ruby-lang.org>.
3
+
4
+ # This program is copyrighted free software by NAKAMURA, Hiroshi. You can
5
+ # redistribute it and/or modify it under the same terms of Ruby's license;
6
+ # either the dual license version in 2003, or any later version.
7
+
8
+
9
+ require 'soap/soap'
10
+ require 'soap/processor'
11
+ require 'soap/mapping'
12
+ require 'soap/mapping/literalregistry'
13
+ require 'soap/rpc/rpc'
14
+ require 'soap/rpc/element'
15
+ require 'soap/header/handlerset'
16
+ require 'soap/filter'
17
+ require 'soap/streamHandler'
18
+ require 'soap/mimemessage'
19
+ require 'soap/header/handlerset'
20
+
21
+
22
+ module SOAP
23
+ module RPC
24
+
25
+
26
+ class Router
27
+ include SOAP
28
+
29
+ attr_reader :actor
30
+ attr_accessor :mapping_registry
31
+ attr_accessor :literal_mapping_registry
32
+ attr_accessor :generate_explicit_type
33
+ attr_accessor :use_default_namespace
34
+ attr_accessor :external_ces
35
+ attr_reader :filterchain
36
+
37
+ def initialize(actor)
38
+ @actor = actor
39
+ @mapping_registry = nil
40
+ @headerhandler = Header::HandlerSet.new
41
+ @literal_mapping_registry = ::SOAP::Mapping::LiteralRegistry.new
42
+ @generate_explicit_type = true
43
+ @use_default_namespace = false
44
+ @external_ces = nil
45
+ @operation_by_soapaction = {}
46
+ @operation_by_qname = {}
47
+ @headerhandlerfactory = []
48
+ @filterchain = Filter::FilterChain.new
49
+ end
50
+
51
+ ###
52
+ ## header handler interface
53
+ #
54
+ def add_request_headerhandler(factory)
55
+ unless factory.respond_to?(:create)
56
+ raise TypeError.new("factory must respond to 'create'")
57
+ end
58
+ @headerhandlerfactory << factory
59
+ end
60
+
61
+ def add_headerhandler(handler)
62
+ @headerhandler.add(handler)
63
+ end
64
+
65
+ ###
66
+ ## servant definition interface
67
+ #
68
+ def add_rpc_request_servant(factory, namespace)
69
+ unless factory.respond_to?(:create)
70
+ raise TypeError.new("factory must respond to 'create'")
71
+ end
72
+ obj = factory.create # a dummy instance for introspection
73
+ ::SOAP::RPC.defined_methods(obj).each do |name|
74
+ begin
75
+ qname = XSD::QName.new(namespace, name)
76
+ param_def = ::SOAP::RPC::SOAPMethod.derive_rpc_param_def(obj, name)
77
+ opt = create_styleuse_option(:rpc, :encoded)
78
+ add_rpc_request_operation(factory, qname, nil, name, param_def, opt)
79
+ rescue SOAP::RPC::MethodDefinitionError => e
80
+ p e if $DEBUG
81
+ end
82
+ end
83
+ end
84
+
85
+ def add_rpc_servant(obj, namespace)
86
+ ::SOAP::RPC.defined_methods(obj).each do |name|
87
+ begin
88
+ qname = XSD::QName.new(namespace, name)
89
+ param_def = ::SOAP::RPC::SOAPMethod.derive_rpc_param_def(obj, name)
90
+ opt = create_styleuse_option(:rpc, :encoded)
91
+ add_rpc_operation(obj, qname, nil, name, param_def, opt)
92
+ rescue SOAP::RPC::MethodDefinitionError => e
93
+ p e if $DEBUG
94
+ end
95
+ end
96
+ end
97
+ alias add_servant add_rpc_servant
98
+
99
+ ###
100
+ ## operation definition interface
101
+ #
102
+ def add_rpc_operation(receiver, qname, soapaction, name, param_def, opt = {})
103
+ ensure_styleuse_option(opt, :rpc, :encoded)
104
+ opt[:request_qname] = qname
105
+ op = ApplicationScopeOperation.new(soapaction, receiver, name, param_def,
106
+ opt)
107
+ if opt[:request_style] != :rpc
108
+ raise RPCRoutingError.new("illegal request_style given")
109
+ end
110
+ assign_operation(soapaction, qname, op)
111
+ end
112
+ alias add_method add_rpc_operation
113
+ alias add_rpc_method add_rpc_operation
114
+
115
+ def add_rpc_request_operation(factory, qname, soapaction, name, param_def, opt = {})
116
+ ensure_styleuse_option(opt, :rpc, :encoded)
117
+ opt[:request_qname] = qname
118
+ op = RequestScopeOperation.new(soapaction, factory, name, param_def, opt)
119
+ if opt[:request_style] != :rpc
120
+ raise RPCRoutingError.new("illegal request_style given")
121
+ end
122
+ assign_operation(soapaction, qname, op)
123
+ end
124
+
125
+ def add_document_operation(receiver, soapaction, name, param_def, opt = {})
126
+ #
127
+ # adopt workaround for doc/lit wrapper method
128
+ # (you should consider to simply use rpc/lit service)
129
+ #
130
+ #unless soapaction
131
+ # raise RPCRoutingError.new("soapaction is a must for document method")
132
+ #end
133
+ ensure_styleuse_option(opt, :document, :literal)
134
+ op = ApplicationScopeOperation.new(soapaction, receiver, name, param_def,
135
+ opt)
136
+ if opt[:request_style] != :document
137
+ raise RPCRoutingError.new("illegal request_style given")
138
+ end
139
+ assign_operation(soapaction, first_input_part_qname(param_def), op)
140
+ end
141
+ alias add_document_method add_document_operation
142
+
143
+ def add_document_request_operation(factory, soapaction, name, param_def, opt = {})
144
+ #
145
+ # adopt workaround for doc/lit wrapper method
146
+ # (you should consider to simply use rpc/lit service)
147
+ #
148
+ #unless soapaction
149
+ # raise RPCRoutingError.new("soapaction is a must for document method")
150
+ #end
151
+ ensure_styleuse_option(opt, :document, :literal)
152
+ op = RequestScopeOperation.new(soapaction, receiver, name, param_def, opt)
153
+ if opt[:request_style] != :document
154
+ raise RPCRoutingError.new("illegal request_style given")
155
+ end
156
+ assign_operation(soapaction, first_input_part_qname(param_def), op)
157
+ end
158
+
159
+ def route(conn_data)
160
+ # we cannot set request_default_encodingsyle before parsing the content.
161
+ env = unmarshal(conn_data)
162
+ if env.nil?
163
+ raise ArgumentError.new("illegal SOAP marshal format")
164
+ end
165
+ op = lookup_operation(conn_data.soapaction, env.body)
166
+ headerhandler = @headerhandler.dup
167
+ @headerhandlerfactory.each do |f|
168
+ headerhandler.add(f.create)
169
+ end
170
+ soap_response = default_encodingstyle = nil
171
+ begin
172
+ receive_headers(headerhandler, env.header)
173
+ soap_response =
174
+ op.call(env.body, @mapping_registry, @literal_mapping_registry,
175
+ create_mapping_opt)
176
+ conn_data.is_fault = true if soap_response.is_a?(SOAPFault)
177
+ default_encodingstyle = op.response_default_encodingstyle
178
+ rescue Exception => e
179
+ # If a wsdl fault was raised by service, the fault declaration details
180
+ # is kept in wsdl_fault. Otherwise (exception is a program fault)
181
+ # wsdl_fault is nil
182
+ wsdl_fault_details = op.faults && op.faults[e.class.name]
183
+ soap_response = fault(e, wsdl_fault_details)
184
+ conn_data.is_fault = true
185
+ default_encodingstyle = nil
186
+ end
187
+ header = call_headers(headerhandler)
188
+ if op.response_use.nil?
189
+ conn_data.send_string = ''
190
+ conn_data.is_nocontent = true
191
+ conn_data
192
+ else
193
+ body = SOAPBody.new(soap_response, conn_data.is_fault)
194
+ env = SOAPEnvelope.new(header, body)
195
+ marshal(conn_data, env, default_encodingstyle)
196
+ end
197
+ end
198
+
199
+ # Create fault response string.
200
+ def create_fault_response(e)
201
+ env = SOAPEnvelope.new(SOAPHeader.new, SOAPBody.new(fault(e, nil), true))
202
+ opt = {}
203
+ opt[:external_content] = nil
204
+ @filterchain.reverse_each do |filter|
205
+ env = filter.on_outbound(env, opt)
206
+ break unless env
207
+ end
208
+ response_string = Processor.marshal(env, opt)
209
+ conn_data = StreamHandler::ConnectionData.new(response_string)
210
+ conn_data.is_fault = true
211
+ if ext = opt[:external_content]
212
+ mimeize(conn_data, ext)
213
+ end
214
+ conn_data
215
+ end
216
+
217
+ private
218
+
219
+ def first_input_part_qname(param_def)
220
+ param_def.each do |inout, paramname, typeinfo|
221
+ if inout == SOAPMethod::IN
222
+ klass, nsdef, namedef = typeinfo
223
+ return XSD::QName.new(nsdef, namedef)
224
+ end
225
+ end
226
+ nil
227
+ end
228
+
229
+ def create_styleuse_option(style, use)
230
+ opt = {}
231
+ opt[:request_style] = opt[:response_style] = style
232
+ opt[:request_use] = opt[:response_use] = use
233
+ opt
234
+ end
235
+
236
+ def ensure_styleuse_option(opt, style, use)
237
+ if opt[:request_style] || opt[:response_style] || opt[:request_use] || opt[:response_use]
238
+ # do not edit
239
+ else
240
+ opt[:request_style] ||= style
241
+ opt[:response_style] ||= style
242
+ opt[:request_use] ||= use
243
+ opt[:response_use] ||= use
244
+ end
245
+ end
246
+
247
+ def assign_operation(soapaction, qname, op)
248
+ assigned = false
249
+ if soapaction and !soapaction.empty?
250
+ @operation_by_soapaction[soapaction] = op
251
+ assigned = true
252
+ end
253
+ if qname
254
+ @operation_by_qname[qname] = op
255
+ assigned = true
256
+ end
257
+ unless assigned
258
+ raise RPCRoutingError.new("cannot assign operation")
259
+ end
260
+ end
261
+
262
+ def lookup_operation(soapaction, body)
263
+ if op = @operation_by_soapaction[soapaction]
264
+ return op
265
+ end
266
+ qname = body.root_node.elename
267
+ if op = @operation_by_qname[qname]
268
+ return op
269
+ end
270
+ if soapaction
271
+ raise RPCRoutingError.new(
272
+ "operation: #{soapaction} #{qname} not supported")
273
+ else
274
+ raise RPCRoutingError.new("operation: #{qname} not supported")
275
+ end
276
+ end
277
+
278
+ def call_headers(headerhandler)
279
+ header = ::SOAP::SOAPHeader.new
280
+ items = headerhandler.on_outbound(header)
281
+ items.each do |item|
282
+ header.add(item.elename.name, item)
283
+ end
284
+ header
285
+ end
286
+
287
+ def receive_headers(headerhandler, header)
288
+ headerhandler.on_inbound(header) if header
289
+ end
290
+
291
+ def unmarshal(conn_data)
292
+ xml = nil
293
+ opt = {}
294
+ contenttype = conn_data.receive_contenttype
295
+ if /#{MIMEMessage::MultipartContentType}/i =~ contenttype
296
+ opt[:external_content] = {}
297
+ mime = MIMEMessage.parse("Content-Type: " + contenttype,
298
+ conn_data.receive_string)
299
+ mime.parts.each do |part|
300
+ value = Attachment.new(part.content)
301
+ value.contentid = part.contentid
302
+ obj = SOAPAttachment.new(value)
303
+ opt[:external_content][value.contentid] = obj if value.contentid
304
+ end
305
+ opt[:charset] =
306
+ StreamHandler.parse_media_type(mime.root.headers['content-type'].str)
307
+ xml = mime.root.content
308
+ else
309
+ opt[:charset] = ::SOAP::StreamHandler.parse_media_type(contenttype)
310
+ xml = conn_data.receive_string
311
+ end
312
+ @filterchain.each do |filter|
313
+ xml = filter.on_inbound(xml, opt)
314
+ break unless xml
315
+ end
316
+ env = Processor.unmarshal(xml, opt)
317
+ charset = opt[:charset]
318
+ conn_data.send_contenttype = "text/xml; charset=\"#{charset}\""
319
+ env
320
+ end
321
+
322
+ def marshal(conn_data, env, default_encodingstyle = nil)
323
+ opt = {}
324
+ opt[:external_content] = nil
325
+ opt[:default_encodingstyle] = default_encodingstyle
326
+ opt[:generate_explicit_type] = @generate_explicit_type
327
+ opt[:use_default_namespace] = @use_default_namespace
328
+ @filterchain.reverse_each do |filter|
329
+ env = filter.on_outbound(env, opt)
330
+ break unless env
331
+ end
332
+ response_string = Processor.marshal(env, opt)
333
+ conn_data.send_string = response_string
334
+ if ext = opt[:external_content]
335
+ mimeize(conn_data, ext)
336
+ end
337
+ conn_data
338
+ end
339
+
340
+ def mimeize(conn_data, ext)
341
+ mime = MIMEMessage.new
342
+ ext.each do |k, v|
343
+ mime.add_attachment(v.data)
344
+ end
345
+ mime.add_part(conn_data.send_string + "\r\n")
346
+ mime.close
347
+ conn_data.send_string = mime.content_str
348
+ conn_data.send_contenttype = mime.headers['content-type'].str
349
+ conn_data
350
+ end
351
+
352
+ # Create fault response.
353
+ def fault(e, wsdl_fault_details)
354
+ if e.is_a?(UnhandledMustUnderstandHeaderError)
355
+ faultcode = FaultCode::MustUnderstand
356
+ else
357
+ faultcode = FaultCode::Server
358
+ end
359
+
360
+ # If the exception represents a WSDL fault, the fault element should
361
+ # be added as the SOAP fault <detail> element. If the exception is a
362
+ # normal program exception, it is wrapped inside a custom SOAP4R
363
+ # SOAP exception element.
364
+ detail = nil
365
+ begin
366
+ if (wsdl_fault_details)
367
+ registry = wsdl_fault_details[:use] == "literal" ?
368
+ @literal_mapping_registry : @mapping_registry
369
+ faultQName = XSD::QName.new(
370
+ wsdl_fault_details[:ns], wsdl_fault_details[:name]
371
+ )
372
+ detail = Mapping.obj2soap(e, registry, faultQName)
373
+ # wrap fault element (SOAPFault swallows top-level element)
374
+ wrapper = SOAP::SOAPElement.new(faultQName)
375
+ wrapper.add(detail)
376
+ detail = wrapper
377
+ else
378
+ # Exception is a normal program exception. Wrap it.
379
+ detail = Mapping.obj2soap(Mapping::SOAPException.new(e),
380
+ @mapping_registry)
381
+ detail.elename ||= XSD::QName::EMPTY # for literal mappingregstry
382
+ end
383
+ rescue
384
+ detail = SOAPString.new("failed to serialize detail object: #{$!}")
385
+ end
386
+
387
+ SOAPFault.new(
388
+ SOAPElement.new(nil, faultcode),
389
+ SOAPString.new(e.to_s),
390
+ SOAPString.new(@actor),
391
+ detail)
392
+ end
393
+
394
+ def create_mapping_opt
395
+ { :external_ces => @external_ces }
396
+ end
397
+
398
+ class Operation
399
+ attr_reader :name
400
+ attr_reader :soapaction
401
+ attr_reader :request_style
402
+ attr_reader :response_style
403
+ attr_reader :request_use
404
+ attr_reader :response_use
405
+ attr_reader :faults
406
+
407
+ def initialize(soapaction, name, param_def, opt)
408
+ @soapaction = soapaction
409
+ @name = name
410
+ @request_style = opt[:request_style]
411
+ @response_style = opt[:response_style]
412
+ @request_use = opt[:request_use]
413
+ @response_use = opt[:response_use]
414
+ @faults = opt[:faults]
415
+ check_style(@request_style)
416
+ check_style(@response_style)
417
+ check_use(@request_use)
418
+ check_use(@response_use)
419
+ if @response_style == :rpc
420
+ request_qname = opt[:request_qname] or raise
421
+ @rpc_method_factory =
422
+ RPC::SOAPMethodRequest.new(request_qname, param_def, @soapaction)
423
+ @rpc_response_qname = opt[:response_qname]
424
+ else
425
+ @doc_request_qnames = []
426
+ @doc_request_qualified = []
427
+ @doc_response_qnames = []
428
+ @doc_response_qualified = []
429
+ param_def.each do |inout, paramname, typeinfo, eleinfo|
430
+ klass, nsdef, namedef = typeinfo
431
+ qualified = eleinfo
432
+ case inout
433
+ when SOAPMethod::IN
434
+ @doc_request_qnames << XSD::QName.new(nsdef, namedef)
435
+ @doc_request_qualified << qualified
436
+ when SOAPMethod::OUT
437
+ @doc_response_qnames << XSD::QName.new(nsdef, namedef)
438
+ @doc_response_qualified << qualified
439
+ else
440
+ raise ArgumentError.new(
441
+ "illegal inout definition for document style: #{inout}")
442
+ end
443
+ end
444
+ end
445
+ end
446
+
447
+ def request_default_encodingstyle
448
+ (@request_use == :encoded) ? EncodingNamespace : LiteralNamespace
449
+ end
450
+
451
+ def response_default_encodingstyle
452
+ (@response_use == :encoded) ? EncodingNamespace : LiteralNamespace
453
+ end
454
+
455
+ def call(body, mapping_registry, literal_mapping_registry, opt)
456
+ if @request_style == :rpc
457
+ values = request_rpc(body, mapping_registry, literal_mapping_registry,
458
+ opt)
459
+ else
460
+ values = request_document(body, mapping_registry,
461
+ literal_mapping_registry, opt)
462
+ end
463
+ result = receiver.method(@name.intern).call(*values)
464
+ return result if result.is_a?(SOAPFault)
465
+ if @response_style == :rpc
466
+ response_rpc(result, mapping_registry, literal_mapping_registry, opt)
467
+ elsif @doc_response_qnames.empty?
468
+ # nothing to do
469
+ else
470
+ response_doc(result, mapping_registry, literal_mapping_registry, opt)
471
+ end
472
+ end
473
+
474
+ private
475
+
476
+ def receiver
477
+ raise NotImplementedError.new('must be defined in derived class')
478
+ end
479
+
480
+ def request_rpc(body, mapping_registry, literal_mapping_registry, opt)
481
+ request = body.request
482
+ unless request.is_a?(SOAPNameAccessible)
483
+ if request.is_a?(SOAPNil)
484
+ # SOAP::Lite/0.69 seems to send xsi:nil="true" element as a request.
485
+ request = SOAPStruct.new(request.elename)
486
+ else
487
+ raise RPCRoutingError.new("not an RPC style")
488
+ end
489
+ end
490
+ if @request_use == :encoded
491
+ request_rpc_enc(request, mapping_registry, opt)
492
+ else
493
+ request_rpc_lit(request, literal_mapping_registry, opt)
494
+ end
495
+ end
496
+
497
+ def request_document(body, mapping_registry, literal_mapping_registry, opt)
498
+ # ToDo: compare names with @doc_request_qnames
499
+ if @request_use == :encoded
500
+ request_doc_enc(body, mapping_registry, opt)
501
+ else
502
+ request_doc_lit(body, literal_mapping_registry, opt)
503
+ end
504
+ end
505
+
506
+ def request_rpc_enc(request, mapping_registry, opt)
507
+ param = Mapping.soap2obj(request, mapping_registry, nil, opt)
508
+ request.collect { |key, value|
509
+ param[key]
510
+ }
511
+ end
512
+
513
+ def request_rpc_lit(request, mapping_registry, opt)
514
+ request.collect { |key, value|
515
+ Mapping.soap2obj(value, mapping_registry, nil, opt)
516
+ }
517
+ end
518
+
519
+ def request_doc_enc(body, mapping_registry, opt)
520
+ body.collect { |key, value|
521
+ Mapping.soap2obj(value, mapping_registry, nil, opt)
522
+ }
523
+ end
524
+
525
+ def request_doc_lit(body, mapping_registry, opt)
526
+ body.collect { |key, value|
527
+ Mapping.soap2obj(value, mapping_registry, nil, opt)
528
+ }
529
+ end
530
+
531
+ def response_rpc(result, mapping_registry, literal_mapping_registry, opt)
532
+ if @response_use == :encoded
533
+ response_rpc_enc(result, mapping_registry, opt)
534
+ else
535
+ response_rpc_lit(result, literal_mapping_registry, opt)
536
+ end
537
+ end
538
+
539
+ def response_doc(result, mapping_registry, literal_mapping_registry, opt)
540
+ if @doc_response_qnames.size == 0
541
+ result = []
542
+ elsif @doc_response_qnames.size == 1
543
+ result = [result]
544
+ end
545
+ if result.size != @doc_response_qnames.size
546
+ raise "required #{@doc_response_qnames.size} responses " +
547
+ "but #{result.size} given"
548
+ end
549
+ if @response_use == :encoded
550
+ response_doc_enc(result, mapping_registry, opt)
551
+ else
552
+ response_doc_lit(result, literal_mapping_registry, opt)
553
+ end
554
+ end
555
+
556
+ def response_rpc_enc(result, mapping_registry, opt)
557
+ soap_response =
558
+ @rpc_method_factory.create_method_response(@rpc_response_qname)
559
+ if soap_response.have_outparam?
560
+ unless result.is_a?(Array)
561
+ raise RPCRoutingError.new("out parameter was not returned")
562
+ end
563
+ outparams = {}
564
+ i = 1
565
+ soap_response.output_params.each do |outparam|
566
+ outparams[outparam] = Mapping.obj2soap(result[i], mapping_registry,
567
+ nil, opt)
568
+ i += 1
569
+ end
570
+ soap_response.set_outparam(outparams)
571
+ soap_response.retval = Mapping.obj2soap(result[0], mapping_registry,
572
+ nil, opt)
573
+ else
574
+ soap_response.retval = Mapping.obj2soap(result, mapping_registry, nil,
575
+ opt)
576
+ end
577
+ soap_response
578
+ end
579
+
580
+ def response_rpc_lit(result, mapping_registry, opt)
581
+ soap_response =
582
+ @rpc_method_factory.create_method_response(@rpc_response_qname)
583
+ if soap_response.have_outparam?
584
+ unless result.is_a?(Array)
585
+ raise RPCRoutingError.new("out parameter was not returned")
586
+ end
587
+ outparams = {}
588
+ i = 1
589
+ soap_response.output_params.each do |outparam|
590
+ outparams[outparam] = Mapping.obj2soap(result[i], mapping_registry,
591
+ XSD::QName.new(nil, outparam), opt)
592
+ i += 1
593
+ end
594
+ soap_response.set_outparam(outparams)
595
+ soap_response.retval = Mapping.obj2soap(result[0], mapping_registry,
596
+ soap_response.elename, opt)
597
+ else
598
+ soap_response.retval = Mapping.obj2soap(result, mapping_registry,
599
+ soap_response.elename, opt)
600
+ end
601
+ soap_response
602
+ end
603
+
604
+ def response_doc_enc(result, mapping_registry, opt)
605
+ (0...result.size).collect { |idx|
606
+ ele = Mapping.obj2soap(result[idx], mapping_registry, nil, opt)
607
+ ele.elename = @doc_response_qnames[idx]
608
+ ele.qualified = @doc_response_qualified[idx]
609
+ ele
610
+ }
611
+ end
612
+
613
+ def response_doc_lit(result, mapping_registry, opt)
614
+ (0...result.size).collect { |idx|
615
+ ele = Mapping.obj2soap(result[idx], mapping_registry,
616
+ @doc_response_qnames[idx])
617
+ ele.encodingstyle = LiteralNamespace
618
+ ele.qualified = @doc_response_qualified[idx]
619
+ ele
620
+ }
621
+ end
622
+
623
+ def check_style(style)
624
+ unless [:rpc, :document].include?(style)
625
+ raise ArgumentError.new("unknown style: #{style}")
626
+ end
627
+ end
628
+
629
+ # nil means oneway
630
+ def check_use(use)
631
+ unless [:encoded, :literal, nil].include?(use)
632
+ raise ArgumentError.new("unknown use: #{use}")
633
+ end
634
+ end
635
+ end
636
+
637
+ class ApplicationScopeOperation < Operation
638
+ def initialize(soapaction, receiver, name, param_def, opt)
639
+ super(soapaction, name, param_def, opt)
640
+ @receiver = receiver
641
+ end
642
+
643
+ private
644
+
645
+ def receiver
646
+ @receiver
647
+ end
648
+ end
649
+
650
+ class RequestScopeOperation < Operation
651
+ def initialize(soapaction, receiver_factory, name, param_def, opt)
652
+ super(soapaction, name, param_def, opt)
653
+ unless receiver_factory.respond_to?(:create)
654
+ raise TypeError.new("factory must respond to 'create'")
655
+ end
656
+ @receiver_factory = receiver_factory
657
+ end
658
+
659
+ private
660
+
661
+ def receiver
662
+ @receiver_factory.create
663
+ end
664
+ end
665
+ end
666
+
667
+
668
+ end
669
+ end