-
Notifications
You must be signed in to change notification settings - Fork 131
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
Updated endpoint parsing #618
Merged
berndverst
merged 13 commits into
dapr:master
from
elena-kolevska:updated-endpoint-parsing
Oct 31, 2023
Merged
Changes from all commits
Commits
Show all changes
13 commits
Select commit
Hold shift + click to select a range
af3c06c
wip
elena-kolevska 60b7b70
Updates endpoint parsing for new spec
elena-kolevska bc7c880
Adds support for unix URIs. Clean up.
elena-kolevska eb362ac
Completes support for all valid grpc endpoints
elena-kolevska 65a8955
Code cleanup
elena-kolevska 2d8d80b
Adds a warning about the http and https schemes being deprecated for …
elena-kolevska af1189a
Updates docs
elena-kolevska bc85dbc
Updates the docs for clarity and correctness
elena-kolevska 71d229b
Adds anoter test for vsock without a port
elena-kolevska d52c1a9
Adds more test cases and handles dns with ipv6
elena-kolevska 8874681
Code cleanup
elena-kolevska cb7fe97
Fixes linter
elena-kolevska 59b4e42
Fixes linter errors
elena-kolevska File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -106,3 +106,6 @@ venv.bak/ | |
|
||
# mypy | ||
.mypy_cache/ | ||
|
||
# OSX specific files | ||
.DS_Store |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,56 +1,183 @@ | ||
from typing import Tuple | ||
|
||
|
||
def parse_endpoint(address: str) -> Tuple[str, str, int]: | ||
scheme = "http" | ||
fqdn = "localhost" | ||
port = 80 | ||
addr = address | ||
|
||
addr_list = address.split("://") | ||
|
||
if len(addr_list) == 2: | ||
# A scheme was explicitly specified | ||
scheme = addr_list[0] | ||
if scheme == "https": | ||
port = 443 | ||
addr = addr_list[1] | ||
|
||
addr_list = addr.split(":") | ||
if len(addr_list) == 2: | ||
# A port was explicitly specified | ||
if len(addr_list[0]) > 0: | ||
fqdn = addr_list[0] | ||
# Account for Endpoints of the type http://localhost:3500/v1.0/invoke | ||
addr_list = addr_list[1].split("/") | ||
port = addr_list[0] # type: ignore | ||
elif len(addr_list) == 1: | ||
# No port was specified | ||
# Account for Endpoints of the type :3500/v1.0/invoke | ||
addr_list = addr_list[0].split("/") | ||
fqdn = addr_list[0] | ||
else: | ||
# IPv6 address | ||
addr_list = addr.split("]:") | ||
if len(addr_list) == 2: | ||
# A port was explicitly specified | ||
fqdn = addr_list[0] | ||
fqdn = fqdn.replace("[", "") | ||
|
||
addr_list = addr_list[1].split("/") | ||
port = addr_list[0] # type: ignore | ||
elif len(addr_list) == 1: | ||
# No port was specified | ||
addr_list = addr_list[0].split("/") | ||
fqdn = addr_list[0] | ||
fqdn = fqdn.replace("[", "") | ||
fqdn = fqdn.replace("]", "") | ||
from warnings import warn | ||
from urllib.parse import urlparse, parse_qs, ParseResult | ||
|
||
|
||
class URIParseConfig: | ||
DEFAULT_SCHEME = "dns" | ||
DEFAULT_HOSTNAME = "localhost" | ||
DEFAULT_PORT = 443 | ||
DEFAULT_AUTHORITY = "" | ||
ACCEPTED_SCHEMES = ["dns", "unix", "unix-abstract", "vsock", "http", "https", "grpc", "grpcs"] | ||
|
||
|
||
class GrpcEndpoint: | ||
_scheme: str | ||
_hostname: str | ||
_port: int | ||
_tls: bool | ||
_authority: str | ||
_url: str | ||
_parsed_url: ParseResult # from urllib.parse | ||
_endpoint: str | ||
|
||
def __init__(self, url: str): | ||
self._authority = URIParseConfig.DEFAULT_AUTHORITY | ||
self._url = url | ||
|
||
self._parsed_url = urlparse(self._preprocess_uri(url)) | ||
self._validate_path_and_query() | ||
|
||
self._set_tls() | ||
self._set_hostname() | ||
self._set_scheme() | ||
self._set_port() | ||
self._set_endpoint() | ||
|
||
def _set_scheme(self): | ||
if len(self._parsed_url.scheme) == 0: | ||
self._scheme = URIParseConfig.DEFAULT_SCHEME | ||
return | ||
|
||
if self._parsed_url.scheme in ["http", "https"]: | ||
self._scheme = URIParseConfig.DEFAULT_SCHEME | ||
warn("http and https schemes are deprecated, use grpc or grpcs instead") | ||
return | ||
|
||
if self._parsed_url.scheme not in URIParseConfig.ACCEPTED_SCHEMES: | ||
raise ValueError(f"invalid scheme '{self._parsed_url.scheme}' in URL '{self._url}'") | ||
|
||
self._scheme = self._parsed_url.scheme | ||
|
||
@property | ||
def scheme(self) -> str: | ||
return self._scheme | ||
|
||
def _set_hostname(self): | ||
if self._parsed_url.hostname is None: | ||
self._hostname = URIParseConfig.DEFAULT_HOSTNAME | ||
return | ||
|
||
if self._parsed_url.hostname.count(":") == 7: | ||
# IPv6 address | ||
self._hostname = f"[{self._parsed_url.hostname}]" | ||
return | ||
|
||
self._hostname = self._parsed_url.hostname | ||
|
||
@property | ||
def hostname(self) -> str: | ||
return self._hostname | ||
|
||
def _set_port(self): | ||
if self._parsed_url.scheme in ["unix", "unix-abstract"]: | ||
self._port = 0 | ||
return | ||
|
||
if self._parsed_url.port is None: | ||
self._port = URIParseConfig.DEFAULT_PORT | ||
return | ||
|
||
self._port = self._parsed_url.port | ||
|
||
@property | ||
def port(self) -> str: | ||
if self._port == 0: | ||
return "" | ||
|
||
return str(self._port) | ||
|
||
@property | ||
def port_as_int(self) -> int: | ||
return self._port | ||
|
||
def _set_endpoint(self): | ||
port = "" if not self._port else f":{self.port}" | ||
|
||
if self._scheme == "unix": | ||
separator = "://" if self._url.startswith("unix://") else ":" | ||
self._endpoint = f"{self._scheme}{separator}{self._hostname}" | ||
return | ||
|
||
if self._scheme == "vsock": | ||
self._endpoint = f"{self._scheme}:{self._hostname}:{self.port}" | ||
return | ||
|
||
if self._scheme == "unix-abstract": | ||
self._endpoint = f"{self._scheme}:{self._hostname}{port}" | ||
return | ||
|
||
if self._scheme == "dns": | ||
authority = f"//{self._authority}/" if self._authority else "" | ||
self._endpoint = f"{self._scheme}:{authority}{self._hostname}{port}" | ||
return | ||
|
||
self._endpoint = f"{self._scheme}:{self._hostname}{port}" | ||
|
||
@property | ||
def endpoint(self) -> str: | ||
return self._endpoint | ||
|
||
# Prepares the uri string in a specific format for parsing by the urlparse function | ||
def _preprocess_uri(self, url: str) -> str: | ||
url_list = url.split(":") | ||
if len(url_list) == 3 and "://" not in url: | ||
# A URI like dns:mydomain:5000 or vsock:mycid:5000 was used | ||
url = url.replace(":", "://", 1) | ||
elif len(url_list) >= 2 and "://" not in url and url_list[ | ||
0] in URIParseConfig.ACCEPTED_SCHEMES: | ||
|
||
# A URI like dns:mydomain or dns:[2001:db8:1f70::999:de8:7648:6e8]:mydomain was used | ||
# Possibly a URI like dns:[2001:db8:1f70::999:de8:7648:6e8]:mydomain was used | ||
url = url.replace(":", "://", 1) | ||
else: | ||
raise ValueError(f"Invalid address: {address}") | ||
url_list = url.split("://") | ||
if len(url_list) == 1: | ||
# If a scheme was not explicitly specified in the URL | ||
# we need to add a default scheme, | ||
# because of how urlparse works | ||
url = f'{URIParseConfig.DEFAULT_SCHEME}://{url}' | ||
else: | ||
# If a scheme was explicitly specified in the URL | ||
# we need to make sure it is a valid scheme | ||
scheme = url_list[0] | ||
if scheme not in URIParseConfig.ACCEPTED_SCHEMES: | ||
raise ValueError(f"invalid scheme '{scheme}' in URL '{url}'") | ||
|
||
# We should do a special check if the scheme is dns, and it uses | ||
# an authority in the format of dns:[//authority/]host[:port] | ||
if scheme.lower() == "dns": | ||
# A URI like dns://authority/mydomain was used | ||
url_list = url.split("/") | ||
if len(url_list) < 4: | ||
raise ValueError(f"invalid dns authority '{url_list[2]}' in URL '{url}'") | ||
self._authority = url_list[2] | ||
url = f'dns://{url_list[3]}' | ||
return url | ||
|
||
def _set_tls(self): | ||
query_dict = parse_qs(self._parsed_url.query) | ||
tls_str = query_dict.get('tls', [""])[0] | ||
tls = tls_str.lower() == 'true' | ||
if self._parsed_url.scheme == "https": | ||
tls = True | ||
|
||
self._tls = tls | ||
|
||
try: | ||
port = int(port) | ||
except ValueError: | ||
raise ValueError(f"invalid port: {port}") | ||
@property | ||
def tls(self) -> bool: | ||
return self._tls | ||
|
||
return scheme, fqdn, port | ||
def _validate_path_and_query(self) -> None: | ||
if self._parsed_url.path: | ||
raise ValueError(f"paths are not supported for gRPC endpoints:" | ||
f" '{self._parsed_url.path}'") | ||
if self._parsed_url.query: | ||
query_dict = parse_qs(self._parsed_url.query) | ||
if 'tls' in query_dict and self._parsed_url.scheme in ["http", "https"]: | ||
raise ValueError( | ||
f"the tls query parameter is not supported for http(s) endpoints: " | ||
f"'{self._parsed_url.query}'") | ||
query_dict.pop('tls', None) | ||
if query_dict: | ||
raise ValueError(f"query parameters are not supported for gRPC endpoints:" | ||
f" '{self._parsed_url.query}'") |
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
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.
cc @elena-kolevska
something's off here, both
grpc
andgrpcs
are not present in the naming resolution doc and are also missing in the go-sdk: https://github.com/dapr/go-sdk/blob/main/client/internal/parse.go#L160There 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 this PR: #700