From 4613cb35e483947748fde50151982a641f0b21e2 Mon Sep 17 00:00:00 2001 From: Anthonios Partheniou Date: Tue, 24 Sep 2024 00:23:40 +0000 Subject: [PATCH 01/14] feat: implement `AbstractOperationsAsyncClient` to support long running operations --- google/api_core/operations_v1/__init__.py | 14 +- .../abstract_operations_async_client.py | 331 +++++++++ .../abstract_operations_base_client.py | 416 ++++++++++++ .../abstract_operations_client.py | 286 +------- google/api_core/operations_v1/pagers.py | 29 +- google/api_core/operations_v1/pagers_async.py | 71 ++ google/api_core/operations_v1/pagers_base.py | 73 ++ .../operations_v1/transports/__init__.py | 4 + .../test_operations_rest_client.py | 634 ++++++++++++++---- 9 files changed, 1428 insertions(+), 430 deletions(-) create mode 100644 google/api_core/operations_v1/abstract_operations_async_client.py create mode 100644 google/api_core/operations_v1/abstract_operations_base_client.py create mode 100644 google/api_core/operations_v1/pagers_async.py create mode 100644 google/api_core/operations_v1/pagers_base.py diff --git a/google/api_core/operations_v1/__init__.py b/google/api_core/operations_v1/__init__.py index 52f83fcd..745bcfdf 100644 --- a/google/api_core/operations_v1/__init__.py +++ b/google/api_core/operations_v1/__init__.py @@ -14,9 +14,7 @@ """Package for interacting with the google.longrunning.operations meta-API.""" -from google.api_core.operations_v1.abstract_operations_client import ( - AbstractOperationsClient, -) +from google.api_core.operations_v1.abstract_operations_client import AbstractOperationsClient from google.api_core.operations_v1.operations_async_client import OperationsAsyncClient from google.api_core.operations_v1.operations_client import OperationsClient from google.api_core.operations_v1.transports.rest import OperationsRestTransport @@ -29,10 +27,14 @@ ] try: - from google.api_core.operations_v1.transports.rest_asyncio import OperationsRestAsyncTransport # noqa: F401 - __all__.append("OperationsRestAsyncTransport") + from google.api_core.operations_v1.transports.rest_asyncio import ( + OperationsRestAsyncTransport, + ) + from google.api_core.operations_v1.abstract_operations_async_client import AbstractOperationsAsyncClient + + __all__.extend(["AbstractOperationsAsyncClient", "OperationsRestAsyncTransport"]) except ImportError: # This import requires the `async_rest` extra. # Don't raise an exception if `OperationsRestAsyncTransport` cannot be imported - # as other transports are still available. + # as other transports are still available pass diff --git a/google/api_core/operations_v1/abstract_operations_async_client.py b/google/api_core/operations_v1/abstract_operations_async_client.py new file mode 100644 index 00000000..12a194f3 --- /dev/null +++ b/google/api_core/operations_v1/abstract_operations_async_client.py @@ -0,0 +1,331 @@ +# -*- coding: utf-8 -*- +# Copyright 2024 Google LLC +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +from typing import Optional, Sequence, Tuple, Union + +from google.api_core import client_options as client_options_lib # type: ignore +from google.api_core import gapic_v1 # type: ignore +from google.api_core.operations_v1 import pagers_async as pagers +from google.api_core.operations_v1.transports.base import ( + DEFAULT_CLIENT_INFO, + OperationsTransport, +) +from google.api_core.operations_v1.abstract_operations_base_client import ( + AbstractOperationsBaseClient, +) + +try: + from google.auth.aio import credentials as ga_credentials # type: ignore +except ImportError as e: # pragma: NO COVER + raise ImportError( + "`google-api-core[async_rest]` is required to use asynchronous rest streaming. " + "Install the `async_rest` extra of `google-api-core` using " + "`pip install google-api-core[async_rest]`." + ) from e + +from google.auth.aio import credentials as ga_credentials # type: ignore +from google.longrunning import operations_pb2 + + +class AbstractOperationsAsyncClient(AbstractOperationsBaseClient): + """Manages long-running operations with an API service for the asynchronous client. + + When an API method normally takes long time to complete, it can be + designed to return [Operation][google.api_core.operations_v1.Operation] to the + client, and the client can use this interface to receive the real + response asynchronously by polling the operation resource, or pass + the operation resource to another API (such as Google Cloud Pub/Sub + API) to receive the response. Any API service that returns + long-running operations should implement the ``Operations`` + interface so developers can have a consistent client experience. + """ + + def __init__( + self, + *, + credentials: Optional[ga_credentials.Credentials] = None, + transport: Union[str, OperationsTransport, None] = None, + client_options: Optional[client_options_lib.ClientOptions] = None, + client_info: gapic_v1.client_info.ClientInfo = DEFAULT_CLIENT_INFO, + ) -> None: + """Instantiates the operations client. + + Args: + credentials (Optional[google.auth.aio.credentials.Credentials]): The + authorization credentials to attach to requests. These + credentials identify the application to the service; if none + are specified, the client will attempt to ascertain the + credentials from the environment. + transport (Union[str, OperationsTransport]): The + transport to use. If set to None, a transport is chosen + automatically. + client_options (google.api_core.client_options.ClientOptions): Custom options for the + client. It won't take effect if a ``transport`` instance is provided. + (1) The ``api_endpoint`` property can be used to override the + default endpoint provided by the client. GOOGLE_API_USE_MTLS_ENDPOINT + environment variable can also be used to override the endpoint: + "always" (always use the default mTLS endpoint), "never" (always + use the default regular endpoint) and "auto" (auto switch to the + default mTLS endpoint if client certificate is present, this is + the default value). However, the ``api_endpoint`` property takes + precedence if provided. + (2) If GOOGLE_API_USE_CLIENT_CERTIFICATE environment variable + is "true", then the ``client_cert_source`` property can be used + to provide client certificate for mutual TLS transport. If + not provided, the default SSL client certificate will be used if + present. If GOOGLE_API_USE_CLIENT_CERTIFICATE is "false" or not + set, no client certificate will be used. + client_info (google.api_core.gapic_v1.client_info.ClientInfo): + The client info used to send a user-agent string along with + API requests. If ``None``, then default info will be used. + Generally, you only need to set this if you're developing + your own client library. + + Raises: + google.auth.exceptions.MutualTLSChannelError: If mutual TLS transport + creation failed for any reason. + """ + super().__init__( + credentials=credentials, # type: ignore + transport=transport, + client_options=client_options, + client_info=client_info, + ) + + async def list_operations( + self, + name: str, + filter_: Optional[str] = None, + *, + page_size: Optional[int] = None, + page_token: Optional[str] = None, + timeout: Optional[float] = None, + metadata: Sequence[Tuple[str, str]] = (), + ) -> pagers.ListOperationsAsyncPager: + r"""Lists operations that match the specified filter in the request. + If the server doesn't support this method, it returns + ``UNIMPLEMENTED``. + + NOTE: the ``name`` binding allows API services to override the + binding to use different resource name schemes, such as + ``users/*/operations``. To override the binding, API services + can add a binding such as ``"/v1/{name=users/*}/operations"`` to + their service configuration. For backwards compatibility, the + default name includes the operations collection id, however + overriding users must ensure the name binding is the parent + resource, without the operations collection id. + + Args: + name (str): + The name of the operation's parent + resource. + filter_ (str): + The standard list filter. + This corresponds to the ``filter`` field + on the ``request`` instance; if ``request`` is provided, this + should not be set. + timeout (float): The timeout for this request. + metadata (Sequence[Tuple[str, str]]): Strings which should be + sent along with the request as metadata. + + Returns: + google.api_core.operations_v1.pagers.ListOperationsPager: + The response message for + [Operations.ListOperations][google.api_core.operations_v1.Operations.ListOperations]. + + Iterating over this object will yield results and + resolve additional pages automatically. + + """ + # Create a protobuf request object. + request = operations_pb2.ListOperationsRequest(name=name, filter=filter_) + if page_size is not None: + request.page_size = page_size + if page_token is not None: + request.page_token = page_token + + # Wrap the RPC method; this adds retry and timeout information, + # and friendly error handling. + rpc = self._transport._wrapped_methods[self._transport.list_operations] + + # Certain fields should be provided within the metadata header; + # add these here. + metadata = tuple(metadata or ()) + ( + gapic_v1.routing_header.to_grpc_metadata((("name", request.name),)), + ) + + # Send the request. + response = await rpc( + request, + timeout=timeout, + metadata=metadata, + ) + + # This method is paged; wrap the response in a pager, which provides + # an `__iter__` convenience method. + response = pagers.ListOperationsAsyncPager( + method=rpc, + request=request, + response=response, + metadata=metadata, + ) + + # Done; return the response. + return response + + async def get_operation( + self, + name: str, + *, + timeout: Optional[float] = None, + metadata: Sequence[Tuple[str, str]] = (), + ) -> operations_pb2.Operation: + r"""Gets the latest state of a long-running operation. + Clients can use this method to poll the operation result + at intervals as recommended by the API service. + + Args: + name (str): + The name of the operation resource. + timeout (float): The timeout for this request. + metadata (Sequence[Tuple[str, str]]): Strings which should be + sent along with the request as metadata. + + Returns: + google.longrunning.operations_pb2.Operation: + This resource represents a long- + running operation that is the result of a + network API call. + + """ + + request = operations_pb2.GetOperationRequest(name=name) + + # Wrap the RPC method; this adds retry and timeout information, + # and friendly error handling. + rpc = self._transport._wrapped_methods[self._transport.get_operation] + + # Certain fields should be provided within the metadata header; + # add these here. + metadata = tuple(metadata or ()) + ( + gapic_v1.routing_header.to_grpc_metadata((("name", request.name),)), + ) + + # Send the request. + response = await rpc( + request, + timeout=timeout, + metadata=metadata, + ) + + # Done; return the response. + return response + + async def delete_operation( + self, + name: str, + *, + timeout: Optional[float] = None, + metadata: Sequence[Tuple[str, str]] = (), + ) -> None: + r"""Deletes a long-running operation. This method indicates that the + client is no longer interested in the operation result. It does + not cancel the operation. If the server doesn't support this + method, it returns ``google.rpc.Code.UNIMPLEMENTED``. + + Args: + name (str): + The name of the operation resource to + be deleted. + + This corresponds to the ``name`` field + on the ``request`` instance; if ``request`` is provided, this + should not be set. + timeout (float): The timeout for this request. + metadata (Sequence[Tuple[str, str]]): Strings which should be + sent along with the request as metadata. + """ + # Create the request object. + request = operations_pb2.DeleteOperationRequest(name=name) + + # Wrap the RPC method; this adds retry and timeout information, + # and friendly error handling. + rpc = self._transport._wrapped_methods[self._transport.delete_operation] + + # Certain fields should be provided within the metadata header; + # add these here. + metadata = tuple(metadata or ()) + ( + gapic_v1.routing_header.to_grpc_metadata((("name", request.name),)), + ) + + # Send the request. + await rpc( + request, + timeout=timeout, + metadata=metadata, + ) + + async def cancel_operation( + self, + name: Optional[str] = None, + *, + timeout: Optional[float] = None, + metadata: Sequence[Tuple[str, str]] = (), + ) -> None: + r"""Starts asynchronous cancellation on a long-running operation. + The server makes a best effort to cancel the operation, but + success is not guaranteed. If the server doesn't support this + method, it returns ``google.rpc.Code.UNIMPLEMENTED``. Clients + can use + [Operations.GetOperation][google.api_core.operations_v1.Operations.GetOperation] + or other methods to check whether the cancellation succeeded or + whether the operation completed despite cancellation. On + successful cancellation, the operation is not deleted; instead, + it becomes an operation with an + [Operation.error][google.api_core.operations_v1.Operation.error] value with + a [google.rpc.Status.code][google.rpc.Status.code] of 1, + corresponding to ``Code.CANCELLED``. + + Args: + name (str): + The name of the operation resource to + be cancelled. + + This corresponds to the ``name`` field + on the ``request`` instance; if ``request`` is provided, this + should not be set. + timeout (float): The timeout for this request. + metadata (Sequence[Tuple[str, str]]): Strings which should be + sent along with the request as metadata. + """ + # Create the request object. + request = operations_pb2.CancelOperationRequest(name=name) + + # Wrap the RPC method; this adds retry and timeout information, + # and friendly error handling. + rpc = self._transport._wrapped_methods[self._transport.cancel_operation] + + # Certain fields should be provided within the metadata header; + # add these here. + metadata = tuple(metadata or ()) + ( + gapic_v1.routing_header.to_grpc_metadata((("name", request.name),)), + ) + + # Send the request. + await rpc( + request, + timeout=timeout, + metadata=metadata, + ) diff --git a/google/api_core/operations_v1/abstract_operations_base_client.py b/google/api_core/operations_v1/abstract_operations_base_client.py new file mode 100644 index 00000000..58135b60 --- /dev/null +++ b/google/api_core/operations_v1/abstract_operations_base_client.py @@ -0,0 +1,416 @@ +# -*- coding: utf-8 -*- +# Copyright 2024 Google LLC +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +from collections import OrderedDict +import os +import re +from typing import Awaitable, Callable, Dict, Optional, Type, Union + +from google.api_core import client_options as client_options_lib # type: ignore +from google.api_core import gapic_v1 # type: ignore +from google.api_core.operations_v1.transports.base import ( + DEFAULT_CLIENT_INFO, + OperationsTransport, +) +from google.api_core.operations_v1.transports.rest import OperationsRestTransport + +try: + from google.api_core.operations_v1.transports.rest_asyncio import ( + OperationsRestAsyncTransport, + ) + + HAS_ASYNC_REST_DEPENDENCIES = True +except ImportError as e: + HAS_ASYNC_REST_DEPENDENCIES = False + ASYNC_REST_EXCEPTION = e + +from google.auth import credentials as ga_credentials # type: ignore +from google.auth.exceptions import MutualTLSChannelError # type: ignore +from google.auth.transport import mtls # type: ignore +from google.longrunning import operations_pb2 +from google.oauth2 import service_account # type: ignore +from google.protobuf import empty_pb2 # type: ignore + + +class AbstractOperationsBaseClientMeta(type): + """Metaclass for the Operations Base client. + + This provides base class-level methods for building and retrieving + support objects (e.g. transport) without polluting the client instance + objects. + """ + + _transport_registry = OrderedDict() # type: Dict[str, Type[OperationsTransport]] + _transport_registry["rest"] = OperationsRestTransport + if HAS_ASYNC_REST_DEPENDENCIES: + _transport_registry["rest_asyncio"] = OperationsRestAsyncTransport + + def get_transport_class( + cls, + label: Optional[str] = None, + ) -> Type[OperationsTransport]: + """Returns an appropriate transport class. + + Args: + label: The name of the desired transport. If none is + provided, then the first transport in the registry is used. + + Returns: + The transport class to use. + """ + # If a specific transport is requested, return that one. + if ( + label == "rest_asyncio" and not HAS_ASYNC_REST_DEPENDENCIES + ): # pragma: NO COVER + raise ASYNC_REST_EXCEPTION + + if label: + return cls._transport_registry[label] + + # No transport is requested; return the default (that is, the first one + # in the dictionary). + return next(iter(cls._transport_registry.values())) + + +class AbstractOperationsBaseClient(metaclass=AbstractOperationsBaseClientMeta): + """Manages long-running operations with an API service. + + When an API method normally takes long time to complete, it can be + designed to return [Operation][google.api_core.operations_v1.Operation] to the + client, and the client can use this interface to receive the real + response asynchronously by polling the operation resource, or pass + the operation resource to another API (such as Google Cloud Pub/Sub + API) to receive the response. Any API service that returns + long-running operations should implement the ``Operations`` + interface so developers can have a consistent client experience. + """ + + @staticmethod + def _get_default_mtls_endpoint(api_endpoint): + """Converts api endpoint to mTLS endpoint. + + Convert "*.sandbox.googleapis.com" and "*.googleapis.com" to + "*.mtls.sandbox.googleapis.com" and "*.mtls.googleapis.com" respectively. + Args: + api_endpoint (Optional[str]): the api endpoint to convert. + Returns: + str: converted mTLS api endpoint. + """ + if not api_endpoint: + return api_endpoint + + mtls_endpoint_re = re.compile( + r"(?P[^.]+)(?P\.mtls)?(?P\.sandbox)?(?P\.googleapis\.com)?" + ) + + m = mtls_endpoint_re.match(api_endpoint) + name, mtls, sandbox, googledomain = m.groups() + if mtls or not googledomain: + return api_endpoint + + if sandbox: + return api_endpoint.replace( + "sandbox.googleapis.com", "mtls.sandbox.googleapis.com" + ) + + return api_endpoint.replace(".googleapis.com", ".mtls.googleapis.com") + + DEFAULT_ENDPOINT = "longrunning.googleapis.com" + DEFAULT_MTLS_ENDPOINT = _get_default_mtls_endpoint.__func__( # type: ignore + DEFAULT_ENDPOINT + ) + + @classmethod + def from_service_account_info(cls, info: dict, *args, **kwargs): + """Creates an instance of this client using the provided credentials + info. + + Args: + info (dict): The service account private key info. + args: Additional arguments to pass to the constructor. + kwargs: Additional arguments to pass to the constructor. + + Returns: + AbstractOperationsClient: The constructed client. + """ + credentials = service_account.Credentials.from_service_account_info(info) + kwargs["credentials"] = credentials + return cls(*args, **kwargs) + + @classmethod + def from_service_account_file(cls, filename: str, *args, **kwargs): + """Creates an instance of this client using the provided credentials + file. + + Args: + filename (str): The path to the service account private key json + file. + args: Additional arguments to pass to the constructor. + kwargs: Additional arguments to pass to the constructor. + + Returns: + AbstractOperationsClient: The constructed client. + """ + credentials = service_account.Credentials.from_service_account_file(filename) + kwargs["credentials"] = credentials + return cls(*args, **kwargs) + + from_service_account_json = from_service_account_file + + @property + def transport(self) -> OperationsTransport: + """Returns the transport used by the client instance. + + Returns: + OperationsTransport: The transport used by the client + instance. + """ + return self._transport + + @staticmethod + def common_billing_account_path( + billing_account: str, + ) -> str: + """Returns a fully-qualified billing_account string.""" + return "billingAccounts/{billing_account}".format( + billing_account=billing_account, + ) + + @staticmethod + def parse_common_billing_account_path(path: str) -> Dict[str, str]: + """Parse a billing_account path into its component segments.""" + m = re.match(r"^billingAccounts/(?P.+?)$", path) + return m.groupdict() if m else {} + + @staticmethod + def common_folder_path( + folder: str, + ) -> str: + """Returns a fully-qualified folder string.""" + return "folders/{folder}".format( + folder=folder, + ) + + @staticmethod + def parse_common_folder_path(path: str) -> Dict[str, str]: + """Parse a folder path into its component segments.""" + m = re.match(r"^folders/(?P.+?)$", path) + return m.groupdict() if m else {} + + @staticmethod + def common_organization_path( + organization: str, + ) -> str: + """Returns a fully-qualified organization string.""" + return "organizations/{organization}".format( + organization=organization, + ) + + @staticmethod + def parse_common_organization_path(path: str) -> Dict[str, str]: + """Parse a organization path into its component segments.""" + m = re.match(r"^organizations/(?P.+?)$", path) + return m.groupdict() if m else {} + + @staticmethod + def common_project_path( + project: str, + ) -> str: + """Returns a fully-qualified project string.""" + return "projects/{project}".format( + project=project, + ) + + @staticmethod + def parse_common_project_path(path: str) -> Dict[str, str]: + """Parse a project path into its component segments.""" + m = re.match(r"^projects/(?P.+?)$", path) + return m.groupdict() if m else {} + + @staticmethod + def common_location_path( + project: str, + location: str, + ) -> str: + """Returns a fully-qualified location string.""" + return "projects/{project}/locations/{location}".format( + project=project, + location=location, + ) + + @staticmethod + def parse_common_location_path(path: str) -> Dict[str, str]: + """Parse a location path into its component segments.""" + m = re.match(r"^projects/(?P.+?)/locations/(?P.+?)$", path) + return m.groupdict() if m else {} + + def __init__( + self, + *, + credentials: Optional[ga_credentials.Credentials] = None, + transport: Union[str, OperationsTransport, None] = None, + client_options: Optional[client_options_lib.ClientOptions] = None, + client_info: gapic_v1.client_info.ClientInfo = DEFAULT_CLIENT_INFO, + ) -> None: + """Instantiates the operations client. + + Args: + credentials (Optional[google.auth.credentials.Credentials]): The + authorization credentials to attach to requests. These + credentials identify the application to the service; if none + are specified, the client will attempt to ascertain the + credentials from the environment. + transport (Union[str, OperationsTransport]): The + transport to use. If set to None, a transport is chosen + automatically. + client_options (google.api_core.client_options.ClientOptions): Custom options for the + client. It won't take effect if a ``transport`` instance is provided. + (1) The ``api_endpoint`` property can be used to override the + default endpoint provided by the client. GOOGLE_API_USE_MTLS_ENDPOINT + environment variable can also be used to override the endpoint: + "always" (always use the default mTLS endpoint), "never" (always + use the default regular endpoint) and "auto" (auto switch to the + default mTLS endpoint if client certificate is present, this is + the default value). However, the ``api_endpoint`` property takes + precedence if provided. + (2) If GOOGLE_API_USE_CLIENT_CERTIFICATE environment variable + is "true", then the ``client_cert_source`` property can be used + to provide client certificate for mutual TLS transport. If + not provided, the default SSL client certificate will be used if + present. If GOOGLE_API_USE_CLIENT_CERTIFICATE is "false" or not + set, no client certificate will be used. + client_info (google.api_core.gapic_v1.client_info.ClientInfo): + The client info used to send a user-agent string along with + API requests. If ``None``, then default info will be used. + Generally, you only need to set this if you're developing + your own client library. + + Raises: + google.auth.exceptions.MutualTLSChannelError: If mutual TLS transport + creation failed for any reason. + """ + if isinstance(client_options, dict): + client_options = client_options_lib.from_dict(client_options) + if client_options is None: + client_options = client_options_lib.ClientOptions() + + # Create SSL credentials for mutual TLS if needed. + use_client_cert = os.getenv( + "GOOGLE_API_USE_CLIENT_CERTIFICATE", "false" + ).lower() + if use_client_cert not in ("true", "false"): + raise ValueError( + "Environment variable `GOOGLE_API_USE_CLIENT_CERTIFICATE` must be either `true` or `false`" + ) + client_cert_source_func = None + is_mtls = False + if use_client_cert == "true": + if client_options.client_cert_source: + is_mtls = True + client_cert_source_func = client_options.client_cert_source + else: + is_mtls = mtls.has_default_client_cert_source() + if is_mtls: + client_cert_source_func = mtls.default_client_cert_source() + else: + client_cert_source_func = None + + # Figure out which api endpoint to use. + if client_options.api_endpoint is not None: + api_endpoint = client_options.api_endpoint + else: + use_mtls_env = os.getenv("GOOGLE_API_USE_MTLS_ENDPOINT", "auto") + if use_mtls_env == "never": + api_endpoint = self.DEFAULT_ENDPOINT + elif use_mtls_env == "always": + api_endpoint = self.DEFAULT_MTLS_ENDPOINT + elif use_mtls_env == "auto": + if is_mtls: + api_endpoint = self.DEFAULT_MTLS_ENDPOINT + else: + api_endpoint = self.DEFAULT_ENDPOINT + else: + raise MutualTLSChannelError( + "Unsupported GOOGLE_API_USE_MTLS_ENDPOINT value. Accepted " + "values: never, auto, always" + ) + + # Save or instantiate the transport. + # Ordinarily, we provide the transport, but allowing a custom transport + # instance provides an extensibility point for unusual situations. + if isinstance(transport, OperationsTransport): + # transport is a OperationsTransport instance. + if credentials or client_options.credentials_file: + raise ValueError( + "When providing a transport instance, " + "provide its credentials directly." + ) + if client_options.scopes: + raise ValueError( + "When providing a transport instance, provide its scopes " + "directly." + ) + self._transport = transport + else: + Transport = type(self).get_transport_class(transport) + self._transport = Transport( + credentials=credentials, + credentials_file=client_options.credentials_file, + host=api_endpoint, + scopes=client_options.scopes, + client_cert_source_for_mtls=client_cert_source_func, + quota_project_id=client_options.quota_project_id, + client_info=client_info, + always_use_jwt_access=True, + ) + + @property + def list_operations( + self, + ) -> Callable[ + [operations_pb2.ListOperationsRequest], + Union[ + operations_pb2.ListOperationsResponse, + Awaitable[operations_pb2.ListOperationsResponse], + ], + ]: + raise NotImplementedError() + + @property + def get_operation( + self, + ) -> Callable[ + [operations_pb2.GetOperationRequest], + Union[operations_pb2.Operation, Awaitable[operations_pb2.Operation]], + ]: + raise NotImplementedError() + + @property + def delete_operation( + self, + ) -> Callable[ + [operations_pb2.DeleteOperationRequest], + Union[empty_pb2.Empty, Awaitable[empty_pb2.Empty]], + ]: + raise NotImplementedError() + + @property + def cancel_operation( + self, + ) -> Callable[ + [operations_pb2.CancelOperationRequest], + Union[empty_pb2.Empty, Awaitable[empty_pb2.Empty]], + ]: + raise NotImplementedError() diff --git a/google/api_core/operations_v1/abstract_operations_client.py b/google/api_core/operations_v1/abstract_operations_client.py index 38f532af..64f2cb7d 100644 --- a/google/api_core/operations_v1/abstract_operations_client.py +++ b/google/api_core/operations_v1/abstract_operations_client.py @@ -13,10 +13,7 @@ # See the License for the specific language governing permissions and # limitations under the License. # -from collections import OrderedDict -import os -import re -from typing import Dict, Optional, Sequence, Tuple, Type, Union +from typing import Optional, Sequence, Tuple, Union from google.api_core import client_options as client_options_lib # type: ignore from google.api_core import gapic_v1 # type: ignore @@ -26,51 +23,17 @@ DEFAULT_CLIENT_INFO, OperationsTransport, ) -from google.api_core.operations_v1.transports.rest import OperationsRestTransport +from google.api_core.operations_v1.abstract_operations_base_client import ( + AbstractOperationsBaseClient, +) from google.auth import credentials as ga_credentials # type: ignore -from google.auth.exceptions import MutualTLSChannelError # type: ignore -from google.auth.transport import mtls # type: ignore from google.longrunning import operations_pb2 -from google.oauth2 import service_account # type: ignore import grpc OptionalRetry = Union[retries.Retry, object] -class AbstractOperationsClientMeta(type): - """Metaclass for the Operations client. - - This provides class-level methods for building and retrieving - support objects (e.g. transport) without polluting the client instance - objects. - """ - - _transport_registry = OrderedDict() # type: Dict[str, Type[OperationsTransport]] - _transport_registry["rest"] = OperationsRestTransport - - def get_transport_class( - cls, - label: Optional[str] = None, - ) -> Type[OperationsTransport]: - """Returns an appropriate transport class. - - Args: - label: The name of the desired transport. If none is - provided, then the first transport in the registry is used. - - Returns: - The transport class to use. - """ - # If a specific transport is requested, return that one. - if label: - return cls._transport_registry[label] - - # No transport is requested; return the default (that is, the first one - # in the dictionary). - return next(iter(cls._transport_registry.values())) - - -class AbstractOperationsClient(metaclass=AbstractOperationsClientMeta): +class AbstractOperationsClient(AbstractOperationsBaseClient): """Manages long-running operations with an API service. When an API method normally takes long time to complete, it can be @@ -83,165 +46,6 @@ class AbstractOperationsClient(metaclass=AbstractOperationsClientMeta): interface so developers can have a consistent client experience. """ - @staticmethod - def _get_default_mtls_endpoint(api_endpoint): - """Converts api endpoint to mTLS endpoint. - - Convert "*.sandbox.googleapis.com" and "*.googleapis.com" to - "*.mtls.sandbox.googleapis.com" and "*.mtls.googleapis.com" respectively. - Args: - api_endpoint (Optional[str]): the api endpoint to convert. - Returns: - str: converted mTLS api endpoint. - """ - if not api_endpoint: - return api_endpoint - - mtls_endpoint_re = re.compile( - r"(?P[^.]+)(?P\.mtls)?(?P\.sandbox)?(?P\.googleapis\.com)?" - ) - - m = mtls_endpoint_re.match(api_endpoint) - name, mtls, sandbox, googledomain = m.groups() - if mtls or not googledomain: - return api_endpoint - - if sandbox: - return api_endpoint.replace( - "sandbox.googleapis.com", "mtls.sandbox.googleapis.com" - ) - - return api_endpoint.replace(".googleapis.com", ".mtls.googleapis.com") - - DEFAULT_ENDPOINT = "longrunning.googleapis.com" - DEFAULT_MTLS_ENDPOINT = _get_default_mtls_endpoint.__func__( # type: ignore - DEFAULT_ENDPOINT - ) - - @classmethod - def from_service_account_info(cls, info: dict, *args, **kwargs): - """Creates an instance of this client using the provided credentials - info. - - Args: - info (dict): The service account private key info. - args: Additional arguments to pass to the constructor. - kwargs: Additional arguments to pass to the constructor. - - Returns: - AbstractOperationsClient: The constructed client. - """ - credentials = service_account.Credentials.from_service_account_info(info) - kwargs["credentials"] = credentials - return cls(*args, **kwargs) - - @classmethod - def from_service_account_file(cls, filename: str, *args, **kwargs): - """Creates an instance of this client using the provided credentials - file. - - Args: - filename (str): The path to the service account private key json - file. - args: Additional arguments to pass to the constructor. - kwargs: Additional arguments to pass to the constructor. - - Returns: - AbstractOperationsClient: The constructed client. - """ - credentials = service_account.Credentials.from_service_account_file(filename) - kwargs["credentials"] = credentials - return cls(*args, **kwargs) - - from_service_account_json = from_service_account_file - - @property - def transport(self) -> OperationsTransport: - """Returns the transport used by the client instance. - - Returns: - OperationsTransport: The transport used by the client - instance. - """ - return self._transport - - @staticmethod - def common_billing_account_path( - billing_account: str, - ) -> str: - """Returns a fully-qualified billing_account string.""" - return "billingAccounts/{billing_account}".format( - billing_account=billing_account, - ) - - @staticmethod - def parse_common_billing_account_path(path: str) -> Dict[str, str]: - """Parse a billing_account path into its component segments.""" - m = re.match(r"^billingAccounts/(?P.+?)$", path) - return m.groupdict() if m else {} - - @staticmethod - def common_folder_path( - folder: str, - ) -> str: - """Returns a fully-qualified folder string.""" - return "folders/{folder}".format( - folder=folder, - ) - - @staticmethod - def parse_common_folder_path(path: str) -> Dict[str, str]: - """Parse a folder path into its component segments.""" - m = re.match(r"^folders/(?P.+?)$", path) - return m.groupdict() if m else {} - - @staticmethod - def common_organization_path( - organization: str, - ) -> str: - """Returns a fully-qualified organization string.""" - return "organizations/{organization}".format( - organization=organization, - ) - - @staticmethod - def parse_common_organization_path(path: str) -> Dict[str, str]: - """Parse a organization path into its component segments.""" - m = re.match(r"^organizations/(?P.+?)$", path) - return m.groupdict() if m else {} - - @staticmethod - def common_project_path( - project: str, - ) -> str: - """Returns a fully-qualified project string.""" - return "projects/{project}".format( - project=project, - ) - - @staticmethod - def parse_common_project_path(path: str) -> Dict[str, str]: - """Parse a project path into its component segments.""" - m = re.match(r"^projects/(?P.+?)$", path) - return m.groupdict() if m else {} - - @staticmethod - def common_location_path( - project: str, - location: str, - ) -> str: - """Returns a fully-qualified location string.""" - return "projects/{project}/locations/{location}".format( - project=project, - location=location, - ) - - @staticmethod - def parse_common_location_path(path: str) -> Dict[str, str]: - """Parse a location path into its component segments.""" - m = re.match(r"^projects/(?P.+?)/locations/(?P.+?)$", path) - return m.groupdict() if m else {} - def __init__( self, *, @@ -287,80 +91,12 @@ def __init__( google.auth.exceptions.MutualTLSChannelError: If mutual TLS transport creation failed for any reason. """ - if isinstance(client_options, dict): - client_options = client_options_lib.from_dict(client_options) - if client_options is None: - client_options = client_options_lib.ClientOptions() - - # Create SSL credentials for mutual TLS if needed. - use_client_cert = os.getenv( - "GOOGLE_API_USE_CLIENT_CERTIFICATE", "false" - ).lower() - if use_client_cert not in ("true", "false"): - raise ValueError( - "Environment variable `GOOGLE_API_USE_CLIENT_CERTIFICATE` must be either `true` or `false`" - ) - client_cert_source_func = None - is_mtls = False - if use_client_cert == "true": - if client_options.client_cert_source: - is_mtls = True - client_cert_source_func = client_options.client_cert_source - else: - is_mtls = mtls.has_default_client_cert_source() - if is_mtls: - client_cert_source_func = mtls.default_client_cert_source() - else: - client_cert_source_func = None - - # Figure out which api endpoint to use. - if client_options.api_endpoint is not None: - api_endpoint = client_options.api_endpoint - else: - use_mtls_env = os.getenv("GOOGLE_API_USE_MTLS_ENDPOINT", "auto") - if use_mtls_env == "never": - api_endpoint = self.DEFAULT_ENDPOINT - elif use_mtls_env == "always": - api_endpoint = self.DEFAULT_MTLS_ENDPOINT - elif use_mtls_env == "auto": - if is_mtls: - api_endpoint = self.DEFAULT_MTLS_ENDPOINT - else: - api_endpoint = self.DEFAULT_ENDPOINT - else: - raise MutualTLSChannelError( - "Unsupported GOOGLE_API_USE_MTLS_ENDPOINT value. Accepted " - "values: never, auto, always" - ) - - # Save or instantiate the transport. - # Ordinarily, we provide the transport, but allowing a custom transport - # instance provides an extensibility point for unusual situations. - if isinstance(transport, OperationsTransport): - # transport is a OperationsTransport instance. - if credentials or client_options.credentials_file: - raise ValueError( - "When providing a transport instance, " - "provide its credentials directly." - ) - if client_options.scopes: - raise ValueError( - "When providing a transport instance, provide its scopes " - "directly." - ) - self._transport = transport - else: - Transport = type(self).get_transport_class(transport) - self._transport = Transport( - credentials=credentials, - credentials_file=client_options.credentials_file, - host=api_endpoint, - scopes=client_options.scopes, - client_cert_source_for_mtls=client_cert_source_func, - quota_project_id=client_options.quota_project_id, - client_info=client_info, - always_use_jwt_access=True, - ) + super().__init__( + credentials=credentials, + transport=transport, + client_options=client_options, + client_info=client_info, + ) def list_operations( self, diff --git a/google/api_core/operations_v1/pagers.py b/google/api_core/operations_v1/pagers.py index b8a47757..132f1c66 100644 --- a/google/api_core/operations_v1/pagers.py +++ b/google/api_core/operations_v1/pagers.py @@ -14,7 +14,6 @@ # limitations under the License. # from typing import ( - Any, Callable, Iterator, Sequence, @@ -22,9 +21,10 @@ ) from google.longrunning import operations_pb2 +from google.api_core.operations_v1.pagers_base import ListOperationsPagerBase -class ListOperationsPager: +class ListOperationsPager(ListOperationsPagerBase): """A pager for iterating through ``list_operations`` requests. This class thinly wraps an initial @@ -50,25 +50,9 @@ def __init__( *, metadata: Sequence[Tuple[str, str]] = () ): - """Instantiate the pager. - - Args: - method (Callable): The method that was originally called, and - which instantiated this pager. - request (google.longrunning.operations_pb2.ListOperationsRequest): - The initial request object. - response (google.longrunning.operations_pb2.ListOperationsResponse): - 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 = request - self._response = response - self._metadata = metadata - - def __getattr__(self, name: str) -> Any: - return getattr(self._response, name) + super().__init__( + method=method, request=request, response=response, metadata=metadata + ) @property def pages(self) -> Iterator[operations_pb2.ListOperationsResponse]: @@ -81,6 +65,3 @@ def pages(self) -> Iterator[operations_pb2.ListOperationsResponse]: def __iter__(self) -> Iterator[operations_pb2.Operation]: for page in self.pages: yield from page.operations - - def __repr__(self) -> str: - return "{0}<{1!r}>".format(self.__class__.__name__, self._response) diff --git a/google/api_core/operations_v1/pagers_async.py b/google/api_core/operations_v1/pagers_async.py new file mode 100644 index 00000000..e2909dd5 --- /dev/null +++ b/google/api_core/operations_v1/pagers_async.py @@ -0,0 +1,71 @@ +# -*- coding: utf-8 -*- +# Copyright 2024 Google LLC +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +from typing import ( + Callable, + AsyncIterator, + Sequence, + Tuple, +) + +from google.longrunning import operations_pb2 +from google.api_core.operations_v1.pagers_base import ListOperationsPagerBase + + +class ListOperationsAsyncPager(ListOperationsPagerBase): + """A pager for iterating through ``list_operations`` requests. + + This class thinly wraps an initial + :class:`google.longrunning.operations_pb2.ListOperationsResponse` object, and + provides an ``__iter__`` method to iterate through its + ``operations`` field. + + If there are more pages, the ``__iter__`` method will make additional + ``ListOperations`` requests and continue to iterate + through the ``operations`` field on the + corresponding responses. + + All the usual :class:`google.longrunning.operations_pb2.ListOperationsResponse` + attributes are available on the pager. If multiple requests are made, only + the most recent response is retained, and thus used for attribute lookup. + """ + + def __init__( + self, + method: Callable[..., operations_pb2.ListOperationsResponse], + request: operations_pb2.ListOperationsRequest, + response: operations_pb2.ListOperationsResponse, + *, + metadata: Sequence[Tuple[str, str]] = () + ): + super().__init__( + method=method, request=request, response=response, metadata=metadata + ) + + @property + async def pages(self) -> AsyncIterator[operations_pb2.ListOperationsResponse]: + 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, metadata=self._metadata) + yield self._response + + def __aiter__(self) -> AsyncIterator[operations_pb2.Operation]: + async def async_generator(): + async for page in self.pages: + for operation in page.operations: + yield operation + + return async_generator() diff --git a/google/api_core/operations_v1/pagers_base.py b/google/api_core/operations_v1/pagers_base.py new file mode 100644 index 00000000..24caf74f --- /dev/null +++ b/google/api_core/operations_v1/pagers_base.py @@ -0,0 +1,73 @@ +# -*- coding: utf-8 -*- +# Copyright 2024 Google LLC +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# +from typing import ( + Any, + Callable, + Sequence, + Tuple, +) + +from google.longrunning import operations_pb2 + + +class ListOperationsPagerBase: + """A pager for iterating through ``list_operations`` requests. + + This class thinly wraps an initial + :class:`google.longrunning.operations_pb2.ListOperationsResponse` object, and + provides an ``__iter__`` method to iterate through its + ``operations`` field. + + If there are more pages, the ``__iter__`` method will make additional + ``ListOperations`` requests and continue to iterate + through the ``operations`` field on the + corresponding responses. + + All the usual :class:`google.longrunning.operations_pb2.ListOperationsResponse` + attributes are available on the pager. If multiple requests are made, only + the most recent response is retained, and thus used for attribute lookup. + """ + + def __init__( + self, + method: Callable[..., operations_pb2.ListOperationsResponse], + request: operations_pb2.ListOperationsRequest, + response: operations_pb2.ListOperationsResponse, + *, + metadata: Sequence[Tuple[str, str]] = () + ): + """Instantiate the pager. + + Args: + method (Callable): The method that was originally called, and + which instantiated this pager. + request (google.longrunning.operations_pb2.ListOperationsRequest): + The initial request object. + response (google.longrunning.operations_pb2.ListOperationsResponse): + 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 = request + self._response = response + self._metadata = metadata + + def __getattr__(self, name: str) -> Any: + return getattr(self._response, name) + + def __repr__(self) -> str: + return "{0}<{1!r}>".format(self.__class__.__name__, self._response) diff --git a/google/api_core/operations_v1/transports/__init__.py b/google/api_core/operations_v1/transports/__init__.py index 2ae79e4e..1771e711 100644 --- a/google/api_core/operations_v1/transports/__init__.py +++ b/google/api_core/operations_v1/transports/__init__.py @@ -37,3 +37,7 @@ # Don't raise an exception if `OperationsRestAsyncTransport` cannot be imported # as other transports are still available. pass + +# Compile a registry of transports. +_transport_registry = OrderedDict() +_transport_registry["rest"] = OperationsRestTransport diff --git a/tests/unit/operations_v1/test_operations_rest_client.py b/tests/unit/operations_v1/test_operations_rest_client.py index 556c35c3..a3caa2f3 100644 --- a/tests/unit/operations_v1/test_operations_rest_client.py +++ b/tests/unit/operations_v1/test_operations_rest_client.py @@ -14,6 +14,7 @@ # limitations under the License. # import os +from typing import Union import mock import pytest @@ -23,13 +24,15 @@ except ImportError: # pragma: NO COVER pytest.skip("No GRPC", allow_module_level=True) from requests import Response # noqa I201 -from requests.sessions import Session +from google.auth.transport.requests import AuthorizedSession from google.api_core import client_options from google.api_core import exceptions as core_exceptions from google.api_core import gapic_v1 from google.api_core.operations_v1 import AbstractOperationsClient + from google.api_core.operations_v1 import pagers +from google.api_core.operations_v1 import pagers_async from google.api_core.operations_v1 import transports from google.auth import credentials as ga_credentials from google.auth.exceptions import MutualTLSChannelError @@ -41,6 +44,8 @@ try: import aiohttp # noqa: F401 import google.auth.aio.transport + from google.auth.aio.transport.sessions import AsyncAuthorizedSession + from google.api_core.operations_v1 import AbstractOperationsAsyncClient from google.auth.aio import credentials as ga_credentials_async GOOGLE_AUTH_AIO_INSTALLED = True @@ -81,17 +86,63 @@ ], } +CLIENTS = [ + AbstractOperationsClient, +] + +CLIENTS_WITH_TRANSPORT = [ + (AbstractOperationsClient, transports.OperationsRestTransport, "rest"), +] +CLIENTS_WITH_CREDENTIALS = [ + ( + AbstractOperationsClient, + transports.OperationsRestTransport, + ga_credentials.AnonymousCredentials(), + ), +] + +if GOOGLE_AUTH_AIO_INSTALLED: + CLIENTS.append(AbstractOperationsAsyncClient) + CLIENTS_WITH_TRANSPORT.append( + ( + AbstractOperationsAsyncClient, + transports.OperationsRestAsyncTransport, + "rest_asyncio", + ) + ), + CLIENTS_WITH_CREDENTIALS.append( + ( + AbstractOperationsAsyncClient, + transports.OperationsRestAsyncTransport, + ga_credentials_async.AnonymousCredentials(), + ) + ), + def client_cert_source_callback(): return b"cert bytes", b"key bytes" -def _get_operations_client(http_options=HTTP_OPTIONS): - transport = transports.rest.OperationsRestTransport( - credentials=ga_credentials.AnonymousCredentials(), http_options=http_options +def _get_session_type(is_async: bool): + return ( + AsyncAuthorizedSession + if is_async and GOOGLE_AUTH_AIO_INSTALLED + else AuthorizedSession ) - return AbstractOperationsClient(transport=transport) + +def _get_operations_client(is_async: bool, http_options=HTTP_OPTIONS): + if is_async and GOOGLE_AUTH_AIO_INSTALLED: + transport = transports.rest_asyncio.OperationsRestAsyncTransport( + credentials=ga_credentials_async.AnonymousCredentials(), + http_options=http_options, + ) + return AbstractOperationsAsyncClient(transport=transport) + else: + transport = transports.rest.OperationsRestTransport( + credentials=ga_credentials.AnonymousCredentials(), http_options=http_options + ) + return AbstractOperationsClient(transport=transport) # If default endpoint is localhost, then default mtls endpoint will be the same. @@ -105,37 +156,40 @@ def modify_default_endpoint(client): ) -def test__get_default_mtls_endpoint(): +# TODO: Add support for mtls in async rest +@pytest.mark.parametrize( + "client_class", + [ + AbstractOperationsClient, + ], +) +def test__get_default_mtls_endpoint(client_class): api_endpoint = "example.googleapis.com" api_mtls_endpoint = "example.mtls.googleapis.com" sandbox_endpoint = "example.sandbox.googleapis.com" sandbox_mtls_endpoint = "example.mtls.sandbox.googleapis.com" non_googleapi = "api.example.com" - assert AbstractOperationsClient._get_default_mtls_endpoint(None) is None + assert client_class._get_default_mtls_endpoint(None) is None + assert client_class._get_default_mtls_endpoint(api_endpoint) == api_mtls_endpoint assert ( - AbstractOperationsClient._get_default_mtls_endpoint(api_endpoint) - == api_mtls_endpoint + client_class._get_default_mtls_endpoint(api_mtls_endpoint) == api_mtls_endpoint ) assert ( - AbstractOperationsClient._get_default_mtls_endpoint(api_mtls_endpoint) - == api_mtls_endpoint - ) - assert ( - AbstractOperationsClient._get_default_mtls_endpoint(sandbox_endpoint) + client_class._get_default_mtls_endpoint(sandbox_endpoint) == sandbox_mtls_endpoint ) assert ( - AbstractOperationsClient._get_default_mtls_endpoint(sandbox_mtls_endpoint) + client_class._get_default_mtls_endpoint(sandbox_mtls_endpoint) == sandbox_mtls_endpoint ) - assert ( - AbstractOperationsClient._get_default_mtls_endpoint(non_googleapi) - == non_googleapi - ) + assert client_class._get_default_mtls_endpoint(non_googleapi) == non_googleapi -@pytest.mark.parametrize("client_class", [AbstractOperationsClient]) +@pytest.mark.parametrize( + "client_class", + CLIENTS, +) def test_operations_client_from_service_account_info(client_class): creds = ga_credentials.AnonymousCredentials() with mock.patch.object( @@ -174,7 +228,10 @@ def test_operations_client_service_account_always_use_jwt(transport_class): use_jwt.assert_not_called() -@pytest.mark.parametrize("client_class", [AbstractOperationsClient]) +@pytest.mark.parametrize( + "client_class", + CLIENTS, +) def test_operations_client_from_service_account_file(client_class): creds = ga_credentials.AnonymousCredentials() with mock.patch.object( @@ -192,20 +249,28 @@ def test_operations_client_from_service_account_file(client_class): assert client.transport._host == "https://longrunning.googleapis.com" -def test_operations_client_get_transport_class(): - transport = AbstractOperationsClient.get_transport_class() +@pytest.mark.parametrize( + "client_class,transport_class,transport_name", + CLIENTS_WITH_TRANSPORT, +) +def test_operations_client_get_transport_class( + client_class, transport_class, transport_name +): + transport = client_class.get_transport_class() available_transports = [ transports.OperationsRestTransport, ] + if GOOGLE_AUTH_AIO_INSTALLED: + available_transports.append(transports.OperationsRestAsyncTransport) assert transport in available_transports - transport = AbstractOperationsClient.get_transport_class("rest") - assert transport == transports.OperationsRestTransport + transport = client_class.get_transport_class(transport_name) + assert transport == transport_class @pytest.mark.parametrize( "client_class,transport_class,transport_name", - [(AbstractOperationsClient, transports.OperationsRestTransport, "rest")], + CLIENTS_WITH_TRANSPORT, ) @mock.patch.object( AbstractOperationsClient, @@ -215,22 +280,21 @@ def test_operations_client_get_transport_class(): def test_operations_client_client_options( client_class, transport_class, transport_name ): - # Check that if channel is provided we won't create a new one. - with mock.patch.object(AbstractOperationsClient, "get_transport_class") as gtc: - transport = transport_class(credentials=ga_credentials.AnonymousCredentials()) - client = client_class(transport=transport) - gtc.assert_not_called() + # # Check that if channel is provided we won't create a new one. + # with mock.patch.object(AbstractOperationsBaseClient, "get_transport_class") as gtc: + # client = client_class(transport=transport_class()) + # gtc.assert_not_called() - # Check that if channel is provided via str we will create a new one. - with mock.patch.object(AbstractOperationsClient, "get_transport_class") as gtc: - client = client_class(transport=transport_name) - gtc.assert_called() + # # Check that if channel is provided via str we will create a new one. + # with mock.patch.object(AbstractOperationsBaseClient, "get_transport_class") as gtc: + # client = client_class(transport=transport_name) + # gtc.assert_called() # Check the case api_endpoint is provided. options = client_options.ClientOptions(api_endpoint="squid.clam.whelk") with mock.patch.object(transport_class, "__init__") as patched: patched.return_value = None - client = client_class(client_options=options) + client = client_class(client_options=options, transport=transport_name) patched.assert_called_once_with( credentials=None, credentials_file=None, @@ -247,7 +311,7 @@ def test_operations_client_client_options( with mock.patch.dict(os.environ, {"GOOGLE_API_USE_MTLS_ENDPOINT": "never"}): with mock.patch.object(transport_class, "__init__") as patched: patched.return_value = None - client = client_class() + client = client_class(transport=transport_name) patched.assert_called_once_with( credentials=None, credentials_file=None, @@ -264,7 +328,7 @@ def test_operations_client_client_options( with mock.patch.dict(os.environ, {"GOOGLE_API_USE_MTLS_ENDPOINT": "always"}): with mock.patch.object(transport_class, "__init__") as patched: patched.return_value = None - client = client_class() + client = client_class(transport=transport_name) patched.assert_called_once_with( credentials=None, credentials_file=None, @@ -293,7 +357,7 @@ def test_operations_client_client_options( options = client_options.ClientOptions(quota_project_id="octopus") with mock.patch.object(transport_class, "__init__") as patched: patched.return_value = None - client = client_class(client_options=options) + client = client_class(client_options=options, transport=transport_name) patched.assert_called_once_with( credentials=None, credentials_file=None, @@ -306,6 +370,7 @@ def test_operations_client_client_options( ) +# TODO: Add support for mtls in async REST @pytest.mark.parametrize( "client_class,transport_class,transport_name,use_client_cert_env", [ @@ -422,7 +487,7 @@ def fake_init(client_cert_source_for_mtls=None, **kwargs): @pytest.mark.parametrize( "client_class,transport_class,transport_name", - [(AbstractOperationsClient, transports.OperationsRestTransport, "rest")], + CLIENTS_WITH_TRANSPORT, ) def test_operations_client_client_options_scopes( client_class, transport_class, transport_name @@ -433,7 +498,7 @@ def test_operations_client_client_options_scopes( ) with mock.patch.object(transport_class, "__init__") as patched: patched.return_value = None - client = client_class(client_options=options) + client = client_class(client_options=options, transport=transport_name) patched.assert_called_once_with( credentials=None, credentials_file=None, @@ -448,7 +513,7 @@ def test_operations_client_client_options_scopes( @pytest.mark.parametrize( "client_class,transport_class,transport_name", - [(AbstractOperationsClient, transports.OperationsRestTransport, "rest")], + CLIENTS_WITH_TRANSPORT, ) def test_operations_client_client_options_credentials_file( client_class, transport_class, transport_name @@ -457,7 +522,7 @@ def test_operations_client_client_options_credentials_file( options = client_options.ClientOptions(credentials_file="credentials.json") with mock.patch.object(transport_class, "__init__") as patched: patched.return_value = None - client = client_class(client_options=options) + client = client_class(client_options=options, transport=transport_name) patched.assert_called_once_with( credentials=None, credentials_file="credentials.json", @@ -470,13 +535,10 @@ def test_operations_client_client_options_credentials_file( ) -def test_list_operations_rest( - transport: str = "rest", request_type=operations_pb2.ListOperationsRequest -): - client = _get_operations_client() - +def test_list_operations_rest(): + client = _get_operations_client(is_async=False) # Mock the http request call within the method and fake a response. - with mock.patch.object(Session, "request") as req: + with mock.patch.object(_get_session_type(is_async=False), "request") as req: # Designate an appropriate value for the returned response. return_value = operations_pb2.ListOperationsResponse( next_page_token="next_page_token_value", @@ -506,10 +568,47 @@ def test_list_operations_rest( assert response.next_page_token == "next_page_token_value" +@pytest.mark.asyncio +async def test_list_operations_rest_async(): + if not GOOGLE_AUTH_AIO_INSTALLED: + pytest.skip("Skipped because google-api-core[async_rest] is not installed") + + client = _get_operations_client(is_async=True) + # Mock the http request call within the method and fake a response. + with mock.patch.object(_get_session_type(is_async=True), "request") as req: + # Designate an appropriate value for the returned response. + return_value = operations_pb2.ListOperationsResponse( + next_page_token="next_page_token_value", + ) + + # Wrap the value into a proper Response obj + response_value = Response() + response_value.status_code = 200 + json_return_value = json_format.MessageToJson(return_value) + response_value._content = json_return_value.encode("UTF-8") + req.return_value = response_value + response = await client.list_operations( + name="operations", filter_="my_filter", page_size=10, page_token="abc" + ) + + actual_args = req.call_args + assert actual_args.args[0] == "GET" + assert actual_args.args[1] == "https://longrunning.googleapis.com/v3/operations" + assert actual_args.kwargs["params"] == [ + ("filter", "my_filter"), + ("pageSize", 10), + ("pageToken", "abc"), + ] + + # Establish that the response is the type that we expect. + assert isinstance(response, pagers_async.ListOperationsAsyncPager) + assert response.next_page_token == "next_page_token_value" + + def test_list_operations_rest_failure(): - client = _get_operations_client(http_options=None) + client = _get_operations_client(is_async=False, http_options=None) - with mock.patch.object(Session, "request") as req: + with mock.patch.object(_get_session_type(is_async=False), "request") as req: response_value = Response() response_value.status_code = 400 mock_request = mock.MagicMock() @@ -521,13 +620,30 @@ def test_list_operations_rest_failure(): client.list_operations(name="operations") +@pytest.mark.asyncio +async def test_list_operations_rest_failure_async(): + if not GOOGLE_AUTH_AIO_INSTALLED and "async": + pytest.skip("Skipped because google-api-core[async_rest] is not installed") + + client = _get_operations_client(is_async=True, http_options=None) + + with mock.patch.object(_get_session_type(is_async=True), "request") as req: + response_value = Response() + response_value.status_code = 400 + mock_request = mock.MagicMock() + mock_request.method = "GET" + mock_request.url = "https://longrunning.googleapis.com:443/v1/operations" + response_value.request = mock_request + req.return_value = response_value + with pytest.raises(core_exceptions.GoogleAPIError): + await client.list_operations(name="operations") + + def test_list_operations_rest_pager(): - client = AbstractOperationsClient( - credentials=ga_credentials.AnonymousCredentials(), - ) + client = _get_operations_client(is_async=False, http_options=None) # Mock the http request call within the method and fake a response. - with mock.patch.object(Session, "request") as req: + with mock.patch.object(_get_session_type(is_async=False), "request") as req: # TODO(kbandes): remove this mock unless there's a good reason for it. # with mock.patch.object(path_template, 'transcode') as transcode: # Set the response as a series of pages @@ -574,13 +690,80 @@ def test_list_operations_rest_pager(): assert page_.next_page_token == token -def test_get_operation_rest( - transport: str = "rest", request_type=operations_pb2.GetOperationRequest -): - client = _get_operations_client() +@pytest.mark.asyncio +async def test_list_operations_rest_pager_async(): + if not GOOGLE_AUTH_AIO_INSTALLED and "async": + pytest.skip("Skipped because google-api-core[async_rest] is not installed") + client = _get_operations_client(is_async=True, http_options=None) # Mock the http request call within the method and fake a response. - with mock.patch.object(Session, "request") as req: + with mock.patch.object(_get_session_type(is_async=True), "request") as req: + # TODO(kbandes): remove this mock unless there's a good reason for it. + # with mock.patch.object(path_template, 'transcode') as transcode: + # Set the response as a series of pages + response = ( + operations_pb2.ListOperationsResponse( + operations=[ + operations_pb2.Operation(), + operations_pb2.Operation(), + operations_pb2.Operation(), + ], + next_page_token="abc", + ), + operations_pb2.ListOperationsResponse( + operations=[], + next_page_token="def", + ), + operations_pb2.ListOperationsResponse( + operations=[operations_pb2.Operation()], + next_page_token="ghi", + ), + operations_pb2.ListOperationsResponse( + operations=[operations_pb2.Operation(), operations_pb2.Operation()], + ), + ) + # Two responses for two calls + response = response + response + + # Wrap the values into proper Response objs + response = tuple(json_format.MessageToJson(x) for x in response) + return_values = tuple(Response() for i in response) + for return_val, response_val in zip(return_values, response): + return_val._content = response_val.encode("UTF-8") + return_val.status_code = 200 + req.side_effect = return_values + + pager = await client.list_operations(name="operations") + + responses = [] + async for response in pager: + responses.append(response) + + results = list(responses) + assert len(results) == 6 + assert all(isinstance(i, operations_pb2.Operation) for i in results) + pager = await client.list_operations(name="operations") + + responses = [] + async for response in pager: + responses.append(response) + + assert len(responses) == 6 + assert all(isinstance(i, operations_pb2.Operation) for i in results) + + pages = [] + + async for page in pager.pages: + pages.append(page) + for page_, token in zip(pages, ["", "", "", "abc", "def", "ghi", ""]): + assert page_.next_page_token == token + + +def test_get_operation_rest(): + client = _get_operations_client(is_async=False) + + # Mock the http request call within the method and fake a response. + with mock.patch.object(_get_session_type(is_async=False), "request") as req: # Designate an appropriate value for the returned response. return_value = operations_pb2.Operation( name="operations/sample1", @@ -609,10 +792,46 @@ def test_get_operation_rest( assert response.done is True +@pytest.mark.asyncio +async def test_get_operation_rest_async(): + if not GOOGLE_AUTH_AIO_INSTALLED: + pytest.skip("Skipped because google-api-core[async_rest] is not installed") + client = _get_operations_client(is_async=True) + + # Mock the http request call within the method and fake a response. + with mock.patch.object(_get_session_type(is_async=True), "request") as req: + # Designate an appropriate value for the returned response. + return_value = operations_pb2.Operation( + name="operations/sample1", + done=True, + error=status_pb2.Status(code=411), + ) + + # Wrap the value into a proper Response obj + response_value = mock.Mock() + response_value.status_code = 200 + json_return_value = json_format.MessageToJson(return_value) + response_value.read = mock.AsyncMock(return_value=json_return_value) + req.return_value = response_value + response = await client.get_operation("operations/sample1") + + actual_args = req.call_args + assert actual_args.args[0] == "GET" + assert ( + actual_args.args[1] + == "https://longrunning.googleapis.com/v3/operations/sample1" + ) + + # Establish that the response is the type that we expect. + assert isinstance(response, operations_pb2.Operation) + assert response.name == "operations/sample1" + assert response.done is True + + def test_get_operation_rest_failure(): - client = _get_operations_client(http_options=None) + client = _get_operations_client(is_async=False, http_options=None) - with mock.patch.object(Session, "request") as req: + with mock.patch.object(_get_session_type(is_async=False), "request") as req: response_value = Response() response_value.status_code = 400 mock_request = mock.MagicMock() @@ -624,13 +843,29 @@ def test_get_operation_rest_failure(): client.get_operation("sample0/operations/sample1") -def test_delete_operation_rest( - transport: str = "rest", request_type=operations_pb2.DeleteOperationRequest -): - client = _get_operations_client() +@pytest.mark.asyncio +async def test_get_operation_rest_failure_async(): + if not GOOGLE_AUTH_AIO_INSTALLED: + pytest.skip("Skipped because google-api-core[async_rest] is not installed") + client = _get_operations_client(is_async=True, http_options=None) + + with mock.patch.object(_get_session_type(is_async=True), "request") as req: + response_value = Response() + response_value.status_code = 400 + mock_request = mock.MagicMock() + mock_request.method = "GET" + mock_request.url = "https://longrunning.googleapis.com/v1/operations/sample1" + response_value.request = mock_request + req.return_value = response_value + with pytest.raises(core_exceptions.GoogleAPIError): + await client.get_operation("sample0/operations/sample1") + + +def test_delete_operation_rest(): + client = _get_operations_client(is_async=False) # Mock the http request call within the method and fake a response. - with mock.patch.object(Session, "request") as req: + with mock.patch.object(_get_session_type(is_async=False), "request") as req: # Wrap the value into a proper Response obj response_value = Response() response_value.status_code = 200 @@ -647,10 +882,34 @@ def test_delete_operation_rest( ) +@pytest.mark.asyncio +async def test_delete_operation_rest_async(): + if not GOOGLE_AUTH_AIO_INSTALLED: + pytest.skip("Skipped because google-api-core[async_rest] is not installed") + client = _get_operations_client(is_async=True) + + # Mock the http request call within the method and fake a response. + with mock.patch.object(_get_session_type(is_async=True), "request") as req: + # Wrap the value into a proper Response obj + response_value = Response() + response_value.status_code = 200 + json_return_value = "" + response_value._content = json_return_value.encode("UTF-8") + req.return_value = response_value + await client.delete_operation(name="operations/sample1") + assert req.call_count == 1 + actual_args = req.call_args + assert actual_args.args[0] == "DELETE" + assert ( + actual_args.args[1] + == "https://longrunning.googleapis.com/v3/operations/sample1" + ) + + def test_delete_operation_rest_failure(): - client = _get_operations_client(http_options=None) + client = _get_operations_client(is_async=False, http_options=None) - with mock.patch.object(Session, "request") as req: + with mock.patch.object(_get_session_type(is_async=False), "request") as req: response_value = Response() response_value.status_code = 400 mock_request = mock.MagicMock() @@ -662,11 +921,29 @@ def test_delete_operation_rest_failure(): client.delete_operation(name="sample0/operations/sample1") -def test_cancel_operation_rest(transport: str = "rest"): - client = _get_operations_client() +@pytest.mark.asyncio +async def test_delete_operation_rest_failure_async(): + if not GOOGLE_AUTH_AIO_INSTALLED: + pytest.skip("Skipped because google-api-core[async_rest] is not installed") + client = _get_operations_client(is_async=True, http_options=None) + + with mock.patch.object(_get_session_type(is_async=True), "request") as req: + response_value = Response() + response_value.status_code = 400 + mock_request = mock.MagicMock() + mock_request.method = "DELETE" + mock_request.url = "https://longrunning.googleapis.com/v1/operations/sample1" + response_value.request = mock_request + req.return_value = response_value + with pytest.raises(core_exceptions.GoogleAPIError): + await client.delete_operation(name="sample0/operations/sample1") + + +def test_cancel_operation_rest(): + client = _get_operations_client(is_async=False) # Mock the http request call within the method and fake a response. - with mock.patch.object(Session, "request") as req: + with mock.patch.object(_get_session_type(is_async=False), "request") as req: # Wrap the value into a proper Response obj response_value = Response() response_value.status_code = 200 @@ -683,10 +960,34 @@ def test_cancel_operation_rest(transport: str = "rest"): ) +@pytest.mark.asyncio +async def test_cancel_operation_rest_async(): + if not GOOGLE_AUTH_AIO_INSTALLED: + pytest.skip("Skipped because google-api-core[async_rest] is not installed") + client = _get_operations_client(is_async=True) + + # Mock the http request call within the method and fake a response. + with mock.patch.object(_get_session_type(is_async=True), "request") as req: + # Wrap the value into a proper Response obj + response_value = Response() + response_value.status_code = 200 + json_return_value = "" + response_value._content = json_return_value.encode("UTF-8") + req.return_value = response_value + await client.cancel_operation(name="operations/sample1") + assert req.call_count == 1 + actual_args = req.call_args + assert actual_args.args[0] == "POST" + assert ( + actual_args.args[1] + == "https://longrunning.googleapis.com/v3/operations/sample1:cancel" + ) + + def test_cancel_operation_rest_failure(): - client = _get_operations_client(http_options=None) + client = _get_operations_client(is_async=False, http_options=None) - with mock.patch.object(Session, "request") as req: + with mock.patch.object(_get_session_type(is_async=False), "request") as req: response_value = Response() response_value.status_code = 400 mock_request = mock.MagicMock() @@ -700,44 +1001,71 @@ def test_cancel_operation_rest_failure(): client.cancel_operation(name="sample0/operations/sample1") -def test_credentials_transport_error(): +@pytest.mark.asyncio +async def test_cancel_operation_rest_failure_async(): + if not GOOGLE_AUTH_AIO_INSTALLED: + pytest.skip("Skipped because google-api-core[async_rest] is not installed") + client = _get_operations_client(is_async=False, http_options=None) + + with mock.patch.object(_get_session_type(is_async=False), "request") as req: + response_value = Response() + response_value.status_code = 400 + mock_request = mock.MagicMock() + mock_request.method = "POST" + mock_request.url = ( + "https://longrunning.googleapis.com/v1/operations/sample1:cancel" + ) + response_value.request = mock_request + req.return_value = response_value + with pytest.raises(core_exceptions.GoogleAPIError): + await client.cancel_operation(name="sample0/operations/sample1") + + +@pytest.mark.parametrize( + "client_class,transport_class,credentials", + CLIENTS_WITH_CREDENTIALS, +) +def test_credentials_transport_error(client_class, transport_class, credentials): + if not GOOGLE_AUTH_AIO_INSTALLED and "async" in str(transport_class).lower(): + pytest.skip("Skipped because google-api-core[async_rest] is not installed") + # It is an error to provide credentials and a transport instance. - transport = transports.OperationsRestTransport( - credentials=ga_credentials.AnonymousCredentials(), - ) + transport = transport_class(credentials=credentials) with pytest.raises(ValueError): - AbstractOperationsClient( + client_class( credentials=ga_credentials.AnonymousCredentials(), transport=transport, ) # It is an error to provide a credentials file and a transport instance. - transport = transports.OperationsRestTransport( - credentials=ga_credentials.AnonymousCredentials(), - ) + transport = transport_class(credentials=credentials) with pytest.raises(ValueError): - AbstractOperationsClient( + client_class( client_options={"credentials_file": "credentials.json"}, transport=transport, ) # It is an error to provide scopes and a transport instance. - transport = transports.OperationsRestTransport( - credentials=ga_credentials.AnonymousCredentials(), - ) + transport = transport_class(credentials=credentials) with pytest.raises(ValueError): - AbstractOperationsClient( + client_class( client_options={"scopes": ["1", "2"]}, transport=transport, ) -def test_transport_instance(): +@pytest.mark.parametrize( + "client_class,transport_class,credentials", + CLIENTS_WITH_CREDENTIALS, +) +def test_transport_instance(client_class, transport_class, credentials): + if not GOOGLE_AUTH_AIO_INSTALLED and "async" in str(transport_class).lower(): + pytest.skip("Skipped because google-api-core[async_rest] is not installed") # A client may be instantiated with a custom transport instance. - transport = transports.OperationsRestTransport( - credentials=ga_credentials.AnonymousCredentials(), + transport = transport_class( + credentials=credentials, ) - client = AbstractOperationsClient(transport=transport) + client = client_class(transport=transport) assert client.transport is transport @@ -820,11 +1148,15 @@ def test_operations_base_transport_with_adc(): adc.assert_called_once() -def test_operations_auth_adc(): +@pytest.mark.parametrize( + "client_class", + CLIENTS, +) +def test_operations_auth_adc(client_class): # If no credentials are provided, we should use ADC credentials. with mock.patch.object(google.auth, "default", autospec=True) as adc: adc.return_value = (ga_credentials.AnonymousCredentials(), None) - AbstractOperationsClient() + client_class() adc.assert_called_once_with( scopes=None, default_scopes=(), @@ -852,8 +1184,12 @@ def test_operations_http_transport_client_cert_source_for_mtls(transport_class): mock_configure_mtls_channel.assert_called_once_with(client_cert_source_callback) -def test_operations_host_no_port(): - client = AbstractOperationsClient( +@pytest.mark.parametrize( + "client_class", + CLIENTS, +) +def test_operations_host_no_port(client_class): + client = client_class( credentials=ga_credentials.AnonymousCredentials(), client_options=client_options.ClientOptions( api_endpoint="longrunning.googleapis.com" @@ -862,8 +1198,12 @@ def test_operations_host_no_port(): assert client.transport._host == "https://longrunning.googleapis.com" -def test_operations_host_with_port(): - client = AbstractOperationsClient( +@pytest.mark.parametrize( + "client_class", + CLIENTS, +) +def test_operations_host_with_port(client_class): + client = client_class( credentials=ga_credentials.AnonymousCredentials(), client_options=client_options.ClientOptions( api_endpoint="longrunning.googleapis.com:8000" @@ -872,116 +1212,160 @@ def test_operations_host_with_port(): assert client.transport._host == "https://longrunning.googleapis.com:8000" -def test_common_billing_account_path(): +@pytest.mark.parametrize( + "client_class", + CLIENTS, +) +def test_common_billing_account_path(client_class): billing_account = "squid" expected = "billingAccounts/{billing_account}".format( billing_account=billing_account, ) - actual = AbstractOperationsClient.common_billing_account_path(billing_account) + actual = client_class.common_billing_account_path(billing_account) assert expected == actual -def test_parse_common_billing_account_path(): +@pytest.mark.parametrize( + "client_class", + CLIENTS, +) +def test_parse_common_billing_account_path(client_class): expected = { "billing_account": "clam", } - path = AbstractOperationsClient.common_billing_account_path(**expected) + path = client_class.common_billing_account_path(**expected) # Check that the path construction is reversible. - actual = AbstractOperationsClient.parse_common_billing_account_path(path) + actual = client_class.parse_common_billing_account_path(path) assert expected == actual -def test_common_folder_path(): +@pytest.mark.parametrize( + "client_class", + CLIENTS, +) +def test_common_folder_path(client_class): folder = "whelk" expected = "folders/{folder}".format( folder=folder, ) - actual = AbstractOperationsClient.common_folder_path(folder) + actual = client_class.common_folder_path(folder) assert expected == actual -def test_parse_common_folder_path(): +@pytest.mark.parametrize( + "client_class", + CLIENTS, +) +def test_parse_common_folder_path(client_class): expected = { "folder": "octopus", } - path = AbstractOperationsClient.common_folder_path(**expected) + path = client_class.common_folder_path(**expected) # Check that the path construction is reversible. - actual = AbstractOperationsClient.parse_common_folder_path(path) + actual = client_class.parse_common_folder_path(path) assert expected == actual -def test_common_organization_path(): +@pytest.mark.parametrize( + "client_class", + CLIENTS, +) +def test_common_organization_path(client_class): organization = "oyster" expected = "organizations/{organization}".format( organization=organization, ) - actual = AbstractOperationsClient.common_organization_path(organization) + actual = client_class.common_organization_path(organization) assert expected == actual -def test_parse_common_organization_path(): +@pytest.mark.parametrize( + "client_class", + CLIENTS, +) +def test_parse_common_organization_path(client_class): expected = { "organization": "nudibranch", } - path = AbstractOperationsClient.common_organization_path(**expected) + path = client_class.common_organization_path(**expected) # Check that the path construction is reversible. - actual = AbstractOperationsClient.parse_common_organization_path(path) + actual = client_class.parse_common_organization_path(path) assert expected == actual -def test_common_project_path(): +@pytest.mark.parametrize( + "client_class", + CLIENTS, +) +def test_common_project_path(client_class): project = "cuttlefish" expected = "projects/{project}".format( project=project, ) - actual = AbstractOperationsClient.common_project_path(project) + actual = client_class.common_project_path(project) assert expected == actual -def test_parse_common_project_path(): +@pytest.mark.parametrize( + "client_class", + CLIENTS, +) +def test_parse_common_project_path(client_class): expected = { "project": "mussel", } - path = AbstractOperationsClient.common_project_path(**expected) + path = client_class.common_project_path(**expected) # Check that the path construction is reversible. - actual = AbstractOperationsClient.parse_common_project_path(path) + actual = client_class.parse_common_project_path(path) assert expected == actual -def test_common_location_path(): +@pytest.mark.parametrize( + "client_class", + CLIENTS, +) +def test_common_location_path(client_class): project = "winkle" location = "nautilus" expected = "projects/{project}/locations/{location}".format( project=project, location=location, ) - actual = AbstractOperationsClient.common_location_path(project, location) + actual = client_class.common_location_path(project, location) assert expected == actual -def test_parse_common_location_path(): +@pytest.mark.parametrize( + "client_class", + CLIENTS, +) +def test_parse_common_location_path(client_class): expected = { "project": "scallop", "location": "abalone", } - path = AbstractOperationsClient.common_location_path(**expected) + path = client_class.common_location_path(**expected) # Check that the path construction is reversible. - actual = AbstractOperationsClient.parse_common_location_path(path) + actual = client_class.parse_common_location_path(path) assert expected == actual -def test_client_withDEFAULT_CLIENT_INFO(): +@pytest.mark.parametrize( + "client_class", + CLIENTS, +) +def test_client_withDEFAULT_CLIENT_INFO(client_class): client_info = gapic_v1.client_info.ClientInfo() with mock.patch.object( transports.OperationsTransport, "_prep_wrapped_messages" ) as prep: - AbstractOperationsClient( + client_class( credentials=ga_credentials.AnonymousCredentials(), client_info=client_info, ) @@ -990,7 +1374,7 @@ def test_client_withDEFAULT_CLIENT_INFO(): with mock.patch.object( transports.OperationsTransport, "_prep_wrapped_messages" ) as prep: - transport_class = AbstractOperationsClient.get_transport_class() + transport_class = client_class.get_transport_class() transport_class( credentials=ga_credentials.AnonymousCredentials(), client_info=client_info, From 009a5ee49579d27d8cccdeedf3aea97a69798ffa Mon Sep 17 00:00:00 2001 From: ohmayr Date: Tue, 1 Oct 2024 21:13:57 +0000 Subject: [PATCH 02/14] remove coverage guards --- .../operations_v1/transports/rest_asyncio.py | 12 ++++-------- 1 file changed, 4 insertions(+), 8 deletions(-) diff --git a/google/api_core/operations_v1/transports/rest_asyncio.py b/google/api_core/operations_v1/transports/rest_asyncio.py index 5c903b90..6c197123 100644 --- a/google/api_core/operations_v1/transports/rest_asyncio.py +++ b/google/api_core/operations_v1/transports/rest_asyncio.py @@ -189,8 +189,7 @@ async def _list_operations( *, timeout: Optional[float] = None, metadata: Sequence[Tuple[str, str]] = (), - # TODO(https://github.com/googleapis/python-api-core/issues/710): Add coverage and remove `# pragma: NO COVER`. - ) -> operations_pb2.ListOperationsResponse: # pragma: NO COVER + ) -> operations_pb2.ListOperationsResponse: r"""Asynchronously call the list operations method over HTTP. Args: @@ -264,8 +263,7 @@ async def _get_operation( *, timeout: Optional[float] = None, metadata: Sequence[Tuple[str, str]] = (), - # TODO(https://github.com/googleapis/python-api-core/issues/710): Add coverage and remove `# pragma: NO COVER`. - ) -> operations_pb2.Operation: # pragma: NO COVER + ) -> operations_pb2.Operation: r"""Asynchronously call the get operation method over HTTP. Args: @@ -340,8 +338,7 @@ async def _delete_operation( *, timeout: Optional[float] = None, metadata: Sequence[Tuple[str, str]] = (), - # TODO(https://github.com/googleapis/python-api-core/issues/710): Add coverage and remove `# pragma: NO COVER`. - ) -> empty_pb2.Empty: # pragma: NO COVER + ) -> empty_pb2.Empty: r"""Asynchronously call the delete operation method over HTTP. Args: @@ -409,8 +406,7 @@ async def _cancel_operation( *, timeout: Optional[float] = None, metadata: Sequence[Tuple[str, str]] = (), - # TODO(https://github.com/googleapis/python-api-core/issues/710): Add coverage and remove `# pragma: NO COVER`. - ) -> empty_pb2.Empty: # pragma: NO COVER + ) -> empty_pb2.Empty: r"""Asynchronously call the cancel operation method over HTTP. Args: From 0f7a7d0f2e5505f49c250608c16a9cfcda488d18 Mon Sep 17 00:00:00 2001 From: ohmayr Date: Wed, 2 Oct 2024 16:28:18 +0000 Subject: [PATCH 03/14] address presubmit failures --- google/api_core/operations_v1/__init__.py | 2 +- .../abstract_operations_async_client.py | 1 - .../abstract_operations_base_client.py | 44 +-------- .../operations_v1/transports/__init__.py | 4 - .../operations_v1/transports/rest_asyncio.py | 4 + .../test_operations_rest_client.py | 93 +++++++++++-------- 6 files changed, 61 insertions(+), 87 deletions(-) diff --git a/google/api_core/operations_v1/__init__.py b/google/api_core/operations_v1/__init__.py index 745bcfdf..48c064e4 100644 --- a/google/api_core/operations_v1/__init__.py +++ b/google/api_core/operations_v1/__init__.py @@ -32,7 +32,7 @@ ) from google.api_core.operations_v1.abstract_operations_async_client import AbstractOperationsAsyncClient - __all__.extend(["AbstractOperationsAsyncClient", "OperationsRestAsyncTransport"]) + __all__ += ["AbstractOperationsAsyncClient", "OperationsRestAsyncTransport"] except ImportError: # This import requires the `async_rest` extra. # Don't raise an exception if `OperationsRestAsyncTransport` cannot be imported diff --git a/google/api_core/operations_v1/abstract_operations_async_client.py b/google/api_core/operations_v1/abstract_operations_async_client.py index 12a194f3..5e6b1521 100644 --- a/google/api_core/operations_v1/abstract_operations_async_client.py +++ b/google/api_core/operations_v1/abstract_operations_async_client.py @@ -35,7 +35,6 @@ "`pip install google-api-core[async_rest]`." ) from e -from google.auth.aio import credentials as ga_credentials # type: ignore from google.longrunning import operations_pb2 diff --git a/google/api_core/operations_v1/abstract_operations_base_client.py b/google/api_core/operations_v1/abstract_operations_base_client.py index 58135b60..e66fa15e 100644 --- a/google/api_core/operations_v1/abstract_operations_base_client.py +++ b/google/api_core/operations_v1/abstract_operations_base_client.py @@ -16,7 +16,7 @@ from collections import OrderedDict import os import re -from typing import Awaitable, Callable, Dict, Optional, Type, Union +from typing import Dict, Optional, Type, Union from google.api_core import client_options as client_options_lib # type: ignore from google.api_core import gapic_v1 # type: ignore @@ -39,9 +39,7 @@ from google.auth import credentials as ga_credentials # type: ignore from google.auth.exceptions import MutualTLSChannelError # type: ignore from google.auth.transport import mtls # type: ignore -from google.longrunning import operations_pb2 from google.oauth2 import service_account # type: ignore -from google.protobuf import empty_pb2 # type: ignore class AbstractOperationsBaseClientMeta(type): @@ -364,6 +362,7 @@ def __init__( ) self._transport = transport else: + # TODO (WIP): This code block will fail becuase async rest layer does not support all params. Transport = type(self).get_transport_class(transport) self._transport = Transport( credentials=credentials, @@ -375,42 +374,3 @@ def __init__( client_info=client_info, always_use_jwt_access=True, ) - - @property - def list_operations( - self, - ) -> Callable[ - [operations_pb2.ListOperationsRequest], - Union[ - operations_pb2.ListOperationsResponse, - Awaitable[operations_pb2.ListOperationsResponse], - ], - ]: - raise NotImplementedError() - - @property - def get_operation( - self, - ) -> Callable[ - [operations_pb2.GetOperationRequest], - Union[operations_pb2.Operation, Awaitable[operations_pb2.Operation]], - ]: - raise NotImplementedError() - - @property - def delete_operation( - self, - ) -> Callable[ - [operations_pb2.DeleteOperationRequest], - Union[empty_pb2.Empty, Awaitable[empty_pb2.Empty]], - ]: - raise NotImplementedError() - - @property - def cancel_operation( - self, - ) -> Callable[ - [operations_pb2.CancelOperationRequest], - Union[empty_pb2.Empty, Awaitable[empty_pb2.Empty]], - ]: - raise NotImplementedError() diff --git a/google/api_core/operations_v1/transports/__init__.py b/google/api_core/operations_v1/transports/__init__.py index 1771e711..2ae79e4e 100644 --- a/google/api_core/operations_v1/transports/__init__.py +++ b/google/api_core/operations_v1/transports/__init__.py @@ -37,7 +37,3 @@ # Don't raise an exception if `OperationsRestAsyncTransport` cannot be imported # as other transports are still available. pass - -# Compile a registry of transports. -_transport_registry = OrderedDict() -_transport_registry["rest"] = OperationsRestTransport diff --git a/google/api_core/operations_v1/transports/rest_asyncio.py b/google/api_core/operations_v1/transports/rest_asyncio.py index 6c197123..5e47dc54 100644 --- a/google/api_core/operations_v1/transports/rest_asyncio.py +++ b/google/api_core/operations_v1/transports/rest_asyncio.py @@ -138,6 +138,7 @@ def _prep_wrapped_messages(self, client_info): ), default_timeout=10.0, client_info=client_info, + kind="rest_asyncio", ), self.get_operation: gapic_v1.method_async.wrap_method( self.get_operation, @@ -152,6 +153,7 @@ def _prep_wrapped_messages(self, client_info): ), default_timeout=10.0, client_info=client_info, + kind="rest_asyncio", ), self.delete_operation: gapic_v1.method_async.wrap_method( self.delete_operation, @@ -166,6 +168,7 @@ def _prep_wrapped_messages(self, client_info): ), default_timeout=10.0, client_info=client_info, + kind="rest_asyncio", ), self.cancel_operation: gapic_v1.method_async.wrap_method( self.cancel_operation, @@ -180,6 +183,7 @@ def _prep_wrapped_messages(self, client_info): ), default_timeout=10.0, client_info=client_info, + kind="rest_asyncio", ), } diff --git a/tests/unit/operations_v1/test_operations_rest_client.py b/tests/unit/operations_v1/test_operations_rest_client.py index a3caa2f3..7e82460d 100644 --- a/tests/unit/operations_v1/test_operations_rest_client.py +++ b/tests/unit/operations_v1/test_operations_rest_client.py @@ -14,7 +14,6 @@ # limitations under the License. # import os -from typing import Union import mock import pytest @@ -86,37 +85,43 @@ ], } -CLIENTS = [ - AbstractOperationsClient, -] - -CLIENTS_WITH_TRANSPORT = [ - (AbstractOperationsClient, transports.OperationsRestTransport, "rest"), -] -CLIENTS_WITH_CREDENTIALS = [ - ( - AbstractOperationsClient, - transports.OperationsRestTransport, - ga_credentials.AnonymousCredentials(), - ), -] if GOOGLE_AUTH_AIO_INSTALLED: - CLIENTS.append(AbstractOperationsAsyncClient) - CLIENTS_WITH_TRANSPORT.append( + CLIENTS = [AbstractOperationsClient, AbstractOperationsAsyncClient] + CLIENTS_WITH_TRANSPORT = [ + (AbstractOperationsClient, transports.OperationsRestTransport, "rest"), ( AbstractOperationsAsyncClient, transports.OperationsRestAsyncTransport, "rest_asyncio", - ) - ), - CLIENTS_WITH_CREDENTIALS.append( + ), + ] + CLIENTS_WITH_CREDENTIALS = [ + ( + AbstractOperationsClient, + transports.OperationsRestTransport, + ga_credentials.AnonymousCredentials(), + ), ( AbstractOperationsAsyncClient, transports.OperationsRestAsyncTransport, ga_credentials_async.AnonymousCredentials(), - ) - ), + ), + ] +else: + CLIENTS = [ + AbstractOperationsClient, + ] + CLIENTS_WITH_TRANSPORT = [ + (AbstractOperationsClient, transports.OperationsRestTransport, "rest"), + ] + CLIENTS_WITH_CREDENTIALS = [ + ( + AbstractOperationsClient, + transports.OperationsRestTransport, + ga_credentials.AnonymousCredentials(), + ), + ] def client_cert_source_callback(): @@ -133,16 +138,16 @@ def _get_session_type(is_async: bool): def _get_operations_client(is_async: bool, http_options=HTTP_OPTIONS): if is_async and GOOGLE_AUTH_AIO_INSTALLED: - transport = transports.rest_asyncio.OperationsRestAsyncTransport( + async_transport = transports.rest_asyncio.OperationsRestAsyncTransport( credentials=ga_credentials_async.AnonymousCredentials(), http_options=http_options, ) - return AbstractOperationsAsyncClient(transport=transport) + return AbstractOperationsAsyncClient(transport=async_transport) else: - transport = transports.rest.OperationsRestTransport( + sync_transport = transports.rest.OperationsRestTransport( credentials=ga_credentials.AnonymousCredentials(), http_options=http_options ) - return AbstractOperationsClient(transport=transport) + return AbstractOperationsClient(transport=sync_transport) # If default endpoint is localhost, then default mtls endpoint will be the same. @@ -582,10 +587,12 @@ async def test_list_operations_rest_async(): ) # Wrap the value into a proper Response obj - response_value = Response() + response_value = mock.Mock() response_value.status_code = 200 json_return_value = json_format.MessageToJson(return_value) - response_value._content = json_return_value.encode("UTF-8") + response_value.read = mock.AsyncMock( + return_value=json_return_value.encode("UTF-8") + ) req.return_value = response_value response = await client.list_operations( name="operations", filter_="my_filter", page_size=10, page_token="abc" @@ -628,8 +635,9 @@ async def test_list_operations_rest_failure_async(): client = _get_operations_client(is_async=True, http_options=None) with mock.patch.object(_get_session_type(is_async=True), "request") as req: - response_value = Response() + response_value = mock.Mock() response_value.status_code = 400 + response_value.read = mock.AsyncMock(return_value=b"{}") mock_request = mock.MagicMock() mock_request.method = "GET" mock_request.url = "https://longrunning.googleapis.com:443/v1/operations" @@ -727,9 +735,9 @@ async def test_list_operations_rest_pager_async(): # Wrap the values into proper Response objs response = tuple(json_format.MessageToJson(x) for x in response) - return_values = tuple(Response() for i in response) + return_values = tuple(mock.Mock() for i in response) for return_val, response_val in zip(return_values, response): - return_val._content = response_val.encode("UTF-8") + return_val.read = mock.AsyncMock(return_value=response_val.encode("UTF-8")) return_val.status_code = 200 req.side_effect = return_values @@ -850,8 +858,9 @@ async def test_get_operation_rest_failure_async(): client = _get_operations_client(is_async=True, http_options=None) with mock.patch.object(_get_session_type(is_async=True), "request") as req: - response_value = Response() + response_value = mock.Mock() response_value.status_code = 400 + response_value.read = mock.AsyncMock(return_value=b"{}") mock_request = mock.MagicMock() mock_request.method = "GET" mock_request.url = "https://longrunning.googleapis.com/v1/operations/sample1" @@ -891,10 +900,12 @@ async def test_delete_operation_rest_async(): # Mock the http request call within the method and fake a response. with mock.patch.object(_get_session_type(is_async=True), "request") as req: # Wrap the value into a proper Response obj - response_value = Response() + response_value = mock.Mock() response_value.status_code = 200 json_return_value = "" - response_value._content = json_return_value.encode("UTF-8") + response_value.read = mock.AsyncMock( + return_value=json_return_value.encode("UTF-8") + ) req.return_value = response_value await client.delete_operation(name="operations/sample1") assert req.call_count == 1 @@ -928,8 +939,9 @@ async def test_delete_operation_rest_failure_async(): client = _get_operations_client(is_async=True, http_options=None) with mock.patch.object(_get_session_type(is_async=True), "request") as req: - response_value = Response() + response_value = mock.Mock() response_value.status_code = 400 + response_value.read = mock.AsyncMock(return_value=b"{}") mock_request = mock.MagicMock() mock_request.method = "DELETE" mock_request.url = "https://longrunning.googleapis.com/v1/operations/sample1" @@ -969,10 +981,12 @@ async def test_cancel_operation_rest_async(): # Mock the http request call within the method and fake a response. with mock.patch.object(_get_session_type(is_async=True), "request") as req: # Wrap the value into a proper Response obj - response_value = Response() + response_value = mock.Mock() response_value.status_code = 200 json_return_value = "" - response_value._content = json_return_value.encode("UTF-8") + response_value.read = mock.AsyncMock( + return_value=json_return_value.encode("UTF-8") + ) req.return_value = response_value await client.cancel_operation(name="operations/sample1") assert req.call_count == 1 @@ -1007,9 +1021,10 @@ async def test_cancel_operation_rest_failure_async(): pytest.skip("Skipped because google-api-core[async_rest] is not installed") client = _get_operations_client(is_async=False, http_options=None) - with mock.patch.object(_get_session_type(is_async=False), "request") as req: - response_value = Response() + with mock.patch.object(_get_session_type(is_async=True), "request") as req: + response_value = mock.Mock() response_value.status_code = 400 + response_value.read = mock.AsyncMock(return_value=b"{}") mock_request = mock.MagicMock() mock_request.method = "POST" mock_request.url = ( From 5b51c642a706da014128f0005c77874dc2c6ff01 Mon Sep 17 00:00:00 2001 From: ohmayr Date: Wed, 2 Oct 2024 16:41:53 +0000 Subject: [PATCH 04/14] fix coverage for cancel operation --- tests/unit/operations_v1/test_operations_rest_client.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/unit/operations_v1/test_operations_rest_client.py b/tests/unit/operations_v1/test_operations_rest_client.py index 7e82460d..51356c3e 100644 --- a/tests/unit/operations_v1/test_operations_rest_client.py +++ b/tests/unit/operations_v1/test_operations_rest_client.py @@ -1019,7 +1019,7 @@ def test_cancel_operation_rest_failure(): async def test_cancel_operation_rest_failure_async(): if not GOOGLE_AUTH_AIO_INSTALLED: pytest.skip("Skipped because google-api-core[async_rest] is not installed") - client = _get_operations_client(is_async=False, http_options=None) + client = _get_operations_client(is_async=True, http_options=None) with mock.patch.object(_get_session_type(is_async=True), "request") as req: response_value = mock.Mock() From 50d751e3d02e882904f33c3d14b257c19bb0b6ea Mon Sep 17 00:00:00 2001 From: ohmayr Date: Wed, 2 Oct 2024 17:21:04 +0000 Subject: [PATCH 05/14] tests cleanup --- tests/unit/operations_v1/test_operations_rest_client.py | 8 ++------ 1 file changed, 2 insertions(+), 6 deletions(-) diff --git a/tests/unit/operations_v1/test_operations_rest_client.py b/tests/unit/operations_v1/test_operations_rest_client.py index 51356c3e..f1adbf04 100644 --- a/tests/unit/operations_v1/test_operations_rest_client.py +++ b/tests/unit/operations_v1/test_operations_rest_client.py @@ -629,7 +629,7 @@ def test_list_operations_rest_failure(): @pytest.mark.asyncio async def test_list_operations_rest_failure_async(): - if not GOOGLE_AUTH_AIO_INSTALLED and "async": + if not GOOGLE_AUTH_AIO_INSTALLED: pytest.skip("Skipped because google-api-core[async_rest] is not installed") client = _get_operations_client(is_async=True, http_options=None) @@ -700,7 +700,7 @@ def test_list_operations_rest_pager(): @pytest.mark.asyncio async def test_list_operations_rest_pager_async(): - if not GOOGLE_AUTH_AIO_INSTALLED and "async": + if not GOOGLE_AUTH_AIO_INSTALLED: pytest.skip("Skipped because google-api-core[async_rest] is not installed") client = _get_operations_client(is_async=True, http_options=None) @@ -1041,8 +1041,6 @@ async def test_cancel_operation_rest_failure_async(): CLIENTS_WITH_CREDENTIALS, ) def test_credentials_transport_error(client_class, transport_class, credentials): - if not GOOGLE_AUTH_AIO_INSTALLED and "async" in str(transport_class).lower(): - pytest.skip("Skipped because google-api-core[async_rest] is not installed") # It is an error to provide credentials and a transport instance. transport = transport_class(credentials=credentials) @@ -1074,8 +1072,6 @@ def test_credentials_transport_error(client_class, transport_class, credentials) CLIENTS_WITH_CREDENTIALS, ) def test_transport_instance(client_class, transport_class, credentials): - if not GOOGLE_AUTH_AIO_INSTALLED and "async" in str(transport_class).lower(): - pytest.skip("Skipped because google-api-core[async_rest] is not installed") # A client may be instantiated with a custom transport instance. transport = transport_class( credentials=credentials, From 5a4af8d2f9cba1f1e3332c44bd19b6e88423b770 Mon Sep 17 00:00:00 2001 From: ohmayr Date: Wed, 2 Oct 2024 19:16:58 +0000 Subject: [PATCH 06/14] fix incorrect tests --- .../abstract_operations_async_client.py | 2 +- .../abstract_operations_base_client.py | 55 ++++-- .../abstract_operations_client.py | 38 ++++ .../test_operations_rest_client.py | 177 ++++++++++-------- 4 files changed, 178 insertions(+), 94 deletions(-) diff --git a/google/api_core/operations_v1/abstract_operations_async_client.py b/google/api_core/operations_v1/abstract_operations_async_client.py index 5e6b1521..80b4621a 100644 --- a/google/api_core/operations_v1/abstract_operations_async_client.py +++ b/google/api_core/operations_v1/abstract_operations_async_client.py @@ -98,7 +98,7 @@ def __init__( """ super().__init__( credentials=credentials, # type: ignore - transport=transport, + transport=transport or "rest_asyncio", client_options=client_options, client_info=client_info, ) diff --git a/google/api_core/operations_v1/abstract_operations_base_client.py b/google/api_core/operations_v1/abstract_operations_base_client.py index e66fa15e..7358ac78 100644 --- a/google/api_core/operations_v1/abstract_operations_base_client.py +++ b/google/api_core/operations_v1/abstract_operations_base_client.py @@ -19,6 +19,7 @@ from typing import Dict, Optional, Type, Union from google.api_core import client_options as client_options_lib # type: ignore +from google.api_core import exceptions as core_exceptions # type: ignore from google.api_core import gapic_v1 # type: ignore from google.api_core.operations_v1.transports.base import ( DEFAULT_CLIENT_INFO, @@ -39,7 +40,6 @@ from google.auth import credentials as ga_credentials # type: ignore from google.auth.exceptions import MutualTLSChannelError # type: ignore from google.auth.transport import mtls # type: ignore -from google.oauth2 import service_account # type: ignore class AbstractOperationsBaseClientMeta(type): @@ -143,9 +143,7 @@ def from_service_account_info(cls, info: dict, *args, **kwargs): Returns: AbstractOperationsClient: The constructed client. """ - credentials = service_account.Credentials.from_service_account_info(info) - kwargs["credentials"] = credentials - return cls(*args, **kwargs) + raise NotImplementedError("`from_service_account_info` is not implemented.") @classmethod def from_service_account_file(cls, filename: str, *args, **kwargs): @@ -161,9 +159,7 @@ def from_service_account_file(cls, filename: str, *args, **kwargs): Returns: AbstractOperationsClient: The constructed client. """ - credentials = service_account.Credentials.from_service_account_file(filename) - kwargs["credentials"] = credentials - return cls(*args, **kwargs) + raise NotImplementedError("`from_service_account_file` is not implemented.") from_service_account_json = from_service_account_file @@ -362,15 +358,38 @@ def __init__( ) self._transport = transport else: - # TODO (WIP): This code block will fail becuase async rest layer does not support all params. Transport = type(self).get_transport_class(transport) - self._transport = Transport( - credentials=credentials, - credentials_file=client_options.credentials_file, - host=api_endpoint, - scopes=client_options.scopes, - client_cert_source_for_mtls=client_cert_source_func, - quota_project_id=client_options.quota_project_id, - client_info=client_info, - always_use_jwt_access=True, - ) + if "async" in str(Transport).lower(): + # TODO(https://github.com/googleapis/gapic-generator-python/issues/2136): Support the following parameters in async rest. + unsupported_params = { + "google.api_core.client_options.ClientOptions.credentials_file": client_options.credentials_file, + "google.api_core.client_options.ClientOptions.scopes": client_options.scopes, + "google.api_core.client_options.ClientOptions.quota_project_id": client_options.quota_project_id, + "google.api_core.client_options.ClientOptions.client_cert_source": client_options.client_cert_source, + "google.api_core.client_options.ClientOptions.api_audience": client_options.api_audience, + } + provided_unsupported_params = [ + name + for name, value in unsupported_params.items() + if value is not None + ] + if provided_unsupported_params: + raise core_exceptions.AsyncRestUnsupportedParameterError( + f"The following provided parameters are not supported for `transport=rest_asyncio`: {', '.join(provided_unsupported_params)}" + ) + self._transport = Transport( + credentials=credentials, + host=api_endpoint, + client_info=client_info, + ) + else: + self._transport = Transport( + credentials=credentials, + credentials_file=client_options.credentials_file, + host=api_endpoint, + scopes=client_options.scopes, + client_cert_source_for_mtls=client_cert_source_func, + quota_project_id=client_options.quota_project_id, + client_info=client_info, + always_use_jwt_access=True, + ) diff --git a/google/api_core/operations_v1/abstract_operations_client.py b/google/api_core/operations_v1/abstract_operations_client.py index 64f2cb7d..fc445362 100644 --- a/google/api_core/operations_v1/abstract_operations_client.py +++ b/google/api_core/operations_v1/abstract_operations_client.py @@ -28,6 +28,7 @@ ) from google.auth import credentials as ga_credentials # type: ignore from google.longrunning import operations_pb2 +from google.oauth2 import service_account # type: ignore import grpc OptionalRetry = Union[retries.Retry, object] @@ -98,6 +99,43 @@ def __init__( client_info=client_info, ) + @classmethod + def from_service_account_info(cls, info: dict, *args, **kwargs): + """Creates an instance of this client using the provided credentials + info. + + Args: + info (dict): The service account private key info. + args: Additional arguments to pass to the constructor. + kwargs: Additional arguments to pass to the constructor. + + Returns: + AbstractOperationsClient: The constructed client. + """ + credentials = service_account.Credentials.from_service_account_info(info) + kwargs["credentials"] = credentials + return cls(*args, **kwargs) + + @classmethod + def from_service_account_file(cls, filename: str, *args, **kwargs): + """Creates an instance of this client using the provided credentials + file. + + Args: + filename (str): The path to the service account private key json + file. + args: Additional arguments to pass to the constructor. + kwargs: Additional arguments to pass to the constructor. + + Returns: + AbstractOperationsClient: The constructed client. + """ + credentials = service_account.Credentials.from_service_account_file(filename) + kwargs["credentials"] = credentials + return cls(*args, **kwargs) + + from_service_account_json = from_service_account_file + def list_operations( self, name: str, diff --git a/tests/unit/operations_v1/test_operations_rest_client.py b/tests/unit/operations_v1/test_operations_rest_client.py index f1adbf04..68bd9287 100644 --- a/tests/unit/operations_v1/test_operations_rest_client.py +++ b/tests/unit/operations_v1/test_operations_rest_client.py @@ -197,16 +197,22 @@ def test__get_default_mtls_endpoint(client_class): ) def test_operations_client_from_service_account_info(client_class): creds = ga_credentials.AnonymousCredentials() - with mock.patch.object( - service_account.Credentials, "from_service_account_info" - ) as factory: - factory.return_value = creds - info = {"valid": True} - client = client_class.from_service_account_info(info) - assert client.transport._credentials == creds - assert isinstance(client, client_class) + if "async" in str(client_class): + # TODO(): Add support for service account info to async REST transport. + with pytest.raises(NotImplementedError): + info = {"valid": True} + client_class.from_service_account_info(info) + else: + with mock.patch.object( + service_account.Credentials, "from_service_account_info" + ) as factory: + factory.return_value = creds + info = {"valid": True} + client = client_class.from_service_account_info(info) + assert client.transport._credentials == creds + assert isinstance(client, client_class) - assert client.transport._host == "https://longrunning.googleapis.com" + assert client.transport._host == "https://longrunning.googleapis.com" @pytest.mark.parametrize( @@ -238,20 +244,26 @@ def test_operations_client_service_account_always_use_jwt(transport_class): CLIENTS, ) def test_operations_client_from_service_account_file(client_class): - creds = ga_credentials.AnonymousCredentials() - with mock.patch.object( - service_account.Credentials, "from_service_account_file" - ) as factory: - factory.return_value = creds - client = client_class.from_service_account_file("dummy/file/path.json") - assert client.transport._credentials == creds - assert isinstance(client, client_class) - client = client_class.from_service_account_json("dummy/file/path.json") - assert client.transport._credentials == creds - assert isinstance(client, client_class) + if "async" in str(client_class): + # TODO(): Add support for service account creds to async REST transport. + with pytest.raises(NotImplementedError): + client_class.from_service_account_file("dummy/file/path.json") + else: + creds = ga_credentials.AnonymousCredentials() + with mock.patch.object( + service_account.Credentials, "from_service_account_file" + ) as factory: + factory.return_value = creds + client = client_class.from_service_account_file("dummy/file/path.json") + assert client.transport._credentials == creds + assert isinstance(client, client_class) + + client = client_class.from_service_account_json("dummy/file/path.json") + assert client.transport._credentials == creds + assert isinstance(client, client_class) - assert client.transport._host == "https://longrunning.googleapis.com" + assert client.transport._host == "https://longrunning.googleapis.com" @pytest.mark.parametrize( @@ -273,9 +285,10 @@ def test_operations_client_get_transport_class( assert transport == transport_class +# TODO(): Update this test case to include async REST once we have support for MTLS. @pytest.mark.parametrize( "client_class,transport_class,transport_name", - CLIENTS_WITH_TRANSPORT, + [(AbstractOperationsClient, transports.OperationsRestTransport, "rest")], ) @mock.patch.object( AbstractOperationsClient, @@ -501,19 +514,24 @@ def test_operations_client_client_options_scopes( options = client_options.ClientOptions( scopes=["1", "2"], ) - with mock.patch.object(transport_class, "__init__") as patched: - patched.return_value = None - client = client_class(client_options=options, transport=transport_name) - patched.assert_called_once_with( - credentials=None, - credentials_file=None, - host=client.DEFAULT_ENDPOINT, - scopes=["1", "2"], - client_cert_source_for_mtls=None, - quota_project_id=None, - client_info=transports.base.DEFAULT_CLIENT_INFO, - always_use_jwt_access=True, - ) + if "async" in str(client_class): + # TODO(): Add support for scopes to async REST transport. + with pytest.raises(core_exceptions.AsyncRestUnsupportedParameterError): + client_class(client_options=options, transport=transport_name) + else: + with mock.patch.object(transport_class, "__init__") as patched: + patched.return_value = None + client = client_class(client_options=options, transport=transport_name) + patched.assert_called_once_with( + credentials=None, + credentials_file=None, + host=client.DEFAULT_ENDPOINT, + scopes=["1", "2"], + client_cert_source_for_mtls=None, + quota_project_id=None, + client_info=transports.base.DEFAULT_CLIENT_INFO, + always_use_jwt_access=True, + ) @pytest.mark.parametrize( @@ -525,19 +543,24 @@ def test_operations_client_client_options_credentials_file( ): # Check the case credentials file is provided. options = client_options.ClientOptions(credentials_file="credentials.json") - with mock.patch.object(transport_class, "__init__") as patched: - patched.return_value = None - client = client_class(client_options=options, transport=transport_name) - patched.assert_called_once_with( - credentials=None, - credentials_file="credentials.json", - host=client.DEFAULT_ENDPOINT, - scopes=None, - client_cert_source_for_mtls=None, - quota_project_id=None, - client_info=transports.base.DEFAULT_CLIENT_INFO, - always_use_jwt_access=True, - ) + if "async" in str(client_class): + # TODO(): Add support for credentials file to async REST transport. + with pytest.raises(core_exceptions.AsyncRestUnsupportedParameterError): + client_class(client_options=options, transport=transport_name) + else: + with mock.patch.object(transport_class, "__init__") as patched: + patched.return_value = None + client = client_class(client_options=options, transport=transport_name) + patched.assert_called_once_with( + credentials=None, + credentials_file="credentials.json", + host=client.DEFAULT_ENDPOINT, + scopes=None, + client_cert_source_for_mtls=None, + quota_project_id=None, + client_info=transports.base.DEFAULT_CLIENT_INFO, + always_use_jwt_access=True, + ) def test_list_operations_rest(): @@ -1167,12 +1190,22 @@ def test_operations_auth_adc(client_class): # If no credentials are provided, we should use ADC credentials. with mock.patch.object(google.auth, "default", autospec=True) as adc: adc.return_value = (ga_credentials.AnonymousCredentials(), None) - client_class() - adc.assert_called_once_with( - scopes=None, - default_scopes=(), - quota_project_id=None, - ) + + if "async" in str(client_class).lower(): + # TODO(): Add support for adc to async REST transport. + # NOTE: Ideally, the logic for adc shouldn't be called if transport + # is set to async REST. If the user does not configure credentials + # of type `google.auth.aio.credentials.Credentials`, + # we should raise an exception to avoid the adc workflow. + with pytest.raises(google.auth.exceptions.InvalidType): + client_class() + else: + client_class() + adc.assert_called_once_with( + scopes=None, + default_scopes=(), + quota_project_id=None, + ) # TODO(https://github.com/googleapis/python-api-core/issues/705): Add @@ -1196,12 +1229,12 @@ def test_operations_http_transport_client_cert_source_for_mtls(transport_class): @pytest.mark.parametrize( - "client_class", - CLIENTS, + "client_class,transport_class,credentials", + CLIENTS_WITH_CREDENTIALS, ) -def test_operations_host_no_port(client_class): +def test_operations_host_no_port(client_class, transport_class, credentials): client = client_class( - credentials=ga_credentials.AnonymousCredentials(), + credentials=credentials, client_options=client_options.ClientOptions( api_endpoint="longrunning.googleapis.com" ), @@ -1210,12 +1243,12 @@ def test_operations_host_no_port(client_class): @pytest.mark.parametrize( - "client_class", - CLIENTS, + "client_class,transport_class,credentials", + CLIENTS_WITH_CREDENTIALS, ) -def test_operations_host_with_port(client_class): +def test_operations_host_with_port(client_class, transport_class, credentials): client = client_class( - credentials=ga_credentials.AnonymousCredentials(), + credentials=credentials, client_options=client_options.ClientOptions( api_endpoint="longrunning.googleapis.com:8000" ), @@ -1367,27 +1400,21 @@ def test_parse_common_location_path(client_class): @pytest.mark.parametrize( - "client_class", - CLIENTS, + "client_class,transport_class,credentials", + CLIENTS_WITH_CREDENTIALS, ) -def test_client_withDEFAULT_CLIENT_INFO(client_class): +def test_client_withDEFAULT_CLIENT_INFO(client_class, transport_class, credentials): client_info = gapic_v1.client_info.ClientInfo() - - with mock.patch.object( - transports.OperationsTransport, "_prep_wrapped_messages" - ) as prep: + with mock.patch.object(transport_class, "_prep_wrapped_messages") as prep: client_class( - credentials=ga_credentials.AnonymousCredentials(), + credentials=credentials, client_info=client_info, ) prep.assert_called_once_with(client_info) - with mock.patch.object( - transports.OperationsTransport, "_prep_wrapped_messages" - ) as prep: - transport_class = client_class.get_transport_class() + with mock.patch.object(transport_class, "_prep_wrapped_messages") as prep: transport_class( - credentials=ga_credentials.AnonymousCredentials(), + credentials=credentials, client_info=client_info, ) prep.assert_called_once_with(client_info) From 2dea508d727a12cd5d31a0179f1ae3bc8941718b Mon Sep 17 00:00:00 2001 From: ohmayr Date: Wed, 2 Oct 2024 20:45:32 +0000 Subject: [PATCH 07/14] file bugs --- .../abstract_operations_async_client.py | 2 ++ .../abstract_operations_base_client.py | 24 +++++++++++-------- .../operations_v1/transports/rest_asyncio.py | 4 ++++ 3 files changed, 20 insertions(+), 10 deletions(-) diff --git a/google/api_core/operations_v1/abstract_operations_async_client.py b/google/api_core/operations_v1/abstract_operations_async_client.py index 80b4621a..ac34941a 100644 --- a/google/api_core/operations_v1/abstract_operations_async_client.py +++ b/google/api_core/operations_v1/abstract_operations_async_client.py @@ -98,6 +98,8 @@ def __init__( """ super().__init__( credentials=credentials, # type: ignore + # NOTE: If a transport is not provided, we force the client to use the async + # REST transport, as it should. transport=transport or "rest_asyncio", client_options=client_options, client_info=client_info, diff --git a/google/api_core/operations_v1/abstract_operations_base_client.py b/google/api_core/operations_v1/abstract_operations_base_client.py index 7358ac78..5db9adf8 100644 --- a/google/api_core/operations_v1/abstract_operations_base_client.py +++ b/google/api_core/operations_v1/abstract_operations_base_client.py @@ -132,23 +132,23 @@ def _get_default_mtls_endpoint(api_endpoint): @classmethod def from_service_account_info(cls, info: dict, *args, **kwargs): - """Creates an instance of this client using the provided credentials - info. + """ + This class method should be overridden by the subclasses. Args: info (dict): The service account private key info. args: Additional arguments to pass to the constructor. kwargs: Additional arguments to pass to the constructor. - Returns: - AbstractOperationsClient: The constructed client. + Raises: + NotImplementedError: If the method is called on the base class. """ raise NotImplementedError("`from_service_account_info` is not implemented.") @classmethod def from_service_account_file(cls, filename: str, *args, **kwargs): - """Creates an instance of this client using the provided credentials - file. + """ + This class method should be overridden by the subclasses. Args: filename (str): The path to the service account private key json @@ -156,8 +156,8 @@ def from_service_account_file(cls, filename: str, *args, **kwargs): args: Additional arguments to pass to the constructor. kwargs: Additional arguments to pass to the constructor. - Returns: - AbstractOperationsClient: The constructed client. + Raises: + NotImplementedError: If the method is called on the base class. """ raise NotImplementedError("`from_service_account_file` is not implemented.") @@ -359,14 +359,18 @@ def __init__( self._transport = transport else: Transport = type(self).get_transport_class(transport) + # NOTE: The conditional logic below to initialize the transport can be removed + # once we have feature parity with the sync transport. if "async" in str(Transport).lower(): - # TODO(https://github.com/googleapis/gapic-generator-python/issues/2136): Support the following parameters in async rest. unsupported_params = { + # TODO(https://github.com/googleapis/python-api-core/issues/715): Add support for `credentials_file` to async REST transport. "google.api_core.client_options.ClientOptions.credentials_file": client_options.credentials_file, + # TODO(https://github.com/googleapis/python-api-core/issues/716): Add support for `scopes` to async REST transport. "google.api_core.client_options.ClientOptions.scopes": client_options.scopes, + # TODO(https://github.com/googleapis/python-api-core/issues/717): Add support for `quota_project_id` to async REST transport. "google.api_core.client_options.ClientOptions.quota_project_id": client_options.quota_project_id, + # TODO(https://github.com/googleapis/python-api-core/issues/718): Add support for `client_cert_source` to async REST transport. "google.api_core.client_options.ClientOptions.client_cert_source": client_options.client_cert_source, - "google.api_core.client_options.ClientOptions.api_audience": client_options.api_audience, } provided_unsupported_params = [ name diff --git a/google/api_core/operations_v1/transports/rest_asyncio.py b/google/api_core/operations_v1/transports/rest_asyncio.py index 5e47dc54..1b5dadbb 100644 --- a/google/api_core/operations_v1/transports/rest_asyncio.py +++ b/google/api_core/operations_v1/transports/rest_asyncio.py @@ -77,6 +77,10 @@ def __init__( url_scheme: str = "https", http_options: Optional[Dict] = None, path_prefix: str = "v1", + # TODO(https://github.com/googleapis/python-api-core/issues/715): Add support for `credentials_file` to async REST transport. + # TODO(https://github.com/googleapis/python-api-core/issues/716): Add support for `scopes` to async REST transport. + # TODO(https://github.com/googleapis/python-api-core/issues/717): Add support for `quota_project_id` to async REST transport. + # TODO(https://github.com/googleapis/python-api-core/issues/718): Add support for `client_cert_source` to async REST transport. ) -> None: """Instantiate the transport. From f448498768dc49a9d4f3e462c03081f226f428de Mon Sep 17 00:00:00 2001 From: ohmayr Date: Wed, 2 Oct 2024 20:49:18 +0000 Subject: [PATCH 08/14] add auth import --- tests/unit/operations_v1/test_operations_rest_client.py | 1 + 1 file changed, 1 insertion(+) diff --git a/tests/unit/operations_v1/test_operations_rest_client.py b/tests/unit/operations_v1/test_operations_rest_client.py index 68bd9287..955b1d1b 100644 --- a/tests/unit/operations_v1/test_operations_rest_client.py +++ b/tests/unit/operations_v1/test_operations_rest_client.py @@ -30,6 +30,7 @@ from google.api_core import gapic_v1 from google.api_core.operations_v1 import AbstractOperationsClient +import google.auth from google.api_core.operations_v1 import pagers from google.api_core.operations_v1 import pagers_async from google.api_core.operations_v1 import transports From ec2baa4e25914927664256b2c008f09009042246 Mon Sep 17 00:00:00 2001 From: ohmayr Date: Mon, 7 Oct 2024 02:09:21 +0000 Subject: [PATCH 09/14] address PR comments --- google/api_core/operations_v1/__init__.py | 2 +- .../abstract_operations_async_client.py | 6 +- .../test_operations_rest_client.py | 77 +++++++------------ 3 files changed, 29 insertions(+), 56 deletions(-) diff --git a/google/api_core/operations_v1/__init__.py b/google/api_core/operations_v1/__init__.py index 48c064e4..3dbc9217 100644 --- a/google/api_core/operations_v1/__init__.py +++ b/google/api_core/operations_v1/__init__.py @@ -36,5 +36,5 @@ except ImportError: # This import requires the `async_rest` extra. # Don't raise an exception if `OperationsRestAsyncTransport` cannot be imported - # as other transports are still available + # as other transports are still available. pass diff --git a/google/api_core/operations_v1/abstract_operations_async_client.py b/google/api_core/operations_v1/abstract_operations_async_client.py index ac34941a..5ca6a2f9 100644 --- a/google/api_core/operations_v1/abstract_operations_async_client.py +++ b/google/api_core/operations_v1/abstract_operations_async_client.py @@ -25,18 +25,16 @@ from google.api_core.operations_v1.abstract_operations_base_client import ( AbstractOperationsBaseClient, ) +from google.longrunning import operations_pb2 try: from google.auth.aio import credentials as ga_credentials # type: ignore except ImportError as e: # pragma: NO COVER raise ImportError( - "`google-api-core[async_rest]` is required to use asynchronous rest streaming. " - "Install the `async_rest` extra of `google-api-core` using " + "The `async_rest` extra of `google-api-core` is required to use long-running operations. Install it by running " "`pip install google-api-core[async_rest]`." ) from e -from google.longrunning import operations_pb2 - class AbstractOperationsAsyncClient(AbstractOperationsBaseClient): """Manages long-running operations with an API service for the asynchronous client. diff --git a/tests/unit/operations_v1/test_operations_rest_client.py b/tests/unit/operations_v1/test_operations_rest_client.py index 955b1d1b..72d4f0ec 100644 --- a/tests/unit/operations_v1/test_operations_rest_client.py +++ b/tests/unit/operations_v1/test_operations_rest_client.py @@ -17,6 +17,7 @@ import mock import pytest +from typing import Any try: import grpc # noqa: F401 @@ -52,25 +53,6 @@ except ImportError: GOOGLE_AUTH_AIO_INSTALLED = False -if GOOGLE_AUTH_AIO_INSTALLED: - TEST_TRANSPORT_CREDS_PARAMS = [ - ( - transports.OperationsRestTransport, - ga_credentials.AnonymousCredentials(), - ), - ( - transports.OperationsRestAsyncTransport, - ga_credentials_async.AnonymousCredentials(), - ), - ] -else: - TEST_TRANSPORT_CREDS_PARAMS = [ - ( - transports.OperationsRestTransport, - ga_credentials.AnonymousCredentials(), - ) - ] - HTTP_OPTIONS = { "google.longrunning.Operations.CancelOperation": [ {"method": "post", "uri": "/v3/{name=operations/*}:cancel", "body": "*"}, @@ -86,43 +68,36 @@ ], } +CLIENTS: list[Any] = [ + AbstractOperationsClient, +] +CLIENTS_WITH_TRANSPORT = [ + [AbstractOperationsClient, transports.OperationsRestTransport, "rest"], +] +CLIENTS_WITH_CREDENTIALS = [ + [ + AbstractOperationsClient, + transports.OperationsRestTransport, + ga_credentials.AnonymousCredentials(), + ], +] if GOOGLE_AUTH_AIO_INSTALLED: - CLIENTS = [AbstractOperationsClient, AbstractOperationsAsyncClient] - CLIENTS_WITH_TRANSPORT = [ - (AbstractOperationsClient, transports.OperationsRestTransport, "rest"), - ( + CLIENTS.append(AbstractOperationsAsyncClient) + CLIENTS_WITH_TRANSPORT.append( + [ AbstractOperationsAsyncClient, transports.OperationsRestAsyncTransport, "rest_asyncio", - ), - ] - CLIENTS_WITH_CREDENTIALS = [ - ( - AbstractOperationsClient, - transports.OperationsRestTransport, - ga_credentials.AnonymousCredentials(), - ), - ( + ] + ) + CLIENTS_WITH_CREDENTIALS.append( + [ AbstractOperationsAsyncClient, transports.OperationsRestAsyncTransport, ga_credentials_async.AnonymousCredentials(), - ), - ] -else: - CLIENTS = [ - AbstractOperationsClient, - ] - CLIENTS_WITH_TRANSPORT = [ - (AbstractOperationsClient, transports.OperationsRestTransport, "rest"), - ] - CLIENTS_WITH_CREDENTIALS = [ - ( - AbstractOperationsClient, - transports.OperationsRestTransport, - ga_credentials.AnonymousCredentials(), - ), - ] + ] + ) def client_cert_source_callback(): @@ -1105,10 +1080,10 @@ def test_transport_instance(client_class, transport_class, credentials): @pytest.mark.parametrize( - "transport_class,credentials", - TEST_TRANSPORT_CREDS_PARAMS, + "client_class,transport_class,credentials", + CLIENTS_WITH_CREDENTIALS, ) -def test_transport_adc(transport_class, credentials): +def test_transport_adc(client_class, transport_class, credentials): # Test default credentials are used if not provided. with mock.patch.object(google.auth, "default") as adc: adc.return_value = (credentials, None) From 1913c76e84d55aace82b0de184ebe87349039ede Mon Sep 17 00:00:00 2001 From: ohmayr Date: Mon, 7 Oct 2024 02:37:20 +0000 Subject: [PATCH 10/14] address PR comments --- .../operations_v1/abstract_operations_async_client.py | 10 +++++++++- .../api_core/operations_v1/transports/rest_asyncio.py | 8 ++++++++ 2 files changed, 17 insertions(+), 1 deletion(-) diff --git a/google/api_core/operations_v1/abstract_operations_async_client.py b/google/api_core/operations_v1/abstract_operations_async_client.py index 5ca6a2f9..873597b0 100644 --- a/google/api_core/operations_v1/abstract_operations_async_client.py +++ b/google/api_core/operations_v1/abstract_operations_async_client.py @@ -97,7 +97,7 @@ def __init__( super().__init__( credentials=credentials, # type: ignore # NOTE: If a transport is not provided, we force the client to use the async - # REST transport, as it should. + # REST transport. transport=transport or "rest_asyncio", client_options=client_options, client_info=client_info, @@ -112,6 +112,8 @@ async def list_operations( page_token: Optional[str] = None, timeout: Optional[float] = None, metadata: Sequence[Tuple[str, str]] = (), + # TODO(https://github.com/googleapis/python-api-core/issues/722): Add `retry` parameter + # to allow configuring retryable error codes. ) -> pagers.ListOperationsAsyncPager: r"""Lists operations that match the specified filter in the request. If the server doesn't support this method, it returns @@ -190,6 +192,8 @@ async def get_operation( *, timeout: Optional[float] = None, metadata: Sequence[Tuple[str, str]] = (), + # TODO(https://github.com/googleapis/python-api-core/issues/722): Add `retry` parameter + # to allow configuring retryable error codes. ) -> operations_pb2.Operation: r"""Gets the latest state of a long-running operation. Clients can use this method to poll the operation result @@ -238,6 +242,8 @@ async def delete_operation( *, timeout: Optional[float] = None, metadata: Sequence[Tuple[str, str]] = (), + # TODO(https://github.com/googleapis/python-api-core/issues/722): Add `retry` parameter + # to allow configuring retryable error codes. ) -> None: r"""Deletes a long-running operation. This method indicates that the client is no longer interested in the operation result. It does @@ -282,6 +288,8 @@ async def cancel_operation( *, timeout: Optional[float] = None, metadata: Sequence[Tuple[str, str]] = (), + # TODO(https://github.com/googleapis/python-api-core/issues/722): Add `retry` parameter + # to allow configuring retryable error codes. ) -> None: r"""Starts asynchronous cancellation on a long-running operation. The server makes a best effort to cancel the operation, but diff --git a/google/api_core/operations_v1/transports/rest_asyncio.py b/google/api_core/operations_v1/transports/rest_asyncio.py index 1b5dadbb..52aeb881 100644 --- a/google/api_core/operations_v1/transports/rest_asyncio.py +++ b/google/api_core/operations_v1/transports/rest_asyncio.py @@ -197,6 +197,8 @@ async def _list_operations( *, timeout: Optional[float] = None, metadata: Sequence[Tuple[str, str]] = (), + # TODO(https://github.com/googleapis/python-api-core/issues/722): Add `retry` parameter + # to allow configuring retryable error codes. ) -> operations_pb2.ListOperationsResponse: r"""Asynchronously call the list operations method over HTTP. @@ -271,6 +273,8 @@ async def _get_operation( *, timeout: Optional[float] = None, metadata: Sequence[Tuple[str, str]] = (), + # TODO(https://github.com/googleapis/python-api-core/issues/722): Add `retry` parameter + # to allow configuring retryable error codes. ) -> operations_pb2.Operation: r"""Asynchronously call the get operation method over HTTP. @@ -346,6 +350,8 @@ async def _delete_operation( *, timeout: Optional[float] = None, metadata: Sequence[Tuple[str, str]] = (), + # TODO(https://github.com/googleapis/python-api-core/issues/722): Add `retry` parameter + # to allow configuring retryable error codes. ) -> empty_pb2.Empty: r"""Asynchronously call the delete operation method over HTTP. @@ -414,6 +420,8 @@ async def _cancel_operation( *, timeout: Optional[float] = None, metadata: Sequence[Tuple[str, str]] = (), + # TODO(https://github.com/googleapis/python-api-core/issues/722): Add `retry` parameter + # to allow configuring retryable error codes. ) -> empty_pb2.Empty: r"""Asynchronously call the cancel operation method over HTTP. From 80e27e4d10ede46ee3feedbc85e415a29bef34c1 Mon Sep 17 00:00:00 2001 From: ohmayr Date: Mon, 7 Oct 2024 04:18:36 +0000 Subject: [PATCH 11/14] fix unit tests and address more comments --- .../abstract_operations_async_client.py | 3 +- .../abstract_operations_base_client.py | 49 ++++--------------- .../operations_v1/transports/rest_asyncio.py | 35 +++++++++++-- .../test_operations_rest_client.py | 4 +- 4 files changed, 43 insertions(+), 48 deletions(-) diff --git a/google/api_core/operations_v1/abstract_operations_async_client.py b/google/api_core/operations_v1/abstract_operations_async_client.py index 873597b0..8e180c81 100644 --- a/google/api_core/operations_v1/abstract_operations_async_client.py +++ b/google/api_core/operations_v1/abstract_operations_async_client.py @@ -66,8 +66,7 @@ def __init__( are specified, the client will attempt to ascertain the credentials from the environment. transport (Union[str, OperationsTransport]): The - transport to use. If set to None, a transport is chosen - automatically. + transport to use. If set to None, this defaults to 'rest_asyncio'. client_options (google.api_core.client_options.ClientOptions): Custom options for the client. It won't take effect if a ``transport`` instance is provided. (1) The ``api_endpoint`` property can be used to override the diff --git a/google/api_core/operations_v1/abstract_operations_base_client.py b/google/api_core/operations_v1/abstract_operations_base_client.py index 5db9adf8..3304a73a 100644 --- a/google/api_core/operations_v1/abstract_operations_base_client.py +++ b/google/api_core/operations_v1/abstract_operations_base_client.py @@ -19,7 +19,6 @@ from typing import Dict, Optional, Type, Union from google.api_core import client_options as client_options_lib # type: ignore -from google.api_core import exceptions as core_exceptions # type: ignore from google.api_core import gapic_v1 # type: ignore from google.api_core.operations_v1.transports.base import ( DEFAULT_CLIENT_INFO, @@ -359,41 +358,13 @@ def __init__( self._transport = transport else: Transport = type(self).get_transport_class(transport) - # NOTE: The conditional logic below to initialize the transport can be removed - # once we have feature parity with the sync transport. - if "async" in str(Transport).lower(): - unsupported_params = { - # TODO(https://github.com/googleapis/python-api-core/issues/715): Add support for `credentials_file` to async REST transport. - "google.api_core.client_options.ClientOptions.credentials_file": client_options.credentials_file, - # TODO(https://github.com/googleapis/python-api-core/issues/716): Add support for `scopes` to async REST transport. - "google.api_core.client_options.ClientOptions.scopes": client_options.scopes, - # TODO(https://github.com/googleapis/python-api-core/issues/717): Add support for `quota_project_id` to async REST transport. - "google.api_core.client_options.ClientOptions.quota_project_id": client_options.quota_project_id, - # TODO(https://github.com/googleapis/python-api-core/issues/718): Add support for `client_cert_source` to async REST transport. - "google.api_core.client_options.ClientOptions.client_cert_source": client_options.client_cert_source, - } - provided_unsupported_params = [ - name - for name, value in unsupported_params.items() - if value is not None - ] - if provided_unsupported_params: - raise core_exceptions.AsyncRestUnsupportedParameterError( - f"The following provided parameters are not supported for `transport=rest_asyncio`: {', '.join(provided_unsupported_params)}" - ) - self._transport = Transport( - credentials=credentials, - host=api_endpoint, - client_info=client_info, - ) - else: - self._transport = Transport( - credentials=credentials, - credentials_file=client_options.credentials_file, - host=api_endpoint, - scopes=client_options.scopes, - client_cert_source_for_mtls=client_cert_source_func, - quota_project_id=client_options.quota_project_id, - client_info=client_info, - always_use_jwt_access=True, - ) + self._transport = Transport( + credentials=credentials, + credentials_file=client_options.credentials_file, + host=api_endpoint, + scopes=client_options.scopes, + client_cert_source_for_mtls=client_cert_source_func, + quota_project_id=client_options.quota_project_id, + client_info=client_info, + always_use_jwt_access=True, + ) diff --git a/google/api_core/operations_v1/transports/rest_asyncio.py b/google/api_core/operations_v1/transports/rest_asyncio.py index 52aeb881..8fdce84a 100644 --- a/google/api_core/operations_v1/transports/rest_asyncio.py +++ b/google/api_core/operations_v1/transports/rest_asyncio.py @@ -72,15 +72,19 @@ def __init__( *, host: str = "longrunning.googleapis.com", credentials: Optional[ga_credentials_async.Credentials] = None, + credentials_file: Optional[str] = None, + scopes: Optional[Sequence[str]] = None, + client_cert_source_for_mtls: Optional[Callable[[], Tuple[bytes, bytes]]] = None, + quota_project_id: Optional[str] = None, client_info: gapic_v1.client_info.ClientInfo = DEFAULT_CLIENT_INFO, always_use_jwt_access: Optional[bool] = False, url_scheme: str = "https", http_options: Optional[Dict] = None, path_prefix: str = "v1", - # TODO(https://github.com/googleapis/python-api-core/issues/715): Add support for `credentials_file` to async REST transport. - # TODO(https://github.com/googleapis/python-api-core/issues/716): Add support for `scopes` to async REST transport. - # TODO(https://github.com/googleapis/python-api-core/issues/717): Add support for `quota_project_id` to async REST transport. - # TODO(https://github.com/googleapis/python-api-core/issues/718): Add support for `client_cert_source` to async REST transport. + # TODO(https://github.com/googleapis/python-api-core/issues/715): Add docstring for `credentials_file` to async REST transport. + # TODO(https://github.com/googleapis/python-api-core/issues/716): Add docstring for `scopes` to async REST transport. + # TODO(https://github.com/googleapis/python-api-core/issues/717): Add docstring for `quota_project_id` to async REST transport. + # TODO(https://github.com/googleapis/python-api-core/issues/718): Add docstring for `client_cert_source` to async REST transport. ) -> None: """Instantiate the transport. @@ -109,12 +113,33 @@ def __init__( "v1" by default. """ + unsupported_params = { + # TODO(https://github.com/googleapis/python-api-core/issues/715): Add support for `credentials_file` to async REST transport. + "google.api_core.client_options.ClientOptions.credentials_file": credentials_file, + # TODO(https://github.com/googleapis/python-api-core/issues/716): Add support for `scopes` to async REST transport. + "google.api_core.client_options.ClientOptions.scopes": scopes, + # TODO(https://github.com/googleapis/python-api-core/issues/717): Add support for `quota_project_id` to async REST transport. + "google.api_core.client_options.ClientOptions.quota_project_id": quota_project_id, + # TODO(https://github.com/googleapis/python-api-core/issues/718): Add support for `client_cert_source` to async REST transport. + "google.api_core.client_options.ClientOptions.client_cert_source": client_cert_source_for_mtls, + # TODO(https://github.com/googleapis/python-api-core/issues/718): Add support for `client_cert_source` to async REST transport. + "google.api_core.client_options.ClientOptions.client_cert_source": client_cert_source_for_mtls, + } + provided_unsupported_params = [ + name for name, value in unsupported_params.items() if value is not None + ] + if provided_unsupported_params: + raise core_exceptions.AsyncRestUnsupportedParameterError( + f"The following provided parameters are not supported for `transport=rest_asyncio`: {', '.join(provided_unsupported_params)}" + ) + super().__init__( host=host, # TODO(https://github.com/googleapis/python-api-core/issues/709): Remove `type: ignore` when the linked issue is resolved. credentials=credentials, # type: ignore client_info=client_info, - always_use_jwt_access=always_use_jwt_access, + # TODO(https://github.com/googleapis/python-api-core/issues/725): Set always_use_jwt_access token when supported. + always_use_jwt_access=False, ) # TODO(https://github.com/googleapis/python-api-core/issues/708): add support for # `default_host` in AsyncAuthorizedSession for feature parity with the synchronous diff --git a/tests/unit/operations_v1/test_operations_rest_client.py b/tests/unit/operations_v1/test_operations_rest_client.py index 72d4f0ec..3fe4e70d 100644 --- a/tests/unit/operations_v1/test_operations_rest_client.py +++ b/tests/unit/operations_v1/test_operations_rest_client.py @@ -17,7 +17,7 @@ import mock import pytest -from typing import Any +from typing import Any, List try: import grpc # noqa: F401 @@ -68,7 +68,7 @@ ], } -CLIENTS: list[Any] = [ +CLIENTS: List[Any] = [ AbstractOperationsClient, ] CLIENTS_WITH_TRANSPORT = [ From dbf4abd53251efa3d8c948a97efc1dda31ed2798 Mon Sep 17 00:00:00 2001 From: ohmayr Date: Mon, 7 Oct 2024 07:11:22 +0000 Subject: [PATCH 12/14] disable retry parameter --- google/api_core/operation_async.py | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/google/api_core/operation_async.py b/google/api_core/operation_async.py index 2fd341d9..4d17da1e 100644 --- a/google/api_core/operation_async.py +++ b/google/api_core/operation_async.py @@ -151,7 +151,9 @@ async def _refresh_and_update(self, retry=async_future.DEFAULT_RETRY): # If the currently cached operation is done, no need to make another # RPC as it will not change once done. if not self._operation.done: - self._operation = await self._refresh(retry=retry) + # NOTE: Temporary change to not pass down retry until we investigate + # if it is used by async gRPC. It is not supported in async REST. + self._operation = await self._refresh() self._set_result_from_operation() async def done(self, retry=async_future.DEFAULT_RETRY): From 53591d8779ac5cf7e8db1e10534db9faae987837 Mon Sep 17 00:00:00 2001 From: ohmayr Date: Mon, 7 Oct 2024 15:34:44 +0000 Subject: [PATCH 13/14] add retry parameter --- google/api_core/operation_async.py | 4 +- .../abstract_operations_async_client.py | 114 +++++++++--------- 2 files changed, 61 insertions(+), 57 deletions(-) diff --git a/google/api_core/operation_async.py b/google/api_core/operation_async.py index 4d17da1e..2fd341d9 100644 --- a/google/api_core/operation_async.py +++ b/google/api_core/operation_async.py @@ -151,9 +151,7 @@ async def _refresh_and_update(self, retry=async_future.DEFAULT_RETRY): # If the currently cached operation is done, no need to make another # RPC as it will not change once done. if not self._operation.done: - # NOTE: Temporary change to not pass down retry until we investigate - # if it is used by async gRPC. It is not supported in async REST. - self._operation = await self._refresh() + self._operation = await self._refresh(retry=retry) self._set_result_from_operation() async def done(self, retry=async_future.DEFAULT_RETRY): diff --git a/google/api_core/operations_v1/abstract_operations_async_client.py b/google/api_core/operations_v1/abstract_operations_async_client.py index 8e180c81..49e39ba1 100644 --- a/google/api_core/operations_v1/abstract_operations_async_client.py +++ b/google/api_core/operations_v1/abstract_operations_async_client.py @@ -101,18 +101,70 @@ def __init__( client_options=client_options, client_info=client_info, ) + + async def get_operation( + self, + name: str, + # TODO(https://github.com/googleapis/python-api-core/issues/722): Leverage `retry` + # to allow configuring retryable error codes. + retry=gapic_v1.method_async.DEFAULT, + *, + timeout: Optional[float] = None, + metadata: Sequence[Tuple[str, str]] = (), + ) -> operations_pb2.Operation: + r"""Gets the latest state of a long-running operation. + Clients can use this method to poll the operation result + at intervals as recommended by the API service. + + Args: + name (str): + The name of the operation resource. + timeout (float): The timeout for this request. + metadata (Sequence[Tuple[str, str]]): Strings which should be + sent along with the request as metadata. + + Returns: + google.longrunning.operations_pb2.Operation: + This resource represents a long- + running operation that is the result of a + network API call. + + """ + + request = operations_pb2.GetOperationRequest(name=name) + + # Wrap the RPC method; this adds retry and timeout information, + # and friendly error handling. + rpc = self._transport._wrapped_methods[self._transport.get_operation] + + # Certain fields should be provided within the metadata header; + # add these here. + metadata = tuple(metadata or ()) + ( + gapic_v1.routing_header.to_grpc_metadata((("name", request.name),)), + ) + + # Send the request. + response = await rpc( + request, + timeout=timeout, + metadata=metadata, + ) + + # Done; return the response. + return response async def list_operations( self, name: str, filter_: Optional[str] = None, + # TODO(https://github.com/googleapis/python-api-core/issues/722): Leverage `retry` + # to allow configuring retryable error codes. + retry=gapic_v1.method_async.DEFAULT, *, page_size: Optional[int] = None, page_token: Optional[str] = None, timeout: Optional[float] = None, metadata: Sequence[Tuple[str, str]] = (), - # TODO(https://github.com/googleapis/python-api-core/issues/722): Add `retry` parameter - # to allow configuring retryable error codes. ) -> pagers.ListOperationsAsyncPager: r"""Lists operations that match the specified filter in the request. If the server doesn't support this method, it returns @@ -185,64 +237,15 @@ async def list_operations( # Done; return the response. return response - async def get_operation( - self, - name: str, - *, - timeout: Optional[float] = None, - metadata: Sequence[Tuple[str, str]] = (), - # TODO(https://github.com/googleapis/python-api-core/issues/722): Add `retry` parameter - # to allow configuring retryable error codes. - ) -> operations_pb2.Operation: - r"""Gets the latest state of a long-running operation. - Clients can use this method to poll the operation result - at intervals as recommended by the API service. - - Args: - name (str): - The name of the operation resource. - timeout (float): The timeout for this request. - metadata (Sequence[Tuple[str, str]]): Strings which should be - sent along with the request as metadata. - - Returns: - google.longrunning.operations_pb2.Operation: - This resource represents a long- - running operation that is the result of a - network API call. - - """ - - request = operations_pb2.GetOperationRequest(name=name) - - # Wrap the RPC method; this adds retry and timeout information, - # and friendly error handling. - rpc = self._transport._wrapped_methods[self._transport.get_operation] - - # Certain fields should be provided within the metadata header; - # add these here. - metadata = tuple(metadata or ()) + ( - gapic_v1.routing_header.to_grpc_metadata((("name", request.name),)), - ) - - # Send the request. - response = await rpc( - request, - timeout=timeout, - metadata=metadata, - ) - - # Done; return the response. - return response - async def delete_operation( self, name: str, + # TODO(https://github.com/googleapis/python-api-core/issues/722): Leverage `retry` + # to allow configuring retryable error codes. + retry=gapic_v1.method_async.DEFAULT, *, timeout: Optional[float] = None, metadata: Sequence[Tuple[str, str]] = (), - # TODO(https://github.com/googleapis/python-api-core/issues/722): Add `retry` parameter - # to allow configuring retryable error codes. ) -> None: r"""Deletes a long-running operation. This method indicates that the client is no longer interested in the operation result. It does @@ -284,6 +287,9 @@ async def delete_operation( async def cancel_operation( self, name: Optional[str] = None, + # TODO(https://github.com/googleapis/python-api-core/issues/722): Leverage `retry` + # to allow configuring retryable error codes. + retry=gapic_v1.method_async.DEFAULT, *, timeout: Optional[float] = None, metadata: Sequence[Tuple[str, str]] = (), From c0daf3d06ea640d506112c89bd2243e4f422724c Mon Sep 17 00:00:00 2001 From: ohmayr Date: Mon, 7 Oct 2024 20:47:59 +0000 Subject: [PATCH 14/14] address PR comments --- google/api_core/operations_v1/__init__.py | 8 +- .../abstract_operations_base_client.py | 4 +- ...ent.py => operations_rest_client_async.py} | 22 +++--- .../operations_v1/transports/__init__.py | 8 +- .../api_core/operations_v1/transports/rest.py | 8 ++ .../operations_v1/transports/rest_asyncio.py | 22 ++++-- .../test_operations_rest_client.py | 76 +++++++++---------- 7 files changed, 82 insertions(+), 66 deletions(-) rename google/api_core/operations_v1/{abstract_operations_async_client.py => operations_rest_client_async.py} (97%) diff --git a/google/api_core/operations_v1/__init__.py b/google/api_core/operations_v1/__init__.py index 3dbc9217..4db32a4c 100644 --- a/google/api_core/operations_v1/__init__.py +++ b/google/api_core/operations_v1/__init__.py @@ -28,13 +28,13 @@ try: from google.api_core.operations_v1.transports.rest_asyncio import ( - OperationsRestAsyncTransport, + AsyncOperationsRestTransport, ) - from google.api_core.operations_v1.abstract_operations_async_client import AbstractOperationsAsyncClient + from google.api_core.operations_v1.operations_rest_client_async import AsyncOperationsRestClient - __all__ += ["AbstractOperationsAsyncClient", "OperationsRestAsyncTransport"] + __all__ += ["AsyncOperationsRestClient", "AsyncOperationsRestTransport"] except ImportError: # This import requires the `async_rest` extra. - # Don't raise an exception if `OperationsRestAsyncTransport` cannot be imported + # Don't raise an exception if `AsyncOperationsRestTransport` cannot be imported # as other transports are still available. pass diff --git a/google/api_core/operations_v1/abstract_operations_base_client.py b/google/api_core/operations_v1/abstract_operations_base_client.py index 3304a73a..160c2a88 100644 --- a/google/api_core/operations_v1/abstract_operations_base_client.py +++ b/google/api_core/operations_v1/abstract_operations_base_client.py @@ -28,7 +28,7 @@ try: from google.api_core.operations_v1.transports.rest_asyncio import ( - OperationsRestAsyncTransport, + AsyncOperationsRestTransport, ) HAS_ASYNC_REST_DEPENDENCIES = True @@ -52,7 +52,7 @@ class AbstractOperationsBaseClientMeta(type): _transport_registry = OrderedDict() # type: Dict[str, Type[OperationsTransport]] _transport_registry["rest"] = OperationsRestTransport if HAS_ASYNC_REST_DEPENDENCIES: - _transport_registry["rest_asyncio"] = OperationsRestAsyncTransport + _transport_registry["rest_asyncio"] = AsyncOperationsRestTransport def get_transport_class( cls, diff --git a/google/api_core/operations_v1/abstract_operations_async_client.py b/google/api_core/operations_v1/operations_rest_client_async.py similarity index 97% rename from google/api_core/operations_v1/abstract_operations_async_client.py rename to google/api_core/operations_v1/operations_rest_client_async.py index 49e39ba1..fbe029b0 100644 --- a/google/api_core/operations_v1/abstract_operations_async_client.py +++ b/google/api_core/operations_v1/operations_rest_client_async.py @@ -36,8 +36,8 @@ ) from e -class AbstractOperationsAsyncClient(AbstractOperationsBaseClient): - """Manages long-running operations with an API service for the asynchronous client. +class AsyncOperationsRestClient(AbstractOperationsBaseClient): + """Manages long-running operations with a REST API service for the asynchronous client. When an API method normally takes long time to complete, it can be designed to return [Operation][google.api_core.operations_v1.Operation] to the @@ -105,10 +105,10 @@ def __init__( async def get_operation( self, name: str, + *, # TODO(https://github.com/googleapis/python-api-core/issues/722): Leverage `retry` # to allow configuring retryable error codes. retry=gapic_v1.method_async.DEFAULT, - *, timeout: Optional[float] = None, metadata: Sequence[Tuple[str, str]] = (), ) -> operations_pb2.Operation: @@ -146,6 +146,7 @@ async def get_operation( # Send the request. response = await rpc( request, + retry=retry, timeout=timeout, metadata=metadata, ) @@ -157,12 +158,12 @@ async def list_operations( self, name: str, filter_: Optional[str] = None, - # TODO(https://github.com/googleapis/python-api-core/issues/722): Leverage `retry` - # to allow configuring retryable error codes. - retry=gapic_v1.method_async.DEFAULT, *, page_size: Optional[int] = None, page_token: Optional[str] = None, + # TODO(https://github.com/googleapis/python-api-core/issues/722): Leverage `retry` + # to allow configuring retryable error codes. + retry=gapic_v1.method_async.DEFAULT, timeout: Optional[float] = None, metadata: Sequence[Tuple[str, str]] = (), ) -> pagers.ListOperationsAsyncPager: @@ -221,6 +222,7 @@ async def list_operations( # Send the request. response = await rpc( request, + retry=retry, timeout=timeout, metadata=metadata, ) @@ -240,10 +242,10 @@ async def list_operations( async def delete_operation( self, name: str, + *, # TODO(https://github.com/googleapis/python-api-core/issues/722): Leverage `retry` # to allow configuring retryable error codes. retry=gapic_v1.method_async.DEFAULT, - *, timeout: Optional[float] = None, metadata: Sequence[Tuple[str, str]] = (), ) -> None: @@ -280,6 +282,7 @@ async def delete_operation( # Send the request. await rpc( request, + retry=retry, timeout=timeout, metadata=metadata, ) @@ -287,14 +290,12 @@ async def delete_operation( async def cancel_operation( self, name: Optional[str] = None, + *, # TODO(https://github.com/googleapis/python-api-core/issues/722): Leverage `retry` # to allow configuring retryable error codes. retry=gapic_v1.method_async.DEFAULT, - *, timeout: Optional[float] = None, metadata: Sequence[Tuple[str, str]] = (), - # TODO(https://github.com/googleapis/python-api-core/issues/722): Add `retry` parameter - # to allow configuring retryable error codes. ) -> None: r"""Starts asynchronous cancellation on a long-running operation. The server makes a best effort to cancel the operation, but @@ -338,6 +339,7 @@ async def cancel_operation( # Send the request. await rpc( request, + retry=retry, timeout=timeout, metadata=metadata, ) diff --git a/google/api_core/operations_v1/transports/__init__.py b/google/api_core/operations_v1/transports/__init__.py index 2ae79e4e..8c24ce6e 100644 --- a/google/api_core/operations_v1/transports/__init__.py +++ b/google/api_core/operations_v1/transports/__init__.py @@ -26,14 +26,14 @@ __all__: Tuple[str, ...] = ("OperationsTransport", "OperationsRestTransport") try: - from .rest_asyncio import OperationsRestAsyncTransport + from .rest_asyncio import AsyncOperationsRestTransport - __all__ += ("OperationsRestAsyncTransport",) + __all__ += ("AsyncOperationsRestTransport",) _transport_registry["rest_asyncio"] = cast( - OperationsTransport, OperationsRestAsyncTransport + OperationsTransport, AsyncOperationsRestTransport ) except ImportError: # This import requires the `async_rest` extra. - # Don't raise an exception if `OperationsRestAsyncTransport` cannot be imported + # Don't raise an exception if `AsyncOperationsRestTransport` cannot be imported # as other transports are still available. pass diff --git a/google/api_core/operations_v1/transports/rest.py b/google/api_core/operations_v1/transports/rest.py index b0599baa..766a6685 100644 --- a/google/api_core/operations_v1/transports/rest.py +++ b/google/api_core/operations_v1/transports/rest.py @@ -142,6 +142,8 @@ def _list_operations( self, request: operations_pb2.ListOperationsRequest, *, + # TODO(https://github.com/googleapis/python-api-core/issues/723): Leverage `retry` + # to allow configuring retryable error codes. retry: OptionalRetry = gapic_v1.method.DEFAULT, timeout: Optional[float] = None, compression: Optional[grpc.Compression] = gapic_v1.method.DEFAULT, @@ -218,6 +220,8 @@ def _get_operation( self, request: operations_pb2.GetOperationRequest, *, + # TODO(https://github.com/googleapis/python-api-core/issues/723): Leverage `retry` + # to allow configuring retryable error codes. retry: OptionalRetry = gapic_v1.method.DEFAULT, timeout: Optional[float] = None, compression: Optional[grpc.Compression] = gapic_v1.method.DEFAULT, @@ -295,6 +299,8 @@ def _delete_operation( self, request: operations_pb2.DeleteOperationRequest, *, + # TODO(https://github.com/googleapis/python-api-core/issues/723): Leverage `retry` + # to allow configuring retryable error codes. retry: OptionalRetry = gapic_v1.method.DEFAULT, timeout: Optional[float] = None, compression: Optional[grpc.Compression] = gapic_v1.method.DEFAULT, @@ -362,6 +368,8 @@ def _cancel_operation( self, request: operations_pb2.CancelOperationRequest, *, + # TODO(https://github.com/googleapis/python-api-core/issues/723): Leverage `retry` + # to allow configuring retryable error codes. retry: OptionalRetry = gapic_v1.method.DEFAULT, timeout: Optional[float] = None, compression: Optional[grpc.Compression] = gapic_v1.method.DEFAULT, diff --git a/google/api_core/operations_v1/transports/rest_asyncio.py b/google/api_core/operations_v1/transports/rest_asyncio.py index 8fdce84a..71c20eb8 100644 --- a/google/api_core/operations_v1/transports/rest_asyncio.py +++ b/google/api_core/operations_v1/transports/rest_asyncio.py @@ -46,7 +46,7 @@ ) -class OperationsRestAsyncTransport(OperationsTransport): +class AsyncOperationsRestTransport(OperationsTransport): """Asynchronous REST backend transport for Operations. Manages async long-running operations with an API service. @@ -220,10 +220,11 @@ async def _list_operations( self, request: operations_pb2.ListOperationsRequest, *, + # TODO(https://github.com/googleapis/python-api-core/issues/722): Leverage `retry` + # to allow configuring retryable error codes. + retry=gapic_v1.method_async.DEFAULT, timeout: Optional[float] = None, metadata: Sequence[Tuple[str, str]] = (), - # TODO(https://github.com/googleapis/python-api-core/issues/722): Add `retry` parameter - # to allow configuring retryable error codes. ) -> operations_pb2.ListOperationsResponse: r"""Asynchronously call the list operations method over HTTP. @@ -296,10 +297,11 @@ async def _get_operation( self, request: operations_pb2.GetOperationRequest, *, + # TODO(https://github.com/googleapis/python-api-core/issues/722): Leverage `retry` + # to allow configuring retryable error codes. + retry=gapic_v1.method_async.DEFAULT, timeout: Optional[float] = None, metadata: Sequence[Tuple[str, str]] = (), - # TODO(https://github.com/googleapis/python-api-core/issues/722): Add `retry` parameter - # to allow configuring retryable error codes. ) -> operations_pb2.Operation: r"""Asynchronously call the get operation method over HTTP. @@ -373,10 +375,11 @@ async def _delete_operation( self, request: operations_pb2.DeleteOperationRequest, *, + # TODO(https://github.com/googleapis/python-api-core/issues/722): Leverage `retry` + # to allow configuring retryable error codes. + retry=gapic_v1.method_async.DEFAULT, timeout: Optional[float] = None, metadata: Sequence[Tuple[str, str]] = (), - # TODO(https://github.com/googleapis/python-api-core/issues/722): Add `retry` parameter - # to allow configuring retryable error codes. ) -> empty_pb2.Empty: r"""Asynchronously call the delete operation method over HTTP. @@ -443,6 +446,9 @@ async def _cancel_operation( self, request: operations_pb2.CancelOperationRequest, *, + # TODO(https://github.com/googleapis/python-api-core/issues/722): Leverage `retry` + # to allow configuring retryable error codes. + retry=gapic_v1.method_async.DEFAULT, timeout: Optional[float] = None, metadata: Sequence[Tuple[str, str]] = (), # TODO(https://github.com/googleapis/python-api-core/issues/722): Add `retry` parameter @@ -551,4 +557,4 @@ def cancel_operation( return self._cancel_operation -__all__ = ("OperationsRestAsyncTransport",) +__all__ = ("AsyncOperationsRestTransport",) diff --git a/tests/unit/operations_v1/test_operations_rest_client.py b/tests/unit/operations_v1/test_operations_rest_client.py index 3fe4e70d..644cf266 100644 --- a/tests/unit/operations_v1/test_operations_rest_client.py +++ b/tests/unit/operations_v1/test_operations_rest_client.py @@ -46,7 +46,7 @@ import aiohttp # noqa: F401 import google.auth.aio.transport from google.auth.aio.transport.sessions import AsyncAuthorizedSession - from google.api_core.operations_v1 import AbstractOperationsAsyncClient + from google.api_core.operations_v1 import AsyncOperationsRestClient from google.auth.aio import credentials as ga_credentials_async GOOGLE_AUTH_AIO_INSTALLED = True @@ -68,13 +68,13 @@ ], } -CLIENTS: List[Any] = [ +PYPARAM_CLIENT: List[Any] = [ AbstractOperationsClient, ] -CLIENTS_WITH_TRANSPORT = [ +PYPARAM_CLIENT_TRANSPORT_NAME = [ [AbstractOperationsClient, transports.OperationsRestTransport, "rest"], ] -CLIENTS_WITH_CREDENTIALS = [ +PYPARAM_CLIENT_TRANSPORT_CREDENTIALS = [ [ AbstractOperationsClient, transports.OperationsRestTransport, @@ -83,18 +83,18 @@ ] if GOOGLE_AUTH_AIO_INSTALLED: - CLIENTS.append(AbstractOperationsAsyncClient) - CLIENTS_WITH_TRANSPORT.append( + PYPARAM_CLIENT.append(AsyncOperationsRestClient) + PYPARAM_CLIENT_TRANSPORT_NAME.append( [ - AbstractOperationsAsyncClient, - transports.OperationsRestAsyncTransport, + AsyncOperationsRestClient, + transports.AsyncOperationsRestTransport, "rest_asyncio", ] ) - CLIENTS_WITH_CREDENTIALS.append( + PYPARAM_CLIENT_TRANSPORT_CREDENTIALS.append( [ - AbstractOperationsAsyncClient, - transports.OperationsRestAsyncTransport, + AsyncOperationsRestClient, + transports.AsyncOperationsRestTransport, ga_credentials_async.AnonymousCredentials(), ] ) @@ -114,11 +114,11 @@ def _get_session_type(is_async: bool): def _get_operations_client(is_async: bool, http_options=HTTP_OPTIONS): if is_async and GOOGLE_AUTH_AIO_INSTALLED: - async_transport = transports.rest_asyncio.OperationsRestAsyncTransport( + async_transport = transports.rest_asyncio.AsyncOperationsRestTransport( credentials=ga_credentials_async.AnonymousCredentials(), http_options=http_options, ) - return AbstractOperationsAsyncClient(transport=async_transport) + return AsyncOperationsRestClient(transport=async_transport) else: sync_transport = transports.rest.OperationsRestTransport( credentials=ga_credentials.AnonymousCredentials(), http_options=http_options @@ -169,7 +169,7 @@ def test__get_default_mtls_endpoint(client_class): @pytest.mark.parametrize( "client_class", - CLIENTS, + PYPARAM_CLIENT, ) def test_operations_client_from_service_account_info(client_class): creds = ga_credentials.AnonymousCredentials() @@ -196,7 +196,7 @@ def test_operations_client_from_service_account_info(client_class): [ transports.OperationsRestTransport, # TODO(https://github.com/googleapis/python-api-core/issues/706): Add support for - # service account credentials in transports.OperationsRestAsyncTransport + # service account credentials in transports.AsyncOperationsRestTransport ], ) def test_operations_client_service_account_always_use_jwt(transport_class): @@ -217,7 +217,7 @@ def test_operations_client_service_account_always_use_jwt(transport_class): @pytest.mark.parametrize( "client_class", - CLIENTS, + PYPARAM_CLIENT, ) def test_operations_client_from_service_account_file(client_class): @@ -244,7 +244,7 @@ def test_operations_client_from_service_account_file(client_class): @pytest.mark.parametrize( "client_class,transport_class,transport_name", - CLIENTS_WITH_TRANSPORT, + PYPARAM_CLIENT_TRANSPORT_NAME, ) def test_operations_client_get_transport_class( client_class, transport_class, transport_name @@ -254,7 +254,7 @@ def test_operations_client_get_transport_class( transports.OperationsRestTransport, ] if GOOGLE_AUTH_AIO_INSTALLED: - available_transports.append(transports.OperationsRestAsyncTransport) + available_transports.append(transports.AsyncOperationsRestTransport) assert transport in available_transports transport = client_class.get_transport_class(transport_name) @@ -481,7 +481,7 @@ def fake_init(client_cert_source_for_mtls=None, **kwargs): @pytest.mark.parametrize( "client_class,transport_class,transport_name", - CLIENTS_WITH_TRANSPORT, + PYPARAM_CLIENT_TRANSPORT_NAME, ) def test_operations_client_client_options_scopes( client_class, transport_class, transport_name @@ -512,7 +512,7 @@ def test_operations_client_client_options_scopes( @pytest.mark.parametrize( "client_class,transport_class,transport_name", - CLIENTS_WITH_TRANSPORT, + PYPARAM_CLIENT_TRANSPORT_NAME, ) def test_operations_client_client_options_credentials_file( client_class, transport_class, transport_name @@ -1037,7 +1037,7 @@ async def test_cancel_operation_rest_failure_async(): @pytest.mark.parametrize( "client_class,transport_class,credentials", - CLIENTS_WITH_CREDENTIALS, + PYPARAM_CLIENT_TRANSPORT_CREDENTIALS, ) def test_credentials_transport_error(client_class, transport_class, credentials): @@ -1068,7 +1068,7 @@ def test_credentials_transport_error(client_class, transport_class, credentials) @pytest.mark.parametrize( "client_class,transport_class,credentials", - CLIENTS_WITH_CREDENTIALS, + PYPARAM_CLIENT_TRANSPORT_CREDENTIALS, ) def test_transport_instance(client_class, transport_class, credentials): # A client may be instantiated with a custom transport instance. @@ -1081,7 +1081,7 @@ def test_transport_instance(client_class, transport_class, credentials): @pytest.mark.parametrize( "client_class,transport_class,credentials", - CLIENTS_WITH_CREDENTIALS, + PYPARAM_CLIENT_TRANSPORT_CREDENTIALS, ) def test_transport_adc(client_class, transport_class, credentials): # Test default credentials are used if not provided. @@ -1160,7 +1160,7 @@ def test_operations_base_transport_with_adc(): @pytest.mark.parametrize( "client_class", - CLIENTS, + PYPARAM_CLIENT, ) def test_operations_auth_adc(client_class): # If no credentials are provided, we should use ADC credentials. @@ -1185,7 +1185,7 @@ def test_operations_auth_adc(client_class): # TODO(https://github.com/googleapis/python-api-core/issues/705): Add -# testing for `transports.OperationsRestAsyncTransport` once MTLS is supported +# testing for `transports.AsyncOperationsRestTransport` once MTLS is supported # in `google.auth.aio.transport`. @pytest.mark.parametrize( "transport_class", @@ -1206,7 +1206,7 @@ def test_operations_http_transport_client_cert_source_for_mtls(transport_class): @pytest.mark.parametrize( "client_class,transport_class,credentials", - CLIENTS_WITH_CREDENTIALS, + PYPARAM_CLIENT_TRANSPORT_CREDENTIALS, ) def test_operations_host_no_port(client_class, transport_class, credentials): client = client_class( @@ -1220,7 +1220,7 @@ def test_operations_host_no_port(client_class, transport_class, credentials): @pytest.mark.parametrize( "client_class,transport_class,credentials", - CLIENTS_WITH_CREDENTIALS, + PYPARAM_CLIENT_TRANSPORT_CREDENTIALS, ) def test_operations_host_with_port(client_class, transport_class, credentials): client = client_class( @@ -1234,7 +1234,7 @@ def test_operations_host_with_port(client_class, transport_class, credentials): @pytest.mark.parametrize( "client_class", - CLIENTS, + PYPARAM_CLIENT, ) def test_common_billing_account_path(client_class): billing_account = "squid" @@ -1247,7 +1247,7 @@ def test_common_billing_account_path(client_class): @pytest.mark.parametrize( "client_class", - CLIENTS, + PYPARAM_CLIENT, ) def test_parse_common_billing_account_path(client_class): expected = { @@ -1262,7 +1262,7 @@ def test_parse_common_billing_account_path(client_class): @pytest.mark.parametrize( "client_class", - CLIENTS, + PYPARAM_CLIENT, ) def test_common_folder_path(client_class): folder = "whelk" @@ -1275,7 +1275,7 @@ def test_common_folder_path(client_class): @pytest.mark.parametrize( "client_class", - CLIENTS, + PYPARAM_CLIENT, ) def test_parse_common_folder_path(client_class): expected = { @@ -1290,7 +1290,7 @@ def test_parse_common_folder_path(client_class): @pytest.mark.parametrize( "client_class", - CLIENTS, + PYPARAM_CLIENT, ) def test_common_organization_path(client_class): organization = "oyster" @@ -1303,7 +1303,7 @@ def test_common_organization_path(client_class): @pytest.mark.parametrize( "client_class", - CLIENTS, + PYPARAM_CLIENT, ) def test_parse_common_organization_path(client_class): expected = { @@ -1318,7 +1318,7 @@ def test_parse_common_organization_path(client_class): @pytest.mark.parametrize( "client_class", - CLIENTS, + PYPARAM_CLIENT, ) def test_common_project_path(client_class): project = "cuttlefish" @@ -1331,7 +1331,7 @@ def test_common_project_path(client_class): @pytest.mark.parametrize( "client_class", - CLIENTS, + PYPARAM_CLIENT, ) def test_parse_common_project_path(client_class): expected = { @@ -1346,7 +1346,7 @@ def test_parse_common_project_path(client_class): @pytest.mark.parametrize( "client_class", - CLIENTS, + PYPARAM_CLIENT, ) def test_common_location_path(client_class): project = "winkle" @@ -1361,7 +1361,7 @@ def test_common_location_path(client_class): @pytest.mark.parametrize( "client_class", - CLIENTS, + PYPARAM_CLIENT, ) def test_parse_common_location_path(client_class): expected = { @@ -1377,7 +1377,7 @@ def test_parse_common_location_path(client_class): @pytest.mark.parametrize( "client_class,transport_class,credentials", - CLIENTS_WITH_CREDENTIALS, + PYPARAM_CLIENT_TRANSPORT_CREDENTIALS, ) def test_client_withDEFAULT_CLIENT_INFO(client_class, transport_class, credentials): client_info = gapic_v1.client_info.ClientInfo()