download.py 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143
  1. from __future__ import absolute_import
  2. import logging
  3. import os
  4. from pip._internal.cli import cmdoptions
  5. from pip._internal.cli.cmdoptions import make_target_python
  6. from pip._internal.cli.req_command import RequirementCommand, with_cleanup
  7. from pip._internal.cli.status_codes import SUCCESS
  8. from pip._internal.req.req_tracker import get_requirement_tracker
  9. from pip._internal.utils.misc import ensure_dir, normalize_path, write_output
  10. from pip._internal.utils.temp_dir import TempDirectory
  11. from pip._internal.utils.typing import MYPY_CHECK_RUNNING
  12. if MYPY_CHECK_RUNNING:
  13. from optparse import Values
  14. from typing import List
  15. logger = logging.getLogger(__name__)
  16. class DownloadCommand(RequirementCommand):
  17. """
  18. Download packages from:
  19. - PyPI (and other indexes) using requirement specifiers.
  20. - VCS project urls.
  21. - Local project directories.
  22. - Local or remote source archives.
  23. pip also supports downloading from "requirements files", which provide
  24. an easy way to specify a whole environment to be downloaded.
  25. """
  26. usage = """
  27. %prog [options] <requirement specifier> [package-index-options] ...
  28. %prog [options] -r <requirements file> [package-index-options] ...
  29. %prog [options] <vcs project url> ...
  30. %prog [options] <local project path> ...
  31. %prog [options] <archive url/path> ..."""
  32. def add_options(self):
  33. # type: () -> None
  34. self.cmd_opts.add_option(cmdoptions.constraints())
  35. self.cmd_opts.add_option(cmdoptions.requirements())
  36. self.cmd_opts.add_option(cmdoptions.build_dir())
  37. self.cmd_opts.add_option(cmdoptions.no_deps())
  38. self.cmd_opts.add_option(cmdoptions.global_options())
  39. self.cmd_opts.add_option(cmdoptions.no_binary())
  40. self.cmd_opts.add_option(cmdoptions.only_binary())
  41. self.cmd_opts.add_option(cmdoptions.prefer_binary())
  42. self.cmd_opts.add_option(cmdoptions.src())
  43. self.cmd_opts.add_option(cmdoptions.pre())
  44. self.cmd_opts.add_option(cmdoptions.require_hashes())
  45. self.cmd_opts.add_option(cmdoptions.progress_bar())
  46. self.cmd_opts.add_option(cmdoptions.no_build_isolation())
  47. self.cmd_opts.add_option(cmdoptions.use_pep517())
  48. self.cmd_opts.add_option(cmdoptions.no_use_pep517())
  49. self.cmd_opts.add_option(
  50. '-d', '--dest', '--destination-dir', '--destination-directory',
  51. dest='download_dir',
  52. metavar='dir',
  53. default=os.curdir,
  54. help=("Download packages into <dir>."),
  55. )
  56. cmdoptions.add_target_python_options(self.cmd_opts)
  57. index_opts = cmdoptions.make_option_group(
  58. cmdoptions.index_group,
  59. self.parser,
  60. )
  61. self.parser.insert_option_group(0, index_opts)
  62. self.parser.insert_option_group(0, self.cmd_opts)
  63. @with_cleanup
  64. def run(self, options, args):
  65. # type: (Values, List[str]) -> int
  66. options.ignore_installed = True
  67. # editable doesn't really make sense for `pip download`, but the bowels
  68. # of the RequirementSet code require that property.
  69. options.editables = []
  70. cmdoptions.check_dist_restriction(options)
  71. options.download_dir = normalize_path(options.download_dir)
  72. ensure_dir(options.download_dir)
  73. session = self.get_default_session(options)
  74. target_python = make_target_python(options)
  75. finder = self._build_package_finder(
  76. options=options,
  77. session=session,
  78. target_python=target_python,
  79. )
  80. req_tracker = self.enter_context(get_requirement_tracker())
  81. directory = TempDirectory(
  82. delete=not options.no_clean,
  83. kind="download",
  84. globally_managed=True,
  85. )
  86. reqs = self.get_requirements(args, options, finder, session)
  87. preparer = self.make_requirement_preparer(
  88. temp_build_dir=directory,
  89. options=options,
  90. req_tracker=req_tracker,
  91. session=session,
  92. finder=finder,
  93. download_dir=options.download_dir,
  94. use_user_site=False,
  95. )
  96. resolver = self.make_resolver(
  97. preparer=preparer,
  98. finder=finder,
  99. options=options,
  100. py_version_info=options.python_version,
  101. )
  102. self.trace_basic_info(finder)
  103. requirement_set = resolver.resolve(
  104. reqs, check_supported_wheels=True
  105. )
  106. downloaded = [] # type: List[str]
  107. for req in requirement_set.requirements.values():
  108. if req.satisfied_by is None:
  109. assert req.name is not None
  110. preparer.save_linked_requirement(req)
  111. downloaded.append(req.name)
  112. if downloaded:
  113. write_output('Successfully downloaded %s', ' '.join(downloaded))
  114. return SUCCESS