zephyr 1.0.0
Sign up to get free protection for your applications and to get access to all the features.
- data/.document +5 -0
- data/Gemfile +10 -0
- data/Gemfile.lock +21 -0
- data/LICENSE.txt +20 -0
- data/README.md +1 -0
- data/Rakefile +53 -0
- data/VERSION +1 -0
- data/lib/zephyr.rb +350 -0
- data/test/helper.rb +18 -0
- data/test/test_zephyr.rb +7 -0
- data/zephyr.gemspec +56 -0
- metadata +125 -0
data/.document
ADDED
data/Gemfile
ADDED
data/Gemfile.lock
ADDED
@@ -0,0 +1,21 @@
|
|
1
|
+
GEM
|
2
|
+
remote: http://rubygems.org/
|
3
|
+
specs:
|
4
|
+
git (1.2.5)
|
5
|
+
jeweler (1.6.4)
|
6
|
+
bundler (~> 1.0)
|
7
|
+
git (>= 1.2.5)
|
8
|
+
rake
|
9
|
+
mime-types (1.16)
|
10
|
+
rake (0.9.2)
|
11
|
+
typhoeus (0.2.4)
|
12
|
+
mime-types
|
13
|
+
mime-types
|
14
|
+
|
15
|
+
PLATFORMS
|
16
|
+
ruby
|
17
|
+
|
18
|
+
DEPENDENCIES
|
19
|
+
bundler (~> 1.0.0)
|
20
|
+
jeweler (~> 1.6.4)
|
21
|
+
typhoeus (>= 0.2.4)
|
data/LICENSE.txt
ADDED
@@ -0,0 +1,20 @@
|
|
1
|
+
Copyright (c) 2011 Matt Knopp
|
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 @@
|
|
1
|
+
## Simple HTTP client using Typheous, derived from the Riak client
|
data/Rakefile
ADDED
@@ -0,0 +1,53 @@
|
|
1
|
+
# encoding: utf-8
|
2
|
+
|
3
|
+
require 'rubygems'
|
4
|
+
require 'bundler'
|
5
|
+
begin
|
6
|
+
Bundler.setup(:default, :development)
|
7
|
+
rescue Bundler::BundlerError => e
|
8
|
+
$stderr.puts e.message
|
9
|
+
$stderr.puts "Run `bundle install` to install missing gems"
|
10
|
+
exit e.status_code
|
11
|
+
end
|
12
|
+
require 'rake'
|
13
|
+
|
14
|
+
require 'jeweler'
|
15
|
+
Jeweler::Tasks.new do |gem|
|
16
|
+
# gem is a Gem::Specification... see http://docs.rubygems.org/read/chapter/20 for more options
|
17
|
+
gem.name = "zephyr"
|
18
|
+
gem.homepage = "http://github.com/mhat/zephyr"
|
19
|
+
gem.license = "MIT"
|
20
|
+
gem.summary = %Q{Simple HTTP client using Typheous, derived from the Riak client}
|
21
|
+
gem.description = %Q{Simple HTTP client using Typheous, derived from the Riak client}
|
22
|
+
gem.email = "matt.knopp@gmail.com"
|
23
|
+
gem.authors = ["Matt Knopp"]
|
24
|
+
# dependencies defined in Gemfile
|
25
|
+
end
|
26
|
+
Jeweler::RubygemsDotOrgTasks.new
|
27
|
+
|
28
|
+
require 'rake/testtask'
|
29
|
+
Rake::TestTask.new(:test) do |test|
|
30
|
+
test.libs << 'lib' << 'test'
|
31
|
+
test.pattern = 'test/**/test_*.rb'
|
32
|
+
test.verbose = true
|
33
|
+
end
|
34
|
+
|
35
|
+
##require 'rcov/rcovtask'
|
36
|
+
##Rcov::RcovTask.new do |test|
|
37
|
+
## test.libs << 'test'
|
38
|
+
## test.pattern = 'test/**/test_*.rb'
|
39
|
+
## test.verbose = true
|
40
|
+
## test.rcov_opts << '--exclude "gems/*"'
|
41
|
+
##end
|
42
|
+
|
43
|
+
task :default => :test
|
44
|
+
|
45
|
+
require 'rake/rdoctask'
|
46
|
+
Rake::RDocTask.new do |rdoc|
|
47
|
+
version = File.exist?('VERSION') ? File.read('VERSION') : ""
|
48
|
+
|
49
|
+
rdoc.rdoc_dir = 'rdoc'
|
50
|
+
rdoc.title = "zephyr #{version}"
|
51
|
+
rdoc.rdoc_files.include('README*')
|
52
|
+
rdoc.rdoc_files.include('lib/**/*.rb')
|
53
|
+
end
|
data/VERSION
ADDED
@@ -0,0 +1 @@
|
|
1
|
+
1.0.0
|
data/lib/zephyr.rb
ADDED
@@ -0,0 +1,350 @@
|
|
1
|
+
# Stolen with a fair bit of modification from the riak-client gem, which is
|
2
|
+
# Copyright 2010 Sean Cribbs, Sonian Inc., and Basho Technologies, Inc.
|
3
|
+
#
|
4
|
+
# Licensed under the Apache License, Version 2.0 (the "License");
|
5
|
+
# you may not use this file except in compliance with the License.
|
6
|
+
# You may obtain a copy of the License at
|
7
|
+
#
|
8
|
+
# http://www.apache.org/licenses/LICENSE-2.0
|
9
|
+
#
|
10
|
+
# Unless required by applicable law or agreed to in writing, software
|
11
|
+
# distributed under the License is distributed on an "AS IS" BASIS,
|
12
|
+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
13
|
+
# See the License for the specific language governing permissions and
|
14
|
+
# limitations under the License.
|
15
|
+
|
16
|
+
# Splits headers into < 8KB chunks
|
17
|
+
# @private
|
18
|
+
module Net::HTTPHeader
|
19
|
+
def each_capitalized
|
20
|
+
# 1.9 check
|
21
|
+
respond_to?(:enum_for) and (block_given? or return enum_for(__method__))
|
22
|
+
@header.each do |k,v|
|
23
|
+
base_length = "#{k}: \r\n".length
|
24
|
+
values = v.map { |i| i.to_s.split(', ') }.flatten
|
25
|
+
while !values.empty?
|
26
|
+
current_line = ""
|
27
|
+
while values.first && current_line.length + base_length + values.first.length + 2 < 8192
|
28
|
+
val = values.shift.strip
|
29
|
+
current_line += current_line.empty? ? val : ", #{val}"
|
30
|
+
end
|
31
|
+
yield capitalize(k), current_line
|
32
|
+
end
|
33
|
+
end
|
34
|
+
end
|
35
|
+
end
|
36
|
+
|
37
|
+
# A simple front-end for doing HTTP requests quickly and simply.
|
38
|
+
class Zephyr
|
39
|
+
def initialize(root_uri = '')
|
40
|
+
@root_uri = URI.parse(root_uri.to_s).freeze
|
41
|
+
end
|
42
|
+
|
43
|
+
def default_headers
|
44
|
+
{
|
45
|
+
'Accept' => 'application/json;q=0.7, */*;q=0.5',
|
46
|
+
'User-Agent' => 'lib-http.rb',
|
47
|
+
}
|
48
|
+
end
|
49
|
+
|
50
|
+
# Performs a HEAD request to the specified resource.
|
51
|
+
#
|
52
|
+
# A request to /users/#{@user.id}/things?q=woof with an Accept header of
|
53
|
+
# "text/plain" which is expecting a 200 OK within 50ms
|
54
|
+
#
|
55
|
+
# http.head(200, 50, ["users", @user.id, "things", {"q" => "woof"}], "Accept" => "text/plain")
|
56
|
+
#
|
57
|
+
# This returns a hash with three keys:
|
58
|
+
# :status The numeric HTTP status code
|
59
|
+
# :body The body of the response entity, if any
|
60
|
+
# :headers A hash of header values
|
61
|
+
def head(expected_statuses, timeout, path_components, headers={})
|
62
|
+
headers = default_headers.merge(headers)
|
63
|
+
verify_path!(path_components)
|
64
|
+
perform(:head, path_components, headers, expected_statuses, timeout)
|
65
|
+
end
|
66
|
+
|
67
|
+
Workfeed::Timer.time_method self, :head, 'http_head'
|
68
|
+
|
69
|
+
# Performs a GET request to the specified resource.
|
70
|
+
#
|
71
|
+
# A request to /users/#{@user.id}/things?q=woof with an Accept header of
|
72
|
+
# "text/plain" which is expecting a 200 OK within 50ms
|
73
|
+
#
|
74
|
+
# http.get(200, 50 ["users", @user.id, "things", {"q" => "woof"}], "Accept" => "text/plain")
|
75
|
+
#
|
76
|
+
# This returns a hash with three keys:
|
77
|
+
# :status The numeric HTTP status code
|
78
|
+
# :body The body of the response entity, if any
|
79
|
+
# :headers A hash of header values
|
80
|
+
def get(expected_statuses, timeout, path_components, headers={})
|
81
|
+
headers = default_headers.merge(headers)
|
82
|
+
verify_path!(path_components)
|
83
|
+
perform(:get, path_components, headers, expected_statuses, timeout)
|
84
|
+
end
|
85
|
+
|
86
|
+
Workfeed::Timer.time_method self, :get, 'http_get'
|
87
|
+
|
88
|
+
# The same thing as #get, but decodes the response entity as JSON (if it's
|
89
|
+
# application/json) and adds it under the :json key in the returned hash.
|
90
|
+
def get_json(expected_statuses, timeout, path_components, headers={}, yajl_opts={})
|
91
|
+
response = get(expected_statuses, timeout, path_components, headers)
|
92
|
+
|
93
|
+
content_type = response[:headers]['content-type']
|
94
|
+
content_type = content_type.first if content_type.respond_to?(:first)
|
95
|
+
|
96
|
+
if content_type.to_s.strip.match /^application\/json/
|
97
|
+
response[:json] = Yajl::Parser.parse(response[:body], yajl_opts)
|
98
|
+
end
|
99
|
+
|
100
|
+
response
|
101
|
+
end
|
102
|
+
|
103
|
+
# Performs a PUT request to the specified resource.
|
104
|
+
#
|
105
|
+
# A request to /users/#{@user.id}/things?q=woof with an Content-Type header of
|
106
|
+
# "text/plain" and a request entity of "yay" which is expecting a 204 No
|
107
|
+
# Content within 1000ms
|
108
|
+
#
|
109
|
+
# http.put(204, 1000, ["users", @user.id, "things", {"q" => "woof"}], "yay", "Content-Type" => "text/plain")
|
110
|
+
#
|
111
|
+
# This returns a hash with three keys:
|
112
|
+
# :status The numeric HTTP status code
|
113
|
+
# :body The body of the response entity, if any
|
114
|
+
# :headers A hash of header values
|
115
|
+
def put(expected_statuses, timeout, path_components, entity, headers={})
|
116
|
+
headers = default_headers.merge(headers)
|
117
|
+
verify_path_and_entity!(path_components, entity)
|
118
|
+
perform(:put, path_components, headers, expected_statuses, timeout, entity)
|
119
|
+
end
|
120
|
+
|
121
|
+
Workfeed::Timer.time_method self, :put, 'http_put'
|
122
|
+
|
123
|
+
# The same thing as #put, but encodes the entity as JSON and specifies
|
124
|
+
# "application/json" as the request entity content type.
|
125
|
+
def put_json(expected_statuses, timeout, path_components, entity, headers={})
|
126
|
+
put(expected_statuses, timeout, path_components, Yajl::Encoder.encode(entity), headers.merge("Content-Type" => "application/json"))
|
127
|
+
end
|
128
|
+
|
129
|
+
# Performs a POST request to the specified resource.
|
130
|
+
#
|
131
|
+
# A request to /users/#{@user.id}/things?q=woof with an Content-Type header of
|
132
|
+
# "text/plain" and a request entity of "yay" which is expecting a 201 Created
|
133
|
+
# within 500ms
|
134
|
+
#
|
135
|
+
# http.post(201, 500, ["users", @user.id, "things", {"q" => "woof"}], "yay", "Content-Type" => "text/plain")
|
136
|
+
#
|
137
|
+
# This returns a hash with three keys:
|
138
|
+
# :status The numeric HTTP status code
|
139
|
+
# :body The body of the response entity, if any
|
140
|
+
# :headers A hash of header values
|
141
|
+
def post(expected_statuses, timeout, path_components, entity, headers={})
|
142
|
+
headers = default_headers.merge(headers)
|
143
|
+
verify_path_and_entity!(path_components, entity)
|
144
|
+
perform(:post, path_components, headers, expected_statuses, timeout, entity)
|
145
|
+
end
|
146
|
+
|
147
|
+
Workfeed::Timer.time_method self, :post, 'http_post'
|
148
|
+
|
149
|
+
# The same thing as #post, but encodes the entity as JSON and specifies
|
150
|
+
# "application/json" as the request entity content type.
|
151
|
+
def post_json(expected_statuses, timeout, path_components, entity, headers={})
|
152
|
+
post(
|
153
|
+
expected_statuses,
|
154
|
+
timeout,
|
155
|
+
path_components,
|
156
|
+
Yajl::Encoder.encode(entity),
|
157
|
+
headers.merge("Content-Type" => "application/json")
|
158
|
+
)
|
159
|
+
end
|
160
|
+
|
161
|
+
# Performs a DELETE request to the specified resource.
|
162
|
+
#
|
163
|
+
# A request to /users/#{@user.id}/things?q=woof which is expecting a 204 No
|
164
|
+
# Content within 666ms
|
165
|
+
#
|
166
|
+
# http.put(200, 666, ["users", @user.id, "things", {"q" => "woof"}])
|
167
|
+
#
|
168
|
+
# This returns a hash with three keys:
|
169
|
+
# :status The numeric HTTP status code
|
170
|
+
# :body The body of the response entity, if any
|
171
|
+
# :headers A hash of header values
|
172
|
+
def delete(expected_statuses, timeout, path_components, headers={})
|
173
|
+
headers = default_headers.merge(headers)
|
174
|
+
verify_path!(path_components)
|
175
|
+
perform(:delete, path_components, headers, expected_statuses, timeout)
|
176
|
+
end
|
177
|
+
|
178
|
+
Workfeed::Timer.time_method self, :delete, 'http_delete'
|
179
|
+
|
180
|
+
# Creates a URI object, combining the root_uri passed on initialization
|
181
|
+
# with the given parts.
|
182
|
+
#
|
183
|
+
# Example:
|
184
|
+
#
|
185
|
+
# http = Zephyr.new 'http://host/'
|
186
|
+
# http.uri(['hi', 'bob', {:foo => 'bar'}]) => http://host/hi/bob?foo=bar
|
187
|
+
#
|
188
|
+
def uri(given_parts = [])
|
189
|
+
@root_uri.dup.tap do |uri|
|
190
|
+
parts = given_parts.dup.unshift(uri.path) # URI#merge is broken.
|
191
|
+
uri.query = build_query_string(parts.pop) if parts.last.is_a? Hash
|
192
|
+
uri.path = ('/%s/' % parts.join('/')).gsub(/\/+/, '/')
|
193
|
+
end
|
194
|
+
end
|
195
|
+
|
196
|
+
# Comes handy in IRB
|
197
|
+
#
|
198
|
+
def inspect
|
199
|
+
'#<%s:0x%s root_uri=%s>' % [ self.class.to_s, object_id.to_s(16), uri.to_s ]
|
200
|
+
end
|
201
|
+
|
202
|
+
def cleanup!
|
203
|
+
Typheous::Hydra.hydra.cleanup
|
204
|
+
end
|
205
|
+
|
206
|
+
# Assembles a query string from a Hash, escaping values and keys. If a
|
207
|
+
# value is an Array, the query parameter simply appears twice.
|
208
|
+
#
|
209
|
+
# Stolen from Rack (with minor changes for sorting).
|
210
|
+
#
|
211
|
+
def build_query_string(params)
|
212
|
+
params.map do |k, v|
|
213
|
+
if v.kind_of? Array
|
214
|
+
build_query_string(v.map { |x| [k, x] })
|
215
|
+
else
|
216
|
+
"#{percent_encode(k)}=#{percent_encode(v)}"
|
217
|
+
end
|
218
|
+
end.sort.join '&'
|
219
|
+
end
|
220
|
+
|
221
|
+
def percent_encode(value)
|
222
|
+
typhoeus_easy.send(:easy_escape, value.to_s, value.to_s.bytesize)
|
223
|
+
end
|
224
|
+
|
225
|
+
private
|
226
|
+
|
227
|
+
# NOTE: This is here only because it provides a binding to
|
228
|
+
# Curb's 'easy_escape' function, which does what we want.
|
229
|
+
# Don't use it to perform requests. Ever.
|
230
|
+
#
|
231
|
+
def typhoeus_easy
|
232
|
+
@_typhoeus_easy ||= Typhoeus::Easy.new.freeze
|
233
|
+
end
|
234
|
+
|
235
|
+
def verify_path_and_entity!(path_components, entity)
|
236
|
+
begin
|
237
|
+
verify_path!(path_components)
|
238
|
+
rescue ArgumentError
|
239
|
+
raise ArgumentError, "You must supply both a resource path and a body."
|
240
|
+
end
|
241
|
+
|
242
|
+
raise ArgumentError, "Request body must be a string or IO." unless String === entity || IO === entity
|
243
|
+
end
|
244
|
+
|
245
|
+
def verify_path!(path_components)
|
246
|
+
path_components = Array(path_components).flatten
|
247
|
+
raise ArgumentError, "Resource path too short" unless path_components.length > 0
|
248
|
+
end
|
249
|
+
|
250
|
+
def valid_response?(expected, actual)
|
251
|
+
Array(expected).map { |code| code.to_i }.include?(actual.to_i)
|
252
|
+
end
|
253
|
+
|
254
|
+
def return_body?(method, code)
|
255
|
+
method != :head && !valid_response?([204,205,304], code)
|
256
|
+
end
|
257
|
+
|
258
|
+
def perform(method, path_components, headers, expect, timeout, data=nil)
|
259
|
+
params = {}
|
260
|
+
params[:headers] = headers
|
261
|
+
params[:timeout] = timeout
|
262
|
+
params[:follow_location] = false
|
263
|
+
|
264
|
+
# if you want debugging
|
265
|
+
# params[:verbose] = true
|
266
|
+
|
267
|
+
# have a vague feeling this isn't going to work as expected
|
268
|
+
if method == :post || method == :put
|
269
|
+
data = data.read if data.respond_to?(:read)
|
270
|
+
params[:body] = data
|
271
|
+
end
|
272
|
+
|
273
|
+
# request has class methods for :delete, :get, :head, :put, and :post
|
274
|
+
response = Typhoeus::Request.send(method, uri(path_components).to_s, params)
|
275
|
+
|
276
|
+
if defined?(Rails)
|
277
|
+
Rails.logger.info "lib/http.rb: #{method.to_s.upcase} #{uri(path_components).to_s}"
|
278
|
+
end
|
279
|
+
|
280
|
+
# be consistent with what came before
|
281
|
+
response_headers = Headers.new.tap do |h|
|
282
|
+
response.headers.split(/\n/).each do |header_line|
|
283
|
+
h.parse(header_line)
|
284
|
+
end
|
285
|
+
end
|
286
|
+
|
287
|
+
if valid_response?(expect, response.code)
|
288
|
+
result = { :headers => response_headers.to_hash, :status => response.code }
|
289
|
+
if return_body?(method, response.code)
|
290
|
+
result[:body] = response.body
|
291
|
+
end
|
292
|
+
result
|
293
|
+
else
|
294
|
+
response_body = response.timed_out? || response.code == 0 ? "Exceeded #{timeout}ms" : response.body
|
295
|
+
raise FailedRequest.new(method, expect, response.code, response_headers.to_hash, response_body, response.timed_out?)
|
296
|
+
end
|
297
|
+
end
|
298
|
+
|
299
|
+
def create_request_headers(hash)
|
300
|
+
h = Headers.new
|
301
|
+
hash.each {|k,v| h.add_field(k,v) }
|
302
|
+
[].tap do |arr|
|
303
|
+
h.each_capitalized do |k,v|
|
304
|
+
arr << "#{k}: #{v}"
|
305
|
+
end
|
306
|
+
end
|
307
|
+
end
|
308
|
+
end
|
309
|
+
Http = YammerHttp
|
310
|
+
|
311
|
+
# Represents headers from an HTTP request or response.
|
312
|
+
# Used internally by HTTP backends for processing headers.
|
313
|
+
class Headers
|
314
|
+
include Net::HTTPHeader
|
315
|
+
|
316
|
+
def initialize
|
317
|
+
initialize_http_header({})
|
318
|
+
end
|
319
|
+
|
320
|
+
# Parse a single header line into its key and value
|
321
|
+
# @param [String] chunk a single header line
|
322
|
+
def self.parse(chunk)
|
323
|
+
line = chunk.strip
|
324
|
+
# thanks Net::HTTPResponse
|
325
|
+
return [nil,nil] if chunk =~ /\AHTTP(?:\/(\d+\.\d+))?\s+(\d\d\d)\s*(.*)\z/in
|
326
|
+
m = /\A([^:]+):\s*/.match(line)
|
327
|
+
[m[1], m.post_match] rescue [nil, nil]
|
328
|
+
end
|
329
|
+
|
330
|
+
# Parses a header line and adds it to the header collection
|
331
|
+
# @param [String] chunk a single header line
|
332
|
+
def parse(chunk)
|
333
|
+
key, value = self.class.parse(chunk)
|
334
|
+
add_field(key, value) if key && value
|
335
|
+
end
|
336
|
+
end
|
337
|
+
|
338
|
+
class FailedRequest < StandardError
|
339
|
+
attr_reader :method
|
340
|
+
attr_reader :expected
|
341
|
+
attr_reader :code
|
342
|
+
attr_reader :headers
|
343
|
+
attr_reader :body
|
344
|
+
attr_reader :timed_out
|
345
|
+
|
346
|
+
def initialize(method, expected_code, received_code, headers, body, timed_out)
|
347
|
+
@method, @expected, @code, @headers, @body, @timed_out = method, expected_code, received_code, headers, body, timed_out
|
348
|
+
super "Expected #{@expected.inspect} from the server but received #{@code}. #{@body}"
|
349
|
+
end
|
350
|
+
end
|
data/test/helper.rb
ADDED
@@ -0,0 +1,18 @@
|
|
1
|
+
require 'rubygems'
|
2
|
+
require 'bundler'
|
3
|
+
begin
|
4
|
+
Bundler.setup(:default, :development)
|
5
|
+
rescue Bundler::BundlerError => e
|
6
|
+
$stderr.puts e.message
|
7
|
+
$stderr.puts "Run `bundle install` to install missing gems"
|
8
|
+
exit e.status_code
|
9
|
+
end
|
10
|
+
require 'test/unit'
|
11
|
+
require 'shoulda'
|
12
|
+
|
13
|
+
$LOAD_PATH.unshift(File.join(File.dirname(__FILE__), '..', 'lib'))
|
14
|
+
$LOAD_PATH.unshift(File.dirname(__FILE__))
|
15
|
+
require 'zephyr'
|
16
|
+
|
17
|
+
class Test::Unit::TestCase
|
18
|
+
end
|
data/test/test_zephyr.rb
ADDED
data/zephyr.gemspec
ADDED
@@ -0,0 +1,56 @@
|
|
1
|
+
# Generated by jeweler
|
2
|
+
# DO NOT EDIT THIS FILE DIRECTLY
|
3
|
+
# Instead, edit Jeweler::Tasks in Rakefile, and run 'rake gemspec'
|
4
|
+
# -*- encoding: utf-8 -*-
|
5
|
+
|
6
|
+
Gem::Specification.new do |s|
|
7
|
+
s.name = %q{zephyr}
|
8
|
+
s.version = "1.0.0"
|
9
|
+
|
10
|
+
s.required_rubygems_version = Gem::Requirement.new(">= 0") if s.respond_to? :required_rubygems_version=
|
11
|
+
s.authors = ["Matt Knopp"]
|
12
|
+
s.date = %q{2011-09-30}
|
13
|
+
s.description = %q{Simple HTTP client using Typheous, derived from the Riak client}
|
14
|
+
s.email = %q{matt.knopp@gmail.com}
|
15
|
+
s.extra_rdoc_files = [
|
16
|
+
"LICENSE.txt",
|
17
|
+
"README.md"
|
18
|
+
]
|
19
|
+
s.files = [
|
20
|
+
".document",
|
21
|
+
"Gemfile",
|
22
|
+
"Gemfile.lock",
|
23
|
+
"LICENSE.txt",
|
24
|
+
"README.md",
|
25
|
+
"Rakefile",
|
26
|
+
"VERSION",
|
27
|
+
"lib/zephyr.rb",
|
28
|
+
"test/helper.rb",
|
29
|
+
"test/test_zephyr.rb",
|
30
|
+
"zephyr.gemspec"
|
31
|
+
]
|
32
|
+
s.homepage = %q{http://github.com/mhat/zephyr}
|
33
|
+
s.licenses = ["MIT"]
|
34
|
+
s.require_paths = ["lib"]
|
35
|
+
s.rubygems_version = %q{1.5.2}
|
36
|
+
s.summary = %q{Simple HTTP client using Typheous, derived from the Riak client}
|
37
|
+
|
38
|
+
if s.respond_to? :specification_version then
|
39
|
+
s.specification_version = 3
|
40
|
+
|
41
|
+
if Gem::Version.new(Gem::VERSION) >= Gem::Version.new('1.2.0') then
|
42
|
+
s.add_runtime_dependency(%q<typhoeus>, [">= 0.2.4"])
|
43
|
+
s.add_development_dependency(%q<bundler>, ["~> 1.0.0"])
|
44
|
+
s.add_development_dependency(%q<jeweler>, ["~> 1.6.4"])
|
45
|
+
else
|
46
|
+
s.add_dependency(%q<typhoeus>, [">= 0.2.4"])
|
47
|
+
s.add_dependency(%q<bundler>, ["~> 1.0.0"])
|
48
|
+
s.add_dependency(%q<jeweler>, ["~> 1.6.4"])
|
49
|
+
end
|
50
|
+
else
|
51
|
+
s.add_dependency(%q<typhoeus>, [">= 0.2.4"])
|
52
|
+
s.add_dependency(%q<bundler>, ["~> 1.0.0"])
|
53
|
+
s.add_dependency(%q<jeweler>, ["~> 1.6.4"])
|
54
|
+
end
|
55
|
+
end
|
56
|
+
|
metadata
ADDED
@@ -0,0 +1,125 @@
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
2
|
+
name: zephyr
|
3
|
+
version: !ruby/object:Gem::Version
|
4
|
+
hash: 23
|
5
|
+
prerelease:
|
6
|
+
segments:
|
7
|
+
- 1
|
8
|
+
- 0
|
9
|
+
- 0
|
10
|
+
version: 1.0.0
|
11
|
+
platform: ruby
|
12
|
+
authors:
|
13
|
+
- Matt Knopp
|
14
|
+
autorequire:
|
15
|
+
bindir: bin
|
16
|
+
cert_chain: []
|
17
|
+
|
18
|
+
date: 2011-09-30 00:00:00 -07:00
|
19
|
+
default_executable:
|
20
|
+
dependencies:
|
21
|
+
- !ruby/object:Gem::Dependency
|
22
|
+
name: typhoeus
|
23
|
+
version_requirements: &id001 !ruby/object:Gem::Requirement
|
24
|
+
none: false
|
25
|
+
requirements:
|
26
|
+
- - ">="
|
27
|
+
- !ruby/object:Gem::Version
|
28
|
+
hash: 31
|
29
|
+
segments:
|
30
|
+
- 0
|
31
|
+
- 2
|
32
|
+
- 4
|
33
|
+
version: 0.2.4
|
34
|
+
prerelease: false
|
35
|
+
type: :runtime
|
36
|
+
requirement: *id001
|
37
|
+
- !ruby/object:Gem::Dependency
|
38
|
+
name: bundler
|
39
|
+
version_requirements: &id002 !ruby/object:Gem::Requirement
|
40
|
+
none: false
|
41
|
+
requirements:
|
42
|
+
- - ~>
|
43
|
+
- !ruby/object:Gem::Version
|
44
|
+
hash: 23
|
45
|
+
segments:
|
46
|
+
- 1
|
47
|
+
- 0
|
48
|
+
- 0
|
49
|
+
version: 1.0.0
|
50
|
+
prerelease: false
|
51
|
+
type: :development
|
52
|
+
requirement: *id002
|
53
|
+
- !ruby/object:Gem::Dependency
|
54
|
+
name: jeweler
|
55
|
+
version_requirements: &id003 !ruby/object:Gem::Requirement
|
56
|
+
none: false
|
57
|
+
requirements:
|
58
|
+
- - ~>
|
59
|
+
- !ruby/object:Gem::Version
|
60
|
+
hash: 7
|
61
|
+
segments:
|
62
|
+
- 1
|
63
|
+
- 6
|
64
|
+
- 4
|
65
|
+
version: 1.6.4
|
66
|
+
prerelease: false
|
67
|
+
type: :development
|
68
|
+
requirement: *id003
|
69
|
+
description: Simple HTTP client using Typheous, derived from the Riak client
|
70
|
+
email: matt.knopp@gmail.com
|
71
|
+
executables: []
|
72
|
+
|
73
|
+
extensions: []
|
74
|
+
|
75
|
+
extra_rdoc_files:
|
76
|
+
- LICENSE.txt
|
77
|
+
- README.md
|
78
|
+
files:
|
79
|
+
- .document
|
80
|
+
- Gemfile
|
81
|
+
- Gemfile.lock
|
82
|
+
- LICENSE.txt
|
83
|
+
- README.md
|
84
|
+
- Rakefile
|
85
|
+
- VERSION
|
86
|
+
- lib/zephyr.rb
|
87
|
+
- test/helper.rb
|
88
|
+
- test/test_zephyr.rb
|
89
|
+
- zephyr.gemspec
|
90
|
+
has_rdoc: true
|
91
|
+
homepage: http://github.com/mhat/zephyr
|
92
|
+
licenses:
|
93
|
+
- MIT
|
94
|
+
post_install_message:
|
95
|
+
rdoc_options: []
|
96
|
+
|
97
|
+
require_paths:
|
98
|
+
- lib
|
99
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
100
|
+
none: false
|
101
|
+
requirements:
|
102
|
+
- - ">="
|
103
|
+
- !ruby/object:Gem::Version
|
104
|
+
hash: 3
|
105
|
+
segments:
|
106
|
+
- 0
|
107
|
+
version: "0"
|
108
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
109
|
+
none: false
|
110
|
+
requirements:
|
111
|
+
- - ">="
|
112
|
+
- !ruby/object:Gem::Version
|
113
|
+
hash: 3
|
114
|
+
segments:
|
115
|
+
- 0
|
116
|
+
version: "0"
|
117
|
+
requirements: []
|
118
|
+
|
119
|
+
rubyforge_project:
|
120
|
+
rubygems_version: 1.5.2
|
121
|
+
signing_key:
|
122
|
+
specification_version: 3
|
123
|
+
summary: Simple HTTP client using Typheous, derived from the Riak client
|
124
|
+
test_files: []
|
125
|
+
|