123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371 |
- package transport
- import (
- "crypto/tls"
- "net"
- "os"
- "time"
- "github.com/cloudflare/backoff"
- "github.com/cloudflare/cfssl/csr"
- "github.com/cloudflare/cfssl/errors"
- "github.com/cloudflare/cfssl/log"
- "github.com/cloudflare/cfssl/revoke"
- "github.com/cloudflare/cfssl/transport/ca"
- "github.com/cloudflare/cfssl/transport/core"
- "github.com/cloudflare/cfssl/transport/kp"
- "github.com/cloudflare/cfssl/transport/roots"
- )
- func envOrDefault(key, def string) string {
- val := os.Getenv(key)
- if val == "" {
- return def
- }
- return val
- }
- var (
- // NewKeyProvider is the function used to build key providers
- // from some identity.
- NewKeyProvider = func(id *core.Identity) (kp.KeyProvider, error) {
- return kp.NewStandardProvider(id)
- }
- // NewCA is used to load a configuration for a certificate
- // authority.
- NewCA = func(id *core.Identity) (ca.CertificateAuthority, error) {
- return ca.NewCFSSLProvider(id, nil)
- }
- )
- // A Transport is capable of providing transport-layer security using
- // TLS.
- type Transport struct {
- // Before defines how long before the certificate expires the
- // transport should start attempting to refresh the
- // certificate. For example, if this is 24h, then 24 hours
- // before the certificate expires the Transport will start
- // attempting to replace it.
- Before time.Duration
- // Provider contains a key management provider.
- Provider kp.KeyProvider
- // CA contains a mechanism for obtaining signed certificates.
- CA ca.CertificateAuthority
- // TrustStore contains the certificates trusted by this
- // transport.
- TrustStore *roots.TrustStore
- // ClientTrustStore contains the certificate authorities to
- // use in verifying client authentication certificates.
- ClientTrustStore *roots.TrustStore
- // Identity contains information about the entity that will be
- // used to construct certificates.
- Identity *core.Identity
- // Backoff is used to control the behaviour of a Transport
- // when it is attempting to automatically update a certificate
- // as part of AutoUpdate.
- Backoff *backoff.Backoff
- // RevokeSoftFail, if true, will cause a failure to check
- // revocation (such that the revocation status of a
- // certificate cannot be checked) to not be treated as an
- // error.
- RevokeSoftFail bool
- }
- // TLSClientAuthClientConfig returns a new client authentication TLS
- // configuration that can be used for a client using client auth
- // connecting to the named host.
- func (tr *Transport) TLSClientAuthClientConfig(host string) (*tls.Config, error) {
- cert, err := tr.getCertificate()
- if err != nil {
- return nil, err
- }
- return &tls.Config{
- Certificates: []tls.Certificate{cert},
- RootCAs: tr.TrustStore.Pool(),
- ServerName: host,
- CipherSuites: core.CipherSuites,
- MinVersion: tls.VersionTLS12,
- ClientAuth: tls.RequireAndVerifyClientCert,
- }, nil
- }
- // TLSClientAuthServerConfig returns a new client authentication TLS
- // configuration for servers expecting mutually authenticated
- // clients. The clientAuth parameter should contain the root pool used
- // to authenticate clients.
- func (tr *Transport) TLSClientAuthServerConfig() (*tls.Config, error) {
- cert, err := tr.getCertificate()
- if err != nil {
- return nil, err
- }
- return &tls.Config{
- Certificates: []tls.Certificate{cert},
- RootCAs: tr.TrustStore.Pool(),
- ClientCAs: tr.ClientTrustStore.Pool(),
- ClientAuth: tls.RequireAndVerifyClientCert,
- CipherSuites: core.CipherSuites,
- MinVersion: tls.VersionTLS12,
- }, nil
- }
- // TLSServerConfig is a general server configuration that should be
- // used for non-client authentication purposes, such as HTTPS.
- func (tr *Transport) TLSServerConfig() (*tls.Config, error) {
- cert, err := tr.getCertificate()
- if err != nil {
- return nil, err
- }
- return &tls.Config{
- Certificates: []tls.Certificate{cert},
- CipherSuites: core.CipherSuites,
- MinVersion: tls.VersionTLS12,
- }, nil
- }
- // New builds a new transport from an identity and a before time. The
- // before time tells the transport how long before the certificate
- // expires to start attempting to update when auto-updating. If before
- // is longer than the certificate's lifetime, every update check will
- // trigger a new certificate to be generated.
- func New(before time.Duration, identity *core.Identity) (*Transport, error) {
- var tr = &Transport{
- Before: before,
- Identity: identity,
- Backoff: &backoff.Backoff{},
- }
- store, err := roots.New(identity.Roots)
- if err != nil {
- return nil, err
- }
- tr.TrustStore = store
- if len(identity.ClientRoots) > 0 {
- store, err = roots.New(identity.ClientRoots)
- if err != nil {
- return nil, err
- }
- tr.ClientTrustStore = store
- }
- tr.Provider, err = NewKeyProvider(identity)
- if err != nil {
- return nil, err
- }
- tr.CA, err = NewCA(identity)
- if err != nil {
- return nil, err
- }
- return tr, nil
- }
- // Lifespan returns how much time is left before the transport's
- // certificate expires, or 0 if the certificate is not present or
- // expired.
- func (tr *Transport) Lifespan() time.Duration {
- cert := tr.Provider.Certificate()
- if cert == nil {
- return 0
- }
- now := time.Now()
- if now.After(cert.NotAfter) {
- return 0
- }
- now = now.Add(tr.Before)
- ls := cert.NotAfter.Sub(now)
- log.Debugf(" LIFESPAN:\t%s", ls)
- if ls < 0 {
- return 0
- }
- return ls
- }
- // RefreshKeys will make sure the Transport has loaded keys and has a
- // valid certificate. It will handle any persistence, check that the
- // certificate is valid (i.e. that its expiry date is within the
- // Before date), and handle certificate reissuance as needed.
- func (tr *Transport) RefreshKeys() (err error) {
- if !tr.Provider.Ready() {
- log.Debug("key and certificate aren't ready, loading")
- err = tr.Provider.Load()
- if err != nil && err != kp.ErrCertificateUnavailable {
- log.Debugf("failed to load keypair: %v", err)
- kr := tr.Identity.Request.KeyRequest
- if kr == nil {
- kr = csr.NewKeyRequest()
- }
- err = tr.Provider.Generate(kr.Algo(), kr.Size())
- if err != nil {
- log.Debugf("failed to generate key: %v", err)
- return err
- }
- }
- }
- lifespan := tr.Lifespan()
- if lifespan < tr.Before {
- log.Debugf("transport's certificate is out of date (lifespan %s)", lifespan)
- req, err := tr.Provider.CertificateRequest(tr.Identity.Request)
- if err != nil {
- log.Debugf("couldn't get a CSR: %v", err)
- if tr.Provider.SignalFailure(err) {
- return tr.RefreshKeys()
- }
- return err
- }
- log.Debug("requesting certificate from CA")
- cert, err := tr.CA.SignCSR(req)
- if err != nil {
- if tr.Provider.SignalFailure(err) {
- return tr.RefreshKeys()
- }
- log.Debugf("failed to get the certificate signed: %v", err)
- return err
- }
- log.Debug("giving the certificate to the provider")
- err = tr.Provider.SetCertificatePEM(cert)
- if err != nil {
- log.Debugf("failed to set the provider's certificate: %v", err)
- if tr.Provider.SignalFailure(err) {
- return tr.RefreshKeys()
- }
- return err
- }
- if tr.Provider.Persistent() {
- log.Debug("storing the certificate")
- err = tr.Provider.Store()
- if err != nil {
- log.Debugf("the provider failed to store the certificate: %v", err)
- if tr.Provider.SignalFailure(err) {
- return tr.RefreshKeys()
- }
- return err
- }
- }
- }
- return nil
- }
- func (tr *Transport) getCertificate() (cert tls.Certificate, err error) {
- if !tr.Provider.Ready() {
- log.Debug("transport isn't ready; attempting to refresh keypair")
- err = tr.RefreshKeys()
- if err != nil {
- log.Debugf("transport couldn't get a certificate: %v", err)
- return
- }
- }
- cert, err = tr.Provider.X509KeyPair()
- if err != nil {
- log.Debugf("couldn't generate an X.509 keypair: %v", err)
- }
- return
- }
- // Dial initiates a TLS connection to an outbound server. It returns a
- // TLS connection to the server.
- func Dial(address string, tr *Transport) (*tls.Conn, error) {
- host, _, err := net.SplitHostPort(address)
- if err != nil {
- // Assume address is a hostname, and that it should
- // use the HTTPS port number.
- host = address
- address = net.JoinHostPort(address, "443")
- }
- cfg, err := tr.TLSClientAuthClientConfig(host)
- if err != nil {
- return nil, err
- }
- conn, err := tls.Dial("tcp", address, cfg)
- if err != nil {
- return nil, err
- }
- state := conn.ConnectionState()
- if len(state.VerifiedChains) == 0 {
- return nil, errors.New(errors.CertificateError, errors.VerifyFailed)
- }
- for _, chain := range state.VerifiedChains {
- for _, cert := range chain {
- revoked, ok := revoke.VerifyCertificate(cert)
- if (!tr.RevokeSoftFail && !ok) || revoked {
- return nil, errors.New(errors.CertificateError, errors.VerifyFailed)
- }
- }
- }
- return conn, nil
- }
- // AutoUpdate will automatically update the listener. If a non-nil
- // certUpdates chan is provided, it will receive timestamps for
- // reissued certificates. If errChan is non-nil, any errors that occur
- // in the updater will be passed along.
- func (tr *Transport) AutoUpdate(certUpdates chan<- time.Time, errChan chan<- error) {
- defer func() {
- if r := recover(); r != nil {
- log.Criticalf("AutoUpdate panicked: %v", r)
- }
- }()
- for {
- // Wait until it's time to update the certificate.
- target := time.Now().Add(tr.Lifespan())
- if PollInterval == 0 {
- <-time.After(tr.Lifespan())
- } else {
- pollWait(target)
- }
- // Keep trying to update the certificate until it's
- // ready.
- for {
- log.Debugf("attempting to refresh keypair")
- err := tr.RefreshKeys()
- if err == nil {
- break
- }
- delay := tr.Backoff.Duration()
- log.Debugf("failed to update certificate, will try again in %s", delay)
- if errChan != nil {
- errChan <- err
- }
- <-time.After(delay)
- }
- log.Debugf("certificate updated")
- if certUpdates != nil {
- certUpdates <- time.Now()
- }
- tr.Backoff.Reset()
- }
- }
|