##// END OF EJS Templates
TEST: Verify base64 return values after decoding.
Scott Sanderson -
Show More
@@ -1,540 +1,542 b''
1 1 # coding: utf-8
2 2 """Test the contents webservice API."""
3 3
4 4 import base64
5 5 import io
6 6 import json
7 7 import os
8 8 import shutil
9 9 from unicodedata import normalize
10 10
11 11 pjoin = os.path.join
12 12
13 13 import requests
14 14
15 15 from IPython.html.utils import url_path_join, url_escape, to_os_path
16 16 from IPython.html.tests.launchnotebook import NotebookTestBase, assert_http_error
17 17 from IPython.nbformat import read, write, from_dict
18 18 from IPython.nbformat.v4 import (
19 19 new_notebook, new_markdown_cell,
20 20 )
21 21 from IPython.nbformat import v2
22 22 from IPython.utils import py3compat
23 23 from IPython.utils.data import uniq_stable
24 24
25 25
26 26 def notebooks_only(dir_model):
27 27 return [nb for nb in dir_model['content'] if nb['type']=='notebook']
28 28
29 29 def dirs_only(dir_model):
30 30 return [x for x in dir_model['content'] if x['type']=='directory']
31 31
32 32
33 33 class API(object):
34 34 """Wrapper for contents API calls."""
35 35 def __init__(self, base_url):
36 36 self.base_url = base_url
37 37
38 38 def _req(self, verb, path, body=None, params=None):
39 39 response = requests.request(verb,
40 40 url_path_join(self.base_url, 'api/contents', path),
41 41 data=body, params=params,
42 42 )
43 43 response.raise_for_status()
44 44 return response
45 45
46 46 def list(self, path='/'):
47 47 return self._req('GET', path)
48 48
49 49 def read(self, path, type=None, format=None):
50 50 params = {}
51 51 if type is not None:
52 52 params['type'] = type
53 53 if format is not None:
54 54 params['format'] = format
55 55 return self._req('GET', path, params=params)
56 56
57 57 def create_untitled(self, path='/', ext='.ipynb'):
58 58 body = None
59 59 if ext:
60 60 body = json.dumps({'ext': ext})
61 61 return self._req('POST', path, body)
62 62
63 63 def mkdir_untitled(self, path='/'):
64 64 return self._req('POST', path, json.dumps({'type': 'directory'}))
65 65
66 66 def copy(self, copy_from, path='/'):
67 67 body = json.dumps({'copy_from':copy_from})
68 68 return self._req('POST', path, body)
69 69
70 70 def create(self, path='/'):
71 71 return self._req('PUT', path)
72 72
73 73 def upload(self, path, body):
74 74 return self._req('PUT', path, body)
75 75
76 76 def mkdir(self, path='/'):
77 77 return self._req('PUT', path, json.dumps({'type': 'directory'}))
78 78
79 79 def copy_put(self, copy_from, path='/'):
80 80 body = json.dumps({'copy_from':copy_from})
81 81 return self._req('PUT', path, body)
82 82
83 83 def save(self, path, body):
84 84 return self._req('PUT', path, body)
85 85
86 86 def delete(self, path='/'):
87 87 return self._req('DELETE', path)
88 88
89 89 def rename(self, path, new_path):
90 90 body = json.dumps({'path': new_path})
91 91 return self._req('PATCH', path, body)
92 92
93 93 def get_checkpoints(self, path):
94 94 return self._req('GET', url_path_join(path, 'checkpoints'))
95 95
96 96 def new_checkpoint(self, path):
97 97 return self._req('POST', url_path_join(path, 'checkpoints'))
98 98
99 99 def restore_checkpoint(self, path, checkpoint_id):
100 100 return self._req('POST', url_path_join(path, 'checkpoints', checkpoint_id))
101 101
102 102 def delete_checkpoint(self, path, checkpoint_id):
103 103 return self._req('DELETE', url_path_join(path, 'checkpoints', checkpoint_id))
104 104
105 105 class APITest(NotebookTestBase):
106 106 """Test the kernels web service API"""
107 107 dirs_nbs = [('', 'inroot'),
108 108 ('Directory with spaces in', 'inspace'),
109 109 (u'unicodΓ©', 'innonascii'),
110 110 ('foo', 'a'),
111 111 ('foo', 'b'),
112 112 ('foo', 'name with spaces'),
113 113 ('foo', u'unicodΓ©'),
114 114 ('foo/bar', 'baz'),
115 115 ('ordering', 'A'),
116 116 ('ordering', 'b'),
117 117 ('ordering', 'C'),
118 118 (u'Γ₯ b', u'Γ§ d'),
119 119 ]
120 120 hidden_dirs = ['.hidden', '__pycache__']
121 121
122 122 dirs = uniq_stable([py3compat.cast_unicode(d) for (d,n) in dirs_nbs])
123 123 del dirs[0] # remove ''
124 124 top_level_dirs = {normalize('NFC', d.split('/')[0]) for d in dirs}
125 125
126 126 @staticmethod
127 127 def _blob_for_name(name):
128 128 return name.encode('utf-8') + b'\xFF'
129 129
130 130 @staticmethod
131 131 def _txt_for_name(name):
132 132 return u'%s text file' % name
133 133
134 134 def to_os_path(self, api_path):
135 135 return to_os_path(api_path, root=self.notebook_dir.name)
136 136
137 137 def make_dir(self, api_path):
138 138 """Create a directory at api_path"""
139 139 os_path = self.to_os_path(api_path)
140 140 try:
141 141 os.makedirs(os_path)
142 142 except OSError:
143 143 print("Directory already exists: %r" % os_path)
144 144
145 145 def make_txt(self, api_path, txt):
146 146 """Make a text file at a given api_path"""
147 147 os_path = self.to_os_path(api_path)
148 148 with io.open(os_path, 'w', encoding='utf-8') as f:
149 149 f.write(txt)
150 150
151 151 def make_blob(self, api_path, blob):
152 152 """Make a binary file at a given api_path"""
153 153 os_path = self.to_os_path(api_path)
154 154 with io.open(os_path, 'wb') as f:
155 155 f.write(blob)
156 156
157 157 def make_nb(self, api_path, nb):
158 158 """Make a notebook file at a given api_path"""
159 159 os_path = self.to_os_path(api_path)
160 160
161 161 with io.open(os_path, 'w', encoding='utf-8') as f:
162 162 write(nb, f, version=4)
163 163
164 164 def delete_dir(self, api_path):
165 165 """Delete a directory at api_path, removing any contents."""
166 166 os_path = self.to_os_path(api_path)
167 167 shutil.rmtree(os_path, ignore_errors=True)
168 168
169 169 def delete_file(self, api_path):
170 170 """Delete a file at the given path if it exists."""
171 171 if self.isfile(api_path):
172 172 os.unlink(self.to_os_path(api_path))
173 173
174 174 def isfile(self, api_path):
175 175 return os.path.isfile(self.to_os_path(api_path))
176 176
177 177 def isdir(self, api_path):
178 178 return os.path.isdir(self.to_os_path(api_path))
179 179
180 180 def setUp(self):
181 181
182 182 for d in (self.dirs + self.hidden_dirs):
183 183 self.make_dir(d)
184 184
185 185 for d, name in self.dirs_nbs:
186 186 # create a notebook
187 187 nb = new_notebook()
188 188 self.make_nb(u'{}/{}.ipynb'.format(d, name), nb)
189 189
190 190 # create a text file
191 191 txt = self._txt_for_name(name)
192 192 self.make_txt(u'{}/{}.txt'.format(d, name), txt)
193 193
194 194 # create a binary file
195 195 blob = self._blob_for_name(name)
196 196 self.make_blob(u'{}/{}.blob'.format(d, name), blob)
197 197
198 198 self.api = API(self.base_url())
199 199
200 200 def tearDown(self):
201 201 for dname in (list(self.top_level_dirs) + self.hidden_dirs):
202 202 self.delete_dir(dname)
203 203 self.delete_file('inroot.ipynb')
204 204
205 205 def test_list_notebooks(self):
206 206 nbs = notebooks_only(self.api.list().json())
207 207 self.assertEqual(len(nbs), 1)
208 208 self.assertEqual(nbs[0]['name'], 'inroot.ipynb')
209 209
210 210 nbs = notebooks_only(self.api.list('/Directory with spaces in/').json())
211 211 self.assertEqual(len(nbs), 1)
212 212 self.assertEqual(nbs[0]['name'], 'inspace.ipynb')
213 213
214 214 nbs = notebooks_only(self.api.list(u'/unicodΓ©/').json())
215 215 self.assertEqual(len(nbs), 1)
216 216 self.assertEqual(nbs[0]['name'], 'innonascii.ipynb')
217 217 self.assertEqual(nbs[0]['path'], u'unicodΓ©/innonascii.ipynb')
218 218
219 219 nbs = notebooks_only(self.api.list('/foo/bar/').json())
220 220 self.assertEqual(len(nbs), 1)
221 221 self.assertEqual(nbs[0]['name'], 'baz.ipynb')
222 222 self.assertEqual(nbs[0]['path'], 'foo/bar/baz.ipynb')
223 223
224 224 nbs = notebooks_only(self.api.list('foo').json())
225 225 self.assertEqual(len(nbs), 4)
226 226 nbnames = { normalize('NFC', n['name']) for n in nbs }
227 227 expected = [ u'a.ipynb', u'b.ipynb', u'name with spaces.ipynb', u'unicodΓ©.ipynb']
228 228 expected = { normalize('NFC', name) for name in expected }
229 229 self.assertEqual(nbnames, expected)
230 230
231 231 nbs = notebooks_only(self.api.list('ordering').json())
232 232 nbnames = [n['name'] for n in nbs]
233 233 expected = ['A.ipynb', 'b.ipynb', 'C.ipynb']
234 234 self.assertEqual(nbnames, expected)
235 235
236 236 def test_list_dirs(self):
237 237 dirs = dirs_only(self.api.list().json())
238 238 dir_names = {normalize('NFC', d['name']) for d in dirs}
239 239 self.assertEqual(dir_names, self.top_level_dirs) # Excluding hidden dirs
240 240
241 241 def test_list_nonexistant_dir(self):
242 242 with assert_http_error(404):
243 243 self.api.list('nonexistant')
244 244
245 245 def test_get_nb_contents(self):
246 246 for d, name in self.dirs_nbs:
247 247 path = url_path_join(d, name + '.ipynb')
248 248 nb = self.api.read(path).json()
249 249 self.assertEqual(nb['name'], u'%s.ipynb' % name)
250 250 self.assertEqual(nb['path'], path)
251 251 self.assertEqual(nb['type'], 'notebook')
252 252 self.assertIn('content', nb)
253 253 self.assertEqual(nb['format'], 'json')
254 254 self.assertIn('content', nb)
255 255 self.assertIn('metadata', nb['content'])
256 256 self.assertIsInstance(nb['content']['metadata'], dict)
257 257
258 258 def test_get_contents_no_such_file(self):
259 259 # Name that doesn't exist - should be a 404
260 260 with assert_http_error(404):
261 261 self.api.read('foo/q.ipynb')
262 262
263 263 def test_get_text_file_contents(self):
264 264 for d, name in self.dirs_nbs:
265 265 path = url_path_join(d, name + '.txt')
266 266 model = self.api.read(path).json()
267 267 self.assertEqual(model['name'], u'%s.txt' % name)
268 268 self.assertEqual(model['path'], path)
269 269 self.assertIn('content', model)
270 270 self.assertEqual(model['format'], 'text')
271 271 self.assertEqual(model['type'], 'file')
272 272 self.assertEqual(model['content'], self._txt_for_name(name))
273 273
274 274 # Name that doesn't exist - should be a 404
275 275 with assert_http_error(404):
276 276 self.api.read('foo/q.txt')
277 277
278 278 # Specifying format=text should fail on a non-UTF-8 file
279 279 with assert_http_error(400):
280 280 self.api.read('foo/bar/baz.blob', type='file', format='text')
281 281
282 282 def test_get_binary_file_contents(self):
283 283 for d, name in self.dirs_nbs:
284 284 path = url_path_join(d, name + '.blob')
285 285 model = self.api.read(path).json()
286 286 self.assertEqual(model['name'], u'%s.blob' % name)
287 287 self.assertEqual(model['path'], path)
288 288 self.assertIn('content', model)
289 289 self.assertEqual(model['format'], 'base64')
290 290 self.assertEqual(model['type'], 'file')
291 b64_data = base64.encodestring(self._blob_for_name(name)).decode('ascii')
292 self.assertEqual(model['content'], b64_data)
291 self.assertEqual(
292 base64.decodestring(model['content']),
293 self._blob_for_name(name),
294 )
293 295
294 296 # Name that doesn't exist - should be a 404
295 297 with assert_http_error(404):
296 298 self.api.read('foo/q.txt')
297 299
298 300 def test_get_bad_type(self):
299 301 with assert_http_error(400):
300 302 self.api.read(u'unicodΓ©', type='file') # this is a directory
301 303
302 304 with assert_http_error(400):
303 305 self.api.read(u'unicodΓ©/innonascii.ipynb', type='directory')
304 306
305 307 def _check_created(self, resp, path, type='notebook'):
306 308 self.assertEqual(resp.status_code, 201)
307 309 location_header = py3compat.str_to_unicode(resp.headers['Location'])
308 310 self.assertEqual(location_header, url_escape(url_path_join(u'/api/contents', path)))
309 311 rjson = resp.json()
310 312 self.assertEqual(rjson['name'], path.rsplit('/', 1)[-1])
311 313 self.assertEqual(rjson['path'], path)
312 314 self.assertEqual(rjson['type'], type)
313 315 isright = self.isdir if type == 'directory' else self.isfile
314 316 assert isright(path)
315 317
316 318 def test_create_untitled(self):
317 319 resp = self.api.create_untitled(path=u'Γ₯ b')
318 320 self._check_created(resp, u'Γ₯ b/Untitled.ipynb')
319 321
320 322 # Second time
321 323 resp = self.api.create_untitled(path=u'Γ₯ b')
322 324 self._check_created(resp, u'Γ₯ b/Untitled1.ipynb')
323 325
324 326 # And two directories down
325 327 resp = self.api.create_untitled(path='foo/bar')
326 328 self._check_created(resp, 'foo/bar/Untitled.ipynb')
327 329
328 330 def test_create_untitled_txt(self):
329 331 resp = self.api.create_untitled(path='foo/bar', ext='.txt')
330 332 self._check_created(resp, 'foo/bar/untitled.txt', type='file')
331 333
332 334 resp = self.api.read(path='foo/bar/untitled.txt')
333 335 model = resp.json()
334 336 self.assertEqual(model['type'], 'file')
335 337 self.assertEqual(model['format'], 'text')
336 338 self.assertEqual(model['content'], '')
337 339
338 340 def test_upload(self):
339 341 nb = new_notebook()
340 342 nbmodel = {'content': nb, 'type': 'notebook'}
341 343 path = u'Γ₯ b/Upload tΓ©st.ipynb'
342 344 resp = self.api.upload(path, body=json.dumps(nbmodel))
343 345 self._check_created(resp, path)
344 346
345 347 def test_mkdir_untitled(self):
346 348 resp = self.api.mkdir_untitled(path=u'Γ₯ b')
347 349 self._check_created(resp, u'Γ₯ b/Untitled Folder', type='directory')
348 350
349 351 # Second time
350 352 resp = self.api.mkdir_untitled(path=u'Γ₯ b')
351 353 self._check_created(resp, u'Γ₯ b/Untitled Folder 1', type='directory')
352 354
353 355 # And two directories down
354 356 resp = self.api.mkdir_untitled(path='foo/bar')
355 357 self._check_created(resp, 'foo/bar/Untitled Folder', type='directory')
356 358
357 359 def test_mkdir(self):
358 360 path = u'Γ₯ b/New βˆ‚ir'
359 361 resp = self.api.mkdir(path)
360 362 self._check_created(resp, path, type='directory')
361 363
362 364 def test_mkdir_hidden_400(self):
363 365 with assert_http_error(400):
364 366 resp = self.api.mkdir(u'Γ₯ b/.hidden')
365 367
366 368 def test_upload_txt(self):
367 369 body = u'ΓΌnicode tΓ©xt'
368 370 model = {
369 371 'content' : body,
370 372 'format' : 'text',
371 373 'type' : 'file',
372 374 }
373 375 path = u'Γ₯ b/Upload tΓ©st.txt'
374 376 resp = self.api.upload(path, body=json.dumps(model))
375 377
376 378 # check roundtrip
377 379 resp = self.api.read(path)
378 380 model = resp.json()
379 381 self.assertEqual(model['type'], 'file')
380 382 self.assertEqual(model['format'], 'text')
381 383 self.assertEqual(model['content'], body)
382 384
383 385 def test_upload_b64(self):
384 386 body = b'\xFFblob'
385 387 b64body = base64.encodestring(body).decode('ascii')
386 388 model = {
387 389 'content' : b64body,
388 390 'format' : 'base64',
389 391 'type' : 'file',
390 392 }
391 393 path = u'Γ₯ b/Upload tΓ©st.blob'
392 394 resp = self.api.upload(path, body=json.dumps(model))
393 395
394 396 # check roundtrip
395 397 resp = self.api.read(path)
396 398 model = resp.json()
397 399 self.assertEqual(model['type'], 'file')
398 400 self.assertEqual(model['path'], path)
399 401 self.assertEqual(model['format'], 'base64')
400 402 decoded = base64.decodestring(model['content'].encode('ascii'))
401 403 self.assertEqual(decoded, body)
402 404
403 405 def test_upload_v2(self):
404 406 nb = v2.new_notebook()
405 407 ws = v2.new_worksheet()
406 408 nb.worksheets.append(ws)
407 409 ws.cells.append(v2.new_code_cell(input='print("hi")'))
408 410 nbmodel = {'content': nb, 'type': 'notebook'}
409 411 path = u'Γ₯ b/Upload tΓ©st.ipynb'
410 412 resp = self.api.upload(path, body=json.dumps(nbmodel))
411 413 self._check_created(resp, path)
412 414 resp = self.api.read(path)
413 415 data = resp.json()
414 416 self.assertEqual(data['content']['nbformat'], 4)
415 417
416 418 def test_copy(self):
417 419 resp = self.api.copy(u'Γ₯ b/Γ§ d.ipynb', u'Γ₯ b')
418 420 self._check_created(resp, u'Γ₯ b/Γ§ d-Copy1.ipynb')
419 421
420 422 resp = self.api.copy(u'Γ₯ b/Γ§ d.ipynb', u'Γ₯ b')
421 423 self._check_created(resp, u'Γ₯ b/Γ§ d-Copy2.ipynb')
422 424
423 425 def test_copy_copy(self):
424 426 resp = self.api.copy(u'Γ₯ b/Γ§ d.ipynb', u'Γ₯ b')
425 427 self._check_created(resp, u'Γ₯ b/Γ§ d-Copy1.ipynb')
426 428
427 429 resp = self.api.copy(u'Γ₯ b/Γ§ d-Copy1.ipynb', u'Γ₯ b')
428 430 self._check_created(resp, u'Γ₯ b/Γ§ d-Copy2.ipynb')
429 431
430 432 def test_copy_path(self):
431 433 resp = self.api.copy(u'foo/a.ipynb', u'Γ₯ b')
432 434 self._check_created(resp, u'Γ₯ b/a.ipynb')
433 435
434 436 resp = self.api.copy(u'foo/a.ipynb', u'Γ₯ b')
435 437 self._check_created(resp, u'Γ₯ b/a-Copy1.ipynb')
436 438
437 439 def test_copy_put_400(self):
438 440 with assert_http_error(400):
439 441 resp = self.api.copy_put(u'Γ₯ b/Γ§ d.ipynb', u'Γ₯ b/cΓΈpy.ipynb')
440 442
441 443 def test_copy_dir_400(self):
442 444 # can't copy directories
443 445 with assert_http_error(400):
444 446 resp = self.api.copy(u'Γ₯ b', u'foo')
445 447
446 448 def test_delete(self):
447 449 for d, name in self.dirs_nbs:
448 450 print('%r, %r' % (d, name))
449 451 resp = self.api.delete(url_path_join(d, name + '.ipynb'))
450 452 self.assertEqual(resp.status_code, 204)
451 453
452 454 for d in self.dirs + ['/']:
453 455 nbs = notebooks_only(self.api.list(d).json())
454 456 print('------')
455 457 print(d)
456 458 print(nbs)
457 459 self.assertEqual(nbs, [])
458 460
459 461 def test_delete_dirs(self):
460 462 # depth-first delete everything, so we don't try to delete empty directories
461 463 for name in sorted(self.dirs + ['/'], key=len, reverse=True):
462 464 listing = self.api.list(name).json()['content']
463 465 for model in listing:
464 466 self.api.delete(model['path'])
465 467 listing = self.api.list('/').json()['content']
466 468 self.assertEqual(listing, [])
467 469
468 470 def test_delete_non_empty_dir(self):
469 471 """delete non-empty dir raises 400"""
470 472 with assert_http_error(400):
471 473 self.api.delete(u'Γ₯ b')
472 474
473 475 def test_rename(self):
474 476 resp = self.api.rename('foo/a.ipynb', 'foo/z.ipynb')
475 477 self.assertEqual(resp.headers['Location'].split('/')[-1], 'z.ipynb')
476 478 self.assertEqual(resp.json()['name'], 'z.ipynb')
477 479 self.assertEqual(resp.json()['path'], 'foo/z.ipynb')
478 480 assert self.isfile('foo/z.ipynb')
479 481
480 482 nbs = notebooks_only(self.api.list('foo').json())
481 483 nbnames = set(n['name'] for n in nbs)
482 484 self.assertIn('z.ipynb', nbnames)
483 485 self.assertNotIn('a.ipynb', nbnames)
484 486
485 487 def test_rename_existing(self):
486 488 with assert_http_error(409):
487 489 self.api.rename('foo/a.ipynb', 'foo/b.ipynb')
488 490
489 491 def test_save(self):
490 492 resp = self.api.read('foo/a.ipynb')
491 493 nbcontent = json.loads(resp.text)['content']
492 494 nb = from_dict(nbcontent)
493 495 nb.cells.append(new_markdown_cell(u'Created by test Β³'))
494 496
495 497 nbmodel= {'content': nb, 'type': 'notebook'}
496 498 resp = self.api.save('foo/a.ipynb', body=json.dumps(nbmodel))
497 499
498 500 nbcontent = self.api.read('foo/a.ipynb').json()['content']
499 501 newnb = from_dict(nbcontent)
500 502 self.assertEqual(newnb.cells[0].source,
501 503 u'Created by test Β³')
502 504
503 505
504 506 def test_checkpoints(self):
505 507 resp = self.api.read('foo/a.ipynb')
506 508 r = self.api.new_checkpoint('foo/a.ipynb')
507 509 self.assertEqual(r.status_code, 201)
508 510 cp1 = r.json()
509 511 self.assertEqual(set(cp1), {'id', 'last_modified'})
510 512 self.assertEqual(r.headers['Location'].split('/')[-1], cp1['id'])
511 513
512 514 # Modify it
513 515 nbcontent = json.loads(resp.text)['content']
514 516 nb = from_dict(nbcontent)
515 517 hcell = new_markdown_cell('Created by test')
516 518 nb.cells.append(hcell)
517 519 # Save
518 520 nbmodel= {'content': nb, 'type': 'notebook'}
519 521 resp = self.api.save('foo/a.ipynb', body=json.dumps(nbmodel))
520 522
521 523 # List checkpoints
522 524 cps = self.api.get_checkpoints('foo/a.ipynb').json()
523 525 self.assertEqual(cps, [cp1])
524 526
525 527 nbcontent = self.api.read('foo/a.ipynb').json()['content']
526 528 nb = from_dict(nbcontent)
527 529 self.assertEqual(nb.cells[0].source, 'Created by test')
528 530
529 531 # Restore cp1
530 532 r = self.api.restore_checkpoint('foo/a.ipynb', cp1['id'])
531 533 self.assertEqual(r.status_code, 204)
532 534 nbcontent = self.api.read('foo/a.ipynb').json()['content']
533 535 nb = from_dict(nbcontent)
534 536 self.assertEqual(nb.cells, [])
535 537
536 538 # Delete cp1
537 539 r = self.api.delete_checkpoint('foo/a.ipynb', cp1['id'])
538 540 self.assertEqual(r.status_code, 204)
539 541 cps = self.api.get_checkpoints('foo/a.ipynb').json()
540 542 self.assertEqual(cps, [])
General Comments 0
You need to be logged in to leave comments. Login now