Thank you for your interest in contributing!
To start, please read the library docs thoroughly. If you don't find what you are looking for, proceed with the steps below.
Please file a bug report. If you are not using the latest version of the library, please upgrade and see if that fixes the issue. If not, please create a minimal example that demonstrates the bug, and instructions on how to create that setup from a new virtual environment. Also include any error tracebacks (unabridged when possible). This will help a lot when diagnosing the bug. Do not use pictures to include the traceback.
You can suggest new features to be implemented via a feature request. You can ask me to implement it, or work on it yourself, but all features should be discussed and agreed upon first before any coding is done.
Please ask it in the discussions section instead of creating an issue. If your question warrants an issue, I'll ask you to create it. Questions about clarifying documentation are appreciated!
Once you have created a feature request, we have agreed on an implementation, and you wish to work on it, follow these steps to create a pull request.
- Fork the repository.
- Clone your fork and create a new branch from the
main
branch. - Set up the environment.
- Make changes and write tests following these guidelines.
- Add documentation when applicable following these guidelines.
- Push the changes to your fork.
- Create a pull request targeting the main branch.
- Sit back while your pull request is reviewed.
Note that a pull request should always be aimed at solving a single issue. If you want multiple issues solved, make separate pull requests for each. Spelling mistakes are the exception, they are always welcome!
Pull requests should be kept as small as possible while following the guidelines mentioned above. Smaller pull request are easier to review and test, which helps them get merged.
Pull requests will be reviewed automatically and manually.
In the automated phase, GitHub Actions will run testing pipelines for all supported operating systems and python versions, and pre-commit CI will check linting rules. If you encounter any errors, try to fix them based on what the pipelines tell you. If coverage is lowered, add tests, noting the guidelines here. Don't be afraid to ask for advice if you're unsure what is wrong.
Note for first-time contributors: Checks are not allowed to run automatically for first-time contributors, so you'll need me to approve them each time you push new code.
Known issues: GitHub Actions might fail unexpectedly when installing dependencies. If this happens, the failed jobs need to be run again a few times to get past this.
In the manual phase, I will review the pull request by adding comments with suggestions for changes. If you agree with the suggestions, implement them, and push the changes to you fork; the pull request wil be updated automatically. You can either amend your previous commits or add more commits, either is fine. If you disagree with the suggestions, provide your reasons for disagreeing, and we can discuss what to do.
Once all automated checks have passed, and I have accepted the pull request, your code will be
merged to the main
branch. Any related issues should be closed as completed.
I'll usually make a new release after each new feature, but if not, you can also ask for one.
- Increment the version in
pyproject.toml
according semantic versioning rules. - Push the change to the
main
branch with the commit messageBump version
. - Draft a new release on GitHub.
- Use
v{version}
(e.g. v1.2.3) for the tag name, andRelease {version}
for the release title, using the same version that's inpyproject.toml
. - Fill in the release description.
- Add any attachments when applicable.
- Use
- Publish the release. This will start the
release
pipeline on GitHub Actions. - Check that the release pipeline was successful. If not, delete the tag from origin
with
git push --delete origin {tag_name}
and fix the issue before trying again.
Note, that the release will be made with the
pyproject.toml
version and not thetag
version, and that this is not checked anywhere, so make sure they match!
- Install Poetry.
- Install Make.
- Windows: Install Chocolatey and then
choco install make
. - Mac: Install Homebrew and then
brew install make
. - Ubuntu:
apt install make
.
- Windows: Install Chocolatey and then
- Run
poetry install
to create a virtual environment and install project dependencies. - Run
make hook
to install the pre-commit hooks.
Run make help
to list all existing development commands and their descriptions.
Tests can be run with make tests
, and individual tests with make test <test_name>
.
This will run tests in you local environment.
You can also test your code in multiple environments with tox. To do this, you must
install python interpreters for all python version the library supports, then run
make tox
.
Linting can be run on-demand with make pre-commit
, or automatically before commits
when installed with make hook
-
All code should be tested with 100% coverage
- Do not write test simply to archive 100% coverage. Instead, write tests for all the ways the feature could be used (use cases), including ways that should not work, and then test for coverage. If you find uncovered code, see if you can remove it, or maybe you simply missed a use case. You should always need more tests to cover the all use cases than to achieve 100% coverage.
- Comments that ignore test coverage (
# pragma: no cover
) should be used very sparingly. They are often not necessary and can lead to undocumented behavior if you are not careful.
-
All code should be typed when possible.
- Tests are an exception to this; typing them is optional.
- Make sure the typing methods used are supported in all python versions
the library supports (e.g., use
List[int]
instead oflist[int]
for Python 3.8 support). CI will yell at you if you don't. - Create all custom types in
query_optimizer/typing.py
and import them from there. This avoids circular imports. - Use of
TypedDict
is encouraged where dicts would be used. - Also import common types like
List
fromquery_optimizer/typing.py
instead of the built-intyping
module. This is to make importing types more consistent across the codebase, and allows conditional import logic with thetyping_extensions
module for newer typing methods likeParamSpec
to be contained in a single place. - Using
mypy
for static type checking is optional, and will likely lead to many "errors" detected.
-
All functions, methods, and classes should include a docstring (*) in reStructuredText format.
- (*) Code that is short and clearly self-documenting does not necessarily need a docstring.
As a dumb example,
def sum(i: int, j: int) -> int: return i + j
does not need a docstring. This applies more broadly to arguments, e.g., when a function might need a docstring, the arguments might not need explicit documentation. - Keep the docstring to the point. Each line of documentation has a maintenance cost. Documentation is not an excuse to write code that is hard to understand. Docstrings should not include code examples, they belong to docs.
- (*) Code that is short and clearly self-documenting does not necessarily need a docstring.
As a dumb example,
-
All code should be linted using the provided pre-commit hooks.
- Easiest way to do this is to install the pre-commit hooks with
make hook
. This will make sure the pre-commit hooks will run automatically when you make a commit. - Comments that ignore linting rules (
# type: ignore
,# fmt: off
,# noqa
) should be used very sparingly. They are often not necessary and can lead to undocumented behavior if you are not careful.
- Easiest way to do this is to install the pre-commit hooks with
- All documentation is written in
docs/
using markdown, and built with mkdocs. - Write in idiomatic english, using simple language.
- Keep examples simple and self-contained. Don't try to list all possible scenarios at once. Give the reader time to understand the basics before going over edge cases.
- Use markdown features, like fenced code blocks, blockquotes, horizontal rules, or links, to emphasize and format text.
- If diagrams are needed, use mermaid.js inside a fenced code block.
- Break up lines around the 100 characters mark. This improves readability on wider monitors without adjusting the window size (and when not using text-wrapping).
- Do not use emojis.
- Double-check for spelling mistakes and grammar.
By contributing, you agree that your contributions will be licensed under the MIT Licence.