2018-08-27 22:21:45 +02:00
|
|
|
package connecter
|
|
|
|
|
|
|
|
import (
|
2018-10-11 21:18:44 +02:00
|
|
|
"context"
|
2018-08-27 22:21:45 +02:00
|
|
|
"fmt"
|
|
|
|
"github.com/problame/go-streamrpc"
|
|
|
|
"github.com/zrepl/zrepl/config"
|
2018-08-31 21:51:44 +02:00
|
|
|
"github.com/zrepl/zrepl/daemon/streamrpcconfig"
|
2018-10-11 21:18:44 +02:00
|
|
|
"github.com/zrepl/zrepl/daemon/transport"
|
|
|
|
"net"
|
|
|
|
"time"
|
2018-08-27 22:21:45 +02:00
|
|
|
)
|
|
|
|
|
2018-10-11 21:18:44 +02:00
|
|
|
|
|
|
|
type HandshakeConnecter struct {
|
|
|
|
connecter streamrpc.Connecter
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c HandshakeConnecter) Connect(ctx context.Context) (net.Conn, error) {
|
|
|
|
conn, err := c.connecter.Connect(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
dl, ok := ctx.Deadline()
|
|
|
|
if !ok {
|
|
|
|
dl = time.Now().Add(10 * time.Second) // FIXME constant
|
|
|
|
}
|
|
|
|
if err := transport.DoHandshakeCurrentVersion(conn, dl); err != nil {
|
|
|
|
conn.Close()
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return conn, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2018-08-31 21:51:44 +02:00
|
|
|
func FromConfig(g *config.Global, in config.ConnectEnum) (*ClientFactory, error) {
|
|
|
|
var (
|
|
|
|
connecter streamrpc.Connecter
|
|
|
|
errConnecter, errRPC error
|
|
|
|
connConf *streamrpc.ConnConfig
|
|
|
|
)
|
2018-08-27 22:21:45 +02:00
|
|
|
switch v := in.Ret.(type) {
|
|
|
|
case *config.SSHStdinserverConnect:
|
2018-08-31 21:51:44 +02:00
|
|
|
connecter, errConnecter = SSHStdinserverConnecterFromConfig(v)
|
|
|
|
connConf, errRPC = streamrpcconfig.FromDaemonConfig(g, v.RPC)
|
2018-08-27 22:21:45 +02:00
|
|
|
case *config.TCPConnect:
|
2018-08-31 21:51:44 +02:00
|
|
|
connecter, errConnecter = TCPConnecterFromConfig(v)
|
|
|
|
connConf, errRPC = streamrpcconfig.FromDaemonConfig(g, v.RPC)
|
2018-08-27 22:21:45 +02:00
|
|
|
case *config.TLSConnect:
|
2018-08-31 21:51:44 +02:00
|
|
|
connecter, errConnecter = TLSConnecterFromConfig(v)
|
|
|
|
connConf, errRPC = streamrpcconfig.FromDaemonConfig(g, v.RPC)
|
2018-09-24 14:43:53 +02:00
|
|
|
case *config.LocalConnect:
|
|
|
|
connecter, errConnecter = LocalConnecterFromConfig(v)
|
|
|
|
connConf, errRPC = streamrpcconfig.FromDaemonConfig(g, v.RPC)
|
2018-08-27 22:21:45 +02:00
|
|
|
default:
|
|
|
|
panic(fmt.Sprintf("implementation error: unknown connecter type %T", v))
|
|
|
|
}
|
2018-08-31 21:51:44 +02:00
|
|
|
|
|
|
|
if errConnecter != nil {
|
|
|
|
return nil, errConnecter
|
|
|
|
}
|
|
|
|
if errRPC != nil {
|
|
|
|
return nil, errRPC
|
|
|
|
}
|
|
|
|
|
2018-09-05 02:25:10 +02:00
|
|
|
config := streamrpc.ClientConfig{ConnConfig: connConf}
|
2018-09-03 22:17:53 +02:00
|
|
|
if err := config.Validate(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2018-10-11 21:18:44 +02:00
|
|
|
|
|
|
|
connecter = HandshakeConnecter{connecter}
|
|
|
|
|
2018-09-03 22:17:53 +02:00
|
|
|
return &ClientFactory{connecter: connecter, config: &config}, nil
|
2018-08-31 21:51:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
type ClientFactory struct {
|
|
|
|
connecter streamrpc.Connecter
|
|
|
|
config *streamrpc.ClientConfig
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f ClientFactory) NewClient() (*streamrpc.Client, error) {
|
|
|
|
return streamrpc.NewClient(f.connecter, f.config)
|
2018-08-27 22:21:45 +02:00
|
|
|
}
|