extension.py 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218
  1. # coding: utf-8
  2. """A tornado based Jupyter lab server."""
  3. # Copyright (c) Jupyter Development Team.
  4. # Distributed under the terms of the Modified BSD License.
  5. # ----------------------------------------------------------------------------
  6. # Module globals
  7. # ----------------------------------------------------------------------------
  8. import os
  9. DEV_NOTE = """You're running JupyterLab from source.
  10. If you're working on the TypeScript sources of JupyterLab, try running
  11. jupyter lab --dev-mode --watch
  12. to have the system incrementally watch and build JupyterLab for you, as you
  13. make changes.
  14. """
  15. CORE_NOTE = """
  16. Running the core application with no additional extensions or settings
  17. """
  18. def load_config(nbapp):
  19. """Load the JupyterLab configuration and defaults for a given application.
  20. """
  21. from jupyterlab_server import LabConfig
  22. from .commands import (
  23. get_app_dir,
  24. get_app_info,
  25. get_workspaces_dir,
  26. get_user_settings_dir,
  27. pjoin
  28. )
  29. app_dir = getattr(nbapp, 'app_dir', get_app_dir())
  30. info = get_app_info(options=dict(app_dir=app_dir))
  31. static_url = info['staticUrl']
  32. user_settings_dir = getattr(
  33. nbapp, 'user_settings_dir', get_user_settings_dir()
  34. )
  35. workspaces_dir = getattr(nbapp, 'workspaces_dir', get_workspaces_dir())
  36. config = LabConfig()
  37. config.app_dir = app_dir
  38. config.app_name = 'JupyterLab'
  39. config.app_namespace = 'jupyterlab'
  40. config.app_settings_dir = pjoin(app_dir, 'settings')
  41. config.app_version = info['version']
  42. config.cache_files = True
  43. config.schemas_dir = pjoin(app_dir, 'schemas')
  44. config.templates_dir = pjoin(app_dir, 'static')
  45. config.themes_dir = pjoin(app_dir, 'themes')
  46. config.user_settings_dir = user_settings_dir
  47. config.workspaces_dir = workspaces_dir
  48. if getattr(nbapp, 'override_static_url', ''):
  49. static_url = nbapp.override_static_url
  50. if getattr(nbapp, 'override_theme_url', ''):
  51. config.themes_url = nbapp.override_theme_url
  52. config.themes_dir = ''
  53. if static_url:
  54. config.static_url = static_url
  55. else:
  56. config.static_dir = pjoin(app_dir, 'static')
  57. return config
  58. def load_jupyter_server_extension(nbapp):
  59. """Load the JupyterLab server extension.
  60. """
  61. # Delay imports to speed up jlpmapp
  62. from json import dumps
  63. from jupyterlab_server import add_handlers
  64. from notebook.utils import url_path_join as ujoin, url_escape
  65. from notebook._version import version_info
  66. from tornado.ioloop import IOLoop
  67. from markupsafe import Markup
  68. from .handlers.build_handler import build_path, Builder, BuildHandler
  69. from .handlers.extension_manager_handler import (
  70. extensions_handler_path, ExtensionManager, ExtensionHandler
  71. )
  72. from .handlers.error_handler import ErrorHandler
  73. from .commands import (
  74. DEV_DIR, HERE, ensure_app, ensure_core, ensure_dev, watch,
  75. watch_dev, get_app_dir
  76. )
  77. web_app = nbapp.web_app
  78. logger = nbapp.log
  79. base_url = nbapp.base_url
  80. # Handle the app_dir
  81. app_dir = getattr(nbapp, 'app_dir', get_app_dir())
  82. # Check for core mode.
  83. core_mode = False
  84. if getattr(nbapp, 'core_mode', False) or app_dir.startswith(HERE):
  85. app_dir = HERE
  86. core_mode = True
  87. logger.info('Running JupyterLab in core mode')
  88. # Check for dev mode.
  89. dev_mode = False
  90. if getattr(nbapp, 'dev_mode', False) or app_dir.startswith(DEV_DIR):
  91. app_dir = DEV_DIR
  92. dev_mode = True
  93. logger.info('Running JupyterLab in dev mode')
  94. # Set the value on nbapp so it will get picked up in load_config
  95. nbapp.app_dir = app_dir
  96. config = load_config(nbapp)
  97. config.app_name = 'JupyterLab'
  98. config.app_namespace = 'jupyterlab'
  99. config.app_url = '/lab'
  100. config.cache_files = True
  101. # Check for watch.
  102. watch_mode = getattr(nbapp, 'watch', False)
  103. if watch_mode and core_mode:
  104. logger.warn('Cannot watch in core mode, did you mean --dev-mode?')
  105. watch_mode = False
  106. if core_mode and dev_mode:
  107. logger.warn('Conflicting modes, choosing dev_mode over core_mode')
  108. core_mode = False
  109. page_config = web_app.settings.setdefault('page_config_data', dict())
  110. page_config['buildAvailable'] = not core_mode and not dev_mode
  111. page_config['buildCheck'] = not core_mode and not dev_mode
  112. page_config['devMode'] = dev_mode
  113. page_config['token'] = nbapp.token
  114. # Client-side code assumes notebookVersion is a JSON-encoded string
  115. # TODO: fix this when we can make such a change
  116. page_config['notebookVersion'] = dumps(version_info)
  117. if nbapp.file_to_run and type(nbapp).__name__ == "LabApp":
  118. relpath = os.path.relpath(nbapp.file_to_run, nbapp.notebook_dir)
  119. uri = url_escape(ujoin('/lab/tree', *relpath.split(os.sep)))
  120. nbapp.default_url = uri
  121. nbapp.file_to_run = ''
  122. # Print messages.
  123. logger.info('JupyterLab extension loaded from %s' % HERE)
  124. logger.info('JupyterLab application directory is %s' % app_dir)
  125. build_url = ujoin(base_url, build_path)
  126. builder = Builder(logger, core_mode, app_dir)
  127. build_handler = (build_url, BuildHandler, {'builder': builder})
  128. handlers = [build_handler]
  129. errored = False
  130. if core_mode:
  131. logger.info(CORE_NOTE.strip())
  132. ensure_core(logger)
  133. elif dev_mode:
  134. if not watch_mode:
  135. ensure_dev(logger)
  136. logger.info(DEV_NOTE)
  137. # Make sure the app dir exists.
  138. else:
  139. msgs = ensure_app(app_dir)
  140. if msgs:
  141. [logger.error(msg) for msg in msgs]
  142. handler = (ujoin(base_url, '/lab'), ErrorHandler, { 'messages': msgs })
  143. handlers.append(handler)
  144. errored = True
  145. if watch_mode:
  146. logger.info('Starting JupyterLab watch mode...')
  147. # Set the ioloop in case the watch fails.
  148. nbapp.ioloop = IOLoop.current()
  149. if dev_mode:
  150. watch_dev(logger)
  151. else:
  152. watch(options=dict(app_dir=app_dir, logger=logger))
  153. page_config['buildAvailable'] = False
  154. config.cache_files = False
  155. if not core_mode and not errored:
  156. ext_url = ujoin(base_url, extensions_handler_path)
  157. ext_manager = ExtensionManager(logger, app_dir)
  158. ext_handler = (ext_url, ExtensionHandler, {'manager': ext_manager})
  159. handlers.append(ext_handler)
  160. # Must add before the root server handlers to avoid shadowing.
  161. web_app.add_handlers('.*$', handlers)
  162. # If running under JupyterHub, add more metadata.
  163. if hasattr(nbapp, 'hub_prefix'):
  164. page_config['hubPrefix'] = nbapp.hub_prefix
  165. page_config['hubHost'] = nbapp.hub_host
  166. page_config['hubUser'] = nbapp.user
  167. # Assume the server_name property indicates running JupyterHub 1.0.
  168. if hasattr(nbapp, 'server_name'):
  169. page_config['hubServerName'] = nbapp.server_name
  170. api_token = os.getenv('JUPYTERHUB_API_TOKEN', '')
  171. page_config['token'] = api_token
  172. # Add the root handlers if we have not errored.
  173. if not errored:
  174. add_handlers(web_app, config)