2018-08-25 12:58:17 +02:00
|
|
|
package cmd
|
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto/tls"
|
|
|
|
"crypto/x509"
|
|
|
|
"net"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/mitchellh/mapstructure"
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
"github.com/zrepl/zrepl/cmd/tlsconf"
|
|
|
|
)
|
|
|
|
|
|
|
|
type TCPListenerFactory struct {
|
|
|
|
Address string
|
|
|
|
tls bool
|
|
|
|
clientCA *x509.CertPool
|
|
|
|
serverCert tls.Certificate
|
|
|
|
clientCommonName string
|
|
|
|
}
|
|
|
|
|
|
|
|
func parseTCPListenerFactory(c JobParsingContext, i map[string]interface{}) (*TCPListenerFactory, error) {
|
|
|
|
|
|
|
|
var in struct {
|
2018-08-25 21:30:25 +02:00
|
|
|
Address string
|
|
|
|
TLS map[string]interface{}
|
2018-08-25 12:58:17 +02:00
|
|
|
}
|
|
|
|
if err := mapstructure.Decode(i, &in); err != nil {
|
|
|
|
return nil, errors.Wrap(err, "mapstructure error")
|
|
|
|
}
|
|
|
|
|
|
|
|
lf := &TCPListenerFactory{}
|
|
|
|
|
|
|
|
if in.Address == "" {
|
|
|
|
return nil, errors.New("must specify field 'address'")
|
|
|
|
}
|
|
|
|
lf.Address = in.Address
|
|
|
|
|
|
|
|
if in.TLS != nil {
|
|
|
|
err := func(i map[string]interface{}) (err error) {
|
|
|
|
var in struct {
|
|
|
|
CA string
|
|
|
|
Cert string
|
|
|
|
Key string
|
|
|
|
ClientCN string `mapstructure:"client_cn"`
|
|
|
|
}
|
|
|
|
if err := mapstructure.Decode(i, &in); err != nil {
|
|
|
|
return errors.Wrap(err, "mapstructure error")
|
|
|
|
}
|
|
|
|
|
|
|
|
if in.CA == "" || in.Cert == "" || in.Key == "" || in.ClientCN == "" {
|
|
|
|
return errors.New("fields 'ca', 'cert', 'key' and 'client_cn' must be specified")
|
|
|
|
}
|
|
|
|
|
|
|
|
lf.clientCommonName = in.ClientCN
|
|
|
|
|
|
|
|
lf.clientCA, err = tlsconf.ParseCAFile(in.CA)
|
|
|
|
if err != nil {
|
2018-08-25 21:30:25 +02:00
|
|
|
return errors.Wrap(err, "cannot parse ca file")
|
2018-08-25 12:58:17 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
lf.serverCert, err = tls.LoadX509KeyPair(in.Cert, in.Key)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "cannot parse cer/key pair")
|
|
|
|
}
|
|
|
|
|
|
|
|
lf.tls = true // mark success
|
|
|
|
return nil
|
|
|
|
}(in.TLS)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "error parsing TLS config in field 'tls'")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return lf, nil
|
|
|
|
}
|
|
|
|
|
2018-08-25 22:30:16 +02:00
|
|
|
var TCPListenerHandshakeTimeout = 10 * time.Second // FIXME make configurable
|
|
|
|
|
2018-08-25 12:58:17 +02:00
|
|
|
func (f *TCPListenerFactory) Listen() (net.Listener, error) {
|
|
|
|
l, err := net.Listen("tcp", f.Address)
|
|
|
|
if !f.tls || err != nil {
|
|
|
|
return l, err
|
|
|
|
}
|
|
|
|
|
2018-08-25 22:30:16 +02:00
|
|
|
tl := tlsconf.NewClientAuthListener(l, f.clientCA, f.serverCert, f.clientCommonName, TCPListenerHandshakeTimeout)
|
2018-08-25 12:58:17 +02:00
|
|
|
return tl, nil
|
|
|
|
}
|