Compare commits
	
		
			85 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | a5effb219a | ||
|  | 2d57f0d122 | ||
|  | a68505b80e | ||
|  | 2f9497e064 | ||
|  | 33964b883e | ||
|  | ec7574086d | ||
|  | 8a42027bc9 | ||
|  | 71737cf696 | ||
|  | 659ddd9c44 | ||
|  | 5b6997870a | ||
|  | cdf7645722 | ||
|  | ca20069ca3 | ||
|  | 59a4a7da43 | ||
|  | 15af4367e5 | ||
|  | ec5683e572 | ||
|  | 20150fdcf3 | ||
|  | d11b7d04c5 | ||
|  | e2d35f4696 | ||
|  | c3f08b9ef2 | ||
|  | 24d44898f4 | ||
|  | 074448c996 | ||
|  | 0fe557bd3c | ||
|  | 1a87ea43a1 | ||
|  | 983e0895a2 | ||
|  | 4a2baf3f0a | ||
|  | 8f0caf1db2 | ||
|  | c50d9e2fdc | ||
|  | 35548cb43e | ||
|  | b711d1e11f | ||
|  | 917de09bb6 | ||
|  | 1f7f39049e | ||
|  | 3d001a2a1a | ||
|  | de61ddab21 | ||
|  | 5e2d9febea | ||
|  | f6af077ffe | ||
|  | 92088ebda8 | ||
|  | c3e3837f71 | ||
|  | 6bd9c7835c | ||
|  | 6ec902c1b5 | ||
|  | 960dba2ae8 | ||
|  | 4b4bdefb6f | ||
|  | dfa0a56b39 | ||
|  | dd4873dfba | ||
|  | 91f586f7d7 | ||
|  | 33fb83faad | ||
|  | 77c04414f5 | ||
|  | 6969ff7ff6 | ||
|  | 13e08fdaa8 | ||
|  | 6775632f77 | ||
|  | b12f1e4e61 | ||
|  | 7e9ba0866c | ||
|  | 3546f55146 | ||
|  | 499489f1d3 | ||
|  | ce9f492f50 | ||
|  | 93a6334015 | ||
|  | 36a14026d8 | ||
|  | 04a2fcd3eb | ||
|  | 5759e323bd | ||
|  | c762c9c549 | ||
|  | 582a12577c | ||
|  | 3616190451 | ||
|  | 9b990ee1bd | ||
|  | 72a77b0d65 | ||
|  | b2b36c8575 | ||
|  | 203105f048 | ||
|  | fe11f74227 | ||
|  | dc7a3e9bdf | ||
|  | f2e8afc609 | ||
|  | dbd438e682 | ||
|  | dce1c89fbe | ||
|  | c78851b1b8 | ||
|  | 4554d91f89 | ||
|  | c0170f4d80 | ||
|  | 559b8833d8 | ||
|  | 7ccef16579 | ||
|  | d8785b4622 | ||
|  | 45e7a30300 | ||
|  | d7559c22f8 | ||
|  | f9c351a98d | ||
|  | feea790116 | ||
|  | 33f74f6a45 | ||
|  | 3d5c12c532 | ||
|  | 706bd5a475 | ||
|  | 52beeb0d73 | ||
|  | 7e2dc595db | 
							
								
								
									
										37
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										37
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							| @@ -3,7 +3,34 @@ name: CI | |||||||
| on: [push, pull_request] | on: [push, pull_request] | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   build: |  | ||||||
|  |   run-tests: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |  | ||||||
|  |     strategy: | ||||||
|  |       matrix: | ||||||
|  |         python-version: [ '3.6', '3.7' ] | ||||||
|  |  | ||||||
|  |     name: Python ${{ matrix.python-version }} test | ||||||
|  |  | ||||||
|  |     steps: | ||||||
|  |       - uses: actions/checkout@v1 | ||||||
|  |       - uses: actions/setup-python@v1 | ||||||
|  |         with: | ||||||
|  |           python-version: ${{ matrix.python-version }} | ||||||
|  |       - uses: dschep/install-pipenv-action@v1 | ||||||
|  |       - name: Install dependencies | ||||||
|  |         run: | | ||||||
|  |           sudo apt install protobuf-compiler libprotobuf-dev | ||||||
|  |           pipenv install --dev --python ${pythonLocation}/python | ||||||
|  |       - name: Run tests | ||||||
|  |         run: | | ||||||
|  |           cp .env.default .env | ||||||
|  |           pipenv run pip install -e . | ||||||
|  |           pipenv run generate | ||||||
|  |           pipenv run test | ||||||
|  |  | ||||||
|  |   build-release: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
| @@ -15,13 +42,7 @@ jobs: | |||||||
|       - name: Install dependencies |       - name: Install dependencies | ||||||
|         run: | |         run: | | ||||||
|           sudo apt install protobuf-compiler libprotobuf-dev |           sudo apt install protobuf-compiler libprotobuf-dev | ||||||
|           pipenv install --dev |           pipenv install --dev --python ${pythonLocation}/python | ||||||
|       - name: Run tests |  | ||||||
|         run: | |  | ||||||
|           cp .env.default .env |  | ||||||
|           pipenv run pip install -e . |  | ||||||
|           pipenv run generate |  | ||||||
|           pipenv run test |  | ||||||
|       - name: Build package |       - name: Build package | ||||||
|         if: github.event_name == 'push' && startsWith(github.event.ref, 'refs/tags') |         if: github.event_name == 'push' && startsWith(github.event.ref, 'refs/tags') | ||||||
|         run: pipenv run python setup.py sdist |         run: pipenv run python setup.py sdist | ||||||
|   | |||||||
							
								
								
									
										9
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										9
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -2,12 +2,11 @@ | |||||||
| .vscode/settings.json | .vscode/settings.json | ||||||
| .mypy_cache | .mypy_cache | ||||||
| .pytest_cache | .pytest_cache | ||||||
| betterproto/tests/*.bin | .python-version | ||||||
| betterproto/tests/*_pb2.py | build/ | ||||||
| betterproto/tests/*.py | betterproto/tests/output_* | ||||||
| !betterproto/tests/generate.py |  | ||||||
| !betterproto/tests/test_*.py |  | ||||||
| **/__pycache__ | **/__pycache__ | ||||||
| dist | dist | ||||||
| **/*.egg-info | **/*.egg-info | ||||||
| output | output | ||||||
|  | .idea | ||||||
							
								
								
									
										64
									
								
								CHANGELOG.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								CHANGELOG.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,64 @@ | |||||||
|  | # Changelog | ||||||
|  |  | ||||||
|  | All notable changes to this project will be documented in this file. | ||||||
|  |  | ||||||
|  | The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), | ||||||
|  | and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). | ||||||
|  |  | ||||||
|  | ## [1.2.4] - 2020-04-26 | ||||||
|  |  | ||||||
|  | - Enforce utf-8 for reading the readme in setup.py [#67](https://github.com/danielgtaylor/python-betterproto/pull/67) | ||||||
|  | - Only import types from grpclib when type checking [#52](https://github.com/danielgtaylor/python-betterproto/pull/52) | ||||||
|  | - Improve performance of serialize/deserialize by caching type information of fields in class [#46](https://github.com/danielgtaylor/python-betterproto/pull/46) | ||||||
|  | - Support using Google's wrapper types as RPC output values [#40](https://github.com/danielgtaylor/python-betterproto/pull/40) | ||||||
|  | - Fixes issue where protoc did not recognize plugin.py as win32 application [#38](https://github.com/danielgtaylor/python-betterproto/pull/38) | ||||||
|  | - Fix services using non-pythonified field names [#34](https://github.com/danielgtaylor/python-betterproto/pull/34) | ||||||
|  | - Add ability to provide metadata, timeout & deadline args to requests [#32](https://github.com/danielgtaylor/python-betterproto/pull/32) | ||||||
|  |  | ||||||
|  | ## [1.2.3] - 2020-04-15 | ||||||
|  |  | ||||||
|  | - Exclude empty lists from `to_dict` by default [#16](https://github.com/danielgtaylor/python-betterproto/pull/16) | ||||||
|  | - Add `include_default_values` parameter for `to_dict` [#12](https://github.com/danielgtaylor/python-betterproto/pull/12) | ||||||
|  | - Fix class names being prepended with duplicates when using protocol buffers that are nested more than once [#21](https://github.com/danielgtaylor/python-betterproto/pull/21) | ||||||
|  | - Add support for python 3.6 [#30](https://github.com/danielgtaylor/python-betterproto/pull/30) | ||||||
|  |  | ||||||
|  | ## [1.2.2] - 2020-01-09 | ||||||
|  |  | ||||||
|  | - Mention lack of Proto 2 support in README. | ||||||
|  | - Fix serialization of constructor parameters [#10](https://github.com/danielgtaylor/python-betterproto/pull/10) | ||||||
|  | - Fix `casing` parameter propagation [#7](https://github.com/danielgtaylor/python-betterproto/pull/7) | ||||||
|  |  | ||||||
|  | ## [1.2.1] - 2019-10-29 | ||||||
|  |  | ||||||
|  | - Fix comment indentation bug in rendered gRPC methods. | ||||||
|  |  | ||||||
|  | ## [1.2.0] - 2019-10-28 | ||||||
|  |  | ||||||
|  | - Generated code output auto-formatting via [Black](https://github.com/psf/black) | ||||||
|  | - Simplified gRPC helper functions | ||||||
|  |  | ||||||
|  | ## [1.1.0] - 2019-10-27 | ||||||
|  |  | ||||||
|  | - Better JSON casing support | ||||||
|  | - Handle field names which clash with Python reserved words | ||||||
|  | - Better handling of default values from type introspection | ||||||
|  | - Support for Google Duration & Timestamp types | ||||||
|  | - Support for Google wrapper types | ||||||
|  | - Documentation updates | ||||||
|  |  | ||||||
|  | ## [1.0.1] - 2019-10-22 | ||||||
|  |  | ||||||
|  | - README to the PyPI details page | ||||||
|  |  | ||||||
|  | ## [1.0.0] - 2019-10-22 | ||||||
|  |  | ||||||
|  | - Initial release | ||||||
|  |  | ||||||
|  | [1.2.4]: https://github.com/danielgtaylor/python-betterproto/compare/v1.2.3...v1.2.4 | ||||||
|  | [1.2.3]: https://github.com/danielgtaylor/python-betterproto/compare/v1.2.2...v1.2.3 | ||||||
|  | [1.2.2]: https://github.com/danielgtaylor/python-betterproto/compare/v1.2.1...v1.2.2 | ||||||
|  | [1.2.1]: https://github.com/danielgtaylor/python-betterproto/compare/v1.2.0...v1.2.1 | ||||||
|  | [1.2.0]: https://github.com/danielgtaylor/python-betterproto/compare/v1.1.0...v1.2.0 | ||||||
|  | [1.1.0]: https://github.com/danielgtaylor/python-betterproto/compare/v1.0.1...v1.1.0 | ||||||
|  | [1.0.1]: https://github.com/danielgtaylor/python-betterproto/compare/v1.0.0...v1.0.1 | ||||||
|  | [1.0.0]: https://github.com/danielgtaylor/python-betterproto/releases/tag/v1.0.0 | ||||||
							
								
								
									
										10
									
								
								Pipfile
									
									
									
									
									
								
							
							
						
						
									
										10
									
								
								Pipfile
									
									
									
									
									
								
							| @@ -8,18 +8,26 @@ flake8 = "*" | |||||||
| mypy = "*" | mypy = "*" | ||||||
| isort = "*" | isort = "*" | ||||||
| pytest = "*" | pytest = "*" | ||||||
|  | pytest-asyncio = "*" | ||||||
| rope = "*" | rope = "*" | ||||||
|  | v = {editable = true,version = "*"} | ||||||
|  |  | ||||||
| [packages] | [packages] | ||||||
| protobuf = "*" | protobuf = "*" | ||||||
| jinja2 = "*" | jinja2 = "*" | ||||||
| grpclib = "*" | grpclib = "*" | ||||||
| stringcase = "*" | stringcase = "*" | ||||||
|  | black = "*" | ||||||
|  | backports-datetime-fromisoformat = "*" | ||||||
|  | dataclasses = "*" | ||||||
|  |  | ||||||
| [requires] | [requires] | ||||||
| python_version = "3.7" | python_version = "3.6" | ||||||
|  |  | ||||||
| [scripts] | [scripts] | ||||||
| plugin = "protoc --plugin=protoc-gen-custom=betterproto/plugin.py --custom_out=output" | plugin = "protoc --plugin=protoc-gen-custom=betterproto/plugin.py --custom_out=output" | ||||||
| generate = "python betterproto/tests/generate.py" | generate = "python betterproto/tests/generate.py" | ||||||
| test = "pytest ./betterproto/tests" | test = "pytest ./betterproto/tests" | ||||||
|  |  | ||||||
|  | [pipenv] | ||||||
|  | allow_prereleases = true | ||||||
|   | |||||||
							
								
								
									
										452
									
								
								Pipfile.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										452
									
								
								Pipfile.lock
									
									
									
										generated
									
									
									
								
							| @@ -1,11 +1,11 @@ | |||||||
| { | { | ||||||
|     "_meta": { |     "_meta": { | ||||||
|         "hash": { |         "hash": { | ||||||
|             "sha256": "28c38cd6c4eafb0b9ac9a64cf623145868fdee163111d3b941b34d23011db6ca" |             "sha256": "b8fc738d4e14598e36269ce0d849489f95562ba047e5663caca9ac02550893ef" | ||||||
|         }, |         }, | ||||||
|         "pipfile-spec": 6, |         "pipfile-spec": 6, | ||||||
|         "requires": { |         "requires": { | ||||||
|             "python_version": "3.7" |             "python_version": "3.6" | ||||||
|         }, |         }, | ||||||
|         "sources": [ |         "sources": [ | ||||||
|             { |             { | ||||||
| @@ -16,19 +16,63 @@ | |||||||
|         ] |         ] | ||||||
|     }, |     }, | ||||||
|     "default": { |     "default": { | ||||||
|         "grpclib": { |         "appdirs": { | ||||||
|             "hashes": [ |             "hashes": [ | ||||||
|                 "sha256:d19e2ea87cb073e5b0825dfee15336fd2b1c09278d271816e04c90faddc107ea" |                 "sha256:9e5896d1372858f8dd3344faf4e5014d21849c756c8d5701f78f8a103b372d92", | ||||||
|  |                 "sha256:d8b24664561d0d34ddfaec54636d502d7cea6e29c3eaf68f3df6180863e2166e" | ||||||
|  |             ], | ||||||
|  |             "version": "==1.4.3" | ||||||
|  |         }, | ||||||
|  |         "attrs": { | ||||||
|  |             "hashes": [ | ||||||
|  |                 "sha256:08a96c641c3a74e44eb59afb61a24f2cb9f4d7188748e76ba4bb5edfa3cb7d1c", | ||||||
|  |                 "sha256:f7b7ce16570fe9965acd6d30101a28f62fb4a7f9e926b3bbc9b61f8b04247e72" | ||||||
|  |             ], | ||||||
|  |             "version": "==19.3.0" | ||||||
|  |         }, | ||||||
|  |         "backports-datetime-fromisoformat": { | ||||||
|  |             "hashes": [ | ||||||
|  |                 "sha256:9577a2a9486cd7383a5f58b23bb8e81cf0821dbbc0eb7c87d3fa198c1df40f5c" | ||||||
|             ], |             ], | ||||||
|             "index": "pypi", |             "index": "pypi", | ||||||
|             "version": "==0.3.0" |             "version": "==1.0.0" | ||||||
|  |         }, | ||||||
|  |         "black": { | ||||||
|  |             "hashes": [ | ||||||
|  |                 "sha256:1b30e59be925fafc1ee4565e5e08abef6b03fe455102883820fe5ee2e4734e0b", | ||||||
|  |                 "sha256:c2edb73a08e9e0e6f65a0e6af18b059b8b1cdd5bef997d7a0b181df93dc81539" | ||||||
|  |             ], | ||||||
|  |             "index": "pypi", | ||||||
|  |             "version": "==19.10b0" | ||||||
|  |         }, | ||||||
|  |         "click": { | ||||||
|  |             "hashes": [ | ||||||
|  |                 "sha256:8a18b4ea89d8820c5d0c7da8a64b2c324b4dabb695804dbfea19b9be9d88c0cc", | ||||||
|  |                 "sha256:e345d143d80bf5ee7534056164e5e112ea5e22716bbb1ce727941f4c8b471b9a" | ||||||
|  |             ], | ||||||
|  |             "version": "==7.1.1" | ||||||
|  |         }, | ||||||
|  |         "dataclasses": { | ||||||
|  |             "hashes": [ | ||||||
|  |                 "sha256:454a69d788c7fda44efd71e259be79577822f5e3f53f029a22d08004e951dc9f", | ||||||
|  |                 "sha256:6988bd2b895eef432d562370bb707d540f32f7360ab13da45340101bc2307d84" | ||||||
|  |             ], | ||||||
|  |             "index": "pypi", | ||||||
|  |             "version": "==0.6" | ||||||
|  |         }, | ||||||
|  |         "grpclib": { | ||||||
|  |             "hashes": [ | ||||||
|  |                 "sha256:b27d56c987b89023d5640fe9668943e49b46703fc85d8182a58c9f3b19120cdc" | ||||||
|  |             ], | ||||||
|  |             "index": "pypi", | ||||||
|  |             "version": "==0.3.2rc1" | ||||||
|         }, |         }, | ||||||
|         "h2": { |         "h2": { | ||||||
|             "hashes": [ |             "hashes": [ | ||||||
|                 "sha256:ac377fcf586314ef3177bfd90c12c7826ab0840edeb03f0f24f511858326049e", |                 "sha256:61e0f6601fa709f35cdb730863b4e5ec7ad449792add80d1410d4174ed139af5", | ||||||
|                 "sha256:b8a32bd282594424c0ac55845377eea13fa54fe4a8db012f3a198ed923dc3ab4" |                 "sha256:875f41ebd6f2c44781259005b157faed1a5031df3ae5aa7bcb4628a6c0782f14" | ||||||
|             ], |             ], | ||||||
|             "version": "==3.1.1" |             "version": "==3.2.0" | ||||||
|         }, |         }, | ||||||
|         "hpack": { |         "hpack": { | ||||||
|             "hashes": [ |             "hashes": [ | ||||||
| @@ -46,107 +90,124 @@ | |||||||
|         }, |         }, | ||||||
|         "jinja2": { |         "jinja2": { | ||||||
|             "hashes": [ |             "hashes": [ | ||||||
|                 "sha256:74320bb91f31270f9551d46522e33af46a80c3d619f4a4bf42b3164d30b5911f", |                 "sha256:c10142f819c2d22bdcd17548c46fa9b77cf4fda45097854c689666bf425e7484", | ||||||
|                 "sha256:9fe95f19286cfefaa917656583d020be14e7859c6b0252588391e47db34527de" |                 "sha256:c922560ac46888d47384de1dbdc3daaa2ea993af4b26a436dec31fa2c19ec668" | ||||||
|             ], |             ], | ||||||
|             "index": "pypi", |             "index": "pypi", | ||||||
|             "version": "==2.10.3" |             "version": "==3.0.0a1" | ||||||
|         }, |         }, | ||||||
|         "markupsafe": { |         "markupsafe": { | ||||||
|             "hashes": [ |             "hashes": [ | ||||||
|                 "sha256:00bc623926325b26bb9605ae9eae8a215691f33cae5df11ca5424f06f2d1f473", |                 "sha256:06358015a4dee8ee23ae426bf885616ab3963622defd829eb45b44e3dee3515f", | ||||||
|                 "sha256:09027a7803a62ca78792ad89403b1b7a73a01c8cb65909cd876f7fcebd79b161", |                 "sha256:0b0c4fc852c5f02c6277ef3b33d23fcbe89b1b227460423e3335374da046b6db", | ||||||
|                 "sha256:09c4b7f37d6c648cb13f9230d847adf22f8171b1ccc4d5682398e77f40309235", |                 "sha256:267677fc42afed5094fc5ea1c4236bbe4b6a00fe4b08e93451e65ae9048139c7", | ||||||
|                 "sha256:1027c282dad077d0bae18be6794e6b6b8c91d58ed8a8d89a89d59693b9131db5", |                 "sha256:303cb70893e2c345588fb5d5b86e0ca369f9bb56942f03064c5e3e75fa7a238a", | ||||||
|                 "sha256:24982cc2533820871eba85ba648cd53d8623687ff11cbb805be4ff7b4c971aff", |                 "sha256:3c9b624a0d9ed5a5093ac4edc4e823e6b125441e60ef35d36e6f4a6fdacd5054", | ||||||
|                 "sha256:29872e92839765e546828bb7754a68c418d927cd064fd4708fab9fe9c8bb116b", |                 "sha256:42033e14cae1f6c86fc0c3e90d04d08ce73ac8e46ba420a0d22d545c2abd4977", | ||||||
|                 "sha256:43a55c2930bbc139570ac2452adf3d70cdbb3cfe5912c71cdce1c2c6bbd9c5d1", |                 "sha256:4e4a99b6af7bdc0856b50020c095848ec050356a001e1f751510aef6ab14d0e0", | ||||||
|                 "sha256:46c99d2de99945ec5cb54f23c8cd5689f6d7177305ebff350a58ce5f8de1669e", |                 "sha256:4eb07faad54bb07427d848f31030a65a49ebb0cec0b30674f91cf1ddd456bfe4", | ||||||
|                 "sha256:500d4957e52ddc3351cabf489e79c91c17f6e0899158447047588650b5e69183", |                 "sha256:63a7161cd8c2bc563feeda45df62f42c860dd0675e2b8da2667f25bb3c95eaba", | ||||||
|                 "sha256:535f6fc4d397c1563d08b88e485c3496cf5784e927af890fb3c3aac7f933ec66", |                 "sha256:68e0fd039b68d2945b4beb947d4023ca7f8e95b708031c345762efba214ea761", | ||||||
|                 "sha256:62fe6c95e3ec8a7fad637b7f3d372c15ec1caa01ab47926cfdf7a75b40e0eac1", |                 "sha256:8092a63397025c2f655acd42784b2a1528339b90b987beb9253f22e8cdbb36c3", | ||||||
|                 "sha256:6dd73240d2af64df90aa7c4e7481e23825ea70af4b4922f8ede5b9e35f78a3b1", |                 "sha256:841218860683c0f2223e24756843d84cc49cccdae6765e04962607754a52d3e0", | ||||||
|                 "sha256:717ba8fe3ae9cc0006d7c451f0bb265ee07739daf76355d06366154ee68d221e", |                 "sha256:94076b2314bd2f6cfae508ad65b4d493e3a58a50112b7a2cbb6287bdbc404ae8", | ||||||
|                 "sha256:79855e1c5b8da654cf486b830bd42c06e8780cea587384cf6545b7d9ac013a0b", |                 "sha256:9d22aff1c5322e402adfb3ce40839a5056c353e711c033798cf4f02eb9f5124d", | ||||||
|                 "sha256:7c1699dfe0cf8ff607dbdcc1e9b9af1755371f92a68f706051cc8c37d447c905", |                 "sha256:b0e4584f62b3e5f5c1a7bcefd2b52f236505e6ef032cc508caa4f4c8dc8d3af1", | ||||||
|                 "sha256:88e5fcfb52ee7b911e8bb6d6aa2fd21fbecc674eadd44118a9cc3863f938e735", |                 "sha256:b1163ffc1384d242964426a8164da12dbcdbc0de18ea36e2c34b898ed38c3b45", | ||||||
|                 "sha256:8defac2f2ccd6805ebf65f5eeb132adcf2ab57aa11fdf4c0dd5169a004710e7d", |                 "sha256:beac28ed60c8e838301226a7a85841d0af2068eba2dcb1a58c2d32d6c05e440e", | ||||||
|                 "sha256:98c7086708b163d425c67c7a91bad6e466bb99d797aa64f965e9d25c12111a5e", |                 "sha256:c29f096ce79c03054a1101d6e5fe6bf04b0bb489165d5e0e9653fb4fe8048ee1", | ||||||
|                 "sha256:9add70b36c5666a2ed02b43b335fe19002ee5235efd4b8a89bfcf9005bebac0d", |                 "sha256:c58779966d53e5f14ba393d64e2402a7926601d1ac8adeb4e83893def79d0428", | ||||||
|                 "sha256:9bf40443012702a1d2070043cb6291650a0841ece432556f784f004937f0f32c", |                 "sha256:cfe14b37908eaf7d5506302987228bff69e1b8e7071ccd4e70fd0283b1b47f0b", | ||||||
|                 "sha256:ade5e387d2ad0d7ebf59146cc00c8044acbd863725f887353a10df825fc8ae21", |                 "sha256:e834249c45aa9837d0753351cdca61a4b8b383cc9ad0ff2325c97ff7b69e72a6", | ||||||
|                 "sha256:b00c1de48212e4cc9603895652c5c410df699856a2853135b3967591e4beebc2", |                 "sha256:eed1b234c4499811ee85bcefa22ef5e466e75d132502226ed29740d593316c1f" | ||||||
|                 "sha256:b1282f8c00509d99fef04d8ba936b156d419be841854fe901d8ae224c59f0be5", |  | ||||||
|                 "sha256:b2051432115498d3562c084a49bba65d97cf251f5a331c64a12ee7e04dacc51b", |  | ||||||
|                 "sha256:ba59edeaa2fc6114428f1637ffff42da1e311e29382d81b339c1817d37ec93c6", |  | ||||||
|                 "sha256:c8716a48d94b06bb3b2524c2b77e055fb313aeb4ea620c8dd03a105574ba704f", |  | ||||||
|                 "sha256:cd5df75523866410809ca100dc9681e301e3c27567cf498077e8551b6d20e42f", |  | ||||||
|                 "sha256:e249096428b3ae81b08327a63a485ad0878de3fb939049038579ac0ef61e17e7" |  | ||||||
|             ], |             ], | ||||||
|             "version": "==1.1.1" |             "version": "==2.0.0a1" | ||||||
|         }, |         }, | ||||||
|         "multidict": { |         "multidict": { | ||||||
|             "hashes": [ |             "hashes": [ | ||||||
|                 "sha256:024b8129695a952ebd93373e45b5d341dbb87c17ce49637b34000093f243dd4f", |                 "sha256:317f96bc0950d249e96d8d29ab556d01dd38888fbe68324f46fd834b430169f1", | ||||||
|                 "sha256:041e9442b11409be5e4fc8b6a97e4bcead758ab1e11768d1e69160bdde18acc3", |                 "sha256:42f56542166040b4474c0c608ed051732033cd821126493cf25b6c276df7dd35", | ||||||
|                 "sha256:045b4dd0e5f6121e6f314d81759abd2c257db4634260abcfe0d3f7083c4908ef", |                 "sha256:4b7df040fb5fe826d689204f9b544af469593fb3ff3a069a6ad3409f742f5928", | ||||||
|                 "sha256:047c0a04e382ef8bd74b0de01407e8d8632d7d1b4db6f2561106af812a68741b", |                 "sha256:544fae9261232a97102e27a926019100a9db75bec7b37feedd74b3aa82f29969", | ||||||
|                 "sha256:068167c2d7bbeebd359665ac4fff756be5ffac9cda02375b5c5a7c4777038e73", |                 "sha256:620b37c3fea181dab09267cd5a84b0f23fa043beb8bc50d8474dd9694de1fa6e", | ||||||
|                 "sha256:148ff60e0fffa2f5fad2eb25aae7bef23d8f3b8bdaf947a65cdbe84a978092bc", |                 "sha256:6e6fef114741c4d7ca46da8449038ec8b1e880bbe68674c01ceeb1ac8a648e78", | ||||||
|                 "sha256:1d1c77013a259971a72ddaa83b9f42c80a93ff12df6a4723be99d858fa30bee3", |                 "sha256:7774e9f6c9af3f12f296131453f7b81dabb7ebdb948483362f5afcaac8a826f1", | ||||||
|                 "sha256:1d48bc124a6b7a55006d97917f695effa9725d05abe8ee78fd60d6588b8344cd", |                 "sha256:85cb26c38c96f76b7ff38b86c9d560dea10cf3459bb5f4caf72fc1bb932c7136", | ||||||
|                 "sha256:31dfa2fc323097f8ad7acd41aa38d7c614dd1960ac6681745b6da124093dc351", |                 "sha256:a326f4240123a2ac66bb163eeba99578e9d63a8654a59f4688a79198f9aa10f8", | ||||||
|                 "sha256:34f82db7f80c49f38b032c5abb605c458bac997a6c3142e0d6c130be6fb2b941", |                 "sha256:ae402f43604e3b2bc41e8ea8b8526c7fa7139ed76b0d64fc48e28125925275b2", | ||||||
|                 "sha256:3d5dd8e5998fb4ace04789d1d008e2bb532de501218519d70bb672c4c5a2fc5d", |                 "sha256:aee283c49601fa4c13adc64c09c978838a7e812f85377ae130a24d7198c0331e", | ||||||
|                 "sha256:4a6ae52bd3ee41ee0f3acf4c60ceb3f44e0e3bc52ab7da1c2b2aa6703363a3d1", |                 "sha256:b51249fdd2923739cd3efc95a3d6c363b67bbf779208e9f37fd5e68540d1a4d4", | ||||||
|                 "sha256:4b02a3b2a2f01d0490dd39321c74273fed0568568ea0e7ea23e02bd1fb10a10b", |                 "sha256:bb519becc46275c594410c6c28a8a0adc66fe24fef154a9addea54c1adb006f5", | ||||||
|                 "sha256:4b843f8e1dd6a3195679d9838eb4670222e8b8d01bc36c9894d6c3538316fa0a", |                 "sha256:c2c37185fb0af79d5c117b8d2764f4321eeb12ba8c141a95d0aa8c2c1d0a11dd", | ||||||
|                 "sha256:5de53a28f40ef3c4fd57aeab6b590c2c663de87a5af76136ced519923d3efbb3", |                 "sha256:dc561313279f9d05a3d0ffa89cd15ae477528ea37aa9795c4654588a3287a9ab", | ||||||
|                 "sha256:61b2b33ede821b94fa99ce0b09c9ece049c7067a33b279f343adfe35108a4ea7", |                 "sha256:e439c9a10a95cb32abd708bb8be83b2134fa93790a4fb0535ca36db3dda94d20", | ||||||
|                 "sha256:6a3a9b0f45fd75dc05d8e93dc21b18fc1670135ec9544d1ad4acbcf6b86781d0", |                 "sha256:fc3b4adc2ee8474cb3cd2a155305d5f8eda0a9c91320f83e55748e1fcb68f8e3" | ||||||
|                 "sha256:76ad8e4c69dadbb31bad17c16baee61c0d1a4a73bed2590b741b2e1a46d3edd0", |  | ||||||
|                 "sha256:7ba19b777dc00194d1b473180d4ca89a054dd18de27d0ee2e42a103ec9b7d014", |  | ||||||
|                 "sha256:7c1b7eab7a49aa96f3db1f716f0113a8a2e93c7375dd3d5d21c4941f1405c9c5", |  | ||||||
|                 "sha256:7fc0eee3046041387cbace9314926aa48b681202f8897f8bff3809967a049036", |  | ||||||
|                 "sha256:8ccd1c5fff1aa1427100ce188557fc31f1e0a383ad8ec42c559aabd4ff08802d", |  | ||||||
|                 "sha256:8e08dd76de80539d613654915a2f5196dbccc67448df291e69a88712ea21e24a", |  | ||||||
|                 "sha256:c18498c50c59263841862ea0501da9f2b3659c00db54abfbf823a80787fde8ce", |  | ||||||
|                 "sha256:c49db89d602c24928e68c0d510f4fcf8989d77defd01c973d6cbe27e684833b1", |  | ||||||
|                 "sha256:ce20044d0317649ddbb4e54dab3c1bcc7483c78c27d3f58ab3d0c7e6bc60d26a", |  | ||||||
|                 "sha256:d1071414dd06ca2eafa90c85a079169bfeb0e5f57fd0b45d44c092546fcd6fd9", |  | ||||||
|                 "sha256:d3be11ac43ab1a3e979dac80843b42226d5d3cccd3986f2e03152720a4297cd7", |  | ||||||
|                 "sha256:db603a1c235d110c860d5f39988ebc8218ee028f07a7cbc056ba6424372ca31b" |  | ||||||
|             ], |             ], | ||||||
|             "version": "==4.5.2" |             "version": "==4.7.5" | ||||||
|  |         }, | ||||||
|  |         "pathspec": { | ||||||
|  |             "hashes": [ | ||||||
|  |                 "sha256:7d91249d21749788d07a2d0f94147accd8f845507400749ea19c1ec9054a12b0", | ||||||
|  |                 "sha256:da45173eb3a6f2a5a487efba21f050af2b41948be6ab52b6a1e3ff22bb8b7061" | ||||||
|  |             ], | ||||||
|  |             "version": "==0.8.0" | ||||||
|         }, |         }, | ||||||
|         "protobuf": { |         "protobuf": { | ||||||
|             "hashes": [ |             "hashes": [ | ||||||
|                 "sha256:125713564d8cfed7610e52444c9769b8dcb0b55e25cc7841f2290ee7bc86636f", |                 "sha256:0bae429443cc4748be2aadfdaf9633297cfaeb24a9a02d0ab15849175ce90fab", | ||||||
|                 "sha256:1accdb7a47e51503be64d9a57543964ba674edac103215576399d2d0e34eac77", |                 "sha256:24e3b6ad259544d717902777b33966a1a069208c885576254c112663e6a5bb0f", | ||||||
|                 "sha256:27003d12d4f68e3cbea9eb67427cab3bfddd47ff90670cb367fcd7a3a89b9657", |                 "sha256:310a7aca6e7f257510d0c750364774034272538d51796ca31d42c3925d12a52a", | ||||||
|                 "sha256:3264f3c431a631b0b31e9db2ae8c927b79fc1a7b1b06b31e8e5bcf2af91fe896", |                 "sha256:52e586072612c1eec18e1174f8e3bb19d08f075fc2e3f91d3b16c919078469d0", | ||||||
|                 "sha256:3c5ab0f5c71ca5af27143e60613729e3488bb45f6d3f143dc918a20af8bab0bf", |                 "sha256:73152776dc75f335c476d11d52ec6f0f6925774802cd48d6189f4d5d7fe753f4", | ||||||
|                 "sha256:45dcf8758873e3f69feab075e5f3177270739f146255225474ee0b90429adef6", |                 "sha256:7774bbbaac81d3ba86de646c39f154afc8156717972bf0450c9dbfa1dc8dbea2", | ||||||
|                 "sha256:56a77d61a91186cc5676d8e11b36a5feb513873e4ae88d2ee5cf530d52bbcd3b", |                 "sha256:82d7ac987715d8d1eb4068bf997f3053468e0ce0287e2729c30601feb6602fee", | ||||||
|                 "sha256:5984e4947bbcef5bd849d6244aec507d31786f2dd3344139adc1489fb403b300", |                 "sha256:8eb9c93798b904f141d9de36a0ba9f9b73cc382869e67c9e642c0aba53b0fc07", | ||||||
|                 "sha256:6b0441da73796dd00821763bb4119674eaf252776beb50ae3883bed179a60b2a", |                 "sha256:adf0e4d57b33881d0c63bb11e7f9038f98ee0c3e334c221f0858f826e8fb0151", | ||||||
|                 "sha256:6f6677c5ade94d4fe75a912926d6796d5c71a2a90c2aeefe0d6f211d75c74789", |                 "sha256:c40973a0aee65422d8cb4e7d7cbded95dfeee0199caab54d5ab25b63bce8135a", | ||||||
|                 "sha256:84a825a9418d7196e2acc48f8746cf1ee75877ed2f30433ab92a133f3eaf8fbe", |                 "sha256:c77c974d1dadf246d789f6dad1c24426137c9091e930dbf50e0a29c1fcf00b1f", | ||||||
|                 "sha256:b842c34fe043ccf78b4a6cf1019d7b80113707d68c88842d061fa2b8fb6ddedc", |                 "sha256:dd9aa4401c36785ea1b6fff0552c674bdd1b641319cb07ed1fe2392388e9b0d7", | ||||||
|                 "sha256:ca33d2f09dae149a1dcf942d2d825ebb06343b77b437198c9e2ef115cf5d5bc1", |                 "sha256:e11df1ac6905e81b815ab6fd518e79be0a58b5dc427a2cf7208980f30694b956", | ||||||
|                 "sha256:db83b5c12c0cd30150bb568e6feb2435c49ce4e68fe2d7b903113f0e221e58fe", |                 "sha256:e2f8a75261c26b2f5f3442b0525d50fd79a71aeca04b5ec270fc123536188306", | ||||||
|                 "sha256:f50f3b1c5c1c1334ca7ce9cad5992f098f460ffd6388a3cabad10b66c2006b09", |                 "sha256:e512b7f3a4dd780f59f1bf22c302740e27b10b5c97e858a6061772668cd6f961", | ||||||
|                 "sha256:f99f127909731cafb841c52f9216e447d3e4afb99b17bebfad327a75aee206de" |                 "sha256:ef2c2e56aaf9ee914d3dccc3408d42661aaf7d9bb78eaa8f17b2e6282f214481", | ||||||
|  |                 "sha256:fac513a9dc2a74b99abd2e17109b53945e364649ca03d9f7a0b96aa8d1807d0a", | ||||||
|  |                 "sha256:fdfb6ad138dbbf92b5dbea3576d7c8ba7463173f7d2cb0ca1bd336ec88ddbd80" | ||||||
|             ], |             ], | ||||||
|             "index": "pypi", |             "index": "pypi", | ||||||
|             "version": "==3.10.0" |             "version": "==3.11.3" | ||||||
|  |         }, | ||||||
|  |         "regex": { | ||||||
|  |             "hashes": [ | ||||||
|  |                 "sha256:08119f707f0ebf2da60d2f24c2f39ca616277bb67ef6c92b72cbf90cbe3a556b", | ||||||
|  |                 "sha256:0ce9537396d8f556bcfc317c65b6a0705320701e5ce511f05fc04421ba05b8a8", | ||||||
|  |                 "sha256:1cbe0fa0b7f673400eb29e9ef41d4f53638f65f9a2143854de6b1ce2899185c3", | ||||||
|  |                 "sha256:2294f8b70e058a2553cd009df003a20802ef75b3c629506be20687df0908177e", | ||||||
|  |                 "sha256:23069d9c07e115537f37270d1d5faea3e0bdded8279081c4d4d607a2ad393683", | ||||||
|  |                 "sha256:24f4f4062eb16c5bbfff6a22312e8eab92c2c99c51a02e39b4eae54ce8255cd1", | ||||||
|  |                 "sha256:295badf61a51add2d428a46b8580309c520d8b26e769868b922750cf3ce67142", | ||||||
|  |                 "sha256:2a3bf8b48f8e37c3a40bb3f854bf0121c194e69a650b209628d951190b862de3", | ||||||
|  |                 "sha256:4385f12aa289d79419fede43f979e372f527892ac44a541b5446617e4406c468", | ||||||
|  |                 "sha256:5635cd1ed0a12b4c42cce18a8d2fb53ff13ff537f09de5fd791e97de27b6400e", | ||||||
|  |                 "sha256:5bfed051dbff32fd8945eccca70f5e22b55e4148d2a8a45141a3b053d6455ae3", | ||||||
|  |                 "sha256:7e1037073b1b7053ee74c3c6c0ada80f3501ec29d5f46e42669378eae6d4405a", | ||||||
|  |                 "sha256:90742c6ff121a9c5b261b9b215cb476eea97df98ea82037ec8ac95d1be7a034f", | ||||||
|  |                 "sha256:a58dd45cb865be0ce1d5ecc4cfc85cd8c6867bea66733623e54bd95131f473b6", | ||||||
|  |                 "sha256:c087bff162158536387c53647411db09b6ee3f9603c334c90943e97b1052a156", | ||||||
|  |                 "sha256:c162a21e0da33eb3d31a3ac17a51db5e634fc347f650d271f0305d96601dc15b", | ||||||
|  |                 "sha256:c9423a150d3a4fc0f3f2aae897a59919acd293f4cb397429b120a5fcd96ea3db", | ||||||
|  |                 "sha256:ccccdd84912875e34c5ad2d06e1989d890d43af6c2242c6fcfa51556997af6cd", | ||||||
|  |                 "sha256:e91ba11da11cf770f389e47c3f5c30473e6d85e06d7fd9dcba0017d2867aab4a", | ||||||
|  |                 "sha256:ea4adf02d23b437684cd388d557bf76e3afa72f7fed5bbc013482cc00c816948", | ||||||
|  |                 "sha256:fb95debbd1a824b2c4376932f2216cc186912e389bdb0e27147778cf6acb3f89" | ||||||
|  |             ], | ||||||
|  |             "version": "==2020.4.4" | ||||||
|         }, |         }, | ||||||
|         "six": { |         "six": { | ||||||
|             "hashes": [ |             "hashes": [ | ||||||
|                 "sha256:3350809f0555b11f552448330d0b52d5f24c91a322ea4a15ef22629740f3761c", |                 "sha256:236bdbdce46e6e6a3d61a337c0f8b763ca1e8717c03b369e87a7ec7ce1319c0a", | ||||||
|                 "sha256:d16a0141ec1a18405cd4ce8b4613101da75da0e9a7aec5bdd4fa804d0e0eba73" |                 "sha256:8f3cd2e254d8f793e7f3d6d9df77b92252b52637291d0f0da013c76ea2724b6c" | ||||||
|             ], |             ], | ||||||
|             "version": "==1.12.0" |             "version": "==1.14.0" | ||||||
|         }, |         }, | ||||||
|         "stringcase": { |         "stringcase": { | ||||||
|             "hashes": [ |             "hashes": [ | ||||||
| @@ -154,16 +215,42 @@ | |||||||
|             ], |             ], | ||||||
|             "index": "pypi", |             "index": "pypi", | ||||||
|             "version": "==1.2.0" |             "version": "==1.2.0" | ||||||
|  |         }, | ||||||
|  |         "toml": { | ||||||
|  |             "hashes": [ | ||||||
|  |                 "sha256:229f81c57791a41d65e399fc06bf0848bab550a9dfd5ed66df18ce5f05e73d5c", | ||||||
|  |                 "sha256:235682dd292d5899d361a811df37e04a8828a5b1da3115886b73cf81ebc9100e" | ||||||
|  |             ], | ||||||
|  |             "version": "==0.10.0" | ||||||
|  |         }, | ||||||
|  |         "typed-ast": { | ||||||
|  |             "hashes": [ | ||||||
|  |                 "sha256:0666aa36131496aed8f7be0410ff974562ab7eeac11ef351def9ea6fa28f6355", | ||||||
|  |                 "sha256:0c2c07682d61a629b68433afb159376e24e5b2fd4641d35424e462169c0a7919", | ||||||
|  |                 "sha256:249862707802d40f7f29f6e1aad8d84b5aa9e44552d2cc17384b209f091276aa", | ||||||
|  |                 "sha256:24995c843eb0ad11a4527b026b4dde3da70e1f2d8806c99b7b4a7cf491612652", | ||||||
|  |                 "sha256:269151951236b0f9a6f04015a9004084a5ab0d5f19b57de779f908621e7d8b75", | ||||||
|  |                 "sha256:4083861b0aa07990b619bd7ddc365eb7fa4b817e99cf5f8d9cf21a42780f6e01", | ||||||
|  |                 "sha256:498b0f36cc7054c1fead3d7fc59d2150f4d5c6c56ba7fb150c013fbc683a8d2d", | ||||||
|  |                 "sha256:4e3e5da80ccbebfff202a67bf900d081906c358ccc3d5e3c8aea42fdfdfd51c1", | ||||||
|  |                 "sha256:6daac9731f172c2a22ade6ed0c00197ee7cc1221aa84cfdf9c31defeb059a907", | ||||||
|  |                 "sha256:715ff2f2df46121071622063fc7543d9b1fd19ebfc4f5c8895af64a77a8c852c", | ||||||
|  |                 "sha256:73d785a950fc82dd2a25897d525d003f6378d1cb23ab305578394694202a58c3", | ||||||
|  |                 "sha256:8c8aaad94455178e3187ab22c8b01a3837f8ee50e09cf31f1ba129eb293ec30b", | ||||||
|  |                 "sha256:8ce678dbaf790dbdb3eba24056d5364fb45944f33553dd5869b7580cdbb83614", | ||||||
|  |                 "sha256:aaee9905aee35ba5905cfb3c62f3e83b3bec7b39413f0a7f19be4e547ea01ebb", | ||||||
|  |                 "sha256:bcd3b13b56ea479b3650b82cabd6b5343a625b0ced5429e4ccad28a8973f301b", | ||||||
|  |                 "sha256:c9e348e02e4d2b4a8b2eedb48210430658df6951fa484e59de33ff773fbd4b41", | ||||||
|  |                 "sha256:d205b1b46085271b4e15f670058ce182bd1199e56b317bf2ec004b6a44f911f6", | ||||||
|  |                 "sha256:d43943ef777f9a1c42bf4e552ba23ac77a6351de620aa9acf64ad54933ad4d34", | ||||||
|  |                 "sha256:d5d33e9e7af3b34a40dc05f498939f0ebf187f07c385fd58d591c533ad8562fe", | ||||||
|  |                 "sha256:fc0fea399acb12edbf8a628ba8d2312f583bdbdb3335635db062fa98cf71fca4", | ||||||
|  |                 "sha256:fe460b922ec15dd205595c9b5b99e2f056fd98ae8f9f56b888e7a17dc2b757e7" | ||||||
|  |             ], | ||||||
|  |             "version": "==1.4.1" | ||||||
|         } |         } | ||||||
|     }, |     }, | ||||||
|     "develop": { |     "develop": { | ||||||
|         "atomicwrites": { |  | ||||||
|             "hashes": [ |  | ||||||
|                 "sha256:03472c30eb2c5d1ba9227e4c2ca66ab8287fbfbbda3888aa93dc2e28fc6811b4", |  | ||||||
|                 "sha256:75a9445bac02d8d058d5e1fe689654ba5a6556a1dfd8ce6ec55a0ed79866cfa6" |  | ||||||
|             ], |  | ||||||
|             "version": "==1.3.0" |  | ||||||
|         }, |  | ||||||
|         "attrs": { |         "attrs": { | ||||||
|             "hashes": [ |             "hashes": [ | ||||||
|                 "sha256:08a96c641c3a74e44eb59afb61a24f2cb9f4d7188748e76ba4bb5edfa3cb7d1c", |                 "sha256:08a96c641c3a74e44eb59afb61a24f2cb9f4d7188748e76ba4bb5edfa3cb7d1c", | ||||||
| @@ -180,19 +267,19 @@ | |||||||
|         }, |         }, | ||||||
|         "flake8": { |         "flake8": { | ||||||
|             "hashes": [ |             "hashes": [ | ||||||
|                 "sha256:19241c1cbc971b9962473e4438a2ca19749a7dd002dd1a946eaba171b4114548", |                 "sha256:45681a117ecc81e870cbf1262835ae4af5e7a8b08e40b944a8a6e6b895914cfb", | ||||||
|                 "sha256:8e9dfa3cecb2400b3738a42c54c3043e821682b9c840b0448c0503f781130696" |                 "sha256:49356e766643ad15072a789a20915d3c91dc89fd313ccd71802303fd67e4deca" | ||||||
|             ], |             ], | ||||||
|             "index": "pypi", |             "index": "pypi", | ||||||
|             "version": "==3.7.8" |             "version": "==3.7.9" | ||||||
|         }, |         }, | ||||||
|         "importlib-metadata": { |         "importlib-metadata": { | ||||||
|             "hashes": [ |             "hashes": [ | ||||||
|                 "sha256:aa18d7378b00b40847790e7c27e11673d7fed219354109d0e7b9e5b25dc3ad26", |                 "sha256:2a688cbaa90e0cc587f1df48bdc97a6eadccdcd9c35fb3f976a09e3b5016d90f", | ||||||
|                 "sha256:d5f18a79777f3aa179c145737780282e27b508fc8fd688cb17c7a813e8bd39af" |                 "sha256:34513a8a0c4962bc66d35b359558fd8a5e10cd472d37aec5f66858addef32c1e" | ||||||
|             ], |             ], | ||||||
|             "markers": "python_version < '3.8'", |             "markers": "python_version < '3.8'", | ||||||
|             "version": "==0.23" |             "version": "==1.6.0" | ||||||
|         }, |         }, | ||||||
|         "isort": { |         "isort": { | ||||||
|             "hashes": [ |             "hashes": [ | ||||||
| @@ -211,30 +298,30 @@ | |||||||
|         }, |         }, | ||||||
|         "more-itertools": { |         "more-itertools": { | ||||||
|             "hashes": [ |             "hashes": [ | ||||||
|                 "sha256:409cd48d4db7052af495b09dec721011634af3753ae1ef92d2b32f73a745f832", |                 "sha256:5dd8bcf33e5f9513ffa06d5ad33d78f31e1931ac9a18f33d37e77a180d393a7c", | ||||||
|                 "sha256:92b8c4b06dac4f0611c0729b2f2ede52b2e1bac1ab48f089c7ddc12e26bb60c4" |                 "sha256:b1ddb932186d8a6ac451e1d95844b382f55e12686d51ca0c68b6f61f2ab7a507" | ||||||
|             ], |             ], | ||||||
|             "version": "==7.2.0" |             "version": "==8.2.0" | ||||||
|         }, |         }, | ||||||
|         "mypy": { |         "mypy": { | ||||||
|             "hashes": [ |             "hashes": [ | ||||||
|                 "sha256:1521c186a3d200c399bd5573c828ea2db1362af7209b2adb1bb8532cea2fb36f", |                 "sha256:15b948e1302682e3682f11f50208b726a246ab4e6c1b39f9264a8796bb416aa2", | ||||||
|                 "sha256:31a046ab040a84a0fc38bc93694876398e62bc9f35eca8ccbf6418b7297f4c00", |                 "sha256:219a3116ecd015f8dca7b5d2c366c973509dfb9a8fc97ef044a36e3da66144a1", | ||||||
|                 "sha256:3b1a411909c84b2ae9b8283b58b48541654b918e8513c20a400bb946aa9111ae", |                 "sha256:3b1fc683fb204c6b4403a1ef23f0b1fac8e4477091585e0c8c54cbdf7d7bb164", | ||||||
|                 "sha256:48c8bc99380575deb39f5d3400ebb6a8a1cb5cc669bbba4d3bb30f904e0a0e7d", |                 "sha256:3beff56b453b6ef94ecb2996bea101a08f1f8a9771d3cbf4988a61e4d9973761", | ||||||
|                 "sha256:540c9caa57a22d0d5d3c69047cc9dd0094d49782603eb03069821b41f9e970e9", |                 "sha256:7687f6455ec3ed7649d1ae574136835a4272b65b3ddcf01ab8704ac65616c5ce", | ||||||
|                 "sha256:672e418425d957e276c291930a3921b4a6413204f53fe7c37cad7bc57b9a3391", |                 "sha256:7ec45a70d40ede1ec7ad7f95b3c94c9cf4c186a32f6bacb1795b60abd2f9ef27", | ||||||
|                 "sha256:6ed3b9b3fdc7193ea7aca6f3c20549b377a56f28769783a8f27191903a54170f", |                 "sha256:86c857510a9b7c3104cf4cde1568f4921762c8f9842e987bc03ed4f160925754", | ||||||
|                 "sha256:9371290aa2cad5ad133e4cdc43892778efd13293406f7340b9ffe99d5ec7c1d9", |                 "sha256:8a627507ef9b307b46a1fea9513d5c98680ba09591253082b4c48697ba05a4ae", | ||||||
|                 "sha256:ace6ac1d0f87d4072f05b5468a084a45b4eda970e4d26704f201e06d47ab2990", |                 "sha256:8dfb69fbf9f3aeed18afffb15e319ca7f8da9642336348ddd6cab2713ddcf8f9", | ||||||
|                 "sha256:b428f883d2b3fe1d052c630642cc6afddd07d5cd7873da948644508be3b9d4a7", |                 "sha256:a34b577cdf6313bf24755f7a0e3f3c326d5c1f4fe7422d1d06498eb25ad0c600", | ||||||
|                 "sha256:d5bf0e6ec8ba346a2cf35cb55bf4adfddbc6b6576fcc9e10863daa523e418dbb", |                 "sha256:a8ffcd53cb5dfc131850851cc09f1c44689c2812d0beb954d8138d4f5fc17f65", | ||||||
|                 "sha256:d7574e283f83c08501607586b3167728c58e8442947e027d2d4c7dcd6d82f453", |                 "sha256:b90928f2d9eb2f33162405f32dde9f6dcead63a0971ca8a1b50eb4ca3e35ceb8", | ||||||
|                 "sha256:dc889c84241a857c263a2b1cd1121507db7d5b5f5e87e77147097230f374d10b", |                 "sha256:c56ffe22faa2e51054c5f7a3bc70a370939c2ed4de308c690e7949230c995913", | ||||||
|                 "sha256:f4748697b349f373002656bf32fede706a0e713d67bfdcf04edf39b1f61d46eb" |                 "sha256:f91c7ae919bbc3f96cd5e5b2e786b2b108343d1d7972ea130f7de27fdd547cf3" | ||||||
|             ], |             ], | ||||||
|             "index": "pypi", |             "index": "pypi", | ||||||
|             "version": "==0.740" |             "version": "==0.770" | ||||||
|         }, |         }, | ||||||
|         "mypy-extensions": { |         "mypy-extensions": { | ||||||
|             "hashes": [ |             "hashes": [ | ||||||
| @@ -245,24 +332,24 @@ | |||||||
|         }, |         }, | ||||||
|         "packaging": { |         "packaging": { | ||||||
|             "hashes": [ |             "hashes": [ | ||||||
|                 "sha256:28b924174df7a2fa32c1953825ff29c61e2f5e082343165438812f00d3a7fc47", |                 "sha256:3c292b474fda1671ec57d46d739d072bfd495a4f51ad01a055121d81e952b7a3", | ||||||
|                 "sha256:d9551545c6d761f3def1677baf08ab2a3ca17c56879e70fecba2fc4dde4ed108" |                 "sha256:82f77b9bee21c1bafbf35a84905d604d5d1223801d639cf3ed140bd651c08752" | ||||||
|             ], |             ], | ||||||
|             "version": "==19.2" |             "version": "==20.3" | ||||||
|         }, |         }, | ||||||
|         "pluggy": { |         "pluggy": { | ||||||
|             "hashes": [ |             "hashes": [ | ||||||
|                 "sha256:0db4b7601aae1d35b4a033282da476845aa19185c1e6964b25cf324b5e4ec3e6", |                 "sha256:15b2acde666561e1298d71b523007ed7364de07029219b604cf808bfa1c765b0", | ||||||
|                 "sha256:fa5fa1622fa6dd5c030e9cad086fa19ef6a0cf6d7a2d12318e10cb49d6d68f34" |                 "sha256:966c145cd83c96502c3c3868f50408687b38434af77734af1e9ca461a4081d2d" | ||||||
|             ], |             ], | ||||||
|             "version": "==0.13.0" |             "version": "==0.13.1" | ||||||
|         }, |         }, | ||||||
|         "py": { |         "py": { | ||||||
|             "hashes": [ |             "hashes": [ | ||||||
|                 "sha256:64f65755aee5b381cea27766a3a147c3f15b9b6b9ac88676de66ba2ae36793fa", |                 "sha256:5e27081401262157467ad6e7f851b7aa402c5852dbcb3dae06768434de5752aa", | ||||||
|                 "sha256:dc639b046a6e2cff5bbe40194ad65936d6ba360b52b3c3fe1d08a82dd50b5e53" |                 "sha256:c20fdd83a5dbc0af9efd622bee9a5564e278f6380fffcacc43ba6f43db2813b0" | ||||||
|             ], |             ], | ||||||
|             "version": "==1.8.0" |             "version": "==1.8.1" | ||||||
|         }, |         }, | ||||||
|         "pycodestyle": { |         "pycodestyle": { | ||||||
|             "hashes": [ |             "hashes": [ | ||||||
| @@ -280,81 +367,98 @@ | |||||||
|         }, |         }, | ||||||
|         "pyparsing": { |         "pyparsing": { | ||||||
|             "hashes": [ |             "hashes": [ | ||||||
|                 "sha256:6f98a7b9397e206d78cc01df10131398f1c8b8510a2f4d97d9abd82e1aacdd80", |                 "sha256:67199f0c41a9c702154efb0e7a8cc08accf830eb003b4d9fa42c4059002e2492", | ||||||
|                 "sha256:d9338df12903bbf5d65a0e4e87c2161968b10d2e489652bb47001d82a9b028b4" |                 "sha256:700d17888d441604b0bd51535908dcb297561b040819cccde647a92439db5a2a" | ||||||
|             ], |             ], | ||||||
|             "version": "==2.4.2" |             "version": "==3.0.0a1" | ||||||
|         }, |         }, | ||||||
|         "pytest": { |         "pytest": { | ||||||
|             "hashes": [ |             "hashes": [ | ||||||
|                 "sha256:7e4800063ccfc306a53c461442526c5571e1462f61583506ce97e4da6a1d88c8", |                 "sha256:0e5b30f5cb04e887b91b1ee519fa3d89049595f428c1db76e73bd7f17b09b172", | ||||||
|                 "sha256:ca563435f4941d0cb34767301c27bc65c510cb82e90b9ecf9cb52dc2c63caaa0" |                 "sha256:84dde37075b8805f3d1f392cc47e38a0e59518fb46a431cfdaf7cf1ce805f970" | ||||||
|             ], |             ], | ||||||
|             "index": "pypi", |             "index": "pypi", | ||||||
|             "version": "==5.2.1" |             "version": "==5.4.1" | ||||||
|  |         }, | ||||||
|  |         "pytest-asyncio": { | ||||||
|  |             "hashes": [ | ||||||
|  |                 "sha256:9fac5100fd716cbecf6ef89233e8590a4ad61d729d1732e0a96b84182df1daaf", | ||||||
|  |                 "sha256:d734718e25cfc32d2bf78d346e99d33724deeba774cc4afdf491530c6184b63b" | ||||||
|  |             ], | ||||||
|  |             "index": "pypi", | ||||||
|  |             "version": "==0.10.0" | ||||||
|         }, |         }, | ||||||
|         "rope": { |         "rope": { | ||||||
|             "hashes": [ |             "hashes": [ | ||||||
|                 "sha256:6b728fdc3e98a83446c27a91fc5d56808a004f8beab7a31ab1d7224cecc7d969", |                 "sha256:52423a7eebb5306a6d63bdc91a7c657db51ac9babfb8341c9a1440831ecf3203", | ||||||
|                 "sha256:c5c5a6a87f7b1a2095fb311135e2a3d1f194f5ecb96900fdd0a9100881f48aaf", |                 "sha256:ae1fa2fd56f64f4cc9be46493ce54bed0dd12dee03980c61a4393d89d84029ad", | ||||||
|                 "sha256:f0dcf719b63200d492b85535ebe5ea9b29e0d0b8aebeb87fe03fc1a65924fdaf" |                 "sha256:d2830142c2e046f5fc26a022fe680675b6f48f81c7fc1f03a950706e746e9dfe" | ||||||
|             ], |             ], | ||||||
|             "index": "pypi", |             "index": "pypi", | ||||||
|             "version": "==0.14.0" |             "version": "==0.16.0" | ||||||
|         }, |         }, | ||||||
|         "six": { |         "six": { | ||||||
|             "hashes": [ |             "hashes": [ | ||||||
|                 "sha256:3350809f0555b11f552448330d0b52d5f24c91a322ea4a15ef22629740f3761c", |                 "sha256:236bdbdce46e6e6a3d61a337c0f8b763ca1e8717c03b369e87a7ec7ce1319c0a", | ||||||
|                 "sha256:d16a0141ec1a18405cd4ce8b4613101da75da0e9a7aec5bdd4fa804d0e0eba73" |                 "sha256:8f3cd2e254d8f793e7f3d6d9df77b92252b52637291d0f0da013c76ea2724b6c" | ||||||
|             ], |             ], | ||||||
|             "version": "==1.12.0" |             "version": "==1.14.0" | ||||||
|         }, |         }, | ||||||
|         "typed-ast": { |         "typed-ast": { | ||||||
|             "hashes": [ |             "hashes": [ | ||||||
|                 "sha256:1170afa46a3799e18b4c977777ce137bb53c7485379d9706af8a59f2ea1aa161", |                 "sha256:0666aa36131496aed8f7be0410ff974562ab7eeac11ef351def9ea6fa28f6355", | ||||||
|                 "sha256:18511a0b3e7922276346bcb47e2ef9f38fb90fd31cb9223eed42c85d1312344e", |                 "sha256:0c2c07682d61a629b68433afb159376e24e5b2fd4641d35424e462169c0a7919", | ||||||
|                 "sha256:262c247a82d005e43b5b7f69aff746370538e176131c32dda9cb0f324d27141e", |                 "sha256:249862707802d40f7f29f6e1aad8d84b5aa9e44552d2cc17384b209f091276aa", | ||||||
|                 "sha256:2b907eb046d049bcd9892e3076c7a6456c93a25bebfe554e931620c90e6a25b0", |                 "sha256:24995c843eb0ad11a4527b026b4dde3da70e1f2d8806c99b7b4a7cf491612652", | ||||||
|                 "sha256:354c16e5babd09f5cb0ee000d54cfa38401d8b8891eefa878ac772f827181a3c", |                 "sha256:269151951236b0f9a6f04015a9004084a5ab0d5f19b57de779f908621e7d8b75", | ||||||
|                 "sha256:48e5b1e71f25cfdef98b013263a88d7145879fbb2d5185f2a0c79fa7ebbeae47", |                 "sha256:4083861b0aa07990b619bd7ddc365eb7fa4b817e99cf5f8d9cf21a42780f6e01", | ||||||
|                 "sha256:4e0b70c6fc4d010f8107726af5fd37921b666f5b31d9331f0bd24ad9a088e631", |                 "sha256:498b0f36cc7054c1fead3d7fc59d2150f4d5c6c56ba7fb150c013fbc683a8d2d", | ||||||
|                 "sha256:630968c5cdee51a11c05a30453f8cd65e0cc1d2ad0d9192819df9978984529f4", |                 "sha256:4e3e5da80ccbebfff202a67bf900d081906c358ccc3d5e3c8aea42fdfdfd51c1", | ||||||
|                 "sha256:66480f95b8167c9c5c5c87f32cf437d585937970f3fc24386f313a4c97b44e34", |                 "sha256:6daac9731f172c2a22ade6ed0c00197ee7cc1221aa84cfdf9c31defeb059a907", | ||||||
|                 "sha256:71211d26ffd12d63a83e079ff258ac9d56a1376a25bc80b1cdcdf601b855b90b", |                 "sha256:715ff2f2df46121071622063fc7543d9b1fd19ebfc4f5c8895af64a77a8c852c", | ||||||
|                 "sha256:7954560051331d003b4e2b3eb822d9dd2e376fa4f6d98fee32f452f52dd6ebb2", |                 "sha256:73d785a950fc82dd2a25897d525d003f6378d1cb23ab305578394694202a58c3", | ||||||
|                 "sha256:838997f4310012cf2e1ad3803bce2f3402e9ffb71ded61b5ee22617b3a7f6b6e", |                 "sha256:8c8aaad94455178e3187ab22c8b01a3837f8ee50e09cf31f1ba129eb293ec30b", | ||||||
|                 "sha256:95bd11af7eafc16e829af2d3df510cecfd4387f6453355188342c3e79a2ec87a", |                 "sha256:8ce678dbaf790dbdb3eba24056d5364fb45944f33553dd5869b7580cdbb83614", | ||||||
|                 "sha256:bc6c7d3fa1325a0c6613512a093bc2a2a15aeec350451cbdf9e1d4bffe3e3233", |                 "sha256:aaee9905aee35ba5905cfb3c62f3e83b3bec7b39413f0a7f19be4e547ea01ebb", | ||||||
|                 "sha256:cc34a6f5b426748a507dd5d1de4c1978f2eb5626d51326e43280941206c209e1", |                 "sha256:bcd3b13b56ea479b3650b82cabd6b5343a625b0ced5429e4ccad28a8973f301b", | ||||||
|                 "sha256:d755f03c1e4a51e9b24d899561fec4ccaf51f210d52abdf8c07ee2849b212a36", |                 "sha256:c9e348e02e4d2b4a8b2eedb48210430658df6951fa484e59de33ff773fbd4b41", | ||||||
|                 "sha256:d7c45933b1bdfaf9f36c579671fec15d25b06c8398f113dab64c18ed1adda01d", |                 "sha256:d205b1b46085271b4e15f670058ce182bd1199e56b317bf2ec004b6a44f911f6", | ||||||
|                 "sha256:d896919306dd0aa22d0132f62a1b78d11aaf4c9fc5b3410d3c666b818191630a", |                 "sha256:d43943ef777f9a1c42bf4e552ba23ac77a6351de620aa9acf64ad54933ad4d34", | ||||||
|                 "sha256:fdc1c9bbf79510b76408840e009ed65958feba92a88833cdceecff93ae8fff66", |                 "sha256:d5d33e9e7af3b34a40dc05f498939f0ebf187f07c385fd58d591c533ad8562fe", | ||||||
|                 "sha256:ffde2fbfad571af120fcbfbbc61c72469e72f550d676c3342492a9dfdefb8f12" |                 "sha256:fc0fea399acb12edbf8a628ba8d2312f583bdbdb3335635db062fa98cf71fca4", | ||||||
|  |                 "sha256:fe460b922ec15dd205595c9b5b99e2f056fd98ae8f9f56b888e7a17dc2b757e7" | ||||||
|             ], |             ], | ||||||
|             "version": "==1.4.0" |             "version": "==1.4.1" | ||||||
|         }, |         }, | ||||||
|         "typing-extensions": { |         "typing-extensions": { | ||||||
|             "hashes": [ |             "hashes": [ | ||||||
|                 "sha256:2ed632b30bb54fc3941c382decfd0ee4148f5c591651c9272473fea2c6397d95", |                 "sha256:6e95524d8a547a91e08f404ae485bbb71962de46967e1b71a0cb89af24e761c5", | ||||||
|                 "sha256:b1edbbf0652660e32ae780ac9433f4231e7339c7f9a8057d0f042fcbcea49b87", |                 "sha256:79ee589a3caca649a9bfd2a8de4709837400dfa00b6cc81962a1e6a1815969ae", | ||||||
|                 "sha256:d8179012ec2c620d3791ca6fe2bf7979d979acdbef1fca0bc56b37411db682ed" |                 "sha256:f8d2bd89d25bc39dabe7d23df520442fa1d8969b82544370e03d88b5a591c392" | ||||||
|             ], |             ], | ||||||
|             "version": "==3.7.4" |             "version": "==3.7.4.2" | ||||||
|  |         }, | ||||||
|  |         "v": { | ||||||
|  |             "hashes": [ | ||||||
|  |                 "sha256:2d5a8f79a36aaebe62ef2c7068e3ec7f86656078202edabfdbf74715dc822d36", | ||||||
|  |                 "sha256:cd6b6b20b4a611f209c88bcdfb7211321f85662efb2bdd53a7b40314d0a84618" | ||||||
|  |             ], | ||||||
|  |             "index": "pypi", | ||||||
|  |             "version": "==0.0.0" | ||||||
|         }, |         }, | ||||||
|         "wcwidth": { |         "wcwidth": { | ||||||
|             "hashes": [ |             "hashes": [ | ||||||
|                 "sha256:3df37372226d6e63e1b1e1eda15c594bca98a22d33a23832a90998faa96bc65e", |                 "sha256:cafe2186b3c009a04067022ce1dcd79cb38d8d65ee4f4791b8888d6599d1bbe1", | ||||||
|                 "sha256:f4ebe71925af7b40a864553f761ed559b43544f8f71746c2d756c7fe788ade7c" |                 "sha256:ee73862862a156bf77ff92b09034fc4825dd3af9cf81bc5b360668d425f3c5f1" | ||||||
|             ], |             ], | ||||||
|             "version": "==0.1.7" |             "version": "==0.1.9" | ||||||
|         }, |         }, | ||||||
|         "zipp": { |         "zipp": { | ||||||
|             "hashes": [ |             "hashes": [ | ||||||
|                 "sha256:3718b1cbcd963c7d4c5511a8240812904164b7f381b647143a89d3b98f9bcd8e", |                 "sha256:aa36550ff0c0b7ef7fa639055d797116ee891440eac1a56f378e2d3179e0320b", | ||||||
|                 "sha256:f06903e9f1f43b12d371004b4ac7b06ab39a44adc747266928ae6debfa7b3335" |                 "sha256:c599e4d75c98f6798c509911d08a22e6c021d074469042177c8c86fb92eefd96" | ||||||
|             ], |             ], | ||||||
|             "version": "==0.6.0" |             "version": "==3.1.0" | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										39
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										39
									
								
								README.md
									
									
									
									
									
								
							| @@ -2,11 +2,11 @@ | |||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| This project aims to provide an improved experience when using Protobuf / gRPC in a modern Python environment by making use of modern language features and generating readable, understandable, idiomatic Python code. It will not support legacy features or environments. The following are supported: | This project aims to provide an improved experience when using Protobuf / gRPC in a modern Python environment by making use of modern language features and generating readable, understandable, idiomatic Python code. It will not support legacy features or environments (e.g. Protobuf 2). The following are supported: | ||||||
|  |  | ||||||
| - Protobuf 3 & gRPC code generation | - Protobuf 3 & gRPC code generation | ||||||
|   - Both binary & JSON serialization is built-in |   - Both binary & JSON serialization is built-in | ||||||
| - Python 3.7+ making use of: | - Python 3.6+ making use of: | ||||||
|   - Enums |   - Enums | ||||||
|   - Dataclasses |   - Dataclasses | ||||||
|   - `async`/`await` |   - `async`/`await` | ||||||
| @@ -46,7 +46,7 @@ First, install the package. Note that the `[compiler]` feature flag tells it to | |||||||
|  |  | ||||||
| ```sh | ```sh | ||||||
| # Install both the library and compiler | # Install both the library and compiler | ||||||
| $ pip install betterproto[compiler] | $ pip install "betterproto[compiler]" | ||||||
|  |  | ||||||
| # Install just the library (to use the generated code output) | # Install just the library (to use the generated code output) | ||||||
| $ pip install betterproto | $ pip install betterproto | ||||||
| @@ -281,7 +281,7 @@ You can do stuff like: | |||||||
| ```py | ```py | ||||||
| >>> t = Test().from_dict({"maybe": True, "ts": "2019-01-01T12:00:00Z", "duration": "1.200s"}) | >>> t = Test().from_dict({"maybe": True, "ts": "2019-01-01T12:00:00Z", "duration": "1.200s"}) | ||||||
| >>> t | >>> t | ||||||
| st(maybe=True, ts=datetime.datetime(2019, 1, 1, 12, 0, tzinfo=datetime.timezone.utc), duration=datetime.timedelta(seconds=1, microseconds=200000)) | Test(maybe=True, ts=datetime.datetime(2019, 1, 1, 12, 0, tzinfo=datetime.timezone.utc), duration=datetime.timedelta(seconds=1, microseconds=200000)) | ||||||
|  |  | ||||||
| >>> t.ts - t.duration | >>> t.ts - t.duration | ||||||
| datetime.datetime(2019, 1, 1, 11, 59, 58, 800000, tzinfo=datetime.timezone.utc) | datetime.datetime(2019, 1, 1, 11, 59, 58, 800000, tzinfo=datetime.timezone.utc) | ||||||
| @@ -296,7 +296,7 @@ datetime.datetime(2019, 1, 1, 11, 59, 58, 800000, tzinfo=datetime.timezone.utc) | |||||||
|  |  | ||||||
| ## Development | ## Development | ||||||
|  |  | ||||||
| First, make sure you have Python 3.7+ and `pipenv` installed, along with the official [Protobuf Compiler](https://github.com/protocolbuffers/protobuf/releases) for your platform. Then: | First, make sure you have Python 3.6+ and `pipenv` installed, along with the official [Protobuf Compiler](https://github.com/protocolbuffers/protobuf/releases) for your platform. Then: | ||||||
|  |  | ||||||
| ```sh | ```sh | ||||||
| # Get set up with the virtual env & dependencies | # Get set up with the virtual env & dependencies | ||||||
| @@ -311,10 +311,26 @@ $ pip install -e . | |||||||
|  |  | ||||||
| There are two types of tests: | There are two types of tests: | ||||||
|  |  | ||||||
| 1. Manually-written tests for some behavior of the library | 1. Standard tests | ||||||
| 2. Proto files and JSON inputs for automated tests | 2. Custom tests | ||||||
|  |  | ||||||
| For #2, you can add a new `*.proto` file into the `betterproto/tests` directory along with a sample `*.json` input and it will get automatically picked up. | #### Standard tests | ||||||
|  |  | ||||||
|  | Adding a standard test case is easy. | ||||||
|  |  | ||||||
|  | - Create a new directory `betterproto/tests/inputs/<name>` | ||||||
|  |   - add `<name>.proto`  with a message called `Test` | ||||||
|  |   - add `<name>.json` with some test data | ||||||
|  |  | ||||||
|  | It will be picked up automatically when you run the tests. | ||||||
|  |  | ||||||
|  | - See also: [Standard Tests Development Guide](betterproto/tests/README.md)  | ||||||
|  |  | ||||||
|  | #### Custom tests | ||||||
|  |  | ||||||
|  | Custom tests are found in `tests/test_*.py` and are run with pytest. | ||||||
|  |  | ||||||
|  | #### Running | ||||||
|  |  | ||||||
| Here's how to run the tests. | Here's how to run the tests. | ||||||
|  |  | ||||||
| @@ -322,8 +338,8 @@ Here's how to run the tests. | |||||||
| # Generate assets from sample .proto files | # Generate assets from sample .proto files | ||||||
| $ pipenv run generate | $ pipenv run generate | ||||||
|  |  | ||||||
| # Run the tests | # Run all tests | ||||||
| $ pipenv run tests | $ pipenv run test | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ### TODO | ### TODO | ||||||
| @@ -340,6 +356,9 @@ $ pipenv run tests | |||||||
| - [x] Refs to nested types | - [x] Refs to nested types | ||||||
| - [x] Imports in proto files | - [x] Imports in proto files | ||||||
| - [x] Well-known Google types | - [x] Well-known Google types | ||||||
|  |   - [ ] Support as request input | ||||||
|  |   - [ ] Support as response output | ||||||
|  |     - [ ] Automatically wrap/unwrap responses | ||||||
| - [x] OneOf support | - [x] OneOf support | ||||||
|   - [x] Basic support on the wire |   - [x] Basic support on the wire | ||||||
|   - [x] Check which was set from the group |   - [x] Check which was set from the group | ||||||
|   | |||||||
| @@ -3,6 +3,7 @@ import enum | |||||||
| import inspect | import inspect | ||||||
| import json | import json | ||||||
| import struct | import struct | ||||||
|  | import sys | ||||||
| from abc import ABC | from abc import ABC | ||||||
| from base64 import b64encode, b64decode | from base64 import b64encode, b64decode | ||||||
| from datetime import datetime, timedelta, timezone | from datetime import datetime, timedelta, timezone | ||||||
| @@ -10,10 +11,12 @@ from typing import ( | |||||||
|     Any, |     Any, | ||||||
|     AsyncGenerator, |     AsyncGenerator, | ||||||
|     Callable, |     Callable, | ||||||
|  |     Collection, | ||||||
|     Dict, |     Dict, | ||||||
|     Generator, |     Generator, | ||||||
|     Iterable, |     Iterable, | ||||||
|     List, |     List, | ||||||
|  |     Mapping, | ||||||
|     Optional, |     Optional, | ||||||
|     SupportsBytes, |     SupportsBytes, | ||||||
|     Tuple, |     Tuple, | ||||||
| @@ -21,14 +24,27 @@ from typing import ( | |||||||
|     TypeVar, |     TypeVar, | ||||||
|     Union, |     Union, | ||||||
|     get_type_hints, |     get_type_hints, | ||||||
|  |     TYPE_CHECKING, | ||||||
| ) | ) | ||||||
|  |  | ||||||
| import grpclib.client |  | ||||||
| import grpclib.const | import grpclib.const | ||||||
| import stringcase | import stringcase | ||||||
|  |  | ||||||
| from .casing import safe_snake_case | from .casing import safe_snake_case | ||||||
|  |  | ||||||
|  | if TYPE_CHECKING: | ||||||
|  |     from grpclib._protocols import IProtoMessage | ||||||
|  |     from grpclib.client import Channel | ||||||
|  |     from grpclib.metadata import Deadline | ||||||
|  |  | ||||||
|  | if not (sys.version_info.major == 3 and sys.version_info.minor >= 7): | ||||||
|  |     # Apply backport of datetime.fromisoformat from 3.7 | ||||||
|  |     from backports.datetime_fromisoformat import MonkeyPatch | ||||||
|  |  | ||||||
|  |     MonkeyPatch.patch_fromisoformat() | ||||||
|  |  | ||||||
|  |  | ||||||
| # Proto 3 data types | # Proto 3 data types | ||||||
| TYPE_ENUM = "enum" | TYPE_ENUM = "enum" | ||||||
| TYPE_BOOL = "bool" | TYPE_BOOL = "bool" | ||||||
| @@ -106,7 +122,11 @@ WIRE_LEN_DELIM_TYPES = [TYPE_STRING, TYPE_BYTES, TYPE_MESSAGE, TYPE_MAP] | |||||||
|  |  | ||||||
|  |  | ||||||
| # Protobuf datetimes start at the Unix Epoch in 1970 in UTC. | # Protobuf datetimes start at the Unix Epoch in 1970 in UTC. | ||||||
| DATETIME_ZERO = datetime(1970, 1, 1, tzinfo=timezone.utc) | def datetime_default_gen(): | ||||||
|  |     return datetime(1970, 1, 1, tzinfo=timezone.utc) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | DATETIME_ZERO = datetime_default_gen() | ||||||
|  |  | ||||||
|  |  | ||||||
| class Casing(enum.Enum): | class Casing(enum.Enum): | ||||||
| @@ -414,44 +434,101 @@ def parse_fields(value: bytes) -> Generator[ParsedField, None, None]: | |||||||
| T = TypeVar("T", bound="Message") | T = TypeVar("T", bound="Message") | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ProtoClassMetadata: | ||||||
|  |     cls: Type["Message"] | ||||||
|  |  | ||||||
|  |     def __init__(self, cls: Type["Message"]): | ||||||
|  |         self.cls = cls | ||||||
|  |         by_field = {} | ||||||
|  |         by_group = {} | ||||||
|  |  | ||||||
|  |         for field in dataclasses.fields(cls): | ||||||
|  |             meta = FieldMetadata.get(field) | ||||||
|  |  | ||||||
|  |             if meta.group: | ||||||
|  |                 # This is part of a one-of group. | ||||||
|  |                 by_field[field.name] = meta.group | ||||||
|  |  | ||||||
|  |                 by_group.setdefault(meta.group, set()).add(field) | ||||||
|  |  | ||||||
|  |         self.oneof_group_by_field = by_field | ||||||
|  |         self.oneof_field_by_group = by_group | ||||||
|  |  | ||||||
|  |         self.init_default_gen() | ||||||
|  |         self.init_cls_by_field() | ||||||
|  |  | ||||||
|  |     def init_default_gen(self): | ||||||
|  |         default_gen = {} | ||||||
|  |  | ||||||
|  |         for field in dataclasses.fields(self.cls): | ||||||
|  |             meta = FieldMetadata.get(field) | ||||||
|  |             default_gen[field.name] = self.cls._get_field_default_gen(field, meta) | ||||||
|  |  | ||||||
|  |         self.default_gen = default_gen | ||||||
|  |  | ||||||
|  |     def init_cls_by_field(self): | ||||||
|  |         field_cls = {} | ||||||
|  |  | ||||||
|  |         for field in dataclasses.fields(self.cls): | ||||||
|  |             meta = FieldMetadata.get(field) | ||||||
|  |             if meta.proto_type == TYPE_MAP: | ||||||
|  |                 assert meta.map_types | ||||||
|  |                 kt = self.cls._cls_for(field, index=0) | ||||||
|  |                 vt = self.cls._cls_for(field, index=1) | ||||||
|  |                 Entry = dataclasses.make_dataclass( | ||||||
|  |                     "Entry", | ||||||
|  |                     [ | ||||||
|  |                         ("key", kt, dataclass_field(1, meta.map_types[0])), | ||||||
|  |                         ("value", vt, dataclass_field(2, meta.map_types[1])), | ||||||
|  |                     ], | ||||||
|  |                     bases=(Message,), | ||||||
|  |                 ) | ||||||
|  |                 field_cls[field.name] = Entry | ||||||
|  |                 field_cls[field.name + ".value"] = vt | ||||||
|  |             else: | ||||||
|  |                 field_cls[field.name] = self.cls._cls_for(field) | ||||||
|  |  | ||||||
|  |         self.cls_by_field = field_cls | ||||||
|  |  | ||||||
|  |  | ||||||
| class Message(ABC): | class Message(ABC): | ||||||
|     """ |     """ | ||||||
|     A protobuf message base class. Generated code will inherit from this and |     A protobuf message base class. Generated code will inherit from this and | ||||||
|     register the message fields which get used by the serializers and parsers |     register the message fields which get used by the serializers and parsers | ||||||
|     to go between Python, binary and JSON protobuf message representations. |     to go between Python, binary and JSON protobuf message representations. | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     _serialized_on_wire: bool |     _serialized_on_wire: bool | ||||||
|     _unknown_fields: bytes |     _unknown_fields: bytes | ||||||
|     _group_map: Dict[str, dict] |     _group_map: Dict[str, dict] | ||||||
|  |  | ||||||
|     def __post_init__(self) -> None: |     def __post_init__(self) -> None: | ||||||
|  |         # Keep track of whether every field was default | ||||||
|  |         all_sentinel = True | ||||||
|  |  | ||||||
|         # Set a default value for each field in the class after `__init__` has |         # Set a default value for each field in the class after `__init__` has | ||||||
|         # already been run. |         # already been run. | ||||||
|         group_map: Dict[str, dict] = {"fields": {}, "groups": {}} |         group_map: Dict[str, dataclasses.Field] = {} | ||||||
|         for field in dataclasses.fields(self): |         for field in dataclasses.fields(self): | ||||||
|             meta = FieldMetadata.get(field) |             meta = FieldMetadata.get(field) | ||||||
|  |  | ||||||
|             if meta.group: |             if meta.group: | ||||||
|                 # This is part of a one-of group. |                 group_map.setdefault(meta.group) | ||||||
|                 group_map["fields"][field.name] = meta.group |  | ||||||
|  |  | ||||||
|                 if meta.group not in group_map["groups"]: |  | ||||||
|                     group_map["groups"][meta.group] = {"current": None, "fields": set()} |  | ||||||
|                 group_map["groups"][meta.group]["fields"].add(field) |  | ||||||
|  |  | ||||||
|             if getattr(self, field.name) != PLACEHOLDER: |             if getattr(self, field.name) != PLACEHOLDER: | ||||||
|                 # Skip anything not set to the sentinel value |                 # Skip anything not set to the sentinel value | ||||||
|  |                 all_sentinel = False | ||||||
|  |  | ||||||
|                 if meta.group: |                 if meta.group: | ||||||
|                     # This was set, so make it the selected value of the one-of. |                     # This was set, so make it the selected value of the one-of. | ||||||
|                     group_map["groups"][meta.group]["current"] = field |                     group_map[meta.group] = field | ||||||
|  |  | ||||||
|                 continue |                 continue | ||||||
|  |  | ||||||
|             setattr(self, field.name, self._get_field_default(field, meta)) |             setattr(self, field.name, self._get_field_default(field, meta)) | ||||||
|  |  | ||||||
|         # Now that all the defaults are set, reset it! |         # Now that all the defaults are set, reset it! | ||||||
|         self.__dict__["_serialized_on_wire"] = False |         self.__dict__["_serialized_on_wire"] = not all_sentinel | ||||||
|         self.__dict__["_unknown_fields"] = b"" |         self.__dict__["_unknown_fields"] = b"" | ||||||
|         self.__dict__["_group_map"] = group_map |         self.__dict__["_group_map"] = group_map | ||||||
|  |  | ||||||
| @@ -460,19 +537,33 @@ class Message(ABC): | |||||||
|             # Track when a field has been set. |             # Track when a field has been set. | ||||||
|             self.__dict__["_serialized_on_wire"] = True |             self.__dict__["_serialized_on_wire"] = True | ||||||
|  |  | ||||||
|         if attr in getattr(self, "_group_map", {}).get("fields", {}): |         if hasattr(self, "_group_map"):  # __post_init__ had already run | ||||||
|             group = self._group_map["fields"][attr] |             if attr in self._betterproto.oneof_group_by_field: | ||||||
|             for field in self._group_map["groups"][group]["fields"]: |                 group = self._betterproto.oneof_group_by_field[attr] | ||||||
|                 if field.name == attr: |                 for field in self._betterproto.oneof_field_by_group[group]: | ||||||
|                     self._group_map["groups"][group]["current"] = field |                     if field.name == attr: | ||||||
|                 else: |                         self._group_map[group] = field | ||||||
|                     super().__setattr__( |                     else: | ||||||
|                         field.name, |                         super().__setattr__( | ||||||
|                         self._get_field_default(field, FieldMetadata.get(field)), |                             field.name, | ||||||
|                     ) |                             self._get_field_default(field, FieldMetadata.get(field)), | ||||||
|  |                         ) | ||||||
|  |  | ||||||
|         super().__setattr__(attr, value) |         super().__setattr__(attr, value) | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def _betterproto(self): | ||||||
|  |         """ | ||||||
|  |         Lazy initialize metadata for each protobuf class. | ||||||
|  |         It may be initialized multiple times in a multi-threaded environment, | ||||||
|  |         but that won't affect the correctness. | ||||||
|  |         """ | ||||||
|  |         meta = getattr(self.__class__, "_betterproto_meta", None) | ||||||
|  |         if not meta: | ||||||
|  |             meta = ProtoClassMetadata(self.__class__) | ||||||
|  |             self.__class__._betterproto_meta = meta | ||||||
|  |         return meta | ||||||
|  |  | ||||||
|     def __bytes__(self) -> bytes: |     def __bytes__(self) -> bytes: | ||||||
|         """ |         """ | ||||||
|         Get the binary encoded Protobuf representation of this instance. |         Get the binary encoded Protobuf representation of this instance. | ||||||
| @@ -491,7 +582,7 @@ class Message(ABC): | |||||||
|             # currently set in a `oneof` group, so it must be serialized even |             # currently set in a `oneof` group, so it must be serialized even | ||||||
|             # if the value is the default zero value. |             # if the value is the default zero value. | ||||||
|             selected_in_group = False |             selected_in_group = False | ||||||
|             if meta.group and self._group_map["groups"][meta.group]["current"] == field: |             if meta.group and self._group_map[meta.group] == field: | ||||||
|                 selected_in_group = True |                 selected_in_group = True | ||||||
|  |  | ||||||
|             serialize_empty = False |             serialize_empty = False | ||||||
| @@ -543,47 +634,50 @@ class Message(ABC): | |||||||
|     # For compatibility with other libraries |     # For compatibility with other libraries | ||||||
|     SerializeToString = __bytes__ |     SerializeToString = __bytes__ | ||||||
|  |  | ||||||
|     def _type_hint(self, field_name: str) -> Type: |     @classmethod | ||||||
|         module = inspect.getmodule(self.__class__) |     def _type_hint(cls, field_name: str) -> Type: | ||||||
|         type_hints = get_type_hints(self.__class__, vars(module)) |         module = inspect.getmodule(cls) | ||||||
|  |         type_hints = get_type_hints(cls, vars(module)) | ||||||
|         return type_hints[field_name] |         return type_hints[field_name] | ||||||
|  |  | ||||||
|     def _cls_for(self, field: dataclasses.Field, index: int = 0) -> Type: |     @classmethod | ||||||
|  |     def _cls_for(cls, field: dataclasses.Field, index: int = 0) -> Type: | ||||||
|         """Get the message class for a field from the type hints.""" |         """Get the message class for a field from the type hints.""" | ||||||
|         cls = self._type_hint(field.name) |         field_cls = cls._type_hint(field.name) | ||||||
|         if hasattr(cls, "__args__") and index >= 0: |         if hasattr(field_cls, "__args__") and index >= 0: | ||||||
|             cls = cls.__args__[index] |             field_cls = field_cls.__args__[index] | ||||||
|         return cls |         return field_cls | ||||||
|  |  | ||||||
|     def _get_field_default(self, field: dataclasses.Field, meta: FieldMetadata) -> Any: |     def _get_field_default(self, field: dataclasses.Field, meta: FieldMetadata) -> Any: | ||||||
|         t = self._type_hint(field.name) |         return self._betterproto.default_gen[field.name]() | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _get_field_default_gen(cls, field: dataclasses.Field, meta: FieldMetadata) -> Any: | ||||||
|  |         t = cls._type_hint(field.name) | ||||||
|  |  | ||||||
|         value: Any = 0 |  | ||||||
|         if hasattr(t, "__origin__"): |         if hasattr(t, "__origin__"): | ||||||
|             if t.__origin__ == dict: |             if t.__origin__ in (dict, Dict): | ||||||
|                 # This is some kind of map (dict in Python). |                 # This is some kind of map (dict in Python). | ||||||
|                 value = {} |                 return dict | ||||||
|             elif t.__origin__ == list: |             elif t.__origin__ in (list, List): | ||||||
|                 # This is some kind of list (repeated) field. |                 # This is some kind of list (repeated) field. | ||||||
|                 value = [] |                 return list | ||||||
|             elif t.__origin__ == Union and t.__args__[1] == type(None): |             elif t.__origin__ == Union and t.__args__[1] == type(None): | ||||||
|                 # This is an optional (wrapped) field. For setting the default we |                 # This is an optional (wrapped) field. For setting the default we | ||||||
|                 # really don't care what kind of field it is. |                 # really don't care what kind of field it is. | ||||||
|                 value = None |                 return type(None) | ||||||
|             else: |             else: | ||||||
|                 value = t() |                 return t | ||||||
|         elif issubclass(t, Enum): |         elif issubclass(t, Enum): | ||||||
|             # Enums always default to zero. |             # Enums always default to zero. | ||||||
|             value = 0 |             return int | ||||||
|         elif t == datetime: |         elif t == datetime: | ||||||
|             # Offsets are relative to 1970-01-01T00:00:00Z |             # Offsets are relative to 1970-01-01T00:00:00Z | ||||||
|             value = DATETIME_ZERO |             return datetime_default_gen | ||||||
|         else: |         else: | ||||||
|             # This is either a primitive scalar or another message type. Calling |             # This is either a primitive scalar or another message type. Calling | ||||||
|             # it should result in its zero value. |             # it should result in its zero value. | ||||||
|             value = t() |             return t | ||||||
|  |  | ||||||
|         return value |  | ||||||
|  |  | ||||||
|     def _postprocess_single( |     def _postprocess_single( | ||||||
|         self, wire_type: int, meta: FieldMetadata, field: dataclasses.Field, value: Any |         self, wire_type: int, meta: FieldMetadata, field: dataclasses.Field, value: Any | ||||||
| @@ -608,7 +702,7 @@ class Message(ABC): | |||||||
|             if meta.proto_type == TYPE_STRING: |             if meta.proto_type == TYPE_STRING: | ||||||
|                 value = value.decode("utf-8") |                 value = value.decode("utf-8") | ||||||
|             elif meta.proto_type == TYPE_MESSAGE: |             elif meta.proto_type == TYPE_MESSAGE: | ||||||
|                 cls = self._cls_for(field) |                 cls = self._betterproto.cls_by_field[field.name] | ||||||
|  |  | ||||||
|                 if cls == datetime: |                 if cls == datetime: | ||||||
|                     value = _Timestamp().parse(value).to_datetime() |                     value = _Timestamp().parse(value).to_datetime() | ||||||
| @@ -622,20 +716,7 @@ class Message(ABC): | |||||||
|                     value = cls().parse(value) |                     value = cls().parse(value) | ||||||
|                     value._serialized_on_wire = True |                     value._serialized_on_wire = True | ||||||
|             elif meta.proto_type == TYPE_MAP: |             elif meta.proto_type == TYPE_MAP: | ||||||
|                 # TODO: This is slow, use a cache to make it faster since each |                 value = self._betterproto.cls_by_field[field.name]().parse(value) | ||||||
|                 #       key/value pair will recreate the class. |  | ||||||
|                 assert meta.map_types |  | ||||||
|                 kt = self._cls_for(field, index=0) |  | ||||||
|                 vt = self._cls_for(field, index=1) |  | ||||||
|                 Entry = dataclasses.make_dataclass( |  | ||||||
|                     "Entry", |  | ||||||
|                     [ |  | ||||||
|                         ("key", kt, dataclass_field(1, meta.map_types[0])), |  | ||||||
|                         ("value", vt, dataclass_field(2, meta.map_types[1])), |  | ||||||
|                     ], |  | ||||||
|                     bases=(Message,), |  | ||||||
|                 ) |  | ||||||
|                 value = Entry().parse(value) |  | ||||||
|  |  | ||||||
|         return value |         return value | ||||||
|  |  | ||||||
| @@ -695,41 +776,50 @@ class Message(ABC): | |||||||
|     def FromString(cls: Type[T], data: bytes) -> T: |     def FromString(cls: Type[T], data: bytes) -> T: | ||||||
|         return cls().parse(data) |         return cls().parse(data) | ||||||
|  |  | ||||||
|     def to_dict(self, casing: Casing = Casing.CAMEL) -> dict: |     def to_dict( | ||||||
|  |         self, casing: Casing = Casing.CAMEL, include_default_values: bool = False | ||||||
|  |     ) -> dict: | ||||||
|         """ |         """ | ||||||
|         Returns a dict representation of this message instance which can be |         Returns a dict representation of this message instance which can be | ||||||
|         used to serialize to e.g. JSON. Defaults to camel casing for |         used to serialize to e.g. JSON. Defaults to camel casing for | ||||||
|         compatibility but can be set to other modes. |         compatibility but can be set to other modes. | ||||||
|  |  | ||||||
|  |         `include_default_values` can be set to `True` to include default | ||||||
|  |         values of fields. E.g. an `int32` type field with `0` value will | ||||||
|  |         not be in returned dict if `include_default_values` is set to | ||||||
|  |         `False`. | ||||||
|         """ |         """ | ||||||
|         output: Dict[str, Any] = {} |         output: Dict[str, Any] = {} | ||||||
|         for field in dataclasses.fields(self): |         for field in dataclasses.fields(self): | ||||||
|             meta = FieldMetadata.get(field) |             meta = FieldMetadata.get(field) | ||||||
|             v = getattr(self, field.name) |             v = getattr(self, field.name) | ||||||
|             cased_name = casing(field.name).rstrip("_") # type: ignore |             cased_name = casing(field.name).rstrip("_")  # type: ignore | ||||||
|             if meta.proto_type == "message": |             if meta.proto_type == "message": | ||||||
|                 if isinstance(v, datetime): |                 if isinstance(v, datetime): | ||||||
|                     if v != DATETIME_ZERO: |                     if v != DATETIME_ZERO or include_default_values: | ||||||
|                         output[cased_name] = _Timestamp.timestamp_to_json(v) |                         output[cased_name] = _Timestamp.timestamp_to_json(v) | ||||||
|                 elif isinstance(v, timedelta): |                 elif isinstance(v, timedelta): | ||||||
|                     if v != timedelta(0): |                     if v != timedelta(0) or include_default_values: | ||||||
|                         output[cased_name] = _Duration.delta_to_json(v) |                         output[cased_name] = _Duration.delta_to_json(v) | ||||||
|                 elif meta.wraps: |                 elif meta.wraps: | ||||||
|                     if v is not None: |                     if v is not None or include_default_values: | ||||||
|                         output[cased_name] = v |                         output[cased_name] = v | ||||||
|                 elif isinstance(v, list): |                 elif isinstance(v, list): | ||||||
|                     # Convert each item. |                     # Convert each item. | ||||||
|                     v = [i.to_dict() for i in v] |                     v = [i.to_dict(casing, include_default_values) for i in v] | ||||||
|                     output[cased_name] = v |                     if v or include_default_values: | ||||||
|                 elif v._serialized_on_wire: |                         output[cased_name] = v | ||||||
|                     output[cased_name] = v.to_dict() |                 else: | ||||||
|  |                     if v._serialized_on_wire or include_default_values: | ||||||
|  |                         output[cased_name] = v.to_dict(casing, include_default_values) | ||||||
|             elif meta.proto_type == "map": |             elif meta.proto_type == "map": | ||||||
|                 for k in v: |                 for k in v: | ||||||
|                     if hasattr(v[k], "to_dict"): |                     if hasattr(v[k], "to_dict"): | ||||||
|                         v[k] = v[k].to_dict() |                         v[k] = v[k].to_dict(casing, include_default_values) | ||||||
|  |  | ||||||
|                 if v: |                 if v or include_default_values: | ||||||
|                     output[cased_name] = v |                     output[cased_name] = v | ||||||
|             elif v != self._get_field_default(field, meta): |             elif v != self._get_field_default(field, meta) or include_default_values: | ||||||
|                 if meta.proto_type in INT_64_TYPES: |                 if meta.proto_type in INT_64_TYPES: | ||||||
|                     if isinstance(v, list): |                     if isinstance(v, list): | ||||||
|                         output[cased_name] = [str(n) for n in v] |                         output[cased_name] = [str(n) for n in v] | ||||||
| @@ -741,7 +831,7 @@ class Message(ABC): | |||||||
|                     else: |                     else: | ||||||
|                         output[cased_name] = b64encode(v).decode("utf8") |                         output[cased_name] = b64encode(v).decode("utf8") | ||||||
|                 elif meta.proto_type == TYPE_ENUM: |                 elif meta.proto_type == TYPE_ENUM: | ||||||
|                     enum_values = list(self._cls_for(field)) # type: ignore |                     enum_values = list(self._betterproto.cls_by_field[field.name])  # type: ignore | ||||||
|                     if isinstance(v, list): |                     if isinstance(v, list): | ||||||
|                         output[cased_name] = [enum_values[e].name for e in v] |                         output[cased_name] = [enum_values[e].name for e in v] | ||||||
|                     else: |                     else: | ||||||
| @@ -767,7 +857,7 @@ class Message(ABC): | |||||||
|                     if meta.proto_type == "message": |                     if meta.proto_type == "message": | ||||||
|                         v = getattr(self, field.name) |                         v = getattr(self, field.name) | ||||||
|                         if isinstance(v, list): |                         if isinstance(v, list): | ||||||
|                             cls = self._cls_for(field) |                             cls = self._betterproto.cls_by_field[field.name] | ||||||
|                             for i in range(len(value[key])): |                             for i in range(len(value[key])): | ||||||
|                                 v.append(cls().from_dict(value[key][i])) |                                 v.append(cls().from_dict(value[key][i])) | ||||||
|                         elif isinstance(v, datetime): |                         elif isinstance(v, datetime): | ||||||
| @@ -784,7 +874,7 @@ class Message(ABC): | |||||||
|                             v.from_dict(value[key]) |                             v.from_dict(value[key]) | ||||||
|                     elif meta.map_types and meta.map_types[1] == TYPE_MESSAGE: |                     elif meta.map_types and meta.map_types[1] == TYPE_MESSAGE: | ||||||
|                         v = getattr(self, field.name) |                         v = getattr(self, field.name) | ||||||
|                         cls = self._cls_for(field, index=1) |                         cls = self._betterproto.cls_by_field[field.name + ".value"] | ||||||
|                         for k in value[key]: |                         for k in value[key]: | ||||||
|                             v[k] = cls().from_dict(value[key][k]) |                             v[k] = cls().from_dict(value[key][k]) | ||||||
|                     else: |                     else: | ||||||
| @@ -800,7 +890,7 @@ class Message(ABC): | |||||||
|                             else: |                             else: | ||||||
|                                 v = b64decode(value[key]) |                                 v = b64decode(value[key]) | ||||||
|                         elif meta.proto_type == TYPE_ENUM: |                         elif meta.proto_type == TYPE_ENUM: | ||||||
|                             enum_cls = self._cls_for(field) |                             enum_cls = self._betterproto.cls_by_field[field.name] | ||||||
|                             if isinstance(v, list): |                             if isinstance(v, list): | ||||||
|                                 v = [enum_cls.from_string(e) for e in v] |                                 v = [enum_cls.from_string(e) for e in v] | ||||||
|                             elif isinstance(v, str): |                             elif isinstance(v, str): | ||||||
| @@ -833,7 +923,7 @@ def serialized_on_wire(message: Message) -> bool: | |||||||
|  |  | ||||||
| def which_one_of(message: Message, group_name: str) -> Tuple[str, Any]: | def which_one_of(message: Message, group_name: str) -> Tuple[str, Any]: | ||||||
|     """Return the name and value of a message's one-of field group.""" |     """Return the name and value of a message's one-of field group.""" | ||||||
|     field = message._group_map["groups"].get(group_name, {}).get("current") |     field = message._group_map.get(group_name) | ||||||
|     if not field: |     if not field: | ||||||
|         return ("", None) |         return ("", None) | ||||||
|     return (field.name, getattr(message, field.name)) |     return (field.name, getattr(message, field.name)) | ||||||
| @@ -902,6 +992,7 @@ class _WrappedMessage(Message): | |||||||
|     Google protobuf wrapper types base class. JSON representation is just the |     Google protobuf wrapper types base class. JSON representation is just the | ||||||
|     value itself. |     value itself. | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     value: Any |     value: Any | ||||||
|  |  | ||||||
|     def to_dict(self, casing: Casing = Casing.CAMEL) -> Any: |     def to_dict(self, casing: Casing = Casing.CAMEL) -> Any: | ||||||
| @@ -973,20 +1064,57 @@ def _get_wrapper(proto_type: str) -> Type: | |||||||
|     }[proto_type] |     }[proto_type] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | _Value = Union[str, bytes] | ||||||
|  | _MetadataLike = Union[Mapping[str, _Value], Collection[Tuple[str, _Value]]] | ||||||
|  |  | ||||||
|  |  | ||||||
| class ServiceStub(ABC): | class ServiceStub(ABC): | ||||||
|     """ |     """ | ||||||
|     Base class for async gRPC service stubs. |     Base class for async gRPC service stubs. | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     def __init__(self, channel: grpclib.client.Channel) -> None: |     def __init__( | ||||||
|  |         self, | ||||||
|  |         channel: "Channel", | ||||||
|  |         *, | ||||||
|  |         timeout: Optional[float] = None, | ||||||
|  |         deadline: Optional["Deadline"] = None, | ||||||
|  |         metadata: Optional[_MetadataLike] = None, | ||||||
|  |     ) -> None: | ||||||
|         self.channel = channel |         self.channel = channel | ||||||
|  |         self.timeout = timeout | ||||||
|  |         self.deadline = deadline | ||||||
|  |         self.metadata = metadata | ||||||
|  |  | ||||||
|  |     def __resolve_request_kwargs( | ||||||
|  |         self, | ||||||
|  |         timeout: Optional[float], | ||||||
|  |         deadline: Optional["Deadline"], | ||||||
|  |         metadata: Optional[_MetadataLike], | ||||||
|  |     ): | ||||||
|  |         return { | ||||||
|  |             "timeout": self.timeout if timeout is None else timeout, | ||||||
|  |             "deadline": self.deadline if deadline is None else deadline, | ||||||
|  |             "metadata": self.metadata if metadata is None else metadata, | ||||||
|  |         } | ||||||
|  |  | ||||||
|     async def _unary_unary( |     async def _unary_unary( | ||||||
|         self, route: str, request_type: Type, response_type: Type[T], request: Any |         self, | ||||||
|  |         route: str, | ||||||
|  |         request: "IProtoMessage", | ||||||
|  |         response_type: Type[T], | ||||||
|  |         *, | ||||||
|  |         timeout: Optional[float] = None, | ||||||
|  |         deadline: Optional["Deadline"] = None, | ||||||
|  |         metadata: Optional[_MetadataLike] = None, | ||||||
|     ) -> T: |     ) -> T: | ||||||
|         """Make a unary request and return the response.""" |         """Make a unary request and return the response.""" | ||||||
|         async with self.channel.request( |         async with self.channel.request( | ||||||
|             route, grpclib.const.Cardinality.UNARY_UNARY, request_type, response_type |             route, | ||||||
|  |             grpclib.const.Cardinality.UNARY_UNARY, | ||||||
|  |             type(request), | ||||||
|  |             response_type, | ||||||
|  |             **self.__resolve_request_kwargs(timeout, deadline, metadata), | ||||||
|         ) as stream: |         ) as stream: | ||||||
|             await stream.send_message(request, end=True) |             await stream.send_message(request, end=True) | ||||||
|             response = await stream.recv_message() |             response = await stream.recv_message() | ||||||
| @@ -994,11 +1122,22 @@ class ServiceStub(ABC): | |||||||
|             return response |             return response | ||||||
|  |  | ||||||
|     async def _unary_stream( |     async def _unary_stream( | ||||||
|         self, route: str, request_type: Type, response_type: Type[T], request: Any |         self, | ||||||
|  |         route: str, | ||||||
|  |         request: "IProtoMessage", | ||||||
|  |         response_type: Type[T], | ||||||
|  |         *, | ||||||
|  |         timeout: Optional[float] = None, | ||||||
|  |         deadline: Optional["Deadline"] = None, | ||||||
|  |         metadata: Optional[_MetadataLike] = None, | ||||||
|     ) -> AsyncGenerator[T, None]: |     ) -> AsyncGenerator[T, None]: | ||||||
|         """Make a unary request and return the stream response iterator.""" |         """Make a unary request and return the stream response iterator.""" | ||||||
|         async with self.channel.request( |         async with self.channel.request( | ||||||
|             route, grpclib.const.Cardinality.UNARY_STREAM, request_type, response_type |             route, | ||||||
|  |             grpclib.const.Cardinality.UNARY_STREAM, | ||||||
|  |             type(request), | ||||||
|  |             response_type, | ||||||
|  |             **self.__resolve_request_kwargs(timeout, deadline, metadata), | ||||||
|         ) as stream: |         ) as stream: | ||||||
|             await stream.send_message(request, end=True) |             await stream.send_message(request, end=True) | ||||||
|             async for message in stream: |             async for message in stream: | ||||||
|   | |||||||
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										2
									
								
								betterproto/plugin.bat
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								betterproto/plugin.bat
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | |||||||
|  | @SET plugin_dir=%~dp0 | ||||||
|  | @python %plugin_dir%/plugin.py %* | ||||||
| @@ -1,21 +1,21 @@ | |||||||
| #!/usr/bin/env python | #!/usr/bin/env python | ||||||
|  |  | ||||||
| import itertools | import itertools | ||||||
| import json |  | ||||||
| import os.path | import os.path | ||||||
| import re |  | ||||||
| import sys | import sys | ||||||
| import textwrap | import textwrap | ||||||
| from typing import Any, List, Tuple | from collections import defaultdict | ||||||
|  | from typing import Dict, List, Optional, Type | ||||||
|  |  | ||||||
| try: | try: | ||||||
|     import jinja2 |     import black | ||||||
| except ImportError: | except ImportError: | ||||||
|     print( |     print( | ||||||
|         "Unable to import `jinja2`. Did you install the compiler feature with `pip install betterproto[compiler]`?" |         "Unable to import `black` formatter. Did you install the compiler feature with `pip install betterproto[compiler]`?" | ||||||
|     ) |     ) | ||||||
|     raise SystemExit(1) |     raise SystemExit(1) | ||||||
|  |  | ||||||
|  | import jinja2 | ||||||
| import stringcase | import stringcase | ||||||
|  |  | ||||||
| from google.protobuf.compiler import plugin_pb2 as plugin | from google.protobuf.compiler import plugin_pb2 as plugin | ||||||
| @@ -23,44 +23,51 @@ from google.protobuf.descriptor_pb2 import ( | |||||||
|     DescriptorProto, |     DescriptorProto, | ||||||
|     EnumDescriptorProto, |     EnumDescriptorProto, | ||||||
|     FieldDescriptorProto, |     FieldDescriptorProto, | ||||||
|     FileDescriptorProto, |  | ||||||
|     ServiceDescriptorProto, |  | ||||||
| ) | ) | ||||||
|  |  | ||||||
| from betterproto.casing import safe_snake_case | from betterproto.casing import safe_snake_case | ||||||
|  |  | ||||||
|  | import google.protobuf.wrappers_pb2 as google_wrappers | ||||||
|  |  | ||||||
| WRAPPER_TYPES = { | WRAPPER_TYPES: Dict[str, Optional[Type]] = defaultdict(lambda: None, { | ||||||
|     "google.protobuf.DoubleValue": "float", |     'google.protobuf.DoubleValue': google_wrappers.DoubleValue, | ||||||
|     "google.protobuf.FloatValue": "float", |     'google.protobuf.FloatValue': google_wrappers.FloatValue, | ||||||
|     "google.protobuf.Int64Value": "int", |     'google.protobuf.Int64Value': google_wrappers.Int64Value, | ||||||
|     "google.protobuf.UInt64Value": "int", |     'google.protobuf.UInt64Value': google_wrappers.UInt64Value, | ||||||
|     "google.protobuf.Int32Value": "int", |     'google.protobuf.Int32Value': google_wrappers.Int32Value, | ||||||
|     "google.protobuf.UInt32Value": "int", |     'google.protobuf.UInt32Value': google_wrappers.UInt32Value, | ||||||
|     "google.protobuf.BoolValue": "bool", |     'google.protobuf.BoolValue': google_wrappers.BoolValue, | ||||||
|     "google.protobuf.StringValue": "str", |     'google.protobuf.StringValue': google_wrappers.StringValue, | ||||||
|     "google.protobuf.BytesValue": "bytes", |     'google.protobuf.BytesValue': google_wrappers.BytesValue, | ||||||
| } | }) | ||||||
|  |  | ||||||
|  |  | ||||||
| def get_ref_type(package: str, imports: set, type_name: str) -> str: | def get_ref_type(package: str, imports: set, type_name: str, unwrap: bool = True) -> str: | ||||||
|     """ |     """ | ||||||
|     Return a Python type name for a proto type reference. Adds the import if |     Return a Python type name for a proto type reference. Adds the import if | ||||||
|     necessary. |     necessary. Unwraps well known type if required. | ||||||
|     """ |     """ | ||||||
|     # If the package name is a blank string, then this should still work |     # If the package name is a blank string, then this should still work | ||||||
|     # because by convention packages are lowercase and message/enum types are |     # because by convention packages are lowercase and message/enum types are | ||||||
|     # pascal-cased. May require refactoring in the future. |     # pascal-cased. May require refactoring in the future. | ||||||
|     type_name = type_name.lstrip(".") |     type_name = type_name.lstrip(".") | ||||||
|  |  | ||||||
|     if type_name in WRAPPER_TYPES: |     # Check if type is wrapper. | ||||||
|         return f"Optional[{WRAPPER_TYPES[type_name]}]" |     wrapper_class = WRAPPER_TYPES[type_name] | ||||||
|  |  | ||||||
|     if type_name == "google.protobuf.Duration": |     if unwrap: | ||||||
|         return "timedelta" |         if wrapper_class: | ||||||
|  |             wrapped_type = type(wrapper_class().value) | ||||||
|  |             return f"Optional[{wrapped_type.__name__}]" | ||||||
|  |  | ||||||
|     if type_name == "google.protobuf.Timestamp": |         if type_name == "google.protobuf.Duration": | ||||||
|         return "datetime" |             return "timedelta" | ||||||
|  |  | ||||||
|  |         if type_name == "google.protobuf.Timestamp": | ||||||
|  |             return "datetime" | ||||||
|  |     elif wrapper_class: | ||||||
|  |         imports.add(f"from {wrapper_class.__module__} import {wrapper_class.__name__}") | ||||||
|  |         return f"{wrapper_class.__name__}" | ||||||
|  |  | ||||||
|     if type_name.startswith(package): |     if type_name.startswith(package): | ||||||
|         parts = type_name.lstrip(package).lstrip(".").split(".") |         parts = type_name.lstrip(package).lstrip(".").split(".") | ||||||
| @@ -121,19 +128,19 @@ def get_py_zero(type_num: int) -> str: | |||||||
|  |  | ||||||
|  |  | ||||||
| def traverse(proto_file): | def traverse(proto_file): | ||||||
|     def _traverse(path, items): |     def _traverse(path, items, prefix=""): | ||||||
|         for i, item in enumerate(items): |         for i, item in enumerate(items): | ||||||
|  |             # Adjust the name since we flatten the heirarchy. | ||||||
|  |             item.name = next_prefix = prefix + item.name | ||||||
|             yield item, path + [i] |             yield item, path + [i] | ||||||
|  |  | ||||||
|             if isinstance(item, DescriptorProto): |             if isinstance(item, DescriptorProto): | ||||||
|                 for enum in item.enum_type: |                 for enum in item.enum_type: | ||||||
|                     enum.name = item.name + enum.name |                     enum.name = next_prefix + enum.name | ||||||
|                     yield enum, path + [i, 4] |                     yield enum, path + [i, 4] | ||||||
|  |  | ||||||
|                 if item.nested_type: |                 if item.nested_type: | ||||||
|                     for n, p in _traverse(path + [i, 3], item.nested_type): |                     for n, p in _traverse(path + [i, 3], item.nested_type, next_prefix): | ||||||
|                         # Adjust the name since we flatten the heirarchy. |  | ||||||
|                         n.name = item.name + n.name |  | ||||||
|                         yield n, p |                         yield n, p | ||||||
|  |  | ||||||
|     return itertools.chain( |     return itertools.chain( | ||||||
| @@ -141,25 +148,26 @@ def traverse(proto_file): | |||||||
|     ) |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
| def get_comment(proto_file, path: List[int]) -> str: | def get_comment(proto_file, path: List[int], indent: int = 4) -> str: | ||||||
|  |     pad = " " * indent | ||||||
|     for sci in proto_file.source_code_info.location: |     for sci in proto_file.source_code_info.location: | ||||||
|         # print(list(sci.path), path, file=sys.stderr) |         # print(list(sci.path), path, file=sys.stderr) | ||||||
|         if list(sci.path) == path and sci.leading_comments: |         if list(sci.path) == path and sci.leading_comments: | ||||||
|             lines = textwrap.wrap( |             lines = textwrap.wrap( | ||||||
|                 sci.leading_comments.strip().replace("\n", ""), width=75 |                 sci.leading_comments.strip().replace("\n", ""), width=79 - indent | ||||||
|             ) |             ) | ||||||
|  |  | ||||||
|             if path[-2] == 2 and path[-4] != 6: |             if path[-2] == 2 and path[-4] != 6: | ||||||
|                 # This is a field |                 # This is a field | ||||||
|                 return "    # " + "\n    # ".join(lines) |                 return f"{pad}# " + f"\n{pad}# ".join(lines) | ||||||
|             else: |             else: | ||||||
|                 # This is a message, enum, service, or method |                 # This is a message, enum, service, or method | ||||||
|                 if len(lines) == 1 and len(lines[0]) < 70: |                 if len(lines) == 1 and len(lines[0]) < 79 - indent - 6: | ||||||
|                     lines[0] = lines[0].strip('"') |                     lines[0] = lines[0].strip('"') | ||||||
|                     return f'    """{lines[0]}"""' |                     return f'{pad}"""{lines[0]}"""' | ||||||
|                 else: |                 else: | ||||||
|                     joined = "\n    ".join(lines) |                     joined = f"\n{pad}".join(lines) | ||||||
|                     return f'    """\n    {joined}\n    """' |                     return f'{pad}"""\n{pad}{joined}\n{pad}"""' | ||||||
|  |  | ||||||
|     return "" |     return "" | ||||||
|  |  | ||||||
| @@ -370,14 +378,14 @@ def generate_code(request, response): | |||||||
|                         { |                         { | ||||||
|                             "name": method.name, |                             "name": method.name, | ||||||
|                             "py_name": stringcase.snakecase(method.name), |                             "py_name": stringcase.snakecase(method.name), | ||||||
|                             "comment": get_comment(proto_file, [6, i, 2, j]), |                             "comment": get_comment(proto_file, [6, i, 2, j], indent=8), | ||||||
|                             "route": f"/{package}.{service.name}/{method.name}", |                             "route": f"/{package}.{service.name}/{method.name}", | ||||||
|                             "input": get_ref_type( |                             "input": get_ref_type( | ||||||
|                                 package, output["imports"], method.input_type |                                 package, output["imports"], method.input_type | ||||||
|                             ).strip('"'), |                             ).strip('"'), | ||||||
|                             "input_message": input_message, |                             "input_message": input_message, | ||||||
|                             "output": get_ref_type( |                             "output": get_ref_type( | ||||||
|                                 package, output["imports"], method.output_type |                                 package, output["imports"], method.output_type, unwrap=False | ||||||
|                             ).strip('"'), |                             ).strip('"'), | ||||||
|                             "client_streaming": method.client_streaming, |                             "client_streaming": method.client_streaming, | ||||||
|                             "server_streaming": method.server_streaming, |                             "server_streaming": method.server_streaming, | ||||||
| @@ -398,8 +406,11 @@ def generate_code(request, response): | |||||||
|         # print(filename, file=sys.stderr) |         # print(filename, file=sys.stderr) | ||||||
|         f.name = filename.replace(".", os.path.sep) + ".py" |         f.name = filename.replace(".", os.path.sep) + ".py" | ||||||
|  |  | ||||||
|         # f.content = json.dumps(output, indent=2) |         # Render and then format the output file. | ||||||
|         f.content = template.render(description=output).rstrip("\n") + "\n" |         f.content = black.format_str( | ||||||
|  |             template.render(description=output), | ||||||
|  |             mode=black.FileMode(target_versions=set([black.TargetVersion.PY37])), | ||||||
|  |         ) | ||||||
|  |  | ||||||
|     inits = set([""]) |     inits = set([""]) | ||||||
|     for f in response.file: |     for f in response.file: | ||||||
|   | |||||||
| @@ -15,8 +15,8 @@ import betterproto | |||||||
| {% if description.services %} | {% if description.services %} | ||||||
| import grpclib | import grpclib | ||||||
| {% endif %} | {% endif %} | ||||||
| {% for i in description.imports %} |  | ||||||
|  |  | ||||||
|  | {% for i in description.imports %} | ||||||
| {{ i }} | {{ i }} | ||||||
| {% endfor %} | {% endfor %} | ||||||
|  |  | ||||||
| @@ -63,7 +63,7 @@ class {{ service.py_name }}Stub(betterproto.ServiceStub): | |||||||
|  |  | ||||||
|     {% endif %} |     {% endif %} | ||||||
|     {% for method in service.methods %} |     {% for method in service.methods %} | ||||||
|     async def {{ method.py_name }}(self{% if method.input_message and method.input_message.properties %}, *, {% for field in method.input_message.properties %}{{ field.name }}: {% if field.zero == "None" and not field.type.startswith("Optional[") %}Optional[{{ field.type }}]{% else %}{{ field.type }}{% endif %} = {{ field.zero }}{% if not loop.last %}, {% endif %}{% endfor %}{% endif %}) -> {% if method.server_streaming %}AsyncGenerator[{{ method.output }}, None]{% else %}{{ method.output }}{% endif %}: |     async def {{ method.py_name }}(self{% if method.input_message and method.input_message.properties %}, *, {% for field in method.input_message.properties %}{{ field.py_name }}: {% if field.zero == "None" and not field.type.startswith("Optional[") %}Optional[{{ field.type }}]{% else %}{{ field.type }}{% endif %} = {{ field.zero }}{% if not loop.last %}, {% endif %}{% endfor %}{% endif %}) -> {% if method.server_streaming %}AsyncGenerator[{{ method.output }}, None]{% else %}{{ method.output }}{% endif %}: | ||||||
|         {% if method.comment %} |         {% if method.comment %} | ||||||
| {{ method.comment }} | {{ method.comment }} | ||||||
|  |  | ||||||
| @@ -71,27 +71,25 @@ class {{ service.py_name }}Stub(betterproto.ServiceStub): | |||||||
|         request = {{ method.input }}() |         request = {{ method.input }}() | ||||||
|         {% for field in method.input_message.properties %} |         {% for field in method.input_message.properties %} | ||||||
|             {% if field.field_type == 'message' %} |             {% if field.field_type == 'message' %} | ||||||
|         if {{ field.name }} is not None: |         if {{ field.py_name }} is not None: | ||||||
|             request.{{ field.name }} = {{ field.name }} |             request.{{ field.py_name }} = {{ field.py_name }} | ||||||
|             {% else %} |             {% else %} | ||||||
|         request.{{ field.name }} = {{ field.name }} |         request.{{ field.py_name }} = {{ field.py_name }} | ||||||
|             {% endif %} |             {% endif %} | ||||||
|         {% endfor %} |         {% endfor %} | ||||||
|  |  | ||||||
|         {% if method.server_streaming %} |         {% if method.server_streaming %} | ||||||
|         async for response in self._unary_stream( |         async for response in self._unary_stream( | ||||||
|             "{{ method.route }}", |             "{{ method.route }}", | ||||||
|             {{ method.input }}, |  | ||||||
|             {{ method.output }}, |  | ||||||
|             request, |             request, | ||||||
|  |             {{ method.output }}, | ||||||
|         ): |         ): | ||||||
|             yield response |             yield response | ||||||
|         {% else %} |         {% else %} | ||||||
|         return await self._unary_unary( |         return await self._unary_unary( | ||||||
|             "{{ method.route }}", |             "{{ method.route }}", | ||||||
|             {{ method.input }}, |  | ||||||
|             {{ method.output }}, |  | ||||||
|             request, |             request, | ||||||
|  |             {{ method.output }}, | ||||||
|         ) |         ) | ||||||
|         {% endif %} |         {% endif %} | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										90
									
								
								betterproto/tests/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										90
									
								
								betterproto/tests/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,90 @@ | |||||||
|  | # Standard Tests Development Guide | ||||||
|  |  | ||||||
|  | Standard test cases are found in [betterproto/tests/inputs](inputs), where each subdirectory represents a testcase, that is verified in isolation. | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  | inputs/ | ||||||
|  |    bool/ | ||||||
|  |    double/ | ||||||
|  |    int32/ | ||||||
|  |    ... | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## Test case directory structure | ||||||
|  |  | ||||||
|  | Each testcase has a `<name>.proto` file with a message called `Test`, a matching `.json` file and optionally a custom test file called `test_*.py`. | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | bool/ | ||||||
|  |   bool.proto | ||||||
|  |   bool.json | ||||||
|  |   test_bool.py  # optional | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### proto | ||||||
|  |  | ||||||
|  | `<name>.proto` — *The protobuf message to test* | ||||||
|  |  | ||||||
|  | ```protobuf | ||||||
|  | syntax = "proto3"; | ||||||
|  |  | ||||||
|  | message Test { | ||||||
|  |     bool value = 1; | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | You can add multiple `.proto` files to the test case, as long as one file matches the directory name.  | ||||||
|  |  | ||||||
|  | ### json | ||||||
|  |  | ||||||
|  | `<name>.json` — *Test-data to validate the message with* | ||||||
|  |  | ||||||
|  | ```json | ||||||
|  | { | ||||||
|  |   "value": true | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### pytest | ||||||
|  |  | ||||||
|  | `test_<name>.py` — *Custom test to validate specific aspects of the generated class* | ||||||
|  |  | ||||||
|  | ```python | ||||||
|  | from betterproto.tests.output_betterproto.bool.bool import Test | ||||||
|  |  | ||||||
|  | def test_value(): | ||||||
|  |     message = Test() | ||||||
|  |     assert not message.value, "Boolean is False by default" | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## Standard tests | ||||||
|  |  | ||||||
|  | The following tests are automatically executed for all cases: | ||||||
|  |  | ||||||
|  | - [x] Can the generated python code imported? | ||||||
|  | - [x] Can the generated message class be instantiated? | ||||||
|  | - [x] Is the generated code compatible with the Google's `grpc_tools.protoc` implementation? | ||||||
|  |  | ||||||
|  | ## Running the tests | ||||||
|  |  | ||||||
|  | - `pipenv run generate` | ||||||
|  |   This generates | ||||||
|  |   - `betterproto/tests/output_betterproto` — *the plugin generated python classes* | ||||||
|  |   - `betterproto/tests/output_reference` — *reference implementation classes* | ||||||
|  | - `pipenv run test` | ||||||
|  |  | ||||||
|  | ## Intentionally Failing tests | ||||||
|  |  | ||||||
|  | The standard test suite includes tests that fail by intention. These tests document known bugs and missing features that are intended to be corrented in the future. | ||||||
|  |  | ||||||
|  | When running `pytest`, they show up as `x` or  `X` in the test results. | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  | betterproto/tests/test_inputs.py ..x...x..x...x.X........xx........x.....x.......x.xx....x...................... [ 84%] | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | - `.` — PASSED | ||||||
|  | - `x` — XFAIL: expected failure | ||||||
|  | - `X` — XPASS: expected failure, but still passed | ||||||
|  |  | ||||||
|  | Test cases marked for expected failure are declared in [inputs/xfail.py](inputs.xfail.py) | ||||||
| @@ -1,84 +1,91 @@ | |||||||
| #!/usr/bin/env python | #!/usr/bin/env python | ||||||
|  | import glob | ||||||
| import os | import os | ||||||
|  | import shutil | ||||||
|  | import sys | ||||||
|  | from typing import Set | ||||||
|  |  | ||||||
|  | from betterproto.tests.util import ( | ||||||
|  |     get_directories, | ||||||
|  |     inputs_path, | ||||||
|  |     output_path_betterproto, | ||||||
|  |     output_path_reference, | ||||||
|  |     protoc_plugin, | ||||||
|  |     protoc_reference, | ||||||
|  | ) | ||||||
|  |  | ||||||
| # Force pure-python implementation instead of C++, otherwise imports | # Force pure-python implementation instead of C++, otherwise imports | ||||||
| # break things because we can't properly reset the symbol database. | # break things because we can't properly reset the symbol database. | ||||||
| os.environ["PROTOCOL_BUFFERS_PYTHON_IMPLEMENTATION"] = "python" | os.environ["PROTOCOL_BUFFERS_PYTHON_IMPLEMENTATION"] = "python" | ||||||
|  |  | ||||||
| import importlib |  | ||||||
| import json |  | ||||||
| import subprocess |  | ||||||
| import sys |  | ||||||
| from typing import Generator, Tuple |  | ||||||
|  |  | ||||||
| from google.protobuf import symbol_database | def clear_directory(path: str): | ||||||
| from google.protobuf.descriptor_pool import DescriptorPool |     for file_or_directory in glob.glob(os.path.join(path, "*")): | ||||||
| from google.protobuf.json_format import MessageToJson, Parse |         if os.path.isdir(file_or_directory): | ||||||
|  |             shutil.rmtree(file_or_directory) | ||||||
|  |         else: | ||||||
|  |             os.remove(file_or_directory) | ||||||
|  |  | ||||||
|  |  | ||||||
| root = os.path.dirname(os.path.realpath(__file__)) | def generate(whitelist: Set[str]): | ||||||
|  |     path_whitelist = {os.path.realpath(e) for e in whitelist if os.path.exists(e)} | ||||||
|  |     name_whitelist = {e for e in whitelist if not os.path.exists(e)} | ||||||
|  |  | ||||||
|  |     test_case_names = set(get_directories(inputs_path)) | ||||||
|  |  | ||||||
|  |     for test_case_name in sorted(test_case_names): | ||||||
|  |         test_case_input_path = os.path.realpath( | ||||||
|  |             os.path.join(inputs_path, test_case_name) | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         if ( | ||||||
|  |             whitelist | ||||||
|  |             and test_case_input_path not in path_whitelist | ||||||
|  |             and test_case_name not in name_whitelist | ||||||
|  |         ): | ||||||
|  |             continue | ||||||
|  |  | ||||||
|  |         test_case_output_path_reference = os.path.join( | ||||||
|  |             output_path_reference, test_case_name | ||||||
|  |         ) | ||||||
|  |         test_case_output_path_betterproto = os.path.join( | ||||||
|  |             output_path_betterproto, test_case_name | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         print(f"Generating output for {test_case_name}") | ||||||
|  |         os.makedirs(test_case_output_path_reference, exist_ok=True) | ||||||
|  |         os.makedirs(test_case_output_path_betterproto, exist_ok=True) | ||||||
|  |  | ||||||
|  |         clear_directory(test_case_output_path_reference) | ||||||
|  |         clear_directory(test_case_output_path_betterproto) | ||||||
|  |  | ||||||
|  |         protoc_reference(test_case_input_path, test_case_output_path_reference) | ||||||
|  |         protoc_plugin(test_case_input_path, test_case_output_path_betterproto) | ||||||
|  |  | ||||||
|  |  | ||||||
| def get_files(end: str) -> Generator[str, None, None]: | HELP = "\n".join( | ||||||
|     for r, dirs, files in os.walk(root): |     [ | ||||||
|         for filename in [f for f in files if f.endswith(end)]: |         "Usage: python generate.py", | ||||||
|             yield os.path.join(r, filename) |         "       python generate.py [DIRECTORIES or NAMES]", | ||||||
|  |         "Generate python classes for standard tests.", | ||||||
|  |         "", | ||||||
|  |         "DIRECTORIES    One or more relative or absolute directories of test-cases to generate classes for.", | ||||||
|  |         "               python generate.py inputs/bool inputs/double inputs/enum", | ||||||
|  |         "", | ||||||
|  |         "NAMES          One or more test-case names to generate classes for.", | ||||||
|  |         "               python generate.py bool double enums", | ||||||
|  |     ] | ||||||
|  | ) | ||||||
|  |  | ||||||
|  |  | ||||||
| def get_base(filename: str) -> str: | def main(): | ||||||
|     return os.path.splitext(os.path.basename(filename))[0] |     if set(sys.argv).intersection({"-h", "--help"}): | ||||||
|  |         print(HELP) | ||||||
|  |         return | ||||||
|  |     whitelist = set(sys.argv[1:]) | ||||||
|  |  | ||||||
|  |     generate(whitelist) | ||||||
| def ensure_ext(filename: str, ext: str) -> str: |  | ||||||
|     if not filename.endswith(ext): |  | ||||||
|         return filename + ext |  | ||||||
|     return filename |  | ||||||
|  |  | ||||||
|  |  | ||||||
| if __name__ == "__main__": | if __name__ == "__main__": | ||||||
|     os.chdir(root) |     main() | ||||||
|  |  | ||||||
|     if len(sys.argv) > 1: |  | ||||||
|         proto_files = [ensure_ext(f, ".proto") for f in sys.argv[1:]] |  | ||||||
|         bases = {get_base(f) for f in proto_files} |  | ||||||
|         json_files = [ |  | ||||||
|             f for f in get_files(".json") if get_base(f).split("-")[0] in bases |  | ||||||
|         ] |  | ||||||
|     else: |  | ||||||
|         proto_files = get_files(".proto") |  | ||||||
|         json_files = get_files(".json") |  | ||||||
|  |  | ||||||
|     for filename in proto_files: |  | ||||||
|         print(f"Generating code for {os.path.basename(filename)}") |  | ||||||
|         subprocess.run( |  | ||||||
|             f"protoc --python_out=. {os.path.basename(filename)}", shell=True |  | ||||||
|         ) |  | ||||||
|         subprocess.run( |  | ||||||
|             f"protoc --plugin=protoc-gen-custom=../plugin.py --custom_out=. {os.path.basename(filename)}", |  | ||||||
|             shell=True, |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|     for filename in json_files: |  | ||||||
|         # Reset the internal symbol database so we can import the `Test` message |  | ||||||
|         # multiple times. Ugh. |  | ||||||
|         sym = symbol_database.Default() |  | ||||||
|         sym.pool = DescriptorPool() |  | ||||||
|  |  | ||||||
|         parts = get_base(filename).split("-") |  | ||||||
|         out = filename.replace(".json", ".bin") |  | ||||||
|         print(f"Using {parts[0]}_pb2 to generate {os.path.basename(out)}") |  | ||||||
|  |  | ||||||
|         imported = importlib.import_module(f"{parts[0]}_pb2") |  | ||||||
|         input_json = open(filename).read() |  | ||||||
|         parsed = Parse(input_json, imported.Test()) |  | ||||||
|         serialized = parsed.SerializeToString() |  | ||||||
|         preserve = "casing" not in filename |  | ||||||
|         serialized_json = MessageToJson(parsed, preserving_proto_field_name=preserve) |  | ||||||
|  |  | ||||||
|         s_loaded = json.loads(serialized_json) |  | ||||||
|         in_loaded = json.loads(input_json) |  | ||||||
|  |  | ||||||
|         if s_loaded != in_loaded: |  | ||||||
|             raise AssertionError("Expected JSON to be equal:", s_loaded, in_loaded) |  | ||||||
|  |  | ||||||
|         open(out, "wb").write(serialized) |  | ||||||
|   | |||||||
							
								
								
									
										6
									
								
								betterproto/tests/inputs/bool/test_bool.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								betterproto/tests/inputs/bool/test_bool.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | from betterproto.tests.output_betterproto.bool.bool import Test | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_value(): | ||||||
|  |     message = Test() | ||||||
|  |     assert not message.value, "Boolean is False by default" | ||||||
| @@ -9,4 +9,9 @@ enum my_enum { | |||||||
| message Test { | message Test { | ||||||
|   int32 camelCase = 1; |   int32 camelCase = 1; | ||||||
|   my_enum snake_case = 2; |   my_enum snake_case = 2; | ||||||
|  |   snake_case_message snake_case_message = 3; | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | message snake_case_message { | ||||||
|  | 
 | ||||||
|  | } | ||||||
							
								
								
									
										22
									
								
								betterproto/tests/inputs/casing/test_casing.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								betterproto/tests/inputs/casing/test_casing.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | |||||||
|  | import betterproto.tests.output_betterproto.casing.casing as casing | ||||||
|  | from betterproto.tests.output_betterproto.casing.casing import Test | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_message_attributes(): | ||||||
|  |     message = Test() | ||||||
|  |     assert hasattr( | ||||||
|  |         message, "snake_case_message" | ||||||
|  |     ), "snake_case field name is same in python" | ||||||
|  |     assert hasattr(message, "camel_case"), "CamelCase field is snake_case in python" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_message_casing(): | ||||||
|  |     assert hasattr( | ||||||
|  |         casing, "SnakeCaseMessage" | ||||||
|  |     ), "snake_case Message name is converted to CamelCase in python" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_enum_casing(): | ||||||
|  |     assert hasattr( | ||||||
|  |         casing, "MyEnum" | ||||||
|  |     ), "snake_case Enum name is converted to CamelCase in python" | ||||||
| @@ -0,0 +1,21 @@ | |||||||
|  | syntax = "proto3"; | ||||||
|  |  | ||||||
|  | import "google/protobuf/wrappers.proto"; | ||||||
|  |  | ||||||
|  | // Tests that wrapped values can be used directly as return values | ||||||
|  |  | ||||||
|  | service Test { | ||||||
|  |     rpc GetDouble (Input) returns (google.protobuf.DoubleValue); | ||||||
|  |     rpc GetFloat (Input) returns (google.protobuf.FloatValue); | ||||||
|  |     rpc GetInt64 (Input) returns (google.protobuf.Int64Value); | ||||||
|  |     rpc GetUInt64 (Input) returns (google.protobuf.UInt64Value); | ||||||
|  |     rpc GetInt32 (Input) returns (google.protobuf.Int32Value); | ||||||
|  |     rpc GetUInt32 (Input) returns (google.protobuf.UInt32Value); | ||||||
|  |     rpc GetBool (Input) returns (google.protobuf.BoolValue); | ||||||
|  |     rpc GetString (Input) returns (google.protobuf.StringValue); | ||||||
|  |     rpc GetBytes (Input) returns (google.protobuf.BytesValue); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | message Input { | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,56 @@ | |||||||
|  | from typing import Any, Callable, Optional | ||||||
|  |  | ||||||
|  | import google.protobuf.wrappers_pb2 as wrappers | ||||||
|  | import pytest | ||||||
|  |  | ||||||
|  | from betterproto.tests.mocks import MockChannel | ||||||
|  | from betterproto.tests.output_betterproto.googletypes_response.googletypes_response import ( | ||||||
|  |     TestStub, | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | test_cases = [ | ||||||
|  |     (TestStub.get_double, wrappers.DoubleValue, 2.5), | ||||||
|  |     (TestStub.get_float, wrappers.FloatValue, 2.5), | ||||||
|  |     (TestStub.get_int64, wrappers.Int64Value, -64), | ||||||
|  |     (TestStub.get_u_int64, wrappers.UInt64Value, 64), | ||||||
|  |     (TestStub.get_int32, wrappers.Int32Value, -32), | ||||||
|  |     (TestStub.get_u_int32, wrappers.UInt32Value, 32), | ||||||
|  |     (TestStub.get_bool, wrappers.BoolValue, True), | ||||||
|  |     (TestStub.get_string, wrappers.StringValue, "string"), | ||||||
|  |     (TestStub.get_bytes, wrappers.BytesValue, bytes(0xFF)[0:4]), | ||||||
|  | ] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.asyncio | ||||||
|  | @pytest.mark.parametrize(["service_method", "wrapper_class", "value"], test_cases) | ||||||
|  | async def test_channel_receives_wrapped_type( | ||||||
|  |     service_method: Callable[[TestStub], Any], wrapper_class: Callable, value | ||||||
|  | ): | ||||||
|  |     wrapped_value = wrapper_class() | ||||||
|  |     wrapped_value.value = value | ||||||
|  |     channel = MockChannel(responses=[wrapped_value]) | ||||||
|  |     service = TestStub(channel) | ||||||
|  |  | ||||||
|  |     await service_method(service) | ||||||
|  |  | ||||||
|  |     assert channel.requests[0]["response_type"] != Optional[type(value)] | ||||||
|  |     assert channel.requests[0]["response_type"] == type(wrapped_value) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.asyncio | ||||||
|  | @pytest.mark.xfail | ||||||
|  | @pytest.mark.parametrize(["service_method", "wrapper_class", "value"], test_cases) | ||||||
|  | async def test_service_unwraps_response( | ||||||
|  |     service_method: Callable[[TestStub], Any], wrapper_class: Callable, value | ||||||
|  | ): | ||||||
|  |     """ | ||||||
|  |     grpclib does not unwrap wrapper values returned by services | ||||||
|  |     """ | ||||||
|  |     wrapped_value = wrapper_class() | ||||||
|  |     wrapped_value.value = value | ||||||
|  |     service = TestStub(MockChannel(responses=[wrapped_value])) | ||||||
|  |  | ||||||
|  |     response_value = await service_method(service) | ||||||
|  |  | ||||||
|  |     assert response_value == value | ||||||
|  |     assert type(response_value) == type(value) | ||||||
| @@ -0,0 +1,24 @@ | |||||||
|  | syntax = "proto3"; | ||||||
|  |  | ||||||
|  | import "google/protobuf/wrappers.proto"; | ||||||
|  |  | ||||||
|  | // Tests that wrapped values are supported as part of output message | ||||||
|  | service Test { | ||||||
|  |     rpc getOutput (Input) returns (Output); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | message Input { | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  | message Output { | ||||||
|  |     google.protobuf.DoubleValue double_value = 1; | ||||||
|  |     google.protobuf.FloatValue float_value = 2; | ||||||
|  |     google.protobuf.Int64Value int64_value = 3; | ||||||
|  |     google.protobuf.UInt64Value uint64_value = 4; | ||||||
|  |     google.protobuf.Int32Value int32_value = 5; | ||||||
|  |     google.protobuf.UInt32Value uint32_value = 6; | ||||||
|  |     google.protobuf.BoolValue bool_value = 7; | ||||||
|  |     google.protobuf.StringValue string_value = 8; | ||||||
|  |     google.protobuf.BytesValue bytes_value = 9; | ||||||
|  | } | ||||||
| @@ -0,0 +1,39 @@ | |||||||
|  | import pytest | ||||||
|  |  | ||||||
|  | from betterproto.tests.mocks import MockChannel | ||||||
|  | from betterproto.tests.output_betterproto.googletypes_response_embedded.googletypes_response_embedded import ( | ||||||
|  |     Output, | ||||||
|  |     TestStub, | ||||||
|  | ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.asyncio | ||||||
|  | async def test_service_passes_through_unwrapped_values_embedded_in_response(): | ||||||
|  |     """ | ||||||
|  |     We do not not need to implement value unwrapping for embedded well-known types, | ||||||
|  |     as this is already handled by grpclib. This test merely shows that this is the case. | ||||||
|  |     """ | ||||||
|  |     output = Output( | ||||||
|  |         double_value=10.0, | ||||||
|  |         float_value=12.0, | ||||||
|  |         int64_value=-13, | ||||||
|  |         uint64_value=14, | ||||||
|  |         int32_value=-15, | ||||||
|  |         uint32_value=16, | ||||||
|  |         bool_value=True, | ||||||
|  |         string_value="string", | ||||||
|  |         bytes_value=bytes(0xFF)[0:4], | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     service = TestStub(MockChannel(responses=[output])) | ||||||
|  |     response = await service.get_output() | ||||||
|  |  | ||||||
|  |     assert response.double_value == 10.0 | ||||||
|  |     assert response.float_value == 12.0 | ||||||
|  |     assert response.int64_value == -13 | ||||||
|  |     assert response.uint64_value == 14 | ||||||
|  |     assert response.int32_value == -15 | ||||||
|  |     assert response.uint32_value == 16 | ||||||
|  |     assert response.bool_value | ||||||
|  |     assert response.string_value == "string" | ||||||
|  |     assert response.bytes_value == bytes(0xFF)[0:4] | ||||||
| @@ -0,0 +1,7 @@ | |||||||
|  | syntax = "proto3"; | ||||||
|  |  | ||||||
|  | package package.childpackage; | ||||||
|  |  | ||||||
|  | message ChildMessage { | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,9 @@ | |||||||
|  | syntax = "proto3"; | ||||||
|  |  | ||||||
|  | import "package_message.proto"; | ||||||
|  |  | ||||||
|  | // Tests generated imports when a message in a package refers to a message in a nested child package. | ||||||
|  |  | ||||||
|  | message Test { | ||||||
|  |   package.PackageMessage message = 1; | ||||||
|  | } | ||||||
| @@ -0,0 +1,9 @@ | |||||||
|  | syntax = "proto3"; | ||||||
|  |  | ||||||
|  | import "child.proto"; | ||||||
|  |  | ||||||
|  | package package; | ||||||
|  |  | ||||||
|  | message PackageMessage { | ||||||
|  |     package.childpackage.ChildMessage c = 1; | ||||||
|  | } | ||||||
| @@ -0,0 +1,7 @@ | |||||||
|  | syntax = "proto3"; | ||||||
|  |  | ||||||
|  | package childpackage; | ||||||
|  |  | ||||||
|  | message Message { | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,9 @@ | |||||||
|  | syntax = "proto3"; | ||||||
|  |  | ||||||
|  | import "child.proto"; | ||||||
|  |  | ||||||
|  | // Tests generated imports when a message in root refers to a message in a child package. | ||||||
|  |  | ||||||
|  | message Test { | ||||||
|  |   childpackage.Message child = 1; | ||||||
|  | } | ||||||
| @@ -0,0 +1,28 @@ | |||||||
|  | syntax = "proto3"; | ||||||
|  |  | ||||||
|  | import "root.proto"; | ||||||
|  | import "other.proto"; | ||||||
|  |  | ||||||
|  | // This test-case verifies that future implementations will support circular dependencies in the generated python files. | ||||||
|  | // | ||||||
|  | // This becomes important when generating 1 python file/module per package, rather than 1 file per proto file. | ||||||
|  | // | ||||||
|  | // Scenario: | ||||||
|  | // | ||||||
|  | //     The proto messages depend on each other in a non-circular way: | ||||||
|  | // | ||||||
|  | //           Test -------> RootPackageMessage <--------------. | ||||||
|  | //             `------------------------------------> OtherPackageMessage | ||||||
|  | // | ||||||
|  | //     Test and RootPackageMessage are in different files, but belong to the same package (root): | ||||||
|  | // | ||||||
|  | //           (Test -------> RootPackageMessage) <------------. | ||||||
|  | //             `------------------------------------> OtherPackageMessage | ||||||
|  | // | ||||||
|  | //     After grouping the packages into single files or modules, a circular dependency is created: | ||||||
|  | // | ||||||
|  | //           (root: Test & RootPackageMessage) <-------> (other: OtherPackageMessage) | ||||||
|  | message Test { | ||||||
|  |   RootPackageMessage message = 1; | ||||||
|  |   other.OtherPackageMessage other =2; | ||||||
|  | } | ||||||
| @@ -0,0 +1,8 @@ | |||||||
|  | syntax = "proto3"; | ||||||
|  |  | ||||||
|  | import "root.proto"; | ||||||
|  | package other; | ||||||
|  |  | ||||||
|  | message OtherPackageMessage { | ||||||
|  |     RootPackageMessage rootPackageMessage = 1; | ||||||
|  | } | ||||||
| @@ -0,0 +1,5 @@ | |||||||
|  | syntax = "proto3"; | ||||||
|  |  | ||||||
|  | message RootPackageMessage { | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -0,0 +1,12 @@ | |||||||
|  | syntax = "proto3"; | ||||||
|  |  | ||||||
|  | import "parent_package_message.proto"; | ||||||
|  |  | ||||||
|  | package parent.child; | ||||||
|  |  | ||||||
|  | // Tests generated imports when a message refers to a message defined in its parent package | ||||||
|  |  | ||||||
|  | message Test { | ||||||
|  |   ParentPackageMessage message_implicit = 1; | ||||||
|  |   parent.ParentPackageMessage message_explicit = 2; | ||||||
|  | } | ||||||
| @@ -0,0 +1,6 @@ | |||||||
|  | syntax = "proto3"; | ||||||
|  |  | ||||||
|  | package parent; | ||||||
|  |  | ||||||
|  | message ParentPackageMessage { | ||||||
|  | } | ||||||
| @@ -0,0 +1,11 @@ | |||||||
|  | syntax = "proto3"; | ||||||
|  |  | ||||||
|  | import "root.proto"; | ||||||
|  |  | ||||||
|  | package child; | ||||||
|  |  | ||||||
|  | // Tests generated imports when a message inside a child-package refers to a message defined in the root. | ||||||
|  |  | ||||||
|  | message Test { | ||||||
|  |   RootMessage message = 1; | ||||||
|  | } | ||||||
| @@ -0,0 +1,5 @@ | |||||||
|  | syntax = "proto3"; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | message RootMessage { | ||||||
|  | } | ||||||
| @@ -0,0 +1,9 @@ | |||||||
|  | syntax = "proto3"; | ||||||
|  |  | ||||||
|  | import "sibling.proto"; | ||||||
|  |  | ||||||
|  | // Tests generated imports when a message in the root package refers to another message in the root package | ||||||
|  |  | ||||||
|  | message Test { | ||||||
|  |   SiblingMessage sibling = 1; | ||||||
|  | } | ||||||
| @@ -0,0 +1,5 @@ | |||||||
|  | syntax = "proto3"; | ||||||
|  |  | ||||||
|  | message SiblingMessage { | ||||||
|  |  | ||||||
|  | } | ||||||
							
								
								
									
										4
									
								
								betterproto/tests/inputs/int32/int32.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								betterproto/tests/inputs/int32/int32.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | |||||||
|  | { | ||||||
|  |   "positive": 150, | ||||||
|  |   "negative": -150 | ||||||
|  | } | ||||||
| @@ -3,5 +3,6 @@ syntax = "proto3"; | |||||||
| // Some documentation about the Test message. | // Some documentation about the Test message. | ||||||
| message Test { | message Test { | ||||||
|     // Some documentation about the count. |     // Some documentation about the count. | ||||||
|     int32 count = 1; |     int32 positive = 1; | ||||||
|  |     int32 negative = 2; | ||||||
| } | } | ||||||
| @@ -5,3 +5,7 @@ message Test { | |||||||
|   int32 with = 2; |   int32 with = 2; | ||||||
|   int32 as = 3; |   int32 as = 3; | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | service TestService { | ||||||
|  |   rpc GetTest(Test) returns (Test) {} | ||||||
|  | } | ||||||
							
								
								
									
										11
									
								
								betterproto/tests/inputs/nestedtwice/nestedtwice.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								betterproto/tests/inputs/nestedtwice/nestedtwice.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | |||||||
|  | { | ||||||
|  |   "root": { | ||||||
|  |     "name": "double-nested", | ||||||
|  |     "parent": { | ||||||
|  |       "child": [{"foo": "hello"}], | ||||||
|  |       "enumChild": ["A"], | ||||||
|  |       "rootParentChild": [{"a": "hello"}], | ||||||
|  |       "bar": true | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										26
									
								
								betterproto/tests/inputs/nestedtwice/nestedtwice.proto
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								betterproto/tests/inputs/nestedtwice/nestedtwice.proto
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | |||||||
|  | syntax = "proto3"; | ||||||
|  |  | ||||||
|  | message Test { | ||||||
|  |   message Root { | ||||||
|  |     message Parent { | ||||||
|  |       message RootParentChild { | ||||||
|  |         string a = 1; | ||||||
|  |       } | ||||||
|  |       enum EnumChild{ | ||||||
|  |         A = 0; | ||||||
|  |         B = 1; | ||||||
|  |       } | ||||||
|  |       message Child { | ||||||
|  |         string foo = 1; | ||||||
|  |       } | ||||||
|  |       reserved 1; | ||||||
|  |       repeated Child child = 2; | ||||||
|  |       repeated EnumChild enumChild=3; | ||||||
|  |       repeated RootParentChild rootParentChild=4; | ||||||
|  |       bool bar = 5; | ||||||
|  |     } | ||||||
|  |     string name = 1; | ||||||
|  |     Parent parent = 2; | ||||||
|  |   } | ||||||
|  |   Root root = 1; | ||||||
|  | } | ||||||
							
								
								
									
										3
									
								
								betterproto/tests/inputs/oneof/oneof-name.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								betterproto/tests/inputs/oneof/oneof-name.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | { | ||||||
|  |   "name": "foobar" | ||||||
|  | } | ||||||
							
								
								
									
										3
									
								
								betterproto/tests/inputs/oneof/oneof.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								betterproto/tests/inputs/oneof/oneof.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | { | ||||||
|  |   "count": 100 | ||||||
|  | } | ||||||
							
								
								
									
										15
									
								
								betterproto/tests/inputs/oneof/test_oneof.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								betterproto/tests/inputs/oneof/test_oneof.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | |||||||
|  | import betterproto | ||||||
|  | from betterproto.tests.output_betterproto.oneof.oneof import Test | ||||||
|  | from betterproto.tests.util import get_test_case_json_data | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_which_count(): | ||||||
|  |     message = Test() | ||||||
|  |     message.from_json(get_test_case_json_data("oneof")) | ||||||
|  |     assert betterproto.which_one_of(message, "foo") == ("count", 100) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_which_name(): | ||||||
|  |     message = Test() | ||||||
|  |     message.from_json(get_test_case_json_data("oneof", "oneof-name.json")) | ||||||
|  |     assert betterproto.which_one_of(message, "foo") == ("name", "foobar") | ||||||
| @@ -0,0 +1,3 @@ | |||||||
|  | { | ||||||
|  |   "signal": "PASS" | ||||||
|  | } | ||||||
| @@ -0,0 +1,3 @@ | |||||||
|  | { | ||||||
|  |   "signal": "RESIGN" | ||||||
|  | } | ||||||
							
								
								
									
										6
									
								
								betterproto/tests/inputs/oneof_enum/oneof_enum.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								betterproto/tests/inputs/oneof_enum/oneof_enum.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | { | ||||||
|  |   "move": { | ||||||
|  |     "x": 2, | ||||||
|  |     "y": 3 | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										18
									
								
								betterproto/tests/inputs/oneof_enum/oneof_enum.proto
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								betterproto/tests/inputs/oneof_enum/oneof_enum.proto
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | |||||||
|  | syntax = "proto3"; | ||||||
|  |  | ||||||
|  | message Test { | ||||||
|  |   oneof action { | ||||||
|  |     Signal signal = 1; | ||||||
|  |     Move move = 2; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | enum Signal { | ||||||
|  |   PASS = 0; | ||||||
|  |   RESIGN = 1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | message Move { | ||||||
|  |   int32 x = 1; | ||||||
|  |   int32 y = 2; | ||||||
|  | } | ||||||
							
								
								
									
										42
									
								
								betterproto/tests/inputs/oneof_enum/test_oneof_enum.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								betterproto/tests/inputs/oneof_enum/test_oneof_enum.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,42 @@ | |||||||
|  | import pytest | ||||||
|  |  | ||||||
|  | import betterproto | ||||||
|  | from betterproto.tests.output_betterproto.oneof_enum.oneof_enum import ( | ||||||
|  |     Move, | ||||||
|  |     Signal, | ||||||
|  |     Test, | ||||||
|  | ) | ||||||
|  | from betterproto.tests.util import get_test_case_json_data | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.xfail | ||||||
|  | def test_which_one_of_returns_enum_with_default_value(): | ||||||
|  |     """ | ||||||
|  |     returns first field when it is enum and set with default value | ||||||
|  |     """ | ||||||
|  |     message = Test() | ||||||
|  |     message.from_json(get_test_case_json_data("oneof_enum", "oneof_enum-enum-0.json")) | ||||||
|  |     assert message.move is None | ||||||
|  |     assert message.signal == Signal.PASS | ||||||
|  |     assert betterproto.which_one_of(message, "action") == ("signal", Signal.PASS) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.xfail | ||||||
|  | def test_which_one_of_returns_enum_with_non_default_value(): | ||||||
|  |     """ | ||||||
|  |     returns first field when it is enum and set with non default value | ||||||
|  |     """ | ||||||
|  |     message = Test() | ||||||
|  |     message.from_json(get_test_case_json_data("oneof_enum", "oneof_enum-enum-1.json")) | ||||||
|  |     assert message.move is None | ||||||
|  |     assert message.signal == Signal.PASS | ||||||
|  |     assert betterproto.which_one_of(message, "action") == ("signal", Signal.RESIGN) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.xfail | ||||||
|  | def test_which_one_of_returns_second_field_when_set(): | ||||||
|  |     message = Test() | ||||||
|  |     message.from_json(get_test_case_json_data("oneof_enum")) | ||||||
|  |     assert message.move == Move(x=2, y=3) | ||||||
|  |     assert message.signal == 0 | ||||||
|  |     assert betterproto.which_one_of(message, "action") == ("move", Move(x=2, y=3)) | ||||||
| @@ -0,0 +1,11 @@ | |||||||
|  | syntax = "proto3"; | ||||||
|  |  | ||||||
|  | package repeatedmessage; | ||||||
|  |  | ||||||
|  | message Test { | ||||||
|  |   repeated Sub greetings = 1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | message Sub { | ||||||
|  |   string greeting = 1; | ||||||
|  | } | ||||||
							
								
								
									
										15
									
								
								betterproto/tests/inputs/service/service.proto
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								betterproto/tests/inputs/service/service.proto
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | |||||||
|  | syntax = "proto3"; | ||||||
|  |  | ||||||
|  | package service; | ||||||
|  |  | ||||||
|  | message DoThingRequest { | ||||||
|  |   int32 iterations = 1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | message DoThingResponse { | ||||||
|  |   int32 successfulIterations = 1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | service Test { | ||||||
|  |   rpc DoThing (DoThingRequest) returns (DoThingResponse); | ||||||
|  | } | ||||||
							
								
								
									
										132
									
								
								betterproto/tests/inputs/service/test_service.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										132
									
								
								betterproto/tests/inputs/service/test_service.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,132 @@ | |||||||
|  | import betterproto | ||||||
|  | import grpclib | ||||||
|  | from grpclib.testing import ChannelFor | ||||||
|  | import pytest | ||||||
|  | from typing import Dict | ||||||
|  |  | ||||||
|  | from betterproto.tests.output_betterproto.service.service import ( | ||||||
|  |     DoThingResponse, | ||||||
|  |     DoThingRequest, | ||||||
|  |     TestStub as ExampleServiceStub, | ||||||
|  | ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ExampleService: | ||||||
|  |     def __init__(self, test_hook=None): | ||||||
|  |         # This lets us pass assertions to the servicer ;) | ||||||
|  |         self.test_hook = test_hook | ||||||
|  |  | ||||||
|  |     async def DoThing( | ||||||
|  |         self, stream: "grpclib.server.Stream[DoThingRequest, DoThingResponse]" | ||||||
|  |     ): | ||||||
|  |         request = await stream.recv_message() | ||||||
|  |         print("self.test_hook", self.test_hook) | ||||||
|  |         if self.test_hook is not None: | ||||||
|  |             self.test_hook(stream) | ||||||
|  |         for iteration in range(request.iterations): | ||||||
|  |             pass | ||||||
|  |         await stream.send_message(DoThingResponse(request.iterations)) | ||||||
|  |  | ||||||
|  |     def __mapping__(self) -> Dict[str, grpclib.const.Handler]: | ||||||
|  |         return { | ||||||
|  |             "/service.Test/DoThing": grpclib.const.Handler( | ||||||
|  |                 self.DoThing, | ||||||
|  |                 grpclib.const.Cardinality.UNARY_UNARY, | ||||||
|  |                 DoThingRequest, | ||||||
|  |                 DoThingResponse, | ||||||
|  |             ), | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | async def _test_stub(stub, iterations=42, **kwargs): | ||||||
|  |     response = await stub.do_thing(iterations=iterations) | ||||||
|  |     assert response.successful_iterations == iterations | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def _get_server_side_test(deadline, metadata): | ||||||
|  |     def server_side_test(stream): | ||||||
|  |         assert stream.deadline._timestamp == pytest.approx( | ||||||
|  |             deadline._timestamp, 1 | ||||||
|  |         ), "The provided deadline should be recieved serverside" | ||||||
|  |         assert ( | ||||||
|  |             stream.metadata["authorization"] == metadata["authorization"] | ||||||
|  |         ), "The provided authorization metadata should be recieved serverside" | ||||||
|  |  | ||||||
|  |     return server_side_test | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.asyncio | ||||||
|  | async def test_simple_service_call(): | ||||||
|  |     async with ChannelFor([ExampleService()]) as channel: | ||||||
|  |         await _test_stub(ExampleServiceStub(channel)) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.asyncio | ||||||
|  | async def test_service_call_with_upfront_request_params(): | ||||||
|  |     # Setting deadline | ||||||
|  |     deadline = grpclib.metadata.Deadline.from_timeout(22) | ||||||
|  |     metadata = {"authorization": "12345"} | ||||||
|  |     async with ChannelFor( | ||||||
|  |         [ExampleService(test_hook=_get_server_side_test(deadline, metadata))] | ||||||
|  |     ) as channel: | ||||||
|  |         await _test_stub( | ||||||
|  |             ExampleServiceStub(channel, deadline=deadline, metadata=metadata) | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     # Setting timeout | ||||||
|  |     timeout = 99 | ||||||
|  |     deadline = grpclib.metadata.Deadline.from_timeout(timeout) | ||||||
|  |     metadata = {"authorization": "12345"} | ||||||
|  |     async with ChannelFor( | ||||||
|  |         [ExampleService(test_hook=_get_server_side_test(deadline, metadata))] | ||||||
|  |     ) as channel: | ||||||
|  |         await _test_stub( | ||||||
|  |             ExampleServiceStub(channel, timeout=timeout, metadata=metadata) | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.asyncio | ||||||
|  | async def test_service_call_lower_level_with_overrides(): | ||||||
|  |     ITERATIONS = 99 | ||||||
|  |  | ||||||
|  |     # Setting deadline | ||||||
|  |     deadline = grpclib.metadata.Deadline.from_timeout(22) | ||||||
|  |     metadata = {"authorization": "12345"} | ||||||
|  |     kwarg_deadline = grpclib.metadata.Deadline.from_timeout(28) | ||||||
|  |     kwarg_metadata = {"authorization": "12345"} | ||||||
|  |     async with ChannelFor( | ||||||
|  |         [ExampleService(test_hook=_get_server_side_test(deadline, metadata))] | ||||||
|  |     ) as channel: | ||||||
|  |         stub = ExampleServiceStub(channel, deadline=deadline, metadata=metadata) | ||||||
|  |         response = await stub._unary_unary( | ||||||
|  |             "/service.Test/DoThing", | ||||||
|  |             DoThingRequest(ITERATIONS), | ||||||
|  |             DoThingResponse, | ||||||
|  |             deadline=kwarg_deadline, | ||||||
|  |             metadata=kwarg_metadata, | ||||||
|  |         ) | ||||||
|  |         assert response.successful_iterations == ITERATIONS | ||||||
|  |  | ||||||
|  |     # Setting timeout | ||||||
|  |     timeout = 99 | ||||||
|  |     deadline = grpclib.metadata.Deadline.from_timeout(timeout) | ||||||
|  |     metadata = {"authorization": "12345"} | ||||||
|  |     kwarg_timeout = 9000 | ||||||
|  |     kwarg_deadline = grpclib.metadata.Deadline.from_timeout(kwarg_timeout) | ||||||
|  |     kwarg_metadata = {"authorization": "09876"} | ||||||
|  |     async with ChannelFor( | ||||||
|  |         [ | ||||||
|  |             ExampleService( | ||||||
|  |                 test_hook=_get_server_side_test(kwarg_deadline, kwarg_metadata) | ||||||
|  |             ) | ||||||
|  |         ] | ||||||
|  |     ) as channel: | ||||||
|  |         stub = ExampleServiceStub(channel, deadline=deadline, metadata=metadata) | ||||||
|  |         response = await stub._unary_unary( | ||||||
|  |             "/service.Test/DoThing", | ||||||
|  |             DoThingRequest(ITERATIONS), | ||||||
|  |             DoThingResponse, | ||||||
|  |             timeout=kwarg_timeout, | ||||||
|  |             metadata=kwarg_metadata, | ||||||
|  |         ) | ||||||
|  |         assert response.successful_iterations == ITERATIONS | ||||||
							
								
								
									
										6
									
								
								betterproto/tests/inputs/signed/signed.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								betterproto/tests/inputs/signed/signed.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | { | ||||||
|  |   "signed32": 150, | ||||||
|  |   "negative32": -150, | ||||||
|  |   "string64": "150", | ||||||
|  |   "negative64": "-150" | ||||||
|  | } | ||||||
							
								
								
									
										9
									
								
								betterproto/tests/inputs/signed/signed.proto
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								betterproto/tests/inputs/signed/signed.proto
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | |||||||
|  | syntax = "proto3"; | ||||||
|  |  | ||||||
|  | message Test { | ||||||
|  |     // todo: rename fields after fixing bug where 'signed_32_positive' will map to 'signed_32Positive' as output json | ||||||
|  |     sint32 signed32 = 1;    //  signed_32_positive | ||||||
|  |     sint32 negative32 = 2;  //  signed_32_negative | ||||||
|  |     sint64 string64 = 3;    //  signed_64_positive | ||||||
|  |     sint64 negative64 = 4;  //  signed_64_negative | ||||||
|  | } | ||||||
							
								
								
									
										10
									
								
								betterproto/tests/inputs/xfail.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								betterproto/tests/inputs/xfail.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | |||||||
|  | # Test cases that are expected to fail, e.g. unimplemented features or bug-fixes. | ||||||
|  | # Remove from list when fixed. | ||||||
|  | tests = { | ||||||
|  |     "import_root_sibling", | ||||||
|  |     "import_child_package_from_package", | ||||||
|  |     "import_root_package_from_child", | ||||||
|  |     "import_parent_package_from_child", | ||||||
|  |     "import_circular_dependency", | ||||||
|  |     "oneof_enum", | ||||||
|  | } | ||||||
| @@ -1,3 +0,0 @@ | |||||||
| { |  | ||||||
|   "count": -150 |  | ||||||
| } |  | ||||||
| @@ -1,3 +0,0 @@ | |||||||
| { |  | ||||||
|   "count": 150 |  | ||||||
| } |  | ||||||
							
								
								
									
										39
									
								
								betterproto/tests/mocks.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								betterproto/tests/mocks.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,39 @@ | |||||||
|  | from typing import List | ||||||
|  |  | ||||||
|  | from grpclib.client import Channel | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class MockChannel(Channel): | ||||||
|  |     # noinspection PyMissingConstructor | ||||||
|  |     def __init__(self, responses=None) -> None: | ||||||
|  |         self.responses = responses if responses else [] | ||||||
|  |         self.requests = [] | ||||||
|  |  | ||||||
|  |     def request(self, route, cardinality, request, response_type, **kwargs): | ||||||
|  |         self.requests.append( | ||||||
|  |             { | ||||||
|  |                 "route": route, | ||||||
|  |                 "cardinality": cardinality, | ||||||
|  |                 "request": request, | ||||||
|  |                 "response_type": response_type, | ||||||
|  |             } | ||||||
|  |         ) | ||||||
|  |         return MockStream(self.responses) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class MockStream: | ||||||
|  |     def __init__(self, responses: List) -> None: | ||||||
|  |         super().__init__() | ||||||
|  |         self.responses = responses | ||||||
|  |  | ||||||
|  |     async def recv_message(self): | ||||||
|  |         return self.responses.pop(0) | ||||||
|  |  | ||||||
|  |     async def send_message(self, *args, **kwargs): | ||||||
|  |         pass | ||||||
|  |  | ||||||
|  |     async def __aexit__(self, exc_type, exc_val, exc_tb): | ||||||
|  |         return True | ||||||
|  |  | ||||||
|  |     async def __aenter__(self): | ||||||
|  |         return self | ||||||
| @@ -1,3 +0,0 @@ | |||||||
| { |  | ||||||
|   "name": "foo" |  | ||||||
| } |  | ||||||
| @@ -1,3 +0,0 @@ | |||||||
| { |  | ||||||
|   "count": 1 |  | ||||||
| } |  | ||||||
| @@ -1,4 +0,0 @@ | |||||||
| { |  | ||||||
|   "signed_32": -150, |  | ||||||
|   "signed_64": "-150" |  | ||||||
| } |  | ||||||
| @@ -1,4 +0,0 @@ | |||||||
| { |  | ||||||
|   "signed_32": 150, |  | ||||||
|   "signed_64": "150" |  | ||||||
| } |  | ||||||
| @@ -1,6 +0,0 @@ | |||||||
| syntax = "proto3"; |  | ||||||
|  |  | ||||||
| message Test { |  | ||||||
|     sint32 signed_32 = 1; |  | ||||||
|     sint64 signed_64 = 2; |  | ||||||
| } |  | ||||||
| @@ -33,6 +33,21 @@ def test_has_field(): | |||||||
|     assert betterproto.serialized_on_wire(foo.bar) == False |     assert betterproto.serialized_on_wire(foo.bar) == False | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_class_init(): | ||||||
|  |     @dataclass | ||||||
|  |     class Bar(betterproto.Message): | ||||||
|  |         name: str = betterproto.string_field(1) | ||||||
|  |  | ||||||
|  |     @dataclass | ||||||
|  |     class Foo(betterproto.Message): | ||||||
|  |         name: str = betterproto.string_field(1) | ||||||
|  |         child: Bar = betterproto.message_field(2) | ||||||
|  |  | ||||||
|  |     foo = Foo(name="foo", child=Bar(name="bar")) | ||||||
|  |  | ||||||
|  |     assert foo.to_dict() == {"name": "foo", "child": {"name": "bar"}} | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_enum_as_int_json(): | def test_enum_as_int_json(): | ||||||
|     class TestEnum(betterproto.Enum): |     class TestEnum(betterproto.Enum): | ||||||
|         ZERO = 0 |         ZERO = 0 | ||||||
| @@ -162,3 +177,89 @@ def test_optional_flag(): | |||||||
|     # Differentiate between not passed and the zero-value. |     # Differentiate between not passed and the zero-value. | ||||||
|     assert Request().parse(b"").flag == None |     assert Request().parse(b"").flag == None | ||||||
|     assert Request().parse(b"\n\x00").flag == False |     assert Request().parse(b"\n\x00").flag == False | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_to_dict_default_values(): | ||||||
|  |     @dataclass | ||||||
|  |     class TestMessage(betterproto.Message): | ||||||
|  |         some_int: int = betterproto.int32_field(1) | ||||||
|  |         some_double: float = betterproto.double_field(2) | ||||||
|  |         some_str: str = betterproto.string_field(3) | ||||||
|  |         some_bool: bool = betterproto.bool_field(4) | ||||||
|  |  | ||||||
|  |     # Empty dict | ||||||
|  |     test = TestMessage().from_dict({}) | ||||||
|  |  | ||||||
|  |     assert test.to_dict(include_default_values=True) == { | ||||||
|  |         "someInt": 0, | ||||||
|  |         "someDouble": 0.0, | ||||||
|  |         "someStr": "", | ||||||
|  |         "someBool": False, | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     # All default values | ||||||
|  |     test = TestMessage().from_dict( | ||||||
|  |         {"someInt": 0, "someDouble": 0.0, "someStr": "", "someBool": False} | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     assert test.to_dict(include_default_values=True) == { | ||||||
|  |         "someInt": 0, | ||||||
|  |         "someDouble": 0.0, | ||||||
|  |         "someStr": "", | ||||||
|  |         "someBool": False, | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     # Some default and some other values | ||||||
|  |     @dataclass | ||||||
|  |     class TestMessage2(betterproto.Message): | ||||||
|  |         some_int: int = betterproto.int32_field(1) | ||||||
|  |         some_double: float = betterproto.double_field(2) | ||||||
|  |         some_str: str = betterproto.string_field(3) | ||||||
|  |         some_bool: bool = betterproto.bool_field(4) | ||||||
|  |         some_default_int: int = betterproto.int32_field(5) | ||||||
|  |         some_default_double: float = betterproto.double_field(6) | ||||||
|  |         some_default_str: str = betterproto.string_field(7) | ||||||
|  |         some_default_bool: bool = betterproto.bool_field(8) | ||||||
|  |  | ||||||
|  |     test = TestMessage2().from_dict( | ||||||
|  |         { | ||||||
|  |             "someInt": 2, | ||||||
|  |             "someDouble": 1.2, | ||||||
|  |             "someStr": "hello", | ||||||
|  |             "someBool": True, | ||||||
|  |             "someDefaultInt": 0, | ||||||
|  |             "someDefaultDouble": 0.0, | ||||||
|  |             "someDefaultStr": "", | ||||||
|  |             "someDefaultBool": False, | ||||||
|  |         } | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     assert test.to_dict(include_default_values=True) == { | ||||||
|  |         "someInt": 2, | ||||||
|  |         "someDouble": 1.2, | ||||||
|  |         "someStr": "hello", | ||||||
|  |         "someBool": True, | ||||||
|  |         "someDefaultInt": 0, | ||||||
|  |         "someDefaultDouble": 0.0, | ||||||
|  |         "someDefaultStr": "", | ||||||
|  |         "someDefaultBool": False, | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     # Nested messages | ||||||
|  |     @dataclass | ||||||
|  |     class TestChildMessage(betterproto.Message): | ||||||
|  |         some_other_int: int = betterproto.int32_field(1) | ||||||
|  |  | ||||||
|  |     @dataclass | ||||||
|  |     class TestParentMessage(betterproto.Message): | ||||||
|  |         some_int: int = betterproto.int32_field(1) | ||||||
|  |         some_double: float = betterproto.double_field(2) | ||||||
|  |         some_message: TestChildMessage = betterproto.message_field(3) | ||||||
|  |  | ||||||
|  |     test = TestParentMessage().from_dict({"someInt": 0, "someDouble": 1.2}) | ||||||
|  |  | ||||||
|  |     assert test.to_dict(include_default_values=True) == { | ||||||
|  |         "someInt": 0, | ||||||
|  |         "someDouble": 1.2, | ||||||
|  |         "someMessage": {"someOtherInt": 0}, | ||||||
|  |     } | ||||||
|   | |||||||
| @@ -1,32 +1,147 @@ | |||||||
| import importlib | import importlib | ||||||
| import json | import json | ||||||
|  | import os | ||||||
|  | import sys | ||||||
|  | from collections import namedtuple | ||||||
|  | from typing import Set | ||||||
|  |  | ||||||
| import pytest | import pytest | ||||||
|  |  | ||||||
| from .generate import get_base, get_files | import betterproto | ||||||
|  | from betterproto.tests.inputs import xfail | ||||||
|  | from betterproto.tests.mocks import MockChannel | ||||||
|  | from betterproto.tests.util import get_directories, get_test_case_json_data, inputs_path | ||||||
|  |  | ||||||
| inputs = get_files(".bin") | # Force pure-python implementation instead of C++, otherwise imports | ||||||
|  | # break things because we can't properly reset the symbol database. | ||||||
|  | os.environ["PROTOCOL_BUFFERS_PYTHON_IMPLEMENTATION"] = "python" | ||||||
|  |  | ||||||
|  | from google.protobuf import symbol_database | ||||||
|  | from google.protobuf.descriptor_pool import DescriptorPool | ||||||
|  | from google.protobuf.json_format import Parse | ||||||
|  |  | ||||||
|  |  | ||||||
| @pytest.mark.parametrize("filename", inputs) | class TestCases: | ||||||
| def test_sample(filename: str) -> None: |     def __init__(self, path, services: Set[str], xfail: Set[str]): | ||||||
|     module = get_base(filename).split("-")[0] |         _all = set(get_directories(path)) | ||||||
|     imported = importlib.import_module(f"betterproto.tests.{module}") |         _services = services | ||||||
|     data_binary = open(filename, "rb").read() |         _messages = _all - services | ||||||
|     data_dict = json.loads(open(filename.replace(".bin", ".json")).read()) |         _messages_with_json = { | ||||||
|     t1 = imported.Test().parse(data_binary) |             test for test in _messages if get_test_case_json_data(test) | ||||||
|     t2 = imported.Test().from_dict(data_dict) |         } | ||||||
|     print(t1) |  | ||||||
|     print(t2) |  | ||||||
|  |  | ||||||
|     # Equality should automagically work for dataclasses! |         self.all = self.apply_xfail_marks(_all, xfail) | ||||||
|     assert t1 == t2 |         self.services = self.apply_xfail_marks(_services, xfail) | ||||||
|  |         self.messages = self.apply_xfail_marks(_messages, xfail) | ||||||
|  |         self.messages_with_json = self.apply_xfail_marks(_messages_with_json, xfail) | ||||||
|  |  | ||||||
|     # Generally this can't be relied on, but here we are aiming to match the |     @staticmethod | ||||||
|     # existing Python implementation and aren't doing anything tricky. |     def apply_xfail_marks(test_set: Set[str], xfail: Set[str]): | ||||||
|     # https://developers.google.com/protocol-buffers/docs/encoding#implications |         return [ | ||||||
|     assert bytes(t1) == data_binary |             pytest.param(test, marks=pytest.mark.xfail) if test in xfail else test | ||||||
|     assert bytes(t2) == data_binary |             for test in test_set | ||||||
|  |         ] | ||||||
|  |  | ||||||
|     assert t1.to_dict() == data_dict |  | ||||||
|     assert t2.to_dict() == data_dict | test_cases = TestCases( | ||||||
|  |     path=inputs_path, | ||||||
|  |     # test cases for services | ||||||
|  |     services={"googletypes_response", "googletypes_response_embedded", "service"}, | ||||||
|  |     xfail=xfail.tests, | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | plugin_output_package = "betterproto.tests.output_betterproto" | ||||||
|  | reference_output_package = "betterproto.tests.output_reference" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | TestData = namedtuple("TestData", "plugin_module, reference_module, json_data") | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.fixture | ||||||
|  | def test_data(request): | ||||||
|  |     test_case_name = request.param | ||||||
|  |  | ||||||
|  |     # Reset the internal symbol database so we can import the `Test` message | ||||||
|  |     # multiple times. Ugh. | ||||||
|  |     sym = symbol_database.Default() | ||||||
|  |     sym.pool = DescriptorPool() | ||||||
|  |  | ||||||
|  |     reference_module_root = os.path.join( | ||||||
|  |         *reference_output_package.split("."), test_case_name | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     sys.path.append(reference_module_root) | ||||||
|  |  | ||||||
|  |     yield ( | ||||||
|  |         TestData( | ||||||
|  |             plugin_module=importlib.import_module( | ||||||
|  |                 f"{plugin_output_package}.{test_case_name}.{test_case_name}" | ||||||
|  |             ), | ||||||
|  |             reference_module=lambda: importlib.import_module( | ||||||
|  |                 f"{reference_output_package}.{test_case_name}.{test_case_name}_pb2" | ||||||
|  |             ), | ||||||
|  |             json_data=get_test_case_json_data(test_case_name), | ||||||
|  |         ) | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     sys.path.remove(reference_module_root) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.parametrize("test_data", test_cases.messages, indirect=True) | ||||||
|  | def test_message_can_instantiated(test_data: TestData) -> None: | ||||||
|  |     plugin_module, *_ = test_data | ||||||
|  |     plugin_module.Test() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.parametrize("test_data", test_cases.messages, indirect=True) | ||||||
|  | def test_message_equality(test_data: TestData) -> None: | ||||||
|  |     plugin_module, *_ = test_data | ||||||
|  |     message1 = plugin_module.Test() | ||||||
|  |     message2 = plugin_module.Test() | ||||||
|  |     assert message1 == message2 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.parametrize("test_data", test_cases.messages_with_json, indirect=True) | ||||||
|  | def test_message_json(repeat, test_data: TestData) -> None: | ||||||
|  |     plugin_module, _, json_data = test_data | ||||||
|  |  | ||||||
|  |     for _ in range(repeat): | ||||||
|  |         message: betterproto.Message = plugin_module.Test() | ||||||
|  |  | ||||||
|  |         message.from_json(json_data) | ||||||
|  |         message_json = message.to_json(0) | ||||||
|  |  | ||||||
|  |         assert json.loads(json_data) == json.loads(message_json) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.parametrize("test_data", test_cases.services, indirect=True) | ||||||
|  | def test_service_can_be_instantiated(test_data: TestData) -> None: | ||||||
|  |     plugin_module, _, json_data = test_data | ||||||
|  |     plugin_module.TestStub(MockChannel()) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.parametrize("test_data", test_cases.messages_with_json, indirect=True) | ||||||
|  | def test_binary_compatibility(repeat, test_data: TestData) -> None: | ||||||
|  |     plugin_module, reference_module, json_data = test_data | ||||||
|  |  | ||||||
|  |     reference_instance = Parse(json_data, reference_module().Test()) | ||||||
|  |     reference_binary_output = reference_instance.SerializeToString() | ||||||
|  |  | ||||||
|  |     for _ in range(repeat): | ||||||
|  |         plugin_instance_from_json: betterproto.Message = plugin_module.Test().from_json( | ||||||
|  |             json_data | ||||||
|  |         ) | ||||||
|  |         plugin_instance_from_binary = plugin_module.Test.FromString( | ||||||
|  |             reference_binary_output | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         # # Generally this can't be relied on, but here we are aiming to match the | ||||||
|  |         # # existing Python implementation and aren't doing anything tricky. | ||||||
|  |         # # https://developers.google.com/protocol-buffers/docs/encoding#implications | ||||||
|  |         assert bytes(plugin_instance_from_json) == reference_binary_output | ||||||
|  |         assert bytes(plugin_instance_from_binary) == reference_binary_output | ||||||
|  |  | ||||||
|  |         assert plugin_instance_from_json == plugin_instance_from_binary | ||||||
|  |         assert ( | ||||||
|  |             plugin_instance_from_json.to_dict() == plugin_instance_from_binary.to_dict() | ||||||
|  |         ) | ||||||
|   | |||||||
							
								
								
									
										61
									
								
								betterproto/tests/util.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										61
									
								
								betterproto/tests/util.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,61 @@ | |||||||
|  | import os | ||||||
|  | import subprocess | ||||||
|  | from typing import Generator | ||||||
|  |  | ||||||
|  | os.environ["PROTOCOL_BUFFERS_PYTHON_IMPLEMENTATION"] = "python" | ||||||
|  |  | ||||||
|  | root_path = os.path.dirname(os.path.realpath(__file__)) | ||||||
|  | inputs_path = os.path.join(root_path, "inputs") | ||||||
|  | output_path_reference = os.path.join(root_path, "output_reference") | ||||||
|  | output_path_betterproto = os.path.join(root_path, "output_betterproto") | ||||||
|  |  | ||||||
|  | if os.name == "nt": | ||||||
|  |     plugin_path = os.path.join(root_path, "..", "plugin.bat") | ||||||
|  | else: | ||||||
|  |     plugin_path = os.path.join(root_path, "..", "plugin.py") | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def get_files(path, end: str) -> Generator[str, None, None]: | ||||||
|  |     for r, dirs, files in os.walk(path): | ||||||
|  |         for filename in [f for f in files if f.endswith(end)]: | ||||||
|  |             yield os.path.join(r, filename) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def get_directories(path): | ||||||
|  |     for root, directories, files in os.walk(path): | ||||||
|  |         for directory in directories: | ||||||
|  |             yield directory | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def relative(file: str, path: str): | ||||||
|  |     return os.path.join(os.path.dirname(file), path) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def read_relative(file: str, path: str): | ||||||
|  |     with open(relative(file, path)) as fh: | ||||||
|  |         return fh.read() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def protoc_plugin(path: str, output_dir: str): | ||||||
|  |     subprocess.run( | ||||||
|  |         f"protoc --plugin=protoc-gen-custom={plugin_path} --custom_out={output_dir} --proto_path={path} {path}/*.proto", | ||||||
|  |         shell=True, | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def protoc_reference(path: str, output_dir: str): | ||||||
|  |     subprocess.run( | ||||||
|  |         f"protoc --python_out={output_dir} --proto_path={path} {path}/*.proto", | ||||||
|  |         shell=True, | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def get_test_case_json_data(test_case_name, json_file_name=None): | ||||||
|  |     test_data_file_name = json_file_name if json_file_name else f"{test_case_name}.json" | ||||||
|  |     test_data_file_path = os.path.join(inputs_path, test_case_name, test_data_file_name) | ||||||
|  |  | ||||||
|  |     if not os.path.exists(test_data_file_path): | ||||||
|  |         return None | ||||||
|  |  | ||||||
|  |     with open(test_data_file_path) as fh: | ||||||
|  |         return fh.read() | ||||||
							
								
								
									
										10
									
								
								conftest.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								conftest.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | |||||||
|  | import pytest | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def pytest_addoption(parser): | ||||||
|  |     parser.addoption("--repeat", type=int, default=1, help="repeat the operation multiple times") | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.fixture(scope="session") | ||||||
|  | def repeat(request): | ||||||
|  |     return request.config.getoption("repeat") | ||||||
| @@ -1,5 +1,5 @@ | |||||||
| [tool.black] | [tool.black] | ||||||
| target-version = ['py37'] | target-version = ['py36'] | ||||||
|  |  | ||||||
| [tool.isort] | [tool.isort] | ||||||
| multi_line_output = 3 | multi_line_output = 3 | ||||||
|   | |||||||
							
								
								
									
										5
									
								
								pytest.ini
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								pytest.ini
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | [pytest] | ||||||
|  | python_files = test_*.py | ||||||
|  | python_classes = | ||||||
|  | norecursedirs = **/output_* | ||||||
|  | addopts = -p no:warnings | ||||||
							
								
								
									
										15
									
								
								setup.py
									
									
									
									
									
								
							
							
						
						
									
										15
									
								
								setup.py
									
									
									
									
									
								
							| @@ -2,9 +2,9 @@ from setuptools import setup, find_packages | |||||||
|  |  | ||||||
| setup( | setup( | ||||||
|     name="betterproto", |     name="betterproto", | ||||||
|     version="1.1.0", |     version="1.2.4", | ||||||
|     description="A better Protobuf / gRPC generator & library", |     description="A better Protobuf / gRPC generator & library", | ||||||
|     long_description=open("README.md", "r").read(), |     long_description=open("README.md", "r", encoding="utf-8").read(), | ||||||
|     long_description_content_type="text/markdown", |     long_description_content_type="text/markdown", | ||||||
|     url="http://github.com/danielgtaylor/python-betterproto", |     url="http://github.com/danielgtaylor/python-betterproto", | ||||||
|     author="Daniel G. Taylor", |     author="Daniel G. Taylor", | ||||||
| @@ -17,8 +17,13 @@ setup( | |||||||
|         exclude=["tests", "*.tests", "*.tests.*", "output", "output.*"] |         exclude=["tests", "*.tests", "*.tests.*", "output", "output.*"] | ||||||
|     ), |     ), | ||||||
|     package_data={"betterproto": ["py.typed", "templates/template.py"]}, |     package_data={"betterproto": ["py.typed", "templates/template.py"]}, | ||||||
|     python_requires=">=3.7", |     python_requires=">=3.6", | ||||||
|     install_requires=["grpclib", "stringcase"], |     install_requires=[ | ||||||
|     extras_require={"compiler": ["jinja2", "protobuf"]}, |         'dataclasses; python_version<"3.7"', | ||||||
|  |         'backports-datetime-fromisoformat; python_version<"3.7"', | ||||||
|  |         "grpclib", | ||||||
|  |         "stringcase", | ||||||
|  |     ], | ||||||
|  |     extras_require={"compiler": ["black", "jinja2", "protobuf"]}, | ||||||
|     zip_safe=False, |     zip_safe=False, | ||||||
| ) | ) | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user