attrs/.github/CONTRIBUTING.md

299 lines
11 KiB
Markdown
Raw Normal View History

# How To Contribute
Thank you for considering contributing to *attrs*!
It's people like *you* who make it such a great tool for everyone.
This document intends to make contribution more accessible by codifying tribal knowledge and expectations.
Don't be afraid to open half-finished PRs, and ask questions if something is unclear!
Please note that this project is released with a Contributor [Code of Conduct](https://github.com/python-attrs/attrs/blob/main/.github/CODE_OF_CONDUCT.md).
By participating in this project you agree to abide by its terms.
Please report any harm to [Hynek Schlawack] in any way you find appropriate.
## Support
In case you'd like to help out but don't want to deal with GitHub, there's a great opportunity:
help your fellow developers on [Stack Overflow](https://stackoverflow.com/questions/tagged/python-attrs)!
The official tag is `python-attrs` and helping out in support frees us up to improve *attrs* instead!
## Workflow
- No contribution is too small!
Please submit as many fixes for typos and grammar bloopers as you can!
- Try to limit each pull request to *one* change only.
2022-08-23 08:12:44 +00:00
- Since we squash on merge, it's up to you how you handle updates to the `main` branch.
Whether you prefer to rebase on `main` or merge `main` into your branch, do whatever is more comfortable for you.
- *Always* add tests and docs for your code.
2022-08-23 08:12:44 +00:00
This is a hard rule; patches with missing tests or documentation won't be merged.
- Make sure your changes pass our [CI].
You won't get any feedback until it's green unless you ask for it.
2021-12-28 05:57:34 +00:00
- For the CI to pass, the coverage must be 100%.
If you have problems to test something, open anyway and ask for advice.
In some situations, we may agree to add an `# pragma: no cover`.
- Once you've addressed review feedback, make sure to bump the pull request with a short note, so we know you're done.
2022-07-28 07:46:31 +00:00
- Dont break backwards-compatibility.
## Local Development Environment
You can (and should) run our test suite using [*tox*].
However, youll probably want a more traditional environment as well.
First, create a [virtual environment](https://virtualenv.pypa.io/) so you don't break your system-wide Python installation.
We recommend using the Python version from the `.python-version-default` file in project's root directory.
If you're using [*direnv*](https://direnv.net), you can automate the creation of a virtual environment with the correct Python version by adding the following `.envrc` to the project root after you've cloned it to your computer:
```bash
layout python python$(cat .python-version-default)
```
If you're using tools that understand `.python-version` files like [*pyenv*](https://github.com/pyenv/pyenv) does, you can make it a link to the `.python-version-default` file.
---
Then, [fork](https://github.com/python-attrs/attrs/fork) the repository on GitHub.
Clone the fork to your computer:
2022-07-28 07:46:31 +00:00
```console
$ git clone git@github.com:<your-username>/attrs.git
2022-07-28 07:46:31 +00:00
```
2023-02-12 08:48:38 +00:00
Or if you prefer to use Git via HTTPS:
2022-07-28 07:46:31 +00:00
```console
$ git clone https://github.com/<your-username>/attrs.git
```
Then add the *attrs* repository as *upstream* remote:
```console
$ git remote add -t main -m main --tags upstream https://github.com/python-attrs/attrs.git
```
The next step is to sync your local copy with the upstream repository:
```console
$ git fetch upstream
2022-07-28 07:46:31 +00:00
```
This is important to obtain eventually missing tags, which are needed to install the development version later on.
See [#1104](https://github.com/python-attrs/attrs/issues/1104) for more information.
2023-02-12 08:48:38 +00:00
Change into the newly created directory and after activating a virtual environment install an editable version of *attrs* along with its tests and docs requirements:
2022-07-28 07:46:31 +00:00
```console
$ cd attrs
2023-02-12 08:48:38 +00:00
$ python -m pip install --upgrade pip wheel # PLEASE don't skip this step
2022-07-28 07:46:31 +00:00
$ python -m pip install -e '.[dev]'
```
At this point,
```console
$ python -m pytest
```
should work and pass.
You can *significantly* speed up the test suite by passing `-n auto` to *pytest* which activates [*pytest-xdist*](https://github.com/pytest-dev/pytest-xdist) and takes advantage of all your CPU cores.
For documentation, you can use:
2022-07-28 07:46:31 +00:00
```console
$ tox run -e docs-serve
2022-07-28 07:46:31 +00:00
```
This will build the documentation, and then watch for changes and rebuild it whenever you save a file.
To just build the documentation and run doctests, use:
```console
$ tox run -e docs
```
You will find the built documentation in `docs/_build/html`.
---
2022-07-28 07:46:31 +00:00
To file a pull request, create a new branch on top of the upstream repository's `main` branch:
```console
$ git fetch upstream
$ git checkout -b my_topical_branch upstream/main
```
Make your changes, push them to your fork (the remote *origin*):
```console
$ git push -u origin
```
and publish the PR in GitHub's web interface!
After your pull request is merged and the branch is no longer needed, delete it:
```console
$ git checkout main
$ git push --delete origin my_topical_branch && git branch -D my_topical_branch
```
Before starting to work on your next pull request, run the following command to sync your local repository with the remote *upstream*:
```console
$ git fetch upstream -u main:main
```
2022-08-23 08:12:44 +00:00
---
To avoid committing code that violates our style guide, we strongly advise you to install [*pre-commit*] and its hooks:
2022-07-28 07:46:31 +00:00
```console
$ pre-commit install
```
This is not strictly necessary, because our [*tox*] file contains an environment that runs:
2022-07-28 07:46:31 +00:00
```console
$ pre-commit run --all-files
```
and our CI has integration with [pre-commit.ci](https://pre-commit.ci).
But it's way more comfortable to run it locally and *git* catching avoidable errors.
## Code
- Obey [PEP 8](https://peps.python.org/pep-0008/) and [PEP 257](https://peps.python.org/pep-0257/).
We use the `"""`-on-separate-lines style for docstrings:
```python
def func(x):
"""
Do something.
:param str x: A very important parameter.
:rtype: str
"""
```
- If you add or change public APIs, tag the docstring using `.. versionadded:: 16.0.0 WHAT` or `.. versionchanged:: 16.2.0 WHAT`.
2023-12-30 12:05:53 +00:00
- We use [Ruff](https://github.com/astral-sh/ruff) to sort our imports, and we use [Black](https://github.com/psf/black) with line length of 79 characters to format our code.
2022-07-28 07:46:31 +00:00
As long as you run our full [*tox*] suite before committing, or install our [*pre-commit*] hooks (ideally you'll do both see [*Local Development Environment*](#local-development-environment) above), you won't have to spend any time on formatting your code at all.
If you don't, [CI] will catch it for you but that seems like a waste of your time!
## Tests
- Write your asserts as `expected == actual` to line them up nicely:
```python
x = f()
assert 42 == x.some_attribute
assert "foo" == x._a_private_attribute
```
- To run the test suite, all you need is a recent [*tox*].
It will ensure the test suite runs with all dependencies against all Python versions just as it will in our [CI].
2023-12-30 12:05:53 +00:00
If you lack some Python versions, you can can always limit the environments like `tox run -e py38,py39`, or make it a non-failure using `tox run --skip-missing-interpreters`.
In that case you should look into [*asdf*](https://asdf-vm.com) or [*pyenv*](https://github.com/pyenv/pyenv), which make it very easy to install many different Python versions in parallel.
- Write [good test docstrings](https://jml.io/pages/test-docstrings.html).
- To ensure new features work well with the rest of the system, they should be also added to our [*Hypothesis*](https://hypothesis.readthedocs.io/) testing strategy, which can be found in `tests/strategies.py`.
- If you've changed or added public APIs, please update our type stubs (files ending in `.pyi`).
## Documentation
2023-12-30 12:05:53 +00:00
- Use [semantic newlines] in [reStructuredText](https://www.sphinx-doc.org/en/stable/usage/restructuredtext/basics.html) and [Markdown](https://docs.github.com/en/get-started/writing-on-github/getting-started-with-writing-and-formatting-on-github/basic-writing-and-formatting-syntax) files (files ending in `.rst` and `.md`):
```rst
This is a sentence.
This is another sentence.
```
- If you start a new section, add two blank lines before and one blank line after the header, except if two headers follow immediately after each other:
```rst
Last line of previous section.
Header of New Top Section
-------------------------
Header of New Section
^^^^^^^^^^^^^^^^^^^^^
First line of new section.
```
- If you add a new feature, demonstrate its awesomeness on the [examples page](https://github.com/python-attrs/attrs/blob/main/docs/examples.md)!
### Changelog
If your change is noteworthy, there needs to be a changelog entry so our users can learn about it!
To avoid merge conflicts, we use the [*Towncrier*](https://pypi.org/project/towncrier) package to manage our changelog.
*towncrier* uses independent *Markdown* files for each pull request so called *news fragments* instead of one monolithic changelog file.
On release, those news fragments are compiled into our [`CHANGELOG.md`](https://github.com/python-attrs/attrs/blob/main/CHANGELOG.md).
You don't need to install *Towncrier* yourself, you just have to abide by a few simple rules:
- For each pull request, add a new file into `changelog.d` with a filename adhering to the `pr#.(change|deprecation|breaking).md` schema:
For example, `changelog.d/42.change.md` for a non-breaking change that is proposed in pull request #42.
- As with other docs, please use [semantic newlines] within news fragments.
- Wrap symbols like modules, functions, or classes into backticks so they are rendered in a `monospace font`.
2021-11-26 06:19:16 +00:00
- Wrap arguments into asterisks like in docstrings:
`Added new argument *an_argument*.`
- If you mention functions or other callables, add parentheses at the end of their names:
2021-12-28 05:57:34 +00:00
`attrs.func()` or `attrs.Class.method()`.
This makes the changelog a lot more readable.
- Prefer simple past tense or constructions with "now".
For example:
2021-12-28 05:57:34 +00:00
+ Added `attrs.validators.func()`.
+ `attrs.func()` now doesn't crash the Large Hadron Collider anymore when passed the *foobar* argument.
- If you want to reference multiple issues, copy the news fragment to another filename.
*Towncrier* will merge all news fragments with identical contents into one entry with multiple links to the respective pull requests.
Example entries:
```md
Added `attrs.validators.func()`.
The feature really *is* awesome.
```
or:
```md
`attrs.func()` now doesn't crash the Large Hadron Collider anymore when passed the *foobar* argument.
The bug really *was* nasty.
```
---
2023-12-30 12:05:53 +00:00
`tox run -e changelog` will render the current changelog to the terminal if you have any doubts.
## Governance
*attrs* is maintained by [team of volunteers](https://github.com/python-attrs) that is always open to new members that share our vision of a fast, lean, and magic-free library that empowers programmers to write better code with less effort.
If you'd like to join, just get a pull request merged and ask to be added in the very same pull request!
**The simple rule is that everyone is welcome to review/merge pull requests of others but nobody is allowed to merge their own code.**
[Hynek Schlawack] acts reluctantly as the [BDFL](https://en.wikipedia.org/wiki/Benevolent_dictator_for_life) and has the final say over design decisions.
[CI]: https://github.com/python-attrs/attrs/actions?query=workflow%3ACI
[Hynek Schlawack]: https://hynek.me/about/
[*pre-commit*]: https://pre-commit.com/
[*tox*]: https://tox.wiki/
[semantic newlines]: https://rhodesmill.org/brandon/2012/one-sentence-per-line/