2018-08-27 22:21:45 +02:00
|
|
|
package serve
|
|
|
|
|
|
|
|
import (
|
|
|
|
"github.com/problame/go-netssh"
|
|
|
|
"github.com/zrepl/zrepl/config"
|
|
|
|
"github.com/zrepl/zrepl/daemon/nethelpers"
|
|
|
|
"io"
|
|
|
|
"net"
|
|
|
|
"path"
|
|
|
|
"time"
|
2018-09-05 01:41:54 +02:00
|
|
|
"context"
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
"sync/atomic"
|
2018-08-27 22:21:45 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
type StdinserverListenerFactory struct {
|
2018-09-05 01:41:54 +02:00
|
|
|
ClientIdentities []string
|
|
|
|
Sockdir string
|
2018-08-27 22:21:45 +02:00
|
|
|
}
|
|
|
|
|
2018-09-05 01:41:54 +02:00
|
|
|
func MultiStdinserverListenerFactoryFromConfig(g *config.Global, in *config.StdinserverServer) (f *multiStdinserverListenerFactory, err error) {
|
2018-08-27 22:21:45 +02:00
|
|
|
|
2018-09-05 01:41:54 +02:00
|
|
|
for _, ci := range in.ClientIdentities {
|
|
|
|
if err := ValidateClientIdentity(ci); err != nil {
|
|
|
|
return nil, errors.Wrapf(err, "invalid client identity %q", ci)
|
|
|
|
}
|
2018-08-27 22:21:45 +02:00
|
|
|
}
|
|
|
|
|
2018-09-05 01:41:54 +02:00
|
|
|
f = &multiStdinserverListenerFactory{
|
|
|
|
ClientIdentities: in.ClientIdentities,
|
|
|
|
Sockdir: g.Serve.StdinServer.SockDir,
|
|
|
|
}
|
2018-08-27 22:21:45 +02:00
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2018-09-05 01:41:54 +02:00
|
|
|
type multiStdinserverListenerFactory struct {
|
|
|
|
ClientIdentities []string
|
|
|
|
Sockdir string
|
|
|
|
}
|
2018-08-27 22:21:45 +02:00
|
|
|
|
2018-09-05 01:41:54 +02:00
|
|
|
func (f *multiStdinserverListenerFactory) Listen() (AuthenticatedListener, error) {
|
|
|
|
return multiStdinserverListenerFromClientIdentities(f.Sockdir, f.ClientIdentities)
|
|
|
|
}
|
2018-08-27 22:21:45 +02:00
|
|
|
|
2018-09-05 01:41:54 +02:00
|
|
|
type multiStdinserverAcceptRes struct {
|
|
|
|
conn AuthenticatedConn
|
|
|
|
err error
|
|
|
|
}
|
|
|
|
|
|
|
|
type MultiStdinserverListener struct {
|
|
|
|
listeners []*stdinserverListener
|
|
|
|
accepts chan multiStdinserverAcceptRes
|
|
|
|
closed int32
|
|
|
|
}
|
|
|
|
|
|
|
|
// client identities must be validated
|
|
|
|
func multiStdinserverListenerFromClientIdentities(sockdir string, cis []string) (*MultiStdinserverListener, error) {
|
|
|
|
listeners := make([]*stdinserverListener, 0, len(cis))
|
|
|
|
var err error
|
|
|
|
for _, ci := range cis {
|
|
|
|
sockpath := path.Join(sockdir, ci)
|
|
|
|
l := &stdinserverListener{clientIdentity: ci}
|
|
|
|
if err = nethelpers.PreparePrivateSockpath(sockpath); err != nil {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
if l.l, err = netssh.Listen(sockpath); err != nil {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
listeners = append(listeners, l)
|
|
|
|
}
|
2018-08-27 22:21:45 +02:00
|
|
|
if err != nil {
|
2018-09-05 01:41:54 +02:00
|
|
|
for _, l := range listeners {
|
|
|
|
l.Close() // FIXME error reporting?
|
|
|
|
}
|
2018-08-27 22:21:45 +02:00
|
|
|
return nil, err
|
|
|
|
}
|
2018-09-05 01:41:54 +02:00
|
|
|
return &MultiStdinserverListener{listeners: listeners}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *MultiStdinserverListener) Accept(ctx context.Context) (AuthenticatedConn, error){
|
|
|
|
|
|
|
|
if m.accepts == nil {
|
|
|
|
m.accepts = make(chan multiStdinserverAcceptRes, len(m.listeners))
|
|
|
|
for i := range m.listeners {
|
|
|
|
go func(i int) {
|
|
|
|
for atomic.LoadInt32(&m.closed) == 0 {
|
|
|
|
conn, err := m.listeners[i].Accept(context.TODO())
|
|
|
|
m.accepts <- multiStdinserverAcceptRes{conn, err}
|
|
|
|
}
|
|
|
|
}(i)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
res := <- m.accepts
|
|
|
|
return res.conn, res.err
|
|
|
|
|
2018-08-27 22:21:45 +02:00
|
|
|
}
|
|
|
|
|
2018-09-05 01:41:54 +02:00
|
|
|
func (m *MultiStdinserverListener) Addr() (net.Addr) {
|
|
|
|
return netsshAddr{}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *MultiStdinserverListener) Close() error {
|
|
|
|
atomic.StoreInt32(&m.closed, 1)
|
|
|
|
var oneErr error
|
|
|
|
for _, l := range m.listeners {
|
|
|
|
if err := l.Close(); err != nil && oneErr == nil {
|
|
|
|
oneErr = err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return oneErr
|
|
|
|
}
|
|
|
|
|
|
|
|
// a single stdinserverListener (part of multiStinserverListener)
|
|
|
|
type stdinserverListener struct {
|
2018-08-27 22:21:45 +02:00
|
|
|
l *netssh.Listener
|
2018-09-05 01:41:54 +02:00
|
|
|
clientIdentity string
|
2018-08-27 22:21:45 +02:00
|
|
|
}
|
|
|
|
|
2018-09-05 01:41:54 +02:00
|
|
|
func (l stdinserverListener) Addr() net.Addr {
|
2018-08-27 22:21:45 +02:00
|
|
|
return netsshAddr{}
|
|
|
|
}
|
|
|
|
|
2018-09-05 01:41:54 +02:00
|
|
|
func (l stdinserverListener) Accept(ctx context.Context) (AuthenticatedConn, error) {
|
2018-08-27 22:21:45 +02:00
|
|
|
c, err := l.l.Accept()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2018-09-05 01:41:54 +02:00
|
|
|
return netsshConnToNetConnAdatper{c, l.clientIdentity}, nil
|
2018-08-27 22:21:45 +02:00
|
|
|
}
|
|
|
|
|
2018-09-05 01:41:54 +02:00
|
|
|
func (l stdinserverListener) Close() (err error) {
|
2018-08-27 22:21:45 +02:00
|
|
|
return l.l.Close()
|
|
|
|
}
|
|
|
|
|
|
|
|
type netsshAddr struct{}
|
|
|
|
|
|
|
|
func (netsshAddr) Network() string { return "netssh" }
|
|
|
|
func (netsshAddr) String() string { return "???" }
|
|
|
|
|
|
|
|
type netsshConnToNetConnAdatper struct {
|
|
|
|
io.ReadWriteCloser // works for both netssh.SSHConn and netssh.ServeConn
|
2018-09-05 01:41:54 +02:00
|
|
|
clientIdentity string
|
2018-08-27 22:21:45 +02:00
|
|
|
}
|
|
|
|
|
2018-09-05 01:41:54 +02:00
|
|
|
func (a netsshConnToNetConnAdatper) ClientIdentity() string { return a.clientIdentity }
|
|
|
|
|
2018-08-27 22:21:45 +02:00
|
|
|
func (netsshConnToNetConnAdatper) LocalAddr() net.Addr { return netsshAddr{} }
|
|
|
|
|
|
|
|
func (netsshConnToNetConnAdatper) RemoteAddr() net.Addr { return netsshAddr{} }
|
|
|
|
|
2018-09-05 01:41:54 +02:00
|
|
|
// FIXME log warning once!
|
2018-08-27 22:21:45 +02:00
|
|
|
func (netsshConnToNetConnAdatper) SetDeadline(t time.Time) error { return nil }
|
|
|
|
|
|
|
|
func (netsshConnToNetConnAdatper) SetReadDeadline(t time.Time) error { return nil }
|
|
|
|
|
|
|
|
func (netsshConnToNetConnAdatper) SetWriteDeadline(t time.Time) error { return nil }
|