Skip to content

Latest commit

 

History

History
354 lines (267 loc) · 15 KB

CONTRIBUTING.md

File metadata and controls

354 lines (267 loc) · 15 KB

Contributing to Kornia

Welcome !! This is the Kornia library contributor's corner. If you are reading this, it means that you have an interest in Differentiable Computer Vision, and are willing to contribute to the project.

Everyone is welcome to get involved with the project. There are different ways to contribute with your two cents:

  1. Ask/Answer questions:

    • Where can you ask questions?
      1. using the GitHub discussion at Kornia repo: GH Discussions
      2. our slack workspace to keep in touch with our core contributors and the community: Join Here
      3. using the #kornia tag in PyTorch Discuss
    • Please, don't use GitHub issues for Q&A.
    • In case you are a developer and want to learn more about the PyTorch ecosystem, we suggest you join the PyTorch slack. You can apply using this form: https://bit.ly/ptslack
  2. Report bugs through GitHub issues:

    • Do a quick search first to see whether others reported a similar issue.
    • In case you find an unreported bug, please open a new ticket.
    • Try to provide as much information as possible. Report using one of the available templates. Some tips:
      • Clear title and description of the issue.
      • Explain how to reproduce the error.
      • Report your package versions to facilitate the task.
      • Try to include a code sample/test that raises the error.
  3. Fix a bug or develop a feature from the roadmap:

    • We will always have an open ticket showing the current roadmap.
    • Pick an unassigned feature (or potentially propose a new one) or an open bug ticket.
    • Follow the instructions from Developing Kornia to setup your development environment and start coding.
    • Check our coding conventions. See more details below.
    • Run the test framework locally and make sure all works as expected before sending a pull request.
    • Open a Pull Request, get the green light from the CI, and get your code merged.
  4. Donate resources to the project:

    • In case you are an organization/institution that wants to give support, sponsor, or just use the project, please contact us.
    • We are open to starting any kind of collaboration and hearing feedback from you.
    • We pretend to provide features on demand. Reach us!
    • Currently looking for some kind of server donation to test CUDA code. (Please contact us).

Developing Kornia

To start to develop, please follow the steps below:

  1. Fork the kornia repository by clicking on the fork button on the repository page. This will create a copy of the Kornia repository under your GitHub account.

  2. Clone your fork of Kornia, and add the Kornia repository as a remote:

    $ git clone [email protected]:<your Github username>/kornia.git
    $ cd kornia
    $ git remote add upstream https://github.com/kornia/kornia.git
  3. Create a new branch with a meaningful name reflecting your contribution. See an example:

    $ git checkout upstream/main -b feat/foo_feature
    # or
    $ git checkout upstream/main -b fix/bar_bug

    🚨 Do not work on the main branch!

  4. Creating a development environment

    Using kornia script

    Assuming that you are on ubuntu, with nvidia-drivers installed. In bash, source the path.bash.inc script. This will install a local conda environment under ./.dev_env, which includes Pytorch and some dependencies (no root required).

    $ source ./path.bash.inc
    $ python setup.py develop
    $ python -c "import kornia; print(kornia.__version__)"

    To install, or update the conda environment run setup_dev_env.sh

    $ ./setup_dev_env.sh

    Manually setup

    Otherwise, use a virtual environment of your preference. We recommend using conda because it facilitates the Pytorch setup, mainly for those who have a GPU available.

    Example of creating and activating a virtualenv under venv name:

    # Using virtualenv
    $ virtualenv venv -p <your python version / alias> # e.g python3.10
    $ ./venv/bin/activate
    
    # Using conda
    $ conda create -p venv python=<language version> # e.g 3.10
    $ conda activate venv/

    Setup Pytorch and Kornia:

    # Installing pytorch: https://pytorch.org/get-started/locally/
    # With pip
    $ pip install torch
    # With conda
    $ conda install pytorch cudatoolkit -c pytorch -c nvidia # For GPU env
    # or
    $ conda install pytorch cpuonly -c pytorch # For CPU env
    
    # Installing Kornia for development
    $ pip install -e .[dev]
    
    # If you want to contribute to the documentation
    $ pip install -e .[docs]

    Attention: If Kornia was already installed in your virtual environment, remove it with pip uninstall kornia before reinstalling it in editable mode with the -e flag.

  5. Develop the code on your branch, and before creating the pull request, make sure to ensure the code passes the checks.

    As you develop your code, you should also create test cases for your code. As well as, In addition to ensuring that the other tests continue to pass. You can run the tests with:

    $ pytest tests/<TEST_TO_RUN>.py --dtype=float32,float64 --device=all

    With the dtype argument, run the tests using tensors with all dtypes desired. Options: bfloat16, float16, float32, float64, and all.

    In the same way, the device, will run the tests using tensors on the device desired. Options: cpu, cuda, tpu, mps, and all.

    Kornia relies on pre-commit to run code quality tools. Make sure to have pre-commit under your dev environment, otherwise, you can install the tools manually and run them with the help of the available commands of the Makefile. Read more about the code standards adopted here.

Benchmarking

We have a benchmark suite configured in benchmarks/. We used the pytest-benchmark library to benchmark our functions units.

Our Makefile has an benchmark command as an alias on how to run our benchmarks.

# To run all suite
$ make benchmark

# To run a specific file you can pass `BENCHMARK_SOURCE`
$ make benchmark BENCHMARK_SOURCE=benchmarks/augmentation/2d_geometric_test.py

# To run a specific benchmark you use `BENCHMARK_SOURCE` as the pytest standard behaviour
$ make benchmark BENCHMARK_SOURCE=benchmarks/augmentation/2d_geometric_test.py::test_aug_2d_elastic_transform

# To update the optimizer backends desired to execute you can pass `BENCHMARK_BACKENDS=`
$ make benchmark BENCHMARK_BACKENDS='inductor,eager'

# To pass other options to the runner, you can use `BENCHMARK_OPTS`
# Example, setup to run the benchmark on cuda on verbose mode
$ make benchmark BENCHMARK_OPTS='--device=cuda -vv'

We use the same tests generator suite, so you can setup the device within --device, the dtype within --dtype, and the optimizer backend within --optimizer.

The optimizer backend supported on the suite, are the torch compile backend on non experimental mode, and the '' or None which will do the same as eager mode and do anything, and 'jit' which will try to torch.jit.script the operation.

You can use the BENCHMARK_OPTS on make benchmark to overload the default options we use on pytest-benchmark.

We are using as default:

  • the warmup, because the optimizer/jit may had an overhead.
  • the group: to display the benchmark per each test
  • the precision: to have a better precision on the results
  • the default for BENCHMARK_BACKENDS are 'inductor,eager'.
  • the default for BENCHMARK_SOURCE is benchmarks/.

You can also run the benchmark within docker:

$ make benchmark-docker

which will build and run the image docker/Dockerfile.benchmark. The benchmark command can be used within BENCHMARK_BACKENDS and BENCHMARK_SOURCE.

Coding Standards

This section provides general guidance for developing code for the project. The following rules will serve as a guide in writing high-quality code that will allow us to scale the project and ensure that the code base remains readable and maintainable.

  • Use meaningful names for variables, functions, and classes.

  • Write small incremental changes:

    • To have a linear and clean commits history, we recommend committing each small change that you do to the source code.
    • Clear commit messages will help to understand the progress of your work.
    • Please, avoid pushing large files.
  • Add tests:

    • Tests are crucial and we expect you to write unit tests for each of the functionalities that you implement. It is also a good idea to group the tests for functionalities

    • At testing/ directory we have a bunch of functions to help you to produce meaningful tests. Feel free, to add any functionality which you that you think is essential and can be used with the test suite. Under this testing directory should go all code which are needed under the tests and aren't tests or pytest configs (fixtures, etc).

      from testing.base import BaseTester
      
      class TestMyFunction(BaseTester):
          # To compare the actual and expected tensors use `self.assert_close(...)`
      
      
          def test_smoke(self, device, dtype):
              # test the function with different parameters arguments, to check if the function at least runs with all the
              # arguments allowed.
              pass
      
          def test_exception(self, device, dtype):
              # tests the exceptions which can occur on your function
      
              # example of how to properly test your exceptions
              # with pytest.raises(<raised Error>) as errinfo:
              #     your_function(<set of parameters that raise the error>)
              # assert '<msg of error>' in str(errinfo)
      
              pass
      
          def test_cardinality(self, device, dtype):
              # test if with different parameters the shape of the output is the expected
              pass
      
          def test_feature_foo(self, device, dtype):
              # test basic functionality
              pass
      
          def test_feature_bar(self, device, dtype):
              # test another functionality
              pass
      
          def test_gradcheck(self, device):
              # test the functionality gradients
              # Uses `self.gradcheck(...)`
              pass
      
          def test_dynamo(self, device, dtype, torch_optimizer):
              #  test the functionality using dynamo optimizer
      
              # Example of how to properly test your function for dynamo
              # inputs = (...)
              # op = your_function
              # op_optimized = torch_optimizer(op)
              # self.assert_close(op(inputs), op_optimized(inputs))
      
              pass
  • Tests should cover different devices (CPU, CUDA, etc), dtypes, and different input batch sizes. The device, and dtype, are generated from the arguments (--dtype and --device) as explained before. These arguments when invoking the tests suits with pytest, will generate all possibilities, providing fixtures for all functions. See an example:

    import pytest
    
    @pytest.mark.parametrize("batch_size", [1, 2, 5])
    def test_smoke(batch_size, device, dtype):
        x = torch.rand(batch_size, 2, 3, device=device, dtype=dtype)
        assert x.shape == (batch_size, 2, 3)
  • We give support to static type checker for Python >= 3.8

    • Please, read MyPy cheatsheet for Python 3.
    • It is recommended to use typing inside the function, when it would increase readability.
    • Try to use all things available under kornia.core, e.g. from kornia.core import Tensor
    • For modules which not support anymore JIT consider, adding from __future__ import annotations, to enable the new features of typing.
    • Always type function input and output, e.g.:
      from __future__ import annotations
      from kornia.core import Tensor
      
      def homography_warp(
        patch_src: Tensor,
        dst_homo_src: Tensor,
        dsize: tuple[int, int],
        mode: str = 'bilinear',
        padding_mode: str = 'zeros'
      ) -> Tensor:
  • We suggest using new Python 3's f-Strings improved string formatting syntax:

    Guidelines: PEP 498 - Literal String Interpolation

  • Format your code:

    • We follow PEP8 style guide.
    • Use pre-commit to autoformat each commit before push: pre-commit.com For doing so, just install it for this repository by running the command: pre-commit install on your terminal
  • Changes to PEP8:

    • Line length is 120 characters.

    • W504 (line break after binary operator) is sometimes acceptable. E.g.

      determinant = A[:, :, 0:1, 0:1] * A[:, :, 1:2, 1:2] -
                    A[:, :, 0:1, 1:2] * A[:, :, 1:2, 0:1])
  • Using 3rd party libraries:

  • Everything from the standard library (https://docs.python.org/3/library/) and PyTorch (https://pytorch.org/) is OK. It doesn’t means, that one should import urllib just because, but doing it when needed is fine.

Pull Request

Once you finish implementing a feature or bug-fix, please send a Pull Request to https://github.com/kornia/kornia through the website.

If you are not familiar with creating a Pull Request, here are some guides:

Once your pull request is created, our continuous build system will check your pull request. Continuous build will test that:

  • pytest all tests pass.
  • Test coverage remains high. Please add unit tests so we maintain our code coverage.
  • Typing with mypy type checks the Python code.
  • If the docs can be generated successfully
  • pre-commit ci
    • ruff accepts the code style (our guidelines are based on PEP8) and checks if the code is well formatted
    • docformatter checks if the code docstrings are well formatted
    • and some other checks. Check our pre-commit config

If your code fails one of these checks, you will be expected to fix your pull request before it is considered.

Licence

By contributing to the project, you agree that your contributions will be licensed under the Apache LICENSE. Check the complete license here