blob: 71ca0fb94239fb1ef3ef5f1c691ebd1f4853494d [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 Gregorio583d9e42011-09-16 15:54:15 -040026import copy
Joe Gregorioba9ea7f2010-08-19 15:49:04 -040027import httplib2
28import os
29import unittest
Joe Gregorio00cf1d92010-09-27 09:22:03 -040030import urlparse
Joe Gregorioa98733f2011-09-16 10:12:28 -040031
ade@google.comc5eb46f2010-09-27 23:35:39 +010032try:
33 from urlparse import parse_qs
34except ImportError:
35 from cgi import parse_qs
Joe Gregorio00cf1d92010-09-27 09:22:03 -040036
ade@google.com6a8c1cb2011-09-06 17:40:00 +010037from apiclient.discovery import build, build_from_document, key2param
Joe Gregoriocb8103d2011-02-11 23:20:52 -050038from apiclient.http import HttpMock
Joe Gregoriof4153422011-03-18 22:45:18 -040039from apiclient.http import tunnel_patch
40from apiclient.http import HttpMockSequence
Joe Gregorioc0e0fe92011-03-04 16:16:55 -050041from apiclient.errors import HttpError
Joe Gregorio49396552011-03-08 10:39:00 -050042from apiclient.errors import InvalidJsonError
Joe Gregoriofdf7c802011-06-30 12:33:38 -040043from apiclient.errors import MediaUploadSizeError
44from apiclient.errors import UnacceptableMimeTypeError
Joe Gregoriocb8103d2011-02-11 23:20:52 -050045
46
47DATA_DIR = os.path.join(os.path.dirname(__file__), 'data')
48
Joe Gregorioa98733f2011-09-16 10:12:28 -040049
Joe Gregoriocb8103d2011-02-11 23:20:52 -050050def datafile(filename):
51 return os.path.join(DATA_DIR, filename)
Joe Gregorioba9ea7f2010-08-19 15:49:04 -040052
53
Joe Gregorioc5c5a372010-09-22 11:42:32 -040054class Utilities(unittest.TestCase):
55 def test_key2param(self):
56 self.assertEqual('max_results', key2param('max-results'))
57 self.assertEqual('x007_bond', key2param('007-bond'))
58
59
Joe Gregorioc0e0fe92011-03-04 16:16:55 -050060class DiscoveryErrors(unittest.TestCase):
61
62 def test_failed_to_parse_discovery_json(self):
63 self.http = HttpMock(datafile('malformed.json'), {'status': '200'})
64 try:
65 buzz = build('buzz', 'v1', self.http)
66 self.fail("should have raised an exception over malformed JSON.")
Joe Gregorio49396552011-03-08 10:39:00 -050067 except InvalidJsonError:
68 pass
Joe Gregorioc0e0fe92011-03-04 16:16:55 -050069
70
ade@google.com6a8c1cb2011-09-06 17:40:00 +010071class DiscoveryFromDocument(unittest.TestCase):
Joe Gregorioa98733f2011-09-16 10:12:28 -040072
ade@google.com6a8c1cb2011-09-06 17:40:00 +010073 def test_can_build_from_local_document(self):
74 discovery = file(datafile('buzz.json')).read()
75 buzz = build_from_document(discovery, base="https://www.googleapis.com/")
76 self.assertTrue(buzz is not None)
Joe Gregorioa98733f2011-09-16 10:12:28 -040077
ade@google.com6a8c1cb2011-09-06 17:40:00 +010078 def test_building_with_base_remembers_base(self):
79 discovery = file(datafile('buzz.json')).read()
Joe Gregorioa98733f2011-09-16 10:12:28 -040080
ade@google.com6a8c1cb2011-09-06 17:40:00 +010081 base = "https://www.example.com/"
82 buzz = build_from_document(discovery, base=base)
83 self.assertEquals(base + "buzz/v1/", buzz._baseUrl)
84
85
Joe Gregorioa98733f2011-09-16 10:12:28 -040086class DiscoveryFromHttp(unittest.TestCase):
Joe Gregorio583d9e42011-09-16 15:54:15 -040087 def setUp(self):
88 self.old_environ = copy.copy(os.environ)
Joe Gregorioa98733f2011-09-16 10:12:28 -040089
Joe Gregorio583d9e42011-09-16 15:54:15 -040090 def tearDown(self):
91 os.environ = self.old_environ
92
93 def test_userip_is_added_to_discovery_uri(self):
Joe Gregorioa98733f2011-09-16 10:12:28 -040094 # build() will raise an HttpError on a 400, use this to pick the request uri
95 # out of the raised exception.
Joe Gregorio583d9e42011-09-16 15:54:15 -040096 os.environ['REMOTE_ADDR'] = '10.0.0.1'
Joe Gregorioa98733f2011-09-16 10:12:28 -040097 try:
98 http = HttpMockSequence([
99 ({'status': '400'}, file(datafile('zoo.json'), 'r').read()),
100 ])
101 zoo = build('zoo', 'v1', http, developerKey='foo',
102 discoveryServiceUrl='http://example.com')
103 self.fail('Should have raised an exception.')
104 except HttpError, e:
Joe Gregorio583d9e42011-09-16 15:54:15 -0400105 self.assertEqual(e.uri, 'http://example.com?userIp=10.0.0.1')
Joe Gregorioa98733f2011-09-16 10:12:28 -0400106
Joe Gregorio583d9e42011-09-16 15:54:15 -0400107 def test_userip_missing_is_not_added_to_discovery_uri(self):
Joe Gregorioa98733f2011-09-16 10:12:28 -0400108 # build() will raise an HttpError on a 400, use this to pick the request uri
109 # out of the raised exception.
110 try:
111 http = HttpMockSequence([
112 ({'status': '400'}, file(datafile('zoo.json'), 'r').read()),
113 ])
114 zoo = build('zoo', 'v1', http, developerKey=None,
115 discoveryServiceUrl='http://example.com')
116 self.fail('Should have raised an exception.')
117 except HttpError, e:
118 self.assertEqual(e.uri, 'http://example.com')
119
120
Joe Gregorioba9ea7f2010-08-19 15:49:04 -0400121class Discovery(unittest.TestCase):
Joe Gregorio2379ecc2010-10-26 10:51:28 -0400122
Joe Gregorioba9ea7f2010-08-19 15:49:04 -0400123 def test_method_error_checking(self):
Joe Gregoriocb8103d2011-02-11 23:20:52 -0500124 self.http = HttpMock(datafile('buzz.json'), {'status': '200'})
Joe Gregorioba9ea7f2010-08-19 15:49:04 -0400125 buzz = build('buzz', 'v1', self.http)
126
127 # Missing required parameters
128 try:
129 buzz.activities().list()
130 self.fail()
131 except TypeError, e:
132 self.assertTrue('Missing' in str(e))
133
134 # Parameter doesn't match regex
135 try:
Joe Gregoriobee86832011-02-22 10:00:19 -0500136 buzz.activities().list(scope='@myself', userId='me')
Joe Gregorioba9ea7f2010-08-19 15:49:04 -0400137 self.fail()
138 except TypeError, e:
Joe Gregorioca876e42011-02-22 19:39:42 -0500139 self.assertTrue('not an allowed value' in str(e))
Joe Gregorioba9ea7f2010-08-19 15:49:04 -0400140
141 # Parameter doesn't match regex
142 try:
143 buzz.activities().list(scope='not@', userId='foo')
144 self.fail()
145 except TypeError, e:
Joe Gregorioca876e42011-02-22 19:39:42 -0500146 self.assertTrue('not an allowed value' in str(e))
Joe Gregorioba9ea7f2010-08-19 15:49:04 -0400147
148 # Unexpected parameter
149 try:
150 buzz.activities().list(flubber=12)
151 self.fail()
152 except TypeError, e:
153 self.assertTrue('unexpected' in str(e))
154
Joe Gregoriobee86832011-02-22 10:00:19 -0500155 def _check_query_types(self, request):
156 parsed = urlparse.urlparse(request.uri)
157 q = parse_qs(parsed[4])
158 self.assertEqual(q['q'], ['foo'])
159 self.assertEqual(q['i'], ['1'])
160 self.assertEqual(q['n'], ['1.0'])
161 self.assertEqual(q['b'], ['false'])
162 self.assertEqual(q['a'], ['[1, 2, 3]'])
163 self.assertEqual(q['o'], ['{\'a\': 1}'])
164 self.assertEqual(q['e'], ['bar'])
165
166 def test_type_coercion(self):
Joe Gregoriof4153422011-03-18 22:45:18 -0400167 http = HttpMock(datafile('zoo.json'), {'status': '200'})
168 zoo = build('zoo', 'v1', http)
Joe Gregoriobee86832011-02-22 10:00:19 -0500169
Joe Gregorioa98733f2011-09-16 10:12:28 -0400170 request = zoo.query(
171 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 -0500172 self._check_query_types(request)
Joe Gregorioa98733f2011-09-16 10:12:28 -0400173 request = zoo.query(
174 q="foo", i=1, n=1, b=False, a=[1,2,3], o={'a':1}, e='bar')
Joe Gregoriobee86832011-02-22 10:00:19 -0500175 self._check_query_types(request)
Joe Gregoriof863f7a2011-02-24 03:24:44 -0500176
Joe Gregorioa98733f2011-09-16 10:12:28 -0400177 request = zoo.query(
178 q="foo", i="1", n="1", b="", a=[1,2,3], o={'a':1}, e='bar')
Joe Gregoriobee86832011-02-22 10:00:19 -0500179 self._check_query_types(request)
180
Joe Gregorio13217952011-02-22 15:37:38 -0500181 def test_optional_stack_query_parameters(self):
Joe Gregoriof4153422011-03-18 22:45:18 -0400182 http = HttpMock(datafile('zoo.json'), {'status': '200'})
183 zoo = build('zoo', 'v1', http)
184 request = zoo.query(trace='html', fields='description')
Joe Gregorio13217952011-02-22 15:37:38 -0500185
Joe Gregorioca876e42011-02-22 19:39:42 -0500186 parsed = urlparse.urlparse(request.uri)
187 q = parse_qs(parsed[4])
188 self.assertEqual(q['trace'], ['html'])
Joe Gregoriof4153422011-03-18 22:45:18 -0400189 self.assertEqual(q['fields'], ['description'])
190
191 def test_patch(self):
192 http = HttpMock(datafile('zoo.json'), {'status': '200'})
193 zoo = build('zoo', 'v1', http)
194 request = zoo.animals().patch(name='lion', body='{"description": "foo"}')
195
196 self.assertEqual(request.method, 'PATCH')
197
198 def test_tunnel_patch(self):
199 http = HttpMockSequence([
200 ({'status': '200'}, file(datafile('zoo.json'), 'r').read()),
201 ({'status': '200'}, 'echo_request_headers_as_json'),
202 ])
203 http = tunnel_patch(http)
204 zoo = build('zoo', 'v1', http)
Joe Gregorioa98733f2011-09-16 10:12:28 -0400205 resp = zoo.animals().patch(
206 name='lion', body='{"description": "foo"}').execute()
Joe Gregoriof4153422011-03-18 22:45:18 -0400207
208 self.assertTrue('x-http-method-override' in resp)
Joe Gregorioca876e42011-02-22 19:39:42 -0500209
ade@google.com850cf552010-08-20 23:24:56 +0100210 def test_buzz_resources(self):
Joe Gregoriocb8103d2011-02-11 23:20:52 -0500211 self.http = HttpMock(datafile('buzz.json'), {'status': '200'})
Joe Gregorioba9ea7f2010-08-19 15:49:04 -0400212 buzz = build('buzz', 'v1', self.http)
213 self.assertTrue(getattr(buzz, 'activities'))
Joe Gregorioba9ea7f2010-08-19 15:49:04 -0400214 self.assertTrue(getattr(buzz, 'photos'))
215 self.assertTrue(getattr(buzz, 'people'))
216 self.assertTrue(getattr(buzz, 'groups'))
217 self.assertTrue(getattr(buzz, 'comments'))
218 self.assertTrue(getattr(buzz, 'related'))
219
Joe Gregorioc5c5a372010-09-22 11:42:32 -0400220 def test_auth(self):
Joe Gregoriocb8103d2011-02-11 23:20:52 -0500221 self.http = HttpMock(datafile('buzz.json'), {'status': '200'})
Joe Gregorioc5c5a372010-09-22 11:42:32 -0400222 buzz = build('buzz', 'v1', self.http)
223 auth = buzz.auth_discovery()
224 self.assertTrue('request' in auth)
225
Joe Gregorio2379ecc2010-10-26 10:51:28 -0400226 def test_full_featured(self):
227 # Zoo should exercise all discovery facets
228 # and should also have no future.json file.
Joe Gregoriocb8103d2011-02-11 23:20:52 -0500229 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
Joe Gregorio2379ecc2010-10-26 10:51:28 -0400230 zoo = build('zoo', 'v1', self.http)
231 self.assertTrue(getattr(zoo, 'animals'))
Joe Gregoriof863f7a2011-02-24 03:24:44 -0500232
Joe Gregoriof4153422011-03-18 22:45:18 -0400233 request = zoo.animals().list(name='bat', projection="full")
Joe Gregorio2379ecc2010-10-26 10:51:28 -0400234 parsed = urlparse.urlparse(request.uri)
235 q = parse_qs(parsed[4])
236 self.assertEqual(q['name'], ['bat'])
Joe Gregoriof4153422011-03-18 22:45:18 -0400237 self.assertEqual(q['projection'], ['full'])
Joe Gregorio2379ecc2010-10-26 10:51:28 -0400238
Joe Gregorio3fada332011-01-07 17:07:45 -0500239 def test_nested_resources(self):
Joe Gregoriocb8103d2011-02-11 23:20:52 -0500240 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
Joe Gregorio3fada332011-01-07 17:07:45 -0500241 zoo = build('zoo', 'v1', self.http)
242 self.assertTrue(getattr(zoo, 'animals'))
243 request = zoo.my().favorites().list(max_results="5")
244 parsed = urlparse.urlparse(request.uri)
245 q = parse_qs(parsed[4])
246 self.assertEqual(q['max-results'], ['5'])
247
Joe Gregoriod92897c2011-07-07 11:44:56 -0400248 def test_methods_with_reserved_names(self):
249 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
250 zoo = build('zoo', 'v1', self.http)
251 self.assertTrue(getattr(zoo, 'animals'))
252 request = zoo.global_().print_().assert_(max_results="5")
253 parsed = urlparse.urlparse(request.uri)
254 self.assertEqual(parsed[2], '/zoo/global/print/assert')
255
Joe Gregorio7a6df3a2011-01-31 21:55:21 -0500256 def test_top_level_functions(self):
Joe Gregoriocb8103d2011-02-11 23:20:52 -0500257 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
Joe Gregorio7a6df3a2011-01-31 21:55:21 -0500258 zoo = build('zoo', 'v1', self.http)
259 self.assertTrue(getattr(zoo, 'query'))
260 request = zoo.query(q="foo")
261 parsed = urlparse.urlparse(request.uri)
262 q = parse_qs(parsed[4])
263 self.assertEqual(q['q'], ['foo'])
Joe Gregorio2379ecc2010-10-26 10:51:28 -0400264
Joe Gregoriofdf7c802011-06-30 12:33:38 -0400265 def test_simple_media_uploads(self):
266 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
267 zoo = build('zoo', 'v1', self.http)
268 doc = getattr(zoo.animals().insert, '__doc__')
269 self.assertTrue('media_body' in doc)
270
Joe Gregorio84d3c1f2011-07-25 10:39:45 -0400271 def test_simple_media_upload_no_max_size_provided(self):
272 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
273 zoo = build('zoo', 'v1', self.http)
274 request = zoo.animals().crossbreed(media_body=datafile('small.png'))
275 self.assertEquals('image/png', request.headers['content-type'])
276 self.assertEquals('PNG', request.body[1:4])
277
Joe Gregoriofdf7c802011-06-30 12:33:38 -0400278 def test_simple_media_raise_correct_exceptions(self):
279 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
280 zoo = build('zoo', 'v1', self.http)
281
282 try:
283 zoo.animals().insert(media_body=datafile('smiley.png'))
284 self.fail("should throw exception if media is too large.")
285 except MediaUploadSizeError:
286 pass
287
288 try:
289 zoo.animals().insert(media_body=datafile('small.jpg'))
290 self.fail("should throw exception if mimetype is unacceptable.")
291 except UnacceptableMimeTypeError:
292 pass
293
294 def test_simple_media_good_upload(self):
295 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
296 zoo = build('zoo', 'v1', self.http)
297
298 request = zoo.animals().insert(media_body=datafile('small.png'))
299 self.assertEquals('image/png', request.headers['content-type'])
300 self.assertEquals('PNG', request.body[1:4])
301
302 def test_multipart_media_raise_correct_exceptions(self):
303 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
304 zoo = build('zoo', 'v1', self.http)
305
306 try:
307 zoo.animals().insert(media_body=datafile('smiley.png'), body={})
308 self.fail("should throw exception if media is too large.")
309 except MediaUploadSizeError:
310 pass
311
312 try:
313 zoo.animals().insert(media_body=datafile('small.jpg'), body={})
314 self.fail("should throw exception if mimetype is unacceptable.")
315 except UnacceptableMimeTypeError:
316 pass
317
318 def test_multipart_media_good_upload(self):
319 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
320 zoo = build('zoo', 'v1', self.http)
321
322 request = zoo.animals().insert(media_body=datafile('small.png'), body={})
Joe Gregorioa98733f2011-09-16 10:12:28 -0400323 self.assertTrue(request.headers['content-type'].startswith(
324 'multipart/related'))
Joe Gregoriofdf7c802011-06-30 12:33:38 -0400325 self.assertEquals('--==', request.body[0:4])
326
327 def test_media_capable_method_without_media(self):
328 self.http = HttpMock(datafile('zoo.json'), {'status': '200'})
329 zoo = build('zoo', 'v1', self.http)
330
331 request = zoo.animals().insert(body={})
332 self.assertTrue(request.headers['content-type'], 'application/json')
Joe Gregorioc5c5a372010-09-22 11:42:32 -0400333
334class Next(unittest.TestCase):
Joe Gregorioc9359072010-10-25 15:26:13 -0400335 def test_next_for_people_liked(self):
Joe Gregorio3c676f92011-07-25 10:38:14 -0400336 """Legacy test for Buzz _next support."""
Joe Gregoriocb8103d2011-02-11 23:20:52 -0500337 self.http = HttpMock(datafile('buzz.json'), {'status': '200'})
Joe Gregorioc9359072010-10-25 15:26:13 -0400338 buzz = build('buzz', 'v1', self.http)
339 people = {'links':
Joe Gregorioc5c5a372010-09-22 11:42:32 -0400340 {'next':
341 [{'href': 'http://www.googleapis.com/next-link'}]}}
Joe Gregorioc9359072010-10-25 15:26:13 -0400342 request = buzz.people().liked_next(people)
343 self.assertEqual(request.uri, 'http://www.googleapis.com/next-link')
Joe Gregorio00cf1d92010-09-27 09:22:03 -0400344
Joe Gregorio3c676f92011-07-25 10:38:14 -0400345 def test_next_successful_none_on_no_next_page_token(self):
346 self.http = HttpMock(datafile('tasks.json'), {'status': '200'})
347 tasks = build('tasks', 'v1', self.http)
348 request = tasks.tasklists().list()
349 self.assertEqual(None, tasks.tasklists().list_next(request, {}))
350
351 def test_next_successful_with_next_page_token(self):
352 self.http = HttpMock(datafile('tasks.json'), {'status': '200'})
353 tasks = build('tasks', 'v1', self.http)
354 request = tasks.tasklists().list()
Joe Gregorioa98733f2011-09-16 10:12:28 -0400355 next_request = tasks.tasklists().list_next(
356 request, {'nextPageToken': '123abc'})
Joe Gregorio3c676f92011-07-25 10:38:14 -0400357 parsed = list(urlparse.urlparse(next_request.uri))
358 q = parse_qs(parsed[4])
359 self.assertEqual(q['pageToken'][0], '123abc')
360
Joe Gregorio555f33c2011-08-19 14:56:07 -0400361 def test_next_with_method_with_no_properties(self):
362 self.http = HttpMock(datafile('latitude.json'), {'status': '200'})
363 service = build('latitude', 'v1', self.http)
364 request = service.currentLocation().get()
Joe Gregorio00cf1d92010-09-27 09:22:03 -0400365
Joe Gregorioa98733f2011-09-16 10:12:28 -0400366
Joe Gregorio00cf1d92010-09-27 09:22:03 -0400367class DeveloperKey(unittest.TestCase):
Joe Gregorioa98733f2011-09-16 10:12:28 -0400368
Joe Gregorio00cf1d92010-09-27 09:22:03 -0400369 def test_param(self):
Joe Gregoriocb8103d2011-02-11 23:20:52 -0500370 self.http = HttpMock(datafile('buzz.json'), {'status': '200'})
Joe Gregorio00cf1d92010-09-27 09:22:03 -0400371 buzz = build('buzz', 'v1', self.http, developerKey='foobie_bletch')
372 activities = {'links':
373 {'next':
374 [{'href': 'http://www.googleapis.com/next-link'}]}}
375 request = buzz.activities().list_next(activities)
376 parsed = urlparse.urlparse(request.uri)
ade@google.comc5eb46f2010-09-27 23:35:39 +0100377 q = parse_qs(parsed[4])
Joe Gregorio00cf1d92010-09-27 09:22:03 -0400378 self.assertEqual(q['key'], ['foobie_bletch'])
379
Joe Gregorioc9359072010-10-25 15:26:13 -0400380 def test_next_for_activities_list(self):
Joe Gregoriocb8103d2011-02-11 23:20:52 -0500381 self.http = HttpMock(datafile('buzz.json'), {'status': '200'})
Joe Gregorioc9359072010-10-25 15:26:13 -0400382 buzz = build('buzz', 'v1', self.http, developerKey='foobie_bletch')
383 activities = {'links':
ade@google.com2ab0de72010-09-27 23:26:54 +0100384 {'next':
385 [{'href': 'http://www.googleapis.com/next-link'}]}}
Joe Gregorioc9359072010-10-25 15:26:13 -0400386 request = buzz.activities().list_next(activities)
387 self.assertEqual(request.uri,
388 'http://www.googleapis.com/next-link?key=foobie_bletch')
389
Joe Gregorioba9ea7f2010-08-19 15:49:04 -0400390
391if __name__ == '__main__':
392 unittest.main()