diff --git a/docs/html/reference/pip_install.rst b/docs/html/reference/pip_install.rst index 96fd1c018ba..f56dc9a0e9f 100644 --- a/docs/html/reference/pip_install.rst +++ b/docs/html/reference/pip_install.rst @@ -694,10 +694,21 @@ does not satisfy the ``--require-hashes`` demand that every package have a local hash. +Local project installs +++++++++++++++++++++++ +pip supports installing local project in both regular mode and editable mode. +You can install local projects by specifying the project path to pip:: + +$ pip install path/to/SomeProject + +During regular installation, pip will copy the entire project directory to a temporary location and install from there. +The exception is that pip will exclude .tox and .nox directories present in the top level of the project from being copied. + + .. _`editable-installs`: "Editable" Installs -+++++++++++++++++++ +~~~~~~~~~~~~~~~~~~~ "Editable" installs are fundamentally `"setuptools develop mode" `_ diff --git a/news/6770.bugfix b/news/6770.bugfix new file mode 100644 index 00000000000..c0ab57ee109 --- /dev/null +++ b/news/6770.bugfix @@ -0,0 +1 @@ +Skip copying .tox and .nox directories to temporary build directories diff --git a/src/pip/_internal/download.py b/src/pip/_internal/download.py index 2c8627b099f..a4db2382438 100644 --- a/src/pip/_internal/download.py +++ b/src/pip/_internal/download.py @@ -967,12 +967,23 @@ def unpack_file_url( of the link file inside download_dir. """ link_path = url_to_path(link.url_without_fragment) - # If it's a url to a local directory if is_dir_url(link): + + def ignore(d, names): + # Pulling in those directories can potentially be very slow, + # exclude the following directories if they appear in the top + # level dir (and only it). + # See discussion at https://github.com/pypa/pip/pull/6770 + return ['.tox', '.nox'] if d == link_path else [] + if os.path.isdir(location): rmtree(location) - shutil.copytree(link_path, location, symlinks=True) + shutil.copytree(link_path, + location, + symlinks=True, + ignore=ignore) + if download_dir: logger.info('Link is a directory, ignoring download_dir') return diff --git a/tests/unit/test_download.py b/tests/unit/test_download.py index 24b725d5d9c..c0712b9a45e 100644 --- a/tests/unit/test_download.py +++ b/tests/unit/test_download.py @@ -431,6 +431,41 @@ def test_unpack_file_url_thats_a_dir(self, tmpdir, data): assert os.path.isdir(os.path.join(self.build_dir, 'fspkg')) +@pytest.mark.parametrize('exclude_dir', [ + '.nox', + '.tox' +]) +def test_unpack_file_url_excludes_expected_dirs(tmpdir, exclude_dir): + src_dir = tmpdir / 'src' + dst_dir = tmpdir / 'dst' + src_included_file = src_dir.joinpath('file.txt') + src_excluded_dir = src_dir.joinpath(exclude_dir) + src_excluded_file = src_dir.joinpath(exclude_dir, 'file.txt') + src_included_dir = src_dir.joinpath('subdir', exclude_dir) + + # set up source directory + src_excluded_dir.mkdir(parents=True) + src_included_dir.mkdir(parents=True) + src_included_file.touch() + src_excluded_file.touch() + + dst_included_file = dst_dir.joinpath('file.txt') + dst_excluded_dir = dst_dir.joinpath(exclude_dir) + dst_excluded_file = dst_dir.joinpath(exclude_dir, 'file.txt') + dst_included_dir = dst_dir.joinpath('subdir', exclude_dir) + + src_link = Link(path_to_url(src_dir)) + unpack_file_url( + src_link, + dst_dir, + download_dir=None + ) + assert not os.path.isdir(dst_excluded_dir) + assert not os.path.isfile(dst_excluded_file) + assert os.path.isfile(dst_included_file) + assert os.path.isdir(dst_included_dir) + + class TestSafeFileCache: """ The no_perms test are useless on Windows since SafeFileCache uses