|
|
"""Implementation of packaging-related magic functions.
|
|
|
"""
|
|
|
#-----------------------------------------------------------------------------
|
|
|
# Copyright (c) 2018 The IPython Development Team.
|
|
|
#
|
|
|
# Distributed under the terms of the Modified BSD License.
|
|
|
#
|
|
|
# The full license is in the file COPYING.txt, distributed with this software.
|
|
|
#-----------------------------------------------------------------------------
|
|
|
|
|
|
import re
|
|
|
import shlex
|
|
|
import sys
|
|
|
from pathlib import Path
|
|
|
|
|
|
from IPython.core.magic import Magics, magics_class, line_magic
|
|
|
|
|
|
|
|
|
def _is_conda_environment():
|
|
|
"""Return True if the current Python executable is in a conda env"""
|
|
|
# TODO: does this need to change on windows?
|
|
|
return Path(sys.prefix, "conda-meta", "history").exists()
|
|
|
|
|
|
|
|
|
def _get_conda_executable():
|
|
|
"""Find the path to the conda executable"""
|
|
|
# Check if there is a conda executable in the same directory as the Python executable.
|
|
|
# This is the case within conda's root environment.
|
|
|
conda = Path(sys.executable).parent / "conda"
|
|
|
if conda.is_file():
|
|
|
return str(conda)
|
|
|
|
|
|
# Otherwise, attempt to extract the executable from conda history.
|
|
|
# This applies in any conda environment.
|
|
|
history = Path(sys.prefix, "conda-meta", "history").read_text()
|
|
|
match = re.search(
|
|
|
r"^#\s*cmd:\s*(?P<command>.*conda)\s[create|install]",
|
|
|
history,
|
|
|
flags=re.MULTILINE,
|
|
|
)
|
|
|
if match:
|
|
|
return match.groupdict()["command"]
|
|
|
|
|
|
# Fallback: assume conda is available on the system path.
|
|
|
return "conda"
|
|
|
|
|
|
|
|
|
CONDA_COMMANDS_REQUIRING_PREFIX = {
|
|
|
'install', 'list', 'remove', 'uninstall', 'update', 'upgrade',
|
|
|
}
|
|
|
CONDA_COMMANDS_REQUIRING_YES = {
|
|
|
'install', 'remove', 'uninstall', 'update', 'upgrade',
|
|
|
}
|
|
|
CONDA_ENV_FLAGS = {'-p', '--prefix', '-n', '--name'}
|
|
|
CONDA_YES_FLAGS = {'-y', '--y'}
|
|
|
|
|
|
|
|
|
@magics_class
|
|
|
class PackagingMagics(Magics):
|
|
|
"""Magics related to packaging & installation"""
|
|
|
|
|
|
@line_magic
|
|
|
def pip(self, line):
|
|
|
"""Run the pip package manager within the current kernel.
|
|
|
|
|
|
Usage:
|
|
|
%pip install [pkgs]
|
|
|
"""
|
|
|
python = sys.executable
|
|
|
if sys.platform == "win32":
|
|
|
python = '"' + python + '"'
|
|
|
else:
|
|
|
python = shlex.quote(python)
|
|
|
|
|
|
self.shell.system(" ".join([python, "-m", "pip", line]))
|
|
|
|
|
|
print("Note: you may need to restart the kernel to use updated packages.")
|
|
|
|
|
|
@line_magic
|
|
|
def conda(self, line):
|
|
|
"""Run the conda package manager within the current kernel.
|
|
|
|
|
|
Usage:
|
|
|
%conda install [pkgs]
|
|
|
"""
|
|
|
if not _is_conda_environment():
|
|
|
raise ValueError("The python kernel does not appear to be a conda environment. "
|
|
|
"Please use ``%pip install`` instead.")
|
|
|
|
|
|
conda = _get_conda_executable()
|
|
|
args = shlex.split(line)
|
|
|
command = args[0] if len(args) > 0 else ""
|
|
|
args = args[1:] if len(args) > 1 else [""]
|
|
|
|
|
|
extra_args = []
|
|
|
|
|
|
# When the subprocess does not allow us to respond "yes" during the installation,
|
|
|
# we need to insert --yes in the argument list for some commands
|
|
|
stdin_disabled = getattr(self.shell, 'kernel', None) is not None
|
|
|
needs_yes = command in CONDA_COMMANDS_REQUIRING_YES
|
|
|
has_yes = set(args).intersection(CONDA_YES_FLAGS)
|
|
|
if stdin_disabled and needs_yes and not has_yes:
|
|
|
extra_args.append("--yes")
|
|
|
|
|
|
# Add --prefix to point conda installation to the current environment
|
|
|
needs_prefix = command in CONDA_COMMANDS_REQUIRING_PREFIX
|
|
|
has_prefix = set(args).intersection(CONDA_ENV_FLAGS)
|
|
|
if needs_prefix and not has_prefix:
|
|
|
extra_args.extend(["--prefix", sys.prefix])
|
|
|
|
|
|
self.shell.system(' '.join([conda, command] + extra_args + args))
|
|
|
print("\nNote: you may need to restart the kernel to use updated packages.")
|
|
|
|