[#383] rpc/client: Export URI-parsing function

Signed-off-by: Evgenii Stratonikov <evgeniy@nspcc.ru>
This commit is contained in:
Evgenii Stratonikov 2022-03-09 12:56:02 +03:00 committed by Alex Vanin
parent fadd47f4fb
commit 50382114f4
2 changed files with 45 additions and 25 deletions

View file

@ -3,6 +3,9 @@ package client
import ( import (
"context" "context"
"errors" "errors"
"fmt"
"net"
"net/url"
"github.com/nspcc-dev/neofs-api-go/v2/rpc/grpc" "github.com/nspcc-dev/neofs-api-go/v2/rpc/grpc"
grpcstd "google.golang.org/grpc" grpcstd "google.golang.org/grpc"
@ -52,3 +55,33 @@ func (c *Client) openGRPCConn() error {
return err return err
} }
// ParseURI parses s as address and returns a host and a flag
// indicating that TLS is enabled. If multi-address is provided
// the argument is returned unchanged.
func ParseURI(s string) (string, bool, error) {
uri, err := url.ParseRequestURI(s)
if err != nil {
return s, false, nil
}
// check if passed string was parsed correctly
// URIs that do not start with a slash after the scheme are interpreted as:
// `scheme:opaque` => if `opaque` is not empty, then it is supposed that URI
// is in `host:port` format
if uri.Host == "" {
uri.Host = uri.Scheme
uri.Scheme = grpcScheme // assume GRPC by default
if uri.Opaque != "" {
uri.Host = net.JoinHostPort(uri.Host, uri.Opaque)
}
}
switch uri.Scheme {
case grpcTLSScheme, grpcScheme:
default:
return "", false, fmt.Errorf("unsupported scheme: %s", uri.Scheme)
}
return uri.Host, uri.Scheme == grpcTLSScheme, nil
}

View file

@ -2,7 +2,6 @@ package client
import ( import (
"crypto/tls" "crypto/tls"
"net/url"
"time" "time"
"google.golang.org/grpc" "google.golang.org/grpc"
@ -58,35 +57,23 @@ func WithNetworkAddress(v string) Option {
// //
// Ignored if WithGRPCConn is provided. // Ignored if WithGRPCConn is provided.
func WithNetworkURIAddress(addr string, tlsCfg *tls.Config) []Option { func WithNetworkURIAddress(addr string, tlsCfg *tls.Config) []Option {
uri, err := url.ParseRequestURI(addr) host, isTLS, err := ParseURI(addr)
if err != nil { if err != nil {
return []Option{WithNetworkAddress(addr)}
}
// check if passed string was parsed correctly
// URIs that do not start with a slash after the scheme are interpreted as:
// `scheme:opaque` => if `opaque` is not empty, then it is supposed that URI
// is in `host:port` format
if uri.Opaque != "" {
return []Option{WithNetworkAddress(addr)}
}
switch uri.Scheme {
case grpcScheme:
tlsCfg = nil
case grpcTLSScheme:
if tlsCfg == nil {
tlsCfg = &tls.Config{}
}
default:
// not supported scheme
return nil return nil
} }
return []Option{ opts := make([]Option, 2)
WithNetworkAddress(uri.Host), opts[0] = WithNetworkAddress(host)
WithTLSCfg(tlsCfg), if isTLS {
if tlsCfg == nil {
tlsCfg = &tls.Config{}
} }
opts[1] = WithTLSCfg(tlsCfg)
} else {
opts[1] = WithTLSCfg(nil)
}
return opts
} }
// WithDialTimeout returns option to specify // WithDialTimeout returns option to specify