lock.py 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175
  1. import sys
  2. from optparse import Values
  3. from pathlib import Path
  4. from pip._vendor import tomli_w
  5. from pip._vendor.packaging.pylock import is_valid_pylock_path
  6. from pip._internal.cache import WheelCache
  7. from pip._internal.cli import cmdoptions
  8. from pip._internal.cli.req_command import (
  9. RequirementCommand,
  10. with_cleanup,
  11. )
  12. from pip._internal.cli.status_codes import SUCCESS
  13. from pip._internal.operations.build.build_tracker import get_build_tracker
  14. from pip._internal.utils.logging import getLogger
  15. from pip._internal.utils.misc import (
  16. get_pip_version,
  17. )
  18. from pip._internal.utils.pylock import pylock_from_install_requirements
  19. from pip._internal.utils.temp_dir import TempDirectory
  20. logger = getLogger(__name__)
  21. class LockCommand(RequirementCommand):
  22. """
  23. EXPERIMENTAL - Lock packages and their dependencies from:
  24. - PyPI (and other indexes) using requirement specifiers.
  25. - VCS project urls.
  26. - Local project directories.
  27. - Local or remote source archives.
  28. pip also supports locking from "requirements files", which provide an easy
  29. way to specify a whole environment to be installed.
  30. The generated lock file is only guaranteed to be valid for the current
  31. python version and platform.
  32. """
  33. usage = """
  34. %prog [options] [-e] <local project path> ...
  35. %prog [options] <requirement specifier> [package-index-options] ...
  36. %prog [options] -r <requirements file> [package-index-options] ...
  37. %prog [options] <archive url/path> ..."""
  38. def add_options(self) -> None:
  39. self.cmd_opts.add_option(
  40. cmdoptions.PipOption(
  41. "--output",
  42. "-o",
  43. dest="output_file",
  44. metavar="path",
  45. type="path",
  46. default="pylock.toml",
  47. help="Lock file name (default=pylock.toml). Use - for stdout.",
  48. )
  49. )
  50. self.cmd_opts.add_option(cmdoptions.requirements())
  51. self.cmd_opts.add_option(cmdoptions.requirements_from_scripts())
  52. self.cmd_opts.add_option(cmdoptions.constraints())
  53. self.cmd_opts.add_option(cmdoptions.build_constraints())
  54. self.cmd_opts.add_option(cmdoptions.no_deps())
  55. self.cmd_opts.add_option(cmdoptions.editable())
  56. self.cmd_opts.add_option(cmdoptions.src())
  57. self.cmd_opts.add_option(cmdoptions.ignore_requires_python())
  58. self.cmd_opts.add_option(cmdoptions.no_build_isolation())
  59. self.cmd_opts.add_option(cmdoptions.use_pep517())
  60. self.cmd_opts.add_option(cmdoptions.check_build_deps())
  61. self.cmd_opts.add_option(cmdoptions.config_settings())
  62. self.cmd_opts.add_option(cmdoptions.require_hashes())
  63. self.cmd_opts.add_option(cmdoptions.progress_bar())
  64. index_opts = cmdoptions.make_option_group(
  65. cmdoptions.index_group,
  66. self.parser,
  67. )
  68. selection_opts = cmdoptions.make_option_group(
  69. cmdoptions.package_selection_group,
  70. self.parser,
  71. )
  72. self.parser.insert_option_group(0, index_opts)
  73. self.parser.insert_option_group(0, selection_opts)
  74. self.parser.insert_option_group(0, self.cmd_opts)
  75. @with_cleanup
  76. def run(self, options: Values, args: list[str]) -> int:
  77. logger.verbose("Using %s", get_pip_version())
  78. logger.warning(
  79. "pip lock is currently an experimental command. "
  80. "It may be removed/changed in a future release "
  81. "without prior warning."
  82. )
  83. cmdoptions.check_build_constraints(options)
  84. cmdoptions.check_release_control_exclusive(options)
  85. session = self.get_default_session(options)
  86. finder = self._build_package_finder(
  87. options=options,
  88. session=session,
  89. ignore_requires_python=options.ignore_requires_python,
  90. )
  91. build_tracker = self.enter_context(get_build_tracker())
  92. directory = TempDirectory(
  93. delete=not options.no_clean,
  94. kind="install",
  95. globally_managed=True,
  96. )
  97. reqs = self.get_requirements(args, options, finder, session)
  98. wheel_cache = WheelCache(options.cache_dir)
  99. # Only when installing is it permitted to use PEP 660.
  100. # In other circumstances (pip wheel, pip download) we generate
  101. # regular (i.e. non editable) metadata and wheels.
  102. for req in reqs:
  103. req.permit_editable_wheels = True
  104. preparer = self.make_requirement_preparer(
  105. temp_build_dir=directory,
  106. options=options,
  107. build_tracker=build_tracker,
  108. session=session,
  109. finder=finder,
  110. use_user_site=False,
  111. verbosity=self.verbosity,
  112. )
  113. resolver = self.make_resolver(
  114. preparer=preparer,
  115. finder=finder,
  116. options=options,
  117. wheel_cache=wheel_cache,
  118. use_user_site=False,
  119. ignore_installed=True,
  120. ignore_requires_python=options.ignore_requires_python,
  121. upgrade_strategy="to-satisfy-only",
  122. )
  123. self.trace_basic_info(finder)
  124. requirement_set = resolver.resolve(reqs, check_supported_wheels=True)
  125. if options.output_file == "-":
  126. base_dir = Path.cwd()
  127. else:
  128. output_file_path = Path(options.output_file)
  129. if not is_valid_pylock_path(output_file_path):
  130. logger.warning(
  131. "%s is not a valid lock file name.",
  132. output_file_path,
  133. )
  134. base_dir = output_file_path.parent
  135. pylock = pylock_from_install_requirements(
  136. requirement_set.requirements.values(), base_dir=base_dir
  137. )
  138. pylock_toml = tomli_w.dumps(pylock.to_dict())
  139. if options.output_file == "-":
  140. sys.stdout.write(pylock_toml)
  141. else:
  142. output_file_path.write_text(pylock_toml, encoding="utf-8")
  143. return SUCCESS