Metadata-Version: 2.1
Name: flake8-import-conventions
Version: 0.0.1
Summary: An opinionated plugin for Flake8 on how certain packages should be imported or aliased.
License: MIT
Keywords: flake8,linter,qa,import,flake8-plugin,code quality
Author: João Palmeiro
Author-email: joaopalmeiro@proton.me
Requires-Python: >=3.7,<4.0
Classifier: License :: OSI Approved :: MIT License
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3.7
Classifier: Programming Language :: Python :: 3.8
Classifier: Programming Language :: Python :: 3.9
Classifier: Programming Language :: Python :: 3.10
Classifier: Programming Language :: Python :: 3.11
Requires-Dist: attrs (>=22.2.0,<23.0.0)
Requires-Dist: flake8 (>=5)
Description-Content-Type: text/markdown

# flake8-import-conventions

An opinionated plugin for Flake8 on how certain packages should be imported or aliased.

It is based on the [`pandas-vet`](https://github.com/deppen8/pandas-vet) and [`flake8-2020`](https://github.com/asottile/flake8-2020) plugins.

## Development

```bash
poetry install --with dev
```

```bash
poetry shell
```

Open the `manual_test.py` file in VS Code to see the error messages.

```bash
pytest tests/ -v
```

or (to see `print()`s)

```bash
pytest tests/ -v -s
```

## Deployment

```bash
poetry check
```

```bash
poetry version minor
```

or

```bash
poetry version patch
```

```bash
git tag
```

```bash
git tag "v$(poetry version --short)"
```

```bash
git push origin "v$(poetry version --short)"
```

## References

- Anthony Sottile's "[a flake8 plugin from scratch (intermediate) anthony explains #025](https://youtu.be/ot5Z4KQPBL8)" tutorial.
- [flake8-pie](https://github.com/sbdchd/flake8-pie).
- [wemake-python-styleguide](https://github.com/wemake-services/wemake-python-styleguide).

## Notes

- [flake8-class-attributes-order](https://github.com/best-doctor/flake8-class-attributes-order).
- [astpretty](https://github.com/asottile/astpretty).
- [babi](https://github.com/asottile/babi) (text editor).
- `self.generic_visit(node)`: call it at the end of each `visit_*` method for the recursion to continue.
- [attrs](https://www.attrs.org/):
  - "Classes without boilerplate".
  - No need to implement object protocols (dunder methods).
  - "It does _nothing_ dynamic at runtime, hence zero runtime overhead."
  - `__attrs_post_init__` method.
- [Dict vs. slotted classes](https://www.attrs.org/en/stable/glossary.html).
- `poetry add flake8@^3.0.0 attrs`.
- `flake8 formatter.py flake8_import_conventions --jobs 1`.
- `poetry install --no-root` (don't install _this_ package).
- `import typing as t` ([flake8-annotations](https://github.com/sco1/flake8-annotations)).
- `import geopandas` ([source](https://github.com/geopandas/geopandas/issues/716)).
- https://github.com/asottile/flake8-2020/commit/10ba36520cca00ecbc812b2b96fcaaf9f8999c06
- Poetry:
  - https://python-poetry.org/docs/#installing-with-the-official-installer
  - `poetry --version` (1.3.2)
  - `poetry init` (https://python-poetry.org/docs/basic-usage/#initialising-a-pre-existing-project)
  - https://python-poetry.org/docs/dependency-specification/#exact-requirements
  - `poetry add attrs`
  - `poetry add mypy black bandit pytest --group dev`
  - `poetry add isort@^5.11.5 --group dev` or `poetry add isort --group dev --allow-prereleases` (https://github.com/PyCQA/isort/issues/2084 + https://pypi.org/project/isort/6.0.0b2/#history + https://github.com/PyCQA/isort/issues/2083)
  - `flake8 = ">=5"` (https://github.com/asottile/flake8-2020/blob/v1.7.0/setup.cfg#L25)
  - https://python-poetry.org/docs/cli/#add (_Allow >=2.0.5 versions, without upper bound_)
  - `poetry config virtualenvs.in-project true --local`
  - `poetry env remove --all`
  - `poetry config --list`
- https://www.attrs.org/en/latest/index.html#getting-started
- Delete local tag: `git tag -d v0.0.1`

**Minimal boilerplate for the `Plugin` class**:

```python
import ast
import importlib.metadata
from typing import Any, Generator, Tuple, Type


class Plugin:
    name = __name__
    version = importlib.metadata.version(__name__)

    def __init__(self, tree: ast.AST) -> None:
        # The AST that represents a single file
        self._tree = tree

    def run(self) -> Generator[Tuple[int, int, str, Type[Any]], None, None]:
        # Tuple[line number, character offset, message]
        # Type[Any] is not being used (use `type(self)`)
        pass
```

**Syntactic sugar for `attr.ib(default=attr.Factory(f))`**:

```python
@attr.s
class C(object):
    x = attr.ib(factory=list)
    # instead of
    # x = attr.ib(default=attr.Factory(list))
```

