-
Notifications
You must be signed in to change notification settings - Fork 90
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
feat: Add attempt_direct_path argument to create_channel #583
Conversation
676ca58
to
e777b53
Compare
e777b53
to
0e974b6
Compare
c4f537d
to
257d4df
Compare
1d40546
to
421becd
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is looking good. I only reviewed the sync file and not the tests yet; I'll pick this back up tomorrow, but feel free to start responding to the comments here (or wait if you prefer; all good).
google/api_core/grpc_helpers.py
Outdated
else: | ||
# Use grpc.compute_engine_channel_credentials in order to support Direct Path. | ||
# See https://grpc.github.io/grpc/python/grpc.html#grpc.compute_engine_channel_credentials | ||
# TODO(b/323073050): Although `grpc.compute_engine_channel_credentials` |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Since this is public code, I would prefer to a public (GitHub) issue in this TODO.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Fixed in e9199a0
google/api_core/grpc_helpers.py
Outdated
attempt_direct_path (Optional[bool]): If set, Direct Path will be attempted when | ||
the request is made. Direct Path provides a proxyless connection which | ||
increases the available throughput, reduces latency, and increases | ||
reliability. Outside of GCE, the direct path request may fallback |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Do we need to spell out what GCE stands for in public docs? Not everyone looking at this file may be using GCE.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Fixed in e7e3cf7
google/api_core/grpc_helpers.py
Outdated
If a `ServiceUnavailable` response is received when the request is sent, it is | ||
recommended that the client repeat the request with `attempt_direct_path` set to `False` | ||
as the Service may not support Direct Path. Using `ssl_credentials` with `attempt_direct_path` | ||
set to `True` will result in `ValueError` as it is not yet supported. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
set to `True` will result in `ValueError` as it is not yet supported. | |
set to `True` will result in `ValueError` as this combination is not yet supported. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Fixed in e7e3cf7
google/api_core/grpc_helpers.py
Outdated
attempt_direct_path (Optional[bool]): If set, Direct Path will be attempted when | ||
the request is made. Direct Path provides a proxyless connection which | ||
increases the available throughput, reduces latency, and increases | ||
reliability. Outside of GCE, the direct path request may fallback | ||
to DNS if this is configured by the Service. This argument should only | ||
be set in a GCE environment and for Services that are known to support Direct Path. | ||
If a `ServiceUnavailable` response is received when the request is sent, it is | ||
recommended that the client repeat the request with `attempt_direct_path` set to `False` | ||
as the Service may not support Direct Path. Using `ssl_credentials` with `attempt_direct_path` | ||
set to `True` will result in `ValueError` as it is not yet supported. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
"Should only be set in a GCE environment" is confusing in light of "Outside of GCE, the request may fall back". Clarify. I suspect you mean something like this:
attempt_direct_path (Optional[bool]): If set, Direct Path will be attempted when | |
the request is made. Direct Path provides a proxyless connection which | |
increases the available throughput, reduces latency, and increases | |
reliability. Outside of GCE, the direct path request may fallback | |
to DNS if this is configured by the Service. This argument should only | |
be set in a GCE environment and for Services that are known to support Direct Path. | |
If a `ServiceUnavailable` response is received when the request is sent, it is | |
recommended that the client repeat the request with `attempt_direct_path` set to `False` | |
as the Service may not support Direct Path. Using `ssl_credentials` with `attempt_direct_path` | |
set to `True` will result in `ValueError` as it is not yet supported. | |
attempt_direct_path (Optional[bool]): If set, Direct Path will be attempted when | |
the request is made. Direct Path is only available within a Google Compute | |
Engine environment and provides a proxyless connection which increases the | |
available throughput, reduces latency, and increases reliability. | |
- This argument should only | |
be set in a GCE environment and for Services that are known to support Direct | |
Path. | |
- If this argument is set outside of GCE, then this request will fail | |
unless the back-end service happens to have configured fall-back to DNS. | |
- If the request causes a `ServiceUnavailable` response, we | |
recommend that the client repeat the request with `attempt_direct_path` set to | |
`False` as the Service may not support Direct Path. | |
- Using `ssl_credentials` with `attempt_direct_path` | |
set to `True` will result in `ValueError` as this combination is not yet | |
supported. |
(And similarly for the async version)
WDYT?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Fixed in e7e3cf7
google/api_core/grpc_helpers.py
Outdated
attempt_direct_path (Optional[bool]): If set, Direct Path will be attempted when | ||
the request is made. Direct Path provides a proxyless connection which | ||
increases the available throughput, reduces latency, and increases | ||
reliability. Outside of GCE, the direct path request may fallback |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
reliability. Outside of GCE, the direct path request may fallback | |
reliability. Outside of GCE, the direct path request may fall back |
but see comment below
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Fixed in e7e3cf7
google/api_core/grpc_helpers.py
Outdated
target (str): The target service address which is converted into a format compatible with Direct Path. | ||
If the target contains `dns:///` or does not have contain `:///`, the target will be converted in | ||
a format compatible with Direct Path, otherwise the original target will be returned. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
target (str): The target service address which is converted into a format compatible with Direct Path. | |
If the target contains `dns:///` or does not have contain `:///`, the target will be converted in | |
a format compatible with Direct Path, otherwise the original target will be returned. | |
target (str): The target service address which is converted into a format compatible with Direct Path. | |
If the target contains `dns:///` or does not contain `:///`, the target will be converted in | |
a format compatible with Direct Path; otherwise the original target will be returned. |
So the idea is that a :///
(except for dns:///
) already denotes Direct Path, so no action is needed, correct? It might be good to be explicit about this assumption.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Done in 44d5845
a format compatible with Direct Path, otherwise the original target will be returned. | ||
""" | ||
|
||
dns_prefix = "dns:///" |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It might be helpful to clarify in a comment what the dns_prefix
means, which I take it is to be explicit about an endpoint living in the Internet (ie outside GCP).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Fixed in e6ea7ec
google/api_core/grpc_helpers.py
Outdated
@@ -288,6 +300,7 @@ def create_channel( | |||
default_scopes=None, | |||
default_host=None, | |||
compression=None, | |||
attempt_direct_path: Optional[bool] = None, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
attempt_direct_path: Optional[bool] = None, | |
attempt_direct_path: Optional[bool] = False, |
Since it's meant to be a Boolean, might as well.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Fixed in 0f3a0e4
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
LGTM! Nothing blocking, but I think some of my suggestions can make things tighter/clearer.
google/api_core/grpc_helpers.py
Outdated
# If `ssl_credentials` is set and `attempt_direct_path` is set to `True`, | ||
# raise ValueError as this is not yet supported. | ||
# See https://github.com/googleapis/python-api-core/issues/590 | ||
if ssl_credentials is not None and attempt_direct_path: |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Could we simplify to just this?:
if ssl_credentials is not None and attempt_direct_path: | |
if ssl_credentials and attempt_direct_path: |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Fixed in 60b8acb
if HAS_GRPC_GCP: # pragma: NO COVER | ||
if compression is not None and compression != grpc.Compression.NoCompression: | ||
_LOGGER.debug( | ||
"Compression argument is being ignored for grpc_gcp.secure_channel creation." | ||
) | ||
if attempt_direct_path: | ||
warnings.warn( |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
For consistency, the _LOGGER.debug
in the previous lines should probably also become a warnings.warn
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
google/api_core/grpc_helpers.py
Outdated
Given a target, return a modified version which is compatible with Direct Path. | ||
|
||
Args: | ||
target (str): The target service address in the format 'hostname:port', 'dns://hostname' or other |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
target (str): The target service address in the format 'hostname:port', 'dns://hostname' or other | |
target (str): The target service address in the format 'hostname[:port]', 'dns://hostname[:port]' or other |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Fixed in af3a21b
google/api_core/grpc_helpers.py
Outdated
|
||
Args: | ||
target (str): The target service address in the format 'hostname:port', 'dns://hostname' or other | ||
compatible format. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
"other compatible format": link to a place that lists compatible formats, or remove this phrase which I think is puzzling by itself.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Removed in af3a21b
google/api_core/grpc_helpers.py
Outdated
direct_path_prefix = ":///" | ||
if direct_path_prefix not in target: |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
direct_path_prefix = ":///" | |
if direct_path_prefix not in target: | |
direct_path_separator = ":///" | |
if direct_path_separator not in target: |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Fixed in c0052e6
return grpc.secure_channel( | ||
target, composite_credentials, compression=compression, **kwargs | ||
) | ||
|
||
|
||
def _modify_target_for_direct_path(target: str) -> str: |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Q: In general, an API endpoint could depend on the URL path and not just the host+port. Is this true of Google APIs? If so, where do we deal with the path part of the URI? (I realize these functions specify only host+port as inputs, so it's clear what they expect and they're doing the right thing, but I was wondering about this more general question.)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
If so, where do we deal with the path part of the URI?
I believe this is part of the transcode
method
Transcodes a grpc request pattern into a proper HTTP request following the rules outlined here,
python-api-core/google/api_core/path_template.py
Lines 250 to 259 in b72929f
def transcode(http_options, message=None, **request_kwargs): | |
"""Transcodes a grpc request pattern into a proper HTTP request following the rules outlined here, | |
https://github.com/googleapis/googleapis/blob/master/google/api/http.proto#L44-L312 | |
Args: | |
http_options (list(dict)): A list of dicts which consist of these keys, | |
'method' (str): The http method | |
'uri' (str): The path template | |
'body' (str): The body field name (optional) | |
(This is a simplified representation of the proto option `google.api.http`) |
@@ -21,7 +21,7 @@ | |||
import asyncio | |||
import functools | |||
|
|||
from typing import Generic, Iterator, AsyncGenerator, TypeVar | |||
from typing import AsyncGenerator, Generic, Iterator, Optional, TypeVar |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
General notes:
- The comments from the sycn version apply here too
- Just a note for the future (not blocking this PR): a lot of the code seems identical or very similar to the sync code. We should consolidate sensibly to eliminate duplication. (I just filed consolidate duplicated code #593 to track more such instances)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks for opening that issue! I also made changes to the async code based on the sync code feedback.
tests/unit/test_grpc_helpers.py
Outdated
def test_create_channel_implicit(grpc_secure_channel, default, composite_creds_call): | ||
def test_create_channel_implicit( | ||
grpc_secure_channel, | ||
default, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
default
is too generic a name. I assume this is meant to take the google.auth.default
you have patched in the decorators. Is there any way this parameter name could be made more descriptive (like auth-default
, say), or does @mock
preclude that?
(I realize this was pre-existing, so no need to spend too much time on this. But if it's easy...)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Fixed in d63402e
@@ -365,52 +365,92 @@ def test_wrap_errors_streaming(wrap_stream_errors): | |||
wrap_stream_errors.assert_called_once_with(callable_) | |||
|
|||
|
|||
@mock.patch("grpc.composite_channel_credentials") | |||
@pytest.mark.parametrize( |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Would it be worth making this test function take a target parameter, and then parametrizing it as we do for test_create_channel_implicit_with_default_host
below? target
seems to be used the same way in both places, and this would allow us to check here the dns:///
and another-c2p:///
cases you parametrize below.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Done in 396d8fa
@@ -298,48 +298,82 @@ def test_wrap_errors_streaming(wrap_stream_errors): | |||
wrap_stream_errors.assert_called_once_with(callable_) | |||
|
|||
|
|||
@mock.patch("grpc.composite_channel_credentials") | |||
@pytest.mark.parametrize( |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Same comments as in sync version apply here.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Done in 396d8fa
b4a6edf
to
396d8fa
Compare
Fixes #482
b/267782870