Coverage for tests/test_api.py: 92.68%
93 statements
« prev ^ index » next coverage.py v7.7.0, created at 2025-03-19 16:19 +0100
« prev ^ index » next coverage.py v7.7.0, created at 2025-03-19 16:19 +0100
1"""Tests for our own API exposition."""
3from __future__ import annotations
5from collections import defaultdict
6from pathlib import Path
7from typing import TYPE_CHECKING
9import griffe
10import pytest
11from mkdocstrings import Inventory
13import yore
15if TYPE_CHECKING:
16 from collections.abc import Iterator
19@pytest.fixture(name="loader", scope="module")
20def _fixture_loader() -> griffe.GriffeLoader:
21 loader = griffe.GriffeLoader()
22 loader.load("yore")
23 loader.resolve_aliases()
24 return loader
27@pytest.fixture(name="internal_api", scope="module")
28def _fixture_internal_api(loader: griffe.GriffeLoader) -> griffe.Module:
29 return loader.modules_collection["yore._internal"]
32@pytest.fixture(name="public_api", scope="module")
33def _fixture_public_api(loader: griffe.GriffeLoader) -> griffe.Module:
34 return loader.modules_collection["yore"]
37def _yield_public_objects(
38 obj: griffe.Module | griffe.Class,
39 *,
40 modules: bool = False,
41 modulelevel: bool = True,
42 inherited: bool = False,
43 special: bool = False,
44) -> Iterator[griffe.Object | griffe.Alias]:
45 for member in obj.all_members.values() if inherited else obj.members.values():
46 try:
47 if member.is_module:
48 if member.is_alias or not member.is_public:
49 continue
50 if modules: 50 ↛ 51line 50 didn't jump to line 51 because the condition on line 50 was never true
51 yield member
52 yield from _yield_public_objects(
53 member, # type: ignore[arg-type]
54 modules=modules,
55 modulelevel=modulelevel,
56 inherited=inherited,
57 special=special,
58 )
59 elif member.is_public and (special or not member.is_special):
60 yield member
61 else:
62 continue
63 if member.is_class and not modulelevel:
64 yield from _yield_public_objects(
65 member, # type: ignore[arg-type]
66 modules=modules,
67 modulelevel=False,
68 inherited=inherited,
69 special=special,
70 )
71 except (griffe.AliasResolutionError, griffe.CyclicAliasError):
72 continue
75@pytest.fixture(name="modulelevel_internal_objects", scope="module")
76def _fixture_modulelevel_internal_objects(internal_api: griffe.Module) -> list[griffe.Object | griffe.Alias]:
77 return list(_yield_public_objects(internal_api, modulelevel=True))
80@pytest.fixture(name="internal_objects", scope="module")
81def _fixture_internal_objects(internal_api: griffe.Module) -> list[griffe.Object | griffe.Alias]:
82 return list(_yield_public_objects(internal_api, modulelevel=False, special=True))
85@pytest.fixture(name="public_objects", scope="module")
86def _fixture_public_objects(public_api: griffe.Module) -> list[griffe.Object | griffe.Alias]:
87 return list(_yield_public_objects(public_api, modulelevel=False, inherited=True, special=True))
90@pytest.fixture(name="inventory", scope="module")
91def _fixture_inventory() -> Inventory:
92 inventory_file = Path(__file__).parent.parent / "site" / "objects.inv"
93 if not inventory_file.exists(): 93 ↛ 94line 93 didn't jump to line 94 because the condition on line 93 was never true
94 raise pytest.skip("The objects inventory is not available.")
95 with inventory_file.open("rb") as file:
96 return Inventory.parse_sphinx(file)
99def test_exposed_objects(modulelevel_internal_objects: list[griffe.Object | griffe.Alias]) -> None:
100 """All public objects in the internal API are exposed under `yore`."""
101 not_exposed = [
102 obj.path for obj in modulelevel_internal_objects if obj.name not in yore.__all__ or not hasattr(yore, obj.name)
103 ]
104 assert not not_exposed, "Objects not exposed:\n" + "\n".join(sorted(not_exposed))
107def test_unique_names(modulelevel_internal_objects: list[griffe.Object | griffe.Alias]) -> None:
108 """All internal objects have unique names."""
109 names_to_paths = defaultdict(list)
110 for obj in modulelevel_internal_objects:
111 names_to_paths[obj.name].append(obj.path)
112 non_unique = [paths for paths in names_to_paths.values() if len(paths) > 1]
113 assert not non_unique, "Non-unique names:\n" + "\n".join(str(paths) for paths in non_unique)
116def test_single_locations(public_api: griffe.Module) -> None:
117 """All objects have a single public location."""
119 def _public_path(obj: griffe.Object | griffe.Alias) -> bool:
120 return obj.is_public and (obj.parent is None or _public_path(obj.parent))
122 multiple_locations = {}
123 for obj_name in yore.__all__:
124 obj = public_api[obj_name]
125 if obj.aliases and ( 125 ↛ 128line 125 didn't jump to line 128 because the condition on line 125 was never true
126 public_aliases := [path for path, alias in obj.aliases.items() if path != obj.path and _public_path(alias)]
127 ):
128 multiple_locations[obj.path] = public_aliases
129 assert not multiple_locations, "Multiple public locations:\n" + "\n".join(
130 f"{path}: {aliases}" for path, aliases in multiple_locations.items()
131 )
134def test_api_matches_inventory(inventory: Inventory, public_objects: list[griffe.Object | griffe.Alias]) -> None:
135 """All public objects are added to the inventory."""
136 ignore_names = {"__bool__", "__getattr__", "__init__", "__post_init__", "__repr__", "__str__"}
137 not_in_inventory = [
138 obj.path for obj in public_objects if obj.name not in ignore_names and obj.path not in inventory
139 ]
140 msg = "Objects not in the inventory (try running `make run mkdocs build`):\n{paths}"
141 assert not not_in_inventory, msg.format(paths="\n".join(sorted(not_in_inventory)))
144def test_inventory_matches_api(
145 inventory: Inventory,
146 public_objects: list[griffe.Object | griffe.Alias],
147 loader: griffe.GriffeLoader,
148) -> None:
149 """The inventory doesn't contain any additional Python object."""
150 not_in_api = []
151 public_api_paths = {obj.path for obj in public_objects}
152 public_api_paths.add("yore")
153 for item in inventory.values():
154 if item.domain == "py" and "(" not in item.name and (item.name == "yore" or item.name.startswith("yore.")):
155 obj = loader.modules_collection[item.name]
156 if obj.path not in public_api_paths and not any(path in public_api_paths for path in obj.aliases): 156 ↛ 157line 156 didn't jump to line 157 because the condition on line 156 was never true
157 not_in_api.append(item.name)
158 msg = "Inventory objects not in public API (try running `make run mkdocs build`):\n{paths}"
159 assert not not_in_api, msg.format(paths="\n".join(sorted(not_in_api)))
162def test_no_module_docstrings_in_internal_api(internal_api: griffe.Module) -> None:
163 """No module docstrings should be written in our internal API.
165 The reasoning is that docstrings are addressed to users of the public API,
166 but internal modules are not exposed to users, so they should not have docstrings.
167 """
169 def _modules(obj: griffe.Module) -> Iterator[griffe.Module]:
170 for member in obj.modules.values():
171 yield member
172 yield from _modules(member)
174 for obj in _modules(internal_api):
175 assert not obj.docstring