reidiculous-actionwebservice-client 3.0.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/CHANGELOG +320 -0
- data/MIT-LICENSE +21 -0
- data/README +381 -0
- data/Rakefile +180 -0
- data/TODO +32 -0
- data/lib/action_web_service.rb +71 -0
- data/lib/action_web_service/api.rb +297 -0
- data/lib/action_web_service/base.rb +38 -0
- data/lib/action_web_service/casting.rb +149 -0
- data/lib/action_web_service/client.rb +3 -0
- data/lib/action_web_service/client/base.rb +28 -0
- data/lib/action_web_service/client/soap_client.rb +113 -0
- data/lib/action_web_service/client/xmlrpc_client.rb +58 -0
- data/lib/action_web_service/container.rb +3 -0
- data/lib/action_web_service/container/action_controller_container.rb +93 -0
- data/lib/action_web_service/container/delegated_container.rb +86 -0
- data/lib/action_web_service/container/direct_container.rb +69 -0
- data/lib/action_web_service/dispatcher.rb +2 -0
- data/lib/action_web_service/dispatcher/abstract.rb +207 -0
- data/lib/action_web_service/dispatcher/action_controller_dispatcher.rb +379 -0
- data/lib/action_web_service/invocation.rb +202 -0
- data/lib/action_web_service/protocol.rb +4 -0
- data/lib/action_web_service/protocol/abstract.rb +115 -0
- data/lib/action_web_service/protocol/discovery.rb +37 -0
- data/lib/action_web_service/protocol/soap_protocol.rb +176 -0
- data/lib/action_web_service/protocol/soap_protocol/marshaler.rb +242 -0
- data/lib/action_web_service/protocol/xmlrpc_protocol.rb +122 -0
- data/lib/action_web_service/scaffolding.rb +281 -0
- data/lib/action_web_service/soap/attachment.rb +107 -0
- data/lib/action_web_service/soap/baseData.rb +942 -0
- data/lib/action_web_service/soap/element.rb +258 -0
- data/lib/action_web_service/soap/encodingstyle/aspDotNetHandler.rb +213 -0
- data/lib/action_web_service/soap/encodingstyle/handler.rb +100 -0
- data/lib/action_web_service/soap/encodingstyle/literalHandler.rb +226 -0
- data/lib/action_web_service/soap/encodingstyle/soapHandler.rb +582 -0
- data/lib/action_web_service/soap/generator.rb +268 -0
- data/lib/action_web_service/soap/header/handler.rb +57 -0
- data/lib/action_web_service/soap/header/handlerset.rb +70 -0
- data/lib/action_web_service/soap/header/simplehandler.rb +44 -0
- data/lib/action_web_service/soap/httpconfigloader.rb +119 -0
- data/lib/action_web_service/soap/mapping.rb +10 -0
- data/lib/action_web_service/soap/mapping/factory.rb +355 -0
- data/lib/action_web_service/soap/mapping/mapping.rb +381 -0
- data/lib/action_web_service/soap/mapping/registry.rb +541 -0
- data/lib/action_web_service/soap/mapping/rubytypeFactory.rb +475 -0
- data/lib/action_web_service/soap/mapping/typeMap.rb +50 -0
- data/lib/action_web_service/soap/mapping/wsdlencodedregistry.rb +280 -0
- data/lib/action_web_service/soap/mapping/wsdlliteralregistry.rb +418 -0
- data/lib/action_web_service/soap/marshal.rb +59 -0
- data/lib/action_web_service/soap/mimemessage.rb +240 -0
- data/lib/action_web_service/soap/netHttpClient.rb +190 -0
- data/lib/action_web_service/soap/parser.rb +251 -0
- data/lib/action_web_service/soap/processor.rb +66 -0
- data/lib/action_web_service/soap/property.rb +333 -0
- data/lib/action_web_service/soap/rpc/cgistub.rb +206 -0
- data/lib/action_web_service/soap/rpc/driver.rb +254 -0
- data/lib/action_web_service/soap/rpc/element.rb +325 -0
- data/lib/action_web_service/soap/rpc/httpserver.rb +129 -0
- data/lib/action_web_service/soap/rpc/proxy.rb +497 -0
- data/lib/action_web_service/soap/rpc/router.rb +594 -0
- data/lib/action_web_service/soap/rpc/rpc.rb +25 -0
- data/lib/action_web_service/soap/rpc/soaplet.rb +162 -0
- data/lib/action_web_service/soap/rpc/standaloneServer.rb +43 -0
- data/lib/action_web_service/soap/soap.rb +140 -0
- data/lib/action_web_service/soap/streamHandler.rb +229 -0
- data/lib/action_web_service/soap/wsdlDriver.rb +575 -0
- data/lib/action_web_service/struct.rb +64 -0
- data/lib/action_web_service/support/class_inheritable_options.rb +28 -0
- data/lib/action_web_service/support/signature_types.rb +227 -0
- data/lib/action_web_service/test_invoke.rb +110 -0
- data/lib/action_web_service/version.rb +9 -0
- data/lib/action_web_service/wsdl/binding.rb +65 -0
- data/lib/action_web_service/wsdl/data.rb +64 -0
- data/lib/action_web_service/wsdl/definitions.rb +250 -0
- data/lib/action_web_service/wsdl/documentation.rb +32 -0
- data/lib/action_web_service/wsdl/import.rb +80 -0
- data/lib/action_web_service/wsdl/importer.rb +38 -0
- data/lib/action_web_service/wsdl/info.rb +39 -0
- data/lib/action_web_service/wsdl/message.rb +54 -0
- data/lib/action_web_service/wsdl/operation.rb +130 -0
- data/lib/action_web_service/wsdl/operationBinding.rb +108 -0
- data/lib/action_web_service/wsdl/param.rb +85 -0
- data/lib/action_web_service/wsdl/parser.rb +163 -0
- data/lib/action_web_service/wsdl/part.rb +52 -0
- data/lib/action_web_service/wsdl/port.rb +84 -0
- data/lib/action_web_service/wsdl/portType.rb +73 -0
- data/lib/action_web_service/wsdl/service.rb +61 -0
- data/lib/action_web_service/wsdl/soap/address.rb +40 -0
- data/lib/action_web_service/wsdl/soap/binding.rb +49 -0
- data/lib/action_web_service/wsdl/soap/body.rb +56 -0
- data/lib/action_web_service/wsdl/soap/cgiStubCreator.rb +76 -0
- data/lib/action_web_service/wsdl/soap/classDefCreator.rb +314 -0
- data/lib/action_web_service/wsdl/soap/classDefCreatorSupport.rb +126 -0
- data/lib/action_web_service/wsdl/soap/clientSkeltonCreator.rb +78 -0
- data/lib/action_web_service/wsdl/soap/complexType.rb +161 -0
- data/lib/action_web_service/wsdl/soap/data.rb +42 -0
- data/lib/action_web_service/wsdl/soap/definitions.rb +149 -0
- data/lib/action_web_service/wsdl/soap/driverCreator.rb +95 -0
- data/lib/action_web_service/wsdl/soap/element.rb +28 -0
- data/lib/action_web_service/wsdl/soap/fault.rb +56 -0
- data/lib/action_web_service/wsdl/soap/header.rb +86 -0
- data/lib/action_web_service/wsdl/soap/headerfault.rb +56 -0
- data/lib/action_web_service/wsdl/soap/mappingRegistryCreator.rb +92 -0
- data/lib/action_web_service/wsdl/soap/methodDefCreator.rb +228 -0
- data/lib/action_web_service/wsdl/soap/operation.rb +122 -0
- data/lib/action_web_service/wsdl/soap/servantSkeltonCreator.rb +67 -0
- data/lib/action_web_service/wsdl/soap/standaloneServerStubCreator.rb +85 -0
- data/lib/action_web_service/wsdl/soap/wsdl2ruby.rb +176 -0
- data/lib/action_web_service/wsdl/types.rb +43 -0
- data/lib/action_web_service/wsdl/wsdl.rb +23 -0
- data/lib/action_web_service/wsdl/xmlSchema/all.rb +69 -0
- data/lib/action_web_service/wsdl/xmlSchema/annotation.rb +34 -0
- data/lib/action_web_service/wsdl/xmlSchema/any.rb +56 -0
- data/lib/action_web_service/wsdl/xmlSchema/attribute.rb +127 -0
- data/lib/action_web_service/wsdl/xmlSchema/choice.rb +69 -0
- data/lib/action_web_service/wsdl/xmlSchema/complexContent.rb +92 -0
- data/lib/action_web_service/wsdl/xmlSchema/complexType.rb +139 -0
- data/lib/action_web_service/wsdl/xmlSchema/content.rb +96 -0
- data/lib/action_web_service/wsdl/xmlSchema/data.rb +80 -0
- data/lib/action_web_service/wsdl/xmlSchema/element.rb +154 -0
- data/lib/action_web_service/wsdl/xmlSchema/enumeration.rb +36 -0
- data/lib/action_web_service/wsdl/xmlSchema/import.rb +65 -0
- data/lib/action_web_service/wsdl/xmlSchema/importer.rb +87 -0
- data/lib/action_web_service/wsdl/xmlSchema/include.rb +54 -0
- data/lib/action_web_service/wsdl/xmlSchema/length.rb +35 -0
- data/lib/action_web_service/wsdl/xmlSchema/parser.rb +166 -0
- data/lib/action_web_service/wsdl/xmlSchema/pattern.rb +36 -0
- data/lib/action_web_service/wsdl/xmlSchema/schema.rb +143 -0
- data/lib/action_web_service/wsdl/xmlSchema/sequence.rb +69 -0
- data/lib/action_web_service/wsdl/xmlSchema/simpleContent.rb +65 -0
- data/lib/action_web_service/wsdl/xmlSchema/simpleExtension.rb +54 -0
- data/lib/action_web_service/wsdl/xmlSchema/simpleRestriction.rb +73 -0
- data/lib/action_web_service/wsdl/xmlSchema/simpleType.rb +73 -0
- data/lib/action_web_service/wsdl/xmlSchema/unique.rb +34 -0
- data/lib/action_web_service/wsdl/xmlSchema/xsd2ruby.rb +107 -0
- data/lib/action_web_service/xsd/charset.rb +187 -0
- data/lib/action_web_service/xsd/codegen.rb +12 -0
- data/lib/action_web_service/xsd/codegen/classdef.rb +203 -0
- data/lib/action_web_service/xsd/codegen/commentdef.rb +34 -0
- data/lib/action_web_service/xsd/codegen/gensupport.rb +166 -0
- data/lib/action_web_service/xsd/codegen/methoddef.rb +63 -0
- data/lib/action_web_service/xsd/codegen/moduledef.rb +191 -0
- data/lib/action_web_service/xsd/datatypes.rb +1269 -0
- data/lib/action_web_service/xsd/datatypes1999.rb +20 -0
- data/lib/action_web_service/xsd/iconvcharset.rb +33 -0
- data/lib/action_web_service/xsd/mapping.rb +42 -0
- data/lib/action_web_service/xsd/namedelements.rb +95 -0
- data/lib/action_web_service/xsd/ns.rb +140 -0
- data/lib/action_web_service/xsd/qname.rb +78 -0
- data/lib/action_web_service/xsd/xmlparser.rb +61 -0
- data/lib/action_web_service/xsd/xmlparser/parser.rb +96 -0
- data/lib/action_web_service/xsd/xmlparser/rexmlparser.rb +54 -0
- data/lib/action_web_service/xsd/xmlparser/xmlparser.rb +50 -0
- data/lib/action_web_service/xsd/xmlparser/xmlscanner.rb +147 -0
- data/lib/actionwebservice.rb +1 -0
- data/setup.rb +1379 -0
- data/test/abstract_client.rb +183 -0
- data/test/abstract_dispatcher.rb +548 -0
- data/test/abstract_unit.rb +45 -0
- data/test/api_test.rb +103 -0
- data/test/apis/auto_load_api.rb +3 -0
- data/test/apis/broken_auto_load_api.rb +2 -0
- data/test/base_test.rb +42 -0
- data/test/casting_test.rb +95 -0
- data/test/client_soap_test.rb +156 -0
- data/test/client_xmlrpc_test.rb +153 -0
- data/test/container_test.rb +73 -0
- data/test/dispatcher_action_controller_soap_test.rb +139 -0
- data/test/dispatcher_action_controller_xmlrpc_test.rb +59 -0
- data/test/fixtures/db_definitions/mysql.sql +8 -0
- data/test/fixtures/db_definitions/sqlite3.sql +8 -0
- data/test/fixtures/users.yml +12 -0
- data/test/gencov +3 -0
- data/test/invocation_test.rb +185 -0
- data/test/run +6 -0
- data/test/scaffolded_controller_test.rb +146 -0
- data/test/struct_test.rb +52 -0
- data/test/test_invoke_test.rb +112 -0
- metadata +265 -0
data/Rakefile
ADDED
@@ -0,0 +1,180 @@
|
|
1
|
+
require 'rubygems'
|
2
|
+
require 'rake'
|
3
|
+
require 'rake/testtask'
|
4
|
+
require 'rake/rdoctask'
|
5
|
+
require 'rake/packagetask'
|
6
|
+
require 'rake/gempackagetask'
|
7
|
+
require 'rake/contrib/rubyforgepublisher'
|
8
|
+
require 'fileutils'
|
9
|
+
require File.join(File.dirname(__FILE__), 'lib', 'action_web_service', 'version')
|
10
|
+
|
11
|
+
PKG_BUILD = ENV['PKG_BUILD'] ? '.' + ENV['PKG_BUILD'] : ''
|
12
|
+
PKG_NAME = 'actionwebservice'
|
13
|
+
PKG_VERSION = ActionWebService::VERSION::STRING + PKG_BUILD
|
14
|
+
PKG_FILE_NAME = "#{PKG_NAME}-#{PKG_VERSION}"
|
15
|
+
PKG_DESTINATION = ENV["RAILS_PKG_DESTINATION"] || "../#{PKG_NAME}"
|
16
|
+
|
17
|
+
RELEASE_NAME = "REL #{PKG_VERSION}"
|
18
|
+
|
19
|
+
RUBY_FORGE_PROJECT = "aws"
|
20
|
+
RUBY_FORGE_USER = "webster132"
|
21
|
+
|
22
|
+
desc "Default Task"
|
23
|
+
task :default => [ :test ]
|
24
|
+
|
25
|
+
|
26
|
+
# Run the unit tests
|
27
|
+
Rake::TestTask.new { |t|
|
28
|
+
t.libs << "test"
|
29
|
+
t.test_files = Dir['test/*_test.rb']
|
30
|
+
t.verbose = true
|
31
|
+
}
|
32
|
+
|
33
|
+
SCHEMA_PATH = File.join(File.dirname(__FILE__), *%w(test fixtures db_definitions))
|
34
|
+
|
35
|
+
desc 'Build the MySQL test database'
|
36
|
+
task :build_database do
|
37
|
+
%x( mysqladmin -uroot create actionwebservice_unittest )
|
38
|
+
%x( mysql -uroot actionwebservice_unittest < #{File.join(SCHEMA_PATH, 'mysql.sql')} )
|
39
|
+
end
|
40
|
+
|
41
|
+
desc 'Build the sqlite3 test database'
|
42
|
+
task :build_sqlite3_database do
|
43
|
+
filename = 'actionwebservice_unittest.db'
|
44
|
+
File.delete filename if File.exist? filename
|
45
|
+
%x(sqlite3 #{filename} < #{File.join(SCHEMA_PATH, 'sqlite3.sql')})
|
46
|
+
end
|
47
|
+
|
48
|
+
|
49
|
+
# Generate the RDoc documentation
|
50
|
+
Rake::RDocTask.new { |rdoc|
|
51
|
+
rdoc.rdoc_dir = 'doc'
|
52
|
+
rdoc.title = "Action Web Service -- Web services for Action Pack"
|
53
|
+
rdoc.options << '--line-numbers' << '--inline-source'
|
54
|
+
rdoc.options << '--charset' << 'utf-8'
|
55
|
+
rdoc.template = "#{ENV['template']}.rb" if ENV['template']
|
56
|
+
rdoc.rdoc_files.include('README')
|
57
|
+
rdoc.rdoc_files.include('CHANGELOG')
|
58
|
+
rdoc.rdoc_files.include('lib/action_web_service.rb')
|
59
|
+
rdoc.rdoc_files.include('lib/action_web_service/*.rb')
|
60
|
+
rdoc.rdoc_files.include('lib/action_web_service/api/*.rb')
|
61
|
+
rdoc.rdoc_files.include('lib/action_web_service/client/*.rb')
|
62
|
+
rdoc.rdoc_files.include('lib/action_web_service/container/*.rb')
|
63
|
+
rdoc.rdoc_files.include('lib/action_web_service/dispatcher/*.rb')
|
64
|
+
rdoc.rdoc_files.include('lib/action_web_service/protocol/*.rb')
|
65
|
+
rdoc.rdoc_files.include('lib/action_web_service/support/*.rb')
|
66
|
+
}
|
67
|
+
|
68
|
+
|
69
|
+
# Create compressed packages
|
70
|
+
spec = Gem::Specification.new do |s|
|
71
|
+
s.platform = Gem::Platform::RUBY
|
72
|
+
s.name = PKG_NAME
|
73
|
+
s.summary = "Web service support for Action Pack."
|
74
|
+
s.description = %q{Adds WSDL/SOAP and XML-RPC web service support to Action Pack}
|
75
|
+
s.version = PKG_VERSION
|
76
|
+
|
77
|
+
s.author = "Leon Breedt, Kent Sibilev"
|
78
|
+
s.email = "bitserf@gmail.com, ksibilev@yahoo.com"
|
79
|
+
s.rubyforge_project = "aws"
|
80
|
+
s.homepage = "http://www.rubyonrails.org"
|
81
|
+
|
82
|
+
s.add_dependency('actionpack', '= 2.3.5' + PKG_BUILD)
|
83
|
+
s.add_dependency('activerecord', '= 2.3.5' + PKG_BUILD)
|
84
|
+
|
85
|
+
s.has_rdoc = true
|
86
|
+
s.requirements << 'none'
|
87
|
+
s.require_path = 'lib'
|
88
|
+
s.autorequire = 'actionwebservice'
|
89
|
+
|
90
|
+
s.files = [ "Rakefile", "setup.rb", "README", "TODO", "CHANGELOG", "MIT-LICENSE" ]
|
91
|
+
s.files = s.files + Dir.glob( "examples/**/*" ).delete_if { |item| item.match( /\.(svn|git)/ ) }
|
92
|
+
s.files = s.files + Dir.glob( "lib/**/*" ).delete_if { |item| item.match( /\.(svn|git)/ ) }
|
93
|
+
s.files = s.files + Dir.glob( "test/**/*" ).delete_if { |item| item.match( /\.(svn|git)/ ) }
|
94
|
+
s.files = s.files + Dir.glob( "generators/**/*" ).delete_if { |item| item.match( /\.(svn|git)/ ) }
|
95
|
+
end
|
96
|
+
Rake::GemPackageTask.new(spec) do |p|
|
97
|
+
p.gem_spec = spec
|
98
|
+
p.need_tar = true
|
99
|
+
p.need_zip = true
|
100
|
+
end
|
101
|
+
|
102
|
+
|
103
|
+
# Publish beta gem
|
104
|
+
desc "Publish the API documentation"
|
105
|
+
task :pgem => [:package] do
|
106
|
+
Rake::SshFilePublisher.new("davidhh@wrath.rubyonrails.org", "public_html/gems/gems", "pkg", "#{PKG_FILE_NAME}.gem").upload
|
107
|
+
`ssh davidhh@wrath.rubyonrails.org './gemupdate.sh'`
|
108
|
+
end
|
109
|
+
|
110
|
+
# Publish documentation
|
111
|
+
desc "Publish the API documentation"
|
112
|
+
task :pdoc => [:rdoc] do
|
113
|
+
Rake::SshDirPublisher.new("davidhh@wrath.rubyonrails.org", "public_html/aws", "doc").upload
|
114
|
+
end
|
115
|
+
|
116
|
+
|
117
|
+
def each_source_file(*args)
|
118
|
+
prefix, includes, excludes, open_file = args
|
119
|
+
prefix ||= File.dirname(__FILE__)
|
120
|
+
open_file = true if open_file.nil?
|
121
|
+
includes ||= %w[lib\/action_web_service\.rb$ lib\/action_web_service\/.*\.rb$]
|
122
|
+
excludes ||= %w[lib\/action_web_service\/vendor]
|
123
|
+
Find.find(prefix) do |file_name|
|
124
|
+
next if file_name =~ /\.svn/
|
125
|
+
file_name.gsub!(/^\.\//, '')
|
126
|
+
continue = false
|
127
|
+
includes.each do |inc|
|
128
|
+
if file_name.match(/#{inc}/)
|
129
|
+
continue = true
|
130
|
+
break
|
131
|
+
end
|
132
|
+
end
|
133
|
+
next unless continue
|
134
|
+
excludes.each do |exc|
|
135
|
+
if file_name.match(/#{exc}/)
|
136
|
+
continue = false
|
137
|
+
break
|
138
|
+
end
|
139
|
+
end
|
140
|
+
next unless continue
|
141
|
+
if open_file
|
142
|
+
File.open(file_name) do |f|
|
143
|
+
yield file_name, f
|
144
|
+
end
|
145
|
+
else
|
146
|
+
yield file_name
|
147
|
+
end
|
148
|
+
end
|
149
|
+
end
|
150
|
+
|
151
|
+
desc "Count lines of the AWS source code"
|
152
|
+
task :lines do
|
153
|
+
total_lines = total_loc = 0
|
154
|
+
puts "Per File:"
|
155
|
+
each_source_file do |file_name, f|
|
156
|
+
file_lines = file_loc = 0
|
157
|
+
while line = f.gets
|
158
|
+
file_lines += 1
|
159
|
+
next if line =~ /^\s*$/
|
160
|
+
next if line =~ /^\s*#/
|
161
|
+
file_loc += 1
|
162
|
+
end
|
163
|
+
puts " #{file_name}: Lines #{file_lines}, LOC #{file_loc}"
|
164
|
+
total_lines += file_lines
|
165
|
+
total_loc += file_loc
|
166
|
+
end
|
167
|
+
puts "Total:"
|
168
|
+
puts " Lines #{total_lines}, LOC #{total_loc}"
|
169
|
+
end
|
170
|
+
|
171
|
+
desc "Publish the release files to RubyForge."
|
172
|
+
task :release => [ :package ] do
|
173
|
+
require 'rubyforge'
|
174
|
+
|
175
|
+
packages = %w( gem tgz zip ).collect{ |ext| "pkg/#{PKG_NAME}-#{PKG_VERSION}.#{ext}" }
|
176
|
+
|
177
|
+
rubyforge = RubyForge.new
|
178
|
+
rubyforge.login
|
179
|
+
rubyforge.add_release(PKG_NAME, PKG_NAME, "REL #{PKG_VERSION}", *packages)
|
180
|
+
end
|
data/TODO
ADDED
@@ -0,0 +1,32 @@
|
|
1
|
+
= Post-1.0
|
2
|
+
- Document/Literal SOAP support
|
3
|
+
- URL-based dispatching, URL identifies method
|
4
|
+
|
5
|
+
- Add :rest dispatching mode, a.l.a. Backpack API. Clean up dispatching
|
6
|
+
in general. Support vanilla XML-format as a "Rails" protocol?
|
7
|
+
XML::Simple deserialization into params?
|
8
|
+
|
9
|
+
web_service_dispatching_mode :rest
|
10
|
+
|
11
|
+
def method1(params)
|
12
|
+
end
|
13
|
+
|
14
|
+
def method2(params)
|
15
|
+
end
|
16
|
+
|
17
|
+
|
18
|
+
/ws/method1
|
19
|
+
<xml>
|
20
|
+
/ws/method2
|
21
|
+
<yaml>
|
22
|
+
|
23
|
+
- Allow locking down a controller to only accept messages for a particular
|
24
|
+
protocol. This will allow us to generate fully conformant error messages
|
25
|
+
in cases where we currently fudge it if we don't know the protocol.
|
26
|
+
|
27
|
+
- Allow AWS user to participate in typecasting, so they can centralize
|
28
|
+
workarounds for buggy input in one place
|
29
|
+
|
30
|
+
= Refactoring
|
31
|
+
- Don't have clean way to go from SOAP Class object to the xsd:NAME type
|
32
|
+
string -- NaHi possibly looking at remedying this situation
|
@@ -0,0 +1,71 @@
|
|
1
|
+
#--
|
2
|
+
# Copyright (C) 2005 Leon Breedt
|
3
|
+
#
|
4
|
+
# Permission is hereby granted, free of charge, to any person obtaining
|
5
|
+
# a copy of this software and associated documentation files (the
|
6
|
+
# "Software"), to deal in the Software without restriction, including
|
7
|
+
# without limitation the rights to use, copy, modify, merge, publish,
|
8
|
+
# distribute, sublicense, and/or sell copies of the Software, and to
|
9
|
+
# permit persons to whom the Software is furnished to do so, subject to
|
10
|
+
# the following conditions:
|
11
|
+
#
|
12
|
+
# The above copyright notice and this permission notice shall be
|
13
|
+
# included in all copies or substantial portions of the Software.
|
14
|
+
#
|
15
|
+
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
16
|
+
# EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
17
|
+
# MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
18
|
+
# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
19
|
+
# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
20
|
+
# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
21
|
+
# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
22
|
+
#++
|
23
|
+
|
24
|
+
begin
|
25
|
+
require 'activesupport'
|
26
|
+
require 'actioncontroller'
|
27
|
+
require 'activerecord'
|
28
|
+
require 'activesupport'
|
29
|
+
require 'actionpack'
|
30
|
+
require 'active_support/core_ext/class/inheritable_attributes'
|
31
|
+
require 'action_dispatch/routing'
|
32
|
+
rescue LoadError
|
33
|
+
require 'rubygems'
|
34
|
+
gem 'activesupport', '3.0.4'
|
35
|
+
gem 'actionpack', '3.0.4'
|
36
|
+
gem 'activerecord', '3.0.4'
|
37
|
+
gem 'activesupport', '3.0.4'
|
38
|
+
end
|
39
|
+
|
40
|
+
$:.unshift(File.dirname(__FILE__) + "/action_web_service/vendor/")
|
41
|
+
|
42
|
+
require 'action_web_service/support/class_inheritable_options'
|
43
|
+
require 'action_web_service/support/signature_types'
|
44
|
+
require 'action_web_service/base'
|
45
|
+
require 'action_web_service/client'
|
46
|
+
require 'action_web_service/invocation'
|
47
|
+
require 'action_web_service/api'
|
48
|
+
require 'action_web_service/casting'
|
49
|
+
require 'action_web_service/struct'
|
50
|
+
require 'action_web_service/container'
|
51
|
+
require 'action_web_service/protocol'
|
52
|
+
require 'action_web_service/dispatcher'
|
53
|
+
require 'action_web_service/scaffolding'
|
54
|
+
|
55
|
+
ActionWebService::Base.class_eval do
|
56
|
+
include ActionWebService::Container::Direct
|
57
|
+
include ActionWebService::Invocation
|
58
|
+
end
|
59
|
+
|
60
|
+
# ActionController::Base.class_eval do
|
61
|
+
# include ActionWebService::Protocol::Discovery
|
62
|
+
# include ActionWebService::Protocol::Soap
|
63
|
+
# include ActionWebService::Protocol::XmlRpc
|
64
|
+
# include ActionWebService::Container::Direct
|
65
|
+
# include ActionWebService::Container::Delegated
|
66
|
+
# include ActionWebService::Container::ActionController
|
67
|
+
# include ActionWebService::Invocation
|
68
|
+
# include ActionWebService::Dispatcher
|
69
|
+
# include ActionWebService::Dispatcher::ActionController
|
70
|
+
# include ActionWebService::Scaffolding
|
71
|
+
# end
|
@@ -0,0 +1,297 @@
|
|
1
|
+
module ActionWebService # :nodoc:
|
2
|
+
module API # :nodoc:
|
3
|
+
# A web service API class specifies the methods that will be available for
|
4
|
+
# invocation for an API. It also contains metadata such as the method type
|
5
|
+
# signature hints.
|
6
|
+
#
|
7
|
+
# It is not intended to be instantiated.
|
8
|
+
#
|
9
|
+
# It is attached to web service implementation classes like
|
10
|
+
# ActionWebService::Base and ActionController::Base derivatives by using
|
11
|
+
# <tt>container.web_service_api</tt>, where <tt>container</tt> is an
|
12
|
+
# ActionController::Base or a ActionWebService::Base.
|
13
|
+
#
|
14
|
+
# See ActionWebService::Container::Direct::ClassMethods for an example
|
15
|
+
# of use.
|
16
|
+
class Base
|
17
|
+
# Whether to transform the public API method names into camel-cased names
|
18
|
+
class_inheritable_option :inflect_names, true
|
19
|
+
|
20
|
+
# By default only HTTP POST requests are processed
|
21
|
+
class_inheritable_option :allowed_http_methods, [ :post ]
|
22
|
+
|
23
|
+
# Whether to allow ActiveRecord::Base models in <tt>:expects</tt>.
|
24
|
+
# The default is +false+; you should be aware of the security implications
|
25
|
+
# of allowing this, and ensure that you don't allow remote callers to
|
26
|
+
# easily overwrite data they should not have access to.
|
27
|
+
class_inheritable_option :allow_active_record_expects, false
|
28
|
+
|
29
|
+
# If present, the name of a method to call when the remote caller
|
30
|
+
# tried to call a nonexistent method. Semantically equivalent to
|
31
|
+
# +method_missing+.
|
32
|
+
class_inheritable_option :default_api_method
|
33
|
+
|
34
|
+
# Disallow instantiation
|
35
|
+
private_class_method :new, :allocate
|
36
|
+
|
37
|
+
class << self
|
38
|
+
include ActionWebService::SignatureTypes
|
39
|
+
|
40
|
+
# API methods have a +name+, which must be the Ruby method name to use when
|
41
|
+
# performing the invocation on the web service object.
|
42
|
+
#
|
43
|
+
# The signatures for the method input parameters and return value can
|
44
|
+
# by specified in +options+.
|
45
|
+
#
|
46
|
+
# A signature is an array of one or more parameter specifiers.
|
47
|
+
# A parameter specifier can be one of the following:
|
48
|
+
#
|
49
|
+
# * A symbol or string representing one of the Action Web Service base types.
|
50
|
+
# See ActionWebService::SignatureTypes for a canonical list of the base types.
|
51
|
+
# * The Class object of the parameter type
|
52
|
+
# * A single-element Array containing one of the two preceding items. This
|
53
|
+
# will cause Action Web Service to treat the parameter at that position
|
54
|
+
# as an array containing only values of the given type.
|
55
|
+
# * A Hash containing as key the name of the parameter, and as value
|
56
|
+
# one of the three preceding items
|
57
|
+
#
|
58
|
+
# If no method input parameter or method return value signatures are given,
|
59
|
+
# the method is assumed to take no parameters and/or return no values of
|
60
|
+
# interest, and any values that are received by the server will be
|
61
|
+
# discarded and ignored.
|
62
|
+
#
|
63
|
+
# Valid options:
|
64
|
+
# [<tt>:expects</tt>] Signature for the method input parameters
|
65
|
+
# [<tt>:returns</tt>] Signature for the method return value
|
66
|
+
# [<tt>:expects_and_returns</tt>] Signature for both input parameters and return value
|
67
|
+
def api_method(name, options={})
|
68
|
+
unless options.is_a?(Hash)
|
69
|
+
raise(ActionWebServiceError, "Expected a Hash for options")
|
70
|
+
end
|
71
|
+
validate_options([:expects, :returns, :expects_and_returns], options.keys)
|
72
|
+
if options[:expects_and_returns]
|
73
|
+
expects = options[:expects_and_returns]
|
74
|
+
returns = options[:expects_and_returns]
|
75
|
+
else
|
76
|
+
expects = options[:expects]
|
77
|
+
returns = options[:returns]
|
78
|
+
end
|
79
|
+
expects = canonical_signature(expects)
|
80
|
+
returns = canonical_signature(returns)
|
81
|
+
if expects
|
82
|
+
expects.each do |type|
|
83
|
+
type = type.element_type if type.is_a?(ArrayType)
|
84
|
+
if type.type_class.ancestors.include?(ActiveRecord::Base) && !allow_active_record_expects
|
85
|
+
raise(ActionWebServiceError, "ActiveRecord model classes not allowed in :expects")
|
86
|
+
end
|
87
|
+
end
|
88
|
+
end
|
89
|
+
name = name.to_sym
|
90
|
+
public_name = public_api_method_name(name)
|
91
|
+
method = Method.new(name, public_name, expects, returns)
|
92
|
+
write_inheritable_hash("api_methods", name => method)
|
93
|
+
write_inheritable_hash("api_public_method_names", public_name => name)
|
94
|
+
end
|
95
|
+
|
96
|
+
# Whether the given method name is a service method on this API
|
97
|
+
#
|
98
|
+
# class ProjectsApi < ActionWebService::API::Base
|
99
|
+
# api_method :getCount, :returns => [:int]
|
100
|
+
# end
|
101
|
+
#
|
102
|
+
# ProjectsApi.has_api_method?('GetCount') #=> false
|
103
|
+
# ProjectsApi.has_api_method?(:getCount) #=> true
|
104
|
+
def has_api_method?(name)
|
105
|
+
api_methods.has_key?(name)
|
106
|
+
end
|
107
|
+
|
108
|
+
# Whether the given public method name has a corresponding service method
|
109
|
+
# on this API
|
110
|
+
#
|
111
|
+
# class ProjectsApi < ActionWebService::API::Base
|
112
|
+
# api_method :getCount, :returns => [:int]
|
113
|
+
# end
|
114
|
+
#
|
115
|
+
# ProjectsApi.has_api_method?(:getCount) #=> false
|
116
|
+
# ProjectsApi.has_api_method?('GetCount') #=> true
|
117
|
+
def has_public_api_method?(public_name)
|
118
|
+
api_public_method_names.has_key?(public_name)
|
119
|
+
end
|
120
|
+
|
121
|
+
# The corresponding public method name for the given service method name
|
122
|
+
#
|
123
|
+
# ProjectsApi.public_api_method_name('GetCount') #=> "GetCount"
|
124
|
+
# ProjectsApi.public_api_method_name(:getCount) #=> "GetCount"
|
125
|
+
def public_api_method_name(name)
|
126
|
+
if inflect_names
|
127
|
+
name.to_s.camelize
|
128
|
+
else
|
129
|
+
name.to_s
|
130
|
+
end
|
131
|
+
end
|
132
|
+
|
133
|
+
# The corresponding service method name for the given public method name
|
134
|
+
#
|
135
|
+
# class ProjectsApi < ActionWebService::API::Base
|
136
|
+
# api_method :getCount, :returns => [:int]
|
137
|
+
# end
|
138
|
+
#
|
139
|
+
# ProjectsApi.api_method_name('GetCount') #=> :getCount
|
140
|
+
def api_method_name(public_name)
|
141
|
+
api_public_method_names[public_name]
|
142
|
+
end
|
143
|
+
|
144
|
+
# A Hash containing all service methods on this API, and their
|
145
|
+
# associated metadata.
|
146
|
+
#
|
147
|
+
# class ProjectsApi < ActionWebService::API::Base
|
148
|
+
# api_method :getCount, :returns => [:int]
|
149
|
+
# api_method :getCompletedCount, :returns => [:int]
|
150
|
+
# end
|
151
|
+
#
|
152
|
+
# ProjectsApi.api_methods #=>
|
153
|
+
# {:getCount=>#<ActionWebService::API::Method:0x24379d8 ...>,
|
154
|
+
# :getCompletedCount=>#<ActionWebService::API::Method:0x2437794 ...>}
|
155
|
+
# ProjectsApi.api_methods[:getCount].public_name #=> "GetCount"
|
156
|
+
def api_methods
|
157
|
+
read_inheritable_attribute("api_methods") || {}
|
158
|
+
end
|
159
|
+
|
160
|
+
# The Method instance for the given public API method name, if any
|
161
|
+
#
|
162
|
+
# class ProjectsApi < ActionWebService::API::Base
|
163
|
+
# api_method :getCount, :returns => [:int]
|
164
|
+
# api_method :getCompletedCount, :returns => [:int]
|
165
|
+
# end
|
166
|
+
#
|
167
|
+
# ProjectsApi.public_api_method_instance('GetCount') #=> <#<ActionWebService::API::Method:0x24379d8 ...>
|
168
|
+
# ProjectsApi.public_api_method_instance(:getCount) #=> nil
|
169
|
+
def public_api_method_instance(public_method_name)
|
170
|
+
api_method_instance(api_method_name(public_method_name))
|
171
|
+
end
|
172
|
+
|
173
|
+
# The Method instance for the given API method name, if any
|
174
|
+
#
|
175
|
+
# class ProjectsApi < ActionWebService::API::Base
|
176
|
+
# api_method :getCount, :returns => [:int]
|
177
|
+
# api_method :getCompletedCount, :returns => [:int]
|
178
|
+
# end
|
179
|
+
#
|
180
|
+
# ProjectsApi.api_method_instance(:getCount) #=> <ActionWebService::API::Method:0x24379d8 ...>
|
181
|
+
# ProjectsApi.api_method_instance('GetCount') #=> <ActionWebService::API::Method:0x24379d8 ...>
|
182
|
+
def api_method_instance(method_name)
|
183
|
+
api_methods[method_name]
|
184
|
+
end
|
185
|
+
|
186
|
+
# The Method instance for the default API method, if any
|
187
|
+
def default_api_method_instance
|
188
|
+
return nil unless name = default_api_method
|
189
|
+
instance = read_inheritable_attribute("default_api_method_instance")
|
190
|
+
if instance && instance.name == name
|
191
|
+
return instance
|
192
|
+
end
|
193
|
+
instance = Method.new(name, public_api_method_name(name), nil, nil)
|
194
|
+
write_inheritable_attribute("default_api_method_instance", instance)
|
195
|
+
instance
|
196
|
+
end
|
197
|
+
|
198
|
+
private
|
199
|
+
def api_public_method_names
|
200
|
+
read_inheritable_attribute("api_public_method_names") || {}
|
201
|
+
end
|
202
|
+
|
203
|
+
def validate_options(valid_option_keys, supplied_option_keys)
|
204
|
+
unknown_option_keys = supplied_option_keys - valid_option_keys
|
205
|
+
unless unknown_option_keys.empty?
|
206
|
+
raise(ActionWebServiceError, "Unknown options: #{unknown_option_keys}")
|
207
|
+
end
|
208
|
+
end
|
209
|
+
end
|
210
|
+
end
|
211
|
+
|
212
|
+
# Represents an API method and its associated metadata, and provides functionality
|
213
|
+
# to assist in commonly performed API method tasks.
|
214
|
+
class Method
|
215
|
+
attr :name
|
216
|
+
attr :public_name
|
217
|
+
attr :expects
|
218
|
+
attr :returns
|
219
|
+
|
220
|
+
def initialize(name, public_name, expects, returns)
|
221
|
+
@name = name
|
222
|
+
@public_name = public_name
|
223
|
+
@expects = expects
|
224
|
+
@returns = returns
|
225
|
+
@caster = ActionWebService::Casting::BaseCaster.new(self)
|
226
|
+
end
|
227
|
+
|
228
|
+
# The list of parameter names for this method
|
229
|
+
def param_names
|
230
|
+
return [] unless @expects
|
231
|
+
@expects.map{ |type| type.name }
|
232
|
+
end
|
233
|
+
|
234
|
+
# Casts a set of Ruby values into the expected Ruby values
|
235
|
+
def cast_expects(params)
|
236
|
+
@caster.cast_expects(params)
|
237
|
+
end
|
238
|
+
|
239
|
+
# Cast a Ruby return value into the expected Ruby value
|
240
|
+
def cast_returns(return_value)
|
241
|
+
@caster.cast_returns(return_value)
|
242
|
+
end
|
243
|
+
|
244
|
+
# Returns the index of the first expected parameter
|
245
|
+
# with the given name
|
246
|
+
def expects_index_of(param_name)
|
247
|
+
return -1 if @expects.nil?
|
248
|
+
(0..(@expects.length-1)).each do |i|
|
249
|
+
return i if @expects[i].name.to_s == param_name.to_s
|
250
|
+
end
|
251
|
+
-1
|
252
|
+
end
|
253
|
+
|
254
|
+
# Returns a hash keyed by parameter name for the given
|
255
|
+
# parameter list
|
256
|
+
def expects_to_hash(params)
|
257
|
+
return {} if @expects.nil?
|
258
|
+
h = {}
|
259
|
+
@expects.zip(params){ |type, param| h[type.name] = param }
|
260
|
+
h
|
261
|
+
end
|
262
|
+
|
263
|
+
# Backwards compatibility with previous API
|
264
|
+
def [](sig_type)
|
265
|
+
case sig_type
|
266
|
+
when :expects
|
267
|
+
@expects.map{|x| compat_signature_entry(x)}
|
268
|
+
when :returns
|
269
|
+
@returns.map{|x| compat_signature_entry(x)}
|
270
|
+
end
|
271
|
+
end
|
272
|
+
|
273
|
+
# String representation of this method
|
274
|
+
def to_s
|
275
|
+
fqn = ""
|
276
|
+
fqn << (@returns ? (@returns[0].human_name(false) + " ") : "void ")
|
277
|
+
fqn << "#{@public_name}("
|
278
|
+
fqn << @expects.map{ |p| p.human_name }.join(", ") if @expects
|
279
|
+
fqn << ")"
|
280
|
+
fqn
|
281
|
+
end
|
282
|
+
|
283
|
+
private
|
284
|
+
def compat_signature_entry(entry)
|
285
|
+
if entry.array?
|
286
|
+
[compat_signature_entry(entry.element_type)]
|
287
|
+
else
|
288
|
+
if entry.spec.is_a?(Hash)
|
289
|
+
{entry.spec.keys.first => entry.type_class}
|
290
|
+
else
|
291
|
+
entry.type_class
|
292
|
+
end
|
293
|
+
end
|
294
|
+
end
|
295
|
+
end
|
296
|
+
end
|
297
|
+
end
|