nbconvertapp.py
311 lines
| 11.2 KiB
| text/x-python
|
PythonLexer
Brian E. Granger
|
r11087 | #!/usr/bin/env python | ||
MinRK
|
r16265 | """NbConvert is a utility for conversion of .ipynb files. | ||
Brian E. Granger
|
r11087 | |||
MinRK
|
r11448 | Command-line interface for the NbConvert conversion utility. | ||
Brian E. Granger
|
r11087 | """ | ||
MinRK
|
r16265 | # Copyright (c) IPython Development Team. | ||
# Distributed under the terms of the Modified BSD License. | ||||
Brian E. Granger
|
r11087 | |||
from __future__ import print_function | ||||
MinRK
|
r11842 | |||
import logging | ||||
Brian E. Granger
|
r11087 | import sys | ||
import os | ||||
Jonathan Frederic
|
r11367 | import glob | ||
Brian E. Granger
|
r11087 | |||
MinRK
|
r11448 | from IPython.core.application import BaseIPythonApplication, base_aliases, base_flags | ||
MinRK
|
r12358 | from IPython.core.profiledir import ProfileDir | ||
MinRK
|
r11453 | from IPython.config import catch_config_error, Configurable | ||
MinRK
|
r11454 | from IPython.utils.traitlets import ( | ||
Unicode, List, Instance, DottedObjectName, Type, CaselessStrEnum, | ||||
) | ||||
Jonathan Frederic
|
r11367 | from IPython.utils.importstring import import_item | ||
Brian E. Granger
|
r11087 | |||
MinRK
|
r11865 | from .exporters.export import get_export_names, exporter_map | ||
Jake Vanderplas
|
r12249 | from IPython.nbconvert import exporters, preprocessors, writers, postprocessors | ||
Jonathan Frederic
|
r11420 | from .utils.base import NbConvertBase | ||
David Wolever
|
r11703 | from .utils.exceptions import ConversionException | ||
Brian E. Granger
|
r11087 | |||
#----------------------------------------------------------------------------- | ||||
Jonathan Frederic
|
r11367 | #Classes and functions | ||
Brian E. Granger
|
r11087 | #----------------------------------------------------------------------------- | ||
Jonathan Frederic
|
r11747 | class DottedOrNone(DottedObjectName): | ||
""" | ||||
A string holding a valid dotted object name in Python, such as A.b3._c | ||||
Also allows for None type.""" | ||||
default_value = u'' | ||||
def validate(self, obj, value): | ||||
if value is not None and len(value) > 0: | ||||
return super(DottedOrNone, self).validate(obj, value) | ||||
else: | ||||
return value | ||||
MinRK
|
r11448 | nbconvert_aliases = {} | ||
nbconvert_aliases.update(base_aliases) | ||||
nbconvert_aliases.update({ | ||||
Jonathan Frederic
|
r11735 | 'to' : 'NbConvertApp.export_format', | ||
Matthias BUSSONNIER
|
r12500 | 'template' : 'TemplateExporter.template_file', | ||
MinRK
|
r11448 | 'writer' : 'NbConvertApp.writer_class', | ||
Jonathan Frederic
|
r12258 | 'post': 'NbConvertApp.postprocessor_class', | ||
damianavila
|
r11894 | 'output': 'NbConvertApp.output_base', | ||
MinRK
|
r12440 | 'reveal-prefix': 'RevealHelpPreprocessor.url_prefix', | ||
MinRK
|
r11448 | }) | ||
nbconvert_flags = {} | ||||
nbconvert_flags.update(base_flags) | ||||
nbconvert_flags.update({ | ||||
'stdout' : ( | ||||
{'NbConvertApp' : {'writer_class' : "StdoutWriter"}}, | ||||
"Write notebook output to stdout instead of files." | ||||
) | ||||
}) | ||||
Jonathan Frederic
|
r11367 | class NbConvertApp(BaseIPythonApplication): | ||
Thomas Kluyver
|
r13597 | """Application used to convert from notebook file type (``*.ipynb``)""" | ||
Brian E. Granger
|
r11087 | |||
Jonathan Frederic
|
r11406 | name = 'ipython-nbconvert' | ||
MinRK
|
r11448 | aliases = nbconvert_aliases | ||
flags = nbconvert_flags | ||||
MinRK
|
r11842 | def _log_level_default(self): | ||
return logging.INFO | ||||
MinRK
|
r11448 | def _classes_default(self): | ||
MinRK
|
r12358 | classes = [NbConvertBase, ProfileDir] | ||
jakobgager
|
r12902 | for pkg in (exporters, preprocessors, writers, postprocessors): | ||
MinRK
|
r11453 | for name in dir(pkg): | ||
cls = getattr(pkg, name) | ||||
if isinstance(cls, type) and issubclass(cls, Configurable): | ||||
classes.append(cls) | ||||
MinRK
|
r12358 | |||
MinRK
|
r11453 | return classes | ||
Paul Ivanov
|
r11248 | |||
Jonathan Frederic
|
r11367 | description = Unicode( | ||
MinRK
|
r11454 | u"""This application is used to convert notebook files (*.ipynb) | ||
Jonathan Frederic
|
r11755 | to various other formats. | ||
WARNING: THE COMMANDLINE INTERFACE MAY CHANGE IN FUTURE RELEASES.""") | ||||
Paul Ivanov
|
r11251 | |||
Matthias BUSSONNIER
|
r11801 | output_base = Unicode('', config=True, help='''overwrite base name use for output files. | ||
can only be use when converting one notebook at a time. | ||||
''') | ||||
Jonathan Frederic
|
r11367 | examples = Unicode(u""" | ||
MinRK
|
r11454 | The simplest way to use nbconvert is | ||
> ipython nbconvert mynotebook.ipynb | ||||
which will convert mynotebook.ipynb to the default format (probably HTML). | ||||
Jonathan Frederic
|
r11741 | You can specify the export format with `--to`. | ||
MinRK
|
r11454 | Options include {0} | ||
dexterdev
|
r16435 | > ipython nbconvert --to latex mynotebook.ipynb | ||
Jonathan Frederic
|
r11741 | |||
Jonathan Frederic
|
r11745 | Both HTML and LaTeX support multiple output templates. LaTeX includes | ||
Jonathan Frederic
|
r11744 | 'basic', 'book', and 'article'. HTML includes 'basic' and 'full'. You | ||
can specify the flavor of the format used. | ||||
Jonathan Frederic
|
r11741 | |||
Jonathan Frederic
|
r11761 | > ipython nbconvert --to html --template basic mynotebook.ipynb | ||
MinRK
|
r11454 | |||
You can also pipe the output to stdout, rather than a file | ||||
> ipython nbconvert mynotebook.ipynb --stdout | ||||
Jonathan Frederic
|
r11742 | |||
MinRK
|
r16265 | PDF is generated via latex | ||
Jonathan Frederic
|
r11742 | |||
MinRK
|
r16265 | > ipython nbconvert mynotebook.ipynb --to pdf | ||
MinRK
|
r11454 | |||
damianavila
|
r11776 | You can get (and serve) a Reveal.js-powered slideshow | ||
> ipython nbconvert myslides.ipynb --to slides --post serve | ||||
Jonathan Frederic
|
r11367 | Multiple notebooks can be given at the command line in a couple of | ||
different ways: | ||||
> ipython nbconvert notebook*.ipynb | ||||
> ipython nbconvert notebook1.ipynb notebook2.ipynb | ||||
MinRK
|
r11469 | |||
or you can specify the notebooks list in a config file, containing:: | ||||
c.NbConvertApp.notebooks = ["my_notebook.ipynb"] | ||||
> ipython nbconvert --config mycfg.py | ||||
MinRK
|
r11454 | """.format(get_export_names())) | ||
Jonathan Frederic
|
r11747 | |||
Jonathan Frederic
|
r11632 | # Writer specific variables | ||
Jonathan Frederic
|
r11367 | writer = Instance('IPython.nbconvert.writers.base.WriterBase', | ||
help="""Instance of the writer class used to write the | ||||
results of the conversion.""") | ||||
writer_class = DottedObjectName('FilesWriter', config=True, | ||||
help="""Writer class used to write the | ||||
results of the conversion""") | ||||
Jonathan Frederic
|
r12054 | writer_aliases = {'fileswriter': 'IPython.nbconvert.writers.files.FilesWriter', | ||
'debugwriter': 'IPython.nbconvert.writers.debug.DebugWriter', | ||||
'stdoutwriter': 'IPython.nbconvert.writers.stdout.StdoutWriter'} | ||||
Jonathan Frederic
|
r11367 | writer_factory = Type() | ||
Paul Ivanov
|
r11251 | |||
Jonathan Frederic
|
r11367 | def _writer_class_changed(self, name, old, new): | ||
Jonathan Frederic
|
r12070 | if new.lower() in self.writer_aliases: | ||
new = self.writer_aliases[new.lower()] | ||||
Jonathan Frederic
|
r11367 | self.writer_factory = import_item(new) | ||
Paul Ivanov
|
r11248 | |||
Jonathan Frederic
|
r11747 | # Post-processor specific variables | ||
Jonathan Frederic
|
r12258 | postprocessor = Instance('IPython.nbconvert.postprocessors.base.PostProcessorBase', | ||
Jonathan Frederic
|
r11747 | help="""Instance of the PostProcessor class used to write the | ||
results of the conversion.""") | ||||
Jonathan Frederic
|
r12258 | postprocessor_class = DottedOrNone(config=True, | ||
Jonathan Frederic
|
r11747 | help="""PostProcessor class used to write the | ||
results of the conversion""") | ||||
MinRK
|
r16418 | postprocessor_aliases = {'serve': 'IPython.nbconvert.postprocessors.serve.ServePostProcessor'} | ||
Jonathan Frederic
|
r12258 | postprocessor_factory = Type() | ||
Jonathan Frederic
|
r11747 | |||
Jonathan Frederic
|
r12258 | def _postprocessor_class_changed(self, name, old, new): | ||
if new.lower() in self.postprocessor_aliases: | ||||
new = self.postprocessor_aliases[new.lower()] | ||||
Jonathan Frederic
|
r11747 | if new: | ||
Jonathan Frederic
|
r12258 | self.postprocessor_factory = import_item(new) | ||
Jonathan Frederic
|
r11747 | |||
Brian E. Granger
|
r11087 | |||
Jonathan Frederic
|
r11632 | # Other configurable variables | ||
MinRK
|
r11454 | export_format = CaselessStrEnum(get_export_names(), | ||
Jonathan Frederic
|
r11735 | default_value="html", | ||
MinRK
|
r11454 | config=True, | ||
help="""The export format to be used.""" | ||||
) | ||||
Brian E. Granger
|
r11087 | |||
Jonathan Frederic
|
r11367 | notebooks = List([], config=True, help="""List of notebooks to convert. | ||
MinRK
|
r11454 | Wildcards are supported. | ||
Filenames passed positionally will be added to the list. | ||||
""") | ||||
Brian E. Granger
|
r11087 | |||
Jonathan Frederic
|
r11367 | @catch_config_error | ||
def initialize(self, argv=None): | ||||
Jonathan Frederic
|
r11652 | self.init_syspath() | ||
jon
|
r16356 | super(NbConvertApp, self).initialize(argv) | ||
MinRK
|
r11448 | self.init_notebooks() | ||
Jonathan Frederic
|
r11367 | self.init_writer() | ||
Jonathan Frederic
|
r12258 | self.init_postprocessor() | ||
Jonathan Frederic
|
r11747 | |||
Brian E. Granger
|
r11087 | |||
Jonathan Frederic
|
r11651 | |||
Jonathan Frederic
|
r11652 | def init_syspath(self): | ||
Jonathan Frederic
|
r11651 | """ | ||
Add the cwd to the sys.path ($PYTHONPATH) | ||||
""" | ||||
Jonathan Frederic
|
r11688 | sys.path.insert(0, os.getcwd()) | ||
Jonathan Frederic
|
r11651 | |||
MinRK
|
r11448 | def init_notebooks(self): | ||
MinRK
|
r11468 | """Construct the list of notebooks. | ||
If notebooks are passed on the command-line, | ||||
they override notebooks specified in config files. | ||||
Glob each notebook to replace notebook patterns with filenames. | ||||
Jonathan Frederic
|
r11367 | """ | ||
Brian E. Granger
|
r11087 | |||
MinRK
|
r11468 | # Specifying notebooks on the command-line overrides (rather than adds) | ||
# the notebook list | ||||
if self.extra_args: | ||||
patterns = self.extra_args | ||||
else: | ||||
patterns = self.notebooks | ||||
Jonathan Frederic
|
r11367 | |||
Jonathan Frederic
|
r11632 | # Use glob to replace all the notebook patterns with filenames. | ||
Jonathan Frederic
|
r11367 | filenames = [] | ||
MinRK
|
r11448 | for pattern in patterns: | ||
Jonathan Frederic
|
r11660 | |||
# Use glob to find matching filenames. Allow the user to convert | ||||
# notebooks without having to type the extension. | ||||
globbed_files = glob.glob(pattern) | ||||
globbed_files.extend(glob.glob(pattern + '.ipynb')) | ||||
MinRK
|
r11851 | if not globbed_files: | ||
self.log.warn("pattern %r matched no files", pattern) | ||||
Jonathan Frederic
|
r11660 | |||
for filename in globbed_files: | ||||
Jonathan Frederic
|
r11367 | if not filename in filenames: | ||
filenames.append(filename) | ||||
self.notebooks = filenames | ||||
def init_writer(self): | ||||
""" | ||||
Initialize the writer (which is stateless) | ||||
""" | ||||
self._writer_class_changed(None, self.writer_class, self.writer_class) | ||||
self.writer = self.writer_factory(parent=self) | ||||
Brian E. Granger
|
r11087 | |||
Jonathan Frederic
|
r12258 | def init_postprocessor(self): | ||
Jonathan Frederic
|
r11747 | """ | ||
Jonathan Frederic
|
r12258 | Initialize the postprocessor (which is stateless) | ||
Jonathan Frederic
|
r11747 | """ | ||
Jonathan Frederic
|
r12258 | self._postprocessor_class_changed(None, self.postprocessor_class, | ||
self.postprocessor_class) | ||||
if self.postprocessor_factory: | ||||
self.postprocessor = self.postprocessor_factory(parent=self) | ||||
Jonathan Frederic
|
r11747 | |||
MinRK
|
r11448 | def start(self): | ||
Brian E. Granger
|
r11087 | """ | ||
MinRK
|
r11448 | Ran after initialization completed | ||
Jonathan Frederic
|
r11367 | """ | ||
Brian E. Granger
|
r11087 | super(NbConvertApp, self).start() | ||
Jonathan Frederic
|
r11400 | self.convert_notebooks() | ||
def convert_notebooks(self): | ||||
""" | ||||
Convert the notebooks in the self.notebook traitlet | ||||
""" | ||||
Jonathan Frederic
|
r11632 | # Export each notebook | ||
Jonathan Frederic
|
r11400 | conversion_success = 0 | ||
Matthias BUSSONNIER
|
r11801 | |||
if self.output_base != '' and len(self.notebooks) > 1: | ||||
MinRK
|
r11865 | self.log.error( | ||
Matthias BUSSONNIER
|
r11801 | """UsageError: --output flag or `NbConvertApp.output_base` config option | ||
cannot be used when converting multiple notebooks. | ||||
MinRK
|
r11865 | """) | ||
Matthias BUSSONNIER
|
r11801 | self.exit(1) | ||
MinRK
|
r11865 | |||
exporter = exporter_map[self.export_format](config=self.config) | ||||
Matthias BUSSONNIER
|
r11801 | |||
Jonathan Frederic
|
r11367 | for notebook_filename in self.notebooks: | ||
MinRK
|
r11842 | self.log.info("Converting notebook %s to %s", notebook_filename, self.export_format) | ||
Jonathan Frederic
|
r11367 | |||
Jonathan Frederic
|
r11632 | # Get a unique key for the notebook and set it in the resources object. | ||
Jonathan Frederic
|
r11367 | basename = os.path.basename(notebook_filename) | ||
notebook_name = basename[:basename.rfind('.')] | ||||
Matthias BUSSONNIER
|
r11801 | if self.output_base: | ||
notebook_name = self.output_base | ||||
Jonathan Frederic
|
r11367 | resources = {} | ||
resources['unique_key'] = notebook_name | ||||
Jonathan Frederic
|
r11631 | resources['output_files_dir'] = '%s_files' % notebook_name | ||
MinRK
|
r11847 | self.log.info("Support files will be in %s", os.path.join(resources['output_files_dir'], '')) | ||
Jonathan Frederic
|
r11367 | |||
Jonathan Frederic
|
r11632 | # Try to export | ||
Jonathan Frederic
|
r11368 | try: | ||
MinRK
|
r11865 | output, resources = exporter.from_filename(notebook_filename, resources=resources) | ||
David Wolever
|
r11703 | except ConversionException as e: | ||
MinRK
|
r11865 | self.log.error("Error while converting '%s'", notebook_filename, | ||
exc_info=True) | ||||
David Wolever
|
r11705 | self.exit(1) | ||
Jonathan Frederic
|
r11400 | else: | ||
Jonathan Frederic
|
r11747 | write_resultes = self.writer.write(output, resources, notebook_name=notebook_name) | ||
#Post-process if post processor has been defined. | ||||
Jonathan Frederic
|
r12258 | if hasattr(self, 'postprocessor') and self.postprocessor: | ||
self.postprocessor(write_resultes) | ||||
Jonathan Frederic
|
r11400 | conversion_success += 1 | ||
Jonathan Frederic
|
r11632 | # If nothing was converted successfully, help the user. | ||
Jonathan Frederic
|
r11400 | if conversion_success == 0: | ||
Jonathan Frederic
|
r11666 | self.print_help() | ||
Jonathan Frederic
|
r11667 | sys.exit(-1) | ||
damianavila
|
r11772 | |||
Brian E. Granger
|
r11087 | #----------------------------------------------------------------------------- | ||
Brian E. Granger
|
r11092 | # Main entry point | ||
Brian E. Granger
|
r11087 | #----------------------------------------------------------------------------- | ||
MinRK
|
r11176 | launch_new_instance = NbConvertApp.launch_instance | ||