blob: b1656c4807c9ad9de4cb29c993cbf028148ca2ba [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 Gregorio910b9b12012-06-12 09:36:30 -040030import StringIO
31
Joe Gregorioa98733f2011-09-16 10:12:28 -040032
ade@google.comc5eb46f2010-09-27 23:35:39 +010033try:
34 from urlparse import parse_qs
35except ImportError:
36 from cgi import parse_qs
Joe Gregorio00cf1d92010-09-27 09:22:03 -040037
ade@google.com6a8c1cb2011-09-06 17:40:00 +010038from apiclient.discovery import build, build_from_document, key2param
Joe Gregorioc0e0fe92011-03-04 16:16:55 -050039from apiclient.errors import HttpError
Joe Gregorio49396552011-03-08 10:39:00 -050040from apiclient.errors import InvalidJsonError
Joe Gregoriofdf7c802011-06-30 12:33:38 -040041from apiclient.errors import MediaUploadSizeError
Joe Gregoriod0bd3882011-11-22 09:49:47 -050042from apiclient.errors import ResumableUploadError
Joe Gregoriofdf7c802011-06-30 12:33:38 -040043from apiclient.errors import UnacceptableMimeTypeError
Joe Gregoriod0bd3882011-11-22 09:49:47 -050044from apiclient.http import HttpMock
45from apiclient.http import HttpMockSequence
46from apiclient.http import MediaFileUpload
Joe Gregorio910b9b12012-06-12 09:36:30 -040047from apiclient.http import MediaIoBaseUpload
Joe Gregoriod0bd3882011-11-22 09:49:47 -050048from apiclient.http import MediaUploadProgress
49from apiclient.http import tunnel_patch
Joe Gregorio910b9b12012-06-12 09:36:30 -040050from oauth2client.anyjson import simplejson
Joe Gregoriocb8103d2011-02-11 23:20:52 -050051
52DATA_DIR = os.path.join(os.path.dirname(__file__), 'data')
53
Joe Gregorioa98733f2011-09-16 10:12:28 -040054
Joe Gregoriocb8103d2011-02-11 23:20:52 -050055def datafile(filename):
56 return os.path.join(DATA_DIR, filename)
Joe Gregorioba9ea7f2010-08-19 15:49:04 -040057
58
Joe Gregorioc5c5a372010-09-22 11:42:32 -040059class Utilities(unittest.TestCase):
60 def test_key2param(self):
61 self.assertEqual('max_results', key2param('max-results'))
62 self.assertEqual('x007_bond', key2param('007-bond'))
63
64
Joe Gregorioc0e0fe92011-03-04 16:16:55 -050065class DiscoveryErrors(unittest.TestCase):
66
67 def test_failed_to_parse_discovery_json(self):
68 self.http = HttpMock(datafile('malformed.json'), {'status': '200'})
69 try:
Joe Gregorio7b70f432011-11-09 10:18:51 -050070 plus = build('plus', 'v1', self.http)
Joe Gregorioc0e0fe92011-03-04 16:16:55 -050071 self.fail("should have raised an exception over malformed JSON.")
Joe Gregorio49396552011-03-08 10:39:00 -050072 except InvalidJsonError:
73 pass
Joe Gregorioc0e0fe92011-03-04 16:16:55 -050074
75
ade@google.com6a8c1cb2011-09-06 17:40:00 +010076class DiscoveryFromDocument(unittest.TestCase):
Joe Gregorioa98733f2011-09-16 10:12:28 -040077
ade@google.com6a8c1cb2011-09-06 17:40:00 +010078 def test_can_build_from_local_document(self):
Joe Gregorio7b70f432011-11-09 10:18:51 -050079 discovery = file(datafile('plus.json')).read()
80 plus = build_from_document(discovery, base="https://www.googleapis.com/")
81 self.assertTrue(plus is not None)
Joe Gregorioa98733f2011-09-16 10:12:28 -040082
ade@google.com6a8c1cb2011-09-06 17:40:00 +010083 def test_building_with_base_remembers_base(self):
Joe Gregorio7b70f432011-11-09 10:18:51 -050084 discovery = file(datafile('plus.json')).read()
Joe Gregorioa98733f2011-09-16 10:12:28 -040085
ade@google.com6a8c1cb2011-09-06 17:40:00 +010086 base = "https://www.example.com/"
Joe Gregorio7b70f432011-11-09 10:18:51 -050087 plus = build_from_document(discovery, base=base)
Joe Gregorioa2838152012-07-16 11:52:17 -040088 self.assertEquals("https://www.googleapis.com/plus/v1/", plus._baseUrl)
ade@google.com6a8c1cb2011-09-06 17:40:00 +010089
90
Joe Gregorioa98733f2011-09-16 10:12:28 -040091class DiscoveryFromHttp(unittest.TestCase):
Joe Gregorio583d9e42011-09-16 15:54:15 -040092 def setUp(self):
Joe Bedafb463cb2011-09-19 17:39:49 -070093 self.old_environ = os.environ.copy()
Joe Gregorioa98733f2011-09-16 10:12:28 -040094
Joe Gregorio583d9e42011-09-16 15:54:15 -040095 def tearDown(self):
96 os.environ = self.old_environ
97
98 def test_userip_is_added_to_discovery_uri(self):
Joe Gregorioa98733f2011-09-16 10:12:28 -040099 # build() will raise an HttpError on a 400, use this to pick the request uri
100 # out of the raised exception.
Joe Gregorio583d9e42011-09-16 15:54:15 -0400101 os.environ['REMOTE_ADDR'] = '10.0.0.1'
Joe Gregorioa98733f2011-09-16 10:12:28 -0400102 try:
103 http = HttpMockSequence([
104 ({'status': '400'}, file(datafile('zoo.json'), 'r').read()),
105 ])
106 zoo = build('zoo', 'v1', http, developerKey='foo',
107 discoveryServiceUrl='http://example.com')
108 self.fail('Should have raised an exception.')
109 except HttpError, e:
Joe Gregorio583d9e42011-09-16 15:54:15 -0400110 self.assertEqual(e.uri, 'http://example.com?userIp=10.0.0.1')
Joe Gregorioa98733f2011-09-16 10:12:28 -0400111
Joe Gregorio583d9e42011-09-16 15:54:15 -0400112 def test_userip_missing_is_not_added_to_discovery_uri(self):
Joe Gregorioa98733f2011-09-16 10:12:28 -0400113 # build() will raise an HttpError on a 400, use this to pick the request uri
114 # out of the raised exception.
115 try:
116 http = HttpMockSequence([
117 ({'status': '400'}, file(datafile('zoo.json'), 'r').read()),
118 ])
119 zoo = build('zoo', 'v1', http, developerKey=None,
120 discoveryServiceUrl='http://example.com')
121 self.fail('Should have raised an exception.')
122 except HttpError, e:
123 self.assertEqual(e.uri, 'http://example.com')
124
125
Joe Gregorioba9ea7f2010-08-19 15:49:04 -0400126class Discovery(unittest.TestCase):
Joe Gregorio2379ecc2010-10-26 10:51:28 -0400127
Joe Gregorioba9ea7f2010-08-19 15:49:04 -0400128 def test_method_error_checking(self):
Joe Gregorio7b70f432011-11-09 10:18:51 -0500129 self.http = HttpMock(datafile('plus.json'), {'status': '200'})
130 plus = build('plus', 'v1', self.http)
Joe Gregorioba9ea7f2010-08-19 15:49:04 -0400131
132 # Missing required parameters
133 try:
Joe Gregorio7b70f432011-11-09 10:18:51 -0500134 plus.activities().list()
Joe Gregorioba9ea7f2010-08-19 15:49:04 -0400135 self.fail()
136 except TypeError, e:
137 self.assertTrue('Missing' in str(e))
138
Joe Gregorio2467afa2012-06-20 12:21:25 -0400139 # Missing required parameters even if supplied as None.
140 try:
141 plus.activities().list(collection=None, userId=None)
142 self.fail()
143 except TypeError, e:
144 self.assertTrue('Missing' in str(e))
145
Joe Gregorioba9ea7f2010-08-19 15:49:04 -0400146 # Parameter doesn't match regex
147 try:
Joe Gregorio7b70f432011-11-09 10:18:51 -0500148 plus.activities().list(collection='not_a_collection_name', userId='me')
Joe Gregorioba9ea7f2010-08-19 15:49:04 -0400149 self.fail()
150 except TypeError, e:
Joe Gregorioca876e42011-02-22 19:39:42 -0500151 self.assertTrue('not an allowed value' in str(e))
Joe Gregorioba9ea7f2010-08-19 15:49:04 -0400152
153 # Unexpected parameter
154 try:
Joe Gregorio7b70f432011-11-09 10:18:51 -0500155 plus.activities().list(flubber=12)
Joe Gregorioba9ea7f2010-08-19 15:49:04 -0400156 self.fail()
157 except TypeError, e:
158 self.assertTrue('unexpected' in str(e))
159
Joe Gregoriobee86832011-02-22 10:00:19 -0500160 def _check_query_types(self, request):
161 parsed = urlparse.urlparse(request.uri)
162 q = parse_qs(parsed[4])
163 self.assertEqual(q['q'], ['foo'])
164 self.assertEqual(q['i'], ['1'])
165 self.assertEqual(q['n'], ['1.0'])
166 self.assertEqual(q['b'], ['false'])
167 self.assertEqual(q['a'], ['[1, 2, 3]'])
168 self.assertEqual(q['o'], ['{\'a\': 1}'])
169 self.assertEqual(q['e'], ['bar'])
170
171 def test_type_coercion(self):
Joe Gregoriof4153422011-03-18 22:45:18 -0400172 http = HttpMock(datafile('zoo.json'), {'status': '200'})
173 zoo = build('zoo', 'v1', http)
Joe Gregoriobee86832011-02-22 10:00:19 -0500174
Joe Gregorioa98733f2011-09-16 10:12:28 -0400175 request = zoo.query(
176 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 -0500177 self._check_query_types(request)
Joe Gregorioa98733f2011-09-16 10:12:28 -0400178 request = zoo.query(
179 q="foo", i=1, n=1, b=False, a=[1,2,3], o={'a':1}, e='bar')
Joe Gregoriobee86832011-02-22 10:00:19 -0500180 self._check_query_types(request)
Joe Gregoriof863f7a2011-02-24 03:24:44 -0500181
Joe Gregorioa98733f2011-09-16 10:12:28 -0400182 request = zoo.query(
Craig Citro1e742822012-03-01 12:59:22 -0800183 q="foo", i="1", n="1", b="", a=[1,2,3], o={'a':1}, e='bar', er='two')
Joe Gregorio6804c7a2011-11-18 14:30:32 -0500184
185 request = zoo.query(
Craig Citro1e742822012-03-01 12:59:22 -0800186 q="foo", i="1", n="1", b="", a=[1,2,3], o={'a':1}, e='bar',
187 er=['one', 'three'], rr=['foo', 'bar'])
Joe Gregoriobee86832011-02-22 10:00:19 -0500188 self._check_query_types(request)
189
Craig Citro1e742822012-03-01 12:59:22 -0800190 # Five is right out.
Joe Gregorio20c26e52012-03-02 15:58:31 -0500191 self.assertRaises(TypeError, zoo.query, er=['one', 'five'])
Craig Citro1e742822012-03-01 12:59:22 -0800192
Joe Gregorio13217952011-02-22 15:37:38 -0500193 def test_optional_stack_query_parameters(self):
Joe Gregoriof4153422011-03-18 22:45:18 -0400194 http = HttpMock(datafile('zoo.json'), {'status': '200'})
195 zoo = build('zoo', 'v1', http)
196 request = zoo.query(trace='html', fields='description')
Joe Gregorio13217952011-02-22 15:37:38 -0500197
Joe Gregorioca876e42011-02-22 19:39:42 -0500198 parsed = urlparse.urlparse(request.uri)
199 q = parse_qs(parsed[4])
200 self.assertEqual(q['trace'], ['html'])
Joe Gregoriof4153422011-03-18 22:45:18 -0400201 self.assertEqual(q['fields'], ['description'])
202
Joe Gregorio2467afa2012-06-20 12:21:25 -0400203 def test_string_params_value_of_none_get_dropped(self):
Joe Gregorio4b4002f2012-06-14 15:41:01 -0400204 http = HttpMock(datafile('zoo.json'), {'status': '200'})
205 zoo = build('zoo', 'v1', http)
Joe Gregorio2467afa2012-06-20 12:21:25 -0400206 request = zoo.query(trace=None, fields='description')
207
208 parsed = urlparse.urlparse(request.uri)
209 q = parse_qs(parsed[4])
210 self.assertFalse('trace' in q)
Joe Gregorio4b4002f2012-06-14 15:41:01 -0400211
Joe Gregorioe08a1662011-12-07 09:48:22 -0500212 def test_model_added_query_parameters(self):
213 http = HttpMock(datafile('zoo.json'), {'status': '200'})
214 zoo = build('zoo', 'v1', http)
215 request = zoo.animals().get(name='Lion')
216
217 parsed = urlparse.urlparse(request.uri)
218 q = parse_qs(parsed[4])
219 self.assertEqual(q['alt'], ['json'])
220 self.assertEqual(request.headers['accept'], 'application/json')
221
222 def test_fallback_to_raw_model(self):
223 http = HttpMock(datafile('zoo.json'), {'status': '200'})
224 zoo = build('zoo', 'v1', http)
225 request = zoo.animals().getmedia(name='Lion')
226
227 parsed = urlparse.urlparse(request.uri)
228 q = parse_qs(parsed[4])
229 self.assertTrue('alt' not in q)
230 self.assertEqual(request.headers['accept'], '*/*')
231
Joe Gregoriof4153422011-03-18 22:45:18 -0400232 def test_patch(self):
233 http = HttpMock(datafile('zoo.json'), {'status': '200'})
234 zoo = build('zoo', 'v1', http)
235 request = zoo.animals().patch(name='lion', body='{"description": "foo"}')
236
237 self.assertEqual(request.method, 'PATCH')
238
239 def test_tunnel_patch(self):
240 http = HttpMockSequence([
241 ({'status': '200'}, file(datafile('zoo.json'), 'r').read()),
242 ({'status': '200'}, 'echo_request_headers_as_json'),
243 ])
244 http = tunnel_patch(http)
245 zoo = build('zoo', 'v1', http)
Joe Gregorioa98733f2011-09-16 10:12:28 -0400246 resp = zoo.animals().patch(
247 name='lion', body='{"description": "foo"}').execute()
Joe Gregoriof4153422011-03-18 22:45:18 -0400248
249 self.assertTrue('x-http-method-override' in resp)
Joe Gregorioca876e42011-02-22 19:39:42 -0500250
Joe Gregorio7b70f432011-11-09 10:18:51 -0500251 def test_plus_resources(self):
252 self.http = HttpMock(datafile('plus.json'), {'status': '200'})
253 plus = build('plus', 'v1', self.http)
254 self.assertTrue(getattr(plus, 'activities'))
255 self.assertTrue(getattr(plus, 'people'))
Joe Gregorioc5c5a372010-09-22 11:42:32 -0400256
Joe Gregorio2379ecc2010-10-26 10:51:28 -0400257 def test_full_featured(self):
258 # Zoo should exercise all discovery facets
259 # and should also have no future.json file.
Joe Gregoriocb8103d2011-02-11 23:20:52 -0500260 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
Joe Gregorio2379ecc2010-10-26 10:51:28 -0400261 zoo = build('zoo', 'v1', self.http)
262 self.assertTrue(getattr(zoo, 'animals'))
Joe Gregoriof863f7a2011-02-24 03:24:44 -0500263
Joe Gregoriof4153422011-03-18 22:45:18 -0400264 request = zoo.animals().list(name='bat', projection="full")
Joe Gregorio2379ecc2010-10-26 10:51:28 -0400265 parsed = urlparse.urlparse(request.uri)
266 q = parse_qs(parsed[4])
267 self.assertEqual(q['name'], ['bat'])
Joe Gregoriof4153422011-03-18 22:45:18 -0400268 self.assertEqual(q['projection'], ['full'])
Joe Gregorio2379ecc2010-10-26 10:51:28 -0400269
Joe Gregorio3fada332011-01-07 17:07:45 -0500270 def test_nested_resources(self):
Joe Gregoriocb8103d2011-02-11 23:20:52 -0500271 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
Joe Gregorio3fada332011-01-07 17:07:45 -0500272 zoo = build('zoo', 'v1', self.http)
273 self.assertTrue(getattr(zoo, 'animals'))
274 request = zoo.my().favorites().list(max_results="5")
275 parsed = urlparse.urlparse(request.uri)
276 q = parse_qs(parsed[4])
277 self.assertEqual(q['max-results'], ['5'])
278
Joe Gregoriod92897c2011-07-07 11:44:56 -0400279 def test_methods_with_reserved_names(self):
280 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
281 zoo = build('zoo', 'v1', self.http)
282 self.assertTrue(getattr(zoo, 'animals'))
283 request = zoo.global_().print_().assert_(max_results="5")
284 parsed = urlparse.urlparse(request.uri)
Joe Gregorioa2838152012-07-16 11:52:17 -0400285 self.assertEqual(parsed[2], '/zoo/v1/global/print/assert')
Joe Gregoriod92897c2011-07-07 11:44:56 -0400286
Joe Gregorio7a6df3a2011-01-31 21:55:21 -0500287 def test_top_level_functions(self):
Joe Gregoriocb8103d2011-02-11 23:20:52 -0500288 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
Joe Gregorio7a6df3a2011-01-31 21:55:21 -0500289 zoo = build('zoo', 'v1', self.http)
290 self.assertTrue(getattr(zoo, 'query'))
291 request = zoo.query(q="foo")
292 parsed = urlparse.urlparse(request.uri)
293 q = parse_qs(parsed[4])
294 self.assertEqual(q['q'], ['foo'])
Joe Gregorio2379ecc2010-10-26 10:51:28 -0400295
Joe Gregoriofdf7c802011-06-30 12:33:38 -0400296 def test_simple_media_uploads(self):
297 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
298 zoo = build('zoo', 'v1', self.http)
299 doc = getattr(zoo.animals().insert, '__doc__')
300 self.assertTrue('media_body' in doc)
301
Joe Gregorio84d3c1f2011-07-25 10:39:45 -0400302 def test_simple_media_upload_no_max_size_provided(self):
303 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
304 zoo = build('zoo', 'v1', self.http)
305 request = zoo.animals().crossbreed(media_body=datafile('small.png'))
306 self.assertEquals('image/png', request.headers['content-type'])
307 self.assertEquals('PNG', request.body[1:4])
308
Joe Gregoriofdf7c802011-06-30 12:33:38 -0400309 def test_simple_media_raise_correct_exceptions(self):
310 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
311 zoo = build('zoo', 'v1', self.http)
312
313 try:
314 zoo.animals().insert(media_body=datafile('smiley.png'))
315 self.fail("should throw exception if media is too large.")
316 except MediaUploadSizeError:
317 pass
318
319 try:
320 zoo.animals().insert(media_body=datafile('small.jpg'))
321 self.fail("should throw exception if mimetype is unacceptable.")
322 except UnacceptableMimeTypeError:
323 pass
324
325 def test_simple_media_good_upload(self):
326 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
327 zoo = build('zoo', 'v1', self.http)
328
329 request = zoo.animals().insert(media_body=datafile('small.png'))
330 self.assertEquals('image/png', request.headers['content-type'])
331 self.assertEquals('PNG', request.body[1:4])
Joe Gregoriode860442012-03-02 15:55:52 -0500332 self.assertEqual(
Joe Gregorioa2838152012-07-16 11:52:17 -0400333 'https://www.googleapis.com/upload/zoo/v1/animals?uploadType=media&alt=json',
Joe Gregoriode860442012-03-02 15:55:52 -0500334 request.uri)
Joe Gregoriofdf7c802011-06-30 12:33:38 -0400335
336 def test_multipart_media_raise_correct_exceptions(self):
337 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
338 zoo = build('zoo', 'v1', self.http)
339
340 try:
341 zoo.animals().insert(media_body=datafile('smiley.png'), body={})
342 self.fail("should throw exception if media is too large.")
343 except MediaUploadSizeError:
344 pass
345
346 try:
347 zoo.animals().insert(media_body=datafile('small.jpg'), body={})
348 self.fail("should throw exception if mimetype is unacceptable.")
349 except UnacceptableMimeTypeError:
350 pass
351
352 def test_multipart_media_good_upload(self):
353 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
354 zoo = build('zoo', 'v1', self.http)
355
356 request = zoo.animals().insert(media_body=datafile('small.png'), body={})
Joe Gregorioa98733f2011-09-16 10:12:28 -0400357 self.assertTrue(request.headers['content-type'].startswith(
358 'multipart/related'))
Joe Gregoriofdf7c802011-06-30 12:33:38 -0400359 self.assertEquals('--==', request.body[0:4])
Joe Gregoriode860442012-03-02 15:55:52 -0500360 self.assertEqual(
Joe Gregorioa2838152012-07-16 11:52:17 -0400361 'https://www.googleapis.com/upload/zoo/v1/animals?uploadType=multipart&alt=json',
Joe Gregoriode860442012-03-02 15:55:52 -0500362 request.uri)
Joe Gregoriofdf7c802011-06-30 12:33:38 -0400363
364 def test_media_capable_method_without_media(self):
365 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
366 zoo = build('zoo', 'v1', self.http)
367
368 request = zoo.animals().insert(body={})
369 self.assertTrue(request.headers['content-type'], 'application/json')
Joe Gregorioc5c5a372010-09-22 11:42:32 -0400370
Joe Gregoriod0bd3882011-11-22 09:49:47 -0500371 def test_resumable_multipart_media_good_upload(self):
372 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
373 zoo = build('zoo', 'v1', self.http)
374
375 media_upload = MediaFileUpload(datafile('small.png'), resumable=True)
376 request = zoo.animals().insert(media_body=media_upload, body={})
377 self.assertTrue(request.headers['content-type'].startswith(
Joe Gregorio945be3e2012-01-27 17:01:06 -0500378 'application/json'))
379 self.assertEquals('{"data": {}}', request.body)
Joe Gregoriod0bd3882011-11-22 09:49:47 -0500380 self.assertEquals(media_upload, request.resumable)
381
382 self.assertEquals('image/png', request.resumable.mimetype())
383
Joe Gregoriod0bd3882011-11-22 09:49:47 -0500384 self.assertNotEquals(request.body, None)
385 self.assertEquals(request.resumable_uri, None)
386
387 http = HttpMockSequence([
388 ({'status': '200',
389 'location': 'http://upload.example.com'}, ''),
390 ({'status': '308',
391 'location': 'http://upload.example.com/2',
392 'range': '0-12'}, ''),
393 ({'status': '308',
394 'location': 'http://upload.example.com/3',
Joe Gregorio945be3e2012-01-27 17:01:06 -0500395 'range': '0-%d' % (media_upload.size() - 2)}, ''),
Joe Gregoriod0bd3882011-11-22 09:49:47 -0500396 ({'status': '200'}, '{"foo": "bar"}'),
397 ])
398
399 status, body = request.next_chunk(http)
400 self.assertEquals(None, body)
401 self.assertTrue(isinstance(status, MediaUploadProgress))
402 self.assertEquals(13, status.resumable_progress)
403
Joe Gregoriod0bd3882011-11-22 09:49:47 -0500404 # Two requests should have been made and the resumable_uri should have been
405 # updated for each one.
406 self.assertEquals(request.resumable_uri, 'http://upload.example.com/2')
407
408 self.assertEquals(media_upload, request.resumable)
409 self.assertEquals(13, request.resumable_progress)
410
411 status, body = request.next_chunk(http)
412 self.assertEquals(request.resumable_uri, 'http://upload.example.com/3')
Joe Gregorio945be3e2012-01-27 17:01:06 -0500413 self.assertEquals(media_upload.size()-1, request.resumable_progress)
414 self.assertEquals('{"data": {}}', request.body)
Joe Gregoriod0bd3882011-11-22 09:49:47 -0500415
416 # Final call to next_chunk should complete the upload.
417 status, body = request.next_chunk(http)
418 self.assertEquals(body, {"foo": "bar"})
419 self.assertEquals(status, None)
420
421
422 def test_resumable_media_good_upload(self):
423 """Not a multipart upload."""
424 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
425 zoo = build('zoo', 'v1', self.http)
426
427 media_upload = MediaFileUpload(datafile('small.png'), resumable=True)
428 request = zoo.animals().insert(media_body=media_upload, body=None)
Joe Gregoriod0bd3882011-11-22 09:49:47 -0500429 self.assertEquals(media_upload, request.resumable)
430
431 self.assertEquals('image/png', request.resumable.mimetype())
432
Joe Gregoriod0bd3882011-11-22 09:49:47 -0500433 self.assertEquals(request.body, None)
434 self.assertEquals(request.resumable_uri, None)
435
436 http = HttpMockSequence([
437 ({'status': '200',
438 'location': 'http://upload.example.com'}, ''),
439 ({'status': '308',
440 'location': 'http://upload.example.com/2',
441 'range': '0-12'}, ''),
442 ({'status': '308',
443 'location': 'http://upload.example.com/3',
Joe Gregorio945be3e2012-01-27 17:01:06 -0500444 'range': '0-%d' % (media_upload.size() - 2)}, ''),
Joe Gregoriod0bd3882011-11-22 09:49:47 -0500445 ({'status': '200'}, '{"foo": "bar"}'),
446 ])
447
448 status, body = request.next_chunk(http)
449 self.assertEquals(None, body)
450 self.assertTrue(isinstance(status, MediaUploadProgress))
451 self.assertEquals(13, status.resumable_progress)
452
453 # Two requests should have been made and the resumable_uri should have been
454 # updated for each one.
455 self.assertEquals(request.resumable_uri, 'http://upload.example.com/2')
456
457 self.assertEquals(media_upload, request.resumable)
458 self.assertEquals(13, request.resumable_progress)
459
460 status, body = request.next_chunk(http)
461 self.assertEquals(request.resumable_uri, 'http://upload.example.com/3')
Joe Gregorio945be3e2012-01-27 17:01:06 -0500462 self.assertEquals(media_upload.size()-1, request.resumable_progress)
Joe Gregoriod0bd3882011-11-22 09:49:47 -0500463 self.assertEquals(request.body, None)
464
465 # Final call to next_chunk should complete the upload.
466 status, body = request.next_chunk(http)
467 self.assertEquals(body, {"foo": "bar"})
468 self.assertEquals(status, None)
469
470
471 def test_resumable_media_good_upload_from_execute(self):
472 """Not a multipart upload."""
473 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
474 zoo = build('zoo', 'v1', self.http)
475
476 media_upload = MediaFileUpload(datafile('small.png'), resumable=True)
477 request = zoo.animals().insert(media_body=media_upload, body=None)
Joe Gregoriode860442012-03-02 15:55:52 -0500478 self.assertEqual(
Joe Gregorioa2838152012-07-16 11:52:17 -0400479 'https://www.googleapis.com/upload/zoo/v1/animals?uploadType=resumable&alt=json',
Joe Gregoriode860442012-03-02 15:55:52 -0500480 request.uri)
Joe Gregoriod0bd3882011-11-22 09:49:47 -0500481
482 http = HttpMockSequence([
483 ({'status': '200',
484 'location': 'http://upload.example.com'}, ''),
485 ({'status': '308',
486 'location': 'http://upload.example.com/2',
487 'range': '0-12'}, ''),
488 ({'status': '308',
489 'location': 'http://upload.example.com/3',
Joe Gregorio945be3e2012-01-27 17:01:06 -0500490 'range': '0-%d' % media_upload.size()}, ''),
Joe Gregoriod0bd3882011-11-22 09:49:47 -0500491 ({'status': '200'}, '{"foo": "bar"}'),
492 ])
493
494 body = request.execute(http)
495 self.assertEquals(body, {"foo": "bar"})
496
497 def test_resumable_media_fail_unknown_response_code_first_request(self):
498 """Not a multipart upload."""
499 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
500 zoo = build('zoo', 'v1', self.http)
501
502 media_upload = MediaFileUpload(datafile('small.png'), resumable=True)
503 request = zoo.animals().insert(media_body=media_upload, body=None)
504
505 http = HttpMockSequence([
506 ({'status': '400',
507 'location': 'http://upload.example.com'}, ''),
508 ])
509
510 self.assertRaises(ResumableUploadError, request.execute, http)
511
512 def test_resumable_media_fail_unknown_response_code_subsequent_request(self):
513 """Not a multipart upload."""
514 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
515 zoo = build('zoo', 'v1', self.http)
516
517 media_upload = MediaFileUpload(datafile('small.png'), resumable=True)
518 request = zoo.animals().insert(media_body=media_upload, body=None)
519
520 http = HttpMockSequence([
521 ({'status': '200',
522 'location': 'http://upload.example.com'}, ''),
523 ({'status': '400'}, ''),
524 ])
525
526 self.assertRaises(HttpError, request.execute, http)
Joe Gregorio910b9b12012-06-12 09:36:30 -0400527 self.assertTrue(request._in_error_state)
Joe Gregoriod0bd3882011-11-22 09:49:47 -0500528
Joe Gregorio910b9b12012-06-12 09:36:30 -0400529 http = HttpMockSequence([
530 ({'status': '308',
531 'range': '0-5'}, ''),
532 ({'status': '308',
533 'range': '0-6'}, ''),
534 ])
535
536 status, body = request.next_chunk(http)
537 self.assertEquals(status.resumable_progress, 7,
538 'Should have first checked length and then tried to PUT more.')
539 self.assertFalse(request._in_error_state)
540
541 # Put it back in an error state.
542 http = HttpMockSequence([
543 ({'status': '400'}, ''),
544 ])
545 self.assertRaises(HttpError, request.execute, http)
546 self.assertTrue(request._in_error_state)
547
548 # Pretend the last request that 400'd actually succeeded.
549 http = HttpMockSequence([
550 ({'status': '200'}, '{"foo": "bar"}'),
551 ])
552 status, body = request.next_chunk(http)
553 self.assertEqual(body, {'foo': 'bar'})
554
555 def test_resumable_media_handle_uploads_of_unknown_size(self):
556 http = HttpMockSequence([
557 ({'status': '200',
558 'location': 'http://upload.example.com'}, ''),
559 ({'status': '200'}, 'echo_request_headers_as_json'),
560 ])
561
562 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
563 zoo = build('zoo', 'v1', self.http)
564
Joe Gregorio4a2c29f2012-07-12 12:52:47 -0400565 fd = StringIO.StringIO('data goes here')
Joe Gregorio910b9b12012-06-12 09:36:30 -0400566
567 # Create an upload that doesn't know the full size of the media.
568 upload = MediaIoBaseUpload(
Joe Gregorio4a2c29f2012-07-12 12:52:47 -0400569 fd=fd, mimetype='image/png', chunksize=10, resumable=True)
Joe Gregorio910b9b12012-06-12 09:36:30 -0400570
571 request = zoo.animals().insert(media_body=upload, body=None)
572 status, body = request.next_chunk(http)
Joe Gregorio44454e42012-06-15 08:38:53 -0400573 self.assertEqual(body, {'Content-Range': 'bytes 0-9/*'},
574 'Should be 10 out of * bytes.')
575
576 def test_resumable_media_handle_uploads_of_unknown_size_eof(self):
577 http = HttpMockSequence([
578 ({'status': '200',
579 'location': 'http://upload.example.com'}, ''),
580 ({'status': '200'}, 'echo_request_headers_as_json'),
581 ])
582
583 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
584 zoo = build('zoo', 'v1', self.http)
585
Joe Gregorio4a2c29f2012-07-12 12:52:47 -0400586 fd = StringIO.StringIO('data goes here')
Joe Gregorio44454e42012-06-15 08:38:53 -0400587
588 # Create an upload that doesn't know the full size of the media.
589 upload = MediaIoBaseUpload(
Joe Gregorio4a2c29f2012-07-12 12:52:47 -0400590 fd=fd, mimetype='image/png', chunksize=15, resumable=True)
Joe Gregorio44454e42012-06-15 08:38:53 -0400591
592 request = zoo.animals().insert(media_body=upload, body=None)
593 status, body = request.next_chunk(http)
594 self.assertEqual(body, {'Content-Range': 'bytes 0-13/14'})
Joe Gregorio910b9b12012-06-12 09:36:30 -0400595
596 def test_resumable_media_handle_resume_of_upload_of_unknown_size(self):
597 http = HttpMockSequence([
598 ({'status': '200',
599 'location': 'http://upload.example.com'}, ''),
600 ({'status': '400'}, ''),
601 ])
602
603 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
604 zoo = build('zoo', 'v1', self.http)
605
606 # Create an upload that doesn't know the full size of the media.
Joe Gregorio4a2c29f2012-07-12 12:52:47 -0400607 fd = StringIO.StringIO('data goes here')
Joe Gregorio910b9b12012-06-12 09:36:30 -0400608
609 upload = MediaIoBaseUpload(
Joe Gregorio4a2c29f2012-07-12 12:52:47 -0400610 fd=fd, mimetype='image/png', chunksize=500, resumable=True)
Joe Gregorio910b9b12012-06-12 09:36:30 -0400611
612 request = zoo.animals().insert(media_body=upload, body=None)
613
614 # Put it in an error state.
615 self.assertRaises(HttpError, request.next_chunk, http)
616
617 http = HttpMockSequence([
618 ({'status': '400',
619 'range': '0-5'}, 'echo_request_headers_as_json'),
620 ])
621 try:
622 # Should resume the upload by first querying the status of the upload.
623 request.next_chunk(http)
624 except HttpError, e:
625 expected = {
626 'Content-Range': 'bytes */*',
627 'content-length': '0'
628 }
629 self.assertEqual(expected, simplejson.loads(e.content),
630 'Should send an empty body when requesting the current upload status.')
Joe Gregoriod0bd3882011-11-22 09:49:47 -0500631
Joe Gregorio708388c2012-06-15 13:43:04 -0400632
Joe Gregorioc5c5a372010-09-22 11:42:32 -0400633class Next(unittest.TestCase):
Joe Gregorio00cf1d92010-09-27 09:22:03 -0400634
Joe Gregorio3c676f92011-07-25 10:38:14 -0400635 def test_next_successful_none_on_no_next_page_token(self):
636 self.http = HttpMock(datafile('tasks.json'), {'status': '200'})
637 tasks = build('tasks', 'v1', self.http)
638 request = tasks.tasklists().list()
639 self.assertEqual(None, tasks.tasklists().list_next(request, {}))
640
641 def test_next_successful_with_next_page_token(self):
642 self.http = HttpMock(datafile('tasks.json'), {'status': '200'})
643 tasks = build('tasks', 'v1', self.http)
644 request = tasks.tasklists().list()
Joe Gregorioa98733f2011-09-16 10:12:28 -0400645 next_request = tasks.tasklists().list_next(
646 request, {'nextPageToken': '123abc'})
Joe Gregorio3c676f92011-07-25 10:38:14 -0400647 parsed = list(urlparse.urlparse(next_request.uri))
648 q = parse_qs(parsed[4])
649 self.assertEqual(q['pageToken'][0], '123abc')
650
Joe Gregorio555f33c2011-08-19 14:56:07 -0400651 def test_next_with_method_with_no_properties(self):
652 self.http = HttpMock(datafile('latitude.json'), {'status': '200'})
653 service = build('latitude', 'v1', self.http)
654 request = service.currentLocation().get()
Joe Gregorio00cf1d92010-09-27 09:22:03 -0400655
Joe Gregorioa98733f2011-09-16 10:12:28 -0400656
Joe Gregorio708388c2012-06-15 13:43:04 -0400657class MediaGet(unittest.TestCase):
658
659 def test_get_media(self):
660 http = HttpMock(datafile('zoo.json'), {'status': '200'})
661 zoo = build('zoo', 'v1', http)
662 request = zoo.animals().get_media(name='Lion')
663
664 parsed = urlparse.urlparse(request.uri)
665 q = parse_qs(parsed[4])
666 self.assertEqual(q['alt'], ['media'])
667 self.assertEqual(request.headers['accept'], '*/*')
668
669 http = HttpMockSequence([
670 ({'status': '200'}, 'standing in for media'),
671 ])
672 response = request.execute(http)
673 self.assertEqual('standing in for media', response)
674
675
Joe Gregorioba9ea7f2010-08-19 15:49:04 -0400676if __name__ == '__main__':
677 unittest.main()