2021-11-09 08:07:49 +00:00
|
|
|
package client
|
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto/ecdsa"
|
|
|
|
"crypto/tls"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/nspcc-dev/neofs-api-go/v2/refs"
|
2021-11-16 18:15:56 +00:00
|
|
|
"github.com/nspcc-dev/neofs-api-go/v2/rpc/client"
|
2021-11-09 08:07:49 +00:00
|
|
|
v2session "github.com/nspcc-dev/neofs-api-go/v2/session"
|
|
|
|
"github.com/nspcc-dev/neofs-sdk-go/session"
|
|
|
|
"github.com/nspcc-dev/neofs-sdk-go/token"
|
|
|
|
"github.com/nspcc-dev/neofs-sdk-go/version"
|
|
|
|
"google.golang.org/grpc"
|
|
|
|
)
|
|
|
|
|
|
|
|
type (
|
|
|
|
CallOption func(*callOptions)
|
|
|
|
|
|
|
|
Option func(*clientOptions)
|
|
|
|
|
|
|
|
callOptions struct {
|
|
|
|
version *version.Version
|
|
|
|
xHeaders []*session.XHeader
|
|
|
|
ttl uint32
|
|
|
|
epoch uint64
|
|
|
|
key *ecdsa.PrivateKey
|
|
|
|
session *session.Token
|
|
|
|
bearer *token.BearerToken
|
2022-01-14 10:38:51 +00:00
|
|
|
// network magic is a client config, but it's convenient to copy it here (see v2MetaHeaderFromOpts)
|
|
|
|
// the value remains the same between calls
|
|
|
|
netMagic uint64
|
2021-11-09 08:07:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
clientOptions struct {
|
|
|
|
key *ecdsa.PrivateKey
|
|
|
|
|
|
|
|
rawOpts []client.Option
|
|
|
|
|
|
|
|
cbRespInfo func(ResponseMetaInfo) error
|
2022-01-11 12:32:29 +00:00
|
|
|
|
|
|
|
// defines if client parses erroneous NeoFS
|
|
|
|
// statuses and returns them as `error`
|
|
|
|
//
|
|
|
|
// default is false
|
|
|
|
parseNeoFSErrors bool
|
2022-01-14 10:38:51 +00:00
|
|
|
|
|
|
|
netMagic uint64
|
2021-11-09 08:07:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
v2SessionReqInfo struct {
|
|
|
|
addr *refs.Address
|
|
|
|
verb v2session.ObjectSessionVerb
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
2021-12-29 13:18:09 +00:00
|
|
|
func (c *Client) defaultCallOptions() *callOptions {
|
2021-11-09 08:07:49 +00:00
|
|
|
return &callOptions{
|
|
|
|
version: version.Current(),
|
|
|
|
ttl: 2,
|
|
|
|
key: c.opts.key,
|
2022-01-14 10:38:51 +00:00
|
|
|
// copy client's static value is a bit overhead, but it is the easiest way at the time of feature intro
|
|
|
|
netMagic: c.opts.netMagic,
|
2021-11-09 08:07:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func WithXHeader(x *session.XHeader) CallOption {
|
|
|
|
return func(opts *callOptions) {
|
|
|
|
opts.xHeaders = append(opts.xHeaders, x)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func WithTTL(ttl uint32) CallOption {
|
|
|
|
return func(opts *callOptions) {
|
|
|
|
opts.ttl = ttl
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// WithKey sets client's key for the next request.
|
|
|
|
func WithKey(key *ecdsa.PrivateKey) CallOption {
|
|
|
|
return func(opts *callOptions) {
|
|
|
|
opts.key = key
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func WithEpoch(epoch uint64) CallOption {
|
|
|
|
return func(opts *callOptions) {
|
|
|
|
opts.epoch = epoch
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func WithSession(token *session.Token) CallOption {
|
|
|
|
return func(opts *callOptions) {
|
|
|
|
opts.session = token
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func WithBearer(token *token.BearerToken) CallOption {
|
|
|
|
return func(opts *callOptions) {
|
|
|
|
opts.bearer = token
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func v2MetaHeaderFromOpts(options *callOptions) *v2session.RequestMetaHeader {
|
|
|
|
meta := new(v2session.RequestMetaHeader)
|
|
|
|
meta.SetVersion(options.version.ToV2())
|
|
|
|
meta.SetTTL(options.ttl)
|
|
|
|
meta.SetEpoch(options.epoch)
|
|
|
|
|
|
|
|
xhdrs := make([]*v2session.XHeader, len(options.xHeaders))
|
|
|
|
for i := range options.xHeaders {
|
|
|
|
xhdrs[i] = options.xHeaders[i].ToV2()
|
|
|
|
}
|
|
|
|
|
|
|
|
meta.SetXHeaders(xhdrs)
|
|
|
|
|
|
|
|
if options.bearer != nil {
|
|
|
|
meta.SetBearerToken(options.bearer.ToV2())
|
|
|
|
}
|
|
|
|
|
|
|
|
meta.SetSessionToken(options.session.ToV2())
|
|
|
|
|
2022-01-14 10:38:51 +00:00
|
|
|
meta.SetNetworkMagic(options.netMagic)
|
|
|
|
|
2021-11-09 08:07:49 +00:00
|
|
|
return meta
|
|
|
|
}
|
|
|
|
|
|
|
|
func defaultClientOptions() *clientOptions {
|
|
|
|
return &clientOptions{
|
|
|
|
rawOpts: make([]client.Option, 0, 4),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// WithAddress returns option to specify
|
|
|
|
// network address of the remote server.
|
|
|
|
//
|
|
|
|
// Ignored if WithGRPCConnection is provided.
|
|
|
|
func WithAddress(addr string) Option {
|
|
|
|
return func(opts *clientOptions) {
|
|
|
|
opts.rawOpts = append(opts.rawOpts, client.WithNetworkAddress(addr))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// WithDialTimeout returns option to set connection timeout to the remote node.
|
|
|
|
//
|
|
|
|
// Ignored if WithGRPCConn is provided.
|
|
|
|
func WithDialTimeout(dur time.Duration) Option {
|
|
|
|
return func(opts *clientOptions) {
|
|
|
|
opts.rawOpts = append(opts.rawOpts, client.WithDialTimeout(dur))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-24 11:24:13 +00:00
|
|
|
// WithRWTimeout returns option to set timeout for single read and write
|
|
|
|
// operation on protobuf message.
|
|
|
|
func WithRWTimeout(dur time.Duration) Option {
|
|
|
|
return func(opts *clientOptions) {
|
|
|
|
opts.rawOpts = append(opts.rawOpts, client.WithRWTimeout(dur))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-11-09 08:07:49 +00:00
|
|
|
// WithTLSConfig returns option to set connection's TLS config to the remote node.
|
|
|
|
//
|
|
|
|
// Ignored if WithGRPCConnection is provided.
|
|
|
|
func WithTLSConfig(cfg *tls.Config) Option {
|
|
|
|
return func(opts *clientOptions) {
|
|
|
|
opts.rawOpts = append(opts.rawOpts, client.WithTLSCfg(cfg))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// WithDefaultPrivateKey returns option to set default private key
|
|
|
|
// used for the work.
|
|
|
|
func WithDefaultPrivateKey(key *ecdsa.PrivateKey) Option {
|
|
|
|
return func(opts *clientOptions) {
|
|
|
|
opts.key = key
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// WithURIAddress returns option to specify
|
|
|
|
// network address of a remote server and connection
|
|
|
|
// scheme for it.
|
|
|
|
//
|
|
|
|
// Format of the URI:
|
|
|
|
//
|
|
|
|
// [scheme://]host:port
|
|
|
|
//
|
|
|
|
// Supported schemes:
|
|
|
|
// - grpc;
|
|
|
|
// - grpcs.
|
|
|
|
//
|
|
|
|
// tls.Cfg second argument is optional and is taken into
|
|
|
|
// account only in case of `grpcs` scheme.
|
|
|
|
//
|
|
|
|
// Falls back to WithNetworkAddress if address is not a valid URI.
|
|
|
|
//
|
|
|
|
// Do not use along with WithAddress and WithTLSConfig.
|
|
|
|
//
|
|
|
|
// Ignored if WithGRPCConnection is provided.
|
|
|
|
func WithURIAddress(addr string, tlsCfg *tls.Config) Option {
|
|
|
|
return func(opts *clientOptions) {
|
|
|
|
opts.rawOpts = append(opts.rawOpts, client.WithNetworkURIAddress(addr, tlsCfg)...)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// WithGRPCConnection returns option to set GRPC connection to
|
|
|
|
// the remote node.
|
|
|
|
func WithGRPCConnection(grpcConn *grpc.ClientConn) Option {
|
|
|
|
return func(opts *clientOptions) {
|
|
|
|
opts.rawOpts = append(opts.rawOpts, client.WithGRPCConn(grpcConn))
|
|
|
|
}
|
|
|
|
}
|
2022-01-11 12:32:29 +00:00
|
|
|
|
|
|
|
// WithNeoFSErrorParsing returns option that makes client parse
|
|
|
|
// erroneous NeoFS statuses and return them as `error` of the method
|
|
|
|
// call.
|
|
|
|
func WithNeoFSErrorParsing() Option {
|
|
|
|
return func(opts *clientOptions) {
|
|
|
|
opts.parseNeoFSErrors = true
|
|
|
|
}
|
|
|
|
}
|
2022-01-14 10:38:51 +00:00
|
|
|
|
|
|
|
// WithNetworkMagic returns option to specify NeoFS network magic.
|
|
|
|
func WithNetworkMagic(magic uint64) Option {
|
|
|
|
return func(opts *clientOptions) {
|
|
|
|
opts.netMagic = magic
|
|
|
|
}
|
|
|
|
}
|