From 4409187f710f4674abbc3aaf69312ab07801c2cb Mon Sep 17 00:00:00 2001 From: David Hotham Date: Tue, 14 Nov 2023 12:21:45 +0000 Subject: [PATCH 1/3] treat collections in packages as immutable justifying the use of a shallow copy in Package.clone() --- src/poetry/core/factory.py | 21 ++++--- src/poetry/core/packages/dependency.py | 11 ++-- src/poetry/core/packages/package.py | 70 ++++++++++++----------- src/poetry/core/packages/specification.py | 2 +- tests/packages/test_dependency.py | 4 +- tests/packages/test_package.py | 10 ++-- tests/packages/test_vcs_dependency.py | 6 +- 7 files changed, 68 insertions(+), 56 deletions(-) diff --git a/src/poetry/core/factory.py b/src/poetry/core/factory.py index aa86ecd4d..e085b0f01 100644 --- a/src/poetry/core/factory.py +++ b/src/poetry/core/factory.py @@ -19,6 +19,8 @@ if TYPE_CHECKING: from collections.abc import Mapping + from packaging.utils import NormalizedName + from poetry.core.packages.dependency import Dependency from poetry.core.packages.dependency_group import DependencyGroup from poetry.core.packages.project_package import ProjectPackage @@ -131,11 +133,13 @@ def configure_package( package.root_dir = root - for author in config.get("authors", []): - package.authors.append(combine_unicode(author)) + package.authors = [ + combine_unicode(author) for author in config.get("authors", []) + ] - for maintainer in config.get("maintainers", []): - package.maintainers.append(combine_unicode(maintainer)) + package.maintainers = [ + combine_unicode(maintainer) for maintainer in config.get("maintainers", []) + ] package.description = config.get("description", "") package.homepage = config.get("homepage") @@ -177,10 +181,11 @@ def configure_package( package=package, group="dev", dependencies=config["dev-dependencies"] ) + package_extras: dict[NormalizedName, list[Dependency]] = {} extras = config.get("extras", {}) for extra_name, requirements in extras.items(): extra_name = canonicalize_name(extra_name) - package.extras[extra_name] = [] + package_extras[extra_name] = [] # Checking for dependency for req in requirements: @@ -188,8 +193,10 @@ def configure_package( for dep in package.requires: if dep.name == req.name: - dep.in_extras.append(extra_name) - package.extras[extra_name].append(dep) + dep._in_extras = [*dep._in_extras, extra_name] + package_extras[extra_name].append(dep) + + package.extras = package_extras if "build" in config: build = config["build"] diff --git a/src/poetry/core/packages/dependency.py b/src/poetry/core/packages/dependency.py index e871a3705..61cce4079 100644 --- a/src/poetry/core/packages/dependency.py +++ b/src/poetry/core/packages/dependency.py @@ -7,6 +7,7 @@ from contextlib import suppress from pathlib import Path from typing import TYPE_CHECKING +from typing import Sequence from typing import TypeVar from packaging.utils import canonicalize_name @@ -79,7 +80,7 @@ def __init__( self._transitive_python_constraint: VersionConstraint | None = None self._transitive_marker: BaseMarker | None = None - self._in_extras: list[NormalizedName] = [] + self._in_extras: Sequence[NormalizedName] = [] self._activated = not self._optional @@ -184,13 +185,15 @@ def marker(self, marker: str | BaseMarker) -> None: # If we have extras, the dependency is optional self.deactivate() + new_in_extras = [] for or_ in markers["extra"]: for op, extra in or_: if op == "==": - self.in_extras.append(canonicalize_name(extra)) + new_in_extras.append(canonicalize_name(extra)) elif op == "" and "||" in extra: for _extra in extra.split(" || "): - self.in_extras.append(canonicalize_name(_extra)) + new_in_extras.append(canonicalize_name(_extra)) + self._in_extras = [*self._in_extras, *new_in_extras] # Recalculate python versions. self._python_versions = "*" @@ -235,7 +238,7 @@ def extras(self) -> frozenset[NormalizedName]: return self._features @property - def in_extras(self) -> list[NormalizedName]: + def in_extras(self) -> Sequence[NormalizedName]: return self._in_extras @property diff --git a/src/poetry/core/packages/package.py b/src/poetry/core/packages/package.py index ddd63564a..a61331c32 100644 --- a/src/poetry/core/packages/package.py +++ b/src/poetry/core/packages/package.py @@ -1,12 +1,13 @@ from __future__ import annotations -import copy import re import warnings from contextlib import contextmanager from typing import TYPE_CHECKING from typing import ClassVar +from typing import Mapping +from typing import Sequence from typing import TypeVar from poetry.core.constraints.version import parse_constraint @@ -97,26 +98,26 @@ def __init__( self.description = "" - self._authors: list[str] = [] - self._maintainers: list[str] = [] + self.authors: Sequence[str] = [] + self.maintainers: Sequence[str] = [] self.homepage: str | None = None self.repository_url: str | None = None self.documentation_url: str | None = None - self.keywords: list[str] = [] + self.keywords: Sequence[str] = [] self._license: License | None = None self.readmes: tuple[Path, ...] = () - self.extras: dict[NormalizedName, list[Dependency]] = {} + self.extras: Mapping[NormalizedName, Sequence[Dependency]] = {} - self._dependency_groups: dict[str, DependencyGroup] = {} + self._dependency_groups: Mapping[str, DependencyGroup] = {} # Category is heading towards deprecation. self._category = "main" - self.files: list[dict[str, str]] = [] + self.files: Sequence[Mapping[str, str]] = [] self.optional = False - self.classifiers: list[str] = [] + self.classifiers: Sequence[str] = [] self._python_versions = "*" self._python_constraint = parse_constraint("*") @@ -177,10 +178,6 @@ def full_pretty_version(self) -> str: ref = self._source_resolved_reference or self._source_reference return f"{self.pretty_version} {ref}" - @property - def authors(self) -> list[str]: - return self._authors - @property def author_name(self) -> str | None: return self._get_author()["name"] @@ -189,10 +186,6 @@ def author_name(self) -> str | None: def author_email(self) -> str | None: return self._get_author()["email"] - @property - def maintainers(self) -> list[str]: - return self._maintainers - @property def maintainer_name(self) -> str | None: return self._get_maintainer()["name"] @@ -238,10 +231,10 @@ def _set_version(self, version: str | Version) -> None: self._version = version def _get_author(self) -> dict[str, str | None]: - if not self._authors: + if not self.authors: return {"name": None, "email": None} - m = AUTHOR_REGEX.match(self._authors[0]) + m = AUTHOR_REGEX.match(self.authors[0]) if m is None: raise ValueError( @@ -255,10 +248,10 @@ def _get_author(self) -> dict[str, str | None]: return {"name": name, "email": email} def _get_maintainer(self) -> dict[str, str | None]: - if not self._maintainers: + if not self.maintainers: return {"name": None, "email": None} - m = AUTHOR_REGEX.match(self._maintainers[0]) + m = AUTHOR_REGEX.match(self.maintainers[0]) if m is None: raise ValueError( @@ -314,7 +307,7 @@ def license(self, value: str | License | None) -> None: def all_classifiers(self) -> list[str]: from poetry.core.constraints.version import Version - classifiers = copy.copy(self.classifiers) + classifiers = list(self.classifiers) # Automatically set python classifiers if self.python_versions == "*": @@ -439,7 +432,9 @@ def dependency_group_names(self, include_optional: bool = False) -> set[str]: } def add_dependency_group(self, group: DependencyGroup) -> None: - self._dependency_groups[group.name] = group + groups = dict(self._dependency_groups) + groups[group.name] = group + self._dependency_groups = groups def has_dependency_group(self, name: str) -> bool: return name in self._dependency_groups @@ -469,11 +464,14 @@ def without_dependency_groups(self: T, groups: Collection[str]) -> T: """ Returns a clone of the package with the given dependency groups excluded. """ - package = self.clone() + updated_groups = { + group_name: group + for group_name, group in self._dependency_groups.items() + if group_name not in groups + } - for group_name in groups: - if group_name in package._dependency_groups: - del package._dependency_groups[group_name] + package = self.clone() + package._dependency_groups = updated_groups return package @@ -481,11 +479,13 @@ def without_optional_dependency_groups(self: T) -> T: """ Returns a clone of the package without optional dependency groups. """ + updated_groups = { + group_name: group + for group_name, group in self._dependency_groups.items() + if not group.is_optional() + } package = self.clone() - - for group_name, group in self._dependency_groups.items(): - if group.is_optional(): - del package._dependency_groups[group_name] + package._dependency_groups = updated_groups return package @@ -500,11 +500,13 @@ def with_dependency_groups( If `only` is set to True, then only the given groups will be selected. """ + updated_groups = { + group_name: group + for group_name, group in self._dependency_groups.items() + if group_name in groups or not only and not group.is_optional() + } package = self.clone() - - for group_name, group in self._dependency_groups.items(): - if (only or group.is_optional()) and group_name not in groups: - del package._dependency_groups[group_name] + package._dependency_groups = updated_groups return package diff --git a/src/poetry/core/packages/specification.py b/src/poetry/core/packages/specification.py index fe95095d1..30c66030f 100644 --- a/src/poetry/core/packages/specification.py +++ b/src/poetry/core/packages/specification.py @@ -175,7 +175,7 @@ def is_same_package_as(self, other: PackageSpecification) -> bool: return self.is_same_source_as(other) def clone(self: T) -> T: - return copy.deepcopy(self) + return copy.copy(self) def with_features(self: T, features: Iterable[str]) -> T: package = self.clone() diff --git a/tests/packages/test_dependency.py b/tests/packages/test_dependency.py index 96c57c5a4..43deb823f 100644 --- a/tests/packages/test_dependency.py +++ b/tests/packages/test_dependency.py @@ -61,7 +61,7 @@ def test_to_pep_508_wilcard() -> None: def test_to_pep_508_in_extras() -> None: dependency = Dependency("Django", "^1.23") - dependency.in_extras.append(canonicalize_name("foo")) + dependency._in_extras = [canonicalize_name("foo")] result = dependency.to_pep_508() assert result == 'Django (>=1.23,<2.0) ; extra == "foo"' @@ -69,7 +69,7 @@ def test_to_pep_508_in_extras() -> None: result = dependency.to_pep_508(with_extras=False) assert result == "Django (>=1.23,<2.0)" - dependency.in_extras.append(canonicalize_name("bar")) + dependency._in_extras = [canonicalize_name("foo"), canonicalize_name("bar")] result = dependency.to_pep_508() assert result == 'Django (>=1.23,<2.0) ; extra == "foo" or extra == "bar"' diff --git a/tests/packages/test_package.py b/tests/packages/test_package.py index e50d77162..f94de4a4e 100644 --- a/tests/packages/test_package.py +++ b/tests/packages/test_package.py @@ -43,11 +43,11 @@ def package_with_groups() -> Package: def test_package_authors() -> None: package = Package("foo", "0.1.0") - package.authors.append("Sébastien Eustace ") + package.authors = ["Sébastien Eustace "] assert package.author_name == "Sébastien Eustace" assert package.author_email == "sebastien@eustace.io" - package.authors.insert(0, "John Doe") + package.authors = ["John Doe", *package.authors] assert package.author_name == "John Doe" assert package.author_email is None @@ -55,7 +55,7 @@ def test_package_authors() -> None: def test_package_authors_invalid() -> None: package = Package("foo", "0.1.0") - package.authors.insert(0, " None: package = Package("foo", "0.1.0") author = name if email is None else f"{name} <{email}>" - package.authors.insert(0, author) + package.authors = [author] assert package.author_name == name assert package.author_email == email @@ -109,7 +109,7 @@ def test_package_authors_valid(name: str, email: str | None) -> None: def test_package_author_names_invalid(name: str) -> None: package = Package("foo", "0.1.0") - package.authors.insert(0, name) + package.authors = [name] with pytest.raises(ValueError): package.author_name # noqa: B018 diff --git a/tests/packages/test_vcs_dependency.py b/tests/packages/test_vcs_dependency.py index 214dd90b4..062cd040e 100644 --- a/tests/packages/test_vcs_dependency.py +++ b/tests/packages/test_vcs_dependency.py @@ -119,7 +119,7 @@ def test_to_pep_508_in_extras() -> None: dependency = VCSDependency( "poetry", "git", "https://github.com/python-poetry/poetry.git" ) - dependency.in_extras.append(canonicalize_name("foo")) + dependency._in_extras = [canonicalize_name("foo")] expected = ( 'poetry @ git+https://github.com/python-poetry/poetry.git ; extra == "foo"' @@ -129,7 +129,7 @@ def test_to_pep_508_in_extras() -> None: dependency = VCSDependency( "poetry", "git", "https://github.com/python-poetry/poetry.git", extras=["bar"] ) - dependency.in_extras.append(canonicalize_name("foo")) + dependency._in_extras = [canonicalize_name("foo")] expected = ( 'poetry[bar] @ git+https://github.com/python-poetry/poetry.git ; extra == "foo"' @@ -140,7 +140,7 @@ def test_to_pep_508_in_extras() -> None: dependency = VCSDependency( "poetry", "git", "https://github.com/python-poetry/poetry.git", "b;ar;" ) - dependency.in_extras.append(canonicalize_name("foo;")) + dependency._in_extras = [canonicalize_name("foo;")] expected = ( "poetry @ git+https://github.com/python-poetry/poetry.git@b;ar; ; extra ==" From 93630bdee08373f1c0c889bcaa23a67fd2ec8c54 Mon Sep 17 00:00:00 2001 From: David Hotham Date: Fri, 17 Nov 2023 19:32:13 +0000 Subject: [PATCH 2/3] more immutable collections --- src/poetry/core/masonry/utils/module.py | 8 ++++---- src/poetry/core/packages/project_package.py | 12 +++++++----- 2 files changed, 11 insertions(+), 9 deletions(-) diff --git a/src/poetry/core/masonry/utils/module.py b/src/poetry/core/masonry/utils/module.py index 4dc201b17..06064be68 100644 --- a/src/poetry/core/masonry/utils/module.py +++ b/src/poetry/core/masonry/utils/module.py @@ -3,6 +3,8 @@ from pathlib import Path from typing import TYPE_CHECKING from typing import Any +from typing import Mapping +from typing import Sequence if TYPE_CHECKING: @@ -18,8 +20,8 @@ def __init__( self, name: str, directory: str = ".", - packages: list[dict[str, Any]] | None = None, - includes: list[dict[str, Any]] | None = None, + packages: Sequence[Mapping[str, Any]] = (), + includes: Sequence[Mapping[str, Any]] = (), ) -> None: from poetry.core.masonry.utils.include import Include from poetry.core.masonry.utils.package_include import PackageInclude @@ -30,8 +32,6 @@ def __init__( self._is_package = False self._path = Path(directory) self._includes: list[Include] = [] - packages = packages or [] - includes = includes or [] if not packages: # It must exist either as a .py file or a directory, but not both diff --git a/src/poetry/core/packages/project_package.py b/src/poetry/core/packages/project_package.py index a6436bc17..f9af3bd19 100644 --- a/src/poetry/core/packages/project_package.py +++ b/src/poetry/core/packages/project_package.py @@ -4,6 +4,8 @@ from typing import TYPE_CHECKING from typing import Any +from typing import Mapping +from typing import Sequence from poetry.core.constraints.version import parse_constraint from poetry.core.version.markers import parse_marker @@ -34,11 +36,11 @@ def __init__( super().__init__(name, version) - self.build_config: dict[str, Any] = {} - self.packages: list[dict[str, Any]] = [] - self.include: list[dict[str, Any]] = [] - self.exclude: list[dict[str, Any]] = [] - self.custom_urls: dict[str, str] = {} + self.build_config: Mapping[str, Any] = {} + self.packages: Sequence[Mapping[str, Any]] = [] + self.include: Sequence[Mapping[str, Any]] = [] + self.exclude: Sequence[Mapping[str, Any]] = [] + self.custom_urls: Mapping[str, str] = {} if self._python_versions == "*": self._python_constraint = parse_constraint("~2.7 || >=3.4") From a281c5bd802b560c8117dfaf3e0d97e305cf6409 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Randy=20D=C3=B6ring?= <30527984+radoering@users.noreply.github.com> Date: Sat, 18 Nov 2023 07:19:02 +0100 Subject: [PATCH 3/3] add comment that attributes must be immutable --- src/poetry/core/packages/dependency.py | 3 +++ src/poetry/core/packages/directory_dependency.py | 2 ++ src/poetry/core/packages/file_dependency.py | 2 ++ src/poetry/core/packages/package.py | 3 +++ src/poetry/core/packages/path_dependency.py | 2 ++ src/poetry/core/packages/project_package.py | 3 +++ src/poetry/core/packages/specification.py | 3 +++ src/poetry/core/packages/url_dependency.py | 2 ++ src/poetry/core/packages/vcs_dependency.py | 2 ++ 9 files changed, 22 insertions(+) diff --git a/src/poetry/core/packages/dependency.py b/src/poetry/core/packages/dependency.py index 61cce4079..de554449d 100644 --- a/src/poetry/core/packages/dependency.py +++ b/src/poetry/core/packages/dependency.py @@ -62,6 +62,9 @@ def __init__( features=extras, ) + # Attributes must be immutable for clone() to be safe! + # (For performance reasons, clone only creates a copy instead of a deep copy). + self._constraint: VersionConstraint self._pretty_constraint: str self.constraint = constraint # type: ignore[assignment] diff --git a/src/poetry/core/packages/directory_dependency.py b/src/poetry/core/packages/directory_dependency.py index 024cdf587..ffa45c2c8 100644 --- a/src/poetry/core/packages/directory_dependency.py +++ b/src/poetry/core/packages/directory_dependency.py @@ -34,6 +34,8 @@ def __init__( base=base, extras=extras, ) + # Attributes must be immutable for clone() to be safe! + # (For performance reasons, clone only creates a copy instead of a deep copy). self._develop = develop # cache this function to avoid multiple IO reads and parsing diff --git a/src/poetry/core/packages/file_dependency.py b/src/poetry/core/packages/file_dependency.py index b6dacfffd..9a6e39a5b 100644 --- a/src/poetry/core/packages/file_dependency.py +++ b/src/poetry/core/packages/file_dependency.py @@ -36,6 +36,8 @@ def __init__( subdirectory=directory, extras=extras, ) + # Attributes must be immutable for clone() to be safe! + # (For performance reasons, clone only creates a copy instead of a deep copy). @property def directory(self) -> str | None: diff --git a/src/poetry/core/packages/package.py b/src/poetry/core/packages/package.py index a61331c32..a841c7d45 100644 --- a/src/poetry/core/packages/package.py +++ b/src/poetry/core/packages/package.py @@ -94,6 +94,9 @@ def __init__( features=features, ) + # Attributes must be immutable for clone() to be safe! + # (For performance reasons, clone only creates a copy instead of a deep copy). + self._set_version(version) self.description = "" diff --git a/src/poetry/core/packages/path_dependency.py b/src/poetry/core/packages/path_dependency.py index 160e6916b..d33473270 100644 --- a/src/poetry/core/packages/path_dependency.py +++ b/src/poetry/core/packages/path_dependency.py @@ -32,6 +32,8 @@ def __init__( subdirectory: str | None = None, extras: Iterable[str] | None = None, ) -> None: + # Attributes must be immutable for clone() to be safe! + # (For performance reasons, clone only creates a copy instead of a deep copy). assert source_type in ("file", "directory") self._path = path self._base = base or Path.cwd() diff --git a/src/poetry/core/packages/project_package.py b/src/poetry/core/packages/project_package.py index f9af3bd19..fad048258 100644 --- a/src/poetry/core/packages/project_package.py +++ b/src/poetry/core/packages/project_package.py @@ -36,6 +36,9 @@ def __init__( super().__init__(name, version) + # Attributes must be immutable for clone() to be safe! + # (For performance reasons, clone only creates a copy instead of a deep copy). + self.build_config: Mapping[str, Any] = {} self.packages: Sequence[Mapping[str, Any]] = [] self.include: Sequence[Mapping[str, Any]] = [] diff --git a/src/poetry/core/packages/specification.py b/src/poetry/core/packages/specification.py index 30c66030f..85b9574e7 100644 --- a/src/poetry/core/packages/specification.py +++ b/src/poetry/core/packages/specification.py @@ -29,6 +29,9 @@ def __init__( ) -> None: from packaging.utils import canonicalize_name + # Attributes must be immutable for clone() to be safe! + # (For performance reasons, clone only creates a copy instead of a deep copy). + self._pretty_name = name self._name = canonicalize_name(name) self._source_type = source_type diff --git a/src/poetry/core/packages/url_dependency.py b/src/poetry/core/packages/url_dependency.py index 8020ed8be..be686e4b3 100644 --- a/src/poetry/core/packages/url_dependency.py +++ b/src/poetry/core/packages/url_dependency.py @@ -21,6 +21,8 @@ def __init__( optional: bool = False, extras: Iterable[str] | None = None, ) -> None: + # Attributes must be immutable for clone() to be safe! + # (For performance reasons, clone only creates a copy instead of a deep copy). self._url = url self._directory = directory diff --git a/src/poetry/core/packages/vcs_dependency.py b/src/poetry/core/packages/vcs_dependency.py index 1ebfeda53..8297f90e6 100644 --- a/src/poetry/core/packages/vcs_dependency.py +++ b/src/poetry/core/packages/vcs_dependency.py @@ -29,6 +29,8 @@ def __init__( develop: bool = False, extras: Iterable[str] | None = None, ) -> None: + # Attributes must be immutable for clone() to be safe! + # (For performance reasons, clone only creates a copy instead of a deep copy). self._vcs = vcs self._source = source