test_contents_api.py
687 lines
| 24.2 KiB
| text/x-python
|
PythonLexer
MinRK
|
r13091 | # coding: utf-8 | ||
MinRK
|
r17524 | """Test the contents webservice API.""" | ||
Zachary Sailer
|
r13041 | |||
MinRK
|
r17525 | import base64 | ||
Scott Sanderson
|
r19748 | from contextlib import contextmanager | ||
Thomas Kluyver
|
r13083 | import io | ||
MinRK
|
r13137 | import json | ||
Zachary Sailer
|
r13041 | import os | ||
Thomas Kluyver
|
r13083 | import shutil | ||
MinRK
|
r13091 | from unicodedata import normalize | ||
Thomas Kluyver
|
r13083 | pjoin = os.path.join | ||
Zachary Sailer
|
r13041 | import requests | ||
Scott Sanderson
|
r19839 | from ..filecheckpoints import GenericFileCheckpoints | ||
Scott Sanderson
|
r19838 | |||
from IPython.config import Config | ||||
Min RK
|
r19604 | from IPython.html.utils import url_path_join, url_escape, to_os_path | ||
Thomas Kluyver
|
r13099 | from IPython.html.tests.launchnotebook import NotebookTestBase, assert_http_error | ||
MinRK
|
r18607 | from IPython.nbformat import read, write, from_dict | ||
from IPython.nbformat.v4 import ( | ||||
new_notebook, new_markdown_cell, | ||||
) | ||||
MinRK
|
r13141 | from IPython.nbformat import v2 | ||
MinRK
|
r13130 | from IPython.utils import py3compat | ||
Thomas Kluyver
|
r13083 | from IPython.utils.data import uniq_stable | ||
Scott Sanderson
|
r19748 | from IPython.utils.tempdir import TemporaryDirectory | ||
Thomas Kluyver
|
r13083 | |||
MinRK
|
r13130 | |||
MinRK
|
r17525 | def notebooks_only(dir_model): | ||
return [nb for nb in dir_model['content'] if nb['type']=='notebook'] | ||||
Brian E. Granger
|
r15079 | |||
MinRK
|
r17525 | def dirs_only(dir_model): | ||
return [x for x in dir_model['content'] if x['type']=='directory'] | ||||
Thomas Kluyver
|
r15522 | |||
Brian E. Granger
|
r15079 | |||
MinRK
|
r17524 | class API(object): | ||
"""Wrapper for contents API calls.""" | ||||
Thomas Kluyver
|
r13083 | def __init__(self, base_url): | ||
self.base_url = base_url | ||||
Thomas Kluyver
|
r18790 | def _req(self, verb, path, body=None, params=None): | ||
Thomas Kluyver
|
r13083 | response = requests.request(verb, | ||
MinRK
|
r17524 | url_path_join(self.base_url, 'api/contents', path), | ||
Thomas Kluyver
|
r18795 | data=body, params=params, | ||
MinRK
|
r13130 | ) | ||
Thomas Kluyver
|
r13083 | response.raise_for_status() | ||
return response | ||||
def list(self, path='/'): | ||||
return self._req('GET', path) | ||||
Min RK
|
r20134 | def read(self, path, type=None, format=None, content=None): | ||
Thomas Kluyver
|
r18790 | params = {} | ||
Min RK
|
r19391 | if type is not None: | ||
params['type'] = type | ||||
Thomas Kluyver
|
r18788 | if format is not None: | ||
Thomas Kluyver
|
r18790 | params['format'] = format | ||
Min RK
|
r20134 | if content == False: | ||
params['content'] = '0' | ||||
Thomas Kluyver
|
r18790 | return self._req('GET', path, params=params) | ||
Thomas Kluyver
|
r13083 | |||
Min RK
|
r18759 | def create_untitled(self, path='/', ext='.ipynb'): | ||
MinRK
|
r17527 | body = None | ||
if ext: | ||||
body = json.dumps({'ext': ext}) | ||||
return self._req('POST', path, body) | ||||
Thomas Kluyver
|
r13083 | |||
Min RK
|
r18758 | def mkdir_untitled(self, path='/'): | ||
return self._req('POST', path, json.dumps({'type': 'directory'})) | ||||
Min RK
|
r18750 | def copy(self, copy_from, path='/'): | ||
MinRK
|
r13137 | body = json.dumps({'copy_from':copy_from}) | ||
return self._req('POST', path, body) | ||||
MinRK
|
r13130 | |||
MinRK
|
r18749 | def create(self, path='/'): | ||
return self._req('PUT', path) | ||||
MinRK
|
r13130 | |||
MinRK
|
r18749 | def upload(self, path, body): | ||
return self._req('PUT', path, body) | ||||
Thomas Kluyver
|
r13084 | |||
MinRK
|
r18749 | def mkdir(self, path='/'): | ||
return self._req('PUT', path, json.dumps({'type': 'directory'})) | ||||
MinRK
|
r17530 | |||
Min RK
|
r18750 | def copy_put(self, copy_from, path='/'): | ||
MinRK
|
r13137 | body = json.dumps({'copy_from':copy_from}) | ||
MinRK
|
r18749 | return self._req('PUT', path, body) | ||
Thomas Kluyver
|
r13086 | |||
MinRK
|
r18749 | def save(self, path, body): | ||
return self._req('PUT', path, body) | ||||
Thomas Kluyver
|
r13085 | |||
MinRK
|
r18749 | def delete(self, path='/'): | ||
return self._req('DELETE', path) | ||||
Thomas Kluyver
|
r13083 | |||
MinRK
|
r18749 | def rename(self, path, new_path): | ||
body = json.dumps({'path': new_path}) | ||||
return self._req('PATCH', path, body) | ||||
Zachary Sailer
|
r13041 | |||
MinRK
|
r18749 | def get_checkpoints(self, path): | ||
return self._req('GET', url_path_join(path, 'checkpoints')) | ||||
Thomas Kluyver
|
r13109 | |||
MinRK
|
r18749 | def new_checkpoint(self, path): | ||
return self._req('POST', url_path_join(path, 'checkpoints')) | ||||
Thomas Kluyver
|
r13109 | |||
MinRK
|
r18749 | def restore_checkpoint(self, path, checkpoint_id): | ||
return self._req('POST', url_path_join(path, 'checkpoints', checkpoint_id)) | ||||
Thomas Kluyver
|
r13109 | |||
MinRK
|
r18749 | def delete_checkpoint(self, path, checkpoint_id): | ||
return self._req('DELETE', url_path_join(path, 'checkpoints', checkpoint_id)) | ||||
Thomas Kluyver
|
r13109 | |||
Zachary Sailer
|
r13041 | class APITest(NotebookTestBase): | ||
"""Test the kernels web service API""" | ||||
Thomas Kluyver
|
r13083 | dirs_nbs = [('', 'inroot'), | ||
('Directory with spaces in', 'inspace'), | ||||
(u'unicodé', 'innonascii'), | ||||
('foo', 'a'), | ||||
('foo', 'b'), | ||||
('foo', 'name with spaces'), | ||||
('foo', u'unicodé'), | ||||
('foo/bar', 'baz'), | ||||
Thomas Kluyver
|
r15523 | ('ordering', 'A'), | ||
('ordering', 'b'), | ||||
('ordering', 'C'), | ||||
(u'å b', u'ç d'), | ||||
Thomas Kluyver
|
r13083 | ] | ||
Thomas Kluyver
|
r15522 | hidden_dirs = ['.hidden', '__pycache__'] | ||
Thomas Kluyver
|
r13083 | |||
Scott Sanderson
|
r19617 | # Don't include root dir. | ||
dirs = uniq_stable([py3compat.cast_unicode(d) for (d,n) in dirs_nbs[1:]]) | ||||
MinRK
|
r15623 | top_level_dirs = {normalize('NFC', d.split('/')[0]) for d in dirs} | ||
Thomas Kluyver
|
r13083 | |||
MinRK
|
r17525 | @staticmethod | ||
def _blob_for_name(name): | ||||
return name.encode('utf-8') + b'\xFF' | ||||
@staticmethod | ||||
def _txt_for_name(name): | ||||
return u'%s text file' % name | ||||
Min RK
|
r19604 | |||
def to_os_path(self, api_path): | ||||
return to_os_path(api_path, root=self.notebook_dir.name) | ||||
def make_dir(self, api_path): | ||||
"""Create a directory at api_path""" | ||||
os_path = self.to_os_path(api_path) | ||||
try: | ||||
os.makedirs(os_path) | ||||
except OSError: | ||||
print("Directory already exists: %r" % os_path) | ||||
Scott Sanderson
|
r19613 | |||
Min RK
|
r19604 | def make_txt(self, api_path, txt): | ||
"""Make a text file at a given api_path""" | ||||
os_path = self.to_os_path(api_path) | ||||
with io.open(os_path, 'w', encoding='utf-8') as f: | ||||
f.write(txt) | ||||
def make_blob(self, api_path, blob): | ||||
"""Make a binary file at a given api_path""" | ||||
os_path = self.to_os_path(api_path) | ||||
with io.open(os_path, 'wb') as f: | ||||
f.write(blob) | ||||
def make_nb(self, api_path, nb): | ||||
"""Make a notebook file at a given api_path""" | ||||
os_path = self.to_os_path(api_path) | ||||
with io.open(os_path, 'w', encoding='utf-8') as f: | ||||
write(nb, f, version=4) | ||||
Scott Sanderson
|
r19613 | |||
def delete_dir(self, api_path): | ||||
"""Delete a directory at api_path, removing any contents.""" | ||||
os_path = self.to_os_path(api_path) | ||||
shutil.rmtree(os_path, ignore_errors=True) | ||||
def delete_file(self, api_path): | ||||
"""Delete a file at the given path if it exists.""" | ||||
if self.isfile(api_path): | ||||
os.unlink(self.to_os_path(api_path)) | ||||
Min RK
|
r19604 | |||
def isfile(self, api_path): | ||||
return os.path.isfile(self.to_os_path(api_path)) | ||||
def isdir(self, api_path): | ||||
return os.path.isdir(self.to_os_path(api_path)) | ||||
Thomas Kluyver
|
r13083 | def setUp(self): | ||
MinRK
|
r17525 | |||
Thomas Kluyver
|
r15522 | for d in (self.dirs + self.hidden_dirs): | ||
Min RK
|
r19604 | self.make_dir(d) | ||
Thomas Kluyver
|
r13083 | |||
for d, name in self.dirs_nbs: | ||||
MinRK
|
r17525 | # create a notebook | ||
Min RK
|
r19604 | nb = new_notebook() | ||
self.make_nb(u'{}/{}.ipynb'.format(d, name), nb) | ||||
MinRK
|
r17525 | # create a text file | ||
Min RK
|
r19604 | txt = self._txt_for_name(name) | ||
self.make_txt(u'{}/{}.txt'.format(d, name), txt) | ||||
MinRK
|
r17525 | # create a binary file | ||
Min RK
|
r19604 | blob = self._blob_for_name(name) | ||
self.make_blob(u'{}/{}.blob'.format(d, name), blob) | ||||
MinRK
|
r17525 | |||
MinRK
|
r17524 | self.api = API(self.base_url()) | ||
Thomas Kluyver
|
r13083 | |||
def tearDown(self): | ||||
Thomas Kluyver
|
r15522 | for dname in (list(self.top_level_dirs) + self.hidden_dirs): | ||
Scott Sanderson
|
r19613 | self.delete_dir(dname) | ||
self.delete_file('inroot.ipynb') | ||||
Thomas Kluyver
|
r13083 | |||
def test_list_notebooks(self): | ||||
MinRK
|
r17524 | nbs = notebooks_only(self.api.list().json()) | ||
Thomas Kluyver
|
r13083 | self.assertEqual(len(nbs), 1) | ||
self.assertEqual(nbs[0]['name'], 'inroot.ipynb') | ||||
MinRK
|
r17524 | nbs = notebooks_only(self.api.list('/Directory with spaces in/').json()) | ||
Thomas Kluyver
|
r13083 | self.assertEqual(len(nbs), 1) | ||
self.assertEqual(nbs[0]['name'], 'inspace.ipynb') | ||||
MinRK
|
r17524 | nbs = notebooks_only(self.api.list(u'/unicodé/').json()) | ||
Thomas Kluyver
|
r13083 | self.assertEqual(len(nbs), 1) | ||
self.assertEqual(nbs[0]['name'], 'innonascii.ipynb') | ||||
MinRK
|
r18749 | self.assertEqual(nbs[0]['path'], u'unicodé/innonascii.ipynb') | ||
Thomas Kluyver
|
r13083 | |||
MinRK
|
r17524 | nbs = notebooks_only(self.api.list('/foo/bar/').json()) | ||
Thomas Kluyver
|
r13083 | self.assertEqual(len(nbs), 1) | ||
self.assertEqual(nbs[0]['name'], 'baz.ipynb') | ||||
MinRK
|
r18749 | self.assertEqual(nbs[0]['path'], 'foo/bar/baz.ipynb') | ||
Thomas Kluyver
|
r13083 | |||
MinRK
|
r17524 | nbs = notebooks_only(self.api.list('foo').json()) | ||
Thomas Kluyver
|
r13083 | self.assertEqual(len(nbs), 4) | ||
MinRK
|
r13091 | nbnames = { normalize('NFC', n['name']) for n in nbs } | ||
expected = [ u'a.ipynb', u'b.ipynb', u'name with spaces.ipynb', u'unicodé.ipynb'] | ||||
expected = { normalize('NFC', name) for name in expected } | ||||
self.assertEqual(nbnames, expected) | ||||
MinRK
|
r17523 | |||
MinRK
|
r17524 | nbs = notebooks_only(self.api.list('ordering').json()) | ||
Thomas Kluyver
|
r15523 | nbnames = [n['name'] for n in nbs] | ||
expected = ['A.ipynb', 'b.ipynb', 'C.ipynb'] | ||||
self.assertEqual(nbnames, expected) | ||||
Thomas Kluyver
|
r13083 | |||
Thomas Kluyver
|
r15522 | def test_list_dirs(self): | ||
MinRK
|
r17524 | dirs = dirs_only(self.api.list().json()) | ||
MinRK
|
r15623 | dir_names = {normalize('NFC', d['name']) for d in dirs} | ||
Thomas Kluyver
|
r15522 | self.assertEqual(dir_names, self.top_level_dirs) # Excluding hidden dirs | ||
Min RK
|
r20134 | def test_get_dir_no_content(self): | ||
for d in self.dirs: | ||||
model = self.api.read(d, content=False).json() | ||||
self.assertEqual(model['path'], d) | ||||
self.assertEqual(model['type'], 'directory') | ||||
self.assertIn('content', model) | ||||
self.assertEqual(model['content'], None) | ||||
Thomas Kluyver
|
r13099 | def test_list_nonexistant_dir(self): | ||
with assert_http_error(404): | ||||
MinRK
|
r17524 | self.api.list('nonexistant') | ||
Thomas Kluyver
|
r13087 | |||
MinRK
|
r17525 | def test_get_nb_contents(self): | ||
Thomas Kluyver
|
r13083 | for d, name in self.dirs_nbs: | ||
MinRK
|
r18749 | path = url_path_join(d, name + '.ipynb') | ||
nb = self.api.read(path).json() | ||||
MinRK
|
r13130 | self.assertEqual(nb['name'], u'%s.ipynb' % name) | ||
MinRK
|
r18749 | self.assertEqual(nb['path'], path) | ||
MinRK
|
r17525 | self.assertEqual(nb['type'], 'notebook') | ||
self.assertIn('content', nb) | ||||
self.assertEqual(nb['format'], 'json') | ||||
Thomas Kluyver
|
r13083 | self.assertIn('metadata', nb['content']) | ||
self.assertIsInstance(nb['content']['metadata'], dict) | ||||
Min RK
|
r20134 | def test_get_nb_no_content(self): | ||
for d, name in self.dirs_nbs: | ||||
path = url_path_join(d, name + '.ipynb') | ||||
nb = self.api.read(path, content=False).json() | ||||
self.assertEqual(nb['name'], u'%s.ipynb' % name) | ||||
self.assertEqual(nb['path'], path) | ||||
self.assertEqual(nb['type'], 'notebook') | ||||
self.assertIn('content', nb) | ||||
self.assertEqual(nb['content'], None) | ||||
MinRK
|
r17525 | def test_get_contents_no_such_file(self): | ||
Thomas Kluyver
|
r13083 | # Name that doesn't exist - should be a 404 | ||
Thomas Kluyver
|
r13099 | with assert_http_error(404): | ||
MinRK
|
r18749 | self.api.read('foo/q.ipynb') | ||
Thomas Kluyver
|
r13083 | |||
MinRK
|
r17525 | def test_get_text_file_contents(self): | ||
for d, name in self.dirs_nbs: | ||||
MinRK
|
r18749 | path = url_path_join(d, name + '.txt') | ||
model = self.api.read(path).json() | ||||
MinRK
|
r17525 | self.assertEqual(model['name'], u'%s.txt' % name) | ||
MinRK
|
r18749 | self.assertEqual(model['path'], path) | ||
MinRK
|
r17525 | self.assertIn('content', model) | ||
self.assertEqual(model['format'], 'text') | ||||
self.assertEqual(model['type'], 'file') | ||||
self.assertEqual(model['content'], self._txt_for_name(name)) | ||||
# Name that doesn't exist - should be a 404 | ||||
with assert_http_error(404): | ||||
MinRK
|
r18749 | self.api.read('foo/q.txt') | ||
MinRK
|
r17525 | |||
Thomas Kluyver
|
r18788 | # Specifying format=text should fail on a non-UTF-8 file | ||
with assert_http_error(400): | ||||
Min RK
|
r19391 | self.api.read('foo/bar/baz.blob', type='file', format='text') | ||
Thomas Kluyver
|
r18788 | |||
MinRK
|
r17525 | def test_get_binary_file_contents(self): | ||
for d, name in self.dirs_nbs: | ||||
MinRK
|
r18749 | path = url_path_join(d, name + '.blob') | ||
model = self.api.read(path).json() | ||||
MinRK
|
r17525 | self.assertEqual(model['name'], u'%s.blob' % name) | ||
MinRK
|
r18749 | self.assertEqual(model['path'], path) | ||
MinRK
|
r17525 | self.assertIn('content', model) | ||
self.assertEqual(model['format'], 'base64') | ||||
self.assertEqual(model['type'], 'file') | ||||
Scott Sanderson
|
r19615 | self.assertEqual( | ||
Scott Sanderson
|
r19616 | base64.decodestring(model['content'].encode('ascii')), | ||
Scott Sanderson
|
r19615 | self._blob_for_name(name), | ||
) | ||||
MinRK
|
r17525 | |||
# Name that doesn't exist - should be a 404 | ||||
with assert_http_error(404): | ||||
MinRK
|
r18749 | self.api.read('foo/q.txt') | ||
MinRK
|
r17525 | |||
Thomas Kluyver
|
r18781 | def test_get_bad_type(self): | ||
with assert_http_error(400): | ||||
Min RK
|
r19391 | self.api.read(u'unicodé', type='file') # this is a directory | ||
Thomas Kluyver
|
r18781 | |||
with assert_http_error(400): | ||||
Min RK
|
r19391 | self.api.read(u'unicodé/innonascii.ipynb', type='directory') | ||
Thomas Kluyver
|
r18781 | |||
MinRK
|
r18749 | def _check_created(self, resp, path, type='notebook'): | ||
Thomas Kluyver
|
r13084 | self.assertEqual(resp.status_code, 201) | ||
MinRK
|
r13130 | location_header = py3compat.str_to_unicode(resp.headers['Location']) | ||
MinRK
|
r18749 | self.assertEqual(location_header, url_escape(url_path_join(u'/api/contents', path))) | ||
MinRK
|
r17528 | rjson = resp.json() | ||
MinRK
|
r18749 | self.assertEqual(rjson['name'], path.rsplit('/', 1)[-1]) | ||
MinRK
|
r17528 | self.assertEqual(rjson['path'], path) | ||
self.assertEqual(rjson['type'], type) | ||||
Min RK
|
r19604 | isright = self.isdir if type == 'directory' else self.isfile | ||
assert isright(path) | ||||
Thomas Kluyver
|
r13083 | |||
def test_create_untitled(self): | ||||
MinRK
|
r17524 | resp = self.api.create_untitled(path=u'Ã¥ b') | ||
Min RK
|
r18813 | self._check_created(resp, u'Ã¥ b/Untitled.ipynb') | ||
Thomas Kluyver
|
r13083 | |||
# Second time | ||||
MinRK
|
r17524 | resp = self.api.create_untitled(path=u'Ã¥ b') | ||
MinRK
|
r18749 | self._check_created(resp, u'Ã¥ b/Untitled1.ipynb') | ||
Thomas Kluyver
|
r13083 | |||
# And two directories down | ||||
MinRK
|
r17524 | resp = self.api.create_untitled(path='foo/bar') | ||
Min RK
|
r18813 | self._check_created(resp, 'foo/bar/Untitled.ipynb') | ||
Thomas Kluyver
|
r13083 | |||
MinRK
|
r17527 | def test_create_untitled_txt(self): | ||
resp = self.api.create_untitled(path='foo/bar', ext='.txt') | ||||
Min RK
|
r18813 | self._check_created(resp, 'foo/bar/untitled.txt', type='file') | ||
MinRK
|
r17527 | |||
Min RK
|
r18813 | resp = self.api.read(path='foo/bar/untitled.txt') | ||
MinRK
|
r17527 | model = resp.json() | ||
self.assertEqual(model['type'], 'file') | ||||
self.assertEqual(model['format'], 'text') | ||||
self.assertEqual(model['content'], '') | ||||
MinRK
|
r13130 | def test_upload(self): | ||
MinRK
|
r18584 | nb = new_notebook() | ||
MinRK
|
r17527 | nbmodel = {'content': nb, 'type': 'notebook'} | ||
MinRK
|
r18749 | path = u'å b/Upload tést.ipynb' | ||
resp = self.api.upload(path, body=json.dumps(nbmodel)) | ||||
self._check_created(resp, path) | ||||
MinRK
|
r17528 | |||
Min RK
|
r18758 | def test_mkdir_untitled(self): | ||
resp = self.api.mkdir_untitled(path=u'Ã¥ b') | ||||
Min RK
|
r18813 | self._check_created(resp, u'Ã¥ b/Untitled Folder', type='directory') | ||
Min RK
|
r18758 | |||
# Second time | ||||
resp = self.api.mkdir_untitled(path=u'Ã¥ b') | ||||
Min RK
|
r18813 | self._check_created(resp, u'Ã¥ b/Untitled Folder 1', type='directory') | ||
Min RK
|
r18758 | |||
# And two directories down | ||||
resp = self.api.mkdir_untitled(path='foo/bar') | ||||
Min RK
|
r18813 | self._check_created(resp, 'foo/bar/Untitled Folder', type='directory') | ||
Min RK
|
r18758 | |||
MinRK
|
r17528 | def test_mkdir(self): | ||
MinRK
|
r18749 | path = u'å b/New ∂ir' | ||
resp = self.api.mkdir(path) | ||||
self._check_created(resp, path, type='directory') | ||||
MinRK
|
r13130 | |||
MinRK
|
r17537 | def test_mkdir_hidden_400(self): | ||
with assert_http_error(400): | ||||
MinRK
|
r18749 | resp = self.api.mkdir(u'Ã¥ b/.hidden') | ||
MinRK
|
r17537 | |||
MinRK
|
r17527 | def test_upload_txt(self): | ||
body = u'ünicode téxt' | ||||
model = { | ||||
'content' : body, | ||||
'format' : 'text', | ||||
'type' : 'file', | ||||
} | ||||
MinRK
|
r18749 | path = u'å b/Upload tést.txt' | ||
resp = self.api.upload(path, body=json.dumps(model)) | ||||
MinRK
|
r17527 | |||
# check roundtrip | ||||
MinRK
|
r18749 | resp = self.api.read(path) | ||
MinRK
|
r17527 | model = resp.json() | ||
self.assertEqual(model['type'], 'file') | ||||
self.assertEqual(model['format'], 'text') | ||||
self.assertEqual(model['content'], body) | ||||
def test_upload_b64(self): | ||||
body = b'\xFFblob' | ||||
b64body = base64.encodestring(body).decode('ascii') | ||||
model = { | ||||
'content' : b64body, | ||||
'format' : 'base64', | ||||
'type' : 'file', | ||||
} | ||||
MinRK
|
r18749 | path = u'å b/Upload tést.blob' | ||
resp = self.api.upload(path, body=json.dumps(model)) | ||||
MinRK
|
r17527 | |||
# check roundtrip | ||||
MinRK
|
r18749 | resp = self.api.read(path) | ||
MinRK
|
r17527 | model = resp.json() | ||
self.assertEqual(model['type'], 'file') | ||||
MinRK
|
r18749 | self.assertEqual(model['path'], path) | ||
MinRK
|
r17527 | self.assertEqual(model['format'], 'base64') | ||
decoded = base64.decodestring(model['content'].encode('ascii')) | ||||
self.assertEqual(decoded, body) | ||||
MinRK
|
r13141 | def test_upload_v2(self): | ||
nb = v2.new_notebook() | ||||
ws = v2.new_worksheet() | ||||
nb.worksheets.append(ws) | ||||
ws.cells.append(v2.new_code_cell(input='print("hi")')) | ||||
MinRK
|
r17527 | nbmodel = {'content': nb, 'type': 'notebook'} | ||
MinRK
|
r18749 | path = u'å b/Upload tést.ipynb' | ||
resp = self.api.upload(path, body=json.dumps(nbmodel)) | ||||
self._check_created(resp, path) | ||||
resp = self.api.read(path) | ||||
MinRK
|
r13141 | data = resp.json() | ||
MinRK
|
r18607 | self.assertEqual(data['content']['nbformat'], 4) | ||
MinRK
|
r13141 | |||
Min RK
|
r18750 | def test_copy(self): | ||
Min RK
|
r18813 | resp = self.api.copy(u'å b/ç d.ipynb', u'å b') | ||
self._check_created(resp, u'å b/ç d-Copy1.ipynb') | ||||
MinRK
|
r18749 | |||
Min RK
|
r18750 | resp = self.api.copy(u'å b/ç d.ipynb', u'å b') | ||
Min RK
|
r18813 | self._check_created(resp, u'å b/ç d-Copy2.ipynb') | ||
def test_copy_copy(self): | ||||
resp = self.api.copy(u'å b/ç d.ipynb', u'å b') | ||||
self._check_created(resp, u'å b/ç d-Copy1.ipynb') | ||||
resp = self.api.copy(u'å b/ç d-Copy1.ipynb', u'å b') | ||||
self._check_created(resp, u'å b/ç d-Copy2.ipynb') | ||||
MinRK
|
r17535 | def test_copy_path(self): | ||
Min RK
|
r18750 | resp = self.api.copy(u'foo/a.ipynb', u'Ã¥ b') | ||
Min RK
|
r18813 | self._check_created(resp, u'Ã¥ b/a.ipynb') | ||
resp = self.api.copy(u'foo/a.ipynb', u'Ã¥ b') | ||||
self._check_created(resp, u'Ã¥ b/a-Copy1.ipynb') | ||||
Min RK
|
r18750 | |||
def test_copy_put_400(self): | ||||
with assert_http_error(400): | ||||
resp = self.api.copy_put(u'å b/ç d.ipynb', u'å b/cøpy.ipynb') | ||||
MinRK
|
r17535 | |||
MinRK
|
r17530 | def test_copy_dir_400(self): | ||
# can't copy directories | ||||
with assert_http_error(400): | ||||
Min RK
|
r18750 | resp = self.api.copy(u'Ã¥ b', u'foo') | ||
MinRK
|
r17530 | |||
Thomas Kluyver
|
r13083 | def test_delete(self): | ||
for d, name in self.dirs_nbs: | ||||
MinRK
|
r18749 | print('%r, %r' % (d, name)) | ||
resp = self.api.delete(url_path_join(d, name + '.ipynb')) | ||||
Thomas Kluyver
|
r13083 | self.assertEqual(resp.status_code, 204) | ||
for d in self.dirs + ['/']: | ||||
MinRK
|
r17524 | nbs = notebooks_only(self.api.list(d).json()) | ||
MinRK
|
r18749 | print('------') | ||
print(d) | ||||
print(nbs) | ||||
self.assertEqual(nbs, []) | ||||
Thomas Kluyver
|
r13083 | |||
MinRK
|
r17530 | def test_delete_dirs(self): | ||
# depth-first delete everything, so we don't try to delete empty directories | ||||
for name in sorted(self.dirs + ['/'], key=len, reverse=True): | ||||
listing = self.api.list(name).json()['content'] | ||||
for model in listing: | ||||
MinRK
|
r18749 | self.api.delete(model['path']) | ||
MinRK
|
r17530 | listing = self.api.list('/').json()['content'] | ||
self.assertEqual(listing, []) | ||||
def test_delete_non_empty_dir(self): | ||||
"""delete non-empty dir raises 400""" | ||||
with assert_http_error(400): | ||||
self.api.delete(u'Ã¥ b') | ||||
Thomas Kluyver
|
r13083 | def test_rename(self): | ||
MinRK
|
r18749 | resp = self.api.rename('foo/a.ipynb', 'foo/z.ipynb') | ||
Thomas Kluyver
|
r13089 | self.assertEqual(resp.headers['Location'].split('/')[-1], 'z.ipynb') | ||
Thomas Kluyver
|
r13083 | self.assertEqual(resp.json()['name'], 'z.ipynb') | ||
MinRK
|
r18749 | self.assertEqual(resp.json()['path'], 'foo/z.ipynb') | ||
Min RK
|
r19604 | assert self.isfile('foo/z.ipynb') | ||
Thomas Kluyver
|
r13083 | |||
MinRK
|
r17524 | nbs = notebooks_only(self.api.list('foo').json()) | ||
Thomas Kluyver
|
r13083 | nbnames = set(n['name'] for n in nbs) | ||
self.assertIn('z.ipynb', nbnames) | ||||
self.assertNotIn('a.ipynb', nbnames) | ||||
Thomas Kluyver
|
r13085 | |||
Scott Sanderson
|
r20785 | def test_checkpoints_follow_file(self): | ||
Scott Sanderson
|
r20784 | |||
# Read initial file state | ||||
orig = self.api.read('foo/a.ipynb') | ||||
# Create a checkpoint of initial state | ||||
r = self.api.new_checkpoint('foo/a.ipynb') | ||||
cp1 = r.json() | ||||
# Modify file and save | ||||
nbcontent = json.loads(orig.text)['content'] | ||||
nb = from_dict(nbcontent) | ||||
hcell = new_markdown_cell('Created by test') | ||||
nb.cells.append(hcell) | ||||
nbmodel = {'content': nb, 'type': 'notebook'} | ||||
self.api.save('foo/a.ipynb', body=json.dumps(nbmodel)) | ||||
# Rename the file. | ||||
self.api.rename('foo/a.ipynb', 'foo/z.ipynb') | ||||
# Looking for checkpoints in the old location should yield no results. | ||||
self.assertEqual(self.api.get_checkpoints('foo/a.ipynb').json(), []) | ||||
# Looking for checkpoints in the new location should work. | ||||
cps = self.api.get_checkpoints('foo/z.ipynb').json() | ||||
Scott Sanderson
|
r20785 | self.assertEqual(cps, [cp1]) | ||
# Delete the file. The checkpoint should be deleted as well. | ||||
self.api.delete('foo/z.ipynb') | ||||
cps = self.api.get_checkpoints('foo/z.ipynb').json() | ||||
self.assertEqual(cps, []) | ||||
Scott Sanderson
|
r20784 | |||
MinRK
|
r13710 | def test_rename_existing(self): | ||
with assert_http_error(409): | ||||
MinRK
|
r18749 | self.api.rename('foo/a.ipynb', 'foo/b.ipynb') | ||
MinRK
|
r13710 | |||
Thomas Kluyver
|
r13085 | def test_save(self): | ||
MinRK
|
r18749 | resp = self.api.read('foo/a.ipynb') | ||
MinRK
|
r13138 | nbcontent = json.loads(resp.text)['content'] | ||
MinRK
|
r18601 | nb = from_dict(nbcontent) | ||
MinRK
|
r18596 | nb.cells.append(new_markdown_cell(u'Created by test ³')) | ||
Thomas Kluyver
|
r13085 | |||
Scott Sanderson
|
r20784 | nbmodel = {'content': nb, 'type': 'notebook'} | ||
MinRK
|
r18749 | resp = self.api.save('foo/a.ipynb', body=json.dumps(nbmodel)) | ||
Thomas Kluyver
|
r13085 | |||
MinRK
|
r18749 | nbcontent = self.api.read('foo/a.ipynb').json()['content'] | ||
MinRK
|
r18601 | newnb = from_dict(nbcontent) | ||
MinRK
|
r18584 | self.assertEqual(newnb.cells[0].source, | ||
Thomas Kluyver
|
r13111 | u'Created by test ³') | ||
Thomas Kluyver
|
r13087 | |||
Thomas Kluyver
|
r13109 | def test_checkpoints(self): | ||
MinRK
|
r18749 | resp = self.api.read('foo/a.ipynb') | ||
r = self.api.new_checkpoint('foo/a.ipynb') | ||||
Thomas Kluyver
|
r13109 | self.assertEqual(r.status_code, 201) | ||
cp1 = r.json() | ||||
MinRK
|
r13122 | self.assertEqual(set(cp1), {'id', 'last_modified'}) | ||
self.assertEqual(r.headers['Location'].split('/')[-1], cp1['id']) | ||||
Thomas Kluyver
|
r13109 | |||
# Modify it | ||||
MinRK
|
r13138 | nbcontent = json.loads(resp.text)['content'] | ||
MinRK
|
r18601 | nb = from_dict(nbcontent) | ||
MinRK
|
r18596 | hcell = new_markdown_cell('Created by test') | ||
MinRK
|
r18584 | nb.cells.append(hcell) | ||
Thomas Kluyver
|
r13109 | # Save | ||
MinRK
|
r18749 | nbmodel= {'content': nb, 'type': 'notebook'} | ||
resp = self.api.save('foo/a.ipynb', body=json.dumps(nbmodel)) | ||||
Thomas Kluyver
|
r13109 | |||
# List checkpoints | ||||
MinRK
|
r18749 | cps = self.api.get_checkpoints('foo/a.ipynb').json() | ||
Thomas Kluyver
|
r13109 | self.assertEqual(cps, [cp1]) | ||
MinRK
|
r18749 | nbcontent = self.api.read('foo/a.ipynb').json()['content'] | ||
MinRK
|
r18601 | nb = from_dict(nbcontent) | ||
MinRK
|
r18584 | self.assertEqual(nb.cells[0].source, 'Created by test') | ||
Thomas Kluyver
|
r13109 | |||
# Restore cp1 | ||||
MinRK
|
r18749 | r = self.api.restore_checkpoint('foo/a.ipynb', cp1['id']) | ||
Thomas Kluyver
|
r13109 | self.assertEqual(r.status_code, 204) | ||
MinRK
|
r18749 | nbcontent = self.api.read('foo/a.ipynb').json()['content'] | ||
MinRK
|
r18601 | nb = from_dict(nbcontent) | ||
MinRK
|
r18584 | self.assertEqual(nb.cells, []) | ||
Thomas Kluyver
|
r13109 | |||
# Delete cp1 | ||||
MinRK
|
r18749 | r = self.api.delete_checkpoint('foo/a.ipynb', cp1['id']) | ||
Thomas Kluyver
|
r13109 | self.assertEqual(r.status_code, 204) | ||
MinRK
|
r18749 | cps = self.api.get_checkpoints('foo/a.ipynb').json() | ||
Thomas Kluyver
|
r13109 | self.assertEqual(cps, []) | ||
Scott Sanderson
|
r19748 | |||
Scott Sanderson
|
r19786 | def test_file_checkpoints(self): | ||
""" | ||||
Test checkpointing of non-notebook files. | ||||
""" | ||||
filename = 'foo/a.txt' | ||||
resp = self.api.read(filename) | ||||
orig_content = json.loads(resp.text)['content'] | ||||
# Create a checkpoint. | ||||
r = self.api.new_checkpoint(filename) | ||||
self.assertEqual(r.status_code, 201) | ||||
cp1 = r.json() | ||||
self.assertEqual(set(cp1), {'id', 'last_modified'}) | ||||
self.assertEqual(r.headers['Location'].split('/')[-1], cp1['id']) | ||||
# Modify the file and save. | ||||
new_content = orig_content + '\nsecond line' | ||||
model = { | ||||
'content': new_content, | ||||
'type': 'file', | ||||
'format': 'text', | ||||
} | ||||
resp = self.api.save(filename, body=json.dumps(model)) | ||||
# List checkpoints | ||||
cps = self.api.get_checkpoints(filename).json() | ||||
self.assertEqual(cps, [cp1]) | ||||
content = self.api.read(filename).json()['content'] | ||||
self.assertEqual(content, new_content) | ||||
# Restore cp1 | ||||
r = self.api.restore_checkpoint(filename, cp1['id']) | ||||
self.assertEqual(r.status_code, 204) | ||||
restored_content = self.api.read(filename).json()['content'] | ||||
self.assertEqual(restored_content, orig_content) | ||||
# Delete cp1 | ||||
r = self.api.delete_checkpoint(filename, cp1['id']) | ||||
self.assertEqual(r.status_code, 204) | ||||
cps = self.api.get_checkpoints(filename).json() | ||||
self.assertEqual(cps, []) | ||||
Scott Sanderson
|
r19748 | @contextmanager | ||
def patch_cp_root(self, dirname): | ||||
""" | ||||
Temporarily patch the root dir of our checkpoint manager. | ||||
""" | ||||
Scott Sanderson
|
r19839 | cpm = self.notebook.contents_manager.checkpoints | ||
Scott Sanderson
|
r19748 | old_dirname = cpm.root_dir | ||
cpm.root_dir = dirname | ||||
try: | ||||
yield | ||||
finally: | ||||
cpm.root_dir = old_dirname | ||||
def test_checkpoints_separate_root(self): | ||||
""" | ||||
Scott Sanderson
|
r19839 | Test that FileCheckpoints functions correctly even when it's | ||
Scott Sanderson
|
r19748 | using a different root dir from FileContentsManager. This also keeps | ||
the implementation honest for use with ContentsManagers that don't map | ||||
models to the filesystem | ||||
Scott Sanderson
|
r19786 | Override this method to a no-op when testing other managers. | ||
""" | ||||
Scott Sanderson
|
r19748 | with TemporaryDirectory() as td: | ||
with self.patch_cp_root(td): | ||||
self.test_checkpoints() | ||||
Scott Sanderson
|
r19786 | |||
with TemporaryDirectory() as td: | ||||
with self.patch_cp_root(td): | ||||
self.test_file_checkpoints() | ||||
Scott Sanderson
|
r19828 | |||
Scott Sanderson
|
r19838 | class GenericFileCheckpointsAPITest(APITest): | ||
""" | ||||
Scott Sanderson
|
r19839 | Run the tests from APITest with GenericFileCheckpoints. | ||
Scott Sanderson
|
r19838 | """ | ||
config = Config() | ||||
Scott Sanderson
|
r19839 | config.FileContentsManager.checkpoints_class = GenericFileCheckpoints | ||
def test_config_did_something(self): | ||||
self.assertIsInstance( | ||||
self.notebook.contents_manager.checkpoints, | ||||
GenericFileCheckpoints, | ||||
) | ||||