Metadata-Version: 2.4
Name: layers-linter
Version: 3.2.2
Summary: A static code analyzer that enforces architectural boundaries for layers and outher libraries
Project-URL: Homepage, https://github.com/pavelmaksimov/layers-linter
Project-URL: Bug Tracker, https://github.com/pavelmaksimov/layers-linter/issues
Requires-Python: >=3.11
Description-Content-Type: text/markdown
Provides-Extra: flake8
Requires-Dist: flake8>=7.2.0; extra == "flake8"

# Layers Linter

A static code analyzer that enforces architectural boundaries for layers and outher libraries

## Overview

Layers Linter helps you maintain a clean architecture by:

1. **Enforcing layer dependencies**: Ensures that dependencies between modules follow the allowed directions
2. **Restricting library usage**: Verifies that libraries are only used in permitted layers
3. **Integrating with your workflow**: Works as a standalone CLI tool or as a Flake8 plugin

The linter analyzes your project's import structure by parsing the AST (Abstract Syntax Tree) of Python files, building a dependency graph, and validating it against your configuration.

## Installation

```bash
# Install from PyPI
pip install layers-linter

# Or install with Flake8 plugin support
pip install layers-linter[flake8]
```

## Usage

### Command Line Interface

Default path to layers.toml config in the current working directory.

```bash
# Basic usage
layers-linter /path/to/your/project

# With custom configuration file
layers-linter /path/to/your/project --config /path/to/your/layers.toml

# Disable checking for modules without a layer
layers-linter /path/to/your/project --no-check-no-layer
```

### Flake8 Plugin

Add to your `.flake8` configuration:

```ini
[flake8]
select = LA
la-config = layers.toml
```

Then run Flake8 as usual:

Default path to layers.toml config in the current working directory.

```bash
flake8 /path/to/your/project [--la-config /path/to/your/config]
```

## Configuration

Layers Linter uses a TOML configuration file to define your project's architecture. By default, it looks for a file named `layers.toml` in the current directory.

### Configuration Format

```toml
# Optional: Modules to exclude from analysis
exclude_modules = ["*.__init__"]

# Layer definitions
[layers]

[layers.dicontainer]
description = ""
# Modules that belong to this layer
contains_modules = [
    "project.container"
]
# Layers that this layer can use (if empty list, it can't use any other layers)
depends_on = ["usecases"]


[layers.usecases]
description = ""
contains_modules = [
    "project.domains.*.use_cases.*",
    "project.domains.*.use_cases",
]
depends_on = []


[layers.presentation]
description = ""
contains_modules = [
    "project.presentation.*",
    "project.domains.*.endpoints"
]
depends_on = ["dicontainer"]


# Library restrictions
[libs]

[libs.presentation]
# Layers that can use this library (if not specified, all layers can use it)
allowed_in = ["fastapi"]
```

### Pattern Matching

The `contains_modules` field supports pattern matching with wildcards:

- `project.module` - Exact match
- `project.module.*` - Module and all submodules
- `project.*.module` - Any module with the specified pattern
- `*.module` - Any module ending with the specified pattern

### Dependency Rules

- **depends_on**: Controls which layers this layer can use
  - Empty list (`[]`): This layer cannot use any other layers
  - Not specified or `"none"`: No restrictions


## Error Codes

- **LA001**: Invalid dependency between layers
  - Example: `Invalid dependency from layer 'presentation' to layer 'repositories'`

- **LA002**: Module without a layer
  - Example: `Module 'project.utils.helpers' does not belong to any layer`

- **LA020**: Restricted library usage
  - Example: `Layers [presentation] cannot use restricted library 'sqlalchemy'`

## Notes

- Imports inside `if typing.TYPE_CHECKING:` blocks are ignored
- The linter builds a complete dependency graph before validation, allowing for comprehensive analysis
