Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/FlyingBird95/openapi-builder

A Flask extension to automatically add OpenAPI documentation for your Flask app.
https://github.com/FlyingBird95/openapi-builder

blueprint flask flask-extension openapi openapi-documentation openapi-generator openapi-specification python swagger swagger-documentation

Last synced: about 1 month ago
JSON representation

A Flask extension to automatically add OpenAPI documentation for your Flask app.

Awesome Lists containing this project

README

        

|GHA tests| |Documentation| |Codecov report| |pre-commit| |black| |pypi| |downloads|

`OpenAPI Builder `_
===================================================================
A Flask_ extension for automatically adding OpenAPI_ documentation to your Flask app.

.. _Flask: https://flask.palletsprojects.com/
.. _OpenAPI: https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.0.3.md

Features
--------
OpenAPI Builder is an extension for your RESTful Flask API to easily build OpenAPI (formerly known as Swagger)
documentation using the `standard specification `_. It is designed to generate the documentation
for your resources, by specifying a few simple parameters per endpoint. OpenAPI Builder is not dependent on a specific
(de-)serialization library (such as marshmallow_ or halogen_), since its goal is abstract them. The most powerful
feature of this package is to inspect the (de-)serialization class. Thus, extending your class will automatically
update your documentation. This ensures that your documentation is always up to date.

To summarize, the key features are:

- **Generate documentation** according to the OpenAPI standard. This is the internet standard for writing RESTful API
documentation.
- **Keep documentation up to date.** This is due to generating the documentation by inspecting the (de-)serialization
class.
- **Reduce up to 90% development time writing documentation.** This is due to automating the repetitive work of writing
documentation.
- **Abstract upon any (de-)serialization class.** It works with the parsing library of your choice.

.. _standard_specification: https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.0.3.md
.. _marshmallow: https://marshmallow.readthedocs.io/en/stable/
.. _halogen: https://halogen.readthedocs.io/en/latest/

How it works
------------
.. image:: img/flow-diagram.png

OpenAPI Builder uses the Flask_ application and schemas to retrieve information about the interaction of your API.
This information is used to generate a specification, which is a JSON-file according to the industry standard Open API
format. This specification can be visualized using any UI that supports the Open API specification, such as
`Swagger UI `_, or Redoc_.

|swagger| |redoc|

.. |swagger| image:: img/swagger-ui.png
:width: 45%
.. |redoc| image:: img/redoc-ui.png
:width: 45%

.. _Flask: https://flask.palletsprojects.com/
.. _swagger_ui: https://swagger.io/tools/swagger-ui/
.. _Redoc: https://redoc.ly/

Quick start
-----------
Installation is simple via PIP:

.. code:: bash

pip install openapi-builder

After installation, it's time to build the documentation for the resources in your Flask application, and expose them
so your external users can read them. For working examples, see `this folder `_.

.. _this-folder: https://github.com/FlyingBird95/openapi-builder/tree/master/examples/

Initialising the extension
~~~~~~~~~~~~~~~~~~~~~~~~~~
The extension can be initialized using the following snippet:

.. code:: python

from flask import Flask
from openapi_builder import OpenApiDocumentation

app = Flask(__name__)

# option 1: initialize directly
documentation = OpenApiDocumentation(app=app)

# option 2: add the app later.
documentation = OpenApiDocumentation()
documentation.init_app(app)

The generated documentation is automatically exposed at http://localhost:5000/documentation. If you
don't want this, you can disable it in the options. The included documentation UI is taken from
the `swagger-codegen page `_.

.. _swagger: https://github.com/swagger-api/swagger-codegen

Adding resources
~~~~~~~~~~~~~~~~
After the documentation UI has been setup, it's time to add documentation for your resources. This can be achieved
using the :code:`add_documentation` decorator. Given an example endpoint for your RESTful API, that might look like
this:

.. code:: python

from flask import Flask
from marshmallow import Schema, fields

class UserSchema(Schema):
"""User response schema."""

name = fields.Str()
"""Name of the user."""

email = fields.Email()
"""Email of the user."""

register_date = fields.DateTime()
"""When the user was registered."""

app = Flask(__name__) # use the same app as used for the UI.

@app.route("/users")
def users():
"""Returns a list of users, serialized using Marshmallow."""
users = [
User(name="John", lastname="Doe", email="[email protected]"),
User(name="Jane", lastname="Doe", email="[email protected]"),
]
return jsonify(UserSchema(many=True).dumps(users))

Documentation for this resource is generated by adding the decorator.

.. code:: python

...
from http import HTTPStatus
from openapi_builder import add_documentation

...

@app.route("/users")
@add_documentation(
responses={HTTPStatus.OK: UserSchema(many=True)},
summary="Returns a list of users.",
description="More extensive information that fully describes the endpoint.",
)
def users():
...

For a full overview of all applicable parameters, see `this configuration page `_.

That's all folks. You can view your documentation at http://localhost:5000/documentation, and inspect the configuration
at: http://localhost:5000/documentation-configuration. Whenever the schema is updated by adding/removing/updating a new
property, it is automatically reflected in the OpenAPI documentation.

.. _doc-page: https://flyingbird95.github.io/openapi-builder/configuration.html

Contributing
------------

Contributions are welcome! If you can see a way to improve this package:

- Do click the fork button
- Make your changes and submit a pull request.

Or to report a bug or request something new, make an issue.

Development
-----------
This section describes development standards for this project.

Black
~~~~~

Black_ is an uncompromising Python code formatter.
By using it, you cede control over minutiae of hand-formatting.
But in return, you no longer have to worry about formatting your code correctly, since black will handle it.
Blackened code looks the same for all authors, ensuring consistent code formatting within your project.

The format used by Black makes code review faster by producing the smaller diffs.

Black's output is always stable.
For a given block of code, a fixed version of black will always produce the same output.
However, it should be noted that different versions of black will produce different outputs.
Black is configured here:

- requirements-dev.txt, `L1 `__
- .pre-commit-config.yaml, `L14 `__,
`L28 `__, and
`L47 `__

.. _black: https://github.com/psf/black

pre-commit
~~~~~~~~~~

This repository comes with a pre-commit_ stack.
This is a set of git hooks which are executed every time a commit is made.
The hooks catch errors as they occur, and will automatically fix some of these errors.

To set up the pre-commit hooks, run the following code from within the repo directory::

pip install -r requirements-dev.txt
pre-commit install

Whenever trying to commit code which is flagged by the pre-commit hooks, the commit will not go through.
Some of the pre-commit hooks (such as black_, isort_) will automatically modify the code to fix the issues.
When this happens, you'll have to stage the changes made by the commit hooks and then commit again.
Other pre-commit hooks will not modify the code and will just tell you about issues which you'll then have to manually fix.

To run the pre-commit stack on all the files at any time::

pre-commit run --all-files

To force a commit to go through without passing the pre-commit hooks use the ``--no-verify`` flag::

git commit --no-verify

The pre-commit stack which comes with the template is highly opinionated, and includes the following operations:

- Code is reformatted to use the black_ style.
Any code inside docstrings will be formatted to black using blackendocs_.
All code cells in Jupyter notebooks are also formatted to black using black_nbconvert_.

- All Jupyter notebooks are cleared using nbstripout_.

- Imports are automatically sorted using isort_.

- flake8_ is run to check for conformity to the python style guide PEP-8_, along with several other formatting issues.

- setup-cfg-fmt_ is used to format any setup.cfg files.

- Several `hooks from pre-commit `_ are used to screen for non-language specific git issues, such as incomplete git merges, overly large files being committed to the repo, bugged JSON and YAML files.
JSON files are also prettified automatically to have standardised indentation.
Entries in requirements.txt files are automatically sorted alphabetically.

- Several `hooks from pre-commit specific to python `_ are used to screen for rST formatting issues, and ensure noqa flags always specify an error code to ignore.

Once it is set up, the pre-commit stack will run locally on every commit.
The pre-commit stack will also run on github with one of the action workflows, which ensures PRs are checked without having to rely on contributors to enable the pre-commit locally.

.. _black_nbconvert: https://github.com/dfm/black_nbconvert
.. _blackendocs: https://github.com/asottile/blacken-docs
.. _flake8: https://gitlab.com/pycqa/flake8
.. _isort: https://github.com/timothycrosley/isort
.. _nbstripout: https://github.com/kynan/nbstripout
.. _PEP-8: https://www.python.org/dev/peps/pep-0008/
.. _pre-commit: https://pre-commit.com/
.. _pre-commit-hooks: https://github.com/pre-commit/pre-commit-hooks
.. _pre-commit-py-hooks: https://github.com/pre-commit/pygrep-hooks
.. _setup-cfg-fmt: https://github.com/asottile/setup-cfg-fmt

Automated documentation
~~~~~~~~~~~~~~~~~~~~~~~

The script ``docs/conf.py`` is based on the Sphinx_ default configuration.
It is set up to work well out of the box, with several features added in.

GitHub Pages
^^^^^^^^^^^^

Documentation is deployed to `GitHub Pages`_ and is available at https://flyingbird95.github.io/openapi-builder/.

The gh-pages documentation is refreshed every time there is a push to the master branch.

Note that only one copy of the documentation is served (the latest version).

.. _GitHub Pages: https://pages.github.com/

Building locally
^^^^^^^^^^^^^^^^

The web documentation can be built locally with::

pip install -r requirements-docs.txt
make -C docs html

And view the documentation like so::

sensible-browser docs/_build/html/index.html

Or build the pdf documentation::

make -C docs latexpdf

On Windows, this becomes::

cd docs
make html
make latexpdf
cd ..

Other documentation features
^^^^^^^^^^^^^^^^^^^^^^^^^^^^

- The README.rst will become part of the generated documentation (via a link file ``docs/source/readme.rst``).
Note that the first line of README.rst is not included in the documentation, since this is expected to contain badges which we render on GitHub, but not include in the documentation pages.

- The docstrings in all modules, functions, classes and methods will be used to build a set of API documentation using autodoc_.
Our ``docs/conf.py`` is also set up to automatically call autodoc whenever it is run, and the output files which it generates are on the gitignore list.
This means it will automatically generate a fresh API description which exactly matches the current docstrings every time the documentation is generated.

- Docstrings can be formatted in plain reST_, or using the `numpy format`_ (recommended), or `Google format`_.
Support for numpy and Google formats is through the napoleon_ extension (which is enabled by default).

- The reference functions in the python core and common packages and they will automatically be hyperlinked to the appropriate documentation in the documentation.
This is done using intersphinx_ mappings, which can be seen at the bottom of the ``docs/conf.py`` file.

- The documentation theme is sphinx-book-theme_.
Alternative themes can be found at sphinx-themes.org_, sphinxthemes.com_, and writethedocs_.

.. _autodoc: http://www.sphinx-doc.org/en/master/usage/extensions/autodoc.html
.. _Google format: https://sphinxcontrib-napoleon.readthedocs.io/en/latest/example_google.html#example-google
.. _intersphinx: http://www.sphinx-doc.org/en/master/usage/extensions/intersphinx.html
.. _napoleon: https://www.sphinx-doc.org/en/master/usage/extensions/napoleon.html
.. _numpy format: https://sphinxcontrib-napoleon.readthedocs.io/en/latest/example_numpy.html#example-numpy-style-python-docstrings
.. _Sphinx: https://www.sphinx-doc.org/
.. _sphinx-book-theme: https://sphinx-book-theme.readthedocs.io/
.. _sphinx-themes.org: https://sphinx-themes.org
.. _sphinxthemes.com: https://sphinxthemes.com/
.. _reST: http://docutils.sourceforge.net/rst.html
.. _writethedocs: https://www.writethedocs.org/guide/tools/sphinx-themes/

Consolidated metadata
~~~~~~~~~~~~~~~~~~~~~

Package metadata is consolidated into one place, the file ``openapi_builder/__meta__.py``.
This is done to only write the metadata once in this centralised location, and everything else (packaging, documentation, etc) picks it up from there.
This is similar to `single-sourcing the package version`_, but for all metadata.

This information is available to end-users with ``import openapi_builder; print(openapi_builder.__meta__)``.
The version information is also accessible at ``openapi_builder.__version__``, as per PEP-396_.

.. _PEP-396: https://www.python.org/dev/peps/pep-0396/#specification
.. _single-sourcing the package version: https://packaging.python.org/guides/single-sourcing-package-version/

setup.py
~~~~~~~~

The ``setup.py`` script is used to build and install the package.

The package can be installed from source with::

pip install .

or alternatively with::

python setup.py install

But do remember that as a developer, the package should be installed in editable mode, using either::

pip install --editable .

or::

python setup.py develop

which will mean changes to the source will affect the installed package immediately without having to reinstall it.

By default, when the package is installed only the main requirements, listed in ``requirements.txt`` will be installed with it.
Requirements listed in ``requirements-dev.txt``, ``requirements-docs.txt``, and ``requirements-test.txt`` are optional extras.
The ``setup.py`` script is configured to include these as extras named ``dev``, ``docs``, and ``test``.
They can be installed along with::

pip install .[dev]

etc.
Any additional files named ``requirements-EXTRANAME.txt`` will also be collected automatically and made available with the corresponding name ``EXTRANAME``.
Another extra named ``all`` captures all of these optional dependencies.

The README file is automatically included in the metadata when setup.py build wheels for PyPI.
The rest of the metadata comes from ``openapi_builder/__meta__.py``.

Our template setup.py file is based on the `example from setuptools documentation `_, and the comprehensive example from `Kenneth Reitz `_ (released under `MIT License `__), with further features added.

.. _kennethreitz/setup.py: https://github.com/kennethreitz/setup.py
.. _setuptools-setup.py: https://setuptools.readthedocs.io/en/latest/setuptools.html#basic-use

GitHub Actions Workflows
~~~~~~~~~~~~~~~~~~~~~~~~

GitHub features the ability to run various workflows whenever code is pushed to the repo or a pull request is opened.
This is one service of several services that can be used to continually run the unit tests and ensure changes can be integrated together without issue.
It is also useful to ensure that style guides are adhered to

Five workflows are included:

docs
The docs workflow ensures the documentation builds correctly, and presents any errors and warnings nicely as annotations.
The available html documentation is automatically deployed to the gh-pages branch and https://flyingbird95.github.io/openapi-builder/.

pre-commit
Runs the pre-commit stack.
Ensures all contributions are compliant, even if a contributor has not set up pre-commit on their local machine.

lint
Checks the code uses the black_ style and tests for flake8_ errors.
Note that the lint workflow is superfluous, due to the pre-commit hooks.

test
Runs the pytest, and pushes coverage reports to Codecov_.

release candidate tests
The release candidate tests workflow runs the unit tests on more Python versions and operating systems than the regular test workflow.
This runs on all tags, plus pushes and PRs to branches named like "v1.2.x", etc.
Wheels are built for all the tested systems, and stored as artifacts for convenience when shipping a new distribution.

When the ``publish`` job is enabled on the release candidate tests workflow, it can also push built release candidates to the `Test PyPI `_ server.

.. _Codecov: https://codecov.io/
.. _ci-packaging: https://packaging.python.org/guides/publishing-package-distribution-releases-using-github-actions-ci-cd-workflows/
.. _github-secrets: https://docs.github.com/en/actions/reference/encrypted-secrets
.. _pypi-api-token: https://pypi.org/help/#apitoken
.. _pypi-publish: https://github.com/pypa/gh-action-pypi-publish
.. _testpypi: https://test.pypi.org/

.. |GHA tests| image:: https://github.com/flyingbird95/openapi-builder/workflows/tests/badge.svg
:target: https://github.com/flyingbird95/openapi-builder/actions?query=workflow%3Atests
:alt: GHA Status
.. |Documentation| image:: https://github.com/flyingbird95/openapi-builder/workflows/docs/badge.svg
:target: https://flyingbird95.github.io/openapi-builder/index.html
:alt: Documentation
.. |Codecov report| image:: https://codecov.io/github/flyingbird95/openapi-builder/coverage.svg?branch=master
:target: https://codecov.io/github/flyingbird95/openapi-builder?branch=master
:alt: Coverage
.. |pre-commit| image:: https://img.shields.io/badge/pre--commit-enabled-brightgreen?logo=pre-commit&logoColor=white
:target: https://github.com/pre-commit/pre-commit
:alt: pre-commit
.. |black| image:: https://img.shields.io/badge/code%20style-black-000000.svg
:target: https://github.com/psf/black
:alt: black
.. |pypi| image:: https://badge.fury.io/py/openapi-builder.svg
:target: https://badge.fury.io/py/openapi-builder
:alt: PyPI
.. |downloads| image:: http://pepy.tech/badge/openapi-builder
:target: https://pepy.tech/project/openapi-builder
:alt: Downloads