From 28435b0c6950c4d4da575407bb098c0ad4e03ebf Mon Sep 17 00:00:00 2001 From: Bu Sun Kim Date: Wed, 17 Jun 2020 20:38:49 +0000 Subject: [PATCH 1/7] fix: pass metadata to pagers --- .../%sub/services/%service/async_client.py.j2 | 1 + .../%sub/services/%service/client.py.j2 | 1 + .../%sub/services/%service/pagers.py.j2 | 14 ++++++++++---- 3 files changed, 12 insertions(+), 4 deletions(-) diff --git a/gapic/templates/%namespace/%name_%version/%sub/services/%service/async_client.py.j2 b/gapic/templates/%namespace/%name_%version/%sub/services/%service/async_client.py.j2 index fb501fe2bc..36a34471f8 100644 --- a/gapic/templates/%namespace/%name_%version/%sub/services/%service/async_client.py.j2 +++ b/gapic/templates/%namespace/%name_%version/%sub/services/%service/async_client.py.j2 @@ -244,6 +244,7 @@ class {{ service.async_client_name }}: method=rpc, request=request, response=response, + metadata=metadata, ) {%- endif %} {%- if not method.void %} diff --git a/gapic/templates/%namespace/%name_%version/%sub/services/%service/client.py.j2 b/gapic/templates/%namespace/%name_%version/%sub/services/%service/client.py.j2 index 48efc9de0b..7e2150b756 100644 --- a/gapic/templates/%namespace/%name_%version/%sub/services/%service/client.py.j2 +++ b/gapic/templates/%namespace/%name_%version/%sub/services/%service/client.py.j2 @@ -367,6 +367,7 @@ class {{ service.client_name }}(metaclass={{ service.client_name }}Meta): method=rpc, request=request, response=response, + metadata=metadata, ) {%- endif %} {%- if not method.void %} diff --git a/gapic/templates/%namespace/%name_%version/%sub/services/%service/pagers.py.j2 b/gapic/templates/%namespace/%name_%version/%sub/services/%service/pagers.py.j2 index 5c069b68fd..3f576fe1bd 100644 --- a/gapic/templates/%namespace/%name_%version/%sub/services/%service/pagers.py.j2 +++ b/gapic/templates/%namespace/%name_%version/%sub/services/%service/pagers.py.j2 @@ -38,7 +38,9 @@ class {{ method.name }}Pager: method: Callable[[{{ method.input.ident }}], {{ method.output.ident }}], request: {{ method.input.ident }}, - response: {{ method.output.ident }}): + response: {{ method.output.ident }}, + *, + metadata: Sequence[Tuple[str, str]] = ()): """Instantiate the pager. Args: @@ -52,6 +54,7 @@ class {{ method.name }}Pager: self._method = method self._request = {{ method.input.ident }}(request) self._response = response + self._metadata = metadata def __getattr__(self, name: str) -> Any: return getattr(self._response, name) @@ -61,7 +64,7 @@ class {{ method.name }}Pager: yield self._response while self._response.next_page_token: self._request.page_token = self._response.next_page_token - self._response = self._method(self._request) + self._response = self._method(self._request, metadata=metadata) yield self._response def __iter__(self) -> {{ method.paged_result_field.ident | replace('Sequence', 'Iterable') }}: @@ -93,7 +96,9 @@ class {{ method.name }}AsyncPager: method: Callable[[{{ method.input.ident }}], Awaitable[{{ method.output.ident }}]], request: {{ method.input.ident }}, - response: {{ method.output.ident }}): + response: {{ method.output.ident }}, + *, + metadata: Sequence[Tuple[str, str]] = ()): """Instantiate the pager. Args: @@ -107,6 +112,7 @@ class {{ method.name }}AsyncPager: self._method = method self._request = {{ method.input.ident }}(request) self._response = response + self._metadata = metadata def __getattr__(self, name: str) -> Any: return getattr(self._response, name) @@ -116,7 +122,7 @@ class {{ method.name }}AsyncPager: yield self._response while self._response.next_page_token: self._request.page_token = self._response.next_page_token - self._response = await self._method(self._request) + self._response = await self._method(self._request, metadata=self._metadata) yield self._response def __aiter__(self) -> {{ method.paged_result_field.ident | replace('Sequence', 'AsyncIterable') }}: From afed8589993cdc66884d83296005d9ace53012b4 Mon Sep 17 00:00:00 2001 From: Bu Sun Kim Date: Wed, 17 Jun 2020 20:46:16 +0000 Subject: [PATCH 2/7] docs: add docstrings --- .../%name_%version/%sub/services/%service/pagers.py.j2 | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/gapic/templates/%namespace/%name_%version/%sub/services/%service/pagers.py.j2 b/gapic/templates/%namespace/%name_%version/%sub/services/%service/pagers.py.j2 index 3f576fe1bd..e6be543b97 100644 --- a/gapic/templates/%namespace/%name_%version/%sub/services/%service/pagers.py.j2 +++ b/gapic/templates/%namespace/%name_%version/%sub/services/%service/pagers.py.j2 @@ -50,6 +50,8 @@ class {{ method.name }}Pager: The initial request object. response (:class:`{{ method.output.ident.sphinx }}`): The initial response object. + metadata (Sequence[Tuple[str, str]]): Strings which should be + sent along with the request as metadata. """ self._method = method self._request = {{ method.input.ident }}(request) @@ -108,6 +110,8 @@ class {{ method.name }}AsyncPager: The initial request object. response (:class:`{{ method.output.ident.sphinx }}`): The initial response object. + metadata (Sequence[Tuple[str, str]]): Strings which should be + sent along with the request as metadata. """ self._method = method self._request = {{ method.input.ident }}(request) From 30201dbf8d5b280e1db9a6867c20794f66ee9796 Mon Sep 17 00:00:00 2001 From: Bu Sun Kim Date: Wed, 17 Jun 2020 20:50:35 +0000 Subject: [PATCH 3/7] lint: add more types --- .../%name_%version/%sub/services/%service/pagers.py.j2 | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/gapic/templates/%namespace/%name_%version/%sub/services/%service/pagers.py.j2 b/gapic/templates/%namespace/%name_%version/%sub/services/%service/pagers.py.j2 index e6be543b97..a55f7fdfb7 100644 --- a/gapic/templates/%namespace/%name_%version/%sub/services/%service/pagers.py.j2 +++ b/gapic/templates/%namespace/%name_%version/%sub/services/%service/pagers.py.j2 @@ -6,7 +6,7 @@ {# This lives within the loop in order to ensure that this template is empty if there are no paged methods. -#} -from typing import Any, AsyncIterable, Awaitable, Callable, Iterable +from typing import Any, AsyncIterable, Awaitable, Callable, Iterable, Sequence, Tuple {% filter sort_lines -%} {% for method in service.methods.values() | selectattr('paged_result_field') -%} From 44105340ebc4ca2797b78920b2475f72b8c38bc3 Mon Sep 17 00:00:00 2001 From: Bu Sun Kim Date: Wed, 17 Jun 2020 21:43:20 +0000 Subject: [PATCH 4/7] test: fix test --- .../%name_%version/%sub/services/%service/pagers.py.j2 | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/gapic/templates/%namespace/%name_%version/%sub/services/%service/pagers.py.j2 b/gapic/templates/%namespace/%name_%version/%sub/services/%service/pagers.py.j2 index a55f7fdfb7..8924783e06 100644 --- a/gapic/templates/%namespace/%name_%version/%sub/services/%service/pagers.py.j2 +++ b/gapic/templates/%namespace/%name_%version/%sub/services/%service/pagers.py.j2 @@ -66,7 +66,7 @@ class {{ method.name }}Pager: yield self._response while self._response.next_page_token: self._request.page_token = self._response.next_page_token - self._response = self._method(self._request, metadata=metadata) + self._response = self._method(self._request, metadata=self._metadata) yield self._response def __iter__(self) -> {{ method.paged_result_field.ident | replace('Sequence', 'Iterable') }}: From b17974051e0e4d4fc3e7a84fdc56c3397f484512 Mon Sep 17 00:00:00 2001 From: Bu Sun Kim Date: Wed, 17 Jun 2020 22:51:00 +0000 Subject: [PATCH 5/7] lint: fix mypy --- .../%name_%version/%sub/services/%service/pagers.py.j2 | 6 ++---- 1 file changed, 2 insertions(+), 4 deletions(-) diff --git a/gapic/templates/%namespace/%name_%version/%sub/services/%service/pagers.py.j2 b/gapic/templates/%namespace/%name_%version/%sub/services/%service/pagers.py.j2 index 8924783e06..cc7bc56100 100644 --- a/gapic/templates/%namespace/%name_%version/%sub/services/%service/pagers.py.j2 +++ b/gapic/templates/%namespace/%name_%version/%sub/services/%service/pagers.py.j2 @@ -35,8 +35,7 @@ class {{ method.name }}Pager: the most recent response is retained, and thus used for attribute lookup. """ def __init__(self, - method: Callable[[{{ method.input.ident }}], - {{ method.output.ident }}], + method: Callable[..., {{ method.output.ident }}], request: {{ method.input.ident }}, response: {{ method.output.ident }}, *, @@ -95,8 +94,7 @@ class {{ method.name }}AsyncPager: the most recent response is retained, and thus used for attribute lookup. """ def __init__(self, - method: Callable[[{{ method.input.ident }}], - Awaitable[{{ method.output.ident }}]], + method: Callable[..., Awaitable[{{ method.output.ident }}]], request: {{ method.input.ident }}, response: {{ method.output.ident }}, *, From 81b528ff7813d003889693150a5938db87249946 Mon Sep 17 00:00:00 2001 From: Bu Sun Kim Date: Thu, 18 Jun 2020 03:23:28 +0000 Subject: [PATCH 6/7] test: check pager metadata --- gapic/schema/wrappers.py | 5 ++++ .../%name_%version/%sub/test_%service.py.j2 | 24 ++++++++++++++++--- 2 files changed, 26 insertions(+), 3 deletions(-) diff --git a/gapic/schema/wrappers.py b/gapic/schema/wrappers.py index 5a632fcc02..dda4bf9747 100644 --- a/gapic/schema/wrappers.py +++ b/gapic/schema/wrappers.py @@ -832,6 +832,11 @@ def has_lro(self) -> bool: """Return whether the service has a long-running method.""" return any([m.lro for m in self.methods.values()]) + @property + def has_pagers(self) -> bool: + """Return whether the service has paged methods.""" + return any([m.paged_result_field for m in self.methods.values()]) + @property def host(self) -> str: """Return the hostname for this service, if specified. diff --git a/gapic/templates/tests/unit/gapic/%name_%version/%sub/test_%service.py.j2 b/gapic/templates/tests/unit/gapic/%name_%version/%sub/test_%service.py.j2 index 550ca1bac4..2812528433 100644 --- a/gapic/templates/tests/unit/gapic/%name_%version/%sub/test_%service.py.j2 +++ b/gapic/templates/tests/unit/gapic/%name_%version/%sub/test_%service.py.j2 @@ -26,6 +26,9 @@ from google.api_core import future from google.api_core import operations_v1 from google.longrunning import operations_pb2 {% endif -%} +{% if service.has_pagers -%} +from google.api_core import gapic_v1 +{% endif -%} {% for method in service.methods.values() -%} {% for ref_type in method.ref_types if not ((ref_type.ident.python_import.package == ('google', 'api_core') and ref_type.ident.python_import.module == 'operation') @@ -635,9 +638,24 @@ def test_{{ method.name|snake_case }}_pager(): ), RuntimeError, ) - results = [i for i in client.{{ method.name|snake_case }}( - request={}, - )] + + metadata = () + {% if method.field_headers -%} + metadata = tuple(metadata) + ( + gapic_v1.routing_header.to_grpc_metadata(( + {%- for field_header in method.field_headers %} + {%- if not method.client_streaming %} + ('{{ field_header }}', ''), + {%- endif %} + {%- endfor %} + )), + ) + {% endif -%} + pager = client.{{ method.name|snake_case }}(request={}) + + assert pager._metadata == metadata + + results = [i for i in pager] assert len(results) == 6 assert all(isinstance(i, {{ method.paged_result_field.message.ident }}) for i in results) From b6751ef898d44bfddc3a12d90f984d4be64dca5b Mon Sep 17 00:00:00 2001 From: Dov Shlachter Date: Tue, 7 Jul 2020 15:18:23 -0700 Subject: [PATCH 7/7] Coverage --- gapic/schema/wrappers.py | 2 +- tests/unit/schema/wrappers/test_service.py | 35 ++++++++++++++++++++++ 2 files changed, 36 insertions(+), 1 deletion(-) diff --git a/gapic/schema/wrappers.py b/gapic/schema/wrappers.py index dda4bf9747..6f7e041f16 100644 --- a/gapic/schema/wrappers.py +++ b/gapic/schema/wrappers.py @@ -835,7 +835,7 @@ def has_lro(self) -> bool: @property def has_pagers(self) -> bool: """Return whether the service has paged methods.""" - return any([m.paged_result_field for m in self.methods.values()]) + return any(m.paged_result_field for m in self.methods.values()) @property def host(self) -> str: diff --git a/tests/unit/schema/wrappers/test_service.py b/tests/unit/schema/wrappers/test_service.py index 86d1aa2e97..8502617b5d 100644 --- a/tests/unit/schema/wrappers/test_service.py +++ b/tests/unit/schema/wrappers/test_service.py @@ -260,3 +260,38 @@ def test_service_any_streaming(): assert service.any_client_streaming == client assert service.any_server_streaming == server + + +def test_has_pagers(): + paged = make_field(name='foos', message=make_message('Foo'), repeated=True) + input_msg = make_message( + name='ListFoosRequest', + fields=( + make_field(name='parent', type=9), # str + make_field(name='page_size', type=5), # int + make_field(name='page_token', type=9), # str + ), + ) + output_msg = make_message( + name='ListFoosResponse', + fields=( + paged, + make_field(name='next_page_token', type=9), # str + ), + ) + method = make_method( + 'ListFoos', + input_message=input_msg, + output_message=output_msg, + ) + + service = make_service(name="Fooer", methods=(method,),) + assert service.has_pagers + + other_service = make_service( + name="Unfooer", + methods=( + get_method("Unfoo", "foo.bar.UnfooReq", "foo.bar.UnFooResp"), + ), + ) + assert not other_service.has_pagers