Type-Driven Development for Python
Project description
pytyped
pytyped
is a Python package whose goal is to enable as much type-driven development as possible in Python.
We believe in using types to automate mundane and repetitive tasks.
Currently, given a type, JSON decoders/encoders and metric extractors can be automatically extracted for that type.
Installation
You can install pytyped
from PyPI:
pip install pytyped
pytyped
is checked on Python 3.6+
.
Using pytyped
to extract JSON decoders/encoders
First, define your type. For example, in the following we want to define an account that can either be a personal account or a business account with personal account having one owner and possibly a co-owner while a business account has the company name as the owner and a list of persons that can represent the company.
from dataclasses import dataclass
from datetime import datetime
from typing import List
from typing import Optional
@dataclass
class Person:
first_name: str
last_name: str
@dataclass
class Account:
created_at: datetime
@dataclass
class PersonalAccount(Account):
owner: Person
co_owner: Optional[Person]
@dataclass
class BusinessAccount(Account):
owner: str
representatives: List[Person]
Second, use an instance of AutoJsonDecoder
and AutoJsonEncoder
to extract JSON decoders and encoders as below:
from pytyped.json.decoder import AutoJsonDecoder
from pytyped.json.decoder import JsonDecoder
from pytyped.json.encoder import AutoJsonEncoder
from pytyped.json.encoder import JsonEncoder
_auto_json_decoder = AutoJsonDecoder()
_auto_json_encoder = AutoJsonEncoder()
account_decoder: JsonDecoder[Account] = _auto_json_decoder.extract(Account)
account_encoder: JsonEncoder[Account] = _auto_json_encoder.extract(Account)
Third, define some instances of the Account
class:
personal_account = PersonalAccount(
created_at = datetime.now(),
owner = Person(first_name = "John", last_name = "Doe"),
co_owner = None
)
business_account = BusinessAccount(
created_at = datetime.now(),
owner = "Doe Ltd.",
representatives = [Person(first_name = "John", last_name = "Doe"), Person(first_name = "Jane", last_name = "Doe")]
)
Finally, use account_encoder
and account_decoder
to convert data in your instances to/from JSON as below:
>>> json = account_encoder.write(personal_account)
>>> json
{'created_at': '2020-08-24T20:00:18.205347', 'owner': {'first_name': 'John', 'last_name': 'Doe'}, 'co_owner': None, 'Account': 'PersonalAccount'}
>>> account_decoder.read(json)
PersonalAccount(created_at=datetime.datetime(2020, 8, 24, 20, 0, 18, 205347), owner=Person(first_name='John', last_name='Doe'), co_owner=None)
>>> json = account_encoder.write(business_account)
>>> json
{'created_at': '2020-08-24T20:00:40.057088', 'owner': 'Doe Ltd.', 'representatives': [{'first_name': 'John', 'last_name': 'Doe'}, {'first_name': 'Jane', 'last_name': 'Doe'}], 'Account': 'BusinessAccount'}
>>> account_decoder.read(json)
BusinessAccount(created_at=datetime.datetime(2020, 8, 24, 20, 0, 40, 57088), owner='Doe Ltd.', representatives=[Person(first_name='John', last_name='Doe'), Person(first_name='Jane', last_name='Doe')])
To illustrate the types of validation that JSON decoders enable for you, consider the following example invalid JSONs:
>>> account_decoder.read({'created_at': '2020-08-24T20:00:40.057088', 'owner': 'Doe Ltd.', 'representatives': [{'first_name': 'John', 'last_name': 'Doe'}, {'first_name': 'Jane', 'last_name': 'Doe'}], 'Account': 'NewAccount'})
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "~/Repos/pytyped/pytyped/json/decoder.py", line 91, in read
raise JsDecodeException(t_or_error)
pytyped.json.decoder.JsDecodeException: Found 1 errors while validating JSON: [
Error when decoding JSON: /Account: Unknown tag value NewAccount (possible values are: PersonalAccount, BusinessAccount).]
>>> account_decoder.read({'created_at': '2020-08-24T20:00:40.057088', 'owner': 'Doe Ltd.', 'representatives': [{'first_name': 'John', 'last_name': 'Doe'}, {'first_name': 'Jane', 'last': 'Doe'}], 'Account': 'BusinessAccount'})
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "/home/shahab/Repos/pytyped/pytyped/json/decoder.py", line 91, in read
raise JsDecodeException(t_or_error)
pytyped.json.decoder.JsDecodeException: Found 1 errors while validating JSON: [
Error when decoding JSON: /representatives[1]/last_name: Non-optional field was not found]
Using pytyped to extract metrics
Similar to extracting JSON decoders/encoders except that pytyped.metrics.exporter.AutoMetricExporter
is used.
Further explanation is WIP.
Defining New Type Automations
You can follow the examples of JSON decoders, JSON encoders, and metric exporters to define new type-based automations.
You just need to extend pytyped.macros.extractor.Extractor
and implement the abstrtact methods there.
Further explanation is WIP.
Issues
Please report any issues to the GitHub repository for this package.
Contributors
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
File details
Details for the file pytyped-0.0.1.tar.gz
.
File metadata
- Download URL: pytyped-0.0.1.tar.gz
- Upload date:
- Size: 18.0 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/3.2.0 pkginfo/1.5.0.1 requests/2.24.0 setuptools/46.1.3 requests-toolbelt/0.9.1 tqdm/4.48.2 CPython/3.6.9
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 37a20bb9960dde1a5dfe88c60a0321dd30d1dece41d86ba466eeae97b37ffe95 |
|
MD5 | fb481d525aeca2289d8b5c62c9caa023 |
|
BLAKE2b-256 | 0c2a168c0ba484db1ef08771d36bc4b121a52cf492c008e78602b53efed73cea |
File details
Details for the file pytyped-0.0.1-py3-none-any.whl
.
File metadata
- Download URL: pytyped-0.0.1-py3-none-any.whl
- Upload date:
- Size: 33.7 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/3.2.0 pkginfo/1.5.0.1 requests/2.24.0 setuptools/46.1.3 requests-toolbelt/0.9.1 tqdm/4.48.2 CPython/3.6.9
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 2137e6cc728a57dd0c46998f8dc9c47e058832877b9ead6de77f0715247794d4 |
|
MD5 | ff8bc095275f272ce2d1258a34f5122f |
|
BLAKE2b-256 | adc1537dead921fb379e46410f665d589970270de119ff5fbb2599c817fa3c72 |