tach


Nametach JSON
Version 0.2.0 PyPI version JSON
download
home_pageNone
SummaryA Python tool to maintain a modular package architecture.
upload_time2024-05-20 02:23:21
maintainerNone
docs_urlNone
authorNone
requires_python>=3.8
licenseNone
keywords python module package guard enforcement boundary enforcer domain architecture
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            [![image](https://img.shields.io/pypi/v/tach.svg)](https://pypi.Python.org/pypi/tach)
[![image](https://img.shields.io/pypi/l/tach.svg)](https://pypi.Python.org/pypi/tach)
[![image](https://img.shields.io/pypi/pyversions/tach.svg)](https://pypi.Python.org/pypi/tach)
[![image](https://github.com/gauge-sh/tach/actions/workflows/ci.yml/badge.svg)](https://github.com/gauge-sh/tach/actions/workflows/ci.yml)
[![Checked with pyright](https://microsoft.github.io/pyright/img/pyright_badge.svg)](https://microsoft.github.io/pyright/)
[![Ruff](https://img.shields.io/endpoint?url=https://raw.githubusercontent.com/astral-sh/ruff/main/assets/badge/v2.json)](https://github.com/astral-sh/ruff)
# tach
a Python tool to enforce modular design


[Docs](https://gauge-sh.github.io/tach/)

[Discord](https://discord.gg/DKVksRtuqS) - come say hi!



https://github.com/gauge-sh/tach/assets/10570340/2f5ed866-124e-4322-afe6-15207727ca38



## What is tach?
`tach` allows you to define boundaries and control dependencies between your Python packages. Each package can also define its public interface.

This enforces a decoupled, modular architecture, and prevents tight coupling.
If a package tries to import from another package that is not listed as a dependency, tach will report an error.
If a package tries to import from another package and does not use its public interface, with `strict: true` set, `tach` will report an error.

`tach` is incredibly lightweight, and has no impact on your runtime. Instead, its checks are performed as a lint check through the CLI.

## Installation
```bash
pip install tach
```

## Quickstart
`tach` comes bundled with a command to interactively define your package boundaries.
Run the following in the root of your Python project to enter the editor:
```bash
tach pkg
```

The interactive editor allows you to mark which directories should be treated as package boundaries.
You can navigate with the arrow keys, mark individual packages with `Enter`, and mark all sibling directories
as packages with `Ctrl + a`.

After identifying your packages, press `Ctrl + s` to initialize the boundaries.
Each package will receive a `package.yml` with a single tag based on the folder name,
and a default `tach.yml` file will be created in the current working directory.

If you want to sync your `tach.yml` with the actual dependencies found in your project, you can use `tach sync`:
```bash
tach sync [--prune]
```

Any dependency errors will be automatically resolved by
adding the corresponding dependencies to your `tach.yml` file. If you supply `--prune`,
any dependency constraints in your `tach.yml` which are not necessary will also be removed.

In case you want to start over, `tach clean` lets you delete all `tach` configuration files so that you can re-initialize or configure your packages manually.
```bash
tach clean
```


## Defining Packages
To define a package, add a `package.yml` to the corresponding Python package. Add at least one 'tag' to identify the package.

Examples:
```python
# core/package.yml
tags: ["core"]
```
```python
# db/package.yml
tags: ["db"]
```
```python
# utils/package.yml
tags: ["utils"]
```
Next, specify the constraints for each tag in `tach.yml` in the root of your project:
```yaml
# [root]/tach.yml
constraints:
- tag: core
  depends_on:
  - db
  - utils
- tag: db
  depends_on:
  - utils
- tag: utils
  depends_on: []
```
With these rules in place, packages with tag `core` can import from packages with tag `db` or `utils`. Packages tagged with `db` can only import from `utils`, and packages tagged with `utils` cannot import from any other packages in the project. 

`tach` will now flag any violation of these boundaries.
```bash
# From the root of your Python project (in this example, `project/`)
> tach check
❌ utils/helpers.py[L10]: Cannot import 'core.PublicAPI'. Tags ['utils'] cannot depend on ['core'].
```

NOTE: If your terminal supports hyperlinks, you can click on the failing file path to go directly to the error.

## Defining Interfaces
If you want to define a public interface for the package, import and reference each object you want exposed in the package's `__init__.py` and add its name to `__all__`:
```python
# db/__init__.py
from db.service import PublicAPI

__all__ = ["PublicAPI"]
```
Turning on `strict: true` in the package's `package.yml` will then enforce that all imports from this package occur through `__init__.py` and are listed in `__all__`
```yaml
# db/package.yml
tags: ["db"]
strict: true
```
```python
# The only valid import from "db"
from db import PublicAPI 
```

### Pre-Commit Hook
`tach` can be installed as a pre-commit hook. See the [docs](https://gauge-sh.github.io/tach/usage/#tach-install) for installation instructions.


## Advanced
`tach` supports specific exceptions. You can mark an import with the `tach-ignore` comment:
```python
# tach-ignore
from db.main import PrivateAPI
```
This will stop `tach` from flagging this import as a boundary violation.

You can also specify multiple tags for a given package:
```python
# utils/package.yml
tags: ["core", "utils"]
```
This will expand the set of packages that "utils" can access to include all packages that "core" and "utils" `depends_on` as defined in `tach.yml`.

By default, `tach` ignores hidden directories and files (paths starting with `.`). To override this behavior, set `exclude_hidden_paths` in `tach.yml`
```yaml
exclude_hidden_paths: false
```

## Details
`tach` works by analyzing the abstract syntax tree (AST) of your codebase. It has no runtime impact, and all operations are performed statically. 

Boundary violations are detected at the import layer. This means that dynamic imports using `importlib` or similar approaches will not be caught by tach.

[PyPi Package](https://pypi.org/project/tach/)

### License
[GNU GPLv3](LICENSE)

            

Raw data

            {
    "_id": null,
    "home_page": null,
    "name": "tach",
    "maintainer": null,
    "docs_url": null,
    "requires_python": ">=3.8",
    "maintainer_email": null,
    "keywords": "python, module, package, guard, enforcement, boundary, enforcer, domain, architecture",
    "author": null,
    "author_email": "Caelean Barnes <caeleanb@gmail.com>, Evan Doyle <evanmdoyle@gmail.com>",
    "download_url": "https://files.pythonhosted.org/packages/27/9f/f09b7cf929d4ebcf19ae62e4e4e3715021162ff844648694be1113dfeda3/tach-0.2.0.tar.gz",
    "platform": null,
    "description": "[![image](https://img.shields.io/pypi/v/tach.svg)](https://pypi.Python.org/pypi/tach)\n[![image](https://img.shields.io/pypi/l/tach.svg)](https://pypi.Python.org/pypi/tach)\n[![image](https://img.shields.io/pypi/pyversions/tach.svg)](https://pypi.Python.org/pypi/tach)\n[![image](https://github.com/gauge-sh/tach/actions/workflows/ci.yml/badge.svg)](https://github.com/gauge-sh/tach/actions/workflows/ci.yml)\n[![Checked with pyright](https://microsoft.github.io/pyright/img/pyright_badge.svg)](https://microsoft.github.io/pyright/)\n[![Ruff](https://img.shields.io/endpoint?url=https://raw.githubusercontent.com/astral-sh/ruff/main/assets/badge/v2.json)](https://github.com/astral-sh/ruff)\n# tach\na Python tool to enforce modular design\n\n\n[Docs](https://gauge-sh.github.io/tach/)\n\n[Discord](https://discord.gg/DKVksRtuqS) - come say hi!\n\n\n\nhttps://github.com/gauge-sh/tach/assets/10570340/2f5ed866-124e-4322-afe6-15207727ca38\n\n\n\n## What is tach?\n`tach` allows you to define boundaries and control dependencies between your Python packages. Each package can also define its public interface.\n\nThis enforces a decoupled, modular architecture, and prevents tight coupling.\nIf a package tries to import from another package that is not listed as a dependency, tach will report an error.\nIf a package tries to import from another package and does not use its public interface, with `strict: true` set, `tach` will report an error.\n\n`tach` is incredibly lightweight, and has no impact on your runtime. Instead, its checks are performed as a lint check through the CLI.\n\n## Installation\n```bash\npip install tach\n```\n\n## Quickstart\n`tach` comes bundled with a command to interactively define your package boundaries.\nRun the following in the root of your Python project to enter the editor:\n```bash\ntach pkg\n```\n\nThe interactive editor allows you to mark which directories should be treated as package boundaries.\nYou can navigate with the arrow keys, mark individual packages with `Enter`, and mark all sibling directories\nas packages with `Ctrl + a`.\n\nAfter identifying your packages, press `Ctrl + s` to initialize the boundaries.\nEach package will receive a `package.yml` with a single tag based on the folder name,\nand a default `tach.yml` file will be created in the current working directory.\n\nIf you want to sync your `tach.yml` with the actual dependencies found in your project, you can use `tach sync`:\n```bash\ntach sync [--prune]\n```\n\nAny dependency errors will be automatically resolved by\nadding the corresponding dependencies to your `tach.yml` file. If you supply `--prune`,\nany dependency constraints in your `tach.yml` which are not necessary will also be removed.\n\nIn case you want to start over, `tach clean` lets you delete all `tach` configuration files so that you can re-initialize or configure your packages manually.\n```bash\ntach clean\n```\n\n\n## Defining Packages\nTo define a package, add a `package.yml` to the corresponding Python package. Add at least one 'tag' to identify the package.\n\nExamples:\n```python\n# core/package.yml\ntags: [\"core\"]\n```\n```python\n# db/package.yml\ntags: [\"db\"]\n```\n```python\n# utils/package.yml\ntags: [\"utils\"]\n```\nNext, specify the constraints for each tag in `tach.yml` in the root of your project:\n```yaml\n# [root]/tach.yml\nconstraints:\n- tag: core\n  depends_on:\n  - db\n  - utils\n- tag: db\n  depends_on:\n  - utils\n- tag: utils\n  depends_on: []\n```\nWith these rules in place, packages with tag `core` can import from packages with tag `db` or `utils`. Packages tagged with `db` can only import from `utils`, and packages tagged with `utils` cannot import from any other packages in the project. \n\n`tach` will now flag any violation of these boundaries.\n```bash\n# From the root of your Python project (in this example, `project/`)\n> tach check\n\u274c utils/helpers.py[L10]: Cannot import 'core.PublicAPI'. Tags ['utils'] cannot depend on ['core'].\n```\n\nNOTE: If your terminal supports hyperlinks, you can click on the failing file path to go directly to the error.\n\n## Defining Interfaces\nIf you want to define a public interface for the package, import and reference each object you want exposed in the package's `__init__.py` and add its name to `__all__`:\n```python\n# db/__init__.py\nfrom db.service import PublicAPI\n\n__all__ = [\"PublicAPI\"]\n```\nTurning on `strict: true` in the package's `package.yml` will then enforce that all imports from this package occur through `__init__.py` and are listed in `__all__`\n```yaml\n# db/package.yml\ntags: [\"db\"]\nstrict: true\n```\n```python\n# The only valid import from \"db\"\nfrom db import PublicAPI \n```\n\n### Pre-Commit Hook\n`tach` can be installed as a pre-commit hook. See the [docs](https://gauge-sh.github.io/tach/usage/#tach-install) for installation instructions.\n\n\n## Advanced\n`tach` supports specific exceptions. You can mark an import with the `tach-ignore` comment:\n```python\n# tach-ignore\nfrom db.main import PrivateAPI\n```\nThis will stop `tach` from flagging this import as a boundary violation.\n\nYou can also specify multiple tags for a given package:\n```python\n# utils/package.yml\ntags: [\"core\", \"utils\"]\n```\nThis will expand the set of packages that \"utils\" can access to include all packages that \"core\" and \"utils\" `depends_on` as defined in `tach.yml`.\n\nBy default, `tach` ignores hidden directories and files (paths starting with `.`). To override this behavior, set `exclude_hidden_paths` in `tach.yml`\n```yaml\nexclude_hidden_paths: false\n```\n\n## Details\n`tach` works by analyzing the abstract syntax tree (AST) of your codebase. It has no runtime impact, and all operations are performed statically. \n\nBoundary violations are detected at the import layer. This means that dynamic imports using `importlib` or similar approaches will not be caught by tach.\n\n[PyPi Package](https://pypi.org/project/tach/)\n\n### License\n[GNU GPLv3](LICENSE)\n",
    "bugtrack_url": null,
    "license": null,
    "summary": "A Python tool to maintain a modular package architecture.",
    "version": "0.2.0",
    "project_urls": {
        "Homepage": "https://github.com/never-over/tach",
        "Issues": "https://github.com/never-over/tach/issues"
    },
    "split_keywords": [
        "python",
        " module",
        " package",
        " guard",
        " enforcement",
        " boundary",
        " enforcer",
        " domain",
        " architecture"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "06d1ba0a5a05968b5b0428b4eb4d880beeeeacb8437f7f6ff3c2ed812d88d062",
                "md5": "04067e208897acacb00e6b461484244e",
                "sha256": "f2df311cbccc6564e2560826484a45b4616c942d18de337be279722af2ba62eb"
            },
            "downloads": -1,
            "filename": "tach-0.2.0-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "04067e208897acacb00e6b461484244e",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": ">=3.8",
            "size": 47428,
            "upload_time": "2024-05-20T02:23:20",
            "upload_time_iso_8601": "2024-05-20T02:23:20.112099Z",
            "url": "https://files.pythonhosted.org/packages/06/d1/ba0a5a05968b5b0428b4eb4d880beeeeacb8437f7f6ff3c2ed812d88d062/tach-0.2.0-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "279ff09b7cf929d4ebcf19ae62e4e4e3715021162ff844648694be1113dfeda3",
                "md5": "f73a287ea4e520755ca13d74a1f1fcfd",
                "sha256": "8de216b159e03d008fd6a3a8b66dffaa170f6e8515b17e1c40600b8cd386e448"
            },
            "downloads": -1,
            "filename": "tach-0.2.0.tar.gz",
            "has_sig": false,
            "md5_digest": "f73a287ea4e520755ca13d74a1f1fcfd",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": ">=3.8",
            "size": 137423,
            "upload_time": "2024-05-20T02:23:21",
            "upload_time_iso_8601": "2024-05-20T02:23:21.921429Z",
            "url": "https://files.pythonhosted.org/packages/27/9f/f09b7cf929d4ebcf19ae62e4e4e3715021162ff844648694be1113dfeda3/tach-0.2.0.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2024-05-20 02:23:21",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "never-over",
    "github_project": "tach",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": true,
    "lcname": "tach"
}
        
Elapsed time: 0.27105s