refactor(tvix/castore/tonic): use match in channel_from_url

Having random if blocks and returning from them is error-prone.

Also, turns out we only need the unprefixed scheme in the fallback case,
so move it down to there.

Change-Id: Ifcb09279c963f8a39e0dbabe145990263f3d7cf9
Reviewed-on: https://cl.tvl.fyi/c/depot/+/10041
Autosubmit: flokli <flokli@flokli.de>
Tested-by: BuildkiteCI
Reviewed-by: raitobezarius <tvl@lahfa.xyz>
This commit is contained in:
Florian Klink 2023-11-15 17:40:42 +02:00 committed by clbot
parent d4d1387409
commit ebfe456251

View file

@ -9,19 +9,16 @@ fn url_wants_wait_connect(url: &url::Url) -> bool {
} }
/// Turn a [url::Url] to a [Channel] if it can be parsed successfully. /// Turn a [url::Url] to a [Channel] if it can be parsed successfully.
/// It supports `grpc+unix:/path/to/socket`, as well as the regular schemes supported /// It supports the following schemes (and URLs):
/// by tonic, for example `grpc+http://[::1]:8000`. /// - `grpc+http://[::1]:8000`, connecting over unencrypted HTTP/2 (h2c)
/// It supports wait-connect=1 as a URL parameter, in which case we don't connect lazily. /// - `grpc+https://[::1]:8000`, connecting over encrypted HTTP/2
/// - `grpc+unix:/path/to/socket`, connecting to a unix domain socket
///
/// All URLs support adding `wait-connect=1` as a URL parameter, in which case
/// the connection is established lazily.
pub async fn channel_from_url(url: &url::Url) -> Result<Channel, self::Error> { pub async fn channel_from_url(url: &url::Url) -> Result<Channel, self::Error> {
// Stringify the URL and remove the grpc+ prefix. match url.scheme() {
// We can't use `url.set_scheme(rest)`, as it disallows "grpc+unix" => {
// setting something http(s) that previously wasn't.
let unprefixed_url_str = match url.to_string().strip_prefix("grpc+") {
None => return Err(Error::MissingGRPCPrefix()),
Some(url_str) => url_str.to_owned(),
};
if url.scheme() == "grpc+unix" {
if url.host_str().is_some() { if url.host_str().is_some() {
return Err(Error::HostSetForUnixSocket()); return Err(Error::HostSetForUnixSocket());
} }
@ -31,32 +28,38 @@ pub async fn channel_from_url(url: &url::Url) -> Result<Channel, self::Error> {
move |_: tonic::transport::Uri| UnixStream::connect(url.path().to_string().clone()) move |_: tonic::transport::Uri| UnixStream::connect(url.path().to_string().clone())
}); });
let channel = if url_wants_wait_connect(url) { // the URL doesn't matter
Endpoint::from_static("http://[::]:50051") let endpoint = Endpoint::from_static("http://[::]:50051");
.connect_with_connector(connector) if url_wants_wait_connect(url) {
.await? Ok(endpoint.connect_with_connector(connector).await?)
} else { } else {
Endpoint::from_static("http://[::]:50051").connect_with_connector_lazy(connector) Ok(endpoint.connect_with_connector_lazy(connector))
};
return Ok(channel);
} }
}
_ => {
// ensure path is empty, not supported with gRPC. // ensure path is empty, not supported with gRPC.
if !url.path().is_empty() { if !url.path().is_empty() {
return Err(Error::PathMayNotBeSet()); return Err(Error::PathMayNotBeSet());
} }
// Stringify the URL and remove the grpc+ prefix.
// We can't use `url.set_scheme(rest)`, as it disallows
// setting something http(s) that previously wasn't.
let unprefixed_url_str = match url.to_string().strip_prefix("grpc+") {
None => return Err(Error::MissingGRPCPrefix()),
Some(url_str) => url_str.to_owned(),
};
// Use the regular tonic transport::Endpoint logic, but unprefixed_url_str, // Use the regular tonic transport::Endpoint logic, but unprefixed_url_str,
// as tonic doesn't know about grpc+http[s]. // as tonic doesn't know about grpc+http[s].
let endpoint = Endpoint::try_from(unprefixed_url_str)?; let endpoint = Endpoint::try_from(unprefixed_url_str)?;
let channel = if url_wants_wait_connect(url) { if url_wants_wait_connect(url) {
endpoint.connect().await? Ok(endpoint.connect().await?)
} else { } else {
endpoint.connect_lazy() Ok(endpoint.connect_lazy())
}; }
}
Ok(channel) }
} }
/// Errors occuring when trying to connect to a backend /// Errors occuring when trying to connect to a backend