3 # Copyright 2013 Google Inc.
5 # Licensed under the Apache License, Version 2.0 (the "License");
6 # you may not use this file except in compliance with the License.
7 # You may obtain a copy of the License at
9 # http://www.apache.org/licenses/LICENSE-2.0
11 # Unless required by applicable law or agreed to in writing, software
12 # distributed under the License is distributed on an "AS IS" BASIS,
13 # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 # See the License for the specific language governing permissions and
15 # limitations under the License.
19 require 'google/api_client'
20 require 'google/api_client/service'
22 fixtures_path = File.expand_path('../../../fixtures', __FILE__)
24 describe Google::APIClient::Service do
25 include ConnectionHelpers
27 APPLICATION_NAME = 'API Client Tests'
29 it 'should error out when called without an API name or version' do
31 Google::APIClient::Service.new
32 end).should raise_error(ArgumentError)
35 it 'should error out when called without an API version' do
37 Google::APIClient::Service.new('foo')
38 end).should raise_error(ArgumentError)
41 it 'should error out when the options hash is not a hash' do
43 Google::APIClient::Service.new('foo', 'v1', 42)
44 end).should raise_error(ArgumentError)
47 describe 'with the AdSense Management API' do
49 it 'should make a valid call for a method with no parameters' do
50 conn = stub_connection do |stub|
51 stub.get('/adsense/v1.3/adclients') do |env|
55 adsense = Google::APIClient::Service.new(
59 :application_name => APPLICATION_NAME,
60 :authenticated => false,
66 req = adsense.adclients.list.execute()
70 it 'should make a valid call for a method with parameters' do
71 conn = stub_connection do |stub|
72 stub.get('/adsense/v1.3/adclients/1/adunits') do |env|
76 adsense = Google::APIClient::Service.new(
80 :application_name => APPLICATION_NAME,
81 :authenticated => false,
86 req = adsense.adunits.list(:adClientId => '1').execute()
89 it 'should make a valid call for a deep method' do
90 conn = stub_connection do |stub|
91 stub.get('/adsense/v1.3/accounts/1/adclients') do |env|
95 adsense = Google::APIClient::Service.new(
99 :application_name => APPLICATION_NAME,
100 :authenticated => false,
105 req = adsense.accounts.adclients.list(:accountId => '1').execute()
108 describe 'with no connection' do
110 @adsense = Google::APIClient::Service.new('adsense', 'v1.3',
111 {:application_name => APPLICATION_NAME, :cache_store => nil})
114 it 'should return a resource when using a valid resource name' do
115 @adsense.accounts.should be_a(Google::APIClient::Service::Resource)
118 it 'should throw an error when using an invalid resource name' do
120 @adsense.invalid_resource
121 end).should raise_error
124 it 'should return a request when using a valid method name' do
125 req = @adsense.adclients.list
126 req.should be_a(Google::APIClient::Service::Request)
127 req.method.id.should == 'adsense.adclients.list'
128 req.parameters.should be_nil
131 it 'should throw an error when using an invalid method name' do
133 @adsense.adclients.invalid_method
134 end).should raise_error
137 it 'should return a valid request with parameters' do
138 req = @adsense.adunits.list(:adClientId => '1')
139 req.should be_a(Google::APIClient::Service::Request)
140 req.method.id.should == 'adsense.adunits.list'
141 req.parameters.should_not be_nil
142 req.parameters[:adClientId].should == '1'
147 describe 'with the Prediction API' do
149 it 'should make a valid call with an object body' do
150 conn = stub_connection do |stub|
151 stub.post('/prediction/v1.5/trainedmodels?project=1') do |env|
152 env.body.should == '{"id":"1"}'
156 prediction = Google::APIClient::Service.new(
160 :application_name => APPLICATION_NAME,
161 :authenticated => false,
166 req = prediction.trainedmodels.insert(:project => '1').body({'id' => '1'}).execute()
170 it 'should make a valid call with a text body' do
171 conn = stub_connection do |stub|
172 stub.post('/prediction/v1.5/trainedmodels?project=1') do |env|
173 env.body.should == '{"id":"1"}'
177 prediction = Google::APIClient::Service.new(
181 :application_name => APPLICATION_NAME,
182 :authenticated => false,
187 req = prediction.trainedmodels.insert(:project => '1').body('{"id":"1"}').execute()
191 describe 'with no connection' do
193 @prediction = Google::APIClient::Service.new('prediction', 'v1.5',
194 {:application_name => APPLICATION_NAME, :cache_store => nil})
197 it 'should return a valid request with a body' do
198 req = @prediction.trainedmodels.insert(:project => '1').body({'id' => '1'})
199 req.should be_a(Google::APIClient::Service::Request)
200 req.method.id.should == 'prediction.trainedmodels.insert'
201 req.body.should == {'id' => '1'}
202 req.parameters.should_not be_nil
203 req.parameters[:project].should == '1'
206 it 'should return a valid request with a body when using resource name' do
207 req = @prediction.trainedmodels.insert(:project => '1').training({'id' => '1'})
208 req.should be_a(Google::APIClient::Service::Request)
209 req.method.id.should == 'prediction.trainedmodels.insert'
210 req.training.should == {'id' => '1'}
211 req.parameters.should_not be_nil
212 req.parameters[:project].should == '1'
217 describe 'with the Drive API' do
221 'title' => 'My movie',
222 'description' => 'The best home movie ever made'
224 @file = File.expand_path('files/sample.txt', fixtures_path)
225 @media = Google::APIClient::UploadIO.new(@file, 'text/plain')
228 it 'should make a valid call with an object body and media upload' do
229 conn = stub_connection do |stub|
230 stub.post('/upload/drive/v1/files?uploadType=multipart') do |env|
231 env.body.should be_a Faraday::CompositeReadIO
235 drive = Google::APIClient::Service.new(
239 :application_name => APPLICATION_NAME,
240 :authenticated => false,
245 req = drive.files.insert(:uploadType => 'multipart').body(@metadata).media(@media).execute()
249 describe 'with no connection' do
251 @drive = Google::APIClient::Service.new('drive', 'v1',
252 {:application_name => APPLICATION_NAME, :cache_store => nil})
255 it 'should return a valid request with a body and media upload' do
256 req = @drive.files.insert(:uploadType => 'multipart').body(@metadata).media(@media)
257 req.should be_a(Google::APIClient::Service::Request)
258 req.method.id.should == 'drive.files.insert'
259 req.body.should == @metadata
260 req.media.should == @media
261 req.parameters.should_not be_nil
262 req.parameters[:uploadType].should == 'multipart'
265 it 'should return a valid request with a body and media upload when using resource name' do
266 req = @drive.files.insert(:uploadType => 'multipart').file(@metadata).media(@media)
267 req.should be_a(Google::APIClient::Service::Request)
268 req.method.id.should == 'drive.files.insert'
269 req.file.should == @metadata
270 req.media.should == @media
271 req.parameters.should_not be_nil
272 req.parameters[:uploadType].should == 'multipart'
277 describe 'with the Discovery API' do
278 it 'should make a valid end-to-end request' do
279 discovery = Google::APIClient::Service.new('discovery', 'v1',
280 {:application_name => APPLICATION_NAME, :authenticated => false,
281 :cache_store => nil})
282 result = discovery.apis.get_rest(:api => 'discovery', :version => 'v1').execute
283 result.should_not be_nil
284 result.data.name.should == 'discovery'
285 result.data.version.should == 'v1'
291 describe Google::APIClient::Service::Result do
293 describe 'with the plus API' do
295 @plus = Google::APIClient::Service.new('plus', 'v1',
296 {:application_name => APPLICATION_NAME, :cache_store => nil})
297 @reference = Google::APIClient::Reference.new({
298 :api_method => @plus.activities.list.method,
301 'collection' => 'public',
305 @request = @plus.activities.list(:userId => 'me', :collection => 'public',
309 @response = double("response")
310 @response.stub(:status).and_return(200)
311 @response.stub(:headers).and_return({
313 'x-google-apiary-auth-scopes' =>
314 'https://www.googleapis.com/auth/plus.me',
315 'content-type' => 'application/json; charset=UTF-8',
316 'date' => 'Mon, 23 Apr 2012 00:00:00 GMT',
317 'cache-control' => 'private, max-age=0, must-revalidate, no-transform',
319 'connection' => 'close'
323 describe 'with a next page token' do
325 @body = <<-END_OF_STRING
327 "kind": "plus#activityFeed",
329 "nextPageToken": "NEXT+PAGE+TOKEN",
330 "selfLink": "https://www.googleapis.com/plus/v1/people/foo/activities/public?",
331 "nextLink": "https://www.googleapis.com/plus/v1/people/foo/activities/public?maxResults=20&pageToken=NEXT%2BPAGE%2BTOKEN",
332 "title": "Plus Public Activity Feed for ",
333 "updated": "2012-04-23T00:00:00.000Z",
338 @response.stub(:body).and_return(@body)
339 base_result = Google::APIClient::Result.new(@reference, @response)
340 @result = Google::APIClient::Service::Result.new(@request, base_result)
343 it 'should indicate a successful response' do
344 @result.error?.should be_false
347 it 'should return the correct next page token' do
348 @result.next_page_token.should == 'NEXT+PAGE+TOKEN'
351 it 'generate a correct request when calling next_page' do
352 next_page_request = @result.next_page
353 next_page_request.parameters.should include('pageToken')
354 next_page_request.parameters['pageToken'].should == 'NEXT+PAGE+TOKEN'
355 @request.parameters.each_pair do |param, value|
356 next_page_request.parameters[param].should == value
360 it 'should return content type correctly' do
361 @result.media_type.should == 'application/json'
364 it 'should return the body correctly' do
365 @result.body.should == @body
368 it 'should return the result data correctly' do
369 @result.data?.should be_true
370 @result.data.class.to_s.should ==
371 'Google::APIClient::Schema::Plus::V1::ActivityFeed'
372 @result.data.kind.should == 'plus#activityFeed'
373 @result.data.etag.should == 'FOO'
374 @result.data.nextPageToken.should == 'NEXT+PAGE+TOKEN'
375 @result.data.selfLink.should ==
376 'https://www.googleapis.com/plus/v1/people/foo/activities/public?'
377 @result.data.nextLink.should ==
378 'https://www.googleapis.com/plus/v1/people/foo/activities/public?' +
379 'maxResults=20&pageToken=NEXT%2BPAGE%2BTOKEN'
380 @result.data.title.should == 'Plus Public Activity Feed for '
381 @result.data.id.should == "123456790"
382 @result.data.items.should be_empty
386 describe 'without a next page token' do
388 @body = <<-END_OF_STRING
390 "kind": "plus#activityFeed",
392 "selfLink": "https://www.googleapis.com/plus/v1/people/foo/activities/public?",
393 "title": "Plus Public Activity Feed for ",
394 "updated": "2012-04-23T00:00:00.000Z",
399 @response.stub(:body).and_return(@body)
400 base_result = Google::APIClient::Result.new(@reference, @response)
401 @result = Google::APIClient::Service::Result.new(@request, base_result)
404 it 'should not return a next page token' do
405 @result.next_page_token.should == nil
408 it 'should return content type correctly' do
409 @result.media_type.should == 'application/json'
412 it 'should return the body correctly' do
413 @result.body.should == @body
416 it 'should return the result data correctly' do
417 @result.data?.should be_true
418 @result.data.class.to_s.should ==
419 'Google::APIClient::Schema::Plus::V1::ActivityFeed'
420 @result.data.kind.should == 'plus#activityFeed'
421 @result.data.etag.should == 'FOO'
422 @result.data.selfLink.should ==
423 'https://www.googleapis.com/plus/v1/people/foo/activities/public?'
424 @result.data.title.should == 'Plus Public Activity Feed for '
425 @result.data.id.should == "123456790"
426 @result.data.items.should be_empty
430 describe 'with JSON error response' do
432 @body = <<-END_OF_STRING
438 "reason": "parseError",
439 "message": "Parse Error"
443 "message": "Parse Error"
447 @response.stub(:body).and_return(@body)
448 @response.stub(:status).and_return(400)
449 base_result = Google::APIClient::Result.new(@reference, @response)
450 @result = Google::APIClient::Service::Result.new(@request, base_result)
453 it 'should return error status correctly' do
454 @result.error?.should be_true
457 it 'should return the correct error message' do
458 @result.error_message.should == 'Parse Error'
461 it 'should return the body correctly' do
462 @result.body.should == @body
466 describe 'with 204 No Content response' do
468 @response.stub(:body).and_return('')
469 @response.stub(:status).and_return(204)
470 @response.stub(:headers).and_return({})
471 base_result = Google::APIClient::Result.new(@reference, @response)
472 @result = Google::APIClient::Service::Result.new(@request, base_result)
475 it 'should indicate no data is available' do
476 @result.data?.should be_false
479 it 'should return nil for data' do
480 @result.data.should == nil
483 it 'should return nil for media_type' do
484 @result.media_type.should == nil
490 describe Google::APIClient::Service::BatchRequest do
491 describe 'with the discovery API' do
493 @discovery = Google::APIClient::Service.new('discovery', 'v1',
494 {:application_name => APPLICATION_NAME, :authorization => nil,
495 :cache_store => nil})
498 describe 'with two valid requests' do
501 @discovery.apis.get_rest(:api => 'plus', :version => 'v1'),
502 @discovery.apis.get_rest(:api => 'discovery', :version => 'v1')
506 it 'should execute both when using a global callback' do
508 batch = @discovery.batch(@calls) do |result|
510 result.status.should == 200
514 block_called.should == 2
517 it 'should execute both when using individual callbacks' do
518 call1_returned, call2_returned = false, false
519 batch = @discovery.batch
521 batch.add(@calls[0]) do |result|
522 call1_returned = true
523 result.status.should == 200
524 result.call_index.should == 0
527 batch.add(@calls[1]) do |result|
528 call2_returned = true
529 result.status.should == 200
530 result.call_index.should == 1
534 call1_returned.should == true
535 call2_returned.should == true
539 describe 'with a valid request and an invalid one' do
542 @discovery.apis.get_rest(:api => 'plus', :version => 'v1'),
543 @discovery.apis.get_rest(:api => 'invalid', :version => 'invalid')
547 it 'should execute both when using a global callback' do
549 batch = @discovery.batch(@calls) do |result|
551 if result.call_index == 0
552 result.status.should == 200
554 result.status.should >= 400
555 result.status.should < 500
560 block_called.should == 2
563 it 'should execute both when using individual callbacks' do
564 call1_returned, call2_returned = false, false
565 batch = @discovery.batch
567 batch.add(@calls[0]) do |result|
568 call1_returned = true
569 result.status.should == 200
570 result.call_index.should == 0
573 batch.add(@calls[1]) do |result|
574 call2_returned = true
575 result.status.should >= 400
576 result.status.should < 500
577 result.call_index.should == 1
581 call1_returned.should == true
582 call2_returned.should == true