soap2r 1.5.8
Sign up to get free protection for your applications and to get access to all the features.
- data/bin/wsdl2ruby.rb +137 -0
- data/bin/xsd2ruby.rb +90 -0
- data/lib/soap/attachment.rb +108 -0
- data/lib/soap/attrproxy.rb +34 -0
- data/lib/soap/baseData.rb +1094 -0
- data/lib/soap/element.rb +277 -0
- data/lib/soap/encodingstyle/aspDotNetHandler.rb +207 -0
- data/lib/soap/encodingstyle/handler.rb +120 -0
- data/lib/soap/encodingstyle/literalHandler.rb +195 -0
- data/lib/soap/encodingstyle/soapHandler.rb +559 -0
- data/lib/soap/filter.rb +13 -0
- data/lib/soap/filter/filterchain.rb +51 -0
- data/lib/soap/filter/handler.rb +31 -0
- data/lib/soap/filter/streamhandler.rb +29 -0
- data/lib/soap/generator.rb +299 -0
- data/lib/soap/header/handler.rb +61 -0
- data/lib/soap/header/handlerset.rb +70 -0
- data/lib/soap/header/mappinghandler.rb +47 -0
- data/lib/soap/header/simplehandler.rb +44 -0
- data/lib/soap/httpconfigloader.rb +141 -0
- data/lib/soap/mapping.rb +12 -0
- data/lib/soap/mapping/encodedregistry.rb +537 -0
- data/lib/soap/mapping/factory.rb +388 -0
- data/lib/soap/mapping/literalregistry.rb +391 -0
- data/lib/soap/mapping/mapping.rb +576 -0
- data/lib/soap/mapping/registry.rb +295 -0
- data/lib/soap/mapping/rubytypeFactory.rb +446 -0
- data/lib/soap/mapping/schemadefinition.rb +170 -0
- data/lib/soap/mapping/typeMap.rb +106 -0
- data/lib/soap/mapping/wsdlencodedregistry.rb +211 -0
- data/lib/soap/mapping/wsdlliteralregistry.rb +248 -0
- data/lib/soap/marshal.rb +59 -0
- data/lib/soap/mimemessage.rb +241 -0
- data/lib/soap/nestedexception.rb +42 -0
- data/lib/soap/netHttpClient.rb +241 -0
- data/lib/soap/ns.rb +34 -0
- data/lib/soap/parser.rb +252 -0
- data/lib/soap/processor.rb +66 -0
- data/lib/soap/property.rb +319 -0
- data/lib/soap/proxy.rb +14 -0
- data/lib/soap/rpc/cgistub.rb +247 -0
- data/lib/soap/rpc/driver.rb +221 -0
- data/lib/soap/rpc/element.rb +374 -0
- data/lib/soap/rpc/httpserver.rb +142 -0
- data/lib/soap/rpc/methodDef.rb +68 -0
- data/lib/soap/rpc/proxy.rb +572 -0
- data/lib/soap/rpc/router.rb +662 -0
- data/lib/soap/rpc/rpc.rb +25 -0
- data/lib/soap/rpc/soaplet.rb +200 -0
- data/lib/soap/rpc/standaloneServer.rb +43 -0
- data/lib/soap/soap.rb +151 -0
- data/lib/soap/streamHandler.rb +301 -0
- data/lib/soap/wsdlDriver.rb +164 -0
- data/lib/wsdl/binding.rb +65 -0
- data/lib/wsdl/data.rb +64 -0
- data/lib/wsdl/definitions.rb +236 -0
- data/lib/wsdl/documentation.rb +32 -0
- data/lib/wsdl/import.rb +80 -0
- data/lib/wsdl/importer.rb +38 -0
- data/lib/wsdl/info.rb +50 -0
- data/lib/wsdl/message.rb +54 -0
- data/lib/wsdl/operation.rb +151 -0
- data/lib/wsdl/operationBinding.rb +240 -0
- data/lib/wsdl/param.rb +93 -0
- data/lib/wsdl/parser.rb +164 -0
- data/lib/wsdl/part.rb +52 -0
- data/lib/wsdl/port.rb +66 -0
- data/lib/wsdl/portType.rb +75 -0
- data/lib/wsdl/service.rb +61 -0
- data/lib/wsdl/soap/address.rb +40 -0
- data/lib/wsdl/soap/binding.rb +49 -0
- data/lib/wsdl/soap/body.rb +58 -0
- data/lib/wsdl/soap/cgiStubCreator.rb +92 -0
- data/lib/wsdl/soap/classDefCreator.rb +433 -0
- data/lib/wsdl/soap/classDefCreatorSupport.rb +240 -0
- data/lib/wsdl/soap/classNameCreator.rb +54 -0
- data/lib/wsdl/soap/clientSkeltonCreator.rb +104 -0
- data/lib/wsdl/soap/complexType.rb +173 -0
- data/lib/wsdl/soap/data.rb +42 -0
- data/lib/wsdl/soap/definitions.rb +200 -0
- data/lib/wsdl/soap/driverCreator.rb +118 -0
- data/lib/wsdl/soap/element.rb +33 -0
- data/lib/wsdl/soap/encodedMappingRegistryCreator.rb +73 -0
- data/lib/wsdl/soap/fault.rb +56 -0
- data/lib/wsdl/soap/header.rb +86 -0
- data/lib/wsdl/soap/headerfault.rb +56 -0
- data/lib/wsdl/soap/literalMappingRegistryCreator.rb +115 -0
- data/lib/wsdl/soap/mappingRegistryCreator.rb +58 -0
- data/lib/wsdl/soap/mappingRegistryCreatorSupport.rb +376 -0
- data/lib/wsdl/soap/methodDefCreator.rb +199 -0
- data/lib/wsdl/soap/operation.rb +112 -0
- data/lib/wsdl/soap/servantSkeltonCreator.rb +89 -0
- data/lib/wsdl/soap/servletStubCreator.rb +104 -0
- data/lib/wsdl/soap/standaloneServerStubCreator.rb +100 -0
- data/lib/wsdl/soap/wsdl2ruby.rb +217 -0
- data/lib/wsdl/types.rb +44 -0
- data/lib/wsdl/wsdl.rb +24 -0
- data/lib/wsdl/xmlSchema/all.rb +24 -0
- data/lib/wsdl/xmlSchema/annotation.rb +34 -0
- data/lib/wsdl/xmlSchema/any.rb +61 -0
- data/lib/wsdl/xmlSchema/anyAttribute.rb +48 -0
- data/lib/wsdl/xmlSchema/attribute.rb +104 -0
- data/lib/wsdl/xmlSchema/attributeGroup.rb +68 -0
- data/lib/wsdl/xmlSchema/choice.rb +58 -0
- data/lib/wsdl/xmlSchema/complexContent.rb +97 -0
- data/lib/wsdl/xmlSchema/complexExtension.rb +119 -0
- data/lib/wsdl/xmlSchema/complexRestriction.rb +104 -0
- data/lib/wsdl/xmlSchema/complexType.rb +193 -0
- data/lib/wsdl/xmlSchema/content.rb +95 -0
- data/lib/wsdl/xmlSchema/data.rb +116 -0
- data/lib/wsdl/xmlSchema/element.rb +153 -0
- data/lib/wsdl/xmlSchema/enumeration.rb +36 -0
- data/lib/wsdl/xmlSchema/fractiondigits.rb +37 -0
- data/lib/wsdl/xmlSchema/group.rb +100 -0
- data/lib/wsdl/xmlSchema/import.rb +53 -0
- data/lib/wsdl/xmlSchema/importHandler.rb +45 -0
- data/lib/wsdl/xmlSchema/importer.rb +102 -0
- data/lib/wsdl/xmlSchema/include.rb +48 -0
- data/lib/wsdl/xmlSchema/length.rb +37 -0
- data/lib/wsdl/xmlSchema/list.rb +48 -0
- data/lib/wsdl/xmlSchema/maxexclusive.rb +37 -0
- data/lib/wsdl/xmlSchema/maxinclusive.rb +37 -0
- data/lib/wsdl/xmlSchema/maxlength.rb +37 -0
- data/lib/wsdl/xmlSchema/minexclusive.rb +37 -0
- data/lib/wsdl/xmlSchema/mininclusive.rb +37 -0
- data/lib/wsdl/xmlSchema/minlength.rb +37 -0
- data/lib/wsdl/xmlSchema/parser.rb +167 -0
- data/lib/wsdl/xmlSchema/pattern.rb +36 -0
- data/lib/wsdl/xmlSchema/ref.rb +33 -0
- data/lib/wsdl/xmlSchema/schema.rb +178 -0
- data/lib/wsdl/xmlSchema/sequence.rb +54 -0
- data/lib/wsdl/xmlSchema/simpleContent.rb +69 -0
- data/lib/wsdl/xmlSchema/simpleExtension.rb +62 -0
- data/lib/wsdl/xmlSchema/simpleRestriction.rb +132 -0
- data/lib/wsdl/xmlSchema/simpleType.rb +87 -0
- data/lib/wsdl/xmlSchema/totaldigits.rb +37 -0
- data/lib/wsdl/xmlSchema/union.rb +35 -0
- data/lib/wsdl/xmlSchema/unique.rb +34 -0
- data/lib/wsdl/xmlSchema/whitespace.rb +37 -0
- data/lib/wsdl/xmlSchema/xsd2ruby.rb +174 -0
- data/lib/xsd/charset.rb +193 -0
- data/lib/xsd/codegen.rb +12 -0
- data/lib/xsd/codegen/classdef.rb +208 -0
- data/lib/xsd/codegen/commentdef.rb +34 -0
- data/lib/xsd/codegen/gensupport.rb +273 -0
- data/lib/xsd/codegen/methoddef.rb +70 -0
- data/lib/xsd/codegen/moduledef.rb +208 -0
- data/lib/xsd/datatypes.rb +1466 -0
- data/lib/xsd/datatypes1999.rb +20 -0
- data/lib/xsd/iconvcharset.rb +39 -0
- data/lib/xsd/mapping.rb +68 -0
- data/lib/xsd/namedelements.rb +132 -0
- data/lib/xsd/ns.rb +182 -0
- data/lib/xsd/qname.rb +79 -0
- data/lib/xsd/xmlparser.rb +76 -0
- data/lib/xsd/xmlparser/libxmlparser.rb +115 -0
- data/lib/xsd/xmlparser/parser.rb +100 -0
- data/lib/xsd/xmlparser/rexmlparser.rb +58 -0
- data/lib/xsd/xmlparser/xmlparser.rb +50 -0
- data/lib/xsd/xmlparser/xmlscanner.rb +149 -0
- metadata +224 -0
@@ -0,0 +1,248 @@
|
|
1
|
+
# SOAP4R - WSDL literal mapping registry.
|
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/baseData'
|
10
|
+
require 'soap/mapping/mapping'
|
11
|
+
require 'soap/mapping/literalregistry'
|
12
|
+
require 'soap/mapping/typeMap'
|
13
|
+
require 'xsd/codegen/gensupport'
|
14
|
+
require 'xsd/namedelements'
|
15
|
+
|
16
|
+
|
17
|
+
module SOAP
|
18
|
+
module Mapping
|
19
|
+
|
20
|
+
|
21
|
+
class WSDLLiteralRegistry < LiteralRegistry
|
22
|
+
attr_reader :definedelements
|
23
|
+
attr_reader :definedtypes
|
24
|
+
|
25
|
+
def initialize(definedtypes = XSD::NamedElements::Empty,
|
26
|
+
definedelements = XSD::NamedElements::Empty)
|
27
|
+
super()
|
28
|
+
@definedtypes = definedtypes
|
29
|
+
@definedelements = definedelements
|
30
|
+
end
|
31
|
+
|
32
|
+
def obj2soap(obj, qname, obj_class = nil)
|
33
|
+
soap_obj = nil
|
34
|
+
if obj.is_a?(SOAPElement)
|
35
|
+
soap_obj = obj
|
36
|
+
elsif eledef = @definedelements[qname]
|
37
|
+
soap_obj = obj2elesoap(obj, eledef)
|
38
|
+
elsif type = @definedtypes[qname]
|
39
|
+
soap_obj = obj2typesoap(obj, type)
|
40
|
+
else
|
41
|
+
soap_obj = any2soap(obj, qname, obj_class)
|
42
|
+
end
|
43
|
+
return soap_obj if soap_obj
|
44
|
+
if @excn_handler_obj2soap
|
45
|
+
soap_obj = @excn_handler_obj2soap.call(obj) { |yield_obj|
|
46
|
+
Mapping.obj2soap(yield_obj, nil, nil, MAPPING_OPT)
|
47
|
+
}
|
48
|
+
return soap_obj if soap_obj
|
49
|
+
end
|
50
|
+
raise MappingError.new("cannot map #{obj.class.name} as #{qname}")
|
51
|
+
end
|
52
|
+
|
53
|
+
# node should be a SOAPElement
|
54
|
+
def soap2obj(node, obj_class = nil)
|
55
|
+
cause = nil
|
56
|
+
begin
|
57
|
+
return any2obj(node, obj_class)
|
58
|
+
rescue MappingError
|
59
|
+
cause = $!
|
60
|
+
end
|
61
|
+
if @excn_handler_soap2obj
|
62
|
+
begin
|
63
|
+
return @excn_handler_soap2obj.call(node) { |yield_node|
|
64
|
+
Mapping.soap2obj(yield_node, nil, nil, MAPPING_OPT)
|
65
|
+
}
|
66
|
+
rescue Exception
|
67
|
+
end
|
68
|
+
end
|
69
|
+
if node.respond_to?(:type)
|
70
|
+
raise MappingError.new("cannot map #{node.type.name} to Ruby object", cause)
|
71
|
+
else
|
72
|
+
raise MappingError.new("cannot map #{node.elename.name} to Ruby object", cause)
|
73
|
+
end
|
74
|
+
end
|
75
|
+
|
76
|
+
private
|
77
|
+
|
78
|
+
def obj2elesoap(obj, eledef)
|
79
|
+
ele = nil
|
80
|
+
qualified = (eledef.elementform == 'qualified')
|
81
|
+
if obj.is_a?(SOAPNil)
|
82
|
+
ele = obj
|
83
|
+
elsif eledef.type
|
84
|
+
if type = @definedtypes[eledef.type]
|
85
|
+
ele = obj2typesoap(obj, type)
|
86
|
+
elsif type = TypeMap[eledef.type]
|
87
|
+
ele = base2soap(obj, type)
|
88
|
+
else
|
89
|
+
raise MappingError.new("cannot find type #{eledef.type}")
|
90
|
+
end
|
91
|
+
elsif eledef.local_complextype
|
92
|
+
ele = obj2typesoap(obj, eledef.local_complextype)
|
93
|
+
elsif eledef.local_simpletype
|
94
|
+
ele = obj2typesoap(obj, eledef.local_simpletype)
|
95
|
+
else
|
96
|
+
raise MappingError.new('illegal schema?')
|
97
|
+
end
|
98
|
+
ele.elename = eledef.name
|
99
|
+
ele.qualified = qualified
|
100
|
+
ele
|
101
|
+
end
|
102
|
+
|
103
|
+
def obj2typesoap(obj, type)
|
104
|
+
ele = nil
|
105
|
+
if type.is_a?(::WSDL::XMLSchema::SimpleType)
|
106
|
+
ele = simpleobj2soap(obj, type)
|
107
|
+
else # complexType
|
108
|
+
if type.simplecontent
|
109
|
+
ele = simpleobj2soap(obj, type.simplecontent)
|
110
|
+
else
|
111
|
+
ele = complexobj2soap(obj, type)
|
112
|
+
end
|
113
|
+
ele.type = type.name
|
114
|
+
if type.base or Mapping.root_type_hint
|
115
|
+
Mapping.reset_root_type_hint
|
116
|
+
ele.force_typed = true
|
117
|
+
end
|
118
|
+
add_definedattributes2soap(obj, ele, type)
|
119
|
+
end
|
120
|
+
ele
|
121
|
+
end
|
122
|
+
|
123
|
+
def simpleobj2soap(obj, type)
|
124
|
+
type.check_lexical_format(obj)
|
125
|
+
return SOAPNil.new if obj.nil?
|
126
|
+
if type.base
|
127
|
+
ele = base2soap(obj, TypeMap[type.base])
|
128
|
+
elsif type.list
|
129
|
+
value = obj.is_a?(Array) ? obj.join(" ") : obj.to_s
|
130
|
+
ele = base2soap(value, SOAP::SOAPString)
|
131
|
+
else
|
132
|
+
raise MappingError.new("unsupported simpleType: #{type}")
|
133
|
+
end
|
134
|
+
ele
|
135
|
+
end
|
136
|
+
|
137
|
+
def complexobj2soap(obj, type)
|
138
|
+
ele = SOAPElement.new(type.name)
|
139
|
+
complexobj2sequencesoap(obj, ele, type, type.choice?, type.choice?)
|
140
|
+
ele
|
141
|
+
end
|
142
|
+
|
143
|
+
def complexobj2sequencesoap(obj, soap, type, nillable, is_choice)
|
144
|
+
added = false
|
145
|
+
type.elements.each do |child_ele|
|
146
|
+
case child_ele
|
147
|
+
when WSDL::XMLSchema::Any
|
148
|
+
any = Mapping.get_attributes_for_any(obj)
|
149
|
+
SOAPElement.from_objs(any).each do |child|
|
150
|
+
soap.add(child)
|
151
|
+
end
|
152
|
+
ele_added = true
|
153
|
+
when WSDL::XMLSchema::Element
|
154
|
+
ele_added = complexobj2soapchildren(obj, soap, child_ele, nillable)
|
155
|
+
when WSDL::XMLSchema::Sequence
|
156
|
+
ele_added = complexobj2sequencesoap(obj, soap, child_ele, nillable, false)
|
157
|
+
when WSDL::XMLSchema::Choice
|
158
|
+
ele_added = complexobj2sequencesoap(obj, soap, child_ele, true, true)
|
159
|
+
else
|
160
|
+
raise MappingError.new("unknown type: #{child_ele}")
|
161
|
+
end
|
162
|
+
added = true if ele_added
|
163
|
+
break if is_choice and ele_added
|
164
|
+
end
|
165
|
+
added
|
166
|
+
end
|
167
|
+
|
168
|
+
def complexobj2soapchildren(obj, soap, child_ele, nillable = false)
|
169
|
+
if child_ele.map_as_array?
|
170
|
+
complexobj2soapchildren_array(obj, soap, child_ele, nillable)
|
171
|
+
else
|
172
|
+
complexobj2soapchildren_single(obj, soap, child_ele, nillable)
|
173
|
+
end
|
174
|
+
end
|
175
|
+
|
176
|
+
def complexobj2soapchildren_array(obj, soap, child_ele, nillable)
|
177
|
+
child = Mapping.get_attribute(obj, child_ele.name.name)
|
178
|
+
if child.nil? and obj.is_a?(::Array)
|
179
|
+
child = obj
|
180
|
+
end
|
181
|
+
if child.nil?
|
182
|
+
return false if nillable
|
183
|
+
if child_soap = nil2soap(child_ele)
|
184
|
+
soap.add(child_soap)
|
185
|
+
return true
|
186
|
+
else
|
187
|
+
return false
|
188
|
+
end
|
189
|
+
end
|
190
|
+
unless child.respond_to?(:each)
|
191
|
+
return false
|
192
|
+
end
|
193
|
+
child.each do |item|
|
194
|
+
if item.is_a?(SOAPElement)
|
195
|
+
soap.add(item)
|
196
|
+
else
|
197
|
+
child_soap = obj2elesoap(item, child_ele)
|
198
|
+
soap.add(child_soap)
|
199
|
+
end
|
200
|
+
end
|
201
|
+
true
|
202
|
+
end
|
203
|
+
|
204
|
+
def complexobj2soapchildren_single(obj, soap, child_ele, nillable)
|
205
|
+
child = Mapping.get_attribute(obj, child_ele.name.name)
|
206
|
+
case child
|
207
|
+
when NilClass
|
208
|
+
return false if nillable
|
209
|
+
if child_soap = nil2soap(child_ele)
|
210
|
+
soap.add(child_soap)
|
211
|
+
true
|
212
|
+
else
|
213
|
+
false
|
214
|
+
end
|
215
|
+
when SOAPElement
|
216
|
+
soap.add(child)
|
217
|
+
true
|
218
|
+
else
|
219
|
+
child_soap = obj2elesoap(child, child_ele)
|
220
|
+
soap.add(child_soap)
|
221
|
+
true
|
222
|
+
end
|
223
|
+
end
|
224
|
+
|
225
|
+
def nil2soap(ele)
|
226
|
+
if ele.nillable
|
227
|
+
obj2elesoap(nil, ele) # add an empty element
|
228
|
+
elsif ele.minoccurs == 0
|
229
|
+
nil # intends no element
|
230
|
+
else
|
231
|
+
warn("nil not allowed: #{ele.name.name}")
|
232
|
+
nil
|
233
|
+
end
|
234
|
+
end
|
235
|
+
|
236
|
+
def add_definedattributes2soap(obj, ele, typedef)
|
237
|
+
if typedef.attributes
|
238
|
+
typedef.attributes.each do |at|
|
239
|
+
value = get_xmlattr_value(obj, at.name)
|
240
|
+
ele.extraattr[at.name] = value unless value.nil?
|
241
|
+
end
|
242
|
+
end
|
243
|
+
end
|
244
|
+
end
|
245
|
+
|
246
|
+
|
247
|
+
end
|
248
|
+
end
|
data/lib/soap/marshal.rb
ADDED
@@ -0,0 +1,59 @@
|
|
1
|
+
# SOAP4R - Marshalling/Unmarshalling Ruby's object using SOAP Encoding.
|
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/mapping"
|
10
|
+
require "soap/processor"
|
11
|
+
|
12
|
+
|
13
|
+
module SOAP
|
14
|
+
|
15
|
+
|
16
|
+
module Marshal
|
17
|
+
# Trying xsd:dateTime data to be recovered as aTime.
|
18
|
+
MarshalMappingRegistry = Mapping::EncodedRegistry.new(
|
19
|
+
:allow_original_mapping => true)
|
20
|
+
MarshalMappingRegistry.add(
|
21
|
+
Time,
|
22
|
+
::SOAP::SOAPDateTime,
|
23
|
+
::SOAP::Mapping::EncodedRegistry::DateTimeFactory
|
24
|
+
)
|
25
|
+
|
26
|
+
class << self
|
27
|
+
public
|
28
|
+
def dump(obj, io = nil)
|
29
|
+
marshal(obj, MarshalMappingRegistry, io)
|
30
|
+
end
|
31
|
+
|
32
|
+
def load(stream)
|
33
|
+
unmarshal(stream, MarshalMappingRegistry)
|
34
|
+
end
|
35
|
+
|
36
|
+
def marshal(obj, mapping_registry = MarshalMappingRegistry, io = nil)
|
37
|
+
elename = Mapping.name2elename(obj.class.to_s)
|
38
|
+
soap_obj = Mapping.obj2soap(obj, mapping_registry)
|
39
|
+
body = SOAPBody.new
|
40
|
+
body.add(elename, soap_obj)
|
41
|
+
env = SOAPEnvelope.new(nil, body)
|
42
|
+
SOAP::Processor.marshal(env, {}, io)
|
43
|
+
end
|
44
|
+
|
45
|
+
def unmarshal(stream, mapping_registry = MarshalMappingRegistry)
|
46
|
+
env = SOAP::Processor.unmarshal(stream)
|
47
|
+
if env.nil?
|
48
|
+
raise ArgumentError.new("Illegal SOAP marshal format.")
|
49
|
+
end
|
50
|
+
Mapping.soap2obj(env.body.root_node, mapping_registry)
|
51
|
+
end
|
52
|
+
end
|
53
|
+
end
|
54
|
+
|
55
|
+
|
56
|
+
end
|
57
|
+
|
58
|
+
|
59
|
+
SOAPMarshal = SOAP::Marshal
|
@@ -0,0 +1,241 @@
|
|
1
|
+
# SOAP4R - MIME Message implementation.
|
2
|
+
# Copyright (C) 2002 Jamie Herre.
|
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/attachment'
|
10
|
+
|
11
|
+
|
12
|
+
module SOAP
|
13
|
+
|
14
|
+
|
15
|
+
# Classes for MIME message handling. Should be put somewhere else!
|
16
|
+
# Tried using the 'tmail' module but found that I needed something
|
17
|
+
# lighter in weight.
|
18
|
+
|
19
|
+
|
20
|
+
class MIMEMessage
|
21
|
+
class MIMEMessageError < StandardError; end
|
22
|
+
|
23
|
+
MultipartContentType = 'multipart/\w+'
|
24
|
+
|
25
|
+
class Header
|
26
|
+
attr_accessor :str, :key, :root
|
27
|
+
|
28
|
+
def initialize
|
29
|
+
@attrs = {}
|
30
|
+
end
|
31
|
+
|
32
|
+
def [](key)
|
33
|
+
@attrs[key]
|
34
|
+
end
|
35
|
+
|
36
|
+
def []=(key, value)
|
37
|
+
@attrs[key] = value
|
38
|
+
end
|
39
|
+
|
40
|
+
def to_s
|
41
|
+
@key + ": " + @str
|
42
|
+
end
|
43
|
+
end
|
44
|
+
|
45
|
+
class Headers < Hash
|
46
|
+
def self.parse(str)
|
47
|
+
new.parse(str)
|
48
|
+
end
|
49
|
+
|
50
|
+
def parse(str)
|
51
|
+
header_cache = nil
|
52
|
+
str.each do |line|
|
53
|
+
case line
|
54
|
+
when /^\A[^\: \t]+:\s*.+$/
|
55
|
+
parse_line(header_cache) if header_cache
|
56
|
+
header_cache = line.sub(/\r?\n\z/, '')
|
57
|
+
when /^\A\s+(.*)$/
|
58
|
+
# a continuous line at the beginning line crashes here.
|
59
|
+
header_cache << line
|
60
|
+
else
|
61
|
+
raise RuntimeError.new("unexpected header: #{line.inspect}")
|
62
|
+
end
|
63
|
+
end
|
64
|
+
parse_line(header_cache) if header_cache
|
65
|
+
self
|
66
|
+
end
|
67
|
+
|
68
|
+
def parse_line(line)
|
69
|
+
if /^\A([^\: \t]+):\s*(.+)\z/ =~ line
|
70
|
+
header = parse_rhs($2.strip)
|
71
|
+
header.key = $1.strip
|
72
|
+
self[header.key.downcase] = header
|
73
|
+
else
|
74
|
+
raise RuntimeError.new("unexpected header line: #{line.inspect}")
|
75
|
+
end
|
76
|
+
end
|
77
|
+
|
78
|
+
def parse_rhs(str)
|
79
|
+
a = str.split(/;+\s+/)
|
80
|
+
header = Header.new
|
81
|
+
header.str = str
|
82
|
+
header.root = a.shift
|
83
|
+
a.each do |pair|
|
84
|
+
if pair =~ /(\w+)\s*=\s*"?([^"]+)"?/
|
85
|
+
header[$1.downcase] = $2
|
86
|
+
else
|
87
|
+
raise RuntimeError.new("unexpected header component: #{pair.inspect}")
|
88
|
+
end
|
89
|
+
end
|
90
|
+
header
|
91
|
+
end
|
92
|
+
|
93
|
+
def add(key, value)
|
94
|
+
if key != nil and value != nil
|
95
|
+
header = parse_rhs(value)
|
96
|
+
header.key = key
|
97
|
+
self[key.downcase] = header
|
98
|
+
end
|
99
|
+
end
|
100
|
+
|
101
|
+
def to_s
|
102
|
+
self.values.collect { |hdr|
|
103
|
+
hdr.to_s
|
104
|
+
}.join("\r\n")
|
105
|
+
end
|
106
|
+
end
|
107
|
+
|
108
|
+
class Part
|
109
|
+
attr_accessor :headers, :body
|
110
|
+
|
111
|
+
def initialize
|
112
|
+
@headers = Headers.new
|
113
|
+
@headers.add("Content-Transfer-Encoding", "8bit")
|
114
|
+
@body = nil
|
115
|
+
@contentid = nil
|
116
|
+
end
|
117
|
+
|
118
|
+
def self.parse(str)
|
119
|
+
new.parse(str)
|
120
|
+
end
|
121
|
+
|
122
|
+
def parse(str)
|
123
|
+
headers, body = str.split(/\r\n\r\n/, 2)
|
124
|
+
if headers != nil and body != nil
|
125
|
+
@headers = Headers.parse(headers)
|
126
|
+
@body = body.sub(/\r\n\z/, '')
|
127
|
+
else
|
128
|
+
raise RuntimeError.new("unexpected part: #{str.inspect}")
|
129
|
+
end
|
130
|
+
self
|
131
|
+
end
|
132
|
+
|
133
|
+
def contentid
|
134
|
+
if @contentid == nil and @headers.key?('content-id')
|
135
|
+
@contentid = @headers['content-id'].str
|
136
|
+
@contentid = $1 if @contentid =~ /^<(.+)>$/
|
137
|
+
end
|
138
|
+
@contentid
|
139
|
+
end
|
140
|
+
|
141
|
+
alias content body
|
142
|
+
|
143
|
+
def to_s
|
144
|
+
@headers.to_s + "\r\n\r\n" + @body
|
145
|
+
end
|
146
|
+
end
|
147
|
+
|
148
|
+
def initialize
|
149
|
+
@parts = []
|
150
|
+
@headers = Headers.new
|
151
|
+
@root = nil
|
152
|
+
@boundary = nil
|
153
|
+
end
|
154
|
+
|
155
|
+
def self.parse(head, str)
|
156
|
+
new.parse(head, str)
|
157
|
+
end
|
158
|
+
|
159
|
+
attr_reader :parts, :headers
|
160
|
+
|
161
|
+
def close
|
162
|
+
@headers.add(
|
163
|
+
"Content-Type",
|
164
|
+
"multipart/related; type=\"text/xml\"; boundary=\"#{boundary}\"; start=\"#{@parts[0].contentid}\""
|
165
|
+
)
|
166
|
+
end
|
167
|
+
|
168
|
+
def parse(head, str)
|
169
|
+
@headers = Headers.parse(head + "\r\n" + "From: jfh\r\n")
|
170
|
+
boundary = @headers['content-type']['boundary']
|
171
|
+
if boundary != nil
|
172
|
+
parts = str.split(/--#{Regexp.quote(boundary)}\s*(?:\r\n|--\r\n)/)
|
173
|
+
part = parts.shift # preamble must be ignored.
|
174
|
+
@parts = parts.collect { |part| Part.parse(part) }
|
175
|
+
else
|
176
|
+
@parts = [Part.parse(str)]
|
177
|
+
end
|
178
|
+
if @parts.length < 1
|
179
|
+
raise MIMEMessageError.new("This message contains no valid parts!")
|
180
|
+
end
|
181
|
+
self
|
182
|
+
end
|
183
|
+
|
184
|
+
def root
|
185
|
+
if @root == nil
|
186
|
+
start = @headers['content-type']['start']
|
187
|
+
@root = (start && @parts.find { |prt| prt.contentid == start }) ||
|
188
|
+
@parts[0]
|
189
|
+
end
|
190
|
+
@root
|
191
|
+
end
|
192
|
+
|
193
|
+
def boundary
|
194
|
+
if @boundary == nil
|
195
|
+
@boundary = "----=Part_" + __id__.to_s + rand.to_s
|
196
|
+
end
|
197
|
+
@boundary
|
198
|
+
end
|
199
|
+
|
200
|
+
def add_part(content)
|
201
|
+
part = Part.new
|
202
|
+
part.headers.add("Content-Type",
|
203
|
+
"text/xml; charset=" + XSD::Charset.xml_encoding_label)
|
204
|
+
part.headers.add("Content-ID", Attachment.contentid(part))
|
205
|
+
part.body = content
|
206
|
+
@parts.unshift(part)
|
207
|
+
end
|
208
|
+
|
209
|
+
def add_attachment(attach)
|
210
|
+
part = Part.new
|
211
|
+
part.headers.add("Content-Type", attach.contenttype)
|
212
|
+
part.headers.add("Content-ID", attach.mime_contentid)
|
213
|
+
part.body = attach.content
|
214
|
+
@parts.unshift(part)
|
215
|
+
end
|
216
|
+
|
217
|
+
def has_parts?
|
218
|
+
(@parts.length > 0)
|
219
|
+
end
|
220
|
+
|
221
|
+
def headers_str
|
222
|
+
@headers.to_s
|
223
|
+
end
|
224
|
+
|
225
|
+
def content_str
|
226
|
+
str = ''
|
227
|
+
@parts.each do |prt|
|
228
|
+
str << "--" + boundary + "\r\n"
|
229
|
+
str << prt.to_s + "\r\n"
|
230
|
+
end
|
231
|
+
str << '--' + boundary + "--\r\n"
|
232
|
+
str
|
233
|
+
end
|
234
|
+
|
235
|
+
def to_s
|
236
|
+
str = headers_str + "\r\n\r\n" + conent_str
|
237
|
+
end
|
238
|
+
end
|
239
|
+
|
240
|
+
|
241
|
+
end
|