Show More
handlers.py
137 lines
| 4.5 KiB
| text/x-python
|
PythonLexer
Thomas Kluyver
|
r13919 | import io | ||
Thomas Kluyver
|
r13827 | import os | ||
Thomas Kluyver
|
r13919 | import zipfile | ||
Thomas Kluyver
|
r13827 | |||
from tornado import web | ||||
Thomas Kluyver
|
r13916 | from ..base.handlers import IPythonHandler, notebook_path_regex | ||
Thomas Kluyver
|
r13827 | from IPython.nbformat.current import to_notebook_json | ||
MinRK
|
r13946 | |||
Thomas Kluyver
|
r13919 | from IPython.utils.py3compat import cast_bytes | ||
def find_resource_files(output_files_dir): | ||||
files = [] | ||||
for dirpath, dirnames, filenames in os.walk(output_files_dir): | ||||
files.extend([os.path.join(dirpath, f) for f in filenames]) | ||||
return files | ||||
def respond_zip(handler, name, output, resources): | ||||
"""Zip up the output and resource files and respond with the zip file. | ||||
Returns True if it has served a zip file, False if there are no resource | ||||
files, in which case we serve the plain output file. | ||||
""" | ||||
# Check if we have resource files we need to zip | ||||
output_files = resources.get('outputs', None) | ||||
if not output_files: | ||||
return False | ||||
# Headers | ||||
zip_filename = os.path.splitext(name)[0] + '.zip' | ||||
handler.set_header('Content-Disposition', | ||||
'attachment; filename="%s"' % zip_filename) | ||||
handler.set_header('Content-Type', 'application/zip') | ||||
# Prepare the zip file | ||||
buffer = io.BytesIO() | ||||
zipf = zipfile.ZipFile(buffer, mode='w', compression=zipfile.ZIP_DEFLATED) | ||||
output_filename = os.path.splitext(name)[0] + '.' + resources['output_extension'] | ||||
zipf.writestr(output_filename, cast_bytes(output, 'utf-8')) | ||||
for filename, data in output_files.items(): | ||||
zipf.writestr(os.path.basename(filename), data) | ||||
zipf.close() | ||||
handler.finish(buffer.getvalue()) | ||||
return True | ||||
Thomas Kluyver
|
r13827 | |||
MinRK
|
r13946 | def get_exporter(format, **kwargs): | ||
"""get an exporter, raising appropriate errors""" | ||||
# if this fails, will raise 500 | ||||
try: | ||||
from IPython.nbconvert.exporters.export import exporter_map | ||||
except ImportError as e: | ||||
raise web.HTTPError(500, "Could not import nbconvert: %s" % e) | ||||
try: | ||||
Exporter = exporter_map[format] | ||||
except KeyError: | ||||
# should this be 400? | ||||
raise web.HTTPError(404, u"No exporter for format: %s" % format) | ||||
MinRK
|
r14044 | try: | ||
return Exporter(**kwargs) | ||||
except Exception as e: | ||||
raise web.HTTPError(500, "Could not construct Exporter: %s" % e) | ||||
MinRK
|
r13946 | |||
Thomas Kluyver
|
r13827 | class NbconvertFileHandler(IPythonHandler): | ||
SUPPORTED_METHODS = ('GET',) | ||||
@web.authenticated | ||||
def get(self, format, path='', name=None): | ||||
MinRK
|
r13946 | |||
MinRK
|
r16266 | exporter = get_exporter(format, config=self.config, log=self.log) | ||
Thomas Kluyver
|
r13827 | |||
path = path.strip('/') | ||||
MinRK
|
r15421 | model = self.notebook_manager.get_notebook(name=name, path=path) | ||
Thomas Kluyver
|
r13827 | |||
MinRK
|
r15421 | self.set_header('Last-Modified', model['last_modified']) | ||
MinRK
|
r13940 | |||
try: | ||||
MinRK
|
r15421 | output, resources = exporter.from_notebook_node(model['content']) | ||
MinRK
|
r13940 | except Exception as e: | ||
raise web.HTTPError(500, "nbconvert failed: %s" % e) | ||||
Thomas Kluyver
|
r13919 | |||
if respond_zip(self, name, output, resources): | ||||
return | ||||
Thomas Kluyver
|
r13830 | # Force download if requested | ||
Thomas Kluyver
|
r13829 | if self.get_argument('download', 'false').lower() == 'true': | ||
Thomas Kluyver
|
r13919 | filename = os.path.splitext(name)[0] + '.' + resources['output_extension'] | ||
Thomas Kluyver
|
r13829 | self.set_header('Content-Disposition', | ||
Thomas Kluyver
|
r13919 | 'attachment; filename="%s"' % filename) | ||
Thomas Kluyver
|
r13830 | # MIME type | ||
Thomas Kluyver
|
r13832 | if exporter.output_mimetype: | ||
self.set_header('Content-Type', | ||||
'%s; charset=utf-8' % exporter.output_mimetype) | ||||
Thomas Kluyver
|
r13830 | |||
Thomas Kluyver
|
r13827 | self.finish(output) | ||
class NbconvertPostHandler(IPythonHandler): | ||||
SUPPORTED_METHODS = ('POST',) | ||||
@web.authenticated | ||||
def post(self, format): | ||||
MinRK
|
r13946 | exporter = get_exporter(format, config=self.config) | ||
Thomas Kluyver
|
r13827 | |||
model = self.get_json_body() | ||||
nbnode = to_notebook_json(model['content']) | ||||
MinRK
|
r13940 | |||
try: | ||||
output, resources = exporter.from_notebook_node(nbnode) | ||||
except Exception as e: | ||||
raise web.HTTPError(500, "nbconvert failed: %s" % e) | ||||
Thomas Kluyver
|
r13919 | |||
if respond_zip(self, nbnode.metadata.name, output, resources): | ||||
return | ||||
Thomas Kluyver
|
r13830 | # MIME type | ||
Thomas Kluyver
|
r13832 | if exporter.output_mimetype: | ||
self.set_header('Content-Type', | ||||
Thomas Kluyver
|
r13919 | '%s; charset=utf-8' % exporter.output_mimetype) | ||
Thomas Kluyver
|
r13830 | |||
Thomas Kluyver
|
r13827 | self.finish(output) | ||
#----------------------------------------------------------------------------- | ||||
# URL to handler mappings | ||||
#----------------------------------------------------------------------------- | ||||
_format_regex = r"(?P<format>\w+)" | ||||
Thomas Kluyver
|
r13916 | |||
Thomas Kluyver
|
r13827 | |||
default_handlers = [ | ||||
Thomas Kluyver
|
r13916 | (r"/nbconvert/%s%s" % (_format_regex, notebook_path_regex), | ||
Thomas Kluyver
|
r13827 | NbconvertFileHandler), | ||
(r"/nbconvert/%s" % _format_regex, NbconvertPostHandler), | ||||
Spencer Nelson
|
r16525 | ] | ||