Skip to main content

Rule your architecture like a real developer

Project description

pytest-archon

build_and_test

pytest-archon is a little tool that helps you structure (large) Python projects. This tool allows you to define architectural boundaries in your code, also known as forbidden dependencies.

Explicitly defined architectural boundaries helps you keep your code in shape. It avoids the creation of circular dependencies. New people on the project are made aware of the structure through a simple set of rules, instead of lore.

Installation

The simple way:

pip install pytest-archon

Usage

pytest-archon can be used to define architectural boundaries from (unit) tests. Because they're tests, they can be closely tied to the actual application.

You can use pytest-archon in tests by simply importing the archrule function. Using this function you can construct import tests:

from pytest_archon import archrule


def test_rule_basic():
    (
        archrule("name", comment="some comment")
        .match("pytest_archon.col*")
        .exclude("pytest_archon.colgate")
        .should_not_import("pytest_archon.import_finder")
        .should_import("pytest_archon.core*")
        .check("pytest_archon")
    )
  • To match the modules and constraints, fnmatch syntax is used (the default). You can also use regular expressions by supplying the use_regex=True argument to archrule(). Example: archrule(..., use_regex=True).match(...).
  • .exclude() is optional
  • .should_import() and .should_not_import() can be combined and can occur multiple times.
  • .may_import() can be used in combination with .should_not_import().
  • .check() needs either a module object or a string

The check() method can have a few optional parameters, that alter the way the checks are performed.

  • Without parameters, the whole file is checked for imports. So imports done in functions and methods are also found. Transitive dependencies are also checked
  • Option only_toplevel_imports=True will only check for toplevel imports. Conditional imports and import in functions and methods are ignored.
  • skip_type_checking=True will check all imports, but skip imports defined in if typing.TYPE_CHECKING blocks.
  • only_direct_imports=True will only check for imports performed by the module directly and will not check transitive imports.
  • If only_toplevel_imports=True is set, skip_type_checking=True has no effect.
  • Options can be combined.
Check toplevel imports Check TYPE_CHECKING imports Check conditional imports, and imports in functions and methods Check transitive imports
no options enabled
skip_type_checking=True
only_toplevel_imports=True
only_direct_imports=True

Example

Domain model has no dependencies

def test_domain():
    # test if the domain model does not import other submodules
    # (the domain model should be standing on its own and be used by other modules)
    (
        archrule("domain", comment="domain does not import any other submodules")
        .match("packageX.domain*") # matches packageX.domain and packageX.domain.*
        .should_not_import("packageX*")
        .may_import("packageX.domain.*")
        .check("packageX")
    )

util module is used at more than one place

You can also supply custom constraints as predicate functions.

If you, for example, have a common or util module, you might want to make sure that it is used at least at two places (otherwise it would not make sense to have a separate module).

from pytest_archon import archrule


def test_utils_are_shared():
    def have_at_least_two_users(util_module, direct_imports, all_imports):
        # iterate through all imports and find modules using the util_module in question
        users = [k for k, v in all_imports.items() if util_module in v]
        # return True if more than two modules use the util_module
        return len(users) > 2

    archrule("util_is_shared").match("pkg.util").should(have_at_least_two_users).check("pkg")

See also

The blog post How to tame your Python codebase is also a good overview.

Similar projects

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

pytest-archon-0.0.6.tar.gz (47.7 kB view details)

Uploaded Source

Built Distribution

pytest_archon-0.0.6-py3-none-any.whl (44.7 kB view details)

Uploaded Python 3

File details

Details for the file pytest-archon-0.0.6.tar.gz.

File metadata

  • Download URL: pytest-archon-0.0.6.tar.gz
  • Upload date:
  • Size: 47.7 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/4.0.2 CPython/3.11.6

File hashes

Hashes for pytest-archon-0.0.6.tar.gz
Algorithm Hash digest
SHA256 5af260a7617fc0a9e8bcd7a62adc42ed1ad466e20ae128b218589f49f73b1bcf
MD5 d52b344156a00bdf5f47f0c5c44e6a38
BLAKE2b-256 5582ae6a09261ad465e43844b5ff201ecf2ab01a2cd64123fb60c2cd702057c5

See more details on using hashes here.

File details

Details for the file pytest_archon-0.0.6-py3-none-any.whl.

File metadata

File hashes

Hashes for pytest_archon-0.0.6-py3-none-any.whl
Algorithm Hash digest
SHA256 f49b6b1f33a541dda1cab8a733b93b8a712594a002f03283294a624d0b394e83
MD5 9a82cc52898e7f74cb032a224b98ba4c
BLAKE2b-256 8f67ac6b2f2e9a9f46f5d1da47fd02107f51d9dff39d80442adc2dc7abb8c35d

See more details on using hashes here.

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page