Skip to content

Commit

Permalink
Unify search path handling
Browse files Browse the repository at this point in the history
- Validate the path types of the entry-points
- Update the entry-points values with the settings value
- Flatten the entry-points to a list or add them as a dict

- Print more debug messages
- Fix the settings type to string

Signed-off-by: Cristian Le <[email protected]>
  • Loading branch information
LecrisUT committed Sep 3, 2024
1 parent 0cf744c commit 836c51e
Show file tree
Hide file tree
Showing 5 changed files with 106 additions and 44 deletions.
1 change: 1 addition & 0 deletions pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -296,6 +296,7 @@ known-local-folder = ["pathutils"]
"importlib_metadata".msg = "Use scikit_build_core._compat.importlib.metadata instead."
"importlib.resources".msg = "Use scikit_build_core._compat.importlib.resources instead."
"importlib_resources".msg = "Use scikit_build_core._compat.importlib.resources instead."
"importlib.readers".msg = "Use scikit_build_core._compat.importlib.readers instead."
"pyproject_metadata".msg = "Use scikit_build_core._vendor.pyproject_metadata instead."


Expand Down
24 changes: 24 additions & 0 deletions src/scikit_build_core/_compat/importlib/readers.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
from __future__ import annotations

import sys
from typing import TYPE_CHECKING

if TYPE_CHECKING:
from pathlib import Path

if sys.version_info < (3, 10):
# Readers and MultiplexedPath were introduced in 3.10, so nothing should output a MultiplexedPath
# It is also tricky because it is unclear if the `resource_loader` when calling `import` would create
# either importlib.readers.MultiplexedPath or importlib_resources.MultiplexedPath.
# Creating a dummy class instead so that if it fails, it fails completely (and mypy is made happy)
class MultiplexedPath:
_paths: list[Path]
else:
# From 3.11 this is an alias of importlib.resources.readers
from importlib.readers import MultiplexedPath

__all__ = ["MultiplexedPath"]


def __dir__() -> list[str]:
return __all__
109 changes: 73 additions & 36 deletions src/scikit_build_core/builder/builder.py
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@

from .. import __version__
from .._compat.importlib import metadata, resources
from .._compat.importlib.readers import MultiplexedPath
from .._logging import logger
from ..resources import find_python
from .generator import set_environment_for_gen
Expand All @@ -23,6 +24,7 @@

if TYPE_CHECKING:
from collections.abc import Generator, Iterable, Mapping, Sequence
from typing import Any

from packaging.version import Version

Expand Down Expand Up @@ -84,6 +86,68 @@ def _filter_env_cmake_args(env_cmake_args: list[str]) -> Generator[str, None, No
yield arg


# Type-hinting for Traversable is rather hard because they were introduced in python 3.11.
# This avoids introducing importlib_resources dependency that may not be used in the actual package loader
def _sanitize_path(path: Any) -> list[Path] | None:
if isinstance(path, MultiplexedPath):

Check warning on line 92 in src/scikit_build_core/builder/builder.py

View check run for this annotation

Codecov / codecov/patch

src/scikit_build_core/builder/builder.py#L92

Added line #L92 was not covered by tests
# pylint: disable-next=protected-access
return path._paths
if isinstance(path, Path):
return [path]
logger.warning("Unknown path type: [{}] {}", type(path), path)
return None

Check warning on line 98 in src/scikit_build_core/builder/builder.py

View check run for this annotation

Codecov / codecov/patch

src/scikit_build_core/builder/builder.py#L94-L98

Added lines #L94 - L98 were not covered by tests


def _handle_search_paths(
entry_point: str,
settings_val: list[str] | dict[str, str] | None,
output: list[Path] | dict[str, list[Path]],
) -> None:
# Sanity checks
if isinstance(output, dict):
assert isinstance(settings_val, dict)

# Get the search paths from the entry points
search_paths_dict = {}
eps = metadata.entry_points(group=entry_point)
if eps:
logger.debug(

Check warning on line 114 in src/scikit_build_core/builder/builder.py

View check run for this annotation

Codecov / codecov/patch

src/scikit_build_core/builder/builder.py#L114

Added line #L114 was not covered by tests
"Loading search paths {} from entry-points: {}", entry_point, len(eps)
)
for ep in eps:
ep_value = _sanitize_path(resources.files(ep.load()))
logger.debug("{}: {} -> {}", ep.name, ep.value, ep_value)
if ep_value:
search_paths_dict[ep.name] = ep_value

Check warning on line 121 in src/scikit_build_core/builder/builder.py

View check run for this annotation

Codecov / codecov/patch

src/scikit_build_core/builder/builder.py#L117-L121

Added lines #L117 - L121 were not covered by tests

# Update the search paths from the settings options
if isinstance(settings_val, dict) and settings_val:
logger.debug(

Check warning on line 125 in src/scikit_build_core/builder/builder.py

View check run for this annotation

Codecov / codecov/patch

src/scikit_build_core/builder/builder.py#L125

Added line #L125 was not covered by tests
"Overriding search paths {} from config: {}", entry_point, settings_val
)
for key, val in settings_val.items():
if val:

Check warning on line 129 in src/scikit_build_core/builder/builder.py

View check run for this annotation

Codecov / codecov/patch

src/scikit_build_core/builder/builder.py#L128-L129

Added lines #L128 - L129 were not covered by tests
# TODO: Allow settings_val to be dict[str, list[str]]?
search_paths_dict[key] = [Path(val)]

Check warning on line 131 in src/scikit_build_core/builder/builder.py

View check run for this annotation

Codecov / codecov/patch

src/scikit_build_core/builder/builder.py#L131

Added line #L131 was not covered by tests
else:
search_paths_dict.pop(key)

Check warning on line 133 in src/scikit_build_core/builder/builder.py

View check run for this annotation

Codecov / codecov/patch

src/scikit_build_core/builder/builder.py#L133

Added line #L133 was not covered by tests

# Write to the output
if isinstance(output, list):
search_paths_list = [
path for ep_values in search_paths_dict.values() for path in ep_values
]
# If the settings options was a list the values are appended as-is
if isinstance(settings_val, list) and settings_val:
logger.debug(

Check warning on line 142 in src/scikit_build_core/builder/builder.py

View check run for this annotation

Codecov / codecov/patch

src/scikit_build_core/builder/builder.py#L142

Added line #L142 was not covered by tests
"Appending search paths {} with config: {}", entry_point, settings_val
)
search_paths_list += map(Path, settings_val)

Check warning on line 145 in src/scikit_build_core/builder/builder.py

View check run for this annotation

Codecov / codecov/patch

src/scikit_build_core/builder/builder.py#L145

Added line #L145 was not covered by tests
output.extend(search_paths_list)
return
output.update(search_paths_dict)


@dataclasses.dataclass
class Builder:
settings: ScikitBuildSettings
Expand Down Expand Up @@ -123,47 +187,20 @@ def configure(
}

# Add any extra CMake modules
module_dirs_dict = {
ep.name: resources.files(ep.load())
for ep in metadata.entry_points(group="cmake.module")
}
if isinstance(self.settings.search.modules, dict):
# Allow to override any entry-point definition
module_dirs_dict.update(self.settings.search.modules)
module_dirs = list(module_dirs_dict.values())
if isinstance(self.settings.search.modules, list):
# If it was a list, append to the entry-point definitions
module_dirs += self.settings.search.modules
# Remove any empty paths
module_dirs = [path for path in module_dirs if path]
self.config.module_dirs.extend(module_dirs)
_handle_search_paths(
"cmake.module", self.settings.search.modules, self.config.module_dirs
)

# Add any extra CMake prefixes
prefix_dirs_dict = {
ep.name: resources.files(ep.load())
for ep in metadata.entry_points(group="cmake.prefix")
}
if isinstance(self.settings.search.prefixes, dict):
# Allow to override any entry-point definition
prefix_dirs_dict.update(self.settings.search.prefixes)
prefix_dirs = list(prefix_dirs_dict.values())
if isinstance(self.settings.search.prefixes, list):
# If it was a list, append to the entry-point definitions
prefix_dirs += self.settings.search.prefixes
# Remove any empty paths
prefix_dirs = [path for path in prefix_dirs if path]
self.config.prefix_dirs.extend(prefix_dirs)
_handle_search_paths(
"cmake.prefix", self.settings.search.prefixes, self.config.prefix_dirs
)

# Add all CMake roots
prefix_roots = {
ep.name: resources.files(ep.load())
for ep in metadata.entry_points(group="cmake.root")
}
# TODO: Check for unique uppercase names
prefix_roots.update(self.settings.search.roots)
# Remove any empty paths
prefix_roots = {pkg: path for pkg, path in prefix_roots.items() if path}
self.config.prefix_roots.update(prefix_roots)
_handle_search_paths(
"cmake.root", self.settings.search.roots, self.config.prefix_roots
)

# Add site-packages to the prefix path for CMake
site_packages = Path(sysconfig.get_path("purelib"))
Expand Down
10 changes: 5 additions & 5 deletions src/scikit_build_core/cmake.py
Original file line number Diff line number Diff line change
Expand Up @@ -79,7 +79,7 @@ class CMaker:
build_type: str
module_dirs: list[Path] = dataclasses.field(default_factory=list)
prefix_dirs: list[Path] = dataclasses.field(default_factory=list)
prefix_roots: dict[str, Path] = dataclasses.field(default_factory=dict)
prefix_roots: dict[str, list[Path]] = dataclasses.field(default_factory=dict)
init_cache_file: Path = dataclasses.field(init=False, default=Path())
env: dict[str, str] = dataclasses.field(init=False, default_factory=os.environ.copy)
single_config: bool = not sysconfig.get_platform().startswith("win")
Expand Down Expand Up @@ -185,14 +185,14 @@ def init_cache(
f.write('set(CMAKE_FIND_ROOT_PATH_MODE_PACKAGE "BOTH" CACHE PATH "")\n')

if self.prefix_roots:
for pkg, path in self.prefix_roots.items():
path_str = str(path).replace("\\", "/")
for pkg, path_list in self.prefix_roots.items():
paths_str = ";".join(map(str, path_list)).replace("\\", "/")
f.write(

Check warning on line 190 in src/scikit_build_core/cmake.py

View check run for this annotation

Codecov / codecov/patch

src/scikit_build_core/cmake.py#L188-L190

Added lines #L188 - L190 were not covered by tests
f'set({pkg}_ROOT [===[{path_str}]===] CACHE PATH "" FORCE)\n'
f'set({pkg}_ROOT [===[{paths_str}]===] CACHE PATH "" FORCE)\n'
)
# Available since CMake 3.27 with CMP0144
f.write(

Check warning on line 194 in src/scikit_build_core/cmake.py

View check run for this annotation

Codecov / codecov/patch

src/scikit_build_core/cmake.py#L194

Added line #L194 was not covered by tests
f'set({pkg.upper()}_ROOT [===[{path_str}]===] CACHE PATH "" FORCE)\n'
f'set({pkg.upper()}_ROOT [===[{paths_str}]===] CACHE PATH "" FORCE)\n'
)

contents = self.init_cache_file.read_text(encoding="utf-8").strip()
Expand Down
6 changes: 3 additions & 3 deletions src/scikit_build_core/settings/skbuild_model.py
Original file line number Diff line number Diff line change
Expand Up @@ -93,19 +93,19 @@ class SearchSettings:
Add the wheel build path to the CMake prefix paths.
"""

modules: Optional[Union[List[Path], Dict[str, Path]]] = None
modules: Optional[Union[List[str], Dict[str, str]]] = None
"""
List or dict of CMake module search paths. Dict from is used to override
another package's entry-point definition. Populates `CMAKE_MODULE_PATH`.
"""

prefixes: Optional[Union[List[Path], Dict[str, Path]]] = None
prefixes: Optional[Union[List[str], Dict[str, str]]] = None
"""
List or dict of CMake prefix search paths. Dict from is used to override
another package's entry-point definition. Populates `CMAKE_PREFIX_PATH`.
"""

roots: Dict[str, Path] = dataclasses.field(default_factory=dict)
roots: Dict[str, str] = dataclasses.field(default_factory=dict)
"""
Dict of package names and prefix paths. Populates `<Pkg>_ROOT`.
"""
Expand Down

0 comments on commit 836c51e

Please sign in to comment.