commands.py 35 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119
  1. # coding: utf-8
  2. """JupyterLab entry points"""
  3. # Copyright (c) Jupyter Development Team.
  4. # Distributed under the terms of the Modified BSD License.
  5. from __future__ import print_function
  6. import errno
  7. import glob
  8. import hashlib
  9. import json
  10. import logging
  11. import pipes
  12. import os
  13. import re
  14. import shutil
  15. import site
  16. import sys
  17. import tarfile
  18. import tempfile
  19. from distutils.version import LooseVersion
  20. from functools import partial
  21. from jupyter_core.paths import jupyter_config_path
  22. from notebook.nbextensions import GREEN_ENABLED, GREEN_OK, RED_DISABLED, RED_X
  23. from os import path as osp
  24. from os.path import join as pjoin
  25. from subprocess import CalledProcessError, Popen, STDOUT, call
  26. from tornado import gen
  27. from tornado.ioloop import IOLoop
  28. from .semver import Range, gte, lt, lte, gt
  29. if sys.platform == 'win32':
  30. from subprocess import list2cmdline
  31. else:
  32. def list2cmdline(cmd_list):
  33. return ' '.join(map(pipes.quote, cmd_list))
  34. here = osp.dirname(osp.abspath(__file__))
  35. logging.basicConfig(format='%(message)s', level=logging.INFO)
  36. def get_app_dir(app_dir=None):
  37. """Get the configured JupyterLab app directory.
  38. """
  39. # Default to the override environment variable.
  40. app_dir = app_dir or os.environ.get('JUPYTERLAB_DIR')
  41. if app_dir:
  42. return os.path.realpath(app_dir)
  43. # Use the default locations for data_files.
  44. app_dir = pjoin(sys.prefix, 'share', 'jupyter', 'lab')
  45. # Check for a user level install.
  46. userbase = getattr(site, 'userbase', '')
  47. if here.startswith(userbase) and not app_dir.startswith(userbase):
  48. app_dir = pjoin(userbase, 'share', 'jupyter', 'lab')
  49. # Check for a system install in '/usr/local/share'.
  50. elif (sys.prefix.startswith('/usr') and not
  51. os.path.exists(app_dir) and
  52. os.path.exists('/usr/local/share/jupyter/lab')):
  53. app_dir = '/usr/local/share/jupyter/lab'
  54. return os.path.realpath(app_dir)
  55. def get_user_settings_dir():
  56. """Get the configured JupyterLab app directory.
  57. """
  58. settings_dir = os.environ.get('JUPYTERLAB_SETTINGS_DIR')
  59. settings_dir = settings_dir or pjoin(
  60. jupyter_config_path()[0], 'lab', 'user-settings'
  61. )
  62. return os.path.realpath(settings_dir)
  63. @gen.coroutine
  64. def run(cmd, **kwargs):
  65. """Run a command in the given working directory.
  66. """
  67. logger = kwargs.pop('logger', logging) or logging
  68. abort_callback = kwargs.pop('abort_callback', None)
  69. logger.info('> ' + list2cmdline(cmd))
  70. kwargs.setdefault('shell', sys.platform == 'win32')
  71. kwargs.setdefault('env', os.environ)
  72. kwargs.setdefault('stderr', STDOUT)
  73. proc = None
  74. yield gen.moment # Sync up to the iterator
  75. try:
  76. proc = Popen(cmd, **kwargs)
  77. # Poll the process once per second until finished.
  78. while 1:
  79. yield gen.sleep(1)
  80. if proc.poll() is not None:
  81. break
  82. if abort_callback and abort_callback():
  83. raise ValueError('Aborted')
  84. except CalledProcessError as error:
  85. output = error.output.decode('utf-8')
  86. logger.info(output)
  87. raise error
  88. finally:
  89. if proc:
  90. proc.wait()
  91. def ensure_dev_build(logger=None):
  92. """Ensure that the dev build assets are there"""
  93. cmd = [get_npm_name(), 'run', 'build']
  94. func = partial(run, cmd, cwd=os.path.dirname(here), logger=logger)
  95. loop = IOLoop.instance()
  96. loop.instance().run_sync(func)
  97. def watch(cwd, logger=None):
  98. """Run watch mode in a given directory"""
  99. loop = IOLoop.instance()
  100. loop.add_callback(run, [get_npm_name(), 'run', 'watch'], cwd=cwd,
  101. logger=logger)
  102. def install_extension(extension, app_dir=None, logger=None):
  103. """Install an extension package into JupyterLab.
  104. Follows the semantics of https://docs.npmjs.com/cli/install.
  105. The extension is first validated.
  106. If link is true, the source directory is linked using `npm link`.
  107. """
  108. func = partial(install_extension_async, extension, app_dir=app_dir,
  109. logger=logger)
  110. return IOLoop.instance().run_sync(func)
  111. @gen.coroutine
  112. def install_extension_async(extension, app_dir=None, logger=None, abort_callback=None):
  113. """Install an extension package into JupyterLab.
  114. Follows the semantics of https://docs.npmjs.com/cli/install.
  115. The extension is first validated.
  116. If link is true, the source directory is linked using `npm link`.
  117. """
  118. yield check_node()
  119. app_dir = get_app_dir(app_dir)
  120. logger = logger or logging
  121. if app_dir == here:
  122. raise ValueError('Cannot install extensions in core app')
  123. extension = _normalize_path(extension)
  124. # Check for a core extensions here.
  125. data = _get_core_data()
  126. if extension in _get_core_extensions():
  127. config = _get_build_config(app_dir)
  128. uninstalled = config.get('uninstalled_core_extensions', [])
  129. if extension in uninstalled:
  130. uninstalled.remove(extension)
  131. config['uninstalled_core_extensions'] = uninstalled
  132. _write_build_config(config, app_dir, logger=logger)
  133. return
  134. # Run npm install if needed
  135. if (os.path.exists(extension) and
  136. os.path.isdir(extension) and
  137. not os.path.exists(pjoin(extension, 'node_modules'))):
  138. yield run([get_npm_name(), 'install'], cwd=extension, logger=logger,
  139. abort_callback=abort_callback)
  140. _ensure_app_dirs(app_dir, logger)
  141. target = pjoin(app_dir, 'extensions', 'temp')
  142. if os.path.exists(target):
  143. shutil.rmtree(target)
  144. os.makedirs(target)
  145. # npm pack the extension
  146. yield run([get_npm_name(), 'pack', extension], cwd=target, logger=logger,
  147. abort_callback=abort_callback)
  148. fnames = glob.glob(pjoin(target, '*.*'))
  149. if not fnames:
  150. raise ValueError('Invalid extension')
  151. fname = os.path.basename(fnames[0])
  152. data = _read_package(pjoin(target, fname))
  153. # Remove the tarball if the package is not an extension.
  154. if not _is_extension(data):
  155. shutil.rmtree(target)
  156. msg = '%s is not a valid JupyterLab extension' % extension
  157. raise ValueError(msg)
  158. # Remove the tarball if the package is not compatible.
  159. core_data = _get_core_data()
  160. deps = data.get('dependencies', dict())
  161. errors = _validate_compatibility(extension, deps, core_data)
  162. if errors:
  163. shutil.rmtree(target)
  164. msg = _format_compatibility_errors(
  165. data['name'], data['version'], errors
  166. )
  167. raise ValueError(msg)
  168. # Check for existing app extension of the same name.
  169. extensions = _get_extensions(app_dir)
  170. if data['name'] in extensions:
  171. other = extensions[data['name']]
  172. path = other['path']
  173. if osp.exists(path) and other['location'] == 'app':
  174. os.remove(path)
  175. # Remove an existing extension tarball.
  176. ext_path = pjoin(app_dir, 'extensions', fname)
  177. if os.path.exists(ext_path):
  178. os.remove(ext_path)
  179. shutil.move(pjoin(target, fname), pjoin(app_dir, 'extensions'))
  180. shutil.rmtree(target)
  181. # Remove any existing package from staging/node_modules to force
  182. # npm to re-install it from the tarball.
  183. target = pjoin(app_dir, 'staging', 'node_modules', data['name'])
  184. if os.path.exists(target):
  185. shutil.rmtree(target)
  186. def link_package(path, app_dir=None, logger=None):
  187. """Link a package against the JupyterLab build."""
  188. func = partial(link_package_async, path, app_dir=app_dir, logger=logger)
  189. return IOLoop.instance().run_sync(func)
  190. @gen.coroutine
  191. def link_package_async(path, app_dir=None, logger=None, abort_callback=None):
  192. """Link a package against the JupyterLab build.
  193. """
  194. logger = logger or logging
  195. app_dir = get_app_dir(app_dir)
  196. if app_dir == here:
  197. raise ValueError('Cannot link packages in core app')
  198. path = _normalize_path(path)
  199. _ensure_app_dirs(app_dir, logger)
  200. # Verify the package.json data.
  201. pkg_path = osp.join(path, 'package.json')
  202. if not osp.exists(pkg_path):
  203. msg = 'Linked package must point to a directory with package.json'
  204. raise ValueError(msg)
  205. with open(pkg_path) as fid:
  206. data = json.load(fid)
  207. # Check for a core extensions here.
  208. core_extensions = _get_core_extensions()
  209. if data['name'] in core_extensions:
  210. raise ValueError('Cannot link a core extension')
  211. is_extension = _is_extension(data)
  212. if is_extension:
  213. yield install_extension_async(path, app_dir, abort_callback=abort_callback)
  214. else:
  215. msg = ('*** Note: Linking non-extension package "%s" (lacks ' +
  216. '`jupyterlab.extension` metadata)')
  217. logger.info(msg % data['name'])
  218. core_data = _get_core_data()
  219. deps = data.get('dependencies', dict())
  220. name = data['name']
  221. errors = _validate_compatibility(name, deps, core_data)
  222. if errors:
  223. msg = _format_compatibility_errors(name, data['version'], errors)
  224. raise ValueError(msg)
  225. config = _get_build_config(app_dir)
  226. config.setdefault('linked_packages', dict())
  227. config['linked_packages'][data['name']] = path
  228. _write_build_config(config, app_dir, logger=logger)
  229. def unlink_package(package, app_dir=None, logger=None):
  230. """Unlink a package from JupyterLab by path or name.
  231. """
  232. logger = logger or logging
  233. package = _normalize_path(package)
  234. name = None
  235. app_dir = get_app_dir(app_dir)
  236. if app_dir == here:
  237. raise ValueError('Cannot link packages in core app')
  238. config = _get_build_config(app_dir)
  239. linked = config.setdefault('linked_packages', dict())
  240. for (key, value) in linked.items():
  241. if value == package or key == package:
  242. name = key
  243. break
  244. if not name:
  245. logger.warn('No package matching "%s" is linked' % package)
  246. return False
  247. del linked[name]
  248. config['linked_packages'] = linked
  249. _write_build_config(config, app_dir, logger=logger)
  250. extensions = _get_extensions(app_dir)
  251. if name in extensions:
  252. uninstall_extension(name, app_dir)
  253. return True
  254. def enable_extension(extension, app_dir=None, logger=None):
  255. """Enable a JupyterLab extension.
  256. """
  257. _toggle_extension(extension, False, app_dir, logger)
  258. def disable_extension(extension, app_dir=None, logger=None):
  259. """Disable a JupyterLab package.
  260. """
  261. _toggle_extension(extension, True, app_dir, logger)
  262. def get_npm_name():
  263. """Get the appropriate npm executable name.
  264. """
  265. return 'npm.cmd' if os.name == 'nt' else 'npm'
  266. @gen.coroutine
  267. def check_node():
  268. """Check for the existence of node and whether it is the right version.
  269. """
  270. try:
  271. yield run(['node', 'node-version-check.js'], cwd=here)
  272. except Exception as e:
  273. raise ValueError('`node` version 5+ is required, see extensions in README')
  274. def build_check(app_dir=None, logger=None):
  275. """Determine whether JupyterLab should be built.
  276. Returns a dictionary with information about the build.
  277. """
  278. func = partial(build_check_async, app_dir=app_dir, logger=logger)
  279. return IOLoop.instance().run_sync(func)
  280. @gen.coroutine
  281. def build_check_async(app_dir=None, logger=None):
  282. """Determine whether JupyterLab should be built.
  283. Returns a tuple of whether a build is recommend and the message.
  284. """
  285. logger = logger or logging
  286. app_dir = get_app_dir(app_dir)
  287. # Check for installed extensions
  288. extensions = _get_extensions(app_dir)
  289. linked = _get_linked_packages(app_dir=app_dir)
  290. # Check for no application.
  291. pkg_path = pjoin(app_dir, 'static', 'package.json')
  292. if not os.path.exists(pkg_path):
  293. raise gen.Return((True,
  294. 'Installed extensions with no built application'))
  295. with open(pkg_path) as fid:
  296. static_data = json.load(fid)
  297. # Look for mismatched version.
  298. static_version = static_data['jupyterlab'].get('version', '')
  299. core_version = static_data['jupyterlab']['version']
  300. if LooseVersion(static_version) != LooseVersion(core_version):
  301. msg = 'Version mismatch: %s (built), %s (current)'
  302. raise gen.Return((True, msg % (static_version, core_version)))
  303. # Look for mismatched extensions.
  304. template_data = _get_package_template(app_dir, logger)
  305. template_exts = template_data['jupyterlab']['extensions']
  306. if set(template_exts) != set(static_data['jupyterlab']['extensions']):
  307. raise gen.Return((True, 'Installed extensions changed'))
  308. template_mime_exts = set(template_data['jupyterlab']['mimeExtensions'])
  309. staging_mime_exts = set(static_data['jupyterlab']['mimeExtensions'])
  310. if template_mime_exts != staging_mime_exts:
  311. raise gen.Return((True, 'Installed extensions changed'))
  312. deps = static_data.get('dependencies', dict())
  313. # Look for mismatched extension paths.
  314. names = set(extensions)
  315. names.update(linked)
  316. for name in names:
  317. # Check for dependencies that were rejected as incompatible.
  318. if name not in template_data['dependencies']:
  319. continue
  320. path = deps[name]
  321. if path.startswith('file:'):
  322. path = path.replace('file:', '')
  323. path = os.path.abspath(pjoin(app_dir, 'staging', path))
  324. template_path = template_data['dependencies'][name]
  325. if sys.platform == 'win32':
  326. path = path.lower()
  327. template_path = template_path.lower()
  328. if path != template_path:
  329. if name in extensions:
  330. raise gen.Return((True, 'Installed extensions changed'))
  331. raise gen.Return((True, 'Linked package changed'))
  332. # Look for linked packages that have changed.
  333. for (name, path) in linked.items():
  334. normed_name = name.replace('@', '').replace('/', '-')
  335. if name in extensions:
  336. root = pjoin(app_dir, 'extensions')
  337. else:
  338. root = pjoin(app_dir, 'staging', 'linked_packages')
  339. path0 = glob.glob(pjoin(root, '%s*.tgz' % normed_name))[0]
  340. existing = _tarsum(path0)
  341. tempdir = tempfile.mkdtemp()
  342. cmd = [get_npm_name(), 'pack', linked[name]]
  343. yield run(cmd, cwd=tempdir, logger=logger)
  344. path1 = glob.glob(pjoin(tempdir, '*.tgz'))[0]
  345. current = _tarsum(path1)
  346. if current != existing:
  347. raise gen.Return((True, 'Linked package changed content'))
  348. raise gen.Return((False, ''))
  349. def validate_compatibility(extension, app_dir=None, logger=None):
  350. """Validate the compatibility of an extension.
  351. """
  352. app_dir = get_app_dir(app_dir)
  353. extensions = _get_extensions(app_dir)
  354. if extension not in extensions:
  355. raise ValueError('%s is not an installed extension')
  356. deps = extensions[extension].get('dependencies', dict())
  357. core_data = _get_core_data()
  358. return _validate_compatibility(extension, deps, core_data)
  359. def uninstall_extension(name, app_dir=None, logger=None):
  360. """Uninstall an extension by name.
  361. """
  362. logger = logger or logging
  363. app_dir = get_app_dir(app_dir)
  364. if app_dir == here:
  365. raise ValueError('Cannot install packages in core app')
  366. # Allow for uninstalled core extensions here.
  367. data = _get_core_data()
  368. if name in _get_core_extensions():
  369. logger.info('Uninstalling core extension %s' % name)
  370. config = _get_build_config(app_dir)
  371. uninstalled = config.get('uninstalled_core_extensions', [])
  372. if name not in uninstalled:
  373. uninstalled.append(name)
  374. config['uninstalled_core_extensions'] = uninstalled
  375. _write_build_config(config, app_dir, logger=logger)
  376. return True
  377. for (extname, data) in _get_extensions(app_dir).items():
  378. path = data['path']
  379. if extname == name:
  380. msg = 'Uninstalling %s from %s' % (name, os.path.dirname(path))
  381. logger.info(msg)
  382. os.remove(path)
  383. return True
  384. logger.warn('No labextension named "%s" installed' % name)
  385. return False
  386. def list_extensions(app_dir=None, logger=None):
  387. """List the extensions.
  388. """
  389. logger = logger or logging
  390. app_dir = get_app_dir(app_dir)
  391. extensions = _get_extensions(app_dir)
  392. disabled = _get_disabled(app_dir)
  393. all_linked = _get_linked_packages(app_dir, logger=logger)
  394. app = []
  395. sys = []
  396. linked = []
  397. errors = dict()
  398. core_data = _get_core_data()
  399. # We want to organize by dir.
  400. sys_path = pjoin(get_app_dir(), 'extensions')
  401. for (key, value) in extensions.items():
  402. deps = extensions[key].get('dependencies', dict())
  403. errors[key] = _validate_compatibility(key, deps, core_data)
  404. if key in all_linked:
  405. linked.append(key)
  406. if value['path'] == sys_path and sys_path != app_dir:
  407. sys.append(key)
  408. continue
  409. app.append(key)
  410. logger.info('JupyterLab v%s' % core_data['jupyterlab']['version'])
  411. logger.info('Known labextensions:')
  412. if app:
  413. logger.info(' app dir: %s' % app_dir)
  414. for item in sorted(app):
  415. logger.info(item)
  416. version = extensions[item]['version']
  417. extra = ''
  418. if is_disabled(item, disabled):
  419. extra += ' %s' % RED_DISABLED
  420. else:
  421. extra += ' %s' % GREEN_ENABLED
  422. if errors[item]:
  423. extra += ' %s' % RED_X
  424. else:
  425. extra += ' %s' % GREEN_OK
  426. logger.info(' %s v%s%s' % (item, version, extra))
  427. if errors[item]:
  428. msg = _format_compatibility_errors(item, version, errors[item])
  429. logger.warn(msg + '\n')
  430. if sys:
  431. logger.info(' sys dir: %s' % sys_path)
  432. for item in sorted(sys):
  433. version = extensions[item]['version']
  434. extra = ''
  435. if item in disabled:
  436. extra += ' %s' % RED_DISABLED
  437. else:
  438. extra += ' %s' % GREEN_ENABLED
  439. logger.info(' %s%s' % (item, extra))
  440. if errors[item]:
  441. extra += ' %s' % RED_X
  442. else:
  443. extra += ' %s' % GREEN_OK
  444. if item in linked:
  445. extra += '*'
  446. logger.info(' %s v%s%s' % (item, version, extra))
  447. if errors[item]:
  448. msg = _format_compatibility_errors(item, version, errors[item])
  449. logger.warn(msg + '\n')
  450. if linked:
  451. logger.info(' linked extensions:')
  452. for item in sorted(linked):
  453. logger.info(' %s: %s' % (item, all_linked[item]))
  454. if len(all_linked) > len(linked):
  455. logger.info(' linked packages:')
  456. for key in sorted(all_linked.keys()):
  457. if (key in linked):
  458. continue
  459. logger.info(' %s: %s' % (key, all_linked[key]))
  460. # Handle uninstalled and disabled core packages
  461. uninstalled_core = _get_uinstalled_core_extensions(app_dir)
  462. if uninstalled_core:
  463. logger.info('\nUninstalled core extensions:')
  464. [logger.info(' %s' % item) for item in sorted(uninstalled_core)]
  465. core_extensions = _get_core_extensions()
  466. disabled_core = []
  467. for key in core_extensions:
  468. if key in disabled:
  469. disabled_core.append(key)
  470. if disabled_core:
  471. logger.info('\nDisabled core extensions:')
  472. [logger.info(' %s' % item) for item in sorted(disabled_core)]
  473. def clean(app_dir=None):
  474. """Clean the JupyterLab application directory."""
  475. app_dir = get_app_dir(app_dir)
  476. if app_dir == here:
  477. raise ValueError('Cannot clean the core app')
  478. for name in ['static', 'staging']:
  479. target = pjoin(app_dir, name)
  480. if osp.exists(target):
  481. shutil.rmtree(target)
  482. def build(app_dir=None, name=None, version=None, logger=None):
  483. """Build the JupyterLab application.
  484. """
  485. func = partial(build_async, app_dir=app_dir, name=name, version=version,
  486. logger=logger)
  487. return IOLoop.instance().run_sync(func)
  488. def is_disabled(name, disabled=[]):
  489. for pattern in disabled:
  490. if name == pattern:
  491. return True
  492. if re.compile(pattern).match(name) != None:
  493. return True
  494. return False
  495. @gen.coroutine
  496. def build_async(app_dir=None, name=None, version=None, logger=None, abort_callback=None):
  497. """Build the JupyterLab application.
  498. """
  499. # Set up the build directory.
  500. logger = logger or logging
  501. app_dir = get_app_dir(app_dir)
  502. # Set up the build directory.
  503. yield check_node()
  504. if app_dir == here:
  505. raise ValueError('Cannot build extensions in the core app')
  506. _ensure_package(app_dir, name=name, version=version, logger=logger)
  507. staging = pjoin(app_dir, 'staging')
  508. extensions = _get_extensions(app_dir)
  509. # Ensure an empty linked_packages directory
  510. linked_packages = pjoin(staging, 'linked_packages')
  511. if osp.exists(linked_packages):
  512. shutil.rmtree(linked_packages)
  513. os.makedirs(linked_packages)
  514. # Install the linked packages.
  515. for (name, path) in _get_linked_packages(app_dir, logger=logger).items():
  516. # Handle linked extensions.
  517. if name in extensions:
  518. yield install_extension_async(path, app_dir, abort_callback=abort_callback)
  519. # Handle linked packages that are not extensions.
  520. elif name not in extensions:
  521. yield _install_linked_package(staging, name, path, logger, abort_callback=abort_callback)
  522. npm = get_npm_name()
  523. # Make sure packages are installed.
  524. yield run([npm, 'install'], cwd=staging, logger=logger, abort_callback=abort_callback)
  525. # Build the app.
  526. yield run([npm, 'run', 'clean'], cwd=staging, logger=logger, abort_callback= abort_callback)
  527. yield run([npm, 'run', 'build'], cwd=staging, logger=logger, abort_callback= abort_callback)
  528. # Move the app to the static dir.
  529. static = pjoin(app_dir, 'static')
  530. if os.path.exists(static):
  531. shutil.rmtree(static)
  532. shutil.copytree(pjoin(app_dir, 'staging', 'build'), static)
  533. @gen.coroutine
  534. def _install_linked_package(staging, name, path, logger, abort_callback=None):
  535. """Install a linked non-extension package using a package tarball
  536. to prevent it from being treated as a symlink.
  537. """
  538. # Remove any existing package from staging/node_modules
  539. target = pjoin(staging, 'node_modules', name)
  540. target = target.replace('/', os.sep)
  541. if os.path.exists(target):
  542. shutil.rmtree(target)
  543. linked = pjoin(staging, 'linked_packages')
  544. target = pjoin(linked, 'temp')
  545. if os.path.exists(target):
  546. shutil.rmtree(target)
  547. os.makedirs(target)
  548. # npm pack the extension
  549. yield run([get_npm_name(), 'pack', path], cwd=target, logger=logger, abort_callback=abort_callback)
  550. fname = os.path.basename(glob.glob(pjoin(target, '*.*'))[0])
  551. data = _read_package(pjoin(target, fname))
  552. # Remove the tarball if the package is not compatible.
  553. core_data = _get_core_data()
  554. deps = data.get('dependencies', dict())
  555. errors = _validate_compatibility(path, deps, core_data)
  556. if errors:
  557. shutil.rmtree(target)
  558. msg = _format_compatibility_errors(
  559. data['name'], data['version'], errors
  560. )
  561. raise ValueError(msg)
  562. # Remove an existing extension tarball.
  563. ext_path = pjoin(linked, fname)
  564. if os.path.exists(ext_path):
  565. os.remove(ext_path)
  566. # Move
  567. shutil.move(pjoin(target, fname), linked)
  568. shutil.rmtree(target)
  569. def _get_build_config(app_dir):
  570. """Get the build config data for the given app dir
  571. """
  572. target = pjoin(app_dir, 'settings', 'build_config.json')
  573. if not os.path.exists(target):
  574. return {}
  575. else:
  576. with open(target) as fid:
  577. return json.load(fid)
  578. def _get_page_config(app_dir):
  579. """Get the page config data for the given app dir
  580. """
  581. target = pjoin(app_dir, 'settings', 'page_config.json')
  582. if not os.path.exists(target):
  583. return {}
  584. else:
  585. with open(target) as fid:
  586. return json.load(fid)
  587. def _validate_compatibility(extension, deps, core_data):
  588. """Validate the compatibility of an extension.
  589. """
  590. core_deps = core_data['dependencies']
  591. singletons = core_data['jupyterlab']['singletonPackages']
  592. errors = []
  593. for (key, value) in deps.items():
  594. if key in singletons:
  595. overlap = _test_overlap(core_deps[key], value)
  596. if overlap is False:
  597. errors.append((key, core_deps[key], value))
  598. return errors
  599. def _get_core_data():
  600. """Get the data for the app template.
  601. """
  602. with open(pjoin(here, 'package.app.json')) as fid:
  603. return json.load(fid)
  604. # Provide the application version.
  605. app_version = _get_core_data()['jupyterlab']['version']
  606. def _test_overlap(spec1, spec2):
  607. """Test whether two version specs overlap.
  608. Returns `None` if we cannot determine compatibility,
  609. otherwise whether there is an overlap
  610. """
  611. # Test for overlapping semver ranges.
  612. r1 = Range(spec1, True)
  613. r2 = Range(spec2, True)
  614. # If either range is empty, we cannot verify.
  615. if not r1.range or not r2.range:
  616. return
  617. x1 = r1.set[0][0].semver
  618. x2 = r1.set[0][-1].semver
  619. y1 = r2.set[0][0].semver
  620. y2 = r2.set[0][-1].semver
  621. o1 = r1.set[0][0].operator
  622. o2 = r2.set[0][0].operator
  623. # We do not handle (<) specifiers.
  624. if (o1.startswith('<') or o2.startswith('<')):
  625. return
  626. # Handle single value specifiers.
  627. lx = lte if x1 == x2 else lt
  628. ly = lte if y1 == y2 else lt
  629. gx = gte if x1 == x2 else gt
  630. gy = gte if x1 == x2 else gt
  631. # Handle unbounded (>) specifiers.
  632. def noop(x, y, z):
  633. return True
  634. if x1 == x2 and o1.startswith('>'):
  635. lx = noop
  636. if y1 == y2 and o2.startswith('>'):
  637. ly = noop
  638. # Check for overlap.
  639. return (
  640. gte(x1, y1, True) and ly(x1, y2, True) or
  641. gy(x2, y1, True) and ly(x2, y2, True) or
  642. gte(y1, x1, True) and lx(y1, x2, True) or
  643. gx(y2, x1, True) and lx(y2, x2, True)
  644. )
  645. def _format_compatibility_errors(name, version, errors):
  646. """Format a message for compatibility errors.
  647. """
  648. msgs = []
  649. l0 = 10
  650. l1 = 10
  651. for error in errors:
  652. pkg, jlab, ext = error
  653. jlab = str(Range(jlab, True))
  654. ext = str(Range(ext, True))
  655. msgs.append((pkg, jlab, ext))
  656. l0 = max(l0, len(pkg) + 1)
  657. l1 = max(l1, len(jlab) + 1)
  658. msg = '\n"%s@%s" is not compatible with the current JupyterLab'
  659. msg = msg % (name, version)
  660. msg += '\nConflicting Dependencies:\n'
  661. msg += 'JupyterLab'.ljust(l0)
  662. msg += 'Extension'.ljust(l1)
  663. msg += 'Package\n'
  664. for (pkg, jlab, ext) in msgs:
  665. msg += jlab.ljust(l0) + ext.ljust(l1) + pkg + '\n'
  666. return msg
  667. def _toggle_extension(extension, value, app_dir=None, logger=None):
  668. """Enable or disable a lab extension.
  669. """
  670. app_dir = get_app_dir(app_dir)
  671. extensions = _get_extensions(app_dir)
  672. config = _get_build_config(app_dir)
  673. disabled = config.get('disabledExtensions', [])
  674. if value and extension not in disabled:
  675. disabled.append(extension)
  676. if not value and extension in disabled:
  677. disabled.remove(extension)
  678. config['disabledExtensions'] = disabled
  679. _write_page_config(config, app_dir, logger=logger)
  680. def _write_build_config(config, app_dir, logger):
  681. """Write the build config to the app dir.
  682. """
  683. _ensure_app_dirs(app_dir, logger)
  684. target = pjoin(app_dir, 'settings', 'build_config.json')
  685. with open(target, 'w') as fid:
  686. json.dump(config, fid, indent=4)
  687. def _write_page_config(config, app_dir, logger):
  688. """Write the build config to the app dir.
  689. """
  690. _ensure_app_dirs(app_dir, logger)
  691. target = pjoin(app_dir, 'settings', 'page_config.json')
  692. with open(target, 'w') as fid:
  693. json.dump(config, fid, indent=4)
  694. def _ensure_package(app_dir, logger=None, name=None, version=None):
  695. """Make sure the build dir is set up.
  696. """
  697. logger = logger or logging
  698. _ensure_app_dirs(app_dir, logger)
  699. # Look for mismatched version.
  700. staging = pjoin(app_dir, 'staging')
  701. pkg_path = pjoin(staging, 'package.json')
  702. if os.path.exists(pkg_path):
  703. with open(pkg_path) as fid:
  704. data = json.load(fid)
  705. if data['jupyterlab'].get('version', '') != version:
  706. shutil.rmtree(staging)
  707. os.makedirs(staging)
  708. for fname in ['index.app.js', 'webpack.config.js', '.npmrc']:
  709. dest = pjoin(staging, fname.replace('.app', ''))
  710. shutil.copy(pjoin(here, fname), dest)
  711. # Template the package.json file.
  712. data = _get_package_template(app_dir, logger)
  713. if version:
  714. data['jupyterlab']['version'] = version
  715. if name:
  716. data['jupyterlab']['name'] = name
  717. data['jupyterlab']['linkedPackages'] = _get_linked_packages(app_dir)
  718. pkg_path = pjoin(staging, 'package.json')
  719. with open(pkg_path, 'w') as fid:
  720. json.dump(data, fid, indent=4)
  721. def _ensure_app_dirs(app_dir, logger):
  722. """Ensure that the application directories exist"""
  723. dirs = ['extensions', 'settings', 'staging', 'schemas', 'themes']
  724. for dname in dirs:
  725. path = pjoin(app_dir, dname)
  726. if not osp.exists(path):
  727. try:
  728. os.makedirs(path)
  729. except OSError as e:
  730. if e.errno != errno.EEXIST:
  731. raise
  732. def _get_package_template(app_dir, logger):
  733. # Get the template the for package.json file.
  734. data = _get_core_data()
  735. extensions = _get_extensions(app_dir)
  736. # Handle extensions
  737. for (key, value) in extensions.items():
  738. # Reject incompatible extensions with a message.
  739. deps = value.get('dependencies', dict())
  740. errors = _validate_compatibility(key, deps, data)
  741. if errors:
  742. msg = _format_compatibility_errors(key, value['version'], errors)
  743. logger.warn(msg + '\n')
  744. continue
  745. data['dependencies'][key] = value['path']
  746. jlab_data = value['jupyterlab']
  747. for item in ['extension', 'mimeExtension']:
  748. ext = jlab_data.get(item, False)
  749. if not ext:
  750. continue
  751. if ext is True:
  752. ext = ''
  753. data['jupyterlab'][item + 's'][key] = ext
  754. # Handle linked packages.
  755. linked = _get_linked_packages(app_dir, logger)
  756. for (key, path) in linked.items():
  757. if key in extensions:
  758. continue
  759. data['dependencies'][key] = path
  760. # Handle uninstalled core extensions.
  761. for item in _get_uinstalled_core_extensions(app_dir):
  762. if item in data['jupyterlab']['extensions']:
  763. data['jupyterlab']['extensions'].pop(item)
  764. else:
  765. data['jupyterlab']['mimeExtensions'].pop(item)
  766. # Remove from dependencies as well.
  767. data['dependencies'].pop(item)
  768. return data
  769. def _is_extension(data):
  770. """Detect if a package is an extension using its metadata.
  771. """
  772. if 'jupyterlab' not in data:
  773. return False
  774. if not isinstance(data['jupyterlab'], dict):
  775. return False
  776. is_extension = data['jupyterlab'].get('extension', False)
  777. is_mime_extension = data['jupyterlab'].get('mimeExtension', False)
  778. return is_extension or is_mime_extension
  779. def _get_uinstalled_core_extensions(app_dir):
  780. """Get the uninstalled core extensions.
  781. """
  782. config = _get_build_config(app_dir)
  783. return config.get('uninstalled_core_extensions', [])
  784. def _validate_package(data, extension):
  785. """Validate package.json data.
  786. """
  787. msg = '%s is not a valid JupyterLab extension' % extension
  788. if not _is_extension(data):
  789. raise ValueError(msg)
  790. def _get_disabled(app_dir):
  791. """Get the disabled extensions.
  792. """
  793. config = _get_page_config(app_dir)
  794. return config.get('disabledExtensions', [])
  795. def _get_core_extensions():
  796. """Get the core extensions.
  797. """
  798. data = _get_core_data()['jupyterlab']
  799. return list(data['extensions']) + list(data['mimeExtensions'])
  800. def _get_extensions(app_dir):
  801. """Get the extensions in a given app dir.
  802. """
  803. extensions = dict()
  804. # Get system level packages
  805. sys_path = pjoin(get_app_dir(), 'extensions')
  806. app_path = pjoin(app_dir, 'extensions')
  807. for target in glob.glob(pjoin(sys_path, '*.tgz')):
  808. location = 'app' if app_path == sys_path else 'system'
  809. data = _read_package(target)
  810. deps = data.get('dependencies', dict())
  811. extensions[data['name']] = dict(path=os.path.realpath(target),
  812. version=data['version'],
  813. jupyterlab=data['jupyterlab'],
  814. dependencies=deps,
  815. location=location)
  816. # Look in app_dir if different
  817. app_path = pjoin(app_dir, 'extensions')
  818. if app_path == sys_path or not os.path.exists(app_path):
  819. return extensions
  820. for target in glob.glob(pjoin(app_path, '*.tgz')):
  821. data = _read_package(target)
  822. deps = data.get('dependencies', dict())
  823. extensions[data['name']] = dict(path=os.path.realpath(target),
  824. version=data['version'],
  825. jupyterlab=data['jupyterlab'],
  826. dependencies=deps,
  827. location='app')
  828. return extensions
  829. def _get_linked_packages(app_dir=None, logger=None):
  830. """Get the linked packages metadata.
  831. """
  832. logger = logger or logging
  833. app_dir = get_app_dir(app_dir)
  834. config = _get_build_config(app_dir)
  835. linked = config.get('linked_packages', dict())
  836. dead = []
  837. for (name, path) in linked.items():
  838. if not os.path.exists(path):
  839. dead.append(name)
  840. if dead:
  841. extensions = _get_extensions(app_dir)
  842. for name in dead:
  843. path = linked[name]
  844. if name in extensions:
  845. uninstall_extension(name)
  846. logger.warn('**Note: Removing dead linked extension "%s"' % name)
  847. else:
  848. logger.warn('**Note: Removing dead linked package "%s"' % name)
  849. del linked[name]
  850. if dead:
  851. config['linked_packages'] = linked
  852. _write_build_config(config, app_dir, logger=logger)
  853. return config.get('linked_packages', dict())
  854. def _read_package(target):
  855. """Read the package data in a given target tarball.
  856. """
  857. tar = tarfile.open(target, "r:gz")
  858. f = tar.extractfile('package/package.json')
  859. data = json.loads(f.read().decode('utf8'))
  860. tar.close()
  861. return data
  862. def _copy_tar_files(fname, source, dest):
  863. """Copy the files from a target path to the destination.
  864. """
  865. tar = tarfile.open(fname, "r:gz")
  866. subdir_and_files = [
  867. tarinfo for tarinfo in tar.getmembers()
  868. if tarinfo.name.startswith('package/' + source)
  869. ]
  870. offset = len('package/' + source + '/')
  871. for member in subdir_and_files:
  872. member.path = member.path[offset:]
  873. tar.extractall(path=dest, members=subdir_and_files)
  874. tar.close()
  875. def _normalize_path(extension):
  876. """Normalize a given extension if it is a path.
  877. """
  878. extension = osp.expanduser(extension)
  879. if osp.exists(extension):
  880. extension = osp.abspath(extension)
  881. return extension
  882. def _tarsum(input_file):
  883. """
  884. Compute the recursive sha sum of a tar file.
  885. """
  886. tar = tarfile.open(input_file, "r:gz")
  887. chunk_size = 100 * 1024
  888. for member in tar:
  889. if not member.isfile():
  890. continue
  891. f = tar.extractfile(member)
  892. h = hashlib.new("sha256")
  893. data = f.read(chunk_size)
  894. while data:
  895. h.update(data)
  896. data = f.read(chunk_size)
  897. return h.hexdigest()