Skip to main content

A package to convert Python type annotations into JSON schemas

Project description

Test

pytojsonschema

Package that uses static analysis - ast - to convert Python 3 function type annotations to JSON schemas.

This allows you to auto-generate the validation schemas for JSON-RPC backend functions written in Python.

Current support is for Python 3.8+ and JSON schema draft 7+.

Getting started

Installation

From a Python 3.8+ environment, run pip install pytojsonschema.

Scan a package

After installing the package, you can open a python terminal from the root of the repo and run:

import os
import pprint

from pytojsonschema.functions import process_package

pprint.pprint(process_package(os.path.join("test", "example")))

The example package will be scanned and JSON schemas will be generated for all the top level functions it can find.

Scan a file

You can also target specific files, which won't include the package namespacing in the result value. Following on the same terminal:

from pytojsonschema.functions import process_file

pprint.pprint(process_file(os.path.join("test", "example", "service.py")))

Include and exclude patterns

Include and exclude unix-like patterns can be used to filter function and module names we want to allow/disallow for scanning. See the difference when you now run this instead:

pprint.pprint(process_package(os.path.join("test", "example"), exclude_patterns=["_*"]))

Similarly, but applied to specific files:

pprint.pprint(process_file(os.path.join("test", "example", "service.py"), exclude_patterns=["_*"]))

Things to take into account:

  • Exclude pattern matching overwrite include matches.
  • __init__.py files are not affected by pattern rules and are always scanned. However, you can still filter its internal functions.

Type annotation rules

Fitting Python's typing model to JSON means not everything is allowed in your function signatures. This is a natural restriction that comes with JSON data serialization. Hopefully, most of the useful stuff you need is allowed.

Allowed types

Base types

Basic types bool, int, float, str, None and typing.Any are allowed. Also, you can build more complex, nested structures with the usage of typing.Union, typing.Optional, typing.Dict (Only str keys are allowed) and typing.List. All these types have a direct, non-ambiguous representation in both JSON and JSON schema.

Custom types

Your functions can also use custom types like the ones defined using an assignment of typing.Union, typing.List, typing.Dict and typing.Optional, as in:

ServicePort = typing.Union[int, float]
ServiceConfig = typing.Dict[str, typing.Any]

You can use one of the new Python 3.8 features, typing.TypedDict, to build stronger validation on dict-like objects (Only class-based syntax). As you can see, you can chain types with no restrictions:

class Service(typing.TypedDict):
    address: str
    port: ServicePort
    config: ServiceConfig
    tags: typing.List[str]
    debug: bool = False

Also, if you need to restrict the choices for a string type, you can use Python enums:

Note 1: Whilst Python itself will not auto-populate default values, you can use them to make the property not required

import enum


class HTTPMethod(enum.Enum):
    GET = "GET"
    POST = "POST"
    PATCH = "PATCH"
    DELETE = "DELETE"


def my_func(http_method: HTTPMethod):
    pass  # My code

Note 1: This only works for enums whose values are strings, as that is the only case JSON schema supports

Note 2: The resulting validation uses the enum values as the valid choices, as that is what JSON schema can understand

Importing types from other files

You can import these custom types within your package and they will be picked up. However, due to the static nature of the scan, custom types coming from external packages can't be followed and hence not supported. In other words, you can only share these types within your package, using relative imports.

Other static analysis tools like mypy use a repository with stub files to solve this issue, see https://mypy.readthedocs.io/en/stable/stubs.html. This is out of the scope for a tiny project like this, at least for now.

Rules

  1. The functions you want to scan need to be type annotated. Kind of obvious requirement, right?

  2. Only the types defined in the previous section can be used. They are the types that can be safely serialised as JSON.

  3. Function arguments are meant to be passed in key-value format, like a json object. This puts a couple of restrictions regarding *args, **kwargs, positional-only and keyword-only arguments:

    The following is allowed:

    • **kwargs: def func(**kwargs): pass
    • keyword-only arguments: def func(*, a): pass

    The following is not allowed:

    • *args: def func(*args): pass
    • positional-only arguments: def func(a, /): pass

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

pytojsonschema-1.10.1.tar.gz (15.1 kB view details)

Uploaded Source

Built Distribution

pytojsonschema-1.10.1-py3-none-any.whl (11.6 kB view details)

Uploaded Python 3

File details

Details for the file pytojsonschema-1.10.1.tar.gz.

File metadata

  • Download URL: pytojsonschema-1.10.1.tar.gz
  • Upload date:
  • Size: 15.1 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.1.1 pkginfo/1.5.0.1 requests/2.23.0 setuptools/41.2.0 requests-toolbelt/0.9.1 tqdm/4.45.0 CPython/3.8.2

File hashes

Hashes for pytojsonschema-1.10.1.tar.gz
Algorithm Hash digest
SHA256 6c56791f91e6d2803dfac627d1a1fa378a08519a86b406a515bf869736d5b9ab
MD5 a734d4d840bc904e0595177196b70332
BLAKE2b-256 216fced544bad2e83e4278f884f74f4783aa27dfeafc49d1afbc2ff903a241f4

See more details on using hashes here.

File details

Details for the file pytojsonschema-1.10.1-py3-none-any.whl.

File metadata

  • Download URL: pytojsonschema-1.10.1-py3-none-any.whl
  • Upload date:
  • Size: 11.6 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.1.1 pkginfo/1.5.0.1 requests/2.23.0 setuptools/41.2.0 requests-toolbelt/0.9.1 tqdm/4.45.0 CPython/3.8.2

File hashes

Hashes for pytojsonschema-1.10.1-py3-none-any.whl
Algorithm Hash digest
SHA256 87be789ae238df2c206a5d0fa76e8f8105ccf6b9a8de5dcf4f9da484915056ce
MD5 75477a56e37717c09c99fd15e1848362
BLAKE2b-256 d7b3fb58dd9cff2892a1dd3ba72af033410ad9d9c470d48f52cb85f40a777b11

See more details on using hashes here.

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page