notebookapp.py
862 lines
| 33.0 KiB
| text/x-python
|
PythonLexer
MinRK
|
r6067 | # coding: utf-8 | ||
MinRK
|
r16374 | """A tornado based IPython notebook server.""" | ||
Brian E. Granger
|
r4609 | |||
MinRK
|
r16374 | # Copyright (c) IPython Development Team. | ||
# Distributed under the terms of the Modified BSD License. | ||||
Brian E. Granger
|
r4609 | |||
Thomas Kluyver
|
r13348 | from __future__ import print_function | ||
MinRK
|
r16375 | |||
Brian E. Granger
|
r4548 | import errno | ||
Thomas Kluyver
|
r14063 | import io | ||
import json | ||||
Brian E. Granger
|
r4339 | import logging | ||
import os | ||||
Bradley M. Froehle
|
r7234 | import random | ||
MinRK
|
r6508 | import select | ||
Brian E. Granger
|
r4345 | import signal | ||
Brian E. Granger
|
r4548 | import socket | ||
Brian E. Granger
|
r4345 | import sys | ||
Fernando Perez
|
r5212 | import threading | ||
MinRK
|
r6509 | import time | ||
Thomas Kluyver
|
r5065 | import webbrowser | ||
Brian E. Granger
|
r4339 | |||
MinRK
|
r10199 | |||
MinRK
|
r10201 | # check for pyzmq 2.1.11 | ||
from IPython.utils.zmqrelated import check_for_zmq | ||||
MinRK
|
r11035 | check_for_zmq('2.1.11', 'IPython.html') | ||
MinRK
|
r10199 | |||
Cameron Bates
|
r8840 | from jinja2 import Environment, FileSystemLoader | ||
Brian E. Granger
|
r4339 | |||
# Install the pyzmq ioloop. This has to be done before anything else from | ||||
# tornado is imported. | ||||
from zmq.eventloop import ioloop | ||||
MinRK
|
r6631 | ioloop.install() | ||
Brian E. Granger
|
r4339 | |||
MinRK
|
r13313 | # check for tornado 3.1.0 | ||
msg = "The IPython Notebook requires tornado >= 3.1.0" | ||||
MinRK
|
r10199 | try: | ||
import tornado | ||||
except ImportError: | ||||
raise ImportError(msg) | ||||
try: | ||||
version_info = tornado.version_info | ||||
except AttributeError: | ||||
raise ImportError(msg + ", but you have < 1.1.0") | ||||
MinRK
|
r13313 | if version_info < (3,1,0): | ||
MinRK
|
r10199 | raise ImportError(msg + ", but you have %s" % tornado.version) | ||
Brian E. Granger
|
r4339 | from tornado import httpserver | ||
from tornado import web | ||||
MinRK
|
r16358 | from tornado.log import LogFormatter | ||
Brian E. Granger
|
r4339 | |||
MinRK
|
r11035 | from IPython.html import DEFAULT_STATIC_FILES_PATH | ||
MinRK
|
r13939 | from .base.handlers import Template404 | ||
MinRK
|
r14645 | from .log import log_request | ||
Brian E. Granger
|
r10666 | from .services.kernels.kernelmanager import MappingKernelManager | ||
from .services.notebooks.nbmanager import NotebookManager | ||||
from .services.notebooks.filenbmanager import FileNotebookManager | ||||
from .services.clusters.clustermanager import ClusterManager | ||||
Zachary Sailer
|
r12982 | from .services.sessions.sessionmanager import SessionManager | ||
Brian E. Granger
|
r10642 | |||
Brian E. Granger
|
r10650 | from .base.handlers import AuthenticatedFileHandler, FileFindHandler | ||
Brian E. Granger
|
r4339 | |||
Thomas Kluyver
|
r16041 | from IPython.config import Config | ||
Fernando Perez
|
r5762 | from IPython.config.application import catch_config_error, boolean_flag | ||
MinRK
|
r16375 | from IPython.core.application import ( | ||
BaseIPythonApplication, base_flags, base_aliases, | ||||
Brian E. Granger
|
r4345 | ) | ||
MinRK
|
r16375 | from IPython.core.profiledir import ProfileDir | ||
from IPython.kernel import KernelManager | ||||
from IPython.kernel.zmq.session import default_secure, Session | ||||
MinRK
|
r15791 | from IPython.nbformat.sign import NotebookNotary | ||
Brian Granger
|
r8180 | from IPython.utils.importstring import import_item | ||
MinRK
|
r10557 | from IPython.utils import submodule | ||
Brian Granger
|
r8180 | from IPython.utils.traitlets import ( | ||
MinRK
|
r10784 | Dict, Unicode, Integer, List, Bool, Bytes, | ||
MinRK
|
r15420 | DottedObjectName, TraitError, | ||
Brian Granger
|
r8180 | ) | ||
MinRK
|
r6067 | from IPython.utils import py3compat | ||
MinRK
|
r12800 | from IPython.utils.path import filefind, get_ipython_dir | ||
Brian E. Granger
|
r4344 | |||
Brian E. Granger
|
r10642 | from .utils import url_path_join | ||
Brian E. Granger
|
r4344 | #----------------------------------------------------------------------------- | ||
# Module globals | ||||
#----------------------------------------------------------------------------- | ||||
Brian E. Granger
|
r4339 | |||
Brian E. Granger
|
r4519 | _examples = """ | ||
ipython notebook # start the notebook | ||||
ipython notebook --profile=sympy # use the sympy profile | ||||
ipython notebook --certfile=mycert.pem # use SSL/TLS certificate | ||||
""" | ||||
Brian E. Granger
|
r4344 | #----------------------------------------------------------------------------- | ||
Andrew Straw
|
r6004 | # Helper functions | ||
#----------------------------------------------------------------------------- | ||||
Bradley M. Froehle
|
r7234 | def random_ports(port, n): | ||
"""Generate a list of n random ports near the given port. | ||||
The first 5 ports will be sequential, and the remaining n-5 will be | ||||
randomly selected in the range [port-2*n, port+2*n]. | ||||
""" | ||||
for i in range(min(5, n)): | ||||
yield port + i | ||||
for i in range(n-5): | ||||
MinRK
|
r12843 | yield max(1, port + random.randint(-2*n, 2*n)) | ||
Bradley M. Froehle
|
r7234 | |||
Brian E. Granger
|
r10647 | def load_handlers(name): | ||
"""Load the (URL pattern, handler) tuples for each component.""" | ||||
MinRK
|
r11035 | name = 'IPython.html.' + name | ||
Brian E. Granger
|
r10647 | mod = __import__(name, fromlist=['default_handlers']) | ||
return mod.default_handlers | ||||
Andrew Straw
|
r6004 | #----------------------------------------------------------------------------- | ||
Brian E. Granger
|
r4344 | # The Tornado web application | ||
#----------------------------------------------------------------------------- | ||||
Brian E. Granger
|
r4339 | |||
Brian E. Granger
|
r4342 | class NotebookWebApplication(web.Application): | ||
Brian E. Granger
|
r4339 | |||
Matthias BUSSONNIER
|
r9284 | def __init__(self, ipython_app, kernel_manager, notebook_manager, | ||
MinRK
|
r15238 | cluster_manager, session_manager, log, base_url, | ||
Matthias BUSSONNIER
|
r15449 | settings_overrides, jinja_env_options): | ||
Brian E. Granger
|
r10647 | |||
Brian E. Granger
|
r10656 | settings = self.init_settings( | ||
ipython_app, kernel_manager, notebook_manager, cluster_manager, | ||||
Matthias BUSSONNIER
|
r15449 | session_manager, log, base_url, settings_overrides, jinja_env_options) | ||
Brian E. Granger
|
r10656 | handlers = self.init_handlers(settings) | ||
super(NotebookWebApplication, self).__init__(handlers, **settings) | ||||
Timo Paulssen
|
r5664 | |||
Brian E. Granger
|
r10656 | def init_settings(self, ipython_app, kernel_manager, notebook_manager, | ||
MinRK
|
r15238 | cluster_manager, session_manager, log, base_url, | ||
Matthias BUSSONNIER
|
r15449 | settings_overrides, jinja_env_options=None): | ||
MinRK
|
r6067 | # Python < 2.6.5 doesn't accept unicode keys in f(**kwargs), and | ||
MinRK
|
r15238 | # base_url will always be unicode, which will in turn | ||
MinRK
|
r6067 | # make the patterns unicode, and ultimately result in unicode | ||
# keys in kwargs to handler._execute(**kwargs) in tornado. | ||||
MinRK
|
r15238 | # This enforces that base_url be ascii in that situation. | ||
MinRK
|
r6067 | # | ||
# Note that the URLs these patterns check against are escaped, | ||||
# and thus guaranteed to be ASCII: 'héllo' is really 'h%C3%A9llo'. | ||||
MinRK
|
r15238 | base_url = py3compat.unicode_to_str(base_url, 'ascii') | ||
Matt Henderson
|
r12194 | template_path = settings_overrides.get("template_path", os.path.join(os.path.dirname(__file__), "templates")) | ||
Matthias BUSSONNIER
|
r15449 | jenv_opt = jinja_env_options if jinja_env_options else {} | ||
env = Environment(loader=FileSystemLoader(template_path),**jenv_opt ) | ||||
Matthias BUSSONNIER
|
r7797 | settings = dict( | ||
MinRK
|
r10355 | # basics | ||
MinRK
|
r14645 | log_function=log_request, | ||
MinRK
|
r15238 | base_url=base_url, | ||
MinRK
|
r10355 | template_path=template_path, | ||
MinRK
|
r7930 | static_path=ipython_app.static_file_path, | ||
MinRK
|
r7923 | static_handler_class = FileFindHandler, | ||
MinRK
|
r15238 | static_url_prefix = url_path_join(base_url,'/static/'), | ||
MinRK
|
r10355 | |||
# authentication | ||||
MinRK
|
r10784 | cookie_secret=ipython_app.cookie_secret, | ||
MinRK
|
r15238 | login_url=url_path_join(base_url,'/login'), | ||
MinRK
|
r10355 | password=ipython_app.password, | ||
# managers | ||||
kernel_manager=kernel_manager, | ||||
notebook_manager=notebook_manager, | ||||
cluster_manager=cluster_manager, | ||||
Zachary Sailer
|
r12982 | session_manager=session_manager, | ||
MinRK
|
r10355 | # IPython stuff | ||
MinRK
|
r12811 | nbextensions_path = ipython_app.nbextensions_path, | ||
MinRK
|
r10388 | mathjax_url=ipython_app.mathjax_url, | ||
MinRK
|
r10355 | config=ipython_app.config, | ||
Matthias BUSSONNIER
|
r15449 | jinja2_env=env, | ||
Matthias BUSSONNIER
|
r7797 | ) | ||
# allow custom overrides for the tornado web app. | ||||
settings.update(settings_overrides) | ||||
Brian E. Granger
|
r10656 | return settings | ||
Matthias BUSSONNIER
|
r7797 | |||
Brian E. Granger
|
r10656 | def init_handlers(self, settings): | ||
# Load the (URL pattern, handler) tuples for each component. | ||||
handlers = [] | ||||
handlers.extend(load_handlers('base.handlers')) | ||||
handlers.extend(load_handlers('tree.handlers')) | ||||
handlers.extend(load_handlers('auth.login')) | ||||
handlers.extend(load_handlers('auth.logout')) | ||||
Brian E. Granger
|
r10665 | handlers.extend(load_handlers('notebook.handlers')) | ||
Thomas Kluyver
|
r13827 | handlers.extend(load_handlers('nbconvert.handlers')) | ||
Brian E. Granger
|
r10665 | handlers.extend(load_handlers('services.kernels.handlers')) | ||
handlers.extend(load_handlers('services.notebooks.handlers')) | ||||
handlers.extend(load_handlers('services.clusters.handlers')) | ||||
Zachary Sailer
|
r12982 | handlers.extend(load_handlers('services.sessions.handlers')) | ||
Thomas Kluyver
|
r13837 | handlers.extend(load_handlers('services.nbconvert.handlers')) | ||
MinRK
|
r15420 | # FIXME: /files/ should be handled by the Contents service when it exists | ||
nbm = settings['notebook_manager'] | ||||
if hasattr(nbm, 'notebook_dir'): | ||||
handlers.extend([ | ||||
(r"/files/(.*)", AuthenticatedFileHandler, {'path' : nbm.notebook_dir}), | ||||
MinRK
|
r12811 | (r"/nbextensions/(.*)", FileFindHandler, {'path' : settings['nbextensions_path']}), | ||
Brian E. Granger
|
r10656 | ]) | ||
MinRK
|
r15238 | # prepend base_url onto the patterns that we match | ||
Andrew Straw
|
r6004 | new_handlers = [] | ||
for handler in handlers: | ||||
MinRK
|
r15238 | pattern = url_path_join(settings['base_url'], handler[0]) | ||
MinRK
|
r10355 | new_handler = tuple([pattern] + list(handler[1:])) | ||
new_handlers.append(new_handler) | ||||
MinRK
|
r13939 | # add 404 on the end, which will catch everything that falls through | ||
new_handlers.append((r'(.*)', Template404)) | ||||
Brian E. Granger
|
r10656 | return new_handlers | ||
Brian E. Granger
|
r4339 | |||
Thomas Kluyver
|
r14177 | class NbserverListApp(BaseIPythonApplication): | ||
description="List currently running notebook servers in this profile." | ||||
flags = dict( | ||||
json=({'NbserverListApp': {'json': True}}, | ||||
"Produce machine-readable JSON output."), | ||||
) | ||||
json = Bool(False, config=True, | ||||
help="If True, each line of output will be a JSON object with the " | ||||
"details from the server info file.") | ||||
def start(self): | ||||
if not self.json: | ||||
print("Currently running servers:") | ||||
Thomas Kluyver
|
r14179 | for serverinfo in list_running_servers(self.profile): | ||
Thomas Kluyver
|
r14177 | if self.json: | ||
print(json.dumps(serverinfo)) | ||||
else: | ||||
Thomas Kluyver
|
r14179 | print(serverinfo['url'], "::", serverinfo['notebook_dir']) | ||
Brian E. Granger
|
r4346 | |||
Brian E. Granger
|
r4344 | #----------------------------------------------------------------------------- | ||
# Aliases and Flags | ||||
#----------------------------------------------------------------------------- | ||||
MinRK
|
r16375 | flags = dict(base_flags) | ||
Thomas Kluyver
|
r5065 | flags['no-browser']=( | ||
MinRK
|
r5128 | {'NotebookApp' : {'open_browser' : False}}, | ||
Thomas Kluyver
|
r5065 | "Don't open the notebook in a browser after startup." | ||
) | ||||
MinRK
|
r16375 | flags['pylab']=( | ||
{'NotebookApp' : {'pylab' : 'warn'}}, | ||||
"DISABLED: use %pylab or %matplotlib in the notebook to enable matplotlib." | ||||
) | ||||
MinRK
|
r5547 | flags['no-mathjax']=( | ||
{'NotebookApp' : {'enable_mathjax' : False}}, | ||||
"""Disable MathJax | ||||
MathJax is the javascript library IPython uses to render math/LaTeX. It is | ||||
very large, so you may want to disable it if you have a slow internet | ||||
connection, or for offline use of the notebook. | ||||
When disabled, equations etc. will appear as their untransformed TeX source. | ||||
""" | ||||
) | ||||
Brian E. Granger
|
r4344 | |||
Fernando Perez
|
r5762 | # Add notebook manager flags | ||
Brian Granger
|
r8183 | flags.update(boolean_flag('script', 'FileNotebookManager.save_script', | ||
Fernando Perez
|
r5762 | 'Auto-save a .py script everytime the .ipynb notebook is saved', | ||
'Do not auto-save .py scripts for every notebook')) | ||||
Fernando Perez
|
r5758 | |||
MinRK
|
r16375 | aliases = dict(base_aliases) | ||
Brian E. Granger
|
r4344 | |||
Brian E. Granger
|
r4519 | aliases.update({ | ||
Brian E. Granger
|
r5104 | 'ip': 'NotebookApp.ip', | ||
'port': 'NotebookApp.port', | ||||
Bradley M. Froehle
|
r7234 | 'port-retries': 'NotebookApp.port_retries', | ||
MinRK
|
r9177 | 'transport': 'KernelManager.transport', | ||
Brian E. Granger
|
r5104 | 'keyfile': 'NotebookApp.keyfile', | ||
'certfile': 'NotebookApp.certfile', | ||||
MinRK
|
r15420 | 'notebook-dir': 'NotebookApp.notebook_dir', | ||
Paul Ivanov
|
r6116 | 'browser': 'NotebookApp.browser', | ||
MinRK
|
r16375 | 'pylab': 'NotebookApp.pylab', | ||
Brian E. Granger
|
r4519 | }) | ||
Brian E. Granger
|
r4344 | |||
#----------------------------------------------------------------------------- | ||||
Brian E. Granger
|
r5104 | # NotebookApp | ||
Brian E. Granger
|
r4344 | #----------------------------------------------------------------------------- | ||
Brian E. Granger
|
r5104 | class NotebookApp(BaseIPythonApplication): | ||
Brian E. Granger
|
r4519 | |||
Brian E. Granger
|
r4344 | name = 'ipython-notebook' | ||
description = """ | ||||
The IPython HTML Notebook. | ||||
This launches a Tornado based HTML Notebook Server that serves up an | ||||
HTML5/Javascript Notebook client. | ||||
""" | ||||
Brian E. Granger
|
r4519 | examples = _examples | ||
MinRK
|
r16375 | aliases = aliases | ||
flags = flags | ||||
Brian E. Granger
|
r4344 | |||
MinRK
|
r16375 | classes = [ | ||
KernelManager, ProfileDir, Session, MappingKernelManager, | ||||
NotebookManager, FileNotebookManager, NotebookNotary, | ||||
] | ||||
Brian E. Granger
|
r4344 | flags = Dict(flags) | ||
aliases = Dict(aliases) | ||||
Thomas Kluyver
|
r14177 | |||
subcommands = dict( | ||||
list=(NbserverListApp, NbserverListApp.description.splitlines()[0]), | ||||
) | ||||
Brian E. Granger
|
r4344 | |||
kernel_argv = List(Unicode) | ||||
MinRK
|
r16358 | |||
_log_formatter_cls = LogFormatter | ||||
Brian E. Granger
|
r4344 | |||
MinRK
|
r10355 | def _log_level_default(self): | ||
return logging.INFO | ||||
Brian E. Granger
|
r4345 | |||
MinRK
|
r16358 | def _log_datefmt_default(self): | ||
"""Exclude date from default date format""" | ||||
return "%H:%M:%S" | ||||
MinRK
|
r10358 | def _log_format_default(self): | ||
"""override default log format to include time""" | ||||
MinRK
|
r16358 | return u"%(color)s[%(levelname)1.1s %(asctime)s.%(msecs).03d %(name)s]%(end_color)s %(message)s" | ||
MinRK
|
r10358 | |||
Paul Ivanov
|
r5847 | # create requested profiles by default, if they don't exist: | ||
auto_create = Bool(True) | ||||
Puneeth Chaganti
|
r6724 | # file to be opened in the notebook server | ||
MinRK
|
r15461 | file_to_run = Unicode('', config=True) | ||
MinRK
|
r15460 | def _file_to_run_changed(self, name, old, new): | ||
path, base = os.path.split(new) | ||||
if path: | ||||
self.file_to_run = base | ||||
MinRK
|
r15461 | self.notebook_dir = path | ||
Puneeth Chaganti
|
r6724 | |||
Brian E. Granger
|
r4519 | # Network related information. | ||
MinRK
|
r16334 | ip = Unicode('localhost', config=True, | ||
Brian E. Granger
|
r4344 | help="The IP address the notebook server will listen on." | ||
) | ||||
Brian E. Granger
|
r4519 | def _ip_changed(self, name, old, new): | ||
if new == u'*': self.ip = u'' | ||||
MinRK
|
r5344 | port = Integer(8888, config=True, | ||
Brian E. Granger
|
r4344 | help="The port the notebook server will listen on." | ||
) | ||||
Bradley M. Froehle
|
r7234 | port_retries = Integer(50, config=True, | ||
help="The number of additional ports to try if the specified port is not available." | ||||
) | ||||
Brian E. Granger
|
r4344 | |||
Brian E. Granger
|
r4519 | certfile = Unicode(u'', config=True, | ||
help="""The full path to an SSL/TLS certificate file.""" | ||||
) | ||||
keyfile = Unicode(u'', config=True, | ||||
help="""The full path to a private key file for usage with SSL/TLS.""" | ||||
) | ||||
MinRK
|
r10784 | |||
cookie_secret = Bytes(b'', config=True, | ||||
help="""The random bytes used to secure cookies. | ||||
By default this is a new random number every time you start the Notebook. | ||||
Set it to a value in a config file to enable logins to persist across server sessions. | ||||
MinRK
|
r10863 | |||
Note: Cookie secrets should be kept private, do not share config files with | ||||
cookie_secret stored in plaintext (you can read the value from a file). | ||||
MinRK
|
r10784 | """ | ||
) | ||||
def _cookie_secret_default(self): | ||||
return os.urandom(1024) | ||||
Brian E. Granger
|
r4344 | |||
MinRK
|
r4705 | password = Unicode(u'', config=True, | ||
Stefan van der Walt
|
r5321 | help="""Hashed password to use for web authentication. | ||
Fernando Perez
|
r5337 | To generate, type in a python/IPython shell: | ||
Stefan van der Walt
|
r5321 | |||
Fernando Perez
|
r5337 | from IPython.lib import passwd; passwd() | ||
Stefan van der Walt
|
r5321 | |||
The string should be of the form type:salt:hashed-password. | ||||
""" | ||||
Satrajit Ghosh
|
r4690 | ) | ||
Paul Ivanov
|
r6116 | |||
Thomas Kluyver
|
r5065 | open_browser = Bool(True, config=True, | ||
Paul Ivanov
|
r6115 | help="""Whether to open in a browser after starting. | ||
The specific browser used is platform dependent and | ||||
determined by the python standard library `webbrowser` | ||||
Paul Ivanov
|
r6116 | module, unless it is overridden using the --browser | ||
(NotebookApp.browser) configuration option. | ||||
Paul Ivanov
|
r6115 | """) | ||
Paul Ivanov
|
r6116 | |||
browser = Unicode(u'', config=True, | ||||
Paul Ivanov
|
r6117 | help="""Specify what command to use to invoke a web | ||
browser when opening the notebook. If not specified, the | ||||
default browser will be determined by the `webbrowser` | ||||
standard library module, which allows setting of the | ||||
BROWSER environment variable to override it. | ||||
Paul Ivanov
|
r6116 | """) | ||
MinRK
|
r5191 | |||
Timo Paulssen
|
r5665 | webapp_settings = Dict(config=True, | ||
help="Supply overrides for the tornado.web.Application that the " | ||||
"IPython notebook uses.") | ||||
Matthias BUSSONNIER
|
r15449 | |||
jinja_environment_options = Dict(config=True, | ||||
help="Supply extra arguments that will be passed to Jinja environment.") | ||||
Timo Paulssen
|
r5665 | |||
MinRK
|
r5547 | enable_mathjax = Bool(True, config=True, | ||
help="""Whether to enable MathJax for typesetting math/TeX | ||||
MathJax is the javascript library IPython uses to render math/LaTeX. It is | ||||
very large, so you may want to disable it if you have a slow internet | ||||
connection, or for offline use of the notebook. | ||||
When disabled, equations etc. will appear as their untransformed TeX source. | ||||
""" | ||||
) | ||||
MinRK
|
r5557 | def _enable_mathjax_changed(self, name, old, new): | ||
"""set mathjax url to empty if mathjax is disabled""" | ||||
if not new: | ||||
self.mathjax_url = u'' | ||||
Andrew Straw
|
r6006 | |||
MinRK
|
r15238 | base_url = Unicode('/', config=True, | ||
Bussonnier Matthias
|
r8938 | help='''The base URL for the notebook server. | ||
Leading and trailing slashes can be omitted, | ||||
and will automatically be added. | ||||
''') | ||||
MinRK
|
r15238 | def _base_url_changed(self, name, old, new): | ||
Bussonnier Matthias
|
r8938 | if not new.startswith('/'): | ||
MinRK
|
r15238 | self.base_url = '/'+new | ||
Bussonnier Matthias
|
r8938 | elif not new.endswith('/'): | ||
MinRK
|
r15238 | self.base_url = new+'/' | ||
base_project_url = Unicode('/', config=True, help="""DEPRECATED use base_url""") | ||||
def _base_project_url_changed(self, name, old, new): | ||||
self.log.warn("base_project_url is deprecated, use base_url") | ||||
self.base_url = new | ||||
Bussonnier Matthias
|
r8938 | |||
MinRK
|
r7923 | extra_static_paths = List(Unicode, config=True, | ||
help="""Extra paths to search for serving static files. | ||||
This allows adding javascript/css to be available from the notebook server machine, | ||||
or overriding individual files in the IPython""" | ||||
) | ||||
def _extra_static_paths_default(self): | ||||
return [os.path.join(self.profile_dir.location, 'static')] | ||||
@property | ||||
def static_file_path(self): | ||||
"""return extra paths + the default location""" | ||||
Thomas Kluyver
|
r9681 | return self.extra_static_paths + [DEFAULT_STATIC_FILES_PATH] | ||
MinRK
|
r12800 | |||
MinRK
|
r12811 | nbextensions_path = List(Unicode, config=True, | ||
help="""paths for Javascript extensions. By default, this is just IPYTHONDIR/nbextensions""" | ||||
MinRK
|
r12800 | ) | ||
MinRK
|
r12811 | def _nbextensions_path_default(self): | ||
return [os.path.join(get_ipython_dir(), 'nbextensions')] | ||||
Andrew Straw
|
r6004 | |||
MinRK
|
r5557 | mathjax_url = Unicode("", config=True, | ||
help="""The url for MathJax.js.""" | ||||
) | ||||
def _mathjax_url_default(self): | ||||
if not self.enable_mathjax: | ||||
return u'' | ||||
Andrew Straw
|
r6002 | static_url_prefix = self.webapp_settings.get("static_url_prefix", | ||
MinRK
|
r15238 | url_path_join(self.base_url, "static") | ||
MinRK
|
r11153 | ) | ||
MinRK
|
r12810 | |||
MinRK
|
r12811 | # try local mathjax, either in nbextensions/mathjax or static/mathjax | ||
MinRK
|
r12810 | for (url_prefix, search_path) in [ | ||
MinRK
|
r15238 | (url_path_join(self.base_url, "nbextensions"), self.nbextensions_path), | ||
MinRK
|
r12810 | (static_url_prefix, self.static_file_path), | ||
]: | ||||
self.log.debug("searching for local mathjax in %s", search_path) | ||||
try: | ||||
mathjax = filefind(os.path.join('mathjax', 'MathJax.js'), search_path) | ||||
except IOError: | ||||
continue | ||||
MinRK
|
r6155 | else: | ||
MinRK
|
r12810 | url = url_path_join(url_prefix, u"mathjax/MathJax.js") | ||
self.log.info("Serving local MathJax from %s at %s", mathjax, url) | ||||
return url | ||||
# no local mathjax, serve from CDN | ||||
if self.certfile: | ||||
# HTTPS: load from Rackspace CDN, because SSL certificate requires it | ||||
host = u"https://c328740.ssl.cf1.rackcdn.com" | ||||
MinRK
|
r7923 | else: | ||
MinRK
|
r12810 | host = u"http://cdn.mathjax.org" | ||
url = host + u"/mathjax/latest/MathJax.js" | ||||
self.log.info("Using MathJax from CDN: %s", url) | ||||
return url | ||||
MinRK
|
r5557 | |||
def _mathjax_url_changed(self, name, old, new): | ||||
if new and not self.enable_mathjax: | ||||
# enable_mathjax=False overrides mathjax_url | ||||
self.mathjax_url = u'' | ||||
else: | ||||
self.log.info("Using MathJax: %s", new) | ||||
Satrajit Ghosh
|
r4690 | |||
MinRK
|
r11035 | notebook_manager_class = DottedObjectName('IPython.html.services.notebooks.filenbmanager.FileNotebookManager', | ||
Brian Granger
|
r8180 | config=True, | ||
MinRK
|
r16423 | help='The notebook manager class to use.' | ||
) | ||||
kernel_manager_class = DottedObjectName('IPython.html.services.kernels.kernelmanager.MappingKernelManager', | ||||
config=True, | ||||
help='The kernel manager class to use.' | ||||
) | ||||
session_manager_class = DottedObjectName('IPython.html.services.sessions.sessionmanager.SessionManager', | ||||
config=True, | ||||
help='The session manager class to use.' | ||||
) | ||||
cluster_manager_class = DottedObjectName('IPython.html.services.clusters.clustermanager.ClusterManager', | ||||
config=True, | ||||
help='The cluster manager class to use.' | ||||
) | ||||
Brian Granger
|
r8180 | |||
Alberto Valverde
|
r10163 | trust_xheaders = Bool(False, config=True, | ||
help=("Whether to trust or not X-Scheme/X-Forwarded-Proto and X-Real-Ip/X-Forwarded-For headers" | ||||
MinRK
|
r14108 | "sent by the upstream reverse proxy. Necessary if the proxy handles SSL") | ||
Alberto Valverde
|
r10163 | ) | ||
MinRK
|
r12284 | |||
Thomas Kluyver
|
r14063 | info_file = Unicode() | ||
def _info_file_default(self): | ||||
info_file = "nbserver-%s.json"%os.getpid() | ||||
return os.path.join(self.profile_dir.security_dir, info_file) | ||||
MinRK
|
r15420 | |||
notebook_dir = Unicode(py3compat.getcwd(), config=True, | ||||
help="The directory to use for notebooks and kernels." | ||||
) | ||||
MinRK
|
r16375 | |||
pylab = Unicode('disabled', config=True, | ||||
help=""" | ||||
DISABLED: use %pylab or %matplotlib in the notebook to enable matplotlib. | ||||
""" | ||||
) | ||||
def _pylab_changed(self, name, old, new): | ||||
MinRK
|
r16380 | """when --pylab is specified, display a warning and exit""" | ||
MinRK
|
r16375 | if new != 'warn': | ||
backend = ' %s' % new | ||||
else: | ||||
backend = '' | ||||
MinRK
|
r16380 | self.log.error("Support for specifying --pylab on the command line has been removed.") | ||
self.log.error( | ||||
MinRK
|
r16375 | "Please use `%pylab{0}` or `%matplotlib{0}` in the notebook itself.".format(backend) | ||
) | ||||
MinRK
|
r16380 | self.exit(1) | ||
MinRK
|
r15420 | |||
def _notebook_dir_changed(self, name, old, new): | ||||
"""Do a bit of validation of the notebook dir.""" | ||||
if not os.path.isabs(new): | ||||
# If we receive a non-absolute path, make it absolute. | ||||
self.notebook_dir = os.path.abspath(new) | ||||
return | ||||
MinRK
|
r15423 | if not os.path.isdir(new): | ||
raise TraitError("No such notebook dir: %r" % new) | ||||
MinRK
|
r15420 | |||
MinRK
|
r15422 | # setting App.notebook_dir implies setting notebook and kernel dirs as well | ||
self.config.FileNotebookManager.notebook_dir = new | ||||
self.config.MappingKernelManager.root_dir = new | ||||
Thomas Kluyver
|
r14063 | |||
Brian E. Granger
|
r4344 | def parse_command_line(self, argv=None): | ||
Brian E. Granger
|
r5104 | super(NotebookApp, self).parse_command_line(argv) | ||
MinRK
|
r12284 | |||
Puneeth Chaganti
|
r6724 | if self.extra_args: | ||
MinRK
|
r13158 | arg0 = self.extra_args[0] | ||
f = os.path.abspath(arg0) | ||||
self.argv.remove(arg0) | ||||
MinRK
|
r13118 | if not os.path.exists(f): | ||
self.log.critical("No such file or directory: %s", f) | ||||
self.exit(1) | ||||
MinRK
|
r15461 | |||
# Use config here, to ensure that it takes higher priority than | ||||
# anything that comes from the profile. | ||||
Thomas Kluyver
|
r16041 | c = Config() | ||
MinRK
|
r7556 | if os.path.isdir(f): | ||
Thomas Kluyver
|
r16041 | c.NotebookApp.notebook_dir = f | ||
Zachary Sailer
|
r13019 | elif os.path.isfile(f): | ||
Thomas Kluyver
|
r16041 | c.NotebookApp.file_to_run = f | ||
self.update_config(c) | ||||
Puneeth Chaganti
|
r6724 | |||
MinRK
|
r12284 | def init_kernel_argv(self): | ||
"""construct the kernel arguments""" | ||||
MinRK
|
r16374 | self.kernel_argv = [] | ||
MinRK
|
r12284 | # Kernel should inherit default config file from frontend | ||
self.kernel_argv.append("--IPKernelApp.parent_appname='%s'" % self.name) | ||||
# Kernel should get *absolute* path to profile directory | ||||
self.kernel_argv.extend(["--profile-dir", self.profile_dir.location]) | ||||
Brian E. Granger
|
r4494 | def init_configurables(self): | ||
MinRK
|
r4963 | # force Session default to be secure | ||
default_secure(self.config) | ||||
MinRK
|
r16423 | kls = import_item(self.kernel_manager_class) | ||
self.kernel_manager = kls( | ||||
MinRK
|
r11064 | parent=self, log=self.log, kernel_argv=self.kernel_argv, | ||
MinRK
|
r4960 | connection_dir = self.profile_dir.security_dir, | ||
Brian E. Granger
|
r4494 | ) | ||
Brian Granger
|
r8180 | kls = import_item(self.notebook_manager_class) | ||
MinRK
|
r11064 | self.notebook_manager = kls(parent=self, log=self.log) | ||
MinRK
|
r16423 | kls = import_item(self.session_manager_class) | ||
self.session_manager = kls(parent=self, log=self.log) | ||||
kls = import_item(self.cluster_manager_class) | ||||
self.cluster_manager = kls(parent=self, log=self.log) | ||||
Brian Granger
|
r6199 | self.cluster_manager.update_profiles() | ||
Brian E. Granger
|
r4344 | |||
Brian E. Granger
|
r4345 | def init_logging(self): | ||
# This prevents double log messages because tornado use a root logger that | ||||
# self.log is a child of. The logging module dipatches log messages to a log | ||||
# and all of its ancenstors until propagate is set to False. | ||||
self.log.propagate = False | ||||
MinRK
|
r10358 | |||
# hook up tornado 3's loggers to our app handlers | ||||
MinRK
|
r16358 | logger = logging.getLogger('tornado') | ||
logger.propagate = True | ||||
logger.parent = self.log | ||||
logger.setLevel(self.log.level) | ||||
MinRK
|
r5799 | |||
def init_webapp(self): | ||||
"""initialize tornado webapp and httpserver""" | ||||
Brian E. Granger
|
r4347 | self.web_app = NotebookWebApplication( | ||
Zachary Sailer
|
r12982 | self, self.kernel_manager, self.notebook_manager, | ||
Paul Ivanov
|
r13042 | self.cluster_manager, self.session_manager, | ||
Matthias BUSSONNIER
|
r15449 | self.log, self.base_url, self.webapp_settings, | ||
self.jinja_environment_options | ||||
Brian E. Granger
|
r4347 | ) | ||
Brian E. Granger
|
r4519 | if self.certfile: | ||
ssl_options = dict(certfile=self.certfile) | ||||
if self.keyfile: | ||||
ssl_options['keyfile'] = self.keyfile | ||||
else: | ||||
ssl_options = None | ||||
MinRK
|
r4705 | self.web_app.password = self.password | ||
Alberto Valverde
|
r10163 | self.http_server = httpserver.HTTPServer(self.web_app, ssl_options=ssl_options, | ||
xheaders=self.trust_xheaders) | ||||
dkua
|
r8723 | if not self.ip: | ||
dkua
|
r8724 | warning = "WARNING: The notebook server is listening on all IP addresses" | ||
dkua
|
r8723 | if ssl_options is None: | ||
Thomas Weißschuh
|
r10564 | self.log.critical(warning + " and not using encryption. This " | ||
dkua
|
r8726 | "is not recommended.") | ||
MinRK
|
r11644 | if not self.password: | ||
Thomas Weißschuh
|
r10564 | self.log.critical(warning + " and not using authentication. " | ||
dkua
|
r8726 | "This is highly insecure and not recommended.") | ||
Bradley M. Froehle
|
r7234 | success = None | ||
for port in random_ports(self.port, self.port_retries+1): | ||||
Brian E. Granger
|
r4548 | try: | ||
self.http_server.listen(port, self.ip) | ||||
Matthias BUSSONNIER
|
r7787 | except socket.error as e: | ||
MinRK
|
r12845 | if e.errno == errno.EADDRINUSE: | ||
self.log.info('The port %i is already in use, trying another random port.' % port) | ||||
continue | ||||
elif e.errno in (errno.EACCES, getattr(errno, 'WSAEACCES', errno.EACCES)): | ||||
self.log.warn("Permission to listen on port %i denied" % port) | ||||
continue | ||||
else: | ||||
Brian E. Granger
|
r4548 | raise | ||
else: | ||||
self.port = port | ||||
Bradley M. Froehle
|
r7234 | success = True | ||
Brian E. Granger
|
r4548 | break | ||
Bradley M. Froehle
|
r7234 | if not success: | ||
self.log.critical('ERROR: the notebook server could not be started because ' | ||||
'no available port could be found.') | ||||
Bradley M. Froehle
|
r7240 | self.exit(1) | ||
MinRK
|
r5799 | |||
Thomas Kluyver
|
r14063 | @property | ||
def display_url(self): | ||||
ip = self.ip if self.ip else '[all ip addresses on your system]' | ||||
return self._url(ip) | ||||
@property | ||||
def connection_url(self): | ||||
MinRK
|
r16334 | ip = self.ip if self.ip else 'localhost' | ||
Thomas Kluyver
|
r14063 | return self._url(ip) | ||
def _url(self, ip): | ||||
proto = 'https' if self.certfile else 'http' | ||||
MinRK
|
r15238 | return "%s://%s:%i%s" % (proto, ip, self.port, self.base_url) | ||
Thomas Kluyver
|
r14063 | |||
MinRK
|
r6498 | def init_signal(self): | ||
MinRK
|
r10200 | if not sys.platform.startswith('win'): | ||
MinRK
|
r6509 | signal.signal(signal.SIGINT, self._handle_sigint) | ||
MinRK
|
r6508 | signal.signal(signal.SIGTERM, self._signal_stop) | ||
Paul Ivanov
|
r10031 | if hasattr(signal, 'SIGUSR1'): | ||
# Windows doesn't support SIGUSR1 | ||||
signal.signal(signal.SIGUSR1, self._signal_info) | ||||
Paul Ivanov
|
r9913 | if hasattr(signal, 'SIGINFO'): | ||
# only on BSD-based systems | ||||
signal.signal(signal.SIGINFO, self._signal_info) | ||||
MinRK
|
r6498 | |||
MinRK
|
r6508 | def _handle_sigint(self, sig, frame): | ||
"""SIGINT handler spawns confirmation dialog""" | ||||
# register more forceful signal handler for ^C^C case | ||||
signal.signal(signal.SIGINT, self._signal_stop) | ||||
# request confirmation dialog in bg thread, to avoid | ||||
# blocking the App | ||||
thread = threading.Thread(target=self._confirm_exit) | ||||
thread.daemon = True | ||||
thread.start() | ||||
def _restore_sigint_handler(self): | ||||
"""callback for restoring original SIGINT handler""" | ||||
signal.signal(signal.SIGINT, self._handle_sigint) | ||||
def _confirm_exit(self): | ||||
"""confirm shutdown on ^C | ||||
A second ^C, or answering 'y' within 5s will cause shutdown, | ||||
otherwise original SIGINT handler will be restored. | ||||
MinRK
|
r7689 | |||
This doesn't work on Windows. | ||||
MinRK
|
r6508 | """ | ||
MinRK
|
r6509 | # FIXME: remove this delay when pyzmq dependency is >= 2.1.11 | ||
time.sleep(0.1) | ||||
Paul Ivanov
|
r9912 | info = self.log.info | ||
info('interrupted') | ||||
Thomas Kluyver
|
r13348 | print(self.notebook_info()) | ||
Paul Ivanov
|
r10019 | sys.stdout.write("Shutdown this notebook server (y/[n])? ") | ||
MinRK
|
r6508 | sys.stdout.flush() | ||
r,w,x = select.select([sys.stdin], [], [], 5) | ||||
if r: | ||||
line = sys.stdin.readline() | ||||
Renaud Richardet
|
r16211 | if line.lower().startswith('y') and 'n' not in line.lower(): | ||
MinRK
|
r6508 | self.log.critical("Shutdown confirmed") | ||
ioloop.IOLoop.instance().stop() | ||||
return | ||||
else: | ||||
Thomas Kluyver
|
r13348 | print("No answer for 5s:", end=' ') | ||
print("resuming operation...") | ||||
MinRK
|
r6508 | # no answer, or answer is no: | ||
# set it back to original SIGINT handler | ||||
# use IOLoop.add_callback because signal.signal must be called | ||||
# from main thread | ||||
ioloop.IOLoop.instance().add_callback(self._restore_sigint_handler) | ||||
def _signal_stop(self, sig, frame): | ||||
MinRK
|
r6498 | self.log.critical("received signal %s, stopping", sig) | ||
ioloop.IOLoop.instance().stop() | ||||
Paul Ivanov
|
r9913 | |||
def _signal_info(self, sig, frame): | ||||
Thomas Kluyver
|
r13348 | print(self.notebook_info()) | ||
MinRK
|
r6498 | |||
MinRK
|
r10557 | def init_components(self): | ||
"""Check the components submodule, and warn if it's unclean""" | ||||
status = submodule.check_submodule_status() | ||||
if status == 'missing': | ||||
self.log.warn("components submodule missing, running `git submodule update`") | ||||
submodule.update_submodules(submodule.ipython_parent()) | ||||
elif status == 'unclean': | ||||
self.log.warn("components submodule unclean, you may see 404s on static/components") | ||||
self.log.warn("run `setup.py submodule` or `git submodule update` to update") | ||||
MinRK
|
r5799 | @catch_config_error | ||
def initialize(self, argv=None): | ||||
super(NotebookApp, self).initialize(argv) | ||||
MinRK
|
r13307 | self.init_logging() | ||
MinRK
|
r12284 | self.init_kernel_argv() | ||
MinRK
|
r5799 | self.init_configurables() | ||
MinRK
|
r10557 | self.init_components() | ||
MinRK
|
r5799 | self.init_webapp() | ||
MinRK
|
r6498 | self.init_signal() | ||
MinRK
|
r5799 | |||
def cleanup_kernels(self): | ||||
Brian E. Granger
|
r9112 | """Shutdown all kernels. | ||
MinRK
|
r5799 | |||
The kernels will shutdown themselves when this process no longer exists, | ||||
but explicit shutdown allows the KernelManagers to cleanup the connection files. | ||||
""" | ||||
self.log.info('Shutting down kernels') | ||||
Brian E. Granger
|
r9112 | self.kernel_manager.shutdown_all() | ||
Brian E. Granger
|
r4344 | |||
Paul Ivanov
|
r10019 | def notebook_info(self): | ||
"Return the current working directory and the server url information" | ||||
Paul Ivanov
|
r13261 | info = self.notebook_manager.info_string() + "\n" | ||
info += "%d active kernels \n" % len(self.kernel_manager._kernels) | ||||
Thomas Kluyver
|
r14063 | return info + "The IPython Notebook is running at: %s" % self.display_url | ||
def server_info(self): | ||||
"""Return a JSONable dict of information about this server.""" | ||||
return {'url': self.connection_url, | ||||
'hostname': self.ip if self.ip else 'localhost', | ||||
'port': self.port, | ||||
'secure': bool(self.certfile), | ||||
MinRK
|
r15238 | 'base_url': self.base_url, | ||
MinRK
|
r15420 | 'notebook_dir': os.path.abspath(self.notebook_dir), | ||
Thomas Kluyver
|
r14063 | } | ||
def write_server_info_file(self): | ||||
"""Write the result of server_info() to the JSON file info_file.""" | ||||
Thomas Kluyver
|
r14079 | with open(self.info_file, 'w') as f: | ||
json.dump(self.server_info(), f, indent=2) | ||||
Thomas Kluyver
|
r14063 | |||
def remove_server_info_file(self): | ||||
"""Remove the nbserver-<pid>.json file created for this server. | ||||
Ignores the error raised when the file has already been removed. | ||||
""" | ||||
try: | ||||
os.unlink(self.info_file) | ||||
except OSError as e: | ||||
if e.errno != errno.ENOENT: | ||||
raise | ||||
Paul Ivanov
|
r9912 | |||
Brian E. Granger
|
r4344 | def start(self): | ||
MinRK
|
r10358 | """ Start the IPython Notebook server app, after initialization | ||
Paul Ivanov
|
r9910 | |||
This method takes no arguments so all configuration and initialization | ||||
must be done prior to calling this method.""" | ||||
Thomas Kluyver
|
r14177 | if self.subapp is not None: | ||
return self.subapp.start() | ||||
Fernando Perez
|
r5786 | info = self.log.info | ||
Paul Ivanov
|
r10019 | for line in self.notebook_info().split("\n"): | ||
info(line) | ||||
MinRK
|
r11228 | info("Use Control-C to stop this server and shut down all kernels (twice to skip confirmation).") | ||
Fernando Perez
|
r5786 | |||
Thomas Kluyver
|
r14063 | self.write_server_info_file() | ||
MinRK
|
r7556 | if self.open_browser or self.file_to_run: | ||
Bradley M. Froehle
|
r7658 | try: | ||
browser = webbrowser.get(self.browser or None) | ||||
except webbrowser.Error as e: | ||||
self.log.warn('No web browser found: %s.' % e) | ||||
browser = None | ||||
MinRK
|
r13118 | |||
MinRK
|
r15460 | if self.file_to_run: | ||
fullpath = os.path.join(self.notebook_dir, self.file_to_run) | ||||
if not os.path.exists(fullpath): | ||||
self.log.critical("%s does not exist" % fullpath) | ||||
self.exit(1) | ||||
uri = url_path_join('notebooks', self.file_to_run) | ||||
Puneeth Chaganti
|
r6724 | else: | ||
MinRK
|
r15460 | uri = 'tree' | ||
Bradley M. Froehle
|
r7658 | if browser: | ||
MinRK
|
r15460 | b = lambda : browser.open(url_path_join(self.connection_url, uri), | ||
Thomas Kluyver
|
r14063 | new=2) | ||
Bradley M. Froehle
|
r7658 | threading.Thread(target=b).start() | ||
MinRK
|
r5799 | try: | ||
ioloop.IOLoop.instance().start() | ||||
except KeyboardInterrupt: | ||||
info("Interrupted...") | ||||
finally: | ||||
self.cleanup_kernels() | ||||
Thomas Kluyver
|
r14063 | self.remove_server_info_file() | ||
Thomas Kluyver
|
r14177 | def list_running_servers(profile='default'): | ||
Thomas Kluyver
|
r14063 | """Iterate over the server info files of running notebook servers. | ||
MinRK
|
r5799 | |||
Thomas Kluyver
|
r14063 | Given a profile name, find nbserver-* files in the security directory of | ||
that profile, and yield dicts of their information, each one pertaining to | ||||
a currently running notebook server instance. | ||||
""" | ||||
pd = ProfileDir.find_profile_dir_by_name(get_ipython_dir(), name=profile) | ||||
for file in os.listdir(pd.security_dir): | ||||
if file.startswith('nbserver-'): | ||||
with io.open(os.path.join(pd.security_dir, file), encoding='utf-8') as f: | ||||
yield json.load(f) | ||||
Brian E. Granger
|
r4344 | |||
#----------------------------------------------------------------------------- | ||||
# Main entry point | ||||
#----------------------------------------------------------------------------- | ||||
Brian E. Granger
|
r4339 | |||
MinRK
|
r11176 | launch_new_instance = NotebookApp.launch_instance | ||
Brian E. Granger
|
r4339 | |||