weary 0.1.0
Sign up to get free protection for your applications and to get access to all the features.
- data/.gitignore +2 -0
- data/LICENSE +20 -0
- data/README.md +14 -0
- data/Rakefile +51 -0
- data/VERSION +1 -0
- data/lib/weary.rb +150 -0
- data/lib/weary/exceptions.rb +3 -0
- data/lib/weary/request.rb +80 -0
- data/lib/weary/resource.rb +74 -0
- data/lib/weary/response.rb +109 -0
- data/spec/weary/request_spec.rb +26 -0
- data/spec/weary_spec.rb +140 -0
- data/weary.gemspec +52 -0
- metadata +68 -0
data/.gitignore
ADDED
data/LICENSE
ADDED
@@ -0,0 +1,20 @@
|
|
1
|
+
Copyright (c) 2009 Mark Wunsch
|
2
|
+
|
3
|
+
Permission is hereby granted, free of charge, to any person obtaining
|
4
|
+
a copy of this software and associated documentation files (the
|
5
|
+
"Software"), to deal in the Software without restriction, including
|
6
|
+
without limitation the rights to use, copy, modify, merge, publish,
|
7
|
+
distribute, sublicense, and/or sell copies of the Software, and to
|
8
|
+
permit persons to whom the Software is furnished to do so, subject to
|
9
|
+
the following conditions:
|
10
|
+
|
11
|
+
The above copyright notice and this permission notice shall be
|
12
|
+
included in all copies or substantial portions of the Software.
|
13
|
+
|
14
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
15
|
+
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
16
|
+
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
17
|
+
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
18
|
+
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
19
|
+
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
20
|
+
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
data/README.md
ADDED
@@ -0,0 +1,14 @@
|
|
1
|
+
# Weary
|
2
|
+
|
3
|
+
_The Weary need REST_
|
4
|
+
|
5
|
+
Weary is a tiny DSL for making the consumption of RESTful web services simple. It is the little brother to [HTTParty](http://github.com/jnunemaker/httparty/ "JNunemaker's HTTParty"). It provides a thin, gossamer-like layer over the Net/HTTP library.
|
6
|
+
|
7
|
+
The things it do:
|
8
|
+
|
9
|
+
+ Quickly build an interface to your favorite REST API.
|
10
|
+
+ Parse XML and JSON with the [Crack](http://github.com/jnunemaker/crack) library.
|
11
|
+
|
12
|
+
## Requirements
|
13
|
+
|
14
|
+
+ Crack
|
data/Rakefile
ADDED
@@ -0,0 +1,51 @@
|
|
1
|
+
require 'rubygems'
|
2
|
+
require 'spec/rake/spectask'
|
3
|
+
|
4
|
+
task :default => :spec
|
5
|
+
|
6
|
+
begin
|
7
|
+
require 'jeweler'
|
8
|
+
Jeweler::Tasks.new do |gemspec|
|
9
|
+
gemspec.name = "weary"
|
10
|
+
gemspec.summary = "A little DSL for consuming RESTful web services"
|
11
|
+
gemspec.email = "mark@markwunsch.com"
|
12
|
+
gemspec.homepage = "http://github.com/mwunsch/weary"
|
13
|
+
gemspec.description = "The Weary need REST: a tiny DSL that makes the consumption of RESTful web services simple."
|
14
|
+
gemspec.authors = "Mark Wunsch"
|
15
|
+
gemspec.has_rdoc = false
|
16
|
+
gemspec.rubyforge_project = "weary"
|
17
|
+
end
|
18
|
+
rescue LoadError
|
19
|
+
puts "Jeweler not available. Install it with: sudo gem install technicalpickles-jeweler -s http://gems.github.com"
|
20
|
+
end
|
21
|
+
|
22
|
+
begin
|
23
|
+
require 'rake/contrib/sshpublisher'
|
24
|
+
namespace :rubyforge do
|
25
|
+
|
26
|
+
desc "Release gem and RDoc documentation to RubyForge"
|
27
|
+
task :release => ["rubyforge:release:gem", "rubyforge:release:docs"]
|
28
|
+
|
29
|
+
namespace :release do
|
30
|
+
desc "Publish RDoc to RubyForge."
|
31
|
+
task :docs => [:rdoc] do
|
32
|
+
config = YAML.load(
|
33
|
+
File.read(File.expand_path('~/.rubyforge/user-config.yml'))
|
34
|
+
)
|
35
|
+
|
36
|
+
host = "#{config['username']}@rubyforge.org"
|
37
|
+
remote_dir = "/var/www/gforge-projects/weary/"
|
38
|
+
local_dir = 'rdoc'
|
39
|
+
|
40
|
+
Rake::SshDirPublisher.new(host, remote_dir, local_dir).upload
|
41
|
+
end
|
42
|
+
end
|
43
|
+
end
|
44
|
+
rescue LoadError
|
45
|
+
puts "Rake SshDirPublisher is unavailable or your rubyforge environment is not configured."
|
46
|
+
end
|
47
|
+
|
48
|
+
Spec::Rake::SpecTask.new do |t|
|
49
|
+
t.spec_files = FileList['spec/**/*_spec.rb']
|
50
|
+
t.spec_opts = ['--color','--format nested']
|
51
|
+
end
|
data/VERSION
ADDED
@@ -0,0 +1 @@
|
|
1
|
+
0.1.0
|
data/lib/weary.rb
ADDED
@@ -0,0 +1,150 @@
|
|
1
|
+
$:.unshift(File.dirname(__FILE__))
|
2
|
+
|
3
|
+
require 'uri'
|
4
|
+
require 'net/http'
|
5
|
+
require 'net/https'
|
6
|
+
|
7
|
+
require 'rubygems'
|
8
|
+
require 'crack'
|
9
|
+
|
10
|
+
gem 'nokogiri'
|
11
|
+
autoload :Yaml, 'yaml'
|
12
|
+
autoload :Nokogiri, 'nokogiri'
|
13
|
+
|
14
|
+
require 'weary/request'
|
15
|
+
require 'weary/response'
|
16
|
+
require 'weary/resource'
|
17
|
+
require 'weary/exceptions'
|
18
|
+
|
19
|
+
|
20
|
+
module Weary
|
21
|
+
|
22
|
+
Methods = { :get => [:get, :GET, /\bget\b/i],
|
23
|
+
:post => [:post, :POST, /\bpost\b/i],
|
24
|
+
:put => [:put, :PUT, /\bput\b/i],
|
25
|
+
:delete => [:delete, :del, :DELETE, :DEL, /\bdelete\b/i],
|
26
|
+
:head => [:head, :HEAD, /\bhead\b/i] }
|
27
|
+
|
28
|
+
# Weary::Query quickly performs a :get request on a URL and parses the request
|
29
|
+
def self.Query(url)
|
30
|
+
req = Weary::Request.new(url, :get).perform
|
31
|
+
req.parse
|
32
|
+
end
|
33
|
+
|
34
|
+
attr_reader :domain, :resources
|
35
|
+
|
36
|
+
def on_domain(domain)
|
37
|
+
parse_domain = URI.extract(domain)
|
38
|
+
raise ArgumentError, 'The domain must be a URL.' if parse_domain.empty?
|
39
|
+
@domain = parse_domain[0]
|
40
|
+
end
|
41
|
+
alias domain= on_domain
|
42
|
+
|
43
|
+
def as_format(format)
|
44
|
+
@default_format = format.to_sym
|
45
|
+
end
|
46
|
+
alias format= as_format
|
47
|
+
|
48
|
+
def construct_url(pattern)
|
49
|
+
@url_pattern = pattern.to_s
|
50
|
+
end
|
51
|
+
alias url= construct_url
|
52
|
+
|
53
|
+
def authenticates_with(username,password)
|
54
|
+
@username = username
|
55
|
+
@password = password
|
56
|
+
return nil
|
57
|
+
end
|
58
|
+
|
59
|
+
def declare_resource(resource, options={})
|
60
|
+
# available options:
|
61
|
+
# :via = get, post, etc. defaults to get
|
62
|
+
# :with = paramaters passed to body or query
|
63
|
+
# :requires = members of :with that must be in the action
|
64
|
+
# :authenticates = boolean; uses basic_authentication
|
65
|
+
# :url = a pattern
|
66
|
+
# :format = to set format, defaults to :json
|
67
|
+
# :no_follow = boolean; defaults to false. do not follow redirects
|
68
|
+
|
69
|
+
|
70
|
+
@resources ||= []
|
71
|
+
|
72
|
+
r = Weary::Resource.new(resource, set_defaults(options))
|
73
|
+
declaration = r.to_hash
|
74
|
+
|
75
|
+
@resources << declaration
|
76
|
+
|
77
|
+
craft_methods(r)
|
78
|
+
return declaration
|
79
|
+
end
|
80
|
+
|
81
|
+
def get(resource,options={})
|
82
|
+
options[:via] = :get
|
83
|
+
declare_resource(resource,options)
|
84
|
+
end
|
85
|
+
|
86
|
+
def post(resource,options={})
|
87
|
+
options[:via] = :post
|
88
|
+
declare_resource(resource,options)
|
89
|
+
end
|
90
|
+
|
91
|
+
def put(resource,options={})
|
92
|
+
options[:via] = :put
|
93
|
+
declare_resource(resource,options)
|
94
|
+
end
|
95
|
+
|
96
|
+
def delete(resource,options={})
|
97
|
+
options[:via] = :delete
|
98
|
+
declare_resource(resource,options)
|
99
|
+
end
|
100
|
+
|
101
|
+
private
|
102
|
+
def set_defaults(hash)
|
103
|
+
hash[:domain] = @domain
|
104
|
+
hash[:via] ||= :get
|
105
|
+
hash[:with] ||= []
|
106
|
+
hash[:with] = hash[:with] | hash[:requires] unless hash[:requires].nil?
|
107
|
+
hash[:format] ||= (@default_format || :json)
|
108
|
+
hash[:authenticates] ||= false
|
109
|
+
hash[:authenticates] = false if hash[:authenticates] == "false"
|
110
|
+
if hash[:authenticates]
|
111
|
+
raise StandardError, "Can not authenticate unless username and password are defined" unless (@username && @password)
|
112
|
+
end
|
113
|
+
hash[:url] ||= (@url_pattern || "<domain><resource>.<format>")
|
114
|
+
hash[:no_follow] ||= false
|
115
|
+
return hash
|
116
|
+
end
|
117
|
+
|
118
|
+
def craft_methods(resource)
|
119
|
+
code = %Q{
|
120
|
+
def #{resource.name}(params={})
|
121
|
+
options ||= {}
|
122
|
+
url = "#{resource.url}"
|
123
|
+
}
|
124
|
+
unless resource.requires.nil?
|
125
|
+
resource.requires.each do |required|
|
126
|
+
code << %Q{raise ArgumentError, "This resource requires parameter: ':#{required}'" unless params.has_key?(:#{required}) \n}
|
127
|
+
end
|
128
|
+
end
|
129
|
+
unless resource.with.nil?
|
130
|
+
with = %Q\[#{resource.with.collect {|x| ":#{x}"}.join(',')}]\
|
131
|
+
code << "unnecessary = params.keys - #{with} \n"
|
132
|
+
code << "unnecessary.each { |x| params.delete(x) } \n"
|
133
|
+
end
|
134
|
+
if resource.via == (:post || :put)
|
135
|
+
code << "options[:body] = params unless params.empty? \n"
|
136
|
+
else
|
137
|
+
code << "options[:query] = params unless params.empty? \n"
|
138
|
+
code << %Q{url << "?" + options[:query].to_params unless options[:query].nil? \n}
|
139
|
+
end
|
140
|
+
if resource.authenticates?
|
141
|
+
code << %Q{options[:basic_auth] = {:username => "#{@username}", :password => "#{@password}"} \n}
|
142
|
+
end
|
143
|
+
code << %Q{
|
144
|
+
Weary::Request.new(url, :#{resource.via}, options).perform
|
145
|
+
end
|
146
|
+
}
|
147
|
+
class_eval code
|
148
|
+
end
|
149
|
+
|
150
|
+
end
|
@@ -0,0 +1,80 @@
|
|
1
|
+
module Weary
|
2
|
+
class Request
|
3
|
+
|
4
|
+
attr_reader :uri
|
5
|
+
attr_accessor :options
|
6
|
+
|
7
|
+
def initialize(url, http_verb= :get, options={})
|
8
|
+
self.method = http_verb
|
9
|
+
self.uri = url
|
10
|
+
self.options = options
|
11
|
+
end
|
12
|
+
|
13
|
+
def uri=(url)
|
14
|
+
@uri = URI.parse(url)
|
15
|
+
end
|
16
|
+
|
17
|
+
def method=(http_verb)
|
18
|
+
@http_verb = case http_verb
|
19
|
+
when *Methods[:get]
|
20
|
+
:get
|
21
|
+
when *Methods[:post]
|
22
|
+
:post
|
23
|
+
when *Methods[:put]
|
24
|
+
:put
|
25
|
+
when *Methods[:delete]
|
26
|
+
:delete
|
27
|
+
when *Methods[:head]
|
28
|
+
:head
|
29
|
+
else
|
30
|
+
raise ArgumentError, "Only GET, POST, PUT, DELETE, and HEAD methods are supported"
|
31
|
+
end
|
32
|
+
end
|
33
|
+
|
34
|
+
def method
|
35
|
+
@http_verb
|
36
|
+
end
|
37
|
+
|
38
|
+
def perform
|
39
|
+
req = http.request(request)
|
40
|
+
response = Response.new(req, @http_verb)
|
41
|
+
unless options[:no_follow]
|
42
|
+
if response.redirected?
|
43
|
+
response.follow_redirect
|
44
|
+
else
|
45
|
+
response
|
46
|
+
end
|
47
|
+
else
|
48
|
+
response
|
49
|
+
end
|
50
|
+
end
|
51
|
+
alias make perform
|
52
|
+
|
53
|
+
private
|
54
|
+
def http
|
55
|
+
connection = Net::HTTP.new(@uri.host, @uri.port)
|
56
|
+
connection.use_ssl = @uri.is_a?(URI::HTTPS)
|
57
|
+
connection.verify_mode = OpenSSL::SSL::VERIFY_NONE if connection.use_ssl
|
58
|
+
connection
|
59
|
+
end
|
60
|
+
|
61
|
+
def request
|
62
|
+
prepare = case @http_verb
|
63
|
+
when :get
|
64
|
+
Net::HTTP::Get.new(@uri.request_uri)
|
65
|
+
when :post
|
66
|
+
Net::HTTP::Post.new(@uri.request_uri)
|
67
|
+
when :put
|
68
|
+
Net::HTTP::Put.new(@uri.request_uri)
|
69
|
+
when :delete
|
70
|
+
Net::HTTP::Delete.new(@uri.request_uri)
|
71
|
+
when :head
|
72
|
+
Net::HTTP::Head.new(@uri.request_uri)
|
73
|
+
end
|
74
|
+
prepare.body = options[:body].is_a?(Hash) ? options[:body].to_params : options[:body] if options[:body]
|
75
|
+
prepare.basic_auth(options[:basic_auth][:username], options[:basic_auth][:password]) if options[:basic_auth]
|
76
|
+
prepare
|
77
|
+
end
|
78
|
+
|
79
|
+
end
|
80
|
+
end
|
@@ -0,0 +1,74 @@
|
|
1
|
+
module Weary
|
2
|
+
class Resource
|
3
|
+
attr_reader :name, :with, :requires, :via, :format, :url
|
4
|
+
|
5
|
+
def initialize(name,options={})
|
6
|
+
@domain = options[:domain]
|
7
|
+
self.name = name
|
8
|
+
self.via = options[:via]
|
9
|
+
self.with = options[:with]
|
10
|
+
self.requires = options[:requires]
|
11
|
+
self.format = options[:format]
|
12
|
+
self.url = options[:url]
|
13
|
+
@authenticates = (options[:authenticates] != false)
|
14
|
+
@follows = (options[:no_follow] == false)
|
15
|
+
end
|
16
|
+
|
17
|
+
def name=(resource)
|
18
|
+
@name = resource.to_s
|
19
|
+
end
|
20
|
+
|
21
|
+
def via=(verb)
|
22
|
+
@via = verb
|
23
|
+
end
|
24
|
+
|
25
|
+
def with=(params)
|
26
|
+
if params.empty?
|
27
|
+
@with = nil
|
28
|
+
else
|
29
|
+
@with = params.collect {|x| x.to_sym }
|
30
|
+
end
|
31
|
+
end
|
32
|
+
|
33
|
+
def url=(pattern)
|
34
|
+
if pattern.index("<domain>")
|
35
|
+
raise StandardError, "Domain flag found but the domain is not defined" if @domain.nil?
|
36
|
+
pattern = pattern.gsub("<domain>", @domain)
|
37
|
+
end
|
38
|
+
pattern = pattern.gsub("<resource>", @name)
|
39
|
+
pattern = pattern.gsub("<format>", @format.to_s)
|
40
|
+
@url = pattern
|
41
|
+
end
|
42
|
+
|
43
|
+
def requires=(params)
|
44
|
+
if (params.nil? || params.empty?)
|
45
|
+
@requires = nil
|
46
|
+
else
|
47
|
+
@requires = params
|
48
|
+
end
|
49
|
+
end
|
50
|
+
|
51
|
+
def format=(type)
|
52
|
+
@format = type
|
53
|
+
end
|
54
|
+
|
55
|
+
def authenticates?
|
56
|
+
@authenticates
|
57
|
+
end
|
58
|
+
|
59
|
+
def follows_redirects?
|
60
|
+
@follows
|
61
|
+
end
|
62
|
+
|
63
|
+
def to_hash
|
64
|
+
{@name.to_sym => {:via => @via,
|
65
|
+
:with => @with,
|
66
|
+
:requires => @requires,
|
67
|
+
:authenticates => authenticates?,
|
68
|
+
:format => @format,
|
69
|
+
:url => @url},
|
70
|
+
:no_follow => !follows_redirects?}
|
71
|
+
end
|
72
|
+
|
73
|
+
end
|
74
|
+
end
|
@@ -0,0 +1,109 @@
|
|
1
|
+
module Weary
|
2
|
+
class Response
|
3
|
+
|
4
|
+
attr_reader :raw, :method, :code, :message, :header, :content_type, :cookie, :body
|
5
|
+
alias mime_type content_type
|
6
|
+
|
7
|
+
def initialize(http_response, http_method)
|
8
|
+
raise ArgumentError, "Must be a Net::HTTPResponse" unless http_response.is_a?(Net::HTTPResponse)
|
9
|
+
@raw = http_response
|
10
|
+
@method = http_method
|
11
|
+
@code = http_response.code.to_i
|
12
|
+
@message = http_response.message
|
13
|
+
@header = http_response.to_hash
|
14
|
+
@content_type = http_response.content_type
|
15
|
+
@cookie = http_response['Set-Cookie']
|
16
|
+
@body = http_response.body
|
17
|
+
self.format = http_response.content_type
|
18
|
+
end
|
19
|
+
|
20
|
+
def redirected?
|
21
|
+
@raw.is_a?(Net::HTTPRedirection)
|
22
|
+
end
|
23
|
+
|
24
|
+
def success?
|
25
|
+
(200..299).include?(@code)
|
26
|
+
end
|
27
|
+
|
28
|
+
def format=(type)
|
29
|
+
@format = case type
|
30
|
+
when 'text/xml', 'application/xml'
|
31
|
+
:xml
|
32
|
+
when 'application/json', 'text/json', 'application/javascript', 'text/javascript'
|
33
|
+
:json
|
34
|
+
when 'text/html'
|
35
|
+
:html
|
36
|
+
when 'application/x-yaml', 'text/yaml'
|
37
|
+
:yaml
|
38
|
+
when 'text/plain'
|
39
|
+
:plain
|
40
|
+
else
|
41
|
+
nil
|
42
|
+
end
|
43
|
+
end
|
44
|
+
|
45
|
+
def format
|
46
|
+
@format
|
47
|
+
end
|
48
|
+
|
49
|
+
def follow_redirect
|
50
|
+
if redirected?
|
51
|
+
Request.new(@raw['location'], @method).perform
|
52
|
+
else
|
53
|
+
nil
|
54
|
+
end
|
55
|
+
end
|
56
|
+
|
57
|
+
def parse
|
58
|
+
raise StandardError, "The Response has no body. #{@method.to_s.upcase} request sent." unless @body
|
59
|
+
handle_errors
|
60
|
+
case @format
|
61
|
+
when :xml, :html
|
62
|
+
Crack::XML.parse @body
|
63
|
+
when :json
|
64
|
+
Crack::JSON.parse @body
|
65
|
+
when :yaml
|
66
|
+
YAML::load @body
|
67
|
+
else
|
68
|
+
@body
|
69
|
+
end
|
70
|
+
end
|
71
|
+
|
72
|
+
def search(selector)
|
73
|
+
raise ArgumentError, "Search can only be used with an XML or HTML document." unless @format != (:xml || :html)
|
74
|
+
doc = Nokogiri.parse(@body)
|
75
|
+
doc.search(selector)
|
76
|
+
end
|
77
|
+
|
78
|
+
private
|
79
|
+
def handle_errors
|
80
|
+
case @code
|
81
|
+
when 301,302
|
82
|
+
raise HTTPError, "#{@message} to #{@raw['location']}"
|
83
|
+
when 200...400
|
84
|
+
return
|
85
|
+
when 400
|
86
|
+
raise HTTPError, "Failed with #{@code}: #{@message}"
|
87
|
+
when 401
|
88
|
+
raise HTTPError, "Failed with #{@code}: #{@message}"
|
89
|
+
when 403
|
90
|
+
raise HTTPError, "Failed with #{@code}: #{@message}"
|
91
|
+
when 404
|
92
|
+
raise HTTPError, "Failed with #{@code}: #{@message}"
|
93
|
+
when 405
|
94
|
+
raise HTTPError, "Failed with #{@code}: #{@message}"
|
95
|
+
when 409
|
96
|
+
raise HTTPError, "Failed with #{@code}: #{@message}"
|
97
|
+
when 422
|
98
|
+
raise HTTPError, "Failed with #{@code}: #{@message}"
|
99
|
+
when 401...500
|
100
|
+
raise HTTPError, "Failed with #{@code}: #{@message}"
|
101
|
+
when 500...600
|
102
|
+
raise HTTPError, "Failed with #{@code}: #{@message}"
|
103
|
+
else
|
104
|
+
raise HTTPError, "Unknown response code: #{@code}"
|
105
|
+
end
|
106
|
+
end
|
107
|
+
|
108
|
+
end
|
109
|
+
end
|
@@ -0,0 +1,26 @@
|
|
1
|
+
require 'rubygems'
|
2
|
+
gem 'rspec'
|
3
|
+
require 'spec'
|
4
|
+
require File.join(File.dirname(__FILE__), '../..', 'lib', 'weary')
|
5
|
+
|
6
|
+
describe Weary::Request do
|
7
|
+
|
8
|
+
it "should contain a url" do
|
9
|
+
test = Weary::Request.new("http://google.com")
|
10
|
+
test.uri.is_a?(URI).should == true
|
11
|
+
end
|
12
|
+
|
13
|
+
it "should parse the http method" do
|
14
|
+
test = Weary::Request.new("http://google.com", "POST")
|
15
|
+
test.method.should == :post
|
16
|
+
end
|
17
|
+
|
18
|
+
it "should craft a Net/HTTP Request" do
|
19
|
+
test = Weary::Request.new("http://google.com").send :http
|
20
|
+
test.class.should == Net::HTTP
|
21
|
+
end
|
22
|
+
|
23
|
+
it "should follow redirects" do
|
24
|
+
pending "Not sure how to test this"
|
25
|
+
end
|
26
|
+
end
|
data/spec/weary_spec.rb
ADDED
@@ -0,0 +1,140 @@
|
|
1
|
+
require 'rubygems'
|
2
|
+
gem 'rspec'
|
3
|
+
require 'spec'
|
4
|
+
require File.join(File.dirname(__FILE__), '..', 'lib', 'weary')
|
5
|
+
|
6
|
+
describe Weary do
|
7
|
+
before do
|
8
|
+
@test = Class.new
|
9
|
+
@test.instance_eval { extend Weary }
|
10
|
+
end
|
11
|
+
|
12
|
+
describe "default domain" do
|
13
|
+
it 'should be set with a url' do
|
14
|
+
@test.on_domain("http://twitter.com/")
|
15
|
+
@test.domain.should == "http://twitter.com/"
|
16
|
+
end
|
17
|
+
|
18
|
+
it "should also be set by it's alias" do
|
19
|
+
@test.domain = "http://twitter.com/"
|
20
|
+
@test.domain.should == "http://twitter.com/"
|
21
|
+
end
|
22
|
+
|
23
|
+
it 'should raise an exception when a url is not present' do
|
24
|
+
lambda { @test.on_domain("foobar") }.should raise_error
|
25
|
+
end
|
26
|
+
|
27
|
+
it 'should only take the first url given' do
|
28
|
+
@test.on_domain("with http://google.com/ and http://yahoo.com/")
|
29
|
+
@test.domain.should == "http://google.com/"
|
30
|
+
end
|
31
|
+
end
|
32
|
+
|
33
|
+
describe "default format" do
|
34
|
+
it 'can be set' do
|
35
|
+
@test.as_format("xml")
|
36
|
+
@test.instance_variable_defined?(:@default_format).should == true
|
37
|
+
end
|
38
|
+
|
39
|
+
it "should also be set by it's alias" do
|
40
|
+
@test.format = "xml"
|
41
|
+
@test.instance_variable_defined?(:@default_format).should == true
|
42
|
+
end
|
43
|
+
|
44
|
+
it 'should be a symbol' do
|
45
|
+
@test.as_format("xml")
|
46
|
+
@test.instance_variable_get(:@default_format).class.should == Symbol
|
47
|
+
end
|
48
|
+
end
|
49
|
+
|
50
|
+
describe "default url pattern" do
|
51
|
+
it 'can be set' do
|
52
|
+
@test.construct_url("<domain><resource>.<format>")
|
53
|
+
@test.instance_variable_defined?(:@url_pattern).should == true
|
54
|
+
end
|
55
|
+
|
56
|
+
it "should also be set by it's alias" do
|
57
|
+
@test.url = "<domain><resource>.<format>"
|
58
|
+
@test.instance_variable_defined?(:@url_pattern).should == true
|
59
|
+
end
|
60
|
+
|
61
|
+
it 'should be a string' do
|
62
|
+
@test.construct_url(123)
|
63
|
+
@test.instance_variable_get(:@url_pattern).class.should == String
|
64
|
+
end
|
65
|
+
end
|
66
|
+
|
67
|
+
describe "basic authentication credentials" do
|
68
|
+
it "should accept a username and password" do
|
69
|
+
@test.authenticates_with("foo","bar")
|
70
|
+
@test.instance_variable_get(:@username).should == "foo"
|
71
|
+
@test.instance_variable_get(:@password).should == "bar"
|
72
|
+
end
|
73
|
+
end
|
74
|
+
|
75
|
+
describe "resource declaration" do
|
76
|
+
before do
|
77
|
+
@test.domain = "http://twitter.com/"
|
78
|
+
end
|
79
|
+
|
80
|
+
it "should adds a new resource" do
|
81
|
+
@test.declare_resource("resource")
|
82
|
+
@test.resources[0].has_key?(:resource).should == true
|
83
|
+
end
|
84
|
+
|
85
|
+
it "should default to a GET request" do
|
86
|
+
@test.declare_resource("resource")[:resource][:via].should == :get
|
87
|
+
end
|
88
|
+
|
89
|
+
it "should default to JSON if no format is defined" do
|
90
|
+
@test.declare_resource("resource")[:resource][:format].should == :json
|
91
|
+
end
|
92
|
+
|
93
|
+
it "should use the declared format, if a specific format is not defined" do
|
94
|
+
@test.format = :xml
|
95
|
+
@test.declare_resource("resource")[:resource][:format].should == :xml
|
96
|
+
end
|
97
|
+
|
98
|
+
it "should override the default format with it's own format" do
|
99
|
+
@test.format = :xml
|
100
|
+
@test.declare_resource("resource",{:format => :yaml})[:resource][:format].should == :yaml
|
101
|
+
end
|
102
|
+
|
103
|
+
it "should form a url if there is a default pattern" do
|
104
|
+
@test.declare_resource("resource")[:resource][:url].should == "http://twitter.com/resource.json"
|
105
|
+
end
|
106
|
+
|
107
|
+
it "should override the default pattern with it's own url" do
|
108
|
+
@test.declare_resource("resource",{:url => "http://foobar.com/<resource>"})[:resource][:url].should == "http://foobar.com/resource"
|
109
|
+
end
|
110
|
+
|
111
|
+
it "should be able to contain a set of allowed parameters" do
|
112
|
+
@test.declare_resource("resource",{:with => [:id]})[:resource][:with].empty?.should == false
|
113
|
+
end
|
114
|
+
|
115
|
+
it "should be able to contain a set of required parameters" do
|
116
|
+
@test.declare_resource("resource",{:requires => [:id]})[:resource][:requires].empty?.should == false
|
117
|
+
end
|
118
|
+
|
119
|
+
it "should merge required parameters into allowed parameters" do
|
120
|
+
@test.declare_resource("resource",{:requires => [:id]})[:resource][:with].empty?.should == false
|
121
|
+
end
|
122
|
+
|
123
|
+
it "should authenticate with username and password" do
|
124
|
+
@test.authenticates_with("foo","bar")
|
125
|
+
@test.declare_resource("resource",{:authenticates => true})[:resource][:authenticates].should == true
|
126
|
+
end
|
127
|
+
|
128
|
+
it "should raise an exception if authentication is required but no credentials are supplied" do
|
129
|
+
lambda do
|
130
|
+
@test.declare_resource("resource",{:authenticates => true})
|
131
|
+
end.should raise_error
|
132
|
+
end
|
133
|
+
|
134
|
+
it "should create a method for an instantiated object" do
|
135
|
+
@test.declare_resource("resource")
|
136
|
+
@test.public_method_defined?(:resource).should == true
|
137
|
+
end
|
138
|
+
|
139
|
+
end
|
140
|
+
end
|
data/weary.gemspec
ADDED
@@ -0,0 +1,52 @@
|
|
1
|
+
# -*- encoding: utf-8 -*-
|
2
|
+
|
3
|
+
Gem::Specification.new do |s|
|
4
|
+
s.name = %q{weary}
|
5
|
+
s.version = "0.1.0"
|
6
|
+
|
7
|
+
s.required_rubygems_version = Gem::Requirement.new(">= 0") if s.respond_to? :required_rubygems_version=
|
8
|
+
s.authors = ["Mark Wunsch"]
|
9
|
+
s.date = %q{2009-06-08}
|
10
|
+
s.description = %q{The Weary need REST: a tiny DSL that makes the consumption of RESTful web services simple.}
|
11
|
+
s.email = %q{mark@markwunsch.com}
|
12
|
+
s.extra_rdoc_files = [
|
13
|
+
"LICENSE",
|
14
|
+
"README.md"
|
15
|
+
]
|
16
|
+
s.files = [
|
17
|
+
".gitignore",
|
18
|
+
"LICENSE",
|
19
|
+
"README.md",
|
20
|
+
"Rakefile",
|
21
|
+
"VERSION",
|
22
|
+
"lib/weary.rb",
|
23
|
+
"lib/weary/exceptions.rb",
|
24
|
+
"lib/weary/request.rb",
|
25
|
+
"lib/weary/resource.rb",
|
26
|
+
"lib/weary/response.rb",
|
27
|
+
"spec/weary/request_spec.rb",
|
28
|
+
"spec/weary_spec.rb",
|
29
|
+
"weary.gemspec"
|
30
|
+
]
|
31
|
+
s.homepage = %q{http://github.com/mwunsch/weary}
|
32
|
+
s.rdoc_options = ["--charset=UTF-8"]
|
33
|
+
s.require_paths = ["lib"]
|
34
|
+
s.rubyforge_project = %q{weary}
|
35
|
+
s.rubygems_version = %q{1.3.1}
|
36
|
+
s.summary = %q{A little DSL for consuming RESTful web services}
|
37
|
+
s.test_files = [
|
38
|
+
"spec/weary/request_spec.rb",
|
39
|
+
"spec/weary_spec.rb",
|
40
|
+
"examples/test.rb"
|
41
|
+
]
|
42
|
+
|
43
|
+
if s.respond_to? :specification_version then
|
44
|
+
current_version = Gem::Specification::CURRENT_SPECIFICATION_VERSION
|
45
|
+
s.specification_version = 2
|
46
|
+
|
47
|
+
if Gem::Version.new(Gem::RubyGemsVersion) >= Gem::Version.new('1.2.0') then
|
48
|
+
else
|
49
|
+
end
|
50
|
+
else
|
51
|
+
end
|
52
|
+
end
|
metadata
ADDED
@@ -0,0 +1,68 @@
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
2
|
+
name: weary
|
3
|
+
version: !ruby/object:Gem::Version
|
4
|
+
version: 0.1.0
|
5
|
+
platform: ruby
|
6
|
+
authors:
|
7
|
+
- Mark Wunsch
|
8
|
+
autorequire:
|
9
|
+
bindir: bin
|
10
|
+
cert_chain: []
|
11
|
+
|
12
|
+
date: 2009-06-08 00:00:00 -04:00
|
13
|
+
default_executable:
|
14
|
+
dependencies: []
|
15
|
+
|
16
|
+
description: "The Weary need REST: a tiny DSL that makes the consumption of RESTful web services simple."
|
17
|
+
email: mark@markwunsch.com
|
18
|
+
executables: []
|
19
|
+
|
20
|
+
extensions: []
|
21
|
+
|
22
|
+
extra_rdoc_files:
|
23
|
+
- LICENSE
|
24
|
+
- README.md
|
25
|
+
files:
|
26
|
+
- .gitignore
|
27
|
+
- LICENSE
|
28
|
+
- README.md
|
29
|
+
- Rakefile
|
30
|
+
- VERSION
|
31
|
+
- lib/weary.rb
|
32
|
+
- lib/weary/exceptions.rb
|
33
|
+
- lib/weary/request.rb
|
34
|
+
- lib/weary/resource.rb
|
35
|
+
- lib/weary/response.rb
|
36
|
+
- spec/weary/request_spec.rb
|
37
|
+
- spec/weary_spec.rb
|
38
|
+
- weary.gemspec
|
39
|
+
has_rdoc: false
|
40
|
+
homepage: http://github.com/mwunsch/weary
|
41
|
+
post_install_message:
|
42
|
+
rdoc_options:
|
43
|
+
- --charset=UTF-8
|
44
|
+
require_paths:
|
45
|
+
- lib
|
46
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
47
|
+
requirements:
|
48
|
+
- - ">="
|
49
|
+
- !ruby/object:Gem::Version
|
50
|
+
version: "0"
|
51
|
+
version:
|
52
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
53
|
+
requirements:
|
54
|
+
- - ">="
|
55
|
+
- !ruby/object:Gem::Version
|
56
|
+
version: "0"
|
57
|
+
version:
|
58
|
+
requirements: []
|
59
|
+
|
60
|
+
rubyforge_project: weary
|
61
|
+
rubygems_version: 1.3.1
|
62
|
+
signing_key:
|
63
|
+
specification_version: 2
|
64
|
+
summary: A little DSL for consuming RESTful web services
|
65
|
+
test_files:
|
66
|
+
- spec/weary/request_spec.rb
|
67
|
+
- spec/weary_spec.rb
|
68
|
+
- examples/test.rb
|