Compare commits
	
		
			8 Commits
		
	
	
		
			285-semant
			...
			v2.0.0b4
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 6dd7baa26c | ||
|  | 573c7292a6 | ||
|  | d77f44ebb7 | ||
|  | 671c0ff4ac | ||
|  | 9cecc8c3ff | ||
|  | bc3cfc5562 | ||
|  | b0a36d12e4 | ||
|  | a4d2d39546 | 
							
								
								
									
										9
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										9
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							| @@ -1,9 +0,0 @@ | |||||||
| ## Description |  | ||||||
|  |  | ||||||
| <!-- Thanks for contributing to betterproto! Add a thorough explanation of what your changes do below this line: --> |  | ||||||
|  |  | ||||||
| ## Checklist |  | ||||||
|  |  | ||||||
| - [ ] This PR targets the `rc` branch (**not** `master`). |  | ||||||
| - [ ] [If this should release a new version to PyPI when merged] The title of the PR follows the [Angular Conventional Commit](https://www.conventionalcommits.org/) syntax (`feat:` or `fix:`, with `BREAKING CHANGE:` in the commit message body if appropriate), and clearly describes the fix or feature. |  | ||||||
| - [ ] Documentation is updated (`README.md` and docstrings). |  | ||||||
							
								
								
									
										4
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							| @@ -15,7 +15,7 @@ jobs: | |||||||
|     strategy: |     strategy: | ||||||
|       matrix: |       matrix: | ||||||
|         os: [Ubuntu, MacOS, Windows] |         os: [Ubuntu, MacOS, Windows] | ||||||
|         python-version: [3.6, 3.7, 3.8, 3.9] |         python-version: ['3.6.7', '3.7', '3.8', '3.9', '3.10'] | ||||||
|         exclude: |         exclude: | ||||||
|           - os: Windows |           - os: Windows | ||||||
|             python-version: 3.6 |             python-version: 3.6 | ||||||
| @@ -66,4 +66,4 @@ jobs: | |||||||
|  |  | ||||||
|       - name: Execute test suite |       - name: Execute test suite | ||||||
|         shell: bash |         shell: bash | ||||||
|         run: poetry run pytest tests/ |         run: poetry run python -m pytest tests/ | ||||||
|   | |||||||
							
								
								
									
										6
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							| @@ -16,18 +16,12 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v2 |       - uses: actions/checkout@v2 | ||||||
|         with: |  | ||||||
|           persist-credentials: false |  | ||||||
|       - name: Set up Python 3.8 |       - name: Set up Python 3.8 | ||||||
|         uses: actions/setup-python@v2 |         uses: actions/setup-python@v2 | ||||||
|         with: |         with: | ||||||
|           python-version: 3.8 |           python-version: 3.8 | ||||||
|       - name: Install poetry |       - name: Install poetry | ||||||
|         run: python -m pip install poetry |         run: python -m pip install poetry | ||||||
|       - name: Semantic Release |  | ||||||
|         uses: cycjimmy/semantic-release-action@v2 |  | ||||||
|         env: |  | ||||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} |  | ||||||
|       - name: Build package |       - name: Build package | ||||||
|         run: poetry build |         run: poetry build | ||||||
|       - name: Publish package to PyPI |       - name: Publish package to PyPI | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -6,7 +6,6 @@ | |||||||
| .pytest_cache | .pytest_cache | ||||||
| .python-version | .python-version | ||||||
| build/ | build/ | ||||||
| node_modules/ |  | ||||||
| tests/output_* | tests/output_* | ||||||
| **/__pycache__ | **/__pycache__ | ||||||
| dist | dist | ||||||
|   | |||||||
							
								
								
									
										13
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										13
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -7,6 +7,17 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 | |||||||
|  |  | ||||||
| - Versions suffixed with `b*` are in `beta` and can be installed with `pip install --pre betterproto`. | - Versions suffixed with `b*` are in `beta` and can be installed with `pip install --pre betterproto`. | ||||||
|  |  | ||||||
|  | ## [2.0.0b4] - 2022-01-03 | ||||||
|  |  | ||||||
|  | - **Breaking**: the minimum Python version has been bumped to `3.6.2` | ||||||
|  |  | ||||||
|  | - Always add `AsyncIterator` to imports if there are services [#264](https://github.com/danielgtaylor/python-betterproto/pull/264) | ||||||
|  | - Allow parsing of messages from `ByteStrings` [#266](https://github.com/danielgtaylor/python-betterproto/pull/266) | ||||||
|  | - Add support for proto3 optional [#281](https://github.com/danielgtaylor/python-betterproto/pull/281) | ||||||
|  |  | ||||||
|  | - Fix compilation of fields with names identical to builtin types [#294](https://github.com/danielgtaylor/python-betterproto/pull/294) | ||||||
|  | - Fix default values for enum service args [#299](https://github.com/danielgtaylor/python-betterproto/pull/299) | ||||||
|  |  | ||||||
| ## [2.0.0b3] - 2021-04-07 | ## [2.0.0b3] - 2021-04-07 | ||||||
|  |  | ||||||
| - Generate grpclib service stubs [#170](https://github.com/danielgtaylor/python-betterproto/pull/170) | - Generate grpclib service stubs [#170](https://github.com/danielgtaylor/python-betterproto/pull/170) | ||||||
| @@ -54,7 +65,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 | |||||||
|  |  | ||||||
| ## [2.0.0b1] - 2020-07-04 | ## [2.0.0b1] - 2020-07-04 | ||||||
|  |  | ||||||
| [Upgrade Guide](./docs/upgrading.md)  | [Upgrade Guide](./docs/upgrading.md) | ||||||
|  |  | ||||||
| > Several bugfixes and improvements required or will require small breaking changes, necessitating a new version. | > Several bugfixes and improvements required or will require small breaking changes, necessitating a new version. | ||||||
| > `2.0.0` will be released once the interface is stable. | > `2.0.0` will be released once the interface is stable. | ||||||
|   | |||||||
							
								
								
									
										10
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										10
									
								
								README.md
									
									
									
									
									
								
							| @@ -498,16 +498,6 @@ protoc \ | |||||||
| - [x] Automate running tests | - [x] Automate running tests | ||||||
| - [ ] Cleanup! | - [ ] Cleanup! | ||||||
|  |  | ||||||
|  |  | ||||||
| ## Release |  | ||||||
|  |  | ||||||
| New versions are versioned and released using [Semantic Release](https://github.com/semantic-release/semantic-release). When new commits |  | ||||||
| using the Angular Conventional Commits syntax land on `master`  or `rc`, those commits are used to determine what new version to release. |  | ||||||
|  |  | ||||||
| All Pull Requests must target the `rc` branch; when merged into `rc` they will publish new release candidate (`rc`) versions to PyPI |  | ||||||
| automatically. When maintainers want to publish a new full release, they simply merge `rc` into `master`. This flow ensures that features |  | ||||||
| and fixes are published quickly and continuously rather than awaiting a manual release process. |  | ||||||
|  |  | ||||||
| ## Community | ## Community | ||||||
|  |  | ||||||
| Join us on [Slack](https://join.slack.com/t/betterproto/shared_invite/zt-f0n0uolx-iN8gBNrkPxtKHTLpG3o1OQ)! | Join us on [Slack](https://join.slack.com/t/betterproto/shared_invite/zt-f0n0uolx-iN8gBNrkPxtKHTLpG3o1OQ)! | ||||||
|   | |||||||
							
								
								
									
										52
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										52
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,52 +0,0 @@ | |||||||
| { |  | ||||||
|     "name": "python-betterproto-semantic-release", |  | ||||||
|     "version": "1.0.0", |  | ||||||
|     "description": "Encapsulate dependencies needed to use semantic-release", |  | ||||||
|     "dependencies": { |  | ||||||
|         "@semantic-release/exec": "^5.0.0", |  | ||||||
|         "@semantic-release/git": "^9.0.0", |  | ||||||
|         "@semantic-release/gitlab": "^6.0.4", |  | ||||||
|         "conventional-changelog-eslint": "^3.0.8", |  | ||||||
|         "semantic-release": "^17.1.1" |  | ||||||
|     }, |  | ||||||
|     "release": { |  | ||||||
|         "branches": [ |  | ||||||
|             "master", |  | ||||||
|             { |  | ||||||
|                 "name": "rc", |  | ||||||
|                 "prerelease": true |  | ||||||
|             } |  | ||||||
|         ], |  | ||||||
|         "plugins": [ |  | ||||||
|             [ |  | ||||||
|                 "@semantic-release/commit-analyzer", |  | ||||||
|                 { |  | ||||||
|                     "preset": "angular" |  | ||||||
|                 } |  | ||||||
|             ], |  | ||||||
|             [ |  | ||||||
|                 "@semantic-release/release-notes-generator", |  | ||||||
|                 { |  | ||||||
|                     "preset": "angular" |  | ||||||
|                 } |  | ||||||
|             ], |  | ||||||
|             [ |  | ||||||
|                 "@semantic-release/exec", |  | ||||||
|                 { |  | ||||||
|                     "prepareCmd": "poetry version ${nextRelease.version}" |  | ||||||
|                 } |  | ||||||
|             ], |  | ||||||
|             "@semantic-release/github", |  | ||||||
|             [ |  | ||||||
|                 "@semantic-release/git", |  | ||||||
|                 { |  | ||||||
|                     "assets": [ |  | ||||||
|                         "pyproject.toml" |  | ||||||
|                     ], |  | ||||||
|                     "message": "Release v${nextRelease.version} [skip ci]" |  | ||||||
|                 } |  | ||||||
|             ] |  | ||||||
|         ], |  | ||||||
|         "repositoryUrl": "ssh://git@github.com/danielgtaylor/python-betterproto.git" |  | ||||||
|     } |  | ||||||
| } |  | ||||||
							
								
								
									
										1330
									
								
								poetry.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										1330
									
								
								poetry.lock
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,6 +1,6 @@ | |||||||
| [tool.poetry] | [tool.poetry] | ||||||
| name = "betterproto" | name = "betterproto" | ||||||
| version = "2.0.0b3" | version = "2.0.0b4" | ||||||
| description = "A better Protobuf / gRPC generator & library" | description = "A better Protobuf / gRPC generator & library" | ||||||
| authors = ["Daniel G. Taylor <danielgtaylor@gmail.com>"] | authors = ["Daniel G. Taylor <danielgtaylor@gmail.com>"] | ||||||
| readme = "README.md" | readme = "README.md" | ||||||
| @@ -12,7 +12,7 @@ packages = [ | |||||||
| ] | ] | ||||||
|  |  | ||||||
| [tool.poetry.dependencies] | [tool.poetry.dependencies] | ||||||
| python = "^3.6" | python = ">=3.6.2,<4.0" | ||||||
| black = { version = ">=19.3b0", optional = true } | black = { version = ">=19.3b0", optional = true } | ||||||
| dataclasses = { version = "^0.7", python = ">=3.6, <3.7" } | dataclasses = { version = "^0.7", python = ">=3.6, <3.7" } | ||||||
| grpclib = "^0.4.1" | grpclib = "^0.4.1" | ||||||
| @@ -21,14 +21,14 @@ python-dateutil = "^2.8" | |||||||
|  |  | ||||||
| [tool.poetry.dev-dependencies] | [tool.poetry.dev-dependencies] | ||||||
| asv = "^0.4.2" | asv = "^0.4.2" | ||||||
| black = "^20.8b1" | black = "^21.11b0" | ||||||
| bpython = "^0.19" | bpython = "^0.19" | ||||||
| grpcio-tools = "^1.30.0" | grpcio-tools = "^1.40.0" | ||||||
| jinja2 = "^2.11.2" | jinja2 = "^2.11.2" | ||||||
| mypy = "^0.770" | mypy = "^0.930" | ||||||
| poethepoet = ">=0.9.0" | poethepoet = ">=0.9.0" | ||||||
| protobuf = "^3.12.2" | protobuf = "^3.12.2" | ||||||
| pytest = "^5.4.2" | pytest = "^6.2.5" | ||||||
| pytest-asyncio = "^0.12.0" | pytest-asyncio = "^0.12.0" | ||||||
| pytest-cov = "^2.9.0" | pytest-cov = "^2.9.0" | ||||||
| pytest-mock = "^3.1.1" | pytest-mock = "^3.1.1" | ||||||
|   | |||||||
| @@ -145,6 +145,8 @@ class FieldMetadata: | |||||||
|     group: Optional[str] = None |     group: Optional[str] = None | ||||||
|     # Describes the wrapped type (e.g. when using google.protobuf.BoolValue) |     # Describes the wrapped type (e.g. when using google.protobuf.BoolValue) | ||||||
|     wraps: Optional[str] = None |     wraps: Optional[str] = None | ||||||
|  |     # Is the field optional | ||||||
|  |     optional: Optional[bool] = False | ||||||
|  |  | ||||||
|     @staticmethod |     @staticmethod | ||||||
|     def get(field: dataclasses.Field) -> "FieldMetadata": |     def get(field: dataclasses.Field) -> "FieldMetadata": | ||||||
| @@ -159,12 +161,15 @@ def dataclass_field( | |||||||
|     map_types: Optional[Tuple[str, str]] = None, |     map_types: Optional[Tuple[str, str]] = None, | ||||||
|     group: Optional[str] = None, |     group: Optional[str] = None, | ||||||
|     wraps: Optional[str] = None, |     wraps: Optional[str] = None, | ||||||
|  |     optional: bool = False, | ||||||
| ) -> dataclasses.Field: | ) -> dataclasses.Field: | ||||||
|     """Creates a dataclass field with attached protobuf metadata.""" |     """Creates a dataclass field with attached protobuf metadata.""" | ||||||
|     return dataclasses.field( |     return dataclasses.field( | ||||||
|         default=PLACEHOLDER, |         default=None if optional else PLACEHOLDER, | ||||||
|         metadata={ |         metadata={ | ||||||
|             "betterproto": FieldMetadata(number, proto_type, map_types, group, wraps) |             "betterproto": FieldMetadata( | ||||||
|  |                 number, proto_type, map_types, group, wraps, optional | ||||||
|  |             ) | ||||||
|         }, |         }, | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
| @@ -174,74 +179,107 @@ def dataclass_field( | |||||||
| # out at runtime. The generated dataclass variables are still typed correctly. | # out at runtime. The generated dataclass variables are still typed correctly. | ||||||
|  |  | ||||||
|  |  | ||||||
| def enum_field(number: int, group: Optional[str] = None) -> Any: | def enum_field(number: int, group: Optional[str] = None, optional: bool = False) -> Any: | ||||||
|     return dataclass_field(number, TYPE_ENUM, group=group) |     return dataclass_field(number, TYPE_ENUM, group=group, optional=optional) | ||||||
|  |  | ||||||
|  |  | ||||||
| def bool_field(number: int, group: Optional[str] = None) -> Any: | def bool_field(number: int, group: Optional[str] = None, optional: bool = False) -> Any: | ||||||
|     return dataclass_field(number, TYPE_BOOL, group=group) |     return dataclass_field(number, TYPE_BOOL, group=group, optional=optional) | ||||||
|  |  | ||||||
|  |  | ||||||
| def int32_field(number: int, group: Optional[str] = None) -> Any: | def int32_field( | ||||||
|     return dataclass_field(number, TYPE_INT32, group=group) |     number: int, group: Optional[str] = None, optional: bool = False | ||||||
|  | ) -> Any: | ||||||
|  |     return dataclass_field(number, TYPE_INT32, group=group, optional=optional) | ||||||
|  |  | ||||||
|  |  | ||||||
| def int64_field(number: int, group: Optional[str] = None) -> Any: | def int64_field( | ||||||
|     return dataclass_field(number, TYPE_INT64, group=group) |     number: int, group: Optional[str] = None, optional: bool = False | ||||||
|  | ) -> Any: | ||||||
|  |     return dataclass_field(number, TYPE_INT64, group=group, optional=optional) | ||||||
|  |  | ||||||
|  |  | ||||||
| def uint32_field(number: int, group: Optional[str] = None) -> Any: | def uint32_field( | ||||||
|     return dataclass_field(number, TYPE_UINT32, group=group) |     number: int, group: Optional[str] = None, optional: bool = False | ||||||
|  | ) -> Any: | ||||||
|  |     return dataclass_field(number, TYPE_UINT32, group=group, optional=optional) | ||||||
|  |  | ||||||
|  |  | ||||||
| def uint64_field(number: int, group: Optional[str] = None) -> Any: | def uint64_field( | ||||||
|     return dataclass_field(number, TYPE_UINT64, group=group) |     number: int, group: Optional[str] = None, optional: bool = False | ||||||
|  | ) -> Any: | ||||||
|  |     return dataclass_field(number, TYPE_UINT64, group=group, optional=optional) | ||||||
|  |  | ||||||
|  |  | ||||||
| def sint32_field(number: int, group: Optional[str] = None) -> Any: | def sint32_field( | ||||||
|     return dataclass_field(number, TYPE_SINT32, group=group) |     number: int, group: Optional[str] = None, optional: bool = False | ||||||
|  | ) -> Any: | ||||||
|  |     return dataclass_field(number, TYPE_SINT32, group=group, optional=optional) | ||||||
|  |  | ||||||
|  |  | ||||||
| def sint64_field(number: int, group: Optional[str] = None) -> Any: | def sint64_field( | ||||||
|     return dataclass_field(number, TYPE_SINT64, group=group) |     number: int, group: Optional[str] = None, optional: bool = False | ||||||
|  | ) -> Any: | ||||||
|  |     return dataclass_field(number, TYPE_SINT64, group=group, optional=optional) | ||||||
|  |  | ||||||
|  |  | ||||||
| def float_field(number: int, group: Optional[str] = None) -> Any: | def float_field( | ||||||
|     return dataclass_field(number, TYPE_FLOAT, group=group) |     number: int, group: Optional[str] = None, optional: bool = False | ||||||
|  | ) -> Any: | ||||||
|  |     return dataclass_field(number, TYPE_FLOAT, group=group, optional=optional) | ||||||
|  |  | ||||||
|  |  | ||||||
| def double_field(number: int, group: Optional[str] = None) -> Any: | def double_field( | ||||||
|     return dataclass_field(number, TYPE_DOUBLE, group=group) |     number: int, group: Optional[str] = None, optional: bool = False | ||||||
|  | ) -> Any: | ||||||
|  |     return dataclass_field(number, TYPE_DOUBLE, group=group, optional=optional) | ||||||
|  |  | ||||||
|  |  | ||||||
| def fixed32_field(number: int, group: Optional[str] = None) -> Any: | def fixed32_field( | ||||||
|     return dataclass_field(number, TYPE_FIXED32, group=group) |     number: int, group: Optional[str] = None, optional: bool = False | ||||||
|  | ) -> Any: | ||||||
|  |     return dataclass_field(number, TYPE_FIXED32, group=group, optional=optional) | ||||||
|  |  | ||||||
|  |  | ||||||
| def fixed64_field(number: int, group: Optional[str] = None) -> Any: | def fixed64_field( | ||||||
|     return dataclass_field(number, TYPE_FIXED64, group=group) |     number: int, group: Optional[str] = None, optional: bool = False | ||||||
|  | ) -> Any: | ||||||
|  |     return dataclass_field(number, TYPE_FIXED64, group=group, optional=optional) | ||||||
|  |  | ||||||
|  |  | ||||||
| def sfixed32_field(number: int, group: Optional[str] = None) -> Any: | def sfixed32_field( | ||||||
|     return dataclass_field(number, TYPE_SFIXED32, group=group) |     number: int, group: Optional[str] = None, optional: bool = False | ||||||
|  | ) -> Any: | ||||||
|  |     return dataclass_field(number, TYPE_SFIXED32, group=group, optional=optional) | ||||||
|  |  | ||||||
|  |  | ||||||
| def sfixed64_field(number: int, group: Optional[str] = None) -> Any: | def sfixed64_field( | ||||||
|     return dataclass_field(number, TYPE_SFIXED64, group=group) |     number: int, group: Optional[str] = None, optional: bool = False | ||||||
|  | ) -> Any: | ||||||
|  |     return dataclass_field(number, TYPE_SFIXED64, group=group, optional=optional) | ||||||
|  |  | ||||||
|  |  | ||||||
| def string_field(number: int, group: Optional[str] = None) -> Any: | def string_field( | ||||||
|     return dataclass_field(number, TYPE_STRING, group=group) |     number: int, group: Optional[str] = None, optional: bool = False | ||||||
|  | ) -> Any: | ||||||
|  |     return dataclass_field(number, TYPE_STRING, group=group, optional=optional) | ||||||
|  |  | ||||||
|  |  | ||||||
| def bytes_field(number: int, group: Optional[str] = None) -> Any: | def bytes_field( | ||||||
|     return dataclass_field(number, TYPE_BYTES, group=group) |     number: int, group: Optional[str] = None, optional: bool = False | ||||||
|  | ) -> Any: | ||||||
|  |     return dataclass_field(number, TYPE_BYTES, group=group, optional=optional) | ||||||
|  |  | ||||||
|  |  | ||||||
| def message_field( | def message_field( | ||||||
|     number: int, group: Optional[str] = None, wraps: Optional[str] = None |     number: int, | ||||||
|  |     group: Optional[str] = None, | ||||||
|  |     wraps: Optional[str] = None, | ||||||
|  |     optional: bool = False, | ||||||
| ) -> Any: | ) -> Any: | ||||||
|     return dataclass_field(number, TYPE_MESSAGE, group=group, wraps=wraps) |     return dataclass_field( | ||||||
|  |         number, TYPE_MESSAGE, group=group, wraps=wraps, optional=optional | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
| def map_field( | def map_field( | ||||||
| @@ -586,7 +624,8 @@ class Message(ABC): | |||||||
|             if meta.group: |             if meta.group: | ||||||
|                 group_current.setdefault(meta.group) |                 group_current.setdefault(meta.group) | ||||||
|  |  | ||||||
|             if self.__raw_get(field_name) != PLACEHOLDER: |             value = self.__raw_get(field_name) | ||||||
|  |             if value != PLACEHOLDER and not (meta.optional and value is None): | ||||||
|                 # Found a non-sentinel value |                 # Found a non-sentinel value | ||||||
|                 all_sentinel = False |                 all_sentinel = False | ||||||
|  |  | ||||||
| @@ -701,12 +740,16 @@ class Message(ABC): | |||||||
|  |  | ||||||
|             if value is None: |             if value is None: | ||||||
|                 # Optional items should be skipped. This is used for the Google |                 # Optional items should be skipped. This is used for the Google | ||||||
|                 # wrapper types. |                 # wrapper types and proto3 field presence/optional fields. | ||||||
|                 continue |                 continue | ||||||
|  |  | ||||||
|             # Being selected in a a group means this field is the one that is |             # Being selected in a a group means this field is the one that is | ||||||
|             # 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. | ||||||
|  |             # | ||||||
|  |             # Note that proto3 field presence/optional fields are put in a | ||||||
|  |             # synthetic single-item oneof by protoc, which helps us ensure we | ||||||
|  |             # send the value even if the value is the default zero value. | ||||||
|             selected_in_group = ( |             selected_in_group = ( | ||||||
|                 meta.group and self._group_current[meta.group] == field_name |                 meta.group and self._group_current[meta.group] == field_name | ||||||
|             ) |             ) | ||||||
| @@ -803,7 +846,7 @@ class Message(ABC): | |||||||
|     @classmethod |     @classmethod | ||||||
|     def _type_hints(cls) -> Dict[str, Type]: |     def _type_hints(cls) -> Dict[str, Type]: | ||||||
|         module = sys.modules[cls.__module__] |         module = sys.modules[cls.__module__] | ||||||
|         return get_type_hints(cls, vars(module)) |         return get_type_hints(cls, module.__dict__, {}) | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def _cls_for(cls, field: dataclasses.Field, index: int = 0) -> Type: |     def _cls_for(cls, field: dataclasses.Field, index: int = 0) -> Type: | ||||||
| @@ -829,8 +872,9 @@ class Message(ABC): | |||||||
|                 # This is some kind of list (repeated) field. |                 # This is some kind of list (repeated) field. | ||||||
|                 return list |                 return list | ||||||
|             elif t.__origin__ is Union and t.__args__[1] is type(None): |             elif t.__origin__ is Union and t.__args__[1] is type(None): | ||||||
|                 # This is an optional (wrapped) field. For setting the default we |                 # This is an optional field (either wrapped, or using proto3 | ||||||
|                 # really don't care what kind of field it is. |                 # field presence). For setting the default we really don't care | ||||||
|  |                 # what kind of field it is. | ||||||
|                 return type(None) |                 return type(None) | ||||||
|             else: |             else: | ||||||
|                 return t |                 return t | ||||||
| @@ -1041,6 +1085,9 @@ class Message(ABC): | |||||||
|                         ] |                         ] | ||||||
|                     if value or include_default_values: |                     if value or include_default_values: | ||||||
|                         output[cased_name] = value |                         output[cased_name] = value | ||||||
|  |                 elif value is None: | ||||||
|  |                     if include_default_values: | ||||||
|  |                         output[cased_name] = value | ||||||
|                 elif ( |                 elif ( | ||||||
|                     value._serialized_on_wire |                     value._serialized_on_wire | ||||||
|                     or include_default_values |                     or include_default_values | ||||||
| @@ -1066,6 +1113,9 @@ class Message(ABC): | |||||||
|                 if meta.proto_type in INT_64_TYPES: |                 if meta.proto_type in INT_64_TYPES: | ||||||
|                     if field_is_repeated: |                     if field_is_repeated: | ||||||
|                         output[cased_name] = [str(n) for n in value] |                         output[cased_name] = [str(n) for n in value] | ||||||
|  |                     elif value is None: | ||||||
|  |                         if include_default_values: | ||||||
|  |                             output[cased_name] = value | ||||||
|                     else: |                     else: | ||||||
|                         output[cased_name] = str(value) |                         output[cased_name] = str(value) | ||||||
|                 elif meta.proto_type == TYPE_BYTES: |                 elif meta.proto_type == TYPE_BYTES: | ||||||
| @@ -1073,6 +1123,8 @@ class Message(ABC): | |||||||
|                         output[cased_name] = [ |                         output[cased_name] = [ | ||||||
|                             b64encode(b).decode("utf8") for b in value |                             b64encode(b).decode("utf8") for b in value | ||||||
|                         ] |                         ] | ||||||
|  |                     elif value is None and include_default_values: | ||||||
|  |                         output[cased_name] = value | ||||||
|                     else: |                     else: | ||||||
|                         output[cased_name] = b64encode(value).decode("utf8") |                         output[cased_name] = b64encode(value).decode("utf8") | ||||||
|                 elif meta.proto_type == TYPE_ENUM: |                 elif meta.proto_type == TYPE_ENUM: | ||||||
| @@ -1085,6 +1137,12 @@ class Message(ABC): | |||||||
|                         else: |                         else: | ||||||
|                             # transparently upgrade single value to repeated |                             # transparently upgrade single value to repeated | ||||||
|                             output[cased_name] = [enum_class(value).name] |                             output[cased_name] = [enum_class(value).name] | ||||||
|  |                     elif value is None: | ||||||
|  |                         if include_default_values: | ||||||
|  |                             output[cased_name] = value | ||||||
|  |                     elif meta.optional: | ||||||
|  |                         enum_class = field_types[field_name].__args__[0] | ||||||
|  |                         output[cased_name] = enum_class(value).name | ||||||
|                     else: |                     else: | ||||||
|                         enum_class = field_types[field_name]  # noqa |                         enum_class = field_types[field_name]  # noqa | ||||||
|                         output[cased_name] = enum_class(value).name |                         output[cased_name] = enum_class(value).name | ||||||
| @@ -1141,6 +1199,9 @@ class Message(ABC): | |||||||
|                         setattr(self, field_name, v) |                         setattr(self, field_name, v) | ||||||
|                     elif meta.wraps: |                     elif meta.wraps: | ||||||
|                         setattr(self, field_name, value[key]) |                         setattr(self, field_name, value[key]) | ||||||
|  |                     elif v is None: | ||||||
|  |                         cls = self._betterproto.cls_by_field[field_name] | ||||||
|  |                         setattr(self, field_name, cls().from_dict(value[key])) | ||||||
|                     else: |                     else: | ||||||
|                         # NOTE: `from_dict` mutates the underlying message, so no |                         # NOTE: `from_dict` mutates the underlying message, so no | ||||||
|                         # assignment here is necessary. |                         # assignment here is necessary. | ||||||
|   | |||||||
| @@ -133,16 +133,6 @@ def lowercase_first(value: str) -> str: | |||||||
|     return value[0:1].lower() + value[1:] |     return value[0:1].lower() + value[1:] | ||||||
|  |  | ||||||
|  |  | ||||||
| def is_reserved_name(value: str) -> bool: |  | ||||||
|     if keyword.iskeyword(value): |  | ||||||
|         return True |  | ||||||
|  |  | ||||||
|     if value in ("bytes", "str"): |  | ||||||
|         return True |  | ||||||
|  |  | ||||||
|     return False |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def sanitize_name(value: str) -> str: | def sanitize_name(value: str) -> str: | ||||||
|     # https://www.python.org/dev/peps/pep-0008/#descriptive-naming-styles |     # https://www.python.org/dev/peps/pep-0008/#descriptive-naming-styles | ||||||
|     return f"{value}_" if is_reserved_name(value) else value |     return f"{value}_" if keyword.iskeyword(value) else value | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ from dataclasses import dataclass | |||||||
| from typing import Dict, List | from typing import Dict, List | ||||||
|  |  | ||||||
| import betterproto | import betterproto | ||||||
|  | from betterproto.grpc.grpclib_server import ServiceBase | ||||||
|  |  | ||||||
|  |  | ||||||
| class Syntax(betterproto.Enum): | class Syntax(betterproto.Enum): | ||||||
| @@ -46,17 +47,6 @@ class FieldCardinality(betterproto.Enum): | |||||||
|     CARDINALITY_REPEATED = 3 |     CARDINALITY_REPEATED = 3 | ||||||
|  |  | ||||||
|  |  | ||||||
| class NullValue(betterproto.Enum): |  | ||||||
|     """ |  | ||||||
|     `NullValue` is a singleton enumeration to represent the null value for the |  | ||||||
|     `Value` type union.  The JSON representation for `NullValue` is JSON |  | ||||||
|     `null`. |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     # Null value. |  | ||||||
|     NULL_VALUE = 0 |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class FieldDescriptorProtoType(betterproto.Enum): | class FieldDescriptorProtoType(betterproto.Enum): | ||||||
|     TYPE_DOUBLE = 1 |     TYPE_DOUBLE = 1 | ||||||
|     TYPE_FLOAT = 2 |     TYPE_FLOAT = 2 | ||||||
| @@ -108,165 +98,15 @@ class MethodOptionsIdempotencyLevel(betterproto.Enum): | |||||||
|     IDEMPOTENT = 2 |     IDEMPOTENT = 2 | ||||||
|  |  | ||||||
|  |  | ||||||
| @dataclass(eq=False, repr=False) | class NullValue(betterproto.Enum): | ||||||
| class Timestamp(betterproto.Message): |  | ||||||
|     """ |     """ | ||||||
|     A Timestamp represents a point in time independent of any time zone or |     `NullValue` is a singleton enumeration to represent the null value for the | ||||||
|     local calendar, encoded as a count of seconds and fractions of seconds at |     `Value` type union.  The JSON representation for `NullValue` is JSON | ||||||
|     nanosecond resolution. The count is relative to an epoch at UTC midnight on |     `null`. | ||||||
|     January 1, 1970, in the proleptic Gregorian calendar which extends the |  | ||||||
|     Gregorian calendar backwards to year one. All minutes are 60 seconds long. |  | ||||||
|     Leap seconds are "smeared" so that no leap second table is needed for |  | ||||||
|     interpretation, using a [24-hour linear |  | ||||||
|     smear](https://developers.google.com/time/smear). The range is from |  | ||||||
|     0001-01-01T00:00:00Z to 9999-12-31T23:59:59.999999999Z. By restricting to |  | ||||||
|     that range, we ensure that we can convert to and from [RFC |  | ||||||
|     3339](https://www.ietf.org/rfc/rfc3339.txt) date strings. # Examples |  | ||||||
|     Example 1: Compute Timestamp from POSIX `time()`.     Timestamp timestamp; |  | ||||||
|     timestamp.set_seconds(time(NULL));     timestamp.set_nanos(0); Example 2: |  | ||||||
|     Compute Timestamp from POSIX `gettimeofday()`.     struct timeval tv; |  | ||||||
|     gettimeofday(&tv, NULL);     Timestamp timestamp; |  | ||||||
|     timestamp.set_seconds(tv.tv_sec);     timestamp.set_nanos(tv.tv_usec * |  | ||||||
|     1000); Example 3: Compute Timestamp from Win32 `GetSystemTimeAsFileTime()`. |  | ||||||
|     FILETIME ft;     GetSystemTimeAsFileTime(&ft);     UINT64 ticks = |  | ||||||
|     (((UINT64)ft.dwHighDateTime) << 32) | ft.dwLowDateTime;     // A Windows |  | ||||||
|     tick is 100 nanoseconds. Windows epoch 1601-01-01T00:00:00Z     // is |  | ||||||
|     11644473600 seconds before Unix epoch 1970-01-01T00:00:00Z.     Timestamp |  | ||||||
|     timestamp;     timestamp.set_seconds((INT64) ((ticks / 10000000) - |  | ||||||
|     11644473600LL));     timestamp.set_nanos((INT32) ((ticks % 10000000) * |  | ||||||
|     100)); Example 4: Compute Timestamp from Java `System.currentTimeMillis()`. |  | ||||||
|     long millis = System.currentTimeMillis();     Timestamp timestamp = |  | ||||||
|     Timestamp.newBuilder().setSeconds(millis / 1000)         .setNanos((int) |  | ||||||
|     ((millis % 1000) * 1000000)).build(); Example 5: Compute Timestamp from |  | ||||||
|     current time in Python.     timestamp = Timestamp() |  | ||||||
|     timestamp.GetCurrentTime() # JSON Mapping In JSON format, the Timestamp |  | ||||||
|     type is encoded as a string in the [RFC |  | ||||||
|     3339](https://www.ietf.org/rfc/rfc3339.txt) format. That is, the format is |  | ||||||
|     "{year}-{month}-{day}T{hour}:{min}:{sec}[.{frac_sec}]Z" where {year} is |  | ||||||
|     always expressed using four digits while {month}, {day}, {hour}, {min}, and |  | ||||||
|     {sec} are zero-padded to two digits each. The fractional seconds, which can |  | ||||||
|     go up to 9 digits (i.e. up to 1 nanosecond resolution), are optional. The |  | ||||||
|     "Z" suffix indicates the timezone ("UTC"); the timezone is required. A |  | ||||||
|     proto3 JSON serializer should always use UTC (as indicated by "Z") when |  | ||||||
|     printing the Timestamp type and a proto3 JSON parser should be able to |  | ||||||
|     accept both UTC and other timezones (as indicated by an offset). For |  | ||||||
|     example, "2017-01-15T01:30:15.01Z" encodes 15.01 seconds past 01:30 UTC on |  | ||||||
|     January 15, 2017. In JavaScript, one can convert a Date object to this |  | ||||||
|     format using the standard [toISOString()](https://developer.mozilla.org/en- |  | ||||||
|     US/docs/Web/JavaScript/Reference/Global_Objects/Date/toISOString) method. |  | ||||||
|     In Python, a standard `datetime.datetime` object can be converted to this |  | ||||||
|     format using |  | ||||||
|     [`strftime`](https://docs.python.org/2/library/time.html#time.strftime) |  | ||||||
|     with the time format spec '%Y-%m-%dT%H:%M:%S.%fZ'. Likewise, in Java, one |  | ||||||
|     can use the Joda Time's [`ISODateTimeFormat.dateTime()`]( |  | ||||||
|     http://www.joda.org/joda- |  | ||||||
|     time/apidocs/org/joda/time/format/ISODateTimeFormat.html#dateTime%2D%2D ) |  | ||||||
|     to obtain a formatter capable of generating timestamps in this format. |  | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     # Represents seconds of UTC time since Unix epoch 1970-01-01T00:00:00Z. Must |     # Null value. | ||||||
|     # be from 0001-01-01T00:00:00Z to 9999-12-31T23:59:59Z inclusive. |     NULL_VALUE = 0 | ||||||
|     seconds: int = betterproto.int64_field(1) |  | ||||||
|     # Non-negative fractions of a second at nanosecond resolution. Negative |  | ||||||
|     # second values with fractions must still have non-negative nanos values that |  | ||||||
|     # count forward in time. Must be from 0 to 999,999,999 inclusive. |  | ||||||
|     nanos: int = betterproto.int32_field(2) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @dataclass(eq=False, repr=False) |  | ||||||
| class FieldMask(betterproto.Message): |  | ||||||
|     """ |  | ||||||
|     `FieldMask` represents a set of symbolic field paths, for example: |  | ||||||
|     paths: "f.a"     paths: "f.b.d" Here `f` represents a field in some root |  | ||||||
|     message, `a` and `b` fields in the message found in `f`, and `d` a field |  | ||||||
|     found in the message in `f.b`. Field masks are used to specify a subset of |  | ||||||
|     fields that should be returned by a get operation or modified by an update |  | ||||||
|     operation. Field masks also have a custom JSON encoding (see below). # |  | ||||||
|     Field Masks in Projections When used in the context of a projection, a |  | ||||||
|     response message or sub-message is filtered by the API to only contain |  | ||||||
|     those fields as specified in the mask. For example, if the mask in the |  | ||||||
|     previous example is applied to a response message as follows:     f { |  | ||||||
|     a : 22       b {         d : 1         x : 2       }       y : 13     } |  | ||||||
|     z: 8 The result will not contain specific values for fields x,y and z |  | ||||||
|     (their value will be set to the default, and omitted in proto text output): |  | ||||||
|     f {       a : 22       b {         d : 1       }     } A repeated field is |  | ||||||
|     not allowed except at the last position of a paths string. If a FieldMask |  | ||||||
|     object is not present in a get operation, the operation applies to all |  | ||||||
|     fields (as if a FieldMask of all fields had been specified). Note that a |  | ||||||
|     field mask does not necessarily apply to the top-level response message. In |  | ||||||
|     case of a REST get operation, the field mask applies directly to the |  | ||||||
|     response, but in case of a REST list operation, the mask instead applies to |  | ||||||
|     each individual message in the returned resource list. In case of a REST |  | ||||||
|     custom method, other definitions may be used. Where the mask applies will |  | ||||||
|     be clearly documented together with its declaration in the API.  In any |  | ||||||
|     case, the effect on the returned resource/resources is required behavior |  | ||||||
|     for APIs. # Field Masks in Update Operations A field mask in update |  | ||||||
|     operations specifies which fields of the targeted resource are going to be |  | ||||||
|     updated. The API is required to only change the values of the fields as |  | ||||||
|     specified in the mask and leave the others untouched. If a resource is |  | ||||||
|     passed in to describe the updated values, the API ignores the values of all |  | ||||||
|     fields not covered by the mask. If a repeated field is specified for an |  | ||||||
|     update operation, new values will be appended to the existing repeated |  | ||||||
|     field in the target resource. Note that a repeated field is only allowed in |  | ||||||
|     the last position of a `paths` string. If a sub-message is specified in the |  | ||||||
|     last position of the field mask for an update operation, then new value |  | ||||||
|     will be merged into the existing sub-message in the target resource. For |  | ||||||
|     example, given the target message:     f {       b {         d: 1 |  | ||||||
|     x: 2       }       c: [1]     } And an update message:     f {       b { |  | ||||||
|     d: 10       }       c: [2]     } then if the field mask is:  paths: ["f.b", |  | ||||||
|     "f.c"] then the result will be:     f {       b {         d: 10         x: |  | ||||||
|     2       }       c: [1, 2]     } An implementation may provide options to |  | ||||||
|     override this default behavior for repeated and message fields. In order to |  | ||||||
|     reset a field's value to the default, the field must be in the mask and set |  | ||||||
|     to the default value in the provided resource. Hence, in order to reset all |  | ||||||
|     fields of a resource, provide a default instance of the resource and set |  | ||||||
|     all fields in the mask, or do not provide a mask as described below. If a |  | ||||||
|     field mask is not present on update, the operation applies to all fields |  | ||||||
|     (as if a field mask of all fields has been specified). Note that in the |  | ||||||
|     presence of schema evolution, this may mean that fields the client does not |  | ||||||
|     know and has therefore not filled into the request will be reset to their |  | ||||||
|     default. If this is unwanted behavior, a specific service may require a |  | ||||||
|     client to always specify a field mask, producing an error if not. As with |  | ||||||
|     get operations, the location of the resource which describes the updated |  | ||||||
|     values in the request message depends on the operation kind. In any case, |  | ||||||
|     the effect of the field mask is required to be honored by the API. ## |  | ||||||
|     Considerations for HTTP REST The HTTP kind of an update operation which |  | ||||||
|     uses a field mask must be set to PATCH instead of PUT in order to satisfy |  | ||||||
|     HTTP semantics (PUT must only be used for full updates). # JSON Encoding of |  | ||||||
|     Field Masks In JSON, a field mask is encoded as a single string where paths |  | ||||||
|     are separated by a comma. Fields name in each path are converted to/from |  | ||||||
|     lower-camel naming conventions. As an example, consider the following |  | ||||||
|     message declarations:     message Profile {       User user = 1; |  | ||||||
|     Photo photo = 2;     }     message User {       string display_name = 1; |  | ||||||
|     string address = 2;     } In proto a field mask for `Profile` may look as |  | ||||||
|     such:     mask {       paths: "user.display_name"       paths: "photo" |  | ||||||
|     } In JSON, the same mask is represented as below:     {       mask: |  | ||||||
|     "user.displayName,photo"     } # Field Masks and Oneof Fields Field masks |  | ||||||
|     treat fields in oneofs just as regular fields. Consider the following |  | ||||||
|     message:     message SampleMessage {       oneof test_oneof { |  | ||||||
|     string name = 4;         SubMessage sub_message = 9;       }     } The |  | ||||||
|     field mask can be:     mask {       paths: "name"     } Or:     mask { |  | ||||||
|     paths: "sub_message"     } Note that oneof type names ("test_oneof" in this |  | ||||||
|     case) cannot be used in paths. ## Field Mask Verification The |  | ||||||
|     implementation of any API method which has a FieldMask type field in the |  | ||||||
|     request should verify the included field paths, and return an |  | ||||||
|     `INVALID_ARGUMENT` error if any path is unmappable. |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     # The set of field mask paths. |  | ||||||
|     paths: List[str] = betterproto.string_field(1) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @dataclass(eq=False, repr=False) |  | ||||||
| class SourceContext(betterproto.Message): |  | ||||||
|     """ |  | ||||||
|     `SourceContext` represents information about the source of a protobuf |  | ||||||
|     element, like the file in which it is defined. |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     # The path-qualified name of the .proto file that contained the associated |  | ||||||
|     # protobuf element.  For example: `"google/protobuf/source_context.proto"`. |  | ||||||
|     file_name: str = betterproto.string_field(1) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @dataclass(eq=False, repr=False) | @dataclass(eq=False, repr=False) | ||||||
| @@ -283,24 +123,25 @@ class Any(betterproto.Message): | |||||||
|     Example 3: Pack and unpack a message in Python.     foo = Foo(...)     any |     Example 3: Pack and unpack a message in Python.     foo = Foo(...)     any | ||||||
|     = Any()     any.Pack(foo)     ...     if any.Is(Foo.DESCRIPTOR): |     = Any()     any.Pack(foo)     ...     if any.Is(Foo.DESCRIPTOR): | ||||||
|     any.Unpack(foo)       ...  Example 4: Pack and unpack a message in Go |     any.Unpack(foo)       ...  Example 4: Pack and unpack a message in Go | ||||||
|     foo := &pb.Foo{...}      any, err := ptypes.MarshalAny(foo)      ... |     foo := &pb.Foo{...}      any, err := anypb.New(foo)      if err != nil { | ||||||
|     foo := &pb.Foo{}      if err := ptypes.UnmarshalAny(any, foo); err != nil { |     ...      }      ...      foo := &pb.Foo{}      if err := | ||||||
|     ...      } The pack methods provided by protobuf library will by default |     any.UnmarshalTo(foo); err != nil {        ...      } The pack methods | ||||||
|     use 'type.googleapis.com/full.type.name' as the type URL and the unpack |     provided by protobuf library will by default use | ||||||
|     methods only use the fully qualified type name after the last '/' in the |     'type.googleapis.com/full.type.name' as the type URL and the unpack methods | ||||||
|     type URL, for example "foo.bar.com/x/y.z" will yield type name "y.z". JSON |     only use the fully qualified type name after the last '/' in the type URL, | ||||||
|     ==== The JSON representation of an `Any` value uses the regular |     for example "foo.bar.com/x/y.z" will yield type name "y.z". JSON ==== The | ||||||
|     representation of the deserialized, embedded message, with an additional |     JSON representation of an `Any` value uses the regular representation of | ||||||
|     field `@type` which contains the type URL. Example:     package |     the deserialized, embedded message, with an additional field `@type` which | ||||||
|     google.profile;     message Person {       string first_name = 1; |     contains the type URL. Example:     package google.profile;     message | ||||||
|     string last_name = 2;     }     {       "@type": |     Person {       string first_name = 1;       string last_name = 2;     } | ||||||
|     "type.googleapis.com/google.profile.Person",       "firstName": <string>, |     {       "@type": "type.googleapis.com/google.profile.Person", | ||||||
|     "lastName": <string>     } If the embedded message type is well-known and |     "firstName": <string>,       "lastName": <string>     } If the embedded | ||||||
|     has a custom JSON representation, that representation will be embedded |     message type is well-known and has a custom JSON representation, that | ||||||
|     adding a field `value` which holds the custom JSON in addition to the |     representation will be embedded adding a field `value` which holds the | ||||||
|     `@type` field. Example (for message [google.protobuf.Duration][]):     { |     custom JSON in addition to the `@type` field. Example (for message | ||||||
|     "@type": "type.googleapis.com/google.protobuf.Duration",       "value": |     [google.protobuf.Duration][]):     {       "@type": | ||||||
|     "1.212s"     } |     "type.googleapis.com/google.protobuf.Duration",       "value": "1.212s" | ||||||
|  |     } | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     # A URL/resource name that uniquely identifies the type of the serialized |     # A URL/resource name that uniquely identifies the type of the serialized | ||||||
| @@ -327,6 +168,18 @@ class Any(betterproto.Message): | |||||||
|     value: bytes = betterproto.bytes_field(2) |     value: bytes = betterproto.bytes_field(2) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @dataclass(eq=False, repr=False) | ||||||
|  | class SourceContext(betterproto.Message): | ||||||
|  |     """ | ||||||
|  |     `SourceContext` represents information about the source of a protobuf | ||||||
|  |     element, like the file in which it is defined. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     # The path-qualified name of the .proto file that contained the associated | ||||||
|  |     # protobuf element.  For example: `"google/protobuf/source_context.proto"`. | ||||||
|  |     file_name: str = betterproto.string_field(1) | ||||||
|  |  | ||||||
|  |  | ||||||
| @dataclass(eq=False, repr=False) | @dataclass(eq=False, repr=False) | ||||||
| class Type(betterproto.Message): | class Type(betterproto.Message): | ||||||
|     """A protocol buffer message type.""" |     """A protocol buffer message type.""" | ||||||
| @@ -510,7 +363,7 @@ class Mixin(betterproto.Message): | |||||||
|     implies that all methods in `AccessControl` are also declared with same |     implies that all methods in `AccessControl` are also declared with same | ||||||
|     name and request/response types in `Storage`. A documentation generator or |     name and request/response types in `Storage`. A documentation generator or | ||||||
|     annotation processor will see the effective `Storage.GetAcl` method after |     annotation processor will see the effective `Storage.GetAcl` method after | ||||||
|     inherting documentation and annotations as follows:     service Storage { |     inheriting documentation and annotations as follows:     service Storage { | ||||||
|     // Get the underlying ACL object.       rpc GetAcl(GetAclRequest) returns |     // Get the underlying ACL object.       rpc GetAcl(GetAclRequest) returns | ||||||
|     (Acl) {         option (google.api.http).get = "/v2/{resource=**}:getAcl"; |     (Acl) {         option (google.api.http).get = "/v2/{resource=**}:getAcl"; | ||||||
|     }       ...     } Note how the version in the path pattern changed from |     }       ...     } Note how the version in the path pattern changed from | ||||||
| @@ -530,215 +383,6 @@ class Mixin(betterproto.Message): | |||||||
|     root: str = betterproto.string_field(2) |     root: str = betterproto.string_field(2) | ||||||
|  |  | ||||||
|  |  | ||||||
| @dataclass(eq=False, repr=False) |  | ||||||
| class Duration(betterproto.Message): |  | ||||||
|     """ |  | ||||||
|     A Duration represents a signed, fixed-length span of time represented as a |  | ||||||
|     count of seconds and fractions of seconds at nanosecond resolution. It is |  | ||||||
|     independent of any calendar and concepts like "day" or "month". It is |  | ||||||
|     related to Timestamp in that the difference between two Timestamp values is |  | ||||||
|     a Duration and it can be added or subtracted from a Timestamp. Range is |  | ||||||
|     approximately +-10,000 years. # Examples Example 1: Compute Duration from |  | ||||||
|     two Timestamps in pseudo code.     Timestamp start = ...;     Timestamp end |  | ||||||
|     = ...;     Duration duration = ...;     duration.seconds = end.seconds - |  | ||||||
|     start.seconds;     duration.nanos = end.nanos - start.nanos;     if |  | ||||||
|     (duration.seconds < 0 && duration.nanos > 0) {       duration.seconds += 1; |  | ||||||
|     duration.nanos -= 1000000000;     } else if (duration.seconds > 0 && |  | ||||||
|     duration.nanos < 0) {       duration.seconds -= 1;       duration.nanos += |  | ||||||
|     1000000000;     } Example 2: Compute Timestamp from Timestamp + Duration in |  | ||||||
|     pseudo code.     Timestamp start = ...;     Duration duration = ...; |  | ||||||
|     Timestamp end = ...;     end.seconds = start.seconds + duration.seconds; |  | ||||||
|     end.nanos = start.nanos + duration.nanos;     if (end.nanos < 0) { |  | ||||||
|     end.seconds -= 1;       end.nanos += 1000000000;     } else if (end.nanos |  | ||||||
|     >= 1000000000) {       end.seconds += 1;       end.nanos -= 1000000000; |  | ||||||
|     } Example 3: Compute Duration from datetime.timedelta in Python.     td = |  | ||||||
|     datetime.timedelta(days=3, minutes=10)     duration = Duration() |  | ||||||
|     duration.FromTimedelta(td) # JSON Mapping In JSON format, the Duration type |  | ||||||
|     is encoded as a string rather than an object, where the string ends in the |  | ||||||
|     suffix "s" (indicating seconds) and is preceded by the number of seconds, |  | ||||||
|     with nanoseconds expressed as fractional seconds. For example, 3 seconds |  | ||||||
|     with 0 nanoseconds should be encoded in JSON format as "3s", while 3 |  | ||||||
|     seconds and 1 nanosecond should be expressed in JSON format as |  | ||||||
|     "3.000000001s", and 3 seconds and 1 microsecond should be expressed in JSON |  | ||||||
|     format as "3.000001s". |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     # Signed seconds of the span of time. Must be from -315,576,000,000 to |  | ||||||
|     # +315,576,000,000 inclusive. Note: these bounds are computed from: 60 |  | ||||||
|     # sec/min * 60 min/hr * 24 hr/day * 365.25 days/year * 10000 years |  | ||||||
|     seconds: int = betterproto.int64_field(1) |  | ||||||
|     # Signed fractions of a second at nanosecond resolution of the span of time. |  | ||||||
|     # Durations less than one second are represented with a 0 `seconds` field and |  | ||||||
|     # a positive or negative `nanos` field. For durations of one second or more, |  | ||||||
|     # a non-zero value for the `nanos` field must be of the same sign as the |  | ||||||
|     # `seconds` field. Must be from -999,999,999 to +999,999,999 inclusive. |  | ||||||
|     nanos: int = betterproto.int32_field(2) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @dataclass(eq=False, repr=False) |  | ||||||
| class Struct(betterproto.Message): |  | ||||||
|     """ |  | ||||||
|     `Struct` represents a structured data value, consisting of fields which map |  | ||||||
|     to dynamically typed values. In some languages, `Struct` might be supported |  | ||||||
|     by a native representation. For example, in scripting languages like JS a |  | ||||||
|     struct is represented as an object. The details of that representation are |  | ||||||
|     described together with the proto support for the language. The JSON |  | ||||||
|     representation for `Struct` is JSON object. |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     # Unordered map of dynamically typed values. |  | ||||||
|     fields: Dict[str, "Value"] = betterproto.map_field( |  | ||||||
|         1, betterproto.TYPE_STRING, betterproto.TYPE_MESSAGE |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @dataclass(eq=False, repr=False) |  | ||||||
| class Value(betterproto.Message): |  | ||||||
|     """ |  | ||||||
|     `Value` represents a dynamically typed value which can be either null, a |  | ||||||
|     number, a string, a boolean, a recursive struct value, or a list of values. |  | ||||||
|     A producer of value is expected to set one of that variants, absence of any |  | ||||||
|     variant indicates an error. The JSON representation for `Value` is JSON |  | ||||||
|     value. |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     # Represents a null value. |  | ||||||
|     null_value: "NullValue" = betterproto.enum_field(1, group="kind") |  | ||||||
|     # Represents a double value. |  | ||||||
|     number_value: float = betterproto.double_field(2, group="kind") |  | ||||||
|     # Represents a string value. |  | ||||||
|     string_value: str = betterproto.string_field(3, group="kind") |  | ||||||
|     # Represents a boolean value. |  | ||||||
|     bool_value: bool = betterproto.bool_field(4, group="kind") |  | ||||||
|     # Represents a structured value. |  | ||||||
|     struct_value: "Struct" = betterproto.message_field(5, group="kind") |  | ||||||
|     # Represents a repeated `Value`. |  | ||||||
|     list_value: "ListValue" = betterproto.message_field(6, group="kind") |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @dataclass(eq=False, repr=False) |  | ||||||
| class ListValue(betterproto.Message): |  | ||||||
|     """ |  | ||||||
|     `ListValue` is a wrapper around a repeated field of values. The JSON |  | ||||||
|     representation for `ListValue` is JSON array. |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     # Repeated field of dynamically typed values. |  | ||||||
|     values: List["Value"] = betterproto.message_field(1) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @dataclass(eq=False, repr=False) |  | ||||||
| class DoubleValue(betterproto.Message): |  | ||||||
|     """ |  | ||||||
|     Wrapper message for `double`. The JSON representation for `DoubleValue` is |  | ||||||
|     JSON number. |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     # The double value. |  | ||||||
|     value: float = betterproto.double_field(1) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @dataclass(eq=False, repr=False) |  | ||||||
| class FloatValue(betterproto.Message): |  | ||||||
|     """ |  | ||||||
|     Wrapper message for `float`. The JSON representation for `FloatValue` is |  | ||||||
|     JSON number. |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     # The float value. |  | ||||||
|     value: float = betterproto.float_field(1) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @dataclass(eq=False, repr=False) |  | ||||||
| class Int64Value(betterproto.Message): |  | ||||||
|     """ |  | ||||||
|     Wrapper message for `int64`. The JSON representation for `Int64Value` is |  | ||||||
|     JSON string. |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     # The int64 value. |  | ||||||
|     value: int = betterproto.int64_field(1) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @dataclass(eq=False, repr=False) |  | ||||||
| class UInt64Value(betterproto.Message): |  | ||||||
|     """ |  | ||||||
|     Wrapper message for `uint64`. The JSON representation for `UInt64Value` is |  | ||||||
|     JSON string. |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     # The uint64 value. |  | ||||||
|     value: int = betterproto.uint64_field(1) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @dataclass(eq=False, repr=False) |  | ||||||
| class Int32Value(betterproto.Message): |  | ||||||
|     """ |  | ||||||
|     Wrapper message for `int32`. The JSON representation for `Int32Value` is |  | ||||||
|     JSON number. |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     # The int32 value. |  | ||||||
|     value: int = betterproto.int32_field(1) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @dataclass(eq=False, repr=False) |  | ||||||
| class UInt32Value(betterproto.Message): |  | ||||||
|     """ |  | ||||||
|     Wrapper message for `uint32`. The JSON representation for `UInt32Value` is |  | ||||||
|     JSON number. |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     # The uint32 value. |  | ||||||
|     value: int = betterproto.uint32_field(1) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @dataclass(eq=False, repr=False) |  | ||||||
| class BoolValue(betterproto.Message): |  | ||||||
|     """ |  | ||||||
|     Wrapper message for `bool`. The JSON representation for `BoolValue` is JSON |  | ||||||
|     `true` and `false`. |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     # The bool value. |  | ||||||
|     value: bool = betterproto.bool_field(1) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @dataclass(eq=False, repr=False) |  | ||||||
| class StringValue(betterproto.Message): |  | ||||||
|     """ |  | ||||||
|     Wrapper message for `string`. The JSON representation for `StringValue` is |  | ||||||
|     JSON string. |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     # The string value. |  | ||||||
|     value: str = betterproto.string_field(1) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @dataclass(eq=False, repr=False) |  | ||||||
| class BytesValue(betterproto.Message): |  | ||||||
|     """ |  | ||||||
|     Wrapper message for `bytes`. The JSON representation for `BytesValue` is |  | ||||||
|     JSON string. |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     # The bytes value. |  | ||||||
|     value: bytes = betterproto.bytes_field(1) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @dataclass(eq=False, repr=False) |  | ||||||
| class Empty(betterproto.Message): |  | ||||||
|     """ |  | ||||||
|     A generic empty message that you can re-use to avoid defining duplicated |  | ||||||
|     empty messages in your APIs. A typical example is to use it as the request |  | ||||||
|     or the response type of an API method. For instance:     service Foo { |  | ||||||
|     rpc Bar(google.protobuf.Empty) returns (google.protobuf.Empty);     } The |  | ||||||
|     JSON representation for `Empty` is empty JSON object `{}`. |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     pass |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @dataclass(eq=False, repr=False) | @dataclass(eq=False, repr=False) | ||||||
| class FileDescriptorSet(betterproto.Message): | class FileDescriptorSet(betterproto.Message): | ||||||
|     """ |     """ | ||||||
| @@ -855,6 +499,23 @@ class FieldDescriptorProto(betterproto.Message): | |||||||
|     # camelCase. |     # camelCase. | ||||||
|     json_name: str = betterproto.string_field(10) |     json_name: str = betterproto.string_field(10) | ||||||
|     options: "FieldOptions" = betterproto.message_field(8) |     options: "FieldOptions" = betterproto.message_field(8) | ||||||
|  |     # If true, this is a proto3 "optional". When a proto3 field is optional, it | ||||||
|  |     # tracks presence regardless of field type. When proto3_optional is true, | ||||||
|  |     # this field must be belong to a oneof to signal to old proto3 clients that | ||||||
|  |     # presence is tracked for this field. This oneof is known as a "synthetic" | ||||||
|  |     # oneof, and this field must be its sole member (each proto3 optional field | ||||||
|  |     # gets its own synthetic oneof). Synthetic oneofs exist in the descriptor | ||||||
|  |     # only, and do not generate any API. Synthetic oneofs must be ordered after | ||||||
|  |     # all "real" oneofs. For message fields, proto3_optional doesn't create any | ||||||
|  |     # semantic change, since non-repeated message fields always track presence. | ||||||
|  |     # However it still indicates the semantic detail of whether the user wrote | ||||||
|  |     # "optional" or not. This can be useful for round-tripping the .proto file. | ||||||
|  |     # For consistency we give message fields a synthetic oneof also, even though | ||||||
|  |     # it is not required to track presence. This is especially important because | ||||||
|  |     # the parser can't tell if a field is a message or an enum, so it must always | ||||||
|  |     # create a synthetic oneof. Proto2 optional fields do not set this flag, | ||||||
|  |     # because they already indicate optional with `LABEL_OPTIONAL`. | ||||||
|  |     proto3_optional: bool = betterproto.bool_field(17) | ||||||
|  |  | ||||||
|  |  | ||||||
| @dataclass(eq=False, repr=False) | @dataclass(eq=False, repr=False) | ||||||
| @@ -937,17 +598,18 @@ class FileOptions(betterproto.Message): | |||||||
|     # inappropriate because proto packages do not normally start with backwards |     # inappropriate because proto packages do not normally start with backwards | ||||||
|     # domain names. |     # domain names. | ||||||
|     java_package: str = betterproto.string_field(1) |     java_package: str = betterproto.string_field(1) | ||||||
|     # If set, all the classes from the .proto file are wrapped in a single outer |     # Controls the name of the wrapper Java class generated for the .proto file. | ||||||
|     # class with the given name.  This applies to both Proto1 (equivalent to the |     # That class will always contain the .proto file's getDescriptor() method as | ||||||
|     # old "--one_java_file" option) and Proto2 (where a .proto always translates |     # well as any top-level extensions defined in the .proto file. If | ||||||
|     # to a single class, but you may want to explicitly choose the class name). |     # java_multiple_files is disabled, then all the other classes from the .proto | ||||||
|  |     # file will be nested inside the single wrapper outer class. | ||||||
|     java_outer_classname: str = betterproto.string_field(8) |     java_outer_classname: str = betterproto.string_field(8) | ||||||
|     # If set true, then the Java code generator will generate a separate .java |     # If enabled, then the Java code generator will generate a separate .java | ||||||
|     # file for each top-level message, enum, and service defined in the .proto |     # file for each top-level message, enum, and service defined in the .proto | ||||||
|     # file.  Thus, these types will *not* be nested inside the outer class named |     # file.  Thus, these types will *not* be nested inside the wrapper class | ||||||
|     # by java_outer_classname.  However, the outer class will still be generated |     # named by java_outer_classname.  However, the wrapper class will still be | ||||||
|     # to contain the file's getDescriptor() method as well as any top-level |     # generated to contain the file's getDescriptor() method as well as any top- | ||||||
|     # extensions defined in the file. |     # level extensions defined in the file. | ||||||
|     java_multiple_files: bool = betterproto.bool_field(10) |     java_multiple_files: bool = betterproto.bool_field(10) | ||||||
|     # This option does nothing. |     # This option does nothing. | ||||||
|     java_generate_equals_and_hash: bool = betterproto.bool_field(20) |     java_generate_equals_and_hash: bool = betterproto.bool_field(20) | ||||||
| @@ -1315,3 +977,363 @@ class GeneratedCodeInfoAnnotation(betterproto.Message): | |||||||
|     # the identified offset. The end offset should be one past the last relevant |     # the identified offset. The end offset should be one past the last relevant | ||||||
|     # byte (so the length of the text = end - begin). |     # byte (so the length of the text = end - begin). | ||||||
|     end: int = betterproto.int32_field(4) |     end: int = betterproto.int32_field(4) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @dataclass(eq=False, repr=False) | ||||||
|  | class Duration(betterproto.Message): | ||||||
|  |     """ | ||||||
|  |     A Duration represents a signed, fixed-length span of time represented as a | ||||||
|  |     count of seconds and fractions of seconds at nanosecond resolution. It is | ||||||
|  |     independent of any calendar and concepts like "day" or "month". It is | ||||||
|  |     related to Timestamp in that the difference between two Timestamp values is | ||||||
|  |     a Duration and it can be added or subtracted from a Timestamp. Range is | ||||||
|  |     approximately +-10,000 years. # Examples Example 1: Compute Duration from | ||||||
|  |     two Timestamps in pseudo code.     Timestamp start = ...;     Timestamp end | ||||||
|  |     = ...;     Duration duration = ...;     duration.seconds = end.seconds - | ||||||
|  |     start.seconds;     duration.nanos = end.nanos - start.nanos;     if | ||||||
|  |     (duration.seconds < 0 && duration.nanos > 0) {       duration.seconds += 1; | ||||||
|  |     duration.nanos -= 1000000000;     } else if (duration.seconds > 0 && | ||||||
|  |     duration.nanos < 0) {       duration.seconds -= 1;       duration.nanos += | ||||||
|  |     1000000000;     } Example 2: Compute Timestamp from Timestamp + Duration in | ||||||
|  |     pseudo code.     Timestamp start = ...;     Duration duration = ...; | ||||||
|  |     Timestamp end = ...;     end.seconds = start.seconds + duration.seconds; | ||||||
|  |     end.nanos = start.nanos + duration.nanos;     if (end.nanos < 0) { | ||||||
|  |     end.seconds -= 1;       end.nanos += 1000000000;     } else if (end.nanos | ||||||
|  |     >= 1000000000) {       end.seconds += 1;       end.nanos -= 1000000000; | ||||||
|  |     } Example 3: Compute Duration from datetime.timedelta in Python.     td = | ||||||
|  |     datetime.timedelta(days=3, minutes=10)     duration = Duration() | ||||||
|  |     duration.FromTimedelta(td) # JSON Mapping In JSON format, the Duration type | ||||||
|  |     is encoded as a string rather than an object, where the string ends in the | ||||||
|  |     suffix "s" (indicating seconds) and is preceded by the number of seconds, | ||||||
|  |     with nanoseconds expressed as fractional seconds. For example, 3 seconds | ||||||
|  |     with 0 nanoseconds should be encoded in JSON format as "3s", while 3 | ||||||
|  |     seconds and 1 nanosecond should be expressed in JSON format as | ||||||
|  |     "3.000000001s", and 3 seconds and 1 microsecond should be expressed in JSON | ||||||
|  |     format as "3.000001s". | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     # Signed seconds of the span of time. Must be from -315,576,000,000 to | ||||||
|  |     # +315,576,000,000 inclusive. Note: these bounds are computed from: 60 | ||||||
|  |     # sec/min * 60 min/hr * 24 hr/day * 365.25 days/year * 10000 years | ||||||
|  |     seconds: int = betterproto.int64_field(1) | ||||||
|  |     # Signed fractions of a second at nanosecond resolution of the span of time. | ||||||
|  |     # Durations less than one second are represented with a 0 `seconds` field and | ||||||
|  |     # a positive or negative `nanos` field. For durations of one second or more, | ||||||
|  |     # a non-zero value for the `nanos` field must be of the same sign as the | ||||||
|  |     # `seconds` field. Must be from -999,999,999 to +999,999,999 inclusive. | ||||||
|  |     nanos: int = betterproto.int32_field(2) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @dataclass(eq=False, repr=False) | ||||||
|  | class Empty(betterproto.Message): | ||||||
|  |     """ | ||||||
|  |     A generic empty message that you can re-use to avoid defining duplicated | ||||||
|  |     empty messages in your APIs. A typical example is to use it as the request | ||||||
|  |     or the response type of an API method. For instance:     service Foo { | ||||||
|  |     rpc Bar(google.protobuf.Empty) returns (google.protobuf.Empty);     } The | ||||||
|  |     JSON representation for `Empty` is empty JSON object `{}`. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     pass | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @dataclass(eq=False, repr=False) | ||||||
|  | class FieldMask(betterproto.Message): | ||||||
|  |     """ | ||||||
|  |     `FieldMask` represents a set of symbolic field paths, for example: | ||||||
|  |     paths: "f.a"     paths: "f.b.d" Here `f` represents a field in some root | ||||||
|  |     message, `a` and `b` fields in the message found in `f`, and `d` a field | ||||||
|  |     found in the message in `f.b`. Field masks are used to specify a subset of | ||||||
|  |     fields that should be returned by a get operation or modified by an update | ||||||
|  |     operation. Field masks also have a custom JSON encoding (see below). # | ||||||
|  |     Field Masks in Projections When used in the context of a projection, a | ||||||
|  |     response message or sub-message is filtered by the API to only contain | ||||||
|  |     those fields as specified in the mask. For example, if the mask in the | ||||||
|  |     previous example is applied to a response message as follows:     f { | ||||||
|  |     a : 22       b {         d : 1         x : 2       }       y : 13     } | ||||||
|  |     z: 8 The result will not contain specific values for fields x,y and z | ||||||
|  |     (their value will be set to the default, and omitted in proto text output): | ||||||
|  |     f {       a : 22       b {         d : 1       }     } A repeated field is | ||||||
|  |     not allowed except at the last position of a paths string. If a FieldMask | ||||||
|  |     object is not present in a get operation, the operation applies to all | ||||||
|  |     fields (as if a FieldMask of all fields had been specified). Note that a | ||||||
|  |     field mask does not necessarily apply to the top-level response message. In | ||||||
|  |     case of a REST get operation, the field mask applies directly to the | ||||||
|  |     response, but in case of a REST list operation, the mask instead applies to | ||||||
|  |     each individual message in the returned resource list. In case of a REST | ||||||
|  |     custom method, other definitions may be used. Where the mask applies will | ||||||
|  |     be clearly documented together with its declaration in the API.  In any | ||||||
|  |     case, the effect on the returned resource/resources is required behavior | ||||||
|  |     for APIs. # Field Masks in Update Operations A field mask in update | ||||||
|  |     operations specifies which fields of the targeted resource are going to be | ||||||
|  |     updated. The API is required to only change the values of the fields as | ||||||
|  |     specified in the mask and leave the others untouched. If a resource is | ||||||
|  |     passed in to describe the updated values, the API ignores the values of all | ||||||
|  |     fields not covered by the mask. If a repeated field is specified for an | ||||||
|  |     update operation, new values will be appended to the existing repeated | ||||||
|  |     field in the target resource. Note that a repeated field is only allowed in | ||||||
|  |     the last position of a `paths` string. If a sub-message is specified in the | ||||||
|  |     last position of the field mask for an update operation, then new value | ||||||
|  |     will be merged into the existing sub-message in the target resource. For | ||||||
|  |     example, given the target message:     f {       b {         d: 1 | ||||||
|  |     x: 2       }       c: [1]     } And an update message:     f {       b { | ||||||
|  |     d: 10       }       c: [2]     } then if the field mask is:  paths: ["f.b", | ||||||
|  |     "f.c"] then the result will be:     f {       b {         d: 10         x: | ||||||
|  |     2       }       c: [1, 2]     } An implementation may provide options to | ||||||
|  |     override this default behavior for repeated and message fields. In order to | ||||||
|  |     reset a field's value to the default, the field must be in the mask and set | ||||||
|  |     to the default value in the provided resource. Hence, in order to reset all | ||||||
|  |     fields of a resource, provide a default instance of the resource and set | ||||||
|  |     all fields in the mask, or do not provide a mask as described below. If a | ||||||
|  |     field mask is not present on update, the operation applies to all fields | ||||||
|  |     (as if a field mask of all fields has been specified). Note that in the | ||||||
|  |     presence of schema evolution, this may mean that fields the client does not | ||||||
|  |     know and has therefore not filled into the request will be reset to their | ||||||
|  |     default. If this is unwanted behavior, a specific service may require a | ||||||
|  |     client to always specify a field mask, producing an error if not. As with | ||||||
|  |     get operations, the location of the resource which describes the updated | ||||||
|  |     values in the request message depends on the operation kind. In any case, | ||||||
|  |     the effect of the field mask is required to be honored by the API. ## | ||||||
|  |     Considerations for HTTP REST The HTTP kind of an update operation which | ||||||
|  |     uses a field mask must be set to PATCH instead of PUT in order to satisfy | ||||||
|  |     HTTP semantics (PUT must only be used for full updates). # JSON Encoding of | ||||||
|  |     Field Masks In JSON, a field mask is encoded as a single string where paths | ||||||
|  |     are separated by a comma. Fields name in each path are converted to/from | ||||||
|  |     lower-camel naming conventions. As an example, consider the following | ||||||
|  |     message declarations:     message Profile {       User user = 1; | ||||||
|  |     Photo photo = 2;     }     message User {       string display_name = 1; | ||||||
|  |     string address = 2;     } In proto a field mask for `Profile` may look as | ||||||
|  |     such:     mask {       paths: "user.display_name"       paths: "photo" | ||||||
|  |     } In JSON, the same mask is represented as below:     {       mask: | ||||||
|  |     "user.displayName,photo"     } # Field Masks and Oneof Fields Field masks | ||||||
|  |     treat fields in oneofs just as regular fields. Consider the following | ||||||
|  |     message:     message SampleMessage {       oneof test_oneof { | ||||||
|  |     string name = 4;         SubMessage sub_message = 9;       }     } The | ||||||
|  |     field mask can be:     mask {       paths: "name"     } Or:     mask { | ||||||
|  |     paths: "sub_message"     } Note that oneof type names ("test_oneof" in this | ||||||
|  |     case) cannot be used in paths. ## Field Mask Verification The | ||||||
|  |     implementation of any API method which has a FieldMask type field in the | ||||||
|  |     request should verify the included field paths, and return an | ||||||
|  |     `INVALID_ARGUMENT` error if any path is unmappable. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     # The set of field mask paths. | ||||||
|  |     paths: List[str] = betterproto.string_field(1) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @dataclass(eq=False, repr=False) | ||||||
|  | class Struct(betterproto.Message): | ||||||
|  |     """ | ||||||
|  |     `Struct` represents a structured data value, consisting of fields which map | ||||||
|  |     to dynamically typed values. In some languages, `Struct` might be supported | ||||||
|  |     by a native representation. For example, in scripting languages like JS a | ||||||
|  |     struct is represented as an object. The details of that representation are | ||||||
|  |     described together with the proto support for the language. The JSON | ||||||
|  |     representation for `Struct` is JSON object. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     # Unordered map of dynamically typed values. | ||||||
|  |     fields: Dict[str, "Value"] = betterproto.map_field( | ||||||
|  |         1, betterproto.TYPE_STRING, betterproto.TYPE_MESSAGE | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @dataclass(eq=False, repr=False) | ||||||
|  | class Value(betterproto.Message): | ||||||
|  |     """ | ||||||
|  |     `Value` represents a dynamically typed value which can be either null, a | ||||||
|  |     number, a string, a boolean, a recursive struct value, or a list of values. | ||||||
|  |     A producer of value is expected to set one of that variants, absence of any | ||||||
|  |     variant indicates an error. The JSON representation for `Value` is JSON | ||||||
|  |     value. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     # Represents a null value. | ||||||
|  |     null_value: "NullValue" = betterproto.enum_field(1, group="kind") | ||||||
|  |     # Represents a double value. | ||||||
|  |     number_value: float = betterproto.double_field(2, group="kind") | ||||||
|  |     # Represents a string value. | ||||||
|  |     string_value: str = betterproto.string_field(3, group="kind") | ||||||
|  |     # Represents a boolean value. | ||||||
|  |     bool_value: bool = betterproto.bool_field(4, group="kind") | ||||||
|  |     # Represents a structured value. | ||||||
|  |     struct_value: "Struct" = betterproto.message_field(5, group="kind") | ||||||
|  |     # Represents a repeated `Value`. | ||||||
|  |     list_value: "ListValue" = betterproto.message_field(6, group="kind") | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @dataclass(eq=False, repr=False) | ||||||
|  | class ListValue(betterproto.Message): | ||||||
|  |     """ | ||||||
|  |     `ListValue` is a wrapper around a repeated field of values. The JSON | ||||||
|  |     representation for `ListValue` is JSON array. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     # Repeated field of dynamically typed values. | ||||||
|  |     values: List["Value"] = betterproto.message_field(1) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @dataclass(eq=False, repr=False) | ||||||
|  | class Timestamp(betterproto.Message): | ||||||
|  |     """ | ||||||
|  |     A Timestamp represents a point in time independent of any time zone or | ||||||
|  |     local calendar, encoded as a count of seconds and fractions of seconds at | ||||||
|  |     nanosecond resolution. The count is relative to an epoch at UTC midnight on | ||||||
|  |     January 1, 1970, in the proleptic Gregorian calendar which extends the | ||||||
|  |     Gregorian calendar backwards to year one. All minutes are 60 seconds long. | ||||||
|  |     Leap seconds are "smeared" so that no leap second table is needed for | ||||||
|  |     interpretation, using a [24-hour linear | ||||||
|  |     smear](https://developers.google.com/time/smear). The range is from | ||||||
|  |     0001-01-01T00:00:00Z to 9999-12-31T23:59:59.999999999Z. By restricting to | ||||||
|  |     that range, we ensure that we can convert to and from [RFC | ||||||
|  |     3339](https://www.ietf.org/rfc/rfc3339.txt) date strings. # Examples | ||||||
|  |     Example 1: Compute Timestamp from POSIX `time()`.     Timestamp timestamp; | ||||||
|  |     timestamp.set_seconds(time(NULL));     timestamp.set_nanos(0); Example 2: | ||||||
|  |     Compute Timestamp from POSIX `gettimeofday()`.     struct timeval tv; | ||||||
|  |     gettimeofday(&tv, NULL);     Timestamp timestamp; | ||||||
|  |     timestamp.set_seconds(tv.tv_sec);     timestamp.set_nanos(tv.tv_usec * | ||||||
|  |     1000); Example 3: Compute Timestamp from Win32 `GetSystemTimeAsFileTime()`. | ||||||
|  |     FILETIME ft;     GetSystemTimeAsFileTime(&ft);     UINT64 ticks = | ||||||
|  |     (((UINT64)ft.dwHighDateTime) << 32) | ft.dwLowDateTime;     // A Windows | ||||||
|  |     tick is 100 nanoseconds. Windows epoch 1601-01-01T00:00:00Z     // is | ||||||
|  |     11644473600 seconds before Unix epoch 1970-01-01T00:00:00Z.     Timestamp | ||||||
|  |     timestamp;     timestamp.set_seconds((INT64) ((ticks / 10000000) - | ||||||
|  |     11644473600LL));     timestamp.set_nanos((INT32) ((ticks % 10000000) * | ||||||
|  |     100)); Example 4: Compute Timestamp from Java `System.currentTimeMillis()`. | ||||||
|  |     long millis = System.currentTimeMillis();     Timestamp timestamp = | ||||||
|  |     Timestamp.newBuilder().setSeconds(millis / 1000)         .setNanos((int) | ||||||
|  |     ((millis % 1000) * 1000000)).build(); Example 5: Compute Timestamp from | ||||||
|  |     Java `Instant.now()`.     Instant now = Instant.now();     Timestamp | ||||||
|  |     timestamp =         Timestamp.newBuilder().setSeconds(now.getEpochSecond()) | ||||||
|  |     .setNanos(now.getNano()).build(); Example 6: Compute Timestamp from current | ||||||
|  |     time in Python.     timestamp = Timestamp()     timestamp.GetCurrentTime() | ||||||
|  |     # JSON Mapping In JSON format, the Timestamp type is encoded as a string in | ||||||
|  |     the [RFC 3339](https://www.ietf.org/rfc/rfc3339.txt) format. That is, the | ||||||
|  |     format is "{year}-{month}-{day}T{hour}:{min}:{sec}[.{frac_sec}]Z" where | ||||||
|  |     {year} is always expressed using four digits while {month}, {day}, {hour}, | ||||||
|  |     {min}, and {sec} are zero-padded to two digits each. The fractional | ||||||
|  |     seconds, which can go up to 9 digits (i.e. up to 1 nanosecond resolution), | ||||||
|  |     are optional. The "Z" suffix indicates the timezone ("UTC"); the timezone | ||||||
|  |     is required. A proto3 JSON serializer should always use UTC (as indicated | ||||||
|  |     by "Z") when printing the Timestamp type and a proto3 JSON parser should be | ||||||
|  |     able to accept both UTC and other timezones (as indicated by an offset). | ||||||
|  |     For example, "2017-01-15T01:30:15.01Z" encodes 15.01 seconds past 01:30 UTC | ||||||
|  |     on January 15, 2017. In JavaScript, one can convert a Date object to this | ||||||
|  |     format using the standard [toISOString()](https://developer.mozilla.org/en- | ||||||
|  |     US/docs/Web/JavaScript/Reference/Global_Objects/Date/toISOString) method. | ||||||
|  |     In Python, a standard `datetime.datetime` object can be converted to this | ||||||
|  |     format using | ||||||
|  |     [`strftime`](https://docs.python.org/2/library/time.html#time.strftime) | ||||||
|  |     with the time format spec '%Y-%m-%dT%H:%M:%S.%fZ'. Likewise, in Java, one | ||||||
|  |     can use the Joda Time's [`ISODateTimeFormat.dateTime()`]( | ||||||
|  |     http://www.joda.org/joda- | ||||||
|  |     time/apidocs/org/joda/time/format/ISODateTimeFormat.html#dateTime%2D%2D ) | ||||||
|  |     to obtain a formatter capable of generating timestamps in this format. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     # Represents seconds of UTC time since Unix epoch 1970-01-01T00:00:00Z. Must | ||||||
|  |     # be from 0001-01-01T00:00:00Z to 9999-12-31T23:59:59Z inclusive. | ||||||
|  |     seconds: int = betterproto.int64_field(1) | ||||||
|  |     # Non-negative fractions of a second at nanosecond resolution. Negative | ||||||
|  |     # second values with fractions must still have non-negative nanos values that | ||||||
|  |     # count forward in time. Must be from 0 to 999,999,999 inclusive. | ||||||
|  |     nanos: int = betterproto.int32_field(2) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @dataclass(eq=False, repr=False) | ||||||
|  | class DoubleValue(betterproto.Message): | ||||||
|  |     """ | ||||||
|  |     Wrapper message for `double`. The JSON representation for `DoubleValue` is | ||||||
|  |     JSON number. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     # The double value. | ||||||
|  |     value: float = betterproto.double_field(1) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @dataclass(eq=False, repr=False) | ||||||
|  | class FloatValue(betterproto.Message): | ||||||
|  |     """ | ||||||
|  |     Wrapper message for `float`. The JSON representation for `FloatValue` is | ||||||
|  |     JSON number. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     # The float value. | ||||||
|  |     value: float = betterproto.float_field(1) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @dataclass(eq=False, repr=False) | ||||||
|  | class Int64Value(betterproto.Message): | ||||||
|  |     """ | ||||||
|  |     Wrapper message for `int64`. The JSON representation for `Int64Value` is | ||||||
|  |     JSON string. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     # The int64 value. | ||||||
|  |     value: int = betterproto.int64_field(1) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @dataclass(eq=False, repr=False) | ||||||
|  | class UInt64Value(betterproto.Message): | ||||||
|  |     """ | ||||||
|  |     Wrapper message for `uint64`. The JSON representation for `UInt64Value` is | ||||||
|  |     JSON string. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     # The uint64 value. | ||||||
|  |     value: int = betterproto.uint64_field(1) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @dataclass(eq=False, repr=False) | ||||||
|  | class Int32Value(betterproto.Message): | ||||||
|  |     """ | ||||||
|  |     Wrapper message for `int32`. The JSON representation for `Int32Value` is | ||||||
|  |     JSON number. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     # The int32 value. | ||||||
|  |     value: int = betterproto.int32_field(1) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @dataclass(eq=False, repr=False) | ||||||
|  | class UInt32Value(betterproto.Message): | ||||||
|  |     """ | ||||||
|  |     Wrapper message for `uint32`. The JSON representation for `UInt32Value` is | ||||||
|  |     JSON number. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     # The uint32 value. | ||||||
|  |     value: int = betterproto.uint32_field(1) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @dataclass(eq=False, repr=False) | ||||||
|  | class BoolValue(betterproto.Message): | ||||||
|  |     """ | ||||||
|  |     Wrapper message for `bool`. The JSON representation for `BoolValue` is JSON | ||||||
|  |     `true` and `false`. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     # The bool value. | ||||||
|  |     value: bool = betterproto.bool_field(1) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @dataclass(eq=False, repr=False) | ||||||
|  | class StringValue(betterproto.Message): | ||||||
|  |     """ | ||||||
|  |     Wrapper message for `string`. The JSON representation for `StringValue` is | ||||||
|  |     JSON string. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     # The string value. | ||||||
|  |     value: str = betterproto.string_field(1) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @dataclass(eq=False, repr=False) | ||||||
|  | class BytesValue(betterproto.Message): | ||||||
|  |     """ | ||||||
|  |     Wrapper message for `bytes`. The JSON representation for `BytesValue` is | ||||||
|  |     JSON string. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     # The bytes value. | ||||||
|  |     value: bytes = betterproto.bytes_field(1) | ||||||
|   | |||||||
| @@ -5,6 +5,12 @@ from dataclasses import dataclass | |||||||
| from typing import List | from typing import List | ||||||
|  |  | ||||||
| import betterproto | import betterproto | ||||||
|  | from betterproto.grpc.grpclib_server import ServiceBase | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class CodeGeneratorResponseFeature(betterproto.Enum): | ||||||
|  |     FEATURE_NONE = 0 | ||||||
|  |     FEATURE_PROTO3_OPTIONAL = 1 | ||||||
|  |  | ||||||
|  |  | ||||||
| @dataclass(eq=False, repr=False) | @dataclass(eq=False, repr=False) | ||||||
| @@ -59,6 +65,9 @@ class CodeGeneratorResponse(betterproto.Message): | |||||||
|     # unparseable -- should be reported by writing a message to stderr and |     # unparseable -- should be reported by writing a message to stderr and | ||||||
|     # exiting with a non-zero status code. |     # exiting with a non-zero status code. | ||||||
|     error: str = betterproto.string_field(1) |     error: str = betterproto.string_field(1) | ||||||
|  |     # A bitmask of supported features that the code generator supports. This is a | ||||||
|  |     # bitwise "or" of values from the Feature enum. | ||||||
|  |     supported_features: int = betterproto.uint64_field(2) | ||||||
|     file: List["CodeGeneratorResponseFile"] = betterproto.message_field(15) |     file: List["CodeGeneratorResponseFile"] = betterproto.message_field(15) | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -108,6 +117,12 @@ class CodeGeneratorResponseFile(betterproto.Message): | |||||||
|     insertion_point: str = betterproto.string_field(2) |     insertion_point: str = betterproto.string_field(2) | ||||||
|     # The file contents. |     # The file contents. | ||||||
|     content: str = betterproto.string_field(15) |     content: str = betterproto.string_field(15) | ||||||
|  |     # Information describing the file content being inserted. If an insertion | ||||||
|  |     # point is used, this information will be appropriately offset and inserted | ||||||
|  |     # into the code generation metadata for the generated files. | ||||||
|  |     generated_code_info: "betterproto_lib_google_protobuf.GeneratedCodeInfo" = ( | ||||||
|  |         betterproto.message_field(16) | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
| import betterproto.lib.google.protobuf as betterproto_lib_google_protobuf | import betterproto.lib.google.protobuf as betterproto_lib_google_protobuf | ||||||
|   | |||||||
| @@ -33,5 +33,5 @@ def outputfile_compiler(output_file: OutputTemplate) -> str: | |||||||
|  |  | ||||||
|     return black.format_str( |     return black.format_str( | ||||||
|         template.render(output_file=output_file), |         template.render(output_file=output_file), | ||||||
|         mode=black.FileMode(target_versions={black.TargetVersion.PY37}), |         mode=black.Mode(), | ||||||
|     ) |     ) | ||||||
|   | |||||||
| @@ -28,11 +28,8 @@ def main() -> None: | |||||||
|     if dump_file: |     if dump_file: | ||||||
|         dump_request(dump_file, request) |         dump_request(dump_file, request) | ||||||
|  |  | ||||||
|     # Create response |  | ||||||
|     response = CodeGeneratorResponse() |  | ||||||
|  |  | ||||||
|     # Generate code |     # Generate code | ||||||
|     generate_code(request, response) |     response = generate_code(request) | ||||||
|  |  | ||||||
|     # Serialise response message |     # Serialise response message | ||||||
|     output = response.SerializeToString() |     output = response.SerializeToString() | ||||||
|   | |||||||
| @@ -30,6 +30,7 @@ reference to `A` to `B`'s `fields` attribute. | |||||||
| """ | """ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | import builtins | ||||||
| import betterproto | import betterproto | ||||||
| from betterproto import which_one_of | from betterproto import which_one_of | ||||||
| from betterproto.casing import sanitize_name | from betterproto.casing import sanitize_name | ||||||
| @@ -58,8 +59,7 @@ from betterproto.lib.google.protobuf.compiler import CodeGeneratorRequest | |||||||
| import re | import re | ||||||
| import textwrap | import textwrap | ||||||
| from dataclasses import dataclass, field | from dataclasses import dataclass, field | ||||||
| from typing import Dict, Iterable, Iterator, List, Optional, Set, Text, Type, Union | from typing import Dict, Iterable, Iterator, List, Optional, Set, Type, Union | ||||||
| import sys |  | ||||||
|  |  | ||||||
| from ..casing import sanitize_name | from ..casing import sanitize_name | ||||||
| from ..compile.importing import get_type_reference, parse_source_type_name | from ..compile.importing import get_type_reference, parse_source_type_name | ||||||
| @@ -237,6 +237,7 @@ class OutputTemplate: | |||||||
|     imports: Set[str] = field(default_factory=set) |     imports: Set[str] = field(default_factory=set) | ||||||
|     datetime_imports: Set[str] = field(default_factory=set) |     datetime_imports: Set[str] = field(default_factory=set) | ||||||
|     typing_imports: Set[str] = field(default_factory=set) |     typing_imports: Set[str] = field(default_factory=set) | ||||||
|  |     builtins_import: bool = False | ||||||
|     messages: List["MessageCompiler"] = field(default_factory=list) |     messages: List["MessageCompiler"] = field(default_factory=list) | ||||||
|     enums: List["EnumDefinitionCompiler"] = field(default_factory=list) |     enums: List["EnumDefinitionCompiler"] = field(default_factory=list) | ||||||
|     services: List["ServiceCompiler"] = field(default_factory=list) |     services: List["ServiceCompiler"] = field(default_factory=list) | ||||||
| @@ -268,6 +269,8 @@ class OutputTemplate: | |||||||
|         imports = set() |         imports = set() | ||||||
|         if any(x for x in self.messages if any(x.deprecated_fields)): |         if any(x for x in self.messages if any(x.deprecated_fields)): | ||||||
|             imports.add("warnings") |             imports.add("warnings") | ||||||
|  |         if self.builtins_import: | ||||||
|  |             imports.add("builtins") | ||||||
|         return imports |         return imports | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -283,6 +286,7 @@ class MessageCompiler(ProtoContentBase): | |||||||
|         default_factory=list |         default_factory=list | ||||||
|     ) |     ) | ||||||
|     deprecated: bool = field(default=False, init=False) |     deprecated: bool = field(default=False, init=False) | ||||||
|  |     builtins_types: Set[str] = field(default_factory=set) | ||||||
|  |  | ||||||
|     def __post_init__(self) -> None: |     def __post_init__(self) -> None: | ||||||
|         # Add message to output file |         # Add message to output file | ||||||
| @@ -376,6 +380,8 @@ class FieldCompiler(MessageCompiler): | |||||||
|         betterproto_field_type = ( |         betterproto_field_type = ( | ||||||
|             f"betterproto.{self.field_type}_field({self.proto_obj.number}{field_args})" |             f"betterproto.{self.field_type}_field({self.proto_obj.number}{field_args})" | ||||||
|         ) |         ) | ||||||
|  |         if self.py_name in dir(builtins): | ||||||
|  |             self.parent.builtins_types.add(self.py_name) | ||||||
|         return f"{name}{annotations} = {betterproto_field_type}" |         return f"{name}{annotations} = {betterproto_field_type}" | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
| @@ -383,6 +389,8 @@ class FieldCompiler(MessageCompiler): | |||||||
|         args = [] |         args = [] | ||||||
|         if self.field_wraps: |         if self.field_wraps: | ||||||
|             args.append(f"wraps={self.field_wraps}") |             args.append(f"wraps={self.field_wraps}") | ||||||
|  |         if self.optional: | ||||||
|  |             args.append(f"optional=True") | ||||||
|         return args |         return args | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
| @@ -408,9 +416,16 @@ class FieldCompiler(MessageCompiler): | |||||||
|             imports.add("Dict") |             imports.add("Dict") | ||||||
|         return imports |         return imports | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def use_builtins(self) -> bool: | ||||||
|  |         return self.py_type in self.parent.builtins_types or ( | ||||||
|  |             self.py_type == self.py_name and self.py_name in dir(builtins) | ||||||
|  |         ) | ||||||
|  |  | ||||||
|     def add_imports_to(self, output_file: OutputTemplate) -> None: |     def add_imports_to(self, output_file: OutputTemplate) -> None: | ||||||
|         output_file.datetime_imports.update(self.datetime_imports) |         output_file.datetime_imports.update(self.datetime_imports) | ||||||
|         output_file.typing_imports.update(self.typing_imports) |         output_file.typing_imports.update(self.typing_imports) | ||||||
|  |         output_file.builtins_import = output_file.builtins_import or self.use_builtins | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def field_wraps(self) -> Optional[str]: |     def field_wraps(self) -> Optional[str]: | ||||||
| @@ -431,6 +446,10 @@ class FieldCompiler(MessageCompiler): | |||||||
|             and not is_map(self.proto_obj, self.parent) |             and not is_map(self.proto_obj, self.parent) | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def optional(self) -> bool: | ||||||
|  |         return self.proto_obj.proto3_optional | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def mutable(self) -> bool: |     def mutable(self) -> bool: | ||||||
|         """True if the field is a mutable type, otherwise False.""" |         """True if the field is a mutable type, otherwise False.""" | ||||||
| @@ -446,10 +465,12 @@ class FieldCompiler(MessageCompiler): | |||||||
|         ) |         ) | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def default_value_string(self) -> Union[Text, None, float, int]: |     def default_value_string(self) -> str: | ||||||
|         """Python representation of the default proto value.""" |         """Python representation of the default proto value.""" | ||||||
|         if self.repeated: |         if self.repeated: | ||||||
|             return "[]" |             return "[]" | ||||||
|  |         if self.optional: | ||||||
|  |             return "None" | ||||||
|         if self.py_type == "int": |         if self.py_type == "int": | ||||||
|             return "0" |             return "0" | ||||||
|         if self.py_type == "float": |         if self.py_type == "float": | ||||||
| @@ -460,6 +481,14 @@ class FieldCompiler(MessageCompiler): | |||||||
|             return '""' |             return '""' | ||||||
|         elif self.py_type == "bytes": |         elif self.py_type == "bytes": | ||||||
|             return 'b""' |             return 'b""' | ||||||
|  |         elif self.field_type == "enum": | ||||||
|  |             enum_proto_obj_name = self.proto_obj.type_name.split(".").pop() | ||||||
|  |             enum = next( | ||||||
|  |                 e | ||||||
|  |                 for e in self.output_file.enums | ||||||
|  |                 if e.proto_obj.name == enum_proto_obj_name | ||||||
|  |             ) | ||||||
|  |             return enum.default_value_string | ||||||
|         else: |         else: | ||||||
|             # Message type |             # Message type | ||||||
|             return "None" |             return "None" | ||||||
| @@ -504,9 +533,14 @@ class FieldCompiler(MessageCompiler): | |||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def annotation(self) -> str: |     def annotation(self) -> str: | ||||||
|  |         py_type = self.py_type | ||||||
|  |         if self.use_builtins: | ||||||
|  |             py_type = f"builtins.{py_type}" | ||||||
|         if self.repeated: |         if self.repeated: | ||||||
|             return f"List[{self.py_type}]" |             return f"List[{py_type}]" | ||||||
|         return self.py_type |         if self.optional: | ||||||
|  |             return f"Optional[{py_type}]" | ||||||
|  |         return py_type | ||||||
|  |  | ||||||
|  |  | ||||||
| @dataclass | @dataclass | ||||||
|   | |||||||
| @@ -8,6 +8,7 @@ from betterproto.lib.google.protobuf import ( | |||||||
| from betterproto.lib.google.protobuf.compiler import ( | from betterproto.lib.google.protobuf.compiler import ( | ||||||
|     CodeGeneratorRequest, |     CodeGeneratorRequest, | ||||||
|     CodeGeneratorResponse, |     CodeGeneratorResponse, | ||||||
|  |     CodeGeneratorResponseFeature, | ||||||
|     CodeGeneratorResponseFile, |     CodeGeneratorResponseFile, | ||||||
| ) | ) | ||||||
| import itertools | import itertools | ||||||
| @@ -60,10 +61,11 @@ def traverse( | |||||||
|     ) |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
| def generate_code( | def generate_code(request: CodeGeneratorRequest) -> CodeGeneratorResponse: | ||||||
|     request: CodeGeneratorRequest, response: CodeGeneratorResponse |     response = CodeGeneratorResponse() | ||||||
| ) -> None: |  | ||||||
|     plugin_options = request.parameter.split(",") if request.parameter else [] |     plugin_options = request.parameter.split(",") if request.parameter else [] | ||||||
|  |     response.supported_features = CodeGeneratorResponseFeature.FEATURE_PROTO3_OPTIONAL | ||||||
|  |  | ||||||
|     request_data = PluginRequestCompiler(plugin_request_obj=request) |     request_data = PluginRequestCompiler(plugin_request_obj=request) | ||||||
|     # Gather output packages |     # Gather output packages | ||||||
| @@ -133,6 +135,8 @@ def generate_code( | |||||||
|     for output_package_name in sorted(output_paths.union(init_files)): |     for output_package_name in sorted(output_paths.union(init_files)): | ||||||
|         print(f"Writing {output_package_name}", file=sys.stderr) |         print(f"Writing {output_package_name}", file=sys.stderr) | ||||||
|  |  | ||||||
|  |     return response | ||||||
|  |  | ||||||
|  |  | ||||||
| def read_protobuf_type( | def read_protobuf_type( | ||||||
|     item: DescriptorProto, |     item: DescriptorProto, | ||||||
|   | |||||||
| @@ -60,13 +60,15 @@ async def generate(whitelist: Set[str], verbose: bool): | |||||||
|         if result != 0: |         if result != 0: | ||||||
|             failed_test_cases.append(test_case_name) |             failed_test_cases.append(test_case_name) | ||||||
|  |  | ||||||
|     if failed_test_cases: |     if len(failed_test_cases) > 0: | ||||||
|         sys.stderr.write( |         sys.stderr.write( | ||||||
|             "\n\033[31;1;4mFailed to generate the following test cases:\033[0m\n" |             "\n\033[31;1;4mFailed to generate the following test cases:\033[0m\n" | ||||||
|         ) |         ) | ||||||
|         for failed_test_case in failed_test_cases: |         for failed_test_case in failed_test_cases: | ||||||
|             sys.stderr.write(f"- {failed_test_case}\n") |             sys.stderr.write(f"- {failed_test_case}\n") | ||||||
|  |  | ||||||
|  |         sys.exit(1) | ||||||
|  |  | ||||||
|  |  | ||||||
| async def generate_test_case_output( | async def generate_test_case_output( | ||||||
|     test_case_input_path: Path, test_case_name: str, verbose: bool |     test_case_input_path: Path, test_case_name: str, verbose: bool | ||||||
| @@ -92,21 +94,41 @@ async def generate_test_case_output( | |||||||
|         protoc(test_case_input_path, test_case_output_path_betterproto, False), |         protoc(test_case_input_path, test_case_output_path_betterproto, False), | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     message = f"Generated output for {test_case_name!r}" |     if ref_code == 0: | ||||||
|     if verbose: |         print(f"\033[31;1;4mGenerated reference output for {test_case_name!r}\033[0m") | ||||||
|         print(f"\033[31;1;4m{message}\033[0m") |  | ||||||
|         if ref_out: |  | ||||||
|             sys.stdout.buffer.write(ref_out) |  | ||||||
|         if ref_err: |  | ||||||
|             sys.stderr.buffer.write(ref_err) |  | ||||||
|         if plg_out: |  | ||||||
|             sys.stdout.buffer.write(plg_out) |  | ||||||
|         if plg_err: |  | ||||||
|             sys.stderr.buffer.write(plg_err) |  | ||||||
|         sys.stdout.buffer.flush() |  | ||||||
|         sys.stderr.buffer.flush() |  | ||||||
|     else: |     else: | ||||||
|         print(message) |         print( | ||||||
|  |             f"\033[31;1;4mFailed to generate reference output for {test_case_name!r}\033[0m" | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     if verbose: | ||||||
|  |         if ref_out: | ||||||
|  |             print("Reference stdout:") | ||||||
|  |             sys.stdout.buffer.write(ref_out) | ||||||
|  |             sys.stdout.buffer.flush() | ||||||
|  |  | ||||||
|  |         if ref_err: | ||||||
|  |             print("Reference stderr:") | ||||||
|  |             sys.stderr.buffer.write(ref_err) | ||||||
|  |             sys.stderr.buffer.flush() | ||||||
|  |  | ||||||
|  |     if plg_code == 0: | ||||||
|  |         print(f"\033[31;1;4mGenerated plugin output for {test_case_name!r}\033[0m") | ||||||
|  |     else: | ||||||
|  |         print( | ||||||
|  |             f"\033[31;1;4mFailed to generate plugin output for {test_case_name!r}\033[0m" | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     if verbose: | ||||||
|  |         if plg_out: | ||||||
|  |             print("Plugin stdout:") | ||||||
|  |             sys.stdout.buffer.write(plg_out) | ||||||
|  |             sys.stdout.buffer.flush() | ||||||
|  |  | ||||||
|  |         if plg_err: | ||||||
|  |             print("Plugin stderr:") | ||||||
|  |             sys.stderr.buffer.write(plg_err) | ||||||
|  |             sys.stderr.buffer.flush() | ||||||
|  |  | ||||||
|     return max(ref_code, plg_code) |     return max(ref_code, plg_code) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -0,0 +1,7 @@ | |||||||
|  | { | ||||||
|  |     "int": 26, | ||||||
|  |     "float": 26.0, | ||||||
|  |     "str": "value-for-str", | ||||||
|  |     "bytes": "001a", | ||||||
|  |     "bool": true | ||||||
|  | } | ||||||
| @@ -0,0 +1,11 @@ | |||||||
|  | syntax = "proto3"; | ||||||
|  |  | ||||||
|  | // Tests that messages may contain fields with names that are identical to their python types (PR #294) | ||||||
|  |  | ||||||
|  | message Test { | ||||||
|  |     int32 int = 1; | ||||||
|  |     float float = 2; | ||||||
|  |     string str = 3; | ||||||
|  |     bytes bytes = 4; | ||||||
|  |     bool bool = 5; | ||||||
|  | } | ||||||
| @@ -0,0 +1,12 @@ | |||||||
|  | { | ||||||
|  |   "test1": 128, | ||||||
|  |   "test2": true, | ||||||
|  |   "test3": "A value", | ||||||
|  |   "test4": "aGVsbG8=", | ||||||
|  |   "test5": { | ||||||
|  |     "test": "Hello" | ||||||
|  |   }, | ||||||
|  |   "test6": "B", | ||||||
|  |   "test7": "8589934592", | ||||||
|  |   "test8": 2.5 | ||||||
|  | } | ||||||
| @@ -0,0 +1,21 @@ | |||||||
|  | syntax = "proto3"; | ||||||
|  |  | ||||||
|  | message InnerTest { | ||||||
|  |     string test = 1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | message Test { | ||||||
|  |     optional uint32 test1 = 1; | ||||||
|  |     optional bool test2 = 2; | ||||||
|  |     optional string test3 = 3; | ||||||
|  |     optional bytes test4 = 4; | ||||||
|  |     optional InnerTest test5 = 5; | ||||||
|  |     optional TestEnum test6 = 6; | ||||||
|  |     optional uint64 test7 = 7; | ||||||
|  |     optional float test8 = 8; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | enum TestEnum { | ||||||
|  |     A = 0; | ||||||
|  |     B = 1; | ||||||
|  | } | ||||||
| @@ -0,0 +1 @@ | |||||||
|  | {} | ||||||
| @@ -0,0 +1,9 @@ | |||||||
|  | { | ||||||
|  |   "test1": 0, | ||||||
|  |   "test2": false, | ||||||
|  |   "test3": "", | ||||||
|  |   "test4": "", | ||||||
|  |   "test6": "A", | ||||||
|  |   "test7": "0", | ||||||
|  |   "test8": 0 | ||||||
|  | } | ||||||
| @@ -0,0 +1,38 @@ | |||||||
|  | import json | ||||||
|  |  | ||||||
|  | from tests.output_betterproto.proto3_field_presence import Test, InnerTest, TestEnum | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_null_fields_json(): | ||||||
|  |     """Ensure that using "null" in JSON is equivalent to not specifying a | ||||||
|  |     field, for fields with explicit presence""" | ||||||
|  |  | ||||||
|  |     def test_json(ref_json: str, obj_json: str) -> None: | ||||||
|  |         """`ref_json` and `obj_json` are JSON strings describing a `Test` object. | ||||||
|  |         Test that deserializing both leads to the same object, and that | ||||||
|  |         `ref_json` is the normalized format.""" | ||||||
|  |         ref_obj = Test().from_json(ref_json) | ||||||
|  |         obj = Test().from_json(obj_json) | ||||||
|  |  | ||||||
|  |         assert obj == ref_obj | ||||||
|  |         assert json.loads(obj.to_json(0)) == json.loads(ref_json) | ||||||
|  |  | ||||||
|  |     test_json("{}", '{ "test1": null, "test2": null, "test3": null }') | ||||||
|  |     test_json("{}", '{ "test4": null, "test5": null, "test6": null }') | ||||||
|  |     test_json("{}", '{ "test7": null, "test8": null }') | ||||||
|  |     test_json('{ "test5": {} }', '{ "test3": null, "test5": {} }') | ||||||
|  |  | ||||||
|  |     # Make sure that if include_default_values is set, None values are | ||||||
|  |     # exported. | ||||||
|  |     obj = Test() | ||||||
|  |     assert obj.to_dict() == {} | ||||||
|  |     assert obj.to_dict(include_default_values=True) == { | ||||||
|  |         "test1": None, | ||||||
|  |         "test2": None, | ||||||
|  |         "test3": None, | ||||||
|  |         "test4": None, | ||||||
|  |         "test5": None, | ||||||
|  |         "test6": None, | ||||||
|  |         "test7": None, | ||||||
|  |         "test8": None, | ||||||
|  |     } | ||||||
| @@ -0,0 +1,3 @@ | |||||||
|  | { | ||||||
|  |   "nested": {} | ||||||
|  | } | ||||||
| @@ -0,0 +1,20 @@ | |||||||
|  | syntax = "proto3"; | ||||||
|  |  | ||||||
|  | message Test { | ||||||
|  |     oneof kind { | ||||||
|  |         Nested nested = 1; | ||||||
|  |         WithOptional with_optional = 2; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | message InnerNested { | ||||||
|  |     optional bool a = 1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | message Nested { | ||||||
|  |     InnerNested inner = 1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | message WithOptional { | ||||||
|  |     optional bool b = 2; | ||||||
|  | } | ||||||
| @@ -0,0 +1,29 @@ | |||||||
|  | from tests.output_betterproto.proto3_field_presence_oneof import ( | ||||||
|  |     Test, | ||||||
|  |     InnerNested, | ||||||
|  |     Nested, | ||||||
|  |     WithOptional, | ||||||
|  | ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_serialization(): | ||||||
|  |     """Ensure that serialization of fields unset but with explicit field | ||||||
|  |     presence do not bloat the serialized payload with length-delimited fields | ||||||
|  |     with length 0""" | ||||||
|  |  | ||||||
|  |     def test_empty_nested(message: Test) -> None: | ||||||
|  |         # '0a' => tag 1, length delimited | ||||||
|  |         # '00' => length: 0 | ||||||
|  |         assert bytes(message) == bytearray.fromhex("0a 00") | ||||||
|  |  | ||||||
|  |     test_empty_nested(Test(nested=Nested())) | ||||||
|  |     test_empty_nested(Test(nested=Nested(inner=None))) | ||||||
|  |     test_empty_nested(Test(nested=Nested(inner=InnerNested(a=None)))) | ||||||
|  |  | ||||||
|  |     def test_empty_with_optional(message: Test) -> None: | ||||||
|  |         # '12' => tag 2, length delimited | ||||||
|  |         # '00' => length: 0 | ||||||
|  |         assert bytes(message) == bytearray.fromhex("12 00") | ||||||
|  |  | ||||||
|  |     test_empty_with_optional(Test(with_optional=WithOptional())) | ||||||
|  |     test_empty_with_optional(Test(with_optional=WithOptional(b=None))) | ||||||
| @@ -2,9 +2,16 @@ syntax = "proto3"; | |||||||
|  |  | ||||||
| package service; | package service; | ||||||
|  |  | ||||||
|  | enum ThingType { | ||||||
|  |   UNKNOWN = 0; | ||||||
|  |   LIVING = 1; | ||||||
|  |   DEAD = 2; | ||||||
|  | } | ||||||
|  |  | ||||||
| message DoThingRequest { | message DoThingRequest { | ||||||
|   string name = 1; |   string name = 1; | ||||||
|   repeated string comments = 2; |   repeated string comments = 2; | ||||||
|  |   ThingType type = 3; | ||||||
| } | } | ||||||
|  |  | ||||||
| message DoThingResponse { | message DoThingResponse { | ||||||
|   | |||||||
| @@ -1,7 +1,8 @@ | |||||||
| import betterproto | import betterproto | ||||||
| from dataclasses import dataclass | from dataclasses import dataclass | ||||||
| from typing import Optional, List, Dict | from typing import Optional, List, Dict | ||||||
| from datetime import datetime, timedelta | from datetime import datetime | ||||||
|  | from inspect import signature | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_has_field(): | def test_has_field(): | ||||||
| @@ -476,3 +477,10 @@ def test_iso_datetime_list(): | |||||||
|  |  | ||||||
|     msg.from_dict({"timestamps": iso_candidates}) |     msg.from_dict({"timestamps": iso_candidates}) | ||||||
|     assert all([isinstance(item, datetime) for item in msg.timestamps]) |     assert all([isinstance(item, datetime) for item in msg.timestamps]) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_enum_service_argument__expected_default_value(): | ||||||
|  |     from tests.output_betterproto.service.service import ThingType, TestStub | ||||||
|  |  | ||||||
|  |     sig = signature(TestStub.do_thing) | ||||||
|  |     assert sig.parameters["type"].default == ThingType.UNKNOWN | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user