rubysl-soap 0.0.1 → 1.0.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/.gitignore +0 -1
- data/.travis.yml +8 -0
- data/README.md +2 -2
- data/Rakefile +0 -1
- data/lib/rubysl/soap.rb +1 -0
- data/lib/rubysl/soap/version.rb +5 -0
- data/lib/soap/attachment.rb +107 -0
- data/lib/soap/baseData.rb +942 -0
- data/lib/soap/element.rb +258 -0
- data/lib/soap/encodingstyle/aspDotNetHandler.rb +213 -0
- data/lib/soap/encodingstyle/handler.rb +100 -0
- data/lib/soap/encodingstyle/literalHandler.rb +226 -0
- data/lib/soap/encodingstyle/soapHandler.rb +582 -0
- data/lib/soap/generator.rb +268 -0
- data/lib/soap/header/handler.rb +57 -0
- data/lib/soap/header/handlerset.rb +70 -0
- data/lib/soap/header/simplehandler.rb +44 -0
- data/lib/soap/httpconfigloader.rb +119 -0
- data/lib/soap/mapping.rb +10 -0
- data/lib/soap/mapping/factory.rb +355 -0
- data/lib/soap/mapping/mapping.rb +381 -0
- data/lib/soap/mapping/registry.rb +541 -0
- data/lib/soap/mapping/rubytypeFactory.rb +475 -0
- data/lib/soap/mapping/typeMap.rb +50 -0
- data/lib/soap/mapping/wsdlencodedregistry.rb +280 -0
- data/lib/soap/mapping/wsdlliteralregistry.rb +418 -0
- data/lib/soap/marshal.rb +59 -0
- data/lib/soap/mimemessage.rb +240 -0
- data/lib/soap/netHttpClient.rb +190 -0
- data/lib/soap/parser.rb +251 -0
- data/lib/soap/processor.rb +66 -0
- data/lib/soap/property.rb +333 -0
- data/lib/soap/rpc/cgistub.rb +206 -0
- data/lib/soap/rpc/driver.rb +254 -0
- data/lib/soap/rpc/element.rb +325 -0
- data/lib/soap/rpc/httpserver.rb +129 -0
- data/lib/soap/rpc/proxy.rb +497 -0
- data/lib/soap/rpc/router.rb +594 -0
- data/lib/soap/rpc/rpc.rb +25 -0
- data/lib/soap/rpc/soaplet.rb +162 -0
- data/lib/soap/rpc/standaloneServer.rb +43 -0
- data/lib/soap/soap.rb +140 -0
- data/lib/soap/streamHandler.rb +229 -0
- data/lib/soap/wsdlDriver.rb +575 -0
- data/rubysl-soap.gemspec +19 -18
- metadata +115 -86
- data/lib/rubysl-soap.rb +0 -7
- data/lib/rubysl-soap/version.rb +0 -5
data/lib/soap/marshal.rb
ADDED
@@ -0,0 +1,59 @@
|
|
1
|
+
# SOAP4R - Marshalling/Unmarshalling Ruby's object using SOAP Encoding.
|
2
|
+
# Copyright (C) 2001, 2003 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::Registry.new(
|
19
|
+
:allow_original_mapping => true)
|
20
|
+
MarshalMappingRegistry.add(
|
21
|
+
Time,
|
22
|
+
::SOAP::SOAPDateTime,
|
23
|
+
::SOAP::Mapping::Registry::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,240 @@
|
|
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/s)
|
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
|
+
end
|
153
|
+
|
154
|
+
def self.parse(head, str)
|
155
|
+
new.parse(head, str)
|
156
|
+
end
|
157
|
+
|
158
|
+
attr_reader :parts, :headers
|
159
|
+
|
160
|
+
def close
|
161
|
+
@headers.add(
|
162
|
+
"Content-Type",
|
163
|
+
"multipart/related; type=\"text/xml\"; boundary=\"#{boundary}\"; start=\"#{@parts[0].contentid}\""
|
164
|
+
)
|
165
|
+
end
|
166
|
+
|
167
|
+
def parse(head, str)
|
168
|
+
@headers = Headers.parse(head + "\r\n" + "From: jfh\r\n")
|
169
|
+
boundary = @headers['content-type']['boundary']
|
170
|
+
if boundary != nil
|
171
|
+
parts = str.split(/--#{Regexp.quote(boundary)}\s*(?:\r\n|--\r\n)/)
|
172
|
+
part = parts.shift # preamble must be ignored.
|
173
|
+
@parts = parts.collect { |part| Part.parse(part) }
|
174
|
+
else
|
175
|
+
@parts = [Part.parse(str)]
|
176
|
+
end
|
177
|
+
if @parts.length < 1
|
178
|
+
raise MIMEMessageError.new("This message contains no valid parts!")
|
179
|
+
end
|
180
|
+
self
|
181
|
+
end
|
182
|
+
|
183
|
+
def root
|
184
|
+
if @root == nil
|
185
|
+
start = @headers['content-type']['start']
|
186
|
+
@root = (start && @parts.find { |prt| prt.contentid == start }) ||
|
187
|
+
@parts[0]
|
188
|
+
end
|
189
|
+
@root
|
190
|
+
end
|
191
|
+
|
192
|
+
def boundary
|
193
|
+
if @boundary == nil
|
194
|
+
@boundary = "----=Part_" + __id__.to_s + rand.to_s
|
195
|
+
end
|
196
|
+
@boundary
|
197
|
+
end
|
198
|
+
|
199
|
+
def add_part(content)
|
200
|
+
part = Part.new
|
201
|
+
part.headers.add("Content-Type",
|
202
|
+
"text/xml; charset=" + XSD::Charset.xml_encoding_label)
|
203
|
+
part.headers.add("Content-ID", Attachment.contentid(part))
|
204
|
+
part.body = content
|
205
|
+
@parts.unshift(part)
|
206
|
+
end
|
207
|
+
|
208
|
+
def add_attachment(attach)
|
209
|
+
part = Part.new
|
210
|
+
part.headers.add("Content-Type", attach.contenttype)
|
211
|
+
part.headers.add("Content-ID", attach.mime_contentid)
|
212
|
+
part.body = attach.content
|
213
|
+
@parts.unshift(part)
|
214
|
+
end
|
215
|
+
|
216
|
+
def has_parts?
|
217
|
+
(@parts.length > 0)
|
218
|
+
end
|
219
|
+
|
220
|
+
def headers_str
|
221
|
+
@headers.to_s
|
222
|
+
end
|
223
|
+
|
224
|
+
def content_str
|
225
|
+
str = ''
|
226
|
+
@parts.each do |prt|
|
227
|
+
str << "--" + boundary + "\r\n"
|
228
|
+
str << prt.to_s + "\r\n"
|
229
|
+
end
|
230
|
+
str << '--' + boundary + "--\r\n"
|
231
|
+
str
|
232
|
+
end
|
233
|
+
|
234
|
+
def to_s
|
235
|
+
str = headers_str + "\r\n\r\n" + content_str
|
236
|
+
end
|
237
|
+
end
|
238
|
+
|
239
|
+
|
240
|
+
end
|
@@ -0,0 +1,190 @@
|
|
1
|
+
# SOAP4R - net/http wrapper
|
2
|
+
# Copyright (C) 2003 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 'net/http'
|
10
|
+
|
11
|
+
|
12
|
+
module SOAP
|
13
|
+
|
14
|
+
|
15
|
+
class NetHttpClient
|
16
|
+
|
17
|
+
SSLEnabled = begin
|
18
|
+
require 'net/https'
|
19
|
+
true
|
20
|
+
rescue LoadError
|
21
|
+
false
|
22
|
+
end
|
23
|
+
|
24
|
+
attr_reader :proxy
|
25
|
+
attr_accessor :no_proxy
|
26
|
+
attr_accessor :debug_dev
|
27
|
+
attr_accessor :ssl_config # ignored for now.
|
28
|
+
attr_accessor :protocol_version # ignored for now.
|
29
|
+
attr_accessor :connect_timeout
|
30
|
+
attr_accessor :send_timeout # ignored for now.
|
31
|
+
attr_accessor :receive_timeout
|
32
|
+
|
33
|
+
def initialize(proxy = nil, agent = nil)
|
34
|
+
@proxy = proxy ? URI.parse(proxy) : nil
|
35
|
+
@agent = agent
|
36
|
+
@debug_dev = nil
|
37
|
+
@session_manager = SessionManager.new
|
38
|
+
@no_proxy = @ssl_config = @protocol_version = nil
|
39
|
+
@connect_timeout = @send_timeout = @receive_timeout = nil
|
40
|
+
end
|
41
|
+
|
42
|
+
def test_loopback_response
|
43
|
+
raise NotImplementedError.new("not supported for now")
|
44
|
+
end
|
45
|
+
|
46
|
+
def proxy=(proxy)
|
47
|
+
if proxy.nil?
|
48
|
+
@proxy = nil
|
49
|
+
else
|
50
|
+
if proxy.is_a?(URI)
|
51
|
+
@proxy = proxy
|
52
|
+
else
|
53
|
+
@proxy = URI.parse(proxy)
|
54
|
+
end
|
55
|
+
if @proxy.scheme == nil or @proxy.scheme.downcase != 'http' or
|
56
|
+
@proxy.host == nil or @proxy.port == nil
|
57
|
+
raise ArgumentError.new("unsupported proxy `#{proxy}'")
|
58
|
+
end
|
59
|
+
end
|
60
|
+
reset_all
|
61
|
+
@proxy
|
62
|
+
end
|
63
|
+
|
64
|
+
def set_basic_auth(uri, user_id, passwd)
|
65
|
+
# net/http does not handle url.
|
66
|
+
@basic_auth = [user_id, passwd]
|
67
|
+
raise NotImplementedError.new("basic_auth is not supported under soap4r + net/http.")
|
68
|
+
end
|
69
|
+
|
70
|
+
def set_cookie_store(filename)
|
71
|
+
raise NotImplementedError.new
|
72
|
+
end
|
73
|
+
|
74
|
+
def save_cookie_store(filename)
|
75
|
+
raise NotImplementedError.new
|
76
|
+
end
|
77
|
+
|
78
|
+
def reset(url)
|
79
|
+
# no persistent connection. ignored.
|
80
|
+
end
|
81
|
+
|
82
|
+
def reset_all
|
83
|
+
# no persistent connection. ignored.
|
84
|
+
end
|
85
|
+
|
86
|
+
def post(url, req_body, header = {})
|
87
|
+
unless url.is_a?(URI)
|
88
|
+
url = URI.parse(url)
|
89
|
+
end
|
90
|
+
extra = header.dup
|
91
|
+
extra['User-Agent'] = @agent if @agent
|
92
|
+
res = start(url) { |http|
|
93
|
+
http.post(url.request_uri, req_body, extra)
|
94
|
+
}
|
95
|
+
Response.new(res)
|
96
|
+
end
|
97
|
+
|
98
|
+
def get_content(url, header = {})
|
99
|
+
unless url.is_a?(URI)
|
100
|
+
url = URI.parse(url)
|
101
|
+
end
|
102
|
+
extra = header.dup
|
103
|
+
extra['User-Agent'] = @agent if @agent
|
104
|
+
res = start(url) { |http|
|
105
|
+
http.get(url.request_uri, extra)
|
106
|
+
}
|
107
|
+
res.body
|
108
|
+
end
|
109
|
+
|
110
|
+
private
|
111
|
+
|
112
|
+
def start(url)
|
113
|
+
http = create_connection(url)
|
114
|
+
response = nil
|
115
|
+
http.start { |worker|
|
116
|
+
response = yield(worker)
|
117
|
+
worker.finish
|
118
|
+
}
|
119
|
+
@debug_dev << response.body if @debug_dev
|
120
|
+
response
|
121
|
+
end
|
122
|
+
|
123
|
+
def create_connection(url)
|
124
|
+
proxy_host = proxy_port = nil
|
125
|
+
unless no_proxy?(url)
|
126
|
+
proxy_host = @proxy.host
|
127
|
+
proxy_port = @proxy.port
|
128
|
+
end
|
129
|
+
http = Net::HTTP::Proxy(proxy_host, proxy_port).new(url.host, url.port)
|
130
|
+
if http.respond_to?(:set_debug_output)
|
131
|
+
http.set_debug_output(@debug_dev)
|
132
|
+
end
|
133
|
+
http.open_timeout = @connect_timeout if @connect_timeout
|
134
|
+
http.read_timeout = @receive_timeout if @receive_timeout
|
135
|
+
case url
|
136
|
+
when URI::HTTPS
|
137
|
+
if SSLEnabled
|
138
|
+
http.use_ssl = true
|
139
|
+
else
|
140
|
+
raise RuntimeError.new("Cannot connect to #{url} (OpenSSL is not installed.)")
|
141
|
+
end
|
142
|
+
when URI::HTTP
|
143
|
+
# OK
|
144
|
+
else
|
145
|
+
raise RuntimeError.new("Cannot connect to #{url} (Not HTTP.)")
|
146
|
+
end
|
147
|
+
http
|
148
|
+
end
|
149
|
+
|
150
|
+
NO_PROXY_HOSTS = ['localhost']
|
151
|
+
|
152
|
+
def no_proxy?(uri)
|
153
|
+
if !@proxy or NO_PROXY_HOSTS.include?(uri.host)
|
154
|
+
return true
|
155
|
+
end
|
156
|
+
if @no_proxy
|
157
|
+
@no_proxy.scan(/([^:,]*)(?::(\d+))?/) do |host, port|
|
158
|
+
if /(\A|\.)#{Regexp.quote(host)}\z/i =~ uri.host &&
|
159
|
+
(!port || uri.port == port.to_i)
|
160
|
+
return true
|
161
|
+
end
|
162
|
+
end
|
163
|
+
else
|
164
|
+
false
|
165
|
+
end
|
166
|
+
end
|
167
|
+
|
168
|
+
class SessionManager
|
169
|
+
attr_accessor :connect_timeout
|
170
|
+
attr_accessor :send_timeout
|
171
|
+
attr_accessor :receive_timeout
|
172
|
+
end
|
173
|
+
|
174
|
+
class Response
|
175
|
+
attr_reader :content
|
176
|
+
attr_reader :status
|
177
|
+
attr_reader :reason
|
178
|
+
attr_reader :contenttype
|
179
|
+
|
180
|
+
def initialize(res)
|
181
|
+
@status = res.code.to_i
|
182
|
+
@reason = res.message
|
183
|
+
@contenttype = res['content-type']
|
184
|
+
@content = res.body
|
185
|
+
end
|
186
|
+
end
|
187
|
+
end
|
188
|
+
|
189
|
+
|
190
|
+
end
|