Skip to content

Commit

Permalink
docs: rewrite CONTRIBUTING.md (#1098)
Browse files Browse the repository at this point in the history
A general overhaul of the entire contribution guide, hopefully making it
easier to follow and understand. Also includes the documentation side of
#196, i.e. outlining the commit/PR naming convention.

This removes the detailed example which, while insightful, was way too
lengthy for new contributors. We could maybe re-add it on a separate
page in the future, but for now it's too much for this guide.
  • Loading branch information
shiftinv authored Sep 5, 2023
1 parent 80c8b32 commit 6689bb6
Show file tree
Hide file tree
Showing 2 changed files with 68 additions and 165 deletions.
232 changes: 67 additions & 165 deletions CONTRIBUTING.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,220 +2,122 @@

# Contributing to disnake

- [Bug Reports](#good-bug-reports)
- [Creating Pull Requests](#creating-a-pull-request)
First off, thanks for taking the time to contribute! It makes the library substantially better. :tada:

The following is a set of guidelines for contributing to the repository. These are not necessarily hard rules, but they streamline the process for everyone involved.

First off, thanks for taking the time to contribute. It makes the library substantially better. :+1:
### Table of Contents

The following is a set of guidelines for contributing to the repository. These are mostly guidelines, not hard rules.
- [Bug Reports](#good-bug-reports)
- [Creating Pull Requests](#creating-a-pull-request)
- [Overview](#overview)
- [Initial setup](#initial-setup)
- [Commit/PR Naming Guidelines](#commitpr-naming-guidelines)


## This is too much to read! I want to ask a question!

Generally speaking questions are better suited in our resources below.
> [!IMPORTANT]
> Please try your best not to create new issues in the issue tracker just to ask questions, unless they provide value to a larger audience.
Generally speaking, questions are better suited in our resources below.

- The official support server: https://discord.gg/disnake
- The official Discord server: https://discord.gg/disnake
- The [FAQ in the documentation](https://docs.disnake.dev/en/latest/faq.html)
- The project's [discussions section](https://github.com/DisnakeDev/disnake/discussions)

Please try your best not to create new issues in the issue tracker just to ask questions. Most of them don't belong there unless they provide value to a larger audience.

---

## Good Bug Reports

Please be aware of the following things when filing bug reports.
To report bugs (or to suggest new features), visit our [issue tracker](https://github.com/DisnakeDev/disnake/issues).
The issue templates will generally walk you through the steps, but please be aware of the following things:

1. Don't open duplicate issues. Please search your issue to see if it has been asked already. Duplicate issues will be closed.
2. When filing a bug about exceptions or tracebacks, please include the *complete* traceback. Without the complete traceback the issue might be **unsolvable** and you will be asked to provide more information.
3. Make sure to provide enough information to make the issue workable. The issue template will generally walk you through the process but they are enumerated here as well:
- A **summary** of your bug report. This is generally a quick sentence or two to describe the issue in human terms.
- Guidance on **how to reproduce the issue**. Ideally, this should have a small code sample that allows us to run and see the issue for ourselves to debug. **Please make sure that the token is not displayed**. If you cannot provide a code snippet, then let us know what the steps were, how often it happens, etc.
- Tell us **what you expected to happen**. That way we can meet that expectation.
- Tell us **what actually happens**. What ends up happening in reality? It's not helpful to say "it fails" or "it doesn't work". Say *how* it failed, do you get an exception? Does it hang? How are the expectations different from reality?
- Tell us **information about your environment**. What version of disnake are you using? How was it installed? What operating system are you running on? These are valuable questions and information that we use.
1. **Don't open duplicate issues**. Before you submit an issue, search the issue tracker to see if an issue for your problem already exists. If you find a similar issue, you can add a comment with additional information or context to help us understand the problem better.
2. **Include the *complete* traceback** when filing a bug report about exceptions or tracebacks. Without the complete traceback, it will be much more difficult for others to understand (and perhaps fix) your issue.
3. **Add a minimal reproducible code snippet** that results in the behavior you're seeing. This helps us quickly confirm a bug or point out a solution to your problem. We cannot reliably investigate bugs without a way to reproduce them.

If the bug report is missing this information then it'll take us longer to fix the issue. We will probably ask for clarification, and barring that if no response was given then the issue will be closed.
If the bug report is missing this information, it'll take us longer to fix the issue. We may ask for clarification, and if no response was given, the issue will be closed.

---

## Creating a Pull Request

Creating a pull request is fairly simple, just make sure it focuses on a single aspect and doesn't manage to have scope creep and it's probably good to go.

### Formatting
Creating a pull request is fairly straightforward. Make sure it focuses on a single aspect and avoids scope creep, then it's probably good to go.

We would greatly appreciate the code submitted to be of a consistent style with other code in disnake. This project follows PEP-8 guidelines (mostly) with a column limit of 100 characters.
If you're unsure about some aspect of development, feel free to use existing files as a guide or reach out via the Discord server.

### Overview

We use [`PDM`](https://pdm.fming.dev/) for development. If PDM is not already installed on your system, you can follow their [installation steps here](https://pdm.fming.dev/latest/#installation) to get started.
The general workflow can be summarized as follows:

Once PDM is installed and avaliable, use the following command to initialise a virtual environment, install the necessary development dependencies, and install the [`pre-commit`](https://pre-commit.com/#quick-start) hooks.
.
```
pdm run setup_env
```
1. Fork + clone the repository.
2. Initialize the development environment: `pdm run setup_env`.
3. Create a new branch.
4. Commit your changes, update documentation if required.
5. Add a changelog entry (e.g. `changelog/1234.feature.rst`).
6. Push the branch to your fork, and [submit a pull request!](https://github.com/DisnakeDev/disnake/compare)

The installed `pre-commit` hooks will automatically run before every commit, which will format/lint the code
to match the project's style. Note that you will have to stage and commit again if anything was updated!
Specific development aspects are further explained below.

Most of the time, running pre-commit will automatically fix any issues that arise, but this is not always the case.
We have a few hooks that *don't* resolve their issues automatically, and must be fixed manually.
One of these is the license header, which must exist in all files unless comments are not supported in those files, or they
are not text files, in which case exceptions can be made. These headers must exist following the format
documented at [https://spdx.dev/ids/](https://spdx.dev/ids/).

### Initial setup

### Scripts
We use [`PDM`](https://pdm.fming.dev/) as our dependency manager. If it isn't already installed on your system, you can follow the installation steps [here](https://pdm.fming.dev/latest/#installation) to get started.

To run all important checks and tests, use `pdm run nox`:
```sh
pdm run nox -R
Once PDM is installed, use the following command to initialize a virtual environment, install the necessary development dependencies, and install the [`pre-commit`](#pre-commit) hooks.
```

You can also choose to only run a single task; run `pdm run --list` to view all available scripts and use `pdm run <name>` to run them.

Some notes (all of the mentioned scripts are automatically run by `pdm run nox -R`, see above):
- If `pre-commit` hooks aren't installed, run `pdm run lint` manually to check and fix the formatting in all files.
**Note**: If the code is formatted incorrectly, `pre-commit` will apply fixes and exit without committing the changes - just stage and commit again.
- For type-checking, run `pdm run pyright`. You can use `pdm run pyright -w` to automatically re-check on every file change.
**Note**: If you're using VSCode and pylance, it will use the same type-checking settings, which generally means that you don't necessarily have to run `pyright` separately. However, there can be version differences which may lead to different results when later run in CI on GitHub.
- Tests can be run using `pdm run test`. If you changed some functionality, you may have to adjust a few tests - if you added new features, it would be great if you added new tests for them as well.

A PR cannot be merged as long as there are any failing checks.

### Changelogs

We use [towncrier](https://github.com/twisted/towncrier) for managing our changelogs. Each change is required to have at least one file in the [`changelog/`](changelog/README.rst) directory. There is more documentation in that directory on how to create a changelog entry.

### Git Commit Guidelines

- Use present tense (e.g. "Add feature" not "Added feature")
- Reference issues or pull requests outside of the first line.
- Please use the shorthand `#123` and not the full URL.

If you do not meet any of these guidelines, don't fret. Chances are they will be fixed upon rebasing but please do try to meet them to remove some of the workload.

---

## How do I add a new feature?

Welcome! If you've made it to this point you are likely a new contributor! This section will go through how to add a new feature to disnake.

Most attributes and data structures are broken up in to a file for each related class. For example, `disnake.Guild` is defined in [disnake/guild.py](disnake/guild.py), and `disnake.GuildPreview` is defined in [disnake/guild_preview.py](disnake/guild_preview.py). For example, writing a new feature to `disnake.Guild` would go in [disnake/guild.py](disnake/guild.py), as part of the `disnake.Guild` class.

### Adding a new API Feature

However, adding a new feature that interfaces with the API requires also updating the [disnake/types](disnake/types) directory to match the relevant [API specifications](https://discord.com/developers/docs). We ask that when making or receiving payloads from the API, they are typed and typehints are used on the functions that are processing said data. For example, take a look at `disnake.abc.Messageable.pins` (defined in [disnake/abc.py](disnake/abc.py)).


```py
async def pins(self) -> List[Message]:
channel = await self._get_channel()
state = self._state
data = await state.http.pins_from(channel.id)
return [state.create_message(channel=channel, data=m) for m in data]
```
*docstring removed for brevity*

Here we have several things occuring. First, we have annotated the return type of this method to return a list of `Message`s. As disnake supports Python 3.8, we use typing imports instead of subscripting built-ins — hence the capital ``List``.

The next interesting thing is `self._state`. The library uses a state-centric design, which means the state is passed around to most objects.
Every Discord model that makes requests uses that internal state and its `http` attribute to make requests to the Discord API. Each endpoint is processed and defined in [disnake/http.py](disnake/http.py) — and it's where `http.pins_from` is defined too, which looks like this:

```py
def pins_from(self, channel_id: Snowflake) -> Response[List[message.Message]]:
return self.request(Route("GET", "/channels/{channel_id}/pins", channel_id=channel_id))
$ pdm run setup_env
```

This is the basic model that all API request methods follow. Define the `Route`, provide the major parameters (in this example `channel_id`), then return a call to `self.request()`.

The `Response[]` part in the typehint is referring to `self.request`, as the important thing here is that `pins_from` is **not** a coroutine. Rather, `pins_from` does preprocessing and `self.request` does the actual work. The result from `pins_from` is awaited by `disnake.abc.Messageable.pins`.

The Route class is how all routes are processed internally. Along with `self.request`, this makes it possible to properly handle all ratelimits. This is why `channel_id` is provided as a kwarg to `Route`, as it is considered a major parameter for ratelimit handling.

#### Writing Documentation
Other tools used in this project include [black](https://black.readthedocs.io/en/stable/) + [isort](https://pycqa.github.io/isort/) (formatters), [ruff](https://beta.ruff.rs/docs/) (linter), and [pyright](https://microsoft.github.io/pyright/#/) (type-checker). For the most part, these automatically run on every commit with no additional action required - see below for details.

While a new feature can be useful, it requires documentation to be usable by everyone. When updating a class or method, we ask that you use
[Sphinx directives](https://www.sphinx-doc.org/en/master/usage/restructuredtext/directives.html#directive-versionadded) in the docstring to note when it was added or updated, and what about it was updated.
All of the following checks also automatically run for every PR on GitHub, so don't worry if you're not sure whether you missed anything. A PR cannot be merged as long as there are any failing checks.

For example, here is the docstring for `pins()`:

```py
"""|coro|
### Commit/PR Naming Guidelines

Retrieves all messages that are currently pinned in the channel.
This project uses the commonly known [conventional commit format](https://www.conventionalcommits.org/en/v1.0.0/).
While not necessarily required (but appreciated) for individual commit messages, please make sure to title your PR according to this schema:

.. note::
Due to a limitation with the Discord API, the :class:`.Message`
objects returned by this method do not contain complete
:attr:`.Message.reactions` data.
Raises
------
HTTPException
Retrieving the pinned messages failed.
Returns
-------
List[:class:`.Message`]
The messages that are currently pinned.
"""
```
<type>(<scope>)<!>: <short summary>
│ │ │ │
│ │ │ └─⫸ Summary in present tense, not capitalized, no period at the end
│ │ │
│ │ └─⫸ [optional] `!` indicates a breaking change
│ │
│ └─⫸ [optional] Commit Scope: The affected area, e.g. gateway, user, ...
└─⫸ Commit Type: feat|fix|docs|style|refactor|perf|test|build|ci|chore|revert
```

If we were to add a new parameter to this method, a few things would need to be added to this docstring. Lets pretend we're adding a parameter, ``oldest_first``.

We use NumPy style docstrings parsed with Sphinx's Napoleon extension — the primary documentation for these docstrings can be found [here](https://www.sphinx-doc.org/en/master/usage/extensions/napoleon.html).

```py
"""
...
Parameters
----------
oldest_first: bool
Whether to order the result by the oldest or newest pins first.
.. versionadded:: 2.9
Examples: `feat: support new avatar format` or `fix(gateway): use correct url for resuming connection`.
Details about the specific commit types can be found [here](https://github.com/commitizen/conventional-commit-types/blob/master/index.json).

...
"""
```

It is important that the section header comes **after** any description and admonitions that exist, as it will stop the parsing of the description.
### Formatting

The end result of these changes would be as follows:
This project follows PEP-8 guidelines (mostly) with a column limit of 100 characters, and uses the tools mentioned above to enforce a consistent coding style.

```py
"""|coro|
The installed [`pre-commit`](https://pre-commit.com/) hooks will automatically run before every commit, which will format/lint the code
to match the project's style. Note that you will have to stage and commit again if anything was updated!
Most of the time, running pre-commit will automatically fix any issues that arise.

Retrieves all messages that are currently pinned in the channel.

.. note::
### Pyright

Due to a limitation with the Discord API, the :class:`.Message`
objects returned by this method do not contain complete
:attr:`.Message.reactions` data.
For type-checking, run `pdm run pyright` (append `-w` to have it automatically re-check on every file change).
> [!NOTE]
> If you're using VSCode and pylance, it will use the same type-checking settings, which generally means that you don't necessarily have to run `pyright` separately.
> However, since we use a specific version of `pyright` (which may not match pylance's version), there can be version differences which may lead to different results.
Parameters
----------
oldest_first: bool
Whether to order the result by the oldest or newest pins first.

.. versionadded:: 2.9
### Changelogs

Raises
------
HTTPException
Retrieving the pinned messages failed.
We use [towncrier](https://github.com/twisted/towncrier) for managing our changelogs. Each change is required to have at least one file in the [`changelog/`](changelog/README.rst) directory, unless it's a trivial change. There is more documentation in that directory on how to create a changelog entry.

Returns
-------
List[:class:`.Message`]
The messages that are currently pinned.
"""
```

*If you're having trouble with adding or modifying documentation, don't be afraid to reach out!
We understand that the documentation can be intimidating, and there are quite a few quirks and limitations to be aware of.*
### Documentation
We use Sphinx to build the project's documentation, which includes [automatically generating](https://www.sphinx-doc.org/en/master/usage/extensions/autodoc.html) the API Reference from docstrings using the [NumPy style](https://sphinxcontrib-napoleon.readthedocs.io/en/latest/example_numpy.html).
To build the documentation locally, use `pdm run docs` and visit http://127.0.0.1:8009/ once built.
1 change: 1 addition & 0 deletions changelog/1098.misc.rst
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
Overhaul and simplify `contribution guide <https://github.com/DisnakeDev/disnake/tree/master/CONTRIBUTING.md>`__.

0 comments on commit 6689bb6

Please sign in to comment.