0.0
No commit activity in last 3 years
No release in over 3 years
GiantClient is a lightweight adapter agnostic wrapper for major HTTP client libraries.
2005
2006
2007
2008
2009
2010
2011
2012
2013
2014
2015
2016
2017
2018
2019
2020
2021
2022
2023
2024
2025
 Dependencies

Development

 Project Readme

##Giant Client

Giant Client is a ruby library which exposes a simple and uniform API for using a variety of http clients. Advantages of Giant Client:

  • Clean, easy-to-use, cruftless API
  • Testing as simple as it gets with a built in test framework that uses the exact same API
  • Take advantage of 5+ http clients while only learning one API
  • Seamlessly swap between http clients with one line of configuration (rather than erasing all of your code)

Installation

Add this line to your application's Gemfile:

gem 'giant_client'

And then execute:

$ bundle

Or install it yourself as:

$ gem install giant_client

Usage

In this guide, I am very exhaustive with options, so it seems long winded, but this is just to show all configuration options, most are optional.

####Initialize a new Giant Client options = { :host => "www.example.com", # optional, defaults to "" :ssl => true, # optional, defaults to false :port => 443, # optional, defaults to 443 when ssl == true, # and to 80 when ssl == false :adapter => :typhoeus # optional, defaults to :net_http, options detailed below }

client = GiantClient.new(options)

######Passing a symbol argument as the first argument to initialize is assumed to be the adapter ######short hand client = GiantClient.new :patron, :host => 'example.com'

####Adapters There are currently five adapters supported. They use the libraries they are named after. :net_http :patron :excon :curb :typhoeus

####The API Giant Client supports the HTTP 'GET', 'POST', 'PUT', 'DELETE', and 'HEAD' methods The 'GET', 'DELETE', and 'HEAD' methods do not support a request body. Attempting to pass one will cause Giant Client to throw a NotImplementedError #####All methods support ssl, host, port, path, query, headers, and body as options.

#####Note: ssl defaults to ssl provided in new (true here) host defaults to host provided in new ('www.example.com' here) port defaults to port provided in new ('443' here)

######GET client.get({ :path => '/path/to/resource', # optional, defaults to '/' :headers => { 'Accepts' => 'application/json' }, # optional, defaults to {} :query => { 'post_id' => '29' } # optional, defaults to {} }) ######POST client.post({ :path => '/path/to/resource', # optional, defaults to '/' :body => 'hey, how's it going?', # optional, defaults to '' }) ######PUT client.put({ :path => '/path/to/resource', # optional, defaults to '/' :body => 'hey, how's it going?' # optional, defaults to '' }) ######DELETE client.delete({ :path => '/path/to/resource?for_good=true' }) ######HEAD client.head({ :path => '/path/to/resource' }) ######Giant Client also exposes a general request method, which each method calls under the hood. client.request(:get, { :path => '/search?q=ahoy', })

####Return Value All of Giant Client's request methods return a uniform response object. This object has accessors for status_code, headers, and body In general, status_code is a number, headers is a Hash, and body is a String. ######Example response = client.request(:get, { :path => '/search?q=ahoy', }) puts response.status_code # e.g. 200 puts response.headers # e.g. {'Content-Type' => 'application/json' } puts response.body # e.g. 'ahoy matee'

######Giant Client provides getters and setters to default port, host, and ssl. # one request over http on port 80 g_client = GiantClient.new( :adapter => :typhoeus, :host => 'www.google.com' ) g_client.get(:path => '/')

# the next on port 8080
# 2 ways to do this:
# way 1
g_client.port = 8080
g_client.get(:path => '/')
# way 2
g_client.get(:path => '/', :port => 8080)

######Timeouts In GiantClient timeouts can be configured on initialization or request by request. If you just leave it alone, the default timeout is 30 seconds. This is roughly the average of the clients supported. Setting a timeout sets both the read timeout and the connect timeout. Setting a Timeout # on configuration client = GiantClient.new( :adapter => :typhoeus, :timeout => 5 ) # five second timeout client.get '/' # will raise a GiantClient::TimeoutError if it takes longer than 5 seconds.

# request specific timeout
client = GiantClient.new :typhoeus           # five second timeout
client.get '/', :timeout => 1                # will raise a GiantClient::TimeoutError if it takes longer than 1 second.

######Passing a string as the options will set path = to that string and all other options to their defaults client.get('/') # same as :path => '/'

######Passing a string and options will set path = to that string and all other options to those specified in the options client.get('/', { :query => "foo=bar" }) # same as :path => '/', :query => "foo=bar"

######The string argument takes precedence client.get('/', {:path => '/this/does/not/matter'}) # same as :path => '/'

######Passing more than 2 arguments or fewer than one argument will raise an ArgumentError client.get('I', 'skipped', 'the', 'README') # ArgumentError

Note: this is true with initialize too

####Testing

Testing is easier than ever if you use GiantClient. Just call GiantClient.new :adapter => :mock and GiantClient will stub out all your requests. When the adapter is set to :mock GiantClient stores all of your requests on a stack: client.requests. This is also true for responses. last_request is shorthand (or longhand, actually) for requests[0] Check it out.

describe 'Mock Adapter' do
  let(:client){ GiantClient.new :host => 'example.com', :adapter => :mock }

  context 'super simple request' do
    before do
      client.get('/')
    end

    it 'should record the url' do
      client.last_request.url.should == 'http://example.com/'
    end
  end
  context 'request with all options set' do
    before do
      opts = {
        :ssl => true,
        :port => 9292,
        :path => '/hey/there',
        :query => { 'howya' => 'doing' },
        :headers => { 'Content-Type' => 'application/awesome' },
        :body => 'test body',
        :timeout => 27
      }
      client.get(opts)
    end

    it 'should record the (correct) url' do
      url = 'https://example.com:9292/hey/there?howya=doing'
      client.last_request.url.should == url
    end

    it 'should record the (correct) ssl' do
      client.last_request.ssl.should == true
    end
    it 'should record the (correct) port' do
      client.last_request.port.should == 9292
    end
    it 'should record the (correct) path' do
      client.last_request.path.should == '/hey/there'
    end
    it 'should record the (correct) query' do
      client.last_request.query.should == { 'howya' => 'doing' }
    end
    it 'should record the (correct) query_string' do
      client.last_request.querystring.should == 'howya=doing'
    end
    it 'should record the (correct) headers' do
      client.last_request.headers.should ==
                            { 'Content-Type' => 'application/awesome' }
    end
    it 'should record the (correct) body' do
      client.last_request.body.should == 'test body'
    end
    it 'should record the (correct) timeout' do
      client.last_request.timeout.should == 27
    end
  end
end

Giant Client creates a response hash for the mock requests. By default it is this:

{
  :status_code => 200,
  :headers     => {},
  :body        => nil
}

You can manipulate the response to a specific request with the respond_with method, which merges in your settings: client.get('/').respond_with(:body => 'hey') client.last_response.should == { :status_code => 200, :headers => {}, :body => 'hey' } You can set arbitrary fields (although there usually isn't reason to)

client.get('/').respond_with(:headers => {}:body => 'hey')
client.last_response.should == {
                                 :status_code => 200,
                                 :headers     => {},
                                 :body        => 'hey'
                               }

As many as you want

client.get('/').respond_with(:headers => {
                                            'Content-Type' => 'application/json',
                                            'X-Powered-By' => 'Internet'
                                         },
                                         :body => 'hey',
                                         :foo  => 'bar'
                            )
client.last_response.should == {
                                 :status_code => 200,
                                 :headers     => {
                                                    'Content-Type' => 'application/json',
                                                    'X-Powered-By' => 'Internet'
                                                 },
                                 :body        => 'hey',
                                 :foo        => 'bar'
                               }

# in practice you'll usually just test a property at a time e.g.
client.last_response[:body].should == 'hey' # much cleaner

For tests with multiple requests / responses, visit spec/examples/mock_adapter_spec.rb

Contributing

  1. Fork it
  2. Create your feature branch (git checkout -b my-new-feature)
  3. Commit your changes (git commit -am 'Added some feature')
  4. Push to the branch (git push origin my-new-feature)
  5. Create new Pull Request