| 
									
										
										
										
											2021-01-20 17:07:21 +01:00
										 |  |  | import collections | 
					
						
							| 
									
										
										
										
											2013-11-11 22:11:55 +10:00
										 |  |  | import os | 
					
						
							|  |  |  | import os.path | 
					
						
							| 
									
										
										
										
											2021-01-20 17:07:21 +01:00
										 |  |  | import subprocess | 
					
						
							| 
									
										
										
										
											2013-11-11 22:11:55 +10:00
										 |  |  | import sys | 
					
						
							| 
									
										
										
										
											2021-01-20 17:07:21 +01:00
										 |  |  | import sysconfig | 
					
						
							| 
									
										
										
										
											2013-11-11 22:11:55 +10:00
										 |  |  | import tempfile | 
					
						
							| 
									
										
										
										
											2019-09-13 13:01:20 -03:00
										 |  |  | from importlib import resources | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-11 22:11:55 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  | __all__ = ["version", "bootstrap"] | 
					
						
							| 
									
										
										
										
											2021-01-20 17:07:21 +01:00
										 |  |  | _PACKAGE_NAMES = ('setuptools', 'pip') | 
					
						
							| 
									
										
										
										
											2022-10-18 15:48:14 +01:00
										 |  |  | _SETUPTOOLS_VERSION = "65.5.0" | 
					
						
							| 
									
										
										
										
											2022-11-05 17:50:39 +00:00
										 |  |  | _PIP_VERSION = "22.3.1" | 
					
						
							| 
									
										
										
										
											2013-11-11 22:11:55 +10:00
										 |  |  | _PROJECTS = [ | 
					
						
							| 
									
										
										
										
											2020-06-15 21:16:48 +02:00
										 |  |  |     ("setuptools", _SETUPTOOLS_VERSION, "py3"), | 
					
						
							| 
									
										
										
										
											2021-01-30 16:42:36 +00:00
										 |  |  |     ("pip", _PIP_VERSION, "py3"), | 
					
						
							| 
									
										
										
										
											2013-11-11 22:11:55 +10:00
										 |  |  | ] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 17:07:21 +01:00
										 |  |  | # Packages bundled in ensurepip._bundled have wheel_name set. | 
					
						
							|  |  |  | # Packages from WHEEL_PKG_DIR have wheel_path set. | 
					
						
							|  |  |  | _Package = collections.namedtuple('Package', | 
					
						
							|  |  |  |                                   ('version', 'wheel_name', 'wheel_path')) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Directory of system wheel packages. Some Linux distribution packaging | 
					
						
							|  |  |  | # policies recommend against bundling dependencies. For example, Fedora | 
					
						
							|  |  |  | # installs wheel packages in the /usr/share/python-wheels/ directory and don't | 
					
						
							|  |  |  | # install the ensurepip._bundled package. | 
					
						
							|  |  |  | _WHEEL_PKG_DIR = sysconfig.get_config_var('WHEEL_PKG_DIR') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def _find_packages(path): | 
					
						
							|  |  |  |     packages = {} | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         filenames = os.listdir(path) | 
					
						
							|  |  |  |     except OSError: | 
					
						
							|  |  |  |         # Ignore: path doesn't exist or permission error | 
					
						
							|  |  |  |         filenames = () | 
					
						
							|  |  |  |     # Make the code deterministic if a directory contains multiple wheel files | 
					
						
							|  |  |  |     # of the same package, but don't attempt to implement correct version | 
					
						
							|  |  |  |     # comparison since this case should not happen. | 
					
						
							|  |  |  |     filenames = sorted(filenames) | 
					
						
							|  |  |  |     for filename in filenames: | 
					
						
							| 
									
										
										
										
											2021-10-06 00:30:38 +03:00
										 |  |  |         # filename is like 'pip-21.2.4-py3-none-any.whl' | 
					
						
							| 
									
										
										
										
											2021-01-20 17:07:21 +01:00
										 |  |  |         if not filename.endswith(".whl"): | 
					
						
							|  |  |  |             continue | 
					
						
							|  |  |  |         for name in _PACKAGE_NAMES: | 
					
						
							|  |  |  |             prefix = name + '-' | 
					
						
							|  |  |  |             if filename.startswith(prefix): | 
					
						
							|  |  |  |                 break | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             continue | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-06 00:30:38 +03:00
										 |  |  |         # Extract '21.2.4' from 'pip-21.2.4-py3-none-any.whl' | 
					
						
							| 
									
										
										
										
											2021-01-20 17:07:21 +01:00
										 |  |  |         version = filename.removeprefix(prefix).partition('-')[0] | 
					
						
							|  |  |  |         wheel_path = os.path.join(path, filename) | 
					
						
							|  |  |  |         packages[name] = _Package(version, None, wheel_path) | 
					
						
							|  |  |  |     return packages | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def _get_packages(): | 
					
						
							|  |  |  |     global _PACKAGES, _WHEEL_PKG_DIR | 
					
						
							|  |  |  |     if _PACKAGES is not None: | 
					
						
							|  |  |  |         return _PACKAGES | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     packages = {} | 
					
						
							|  |  |  |     for name, version, py_tag in _PROJECTS: | 
					
						
							|  |  |  |         wheel_name = f"{name}-{version}-{py_tag}-none-any.whl" | 
					
						
							|  |  |  |         packages[name] = _Package(version, wheel_name, None) | 
					
						
							|  |  |  |     if _WHEEL_PKG_DIR: | 
					
						
							|  |  |  |         dir_packages = _find_packages(_WHEEL_PKG_DIR) | 
					
						
							|  |  |  |         # only used the wheel package directory if all packages are found there | 
					
						
							|  |  |  |         if all(name in dir_packages for name in _PACKAGE_NAMES): | 
					
						
							|  |  |  |             packages = dir_packages | 
					
						
							|  |  |  |     _PACKAGES = packages | 
					
						
							|  |  |  |     return packages | 
					
						
							|  |  |  | _PACKAGES = None | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-11 22:11:55 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-30 17:15:09 +10:00
										 |  |  | def _run_pip(args, additional_paths=None): | 
					
						
							| 
									
										
										
										
											2021-06-06 21:14:47 -05:00
										 |  |  |     # Run the bootstrapping in a subprocess to avoid leaking any state that happens | 
					
						
							|  |  |  |     # after pip has executed. Particularly, this avoids the case when pip holds onto | 
					
						
							| 
									
										
										
										
											2020-10-04 17:45:31 +01:00
										 |  |  |     # the files in *additional_paths*, preventing us to remove them at the end of the | 
					
						
							|  |  |  |     # invocation. | 
					
						
							|  |  |  |     code = f"""
 | 
					
						
							|  |  |  | import runpy | 
					
						
							|  |  |  | import sys | 
					
						
							|  |  |  | sys.path = {additional_paths or []} + sys.path | 
					
						
							|  |  |  | sys.argv[1:] = {args} | 
					
						
							|  |  |  | runpy.run_module("pip", run_name="__main__", alter_sys=True) | 
					
						
							|  |  |  | """
 | 
					
						
							| 
									
										
										
										
											2022-07-05 10:23:44 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     cmd = [ | 
					
						
							|  |  |  |         sys.executable, | 
					
						
							|  |  |  |         '-W', | 
					
						
							|  |  |  |         'ignore::DeprecationWarning', | 
					
						
							|  |  |  |         '-c', | 
					
						
							|  |  |  |         code, | 
					
						
							|  |  |  |     ] | 
					
						
							|  |  |  |     if sys.flags.isolated: | 
					
						
							|  |  |  |         # run code in isolated mode if currently running isolated | 
					
						
							|  |  |  |         cmd.insert(1, '-I') | 
					
						
							|  |  |  |     return subprocess.run(cmd, check=True).returncode | 
					
						
							| 
									
										
										
										
											2013-11-11 22:11:55 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def version(): | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     Returns a string specifying the bundled version of pip. | 
					
						
							|  |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2021-01-20 17:07:21 +01:00
										 |  |  |     return _get_packages()['pip'].version | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-11 22:11:55 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-04 23:02:36 +10:00
										 |  |  | def _disable_pip_configuration_settings(): | 
					
						
							| 
									
										
										
										
											2013-12-23 17:39:12 +10:00
										 |  |  |     # We deliberately ignore all pip environment variables | 
					
						
							|  |  |  |     # when invoking pip | 
					
						
							|  |  |  |     # See http://bugs.python.org/issue19734 for details | 
					
						
							|  |  |  |     keys_to_remove = [k for k in os.environ if k.startswith("PIP_")] | 
					
						
							|  |  |  |     for k in keys_to_remove: | 
					
						
							|  |  |  |         del os.environ[k] | 
					
						
							| 
									
										
										
										
											2014-02-04 23:02:36 +10:00
										 |  |  |     # We also ignore the settings in the default pip configuration file | 
					
						
							|  |  |  |     # See http://bugs.python.org/issue20053 for details | 
					
						
							|  |  |  |     os.environ['PIP_CONFIG_FILE'] = os.devnull | 
					
						
							| 
									
										
										
										
											2013-12-23 17:39:12 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-11 22:11:55 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  | def bootstrap(*, root=None, upgrade=False, user=False, | 
					
						
							|  |  |  |               altinstall=False, default_pip=False, | 
					
						
							|  |  |  |               verbosity=0): | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     Bootstrap pip into the current Python installation (or the given root | 
					
						
							|  |  |  |     directory). | 
					
						
							| 
									
										
										
										
											2013-12-23 16:16:07 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-21 13:07:45 +03:00
										 |  |  |     Note that calling this function will alter both sys.path and os.environ. | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     # Discard the return value | 
					
						
							|  |  |  |     _bootstrap(root=root, upgrade=upgrade, user=user, | 
					
						
							|  |  |  |                altinstall=altinstall, default_pip=default_pip, | 
					
						
							|  |  |  |                verbosity=verbosity) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def _bootstrap(*, root=None, upgrade=False, user=False, | 
					
						
							|  |  |  |               altinstall=False, default_pip=False, | 
					
						
							|  |  |  |               verbosity=0): | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     Bootstrap pip into the current Python installation (or the given root | 
					
						
							|  |  |  |     directory). Returns pip command status code. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-23 16:16:07 +10:00
										 |  |  |     Note that calling this function will alter both sys.path and os.environ. | 
					
						
							| 
									
										
										
										
											2013-11-11 22:11:55 +10:00
										 |  |  |     """
 | 
					
						
							|  |  |  |     if altinstall and default_pip: | 
					
						
							|  |  |  |         raise ValueError("Cannot use altinstall and default_pip together") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-24 08:42:54 -07:00
										 |  |  |     sys.audit("ensurepip.bootstrap", root) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-04 23:02:36 +10:00
										 |  |  |     _disable_pip_configuration_settings() | 
					
						
							| 
									
										
										
										
											2013-12-23 16:16:07 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-11 22:11:55 +10:00
										 |  |  |     # By default, installing pip and setuptools installs all of the | 
					
						
							|  |  |  |     # following scripts (X.Y == running Python version): | 
					
						
							|  |  |  |     # | 
					
						
							|  |  |  |     #   pip, pipX, pipX.Y, easy_install, easy_install-X.Y | 
					
						
							|  |  |  |     # | 
					
						
							|  |  |  |     # pip 1.5+ allows ensurepip to request that some of those be left out | 
					
						
							|  |  |  |     if altinstall: | 
					
						
							|  |  |  |         # omit pip, pipX and easy_install | 
					
						
							|  |  |  |         os.environ["ENSUREPIP_OPTIONS"] = "altinstall" | 
					
						
							|  |  |  |     elif not default_pip: | 
					
						
							|  |  |  |         # omit pip and easy_install | 
					
						
							|  |  |  |         os.environ["ENSUREPIP_OPTIONS"] = "install" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     with tempfile.TemporaryDirectory() as tmpdir: | 
					
						
							|  |  |  |         # Put our bundled wheels into a temporary directory and construct the | 
					
						
							|  |  |  |         # additional paths that need added to sys.path | 
					
						
							|  |  |  |         additional_paths = [] | 
					
						
							| 
									
										
										
										
											2021-01-20 17:07:21 +01:00
										 |  |  |         for name, package in _get_packages().items(): | 
					
						
							|  |  |  |             if package.wheel_name: | 
					
						
							|  |  |  |                 # Use bundled wheel package | 
					
						
							|  |  |  |                 wheel_name = package.wheel_name | 
					
						
							| 
									
										
										
										
											2021-06-06 21:14:47 -05:00
										 |  |  |                 wheel_path = resources.files("ensurepip") / "_bundled" / wheel_name | 
					
						
							|  |  |  |                 whl = wheel_path.read_bytes() | 
					
						
							| 
									
										
										
										
											2021-01-20 17:07:21 +01:00
										 |  |  |             else: | 
					
						
							|  |  |  |                 # Use the wheel package directory | 
					
						
							|  |  |  |                 with open(package.wheel_path, "rb") as fp: | 
					
						
							|  |  |  |                     whl = fp.read() | 
					
						
							|  |  |  |                 wheel_name = os.path.basename(package.wheel_path) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             filename = os.path.join(tmpdir, wheel_name) | 
					
						
							|  |  |  |             with open(filename, "wb") as fp: | 
					
						
							| 
									
										
										
										
											2013-11-11 22:11:55 +10:00
										 |  |  |                 fp.write(whl) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 17:07:21 +01:00
										 |  |  |             additional_paths.append(filename) | 
					
						
							| 
									
										
										
										
											2013-11-11 22:11:55 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # Construct the arguments to be passed to the pip command | 
					
						
							| 
									
										
										
										
											2020-06-15 19:28:46 +02:00
										 |  |  |         args = ["install", "--no-cache-dir", "--no-index", "--find-links", tmpdir] | 
					
						
							| 
									
										
										
										
											2013-11-11 22:11:55 +10:00
										 |  |  |         if root: | 
					
						
							|  |  |  |             args += ["--root", root] | 
					
						
							|  |  |  |         if upgrade: | 
					
						
							|  |  |  |             args += ["--upgrade"] | 
					
						
							|  |  |  |         if user: | 
					
						
							|  |  |  |             args += ["--user"] | 
					
						
							|  |  |  |         if verbosity: | 
					
						
							|  |  |  |             args += ["-" + "v" * verbosity] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 17:07:21 +01:00
										 |  |  |         return _run_pip([*args, *_PACKAGE_NAMES], additional_paths) | 
					
						
							| 
									
										
										
										
											2013-11-30 17:15:09 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-23 18:20:34 +10:00
										 |  |  | def _uninstall_helper(*, verbosity=0): | 
					
						
							| 
									
										
										
										
											2013-12-23 17:39:12 +10:00
										 |  |  |     """Helper to support a clean default uninstall process on Windows
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Note that calling this function may alter os.environ. | 
					
						
							|  |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2013-11-30 17:15:09 +10:00
										 |  |  |     # Nothing to do if pip was never installed, or has been removed | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         import pip | 
					
						
							|  |  |  |     except ImportError: | 
					
						
							|  |  |  |         return | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 17:07:21 +01:00
										 |  |  |     # If the installed pip version doesn't match the available one, | 
					
						
							|  |  |  |     # leave it alone | 
					
						
							|  |  |  |     available_version = version() | 
					
						
							|  |  |  |     if pip.__version__ != available_version: | 
					
						
							|  |  |  |         print(f"ensurepip will only uninstall a matching version " | 
					
						
							|  |  |  |               f"({pip.__version__!r} installed, " | 
					
						
							|  |  |  |               f"{available_version!r} available)", | 
					
						
							|  |  |  |               file=sys.stderr) | 
					
						
							| 
									
										
										
										
											2014-02-28 23:35:05 +10:00
										 |  |  |         return | 
					
						
							| 
									
										
										
										
											2013-11-30 17:15:09 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-04 23:02:36 +10:00
										 |  |  |     _disable_pip_configuration_settings() | 
					
						
							| 
									
										
										
										
											2013-12-23 17:39:12 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-30 17:15:09 +10:00
										 |  |  |     # Construct the arguments to be passed to the pip command | 
					
						
							| 
									
										
										
										
											2015-06-02 10:37:08 -04:00
										 |  |  |     args = ["uninstall", "-y", "--disable-pip-version-check"] | 
					
						
							| 
									
										
										
										
											2013-11-30 17:15:09 +10:00
										 |  |  |     if verbosity: | 
					
						
							|  |  |  |         args += ["-" + "v" * verbosity] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 17:07:21 +01:00
										 |  |  |     return _run_pip([*args, *reversed(_PACKAGE_NAMES)]) | 
					
						
							| 
									
										
										
										
											2013-12-23 18:20:34 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def _main(argv=None): | 
					
						
							|  |  |  |     import argparse | 
					
						
							|  |  |  |     parser = argparse.ArgumentParser(prog="python -m ensurepip") | 
					
						
							|  |  |  |     parser.add_argument( | 
					
						
							|  |  |  |         "--version", | 
					
						
							|  |  |  |         action="version", | 
					
						
							|  |  |  |         version="pip {}".format(version()), | 
					
						
							|  |  |  |         help="Show the version of pip that is bundled with this Python.", | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     parser.add_argument( | 
					
						
							|  |  |  |         "-v", "--verbose", | 
					
						
							|  |  |  |         action="count", | 
					
						
							|  |  |  |         default=0, | 
					
						
							|  |  |  |         dest="verbosity", | 
					
						
							|  |  |  |         help=("Give more output. Option is additive, and can be used up to 3 " | 
					
						
							|  |  |  |               "times."), | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     parser.add_argument( | 
					
						
							|  |  |  |         "-U", "--upgrade", | 
					
						
							|  |  |  |         action="store_true", | 
					
						
							|  |  |  |         default=False, | 
					
						
							|  |  |  |         help="Upgrade pip and dependencies, even if already installed.", | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     parser.add_argument( | 
					
						
							|  |  |  |         "--user", | 
					
						
							|  |  |  |         action="store_true", | 
					
						
							|  |  |  |         default=False, | 
					
						
							|  |  |  |         help="Install using the user scheme.", | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     parser.add_argument( | 
					
						
							|  |  |  |         "--root", | 
					
						
							|  |  |  |         default=None, | 
					
						
							|  |  |  |         help="Install everything relative to this alternate root directory.", | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     parser.add_argument( | 
					
						
							|  |  |  |         "--altinstall", | 
					
						
							|  |  |  |         action="store_true", | 
					
						
							|  |  |  |         default=False, | 
					
						
							| 
									
										
										
										
											2018-05-30 09:45:55 +02:00
										 |  |  |         help=("Make an alternate install, installing only the X.Y versioned " | 
					
						
							|  |  |  |               "scripts (Default: pipX, pipX.Y, easy_install-X.Y)."), | 
					
						
							| 
									
										
										
										
											2013-12-23 18:20:34 +10:00
										 |  |  |     ) | 
					
						
							|  |  |  |     parser.add_argument( | 
					
						
							|  |  |  |         "--default-pip", | 
					
						
							|  |  |  |         action="store_true", | 
					
						
							|  |  |  |         default=False, | 
					
						
							|  |  |  |         help=("Make a default pip install, installing the unqualified pip " | 
					
						
							| 
									
										
										
										
											2018-05-30 09:45:55 +02:00
										 |  |  |               "and easy_install in addition to the versioned scripts."), | 
					
						
							| 
									
										
										
										
											2013-12-23 18:20:34 +10:00
										 |  |  |     ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     args = parser.parse_args(argv) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-21 13:07:45 +03:00
										 |  |  |     return _bootstrap( | 
					
						
							| 
									
										
										
										
											2013-12-23 18:20:34 +10:00
										 |  |  |         root=args.root, | 
					
						
							|  |  |  |         upgrade=args.upgrade, | 
					
						
							|  |  |  |         user=args.user, | 
					
						
							|  |  |  |         verbosity=args.verbosity, | 
					
						
							|  |  |  |         altinstall=args.altinstall, | 
					
						
							|  |  |  |         default_pip=args.default_pip, | 
					
						
							|  |  |  |     ) |