Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Standalone contract #222

Open
wants to merge 2 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
27 changes: 27 additions & 0 deletions docs/contract_types.rst
Original file line number Diff line number Diff line change
Expand Up @@ -318,6 +318,33 @@ Note: you are not allowed to mix different kinds of separators on the same line.
mypackage.blue | mypackage.green : mypackage.yellow # Invalid as it mixes separators.
mypackage.low

Standalone modules
------------------

*Type name:* ``standalone``

Standalone contracts check that a set of modules are standalone, that is not importing
or imported by any other modules in the graph.

**Example:**

.. code-block:: ini

[importlinter:contract:my-standalone-contract]
name = My standalone contract
type = standalone
modules =
mypackage.bar
mypackage.baz
ignore_imports =
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think it would be better to remove the ignore_imports section from the example, just so it's minimal.

mypackage.bar.green -> mypackage.utils
mypackage.foo.purple -> mypackage.baz.blue

**Configuration options**

- ``modules``: A list of modules/subpackages that should be independent of each other.
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This line needs to be updated as it's from the independence contract.

- ``ignore_imports``: See :ref:`Shared options`.
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We should also add unmatched_ignore_imports_alerting as an option.



Custom contract types
---------------------
Expand Down
1 change: 1 addition & 0 deletions src/importlinter/application/use_cases.py
Original file line number Diff line number Diff line change
Expand Up @@ -248,6 +248,7 @@ def _get_built_in_contract_types() -> List[Tuple[str, Type[Contract]]]:
"forbidden: importlinter.contracts.forbidden.ForbiddenContract",
"layers: importlinter.contracts.layers.LayersContract",
"independence: importlinter.contracts.independence.IndependenceContract",
"standalone: importlinter.contracts.standalone.StandaloneContract",
],
)
)
Expand Down
64 changes: 64 additions & 0 deletions src/importlinter/contracts/standalone.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,64 @@
from __future__ import annotations

from grimp import ImportGraph

from importlinter.application import contract_utils, output
from importlinter.domain import fields
from importlinter.domain.contract import Contract, ContractCheck


class StandaloneContract(Contract):
"""
Standalone contracts check that a set of modules are standalone, that is not importing
or imported by any other modules in the graph.

Configuration options:

- modules: A list of Modules that should be standalone.
- ignore_imports: A set of ImportExpressions. These imports will be ignored: if the import
would cause a contract to be broken, adding it to the set will cause
the contract be kept instead. (Optional.)
"""

type_name = "standalone"

modules = fields.ListField(subfield=fields.ModuleField())
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Any reason not to make this a SetField? Order isn't important...

ignore_imports = fields.SetField(subfield=fields.ImportExpressionField(), required=False)

def check(self, graph: ImportGraph, verbose: bool) -> ContractCheck:
warnings = contract_utils.remove_ignored_imports(
graph=graph,
ignore_imports=self.ignore_imports, # type: ignore
unmatched_alerting="none", # type: ignore
)

self._check_all_modules_exist_in_graph(graph)

violations = {}
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Would be helpful to type annotate what type of element this set contains.

for module in self.modules: # type: ignore
imports = graph.find_modules_directly_imported_by(module.name)
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Minor suggestion: rename imports to importeds and imported_by to importers. It works better when we're pulling out individual items (for import in imports is a syntax error.)

imported_by = graph.find_modules_that_directly_import(module.name)
if imported_by or imports:
violations[module.name] = [
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I find this block a bit difficult to read. Maybe better to tackle the importeds and the importers in two separate statements?

It would also be nice to use the DirectImport type, would be more expressive than using tuples.

(module.name, import_expression) for import_expression in imported_by
] + [(import_expression, module.name) for import_expression in imports]
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What about calling graph.get_import_details so we can get richer information about the illegal imports, such as line numbers? These could then be rendered using importlinter.contracts._common._render_direct_import (that private function should be main public). That would make the rendering consistent with what the other contracts are showing, i.e.

mypackage.three must be standalone:

   -   mypackage.three -> mypackage.green (l.4)
   -   mypackage.blue -> mypackage.three (l.14, 24)

What do you think? Would be nice to have line numbers.


kept = all(len(violation) == 0 for violation in violations.values())
return ContractCheck(
kept=kept,
warnings=warnings,
metadata={"violations": violations},
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Could we put these under a key called invalid_imports? Makes the naming more consistent with what's in the metadata in the other contracts.

)

def render_broken_contract(self, check: "ContractCheck") -> None:
for module_name, connections in check.metadata["violations"].items():
output.print(f"{module_name} must be standalone:")
output.new_line()
for upstream, downstream in connections:
output.print_error(f"- {downstream} is not allowed to import {upstream}")
output.new_line()

def _check_all_modules_exist_in_graph(self, graph: ImportGraph) -> None:
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Now that this is used by two contracts, would you mind moving this to contract_utils?

for module in self.modules: # type: ignore
if module.name not in graph.modules:
raise ValueError(f"Module '{module.name}' does not exist.")
3 changes: 3 additions & 0 deletions src/importlinter/domain/imports.py
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,9 @@ def __init__(self, name: str) -> None:
"""
self.name = name

def has_wildcard_expression(self) -> bool:
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't think this commit belongs in this PR, correct?

It would be interesting to support module expressions in standalone contracts, but perhaps that's left for another day!

return "*" in self.name

def __str__(self) -> str:
return self.name

Expand Down
162 changes: 162 additions & 0 deletions tests/unit/contracts/test_standalone.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,162 @@
from __future__ import annotations
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Nice tests, thanks!


import pytest
from grimp.adaptors.graph import ImportGraph

from importlinter.application.app_config import settings
from importlinter.contracts.standalone import StandaloneContract
from importlinter.domain.contract import ContractCheck
from tests.adapters.printing import FakePrinter
from tests.adapters.timing import FakeTimer


@pytest.fixture(scope="module", autouse=True)
def configure():
settings.configure(TIMER=FakeTimer())


class TestStandaloneContract:
def _build_default_graph(self):
graph = ImportGraph()
for module in (
"mypackage",
"mypackage.blue",
"mypackage.blue.alpha",
"mypackage.blue.beta",
"mypackage.blue.beta.foo",
"mypackage.blue.foo",
"mypackage.blue.hello",
"mypackage.blue.world",
"mypackage.green",
"mypackage.green.bar",
"mypackage.yellow",
"mypackage.yellow.gamma",
"mypackage.yellow.delta",
"mypackage.other",
"mypackage.other.sub",
"mypackage.other.sub2",
):
graph.add_module(module)
return graph

def _check_default_contract(self, graph):
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Not a blocker, but IMO it would be better if we passed in the set of modules that we want to be standalone from the test itself - they're a bit hidden in here.

contract = StandaloneContract(
name="Standalone contract",
session_options={"root_packages": ["mypackage"]},
contract_options={"modules": ("mypackage.green", "mypackage.yellow")},
)
return contract.check(graph=graph, verbose=False)

def test_when_modules_are_standalone(self):
graph = self._build_default_graph()
graph.add_import(
importer="mypackage.blue",
imported="mypackage.other",
line_number=10,
line_contents="-",
)
graph.add_import(
importer="mypackage.other",
imported="mypackage.blue.world",
line_number=11,
line_contents="-",
)

contract_check = self._check_default_contract(graph)

assert contract_check.kept, contract_check.metadata

def test_non_standalone_imported(self):
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Mind adding a test case for multiple importeds and importers to/from multiple modules, just to check the logic of handling multiple violations?

graph = self._build_default_graph()
graph.add_import(
importer="mypackage.blue",
imported="mypackage.green",
line_number=10,
line_contents="-",
)
contract_check = self._check_default_contract(graph)

assert not contract_check.kept

expected_metadata = {
"violations": {"mypackage.green": [("mypackage.green", "mypackage.blue")]}
}
assert expected_metadata == contract_check.metadata

def test_non_standalone_imports(self):
graph = self._build_default_graph()
graph.add_import(
importer="mypackage.yellow",
imported="mypackage.other",
line_number=10,
line_contents="-",
)

contract_check = self._check_default_contract(graph)

assert not contract_check.kept

expected_metadata = {
"violations": {"mypackage.yellow": [("mypackage.other", "mypackage.yellow")]}
}
assert expected_metadata == contract_check.metadata

def test_standalone_ignore(self):
graph = self._build_default_graph()
graph.add_import(
importer="mypackage.yellow",
imported="mypackage.other",
line_number=10,
line_contents="-",
)

contract = StandaloneContract(
name="Standalone contract",
session_options={"root_packages": ["mypackage"]},
contract_options={
"modules": ("mypackage.green", "mypackage.yellow"),
"ignore_imports": ["mypackage.yellow -> mypackage.other"],
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Very minor comment, but we could allow passing in the ignore_imports as an optional argument to the utility method.

},
)
contract_check = contract.check(graph=graph, verbose=False)

assert contract_check.kept


def test_render_broken_contract():
settings.configure(PRINTER=FakePrinter())
contract = StandaloneContract(
name="Standalone contract",
session_options={"root_packages": ["mypackage"]},
contract_options={"modules": ["mypackage.green"]},
)
check = ContractCheck(
kept=False,
metadata={
"violations": {
"mypackage": [
("mypackage.blue.foo", "mypackage.utils.red"),
("mypackage.blue.red", "mypackage.utils.yellow"),
],
"mypackage.green": [
("mypackage.green.a.b", "mypackage.green.b.a"),
],
}
},
)

contract.render_broken_contract(check)

settings.PRINTER.pop_and_assert(
"""
mypackage must be standalone:

- mypackage.utils.red is not allowed to import mypackage.blue.foo
- mypackage.utils.yellow is not allowed to import mypackage.blue.red

mypackage.green must be standalone:

- mypackage.green.b.a is not allowed to import mypackage.green.a.b

""" # noqa
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why the noqa?

)
Loading