blob: e1ef94febcfb7d3f21cacc95d56923260ace6175 [file] [log] [blame]
Joe Gregorioba9ea7f2010-08-19 15:49:04 -04001#!/usr/bin/python2.4
Joe Gregoriof863f7a2011-02-24 03:24:44 -05002# -*- coding: utf-8 -*-
Joe Gregorioba9ea7f2010-08-19 15:49:04 -04003#
Joe Gregorio6d5e94f2010-08-25 23:49:30 -04004# Copyright 2010 Google Inc.
5#
6# Licensed under the Apache License, Version 2.0 (the "License");
7# you may not use this file except in compliance with the License.
8# You may obtain a copy of the License at
9#
10# http://www.apache.org/licenses/LICENSE-2.0
11#
12# Unless required by applicable law or agreed to in writing, software
13# distributed under the License is distributed on an "AS IS" BASIS,
14# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15# See the License for the specific language governing permissions and
16# limitations under the License.
17
Joe Gregorioba9ea7f2010-08-19 15:49:04 -040018
19"""Discovery document tests
20
21Unit tests for objects created from discovery documents.
22"""
23
24__author__ = 'jcgregorio@google.com (Joe Gregorio)'
25
Joe Gregorioba9ea7f2010-08-19 15:49:04 -040026import httplib2
27import os
28import unittest
Joe Gregorio00cf1d92010-09-27 09:22:03 -040029import urlparse
Joe Gregorioa98733f2011-09-16 10:12:28 -040030
ade@google.comc5eb46f2010-09-27 23:35:39 +010031try:
32 from urlparse import parse_qs
33except ImportError:
34 from cgi import parse_qs
Joe Gregorio00cf1d92010-09-27 09:22:03 -040035
ade@google.com6a8c1cb2011-09-06 17:40:00 +010036from apiclient.discovery import build, build_from_document, key2param
Joe Gregorioc0e0fe92011-03-04 16:16:55 -050037from apiclient.errors import HttpError
Joe Gregorio49396552011-03-08 10:39:00 -050038from apiclient.errors import InvalidJsonError
Joe Gregoriofdf7c802011-06-30 12:33:38 -040039from apiclient.errors import MediaUploadSizeError
Joe Gregoriod0bd3882011-11-22 09:49:47 -050040from apiclient.errors import ResumableUploadError
Joe Gregoriofdf7c802011-06-30 12:33:38 -040041from apiclient.errors import UnacceptableMimeTypeError
Joe Gregoriod0bd3882011-11-22 09:49:47 -050042from apiclient.http import HttpMock
43from apiclient.http import HttpMockSequence
44from apiclient.http import MediaFileUpload
45from apiclient.http import MediaUploadProgress
46from apiclient.http import tunnel_patch
Joe Gregoriocb8103d2011-02-11 23:20:52 -050047
48
49DATA_DIR = os.path.join(os.path.dirname(__file__), 'data')
50
Joe Gregorioa98733f2011-09-16 10:12:28 -040051
Joe Gregoriocb8103d2011-02-11 23:20:52 -050052def datafile(filename):
53 return os.path.join(DATA_DIR, filename)
Joe Gregorioba9ea7f2010-08-19 15:49:04 -040054
55
Joe Gregorioc5c5a372010-09-22 11:42:32 -040056class Utilities(unittest.TestCase):
57 def test_key2param(self):
58 self.assertEqual('max_results', key2param('max-results'))
59 self.assertEqual('x007_bond', key2param('007-bond'))
60
61
Joe Gregorioc0e0fe92011-03-04 16:16:55 -050062class DiscoveryErrors(unittest.TestCase):
63
64 def test_failed_to_parse_discovery_json(self):
65 self.http = HttpMock(datafile('malformed.json'), {'status': '200'})
66 try:
Joe Gregorio7b70f432011-11-09 10:18:51 -050067 plus = build('plus', 'v1', self.http)
Joe Gregorioc0e0fe92011-03-04 16:16:55 -050068 self.fail("should have raised an exception over malformed JSON.")
Joe Gregorio49396552011-03-08 10:39:00 -050069 except InvalidJsonError:
70 pass
Joe Gregorioc0e0fe92011-03-04 16:16:55 -050071
72
ade@google.com6a8c1cb2011-09-06 17:40:00 +010073class DiscoveryFromDocument(unittest.TestCase):
Joe Gregorioa98733f2011-09-16 10:12:28 -040074
ade@google.com6a8c1cb2011-09-06 17:40:00 +010075 def test_can_build_from_local_document(self):
Joe Gregorio7b70f432011-11-09 10:18:51 -050076 discovery = file(datafile('plus.json')).read()
77 plus = build_from_document(discovery, base="https://www.googleapis.com/")
78 self.assertTrue(plus is not None)
Joe Gregorioa98733f2011-09-16 10:12:28 -040079
ade@google.com6a8c1cb2011-09-06 17:40:00 +010080 def test_building_with_base_remembers_base(self):
Joe Gregorio7b70f432011-11-09 10:18:51 -050081 discovery = file(datafile('plus.json')).read()
Joe Gregorioa98733f2011-09-16 10:12:28 -040082
ade@google.com6a8c1cb2011-09-06 17:40:00 +010083 base = "https://www.example.com/"
Joe Gregorio7b70f432011-11-09 10:18:51 -050084 plus = build_from_document(discovery, base=base)
85 self.assertEquals(base + "plus/v1/", plus._baseUrl)
ade@google.com6a8c1cb2011-09-06 17:40:00 +010086
87
Joe Gregorioa98733f2011-09-16 10:12:28 -040088class DiscoveryFromHttp(unittest.TestCase):
Joe Gregorio583d9e42011-09-16 15:54:15 -040089 def setUp(self):
Joe Bedafb463cb2011-09-19 17:39:49 -070090 self.old_environ = os.environ.copy()
Joe Gregorioa98733f2011-09-16 10:12:28 -040091
Joe Gregorio583d9e42011-09-16 15:54:15 -040092 def tearDown(self):
93 os.environ = self.old_environ
94
95 def test_userip_is_added_to_discovery_uri(self):
Joe Gregorioa98733f2011-09-16 10:12:28 -040096 # build() will raise an HttpError on a 400, use this to pick the request uri
97 # out of the raised exception.
Joe Gregorio583d9e42011-09-16 15:54:15 -040098 os.environ['REMOTE_ADDR'] = '10.0.0.1'
Joe Gregorioa98733f2011-09-16 10:12:28 -040099 try:
100 http = HttpMockSequence([
101 ({'status': '400'}, file(datafile('zoo.json'), 'r').read()),
102 ])
103 zoo = build('zoo', 'v1', http, developerKey='foo',
104 discoveryServiceUrl='http://example.com')
105 self.fail('Should have raised an exception.')
106 except HttpError, e:
Joe Gregorio583d9e42011-09-16 15:54:15 -0400107 self.assertEqual(e.uri, 'http://example.com?userIp=10.0.0.1')
Joe Gregorioa98733f2011-09-16 10:12:28 -0400108
Joe Gregorio583d9e42011-09-16 15:54:15 -0400109 def test_userip_missing_is_not_added_to_discovery_uri(self):
Joe Gregorioa98733f2011-09-16 10:12:28 -0400110 # build() will raise an HttpError on a 400, use this to pick the request uri
111 # out of the raised exception.
112 try:
113 http = HttpMockSequence([
114 ({'status': '400'}, file(datafile('zoo.json'), 'r').read()),
115 ])
116 zoo = build('zoo', 'v1', http, developerKey=None,
117 discoveryServiceUrl='http://example.com')
118 self.fail('Should have raised an exception.')
119 except HttpError, e:
120 self.assertEqual(e.uri, 'http://example.com')
121
122
Joe Gregorioba9ea7f2010-08-19 15:49:04 -0400123class Discovery(unittest.TestCase):
Joe Gregorio2379ecc2010-10-26 10:51:28 -0400124
Joe Gregorioba9ea7f2010-08-19 15:49:04 -0400125 def test_method_error_checking(self):
Joe Gregorio7b70f432011-11-09 10:18:51 -0500126 self.http = HttpMock(datafile('plus.json'), {'status': '200'})
127 plus = build('plus', 'v1', self.http)
Joe Gregorioba9ea7f2010-08-19 15:49:04 -0400128
129 # Missing required parameters
130 try:
Joe Gregorio7b70f432011-11-09 10:18:51 -0500131 plus.activities().list()
Joe Gregorioba9ea7f2010-08-19 15:49:04 -0400132 self.fail()
133 except TypeError, e:
134 self.assertTrue('Missing' in str(e))
135
136 # Parameter doesn't match regex
137 try:
Joe Gregorio7b70f432011-11-09 10:18:51 -0500138 plus.activities().list(collection='not_a_collection_name', userId='me')
Joe Gregorioba9ea7f2010-08-19 15:49:04 -0400139 self.fail()
140 except TypeError, e:
Joe Gregorioca876e42011-02-22 19:39:42 -0500141 self.assertTrue('not an allowed value' in str(e))
Joe Gregorioba9ea7f2010-08-19 15:49:04 -0400142
143 # Unexpected parameter
144 try:
Joe Gregorio7b70f432011-11-09 10:18:51 -0500145 plus.activities().list(flubber=12)
Joe Gregorioba9ea7f2010-08-19 15:49:04 -0400146 self.fail()
147 except TypeError, e:
148 self.assertTrue('unexpected' in str(e))
149
Joe Gregoriobee86832011-02-22 10:00:19 -0500150 def _check_query_types(self, request):
151 parsed = urlparse.urlparse(request.uri)
152 q = parse_qs(parsed[4])
153 self.assertEqual(q['q'], ['foo'])
154 self.assertEqual(q['i'], ['1'])
155 self.assertEqual(q['n'], ['1.0'])
156 self.assertEqual(q['b'], ['false'])
157 self.assertEqual(q['a'], ['[1, 2, 3]'])
158 self.assertEqual(q['o'], ['{\'a\': 1}'])
159 self.assertEqual(q['e'], ['bar'])
160
161 def test_type_coercion(self):
Joe Gregoriof4153422011-03-18 22:45:18 -0400162 http = HttpMock(datafile('zoo.json'), {'status': '200'})
163 zoo = build('zoo', 'v1', http)
Joe Gregoriobee86832011-02-22 10:00:19 -0500164
Joe Gregorioa98733f2011-09-16 10:12:28 -0400165 request = zoo.query(
166 q="foo", i=1.0, n=1.0, b=0, a=[1,2,3], o={'a':1}, e='bar')
Joe Gregoriobee86832011-02-22 10:00:19 -0500167 self._check_query_types(request)
Joe Gregorioa98733f2011-09-16 10:12:28 -0400168 request = zoo.query(
169 q="foo", i=1, n=1, b=False, a=[1,2,3], o={'a':1}, e='bar')
Joe Gregoriobee86832011-02-22 10:00:19 -0500170 self._check_query_types(request)
Joe Gregoriof863f7a2011-02-24 03:24:44 -0500171
Joe Gregorioa98733f2011-09-16 10:12:28 -0400172 request = zoo.query(
173 q="foo", i="1", n="1", b="", a=[1,2,3], o={'a':1}, e='bar')
Joe Gregorio6804c7a2011-11-18 14:30:32 -0500174
175 request = zoo.query(
176 q="foo", i="1", n="1", b="", a=[1,2,3], o={'a':1}, e='bar', rr=['foo',
177 'bar'])
Joe Gregoriobee86832011-02-22 10:00:19 -0500178 self._check_query_types(request)
179
Joe Gregorio13217952011-02-22 15:37:38 -0500180 def test_optional_stack_query_parameters(self):
Joe Gregoriof4153422011-03-18 22:45:18 -0400181 http = HttpMock(datafile('zoo.json'), {'status': '200'})
182 zoo = build('zoo', 'v1', http)
183 request = zoo.query(trace='html', fields='description')
Joe Gregorio13217952011-02-22 15:37:38 -0500184
Joe Gregorioca876e42011-02-22 19:39:42 -0500185 parsed = urlparse.urlparse(request.uri)
186 q = parse_qs(parsed[4])
187 self.assertEqual(q['trace'], ['html'])
Joe Gregoriof4153422011-03-18 22:45:18 -0400188 self.assertEqual(q['fields'], ['description'])
189
Joe Gregorioe08a1662011-12-07 09:48:22 -0500190 def test_model_added_query_parameters(self):
191 http = HttpMock(datafile('zoo.json'), {'status': '200'})
192 zoo = build('zoo', 'v1', http)
193 request = zoo.animals().get(name='Lion')
194
195 parsed = urlparse.urlparse(request.uri)
196 q = parse_qs(parsed[4])
197 self.assertEqual(q['alt'], ['json'])
198 self.assertEqual(request.headers['accept'], 'application/json')
199
200 def test_fallback_to_raw_model(self):
201 http = HttpMock(datafile('zoo.json'), {'status': '200'})
202 zoo = build('zoo', 'v1', http)
203 request = zoo.animals().getmedia(name='Lion')
204
205 parsed = urlparse.urlparse(request.uri)
206 q = parse_qs(parsed[4])
207 self.assertTrue('alt' not in q)
208 self.assertEqual(request.headers['accept'], '*/*')
209
Joe Gregoriof4153422011-03-18 22:45:18 -0400210 def test_patch(self):
211 http = HttpMock(datafile('zoo.json'), {'status': '200'})
212 zoo = build('zoo', 'v1', http)
213 request = zoo.animals().patch(name='lion', body='{"description": "foo"}')
214
215 self.assertEqual(request.method, 'PATCH')
216
217 def test_tunnel_patch(self):
218 http = HttpMockSequence([
219 ({'status': '200'}, file(datafile('zoo.json'), 'r').read()),
220 ({'status': '200'}, 'echo_request_headers_as_json'),
221 ])
222 http = tunnel_patch(http)
223 zoo = build('zoo', 'v1', http)
Joe Gregorioa98733f2011-09-16 10:12:28 -0400224 resp = zoo.animals().patch(
225 name='lion', body='{"description": "foo"}').execute()
Joe Gregoriof4153422011-03-18 22:45:18 -0400226
227 self.assertTrue('x-http-method-override' in resp)
Joe Gregorioca876e42011-02-22 19:39:42 -0500228
Joe Gregorio7b70f432011-11-09 10:18:51 -0500229 def test_plus_resources(self):
230 self.http = HttpMock(datafile('plus.json'), {'status': '200'})
231 plus = build('plus', 'v1', self.http)
232 self.assertTrue(getattr(plus, 'activities'))
233 self.assertTrue(getattr(plus, 'people'))
Joe Gregorioc5c5a372010-09-22 11:42:32 -0400234
Joe Gregorio2379ecc2010-10-26 10:51:28 -0400235 def test_full_featured(self):
236 # Zoo should exercise all discovery facets
237 # and should also have no future.json file.
Joe Gregoriocb8103d2011-02-11 23:20:52 -0500238 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
Joe Gregorio2379ecc2010-10-26 10:51:28 -0400239 zoo = build('zoo', 'v1', self.http)
240 self.assertTrue(getattr(zoo, 'animals'))
Joe Gregoriof863f7a2011-02-24 03:24:44 -0500241
Joe Gregoriof4153422011-03-18 22:45:18 -0400242 request = zoo.animals().list(name='bat', projection="full")
Joe Gregorio2379ecc2010-10-26 10:51:28 -0400243 parsed = urlparse.urlparse(request.uri)
244 q = parse_qs(parsed[4])
245 self.assertEqual(q['name'], ['bat'])
Joe Gregoriof4153422011-03-18 22:45:18 -0400246 self.assertEqual(q['projection'], ['full'])
Joe Gregorio2379ecc2010-10-26 10:51:28 -0400247
Joe Gregorio3fada332011-01-07 17:07:45 -0500248 def test_nested_resources(self):
Joe Gregoriocb8103d2011-02-11 23:20:52 -0500249 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
Joe Gregorio3fada332011-01-07 17:07:45 -0500250 zoo = build('zoo', 'v1', self.http)
251 self.assertTrue(getattr(zoo, 'animals'))
252 request = zoo.my().favorites().list(max_results="5")
253 parsed = urlparse.urlparse(request.uri)
254 q = parse_qs(parsed[4])
255 self.assertEqual(q['max-results'], ['5'])
256
Joe Gregoriod92897c2011-07-07 11:44:56 -0400257 def test_methods_with_reserved_names(self):
258 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
259 zoo = build('zoo', 'v1', self.http)
260 self.assertTrue(getattr(zoo, 'animals'))
261 request = zoo.global_().print_().assert_(max_results="5")
262 parsed = urlparse.urlparse(request.uri)
263 self.assertEqual(parsed[2], '/zoo/global/print/assert')
264
Joe Gregorio7a6df3a2011-01-31 21:55:21 -0500265 def test_top_level_functions(self):
Joe Gregoriocb8103d2011-02-11 23:20:52 -0500266 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
Joe Gregorio7a6df3a2011-01-31 21:55:21 -0500267 zoo = build('zoo', 'v1', self.http)
268 self.assertTrue(getattr(zoo, 'query'))
269 request = zoo.query(q="foo")
270 parsed = urlparse.urlparse(request.uri)
271 q = parse_qs(parsed[4])
272 self.assertEqual(q['q'], ['foo'])
Joe Gregorio2379ecc2010-10-26 10:51:28 -0400273
Joe Gregoriofdf7c802011-06-30 12:33:38 -0400274 def test_simple_media_uploads(self):
275 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
276 zoo = build('zoo', 'v1', self.http)
277 doc = getattr(zoo.animals().insert, '__doc__')
278 self.assertTrue('media_body' in doc)
279
Joe Gregorio84d3c1f2011-07-25 10:39:45 -0400280 def test_simple_media_upload_no_max_size_provided(self):
281 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
282 zoo = build('zoo', 'v1', self.http)
283 request = zoo.animals().crossbreed(media_body=datafile('small.png'))
284 self.assertEquals('image/png', request.headers['content-type'])
285 self.assertEquals('PNG', request.body[1:4])
286
Joe Gregoriofdf7c802011-06-30 12:33:38 -0400287 def test_simple_media_raise_correct_exceptions(self):
288 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
289 zoo = build('zoo', 'v1', self.http)
290
291 try:
292 zoo.animals().insert(media_body=datafile('smiley.png'))
293 self.fail("should throw exception if media is too large.")
294 except MediaUploadSizeError:
295 pass
296
297 try:
298 zoo.animals().insert(media_body=datafile('small.jpg'))
299 self.fail("should throw exception if mimetype is unacceptable.")
300 except UnacceptableMimeTypeError:
301 pass
302
303 def test_simple_media_good_upload(self):
304 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
305 zoo = build('zoo', 'v1', self.http)
306
307 request = zoo.animals().insert(media_body=datafile('small.png'))
308 self.assertEquals('image/png', request.headers['content-type'])
309 self.assertEquals('PNG', request.body[1:4])
310
311 def test_multipart_media_raise_correct_exceptions(self):
312 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
313 zoo = build('zoo', 'v1', self.http)
314
315 try:
316 zoo.animals().insert(media_body=datafile('smiley.png'), body={})
317 self.fail("should throw exception if media is too large.")
318 except MediaUploadSizeError:
319 pass
320
321 try:
322 zoo.animals().insert(media_body=datafile('small.jpg'), body={})
323 self.fail("should throw exception if mimetype is unacceptable.")
324 except UnacceptableMimeTypeError:
325 pass
326
327 def test_multipart_media_good_upload(self):
328 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
329 zoo = build('zoo', 'v1', self.http)
330
331 request = zoo.animals().insert(media_body=datafile('small.png'), body={})
Joe Gregorioa98733f2011-09-16 10:12:28 -0400332 self.assertTrue(request.headers['content-type'].startswith(
333 'multipart/related'))
Joe Gregoriofdf7c802011-06-30 12:33:38 -0400334 self.assertEquals('--==', request.body[0:4])
335
336 def test_media_capable_method_without_media(self):
337 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
338 zoo = build('zoo', 'v1', self.http)
339
340 request = zoo.animals().insert(body={})
341 self.assertTrue(request.headers['content-type'], 'application/json')
Joe Gregorioc5c5a372010-09-22 11:42:32 -0400342
Joe Gregoriod0bd3882011-11-22 09:49:47 -0500343 def test_resumable_multipart_media_good_upload(self):
344 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
345 zoo = build('zoo', 'v1', self.http)
346
347 media_upload = MediaFileUpload(datafile('small.png'), resumable=True)
348 request = zoo.animals().insert(media_body=media_upload, body={})
349 self.assertTrue(request.headers['content-type'].startswith(
350 'multipart/related'))
351 self.assertEquals('--==', request.body[0:4])
352 self.assertEquals(media_upload, request.resumable)
353
354 self.assertEquals('image/png', request.resumable.mimetype())
355
356 self.assertTrue(len(request.multipart_boundary) > 0)
357 self.assertNotEquals(request.body, None)
358 self.assertEquals(request.resumable_uri, None)
359
360 http = HttpMockSequence([
361 ({'status': '200',
362 'location': 'http://upload.example.com'}, ''),
363 ({'status': '308',
364 'location': 'http://upload.example.com/2',
365 'range': '0-12'}, ''),
366 ({'status': '308',
367 'location': 'http://upload.example.com/3',
368 'range': '0-%d' % (request.total_size - 2)}, ''),
369 ({'status': '200'}, '{"foo": "bar"}'),
370 ])
371
372 status, body = request.next_chunk(http)
373 self.assertEquals(None, body)
374 self.assertTrue(isinstance(status, MediaUploadProgress))
375 self.assertEquals(13, status.resumable_progress)
376
377 # request.body is not None because the server only acknowledged 12 bytes,
378 # which is less than the size of the body, so we need to send it again.
379 self.assertNotEquals(request.body, None)
380
381 # Two requests should have been made and the resumable_uri should have been
382 # updated for each one.
383 self.assertEquals(request.resumable_uri, 'http://upload.example.com/2')
384
385 self.assertEquals(media_upload, request.resumable)
386 self.assertEquals(13, request.resumable_progress)
387
388 status, body = request.next_chunk(http)
389 self.assertEquals(request.resumable_uri, 'http://upload.example.com/3')
390 self.assertEquals(request.total_size-1, request.resumable_progress)
391 self.assertEquals(request.body, None)
392
393 # Final call to next_chunk should complete the upload.
394 status, body = request.next_chunk(http)
395 self.assertEquals(body, {"foo": "bar"})
396 self.assertEquals(status, None)
397
398
399 def test_resumable_media_good_upload(self):
400 """Not a multipart upload."""
401 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
402 zoo = build('zoo', 'v1', self.http)
403
404 media_upload = MediaFileUpload(datafile('small.png'), resumable=True)
405 request = zoo.animals().insert(media_body=media_upload, body=None)
406 self.assertTrue(request.headers['content-type'].startswith(
407 'image/png'))
408 self.assertEquals(media_upload, request.resumable)
409
410 self.assertEquals('image/png', request.resumable.mimetype())
411
Joe Gregoriobd512b52011-12-06 15:39:26 -0500412 self.assertEquals(request.multipart_boundary, '--')
Joe Gregoriod0bd3882011-11-22 09:49:47 -0500413 self.assertEquals(request.body, None)
414 self.assertEquals(request.resumable_uri, None)
415
416 http = HttpMockSequence([
417 ({'status': '200',
418 'location': 'http://upload.example.com'}, ''),
419 ({'status': '308',
420 'location': 'http://upload.example.com/2',
421 'range': '0-12'}, ''),
422 ({'status': '308',
423 'location': 'http://upload.example.com/3',
424 'range': '0-%d' % (request.total_size - 2)}, ''),
425 ({'status': '200'}, '{"foo": "bar"}'),
426 ])
427
428 status, body = request.next_chunk(http)
429 self.assertEquals(None, body)
430 self.assertTrue(isinstance(status, MediaUploadProgress))
431 self.assertEquals(13, status.resumable_progress)
432
433 # Two requests should have been made and the resumable_uri should have been
434 # updated for each one.
435 self.assertEquals(request.resumable_uri, 'http://upload.example.com/2')
436
437 self.assertEquals(media_upload, request.resumable)
438 self.assertEquals(13, request.resumable_progress)
439
440 status, body = request.next_chunk(http)
441 self.assertEquals(request.resumable_uri, 'http://upload.example.com/3')
442 self.assertEquals(request.total_size-1, request.resumable_progress)
443 self.assertEquals(request.body, None)
444
445 # Final call to next_chunk should complete the upload.
446 status, body = request.next_chunk(http)
447 self.assertEquals(body, {"foo": "bar"})
448 self.assertEquals(status, None)
449
450
451 def test_resumable_media_good_upload_from_execute(self):
452 """Not a multipart upload."""
453 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
454 zoo = build('zoo', 'v1', self.http)
455
456 media_upload = MediaFileUpload(datafile('small.png'), resumable=True)
457 request = zoo.animals().insert(media_body=media_upload, body=None)
458
459 http = HttpMockSequence([
460 ({'status': '200',
461 'location': 'http://upload.example.com'}, ''),
462 ({'status': '308',
463 'location': 'http://upload.example.com/2',
464 'range': '0-12'}, ''),
465 ({'status': '308',
466 'location': 'http://upload.example.com/3',
467 'range': '0-%d' % (request.total_size - 2)}, ''),
468 ({'status': '200'}, '{"foo": "bar"}'),
469 ])
470
471 body = request.execute(http)
472 self.assertEquals(body, {"foo": "bar"})
473
474 def test_resumable_media_fail_unknown_response_code_first_request(self):
475 """Not a multipart upload."""
476 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
477 zoo = build('zoo', 'v1', self.http)
478
479 media_upload = MediaFileUpload(datafile('small.png'), resumable=True)
480 request = zoo.animals().insert(media_body=media_upload, body=None)
481
482 http = HttpMockSequence([
483 ({'status': '400',
484 'location': 'http://upload.example.com'}, ''),
485 ])
486
487 self.assertRaises(ResumableUploadError, request.execute, http)
488
489 def test_resumable_media_fail_unknown_response_code_subsequent_request(self):
490 """Not a multipart upload."""
491 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
492 zoo = build('zoo', 'v1', self.http)
493
494 media_upload = MediaFileUpload(datafile('small.png'), resumable=True)
495 request = zoo.animals().insert(media_body=media_upload, body=None)
496
497 http = HttpMockSequence([
498 ({'status': '200',
499 'location': 'http://upload.example.com'}, ''),
500 ({'status': '400'}, ''),
501 ])
502
503 self.assertRaises(HttpError, request.execute, http)
504
505
Joe Gregorioc5c5a372010-09-22 11:42:32 -0400506class Next(unittest.TestCase):
Joe Gregorio00cf1d92010-09-27 09:22:03 -0400507
Joe Gregorio3c676f92011-07-25 10:38:14 -0400508 def test_next_successful_none_on_no_next_page_token(self):
509 self.http = HttpMock(datafile('tasks.json'), {'status': '200'})
510 tasks = build('tasks', 'v1', self.http)
511 request = tasks.tasklists().list()
512 self.assertEqual(None, tasks.tasklists().list_next(request, {}))
513
514 def test_next_successful_with_next_page_token(self):
515 self.http = HttpMock(datafile('tasks.json'), {'status': '200'})
516 tasks = build('tasks', 'v1', self.http)
517 request = tasks.tasklists().list()
Joe Gregorioa98733f2011-09-16 10:12:28 -0400518 next_request = tasks.tasklists().list_next(
519 request, {'nextPageToken': '123abc'})
Joe Gregorio3c676f92011-07-25 10:38:14 -0400520 parsed = list(urlparse.urlparse(next_request.uri))
521 q = parse_qs(parsed[4])
522 self.assertEqual(q['pageToken'][0], '123abc')
523
Joe Gregorio555f33c2011-08-19 14:56:07 -0400524 def test_next_with_method_with_no_properties(self):
525 self.http = HttpMock(datafile('latitude.json'), {'status': '200'})
526 service = build('latitude', 'v1', self.http)
527 request = service.currentLocation().get()
Joe Gregorio00cf1d92010-09-27 09:22:03 -0400528
Joe Gregorioa98733f2011-09-16 10:12:28 -0400529
Joe Gregorioba9ea7f2010-08-19 15:49:04 -0400530if __name__ == '__main__':
531 unittest.main()