mirror of
https://github.com/caddyserver/caddy.git
synced 2025-01-22 16:46:53 +01:00
214 lines
6.2 KiB
Go
214 lines
6.2 KiB
Go
package caddytls
|
|
|
|
import (
|
|
"crypto/rand"
|
|
"crypto/tls"
|
|
"encoding/json"
|
|
"fmt"
|
|
"io"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/caddyserver/caddy2"
|
|
)
|
|
|
|
// SessionTicketService configures and manages TLS session tickets.
|
|
type SessionTicketService struct {
|
|
KeySource json.RawMessage `json:"key_source,omitempty"`
|
|
RotationInterval caddy2.Duration `json:"rotation_interval,omitempty"`
|
|
MaxKeys int `json:"max_keys,omitempty"`
|
|
DisableRotation bool `json:"disable_rotation,omitempty"`
|
|
Disabled bool `json:"disabled,omitempty"`
|
|
|
|
keySource STEKProvider
|
|
configs map[*tls.Config]struct{}
|
|
stopChan chan struct{}
|
|
currentKeys [][32]byte
|
|
mu *sync.Mutex
|
|
}
|
|
|
|
func (s *SessionTicketService) provision(ctx caddy2.Context) error {
|
|
s.configs = make(map[*tls.Config]struct{})
|
|
s.mu = new(sync.Mutex)
|
|
|
|
// establish sane defaults
|
|
if s.RotationInterval == 0 {
|
|
s.RotationInterval = caddy2.Duration(defaultSTEKRotationInterval)
|
|
}
|
|
if s.MaxKeys <= 0 {
|
|
s.MaxKeys = defaultMaxSTEKs
|
|
}
|
|
if s.KeySource == nil {
|
|
s.KeySource = json.RawMessage(`{"provider":"standard"}`)
|
|
}
|
|
|
|
// load the STEK module, which will provide keys
|
|
val, err := ctx.LoadModuleInline("provider", "tls.stek", s.KeySource)
|
|
if err != nil {
|
|
return fmt.Errorf("loading TLS session ticket ephemeral keys provider module: %s", err)
|
|
}
|
|
s.keySource = val.(STEKProvider)
|
|
s.KeySource = nil // allow GC to deallocate - TODO: Does this help?
|
|
|
|
// if session tickets or just rotation are
|
|
// disabled, no need to start service
|
|
if s.Disabled || s.DisableRotation {
|
|
return nil
|
|
}
|
|
|
|
// start the STEK module; this ensures we have
|
|
// a starting key before any config needs one
|
|
return s.start()
|
|
}
|
|
|
|
// start loads the starting STEKs and spawns a goroutine
|
|
// which loops to rotate the STEKs, which continues until
|
|
// stop() is called. If start() was already called, this
|
|
// is a no-op.
|
|
func (s *SessionTicketService) start() error {
|
|
if s.stopChan != nil {
|
|
return nil
|
|
}
|
|
s.stopChan = make(chan struct{})
|
|
|
|
// initializing the key source gives us our
|
|
// initial key(s) to start with; if successful,
|
|
// we need to be sure to call Next() so that
|
|
// the key source can know when it is done
|
|
initialKeys, err := s.keySource.Initialize(s)
|
|
if err != nil {
|
|
return fmt.Errorf("setting STEK module configuration: %v", err)
|
|
}
|
|
|
|
s.mu.Lock()
|
|
s.currentKeys = initialKeys
|
|
s.mu.Unlock()
|
|
|
|
// keep the keys rotated
|
|
go s.stayUpdated()
|
|
|
|
return nil
|
|
}
|
|
|
|
// stayUpdated is a blocking function which rotates
|
|
// the keys whenever new ones are sent. It reads
|
|
// from keysChan until s.stop() is called.
|
|
func (s *SessionTicketService) stayUpdated() {
|
|
// this call is essential when Initialize()
|
|
// returns without error, because the stop
|
|
// channel is the only way the key source
|
|
// will know when to clean up
|
|
keysChan := s.keySource.Next(s.stopChan)
|
|
|
|
for {
|
|
select {
|
|
case newKeys := <-keysChan:
|
|
s.mu.Lock()
|
|
s.currentKeys = newKeys
|
|
configs := s.configs
|
|
s.mu.Unlock()
|
|
for cfg := range configs {
|
|
cfg.SetSessionTicketKeys(newKeys)
|
|
}
|
|
case <-s.stopChan:
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
// stop terminates the key rotation goroutine.
|
|
func (s *SessionTicketService) stop() {
|
|
if s.stopChan != nil {
|
|
close(s.stopChan)
|
|
}
|
|
}
|
|
|
|
// register sets the session ticket keys on cfg
|
|
// and keeps them updated. Any values registered
|
|
// must be unregistered, or they will not be
|
|
// garbage-collected. s.start() must have been
|
|
// called first. If session tickets are disabled
|
|
// or if ticket key rotation is disabled, this
|
|
// function is a no-op.
|
|
func (s *SessionTicketService) register(cfg *tls.Config) {
|
|
if s.Disabled || s.DisableRotation {
|
|
return
|
|
}
|
|
s.mu.Lock()
|
|
cfg.SetSessionTicketKeys(s.currentKeys)
|
|
s.configs[cfg] = struct{}{}
|
|
s.mu.Unlock()
|
|
}
|
|
|
|
// unregister stops session key management on cfg and
|
|
// removes the internal stored reference to cfg. If
|
|
// session tickets are disabled or if ticket key rotation
|
|
// is disabled, this function is a no-op.
|
|
func (s *SessionTicketService) unregister(cfg *tls.Config) {
|
|
if s.Disabled || s.DisableRotation {
|
|
return
|
|
}
|
|
s.mu.Lock()
|
|
delete(s.configs, cfg)
|
|
s.mu.Unlock()
|
|
}
|
|
|
|
// RotateSTEKs rotates the keys in keys by producing a new key and eliding
|
|
// the oldest one. The new slice of keys is returned.
|
|
func (s SessionTicketService) RotateSTEKs(keys [][32]byte) ([][32]byte, error) {
|
|
// produce a new key
|
|
newKey, err := s.generateSTEK()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("generating STEK: %v", err)
|
|
}
|
|
|
|
// we need to prepend this new key to the list of
|
|
// keys so that it is preferred, but we need to be
|
|
// careful that we do not grow the slice larger
|
|
// than MaxKeys, otherwise we'll be storing one
|
|
// more key in memory than we expect; so be sure
|
|
// that the slice does not grow beyond the limit
|
|
// even for a brief period of time, since there's
|
|
// no guarantee when that extra allocation will
|
|
// be overwritten; this is why we first trim the
|
|
// length to one less the max, THEN prepend the
|
|
// new key
|
|
if len(keys) >= s.MaxKeys {
|
|
keys[len(keys)-1] = [32]byte{} // zero-out memory of oldest key
|
|
keys = keys[:s.MaxKeys-1] // trim length of slice
|
|
}
|
|
keys = append([][32]byte{newKey}, keys...) // prepend new key
|
|
|
|
return keys, nil
|
|
}
|
|
|
|
// generateSTEK generates key material suitable for use as a
|
|
// session ticket ephemeral key.
|
|
func (s *SessionTicketService) generateSTEK() ([32]byte, error) {
|
|
var newTicketKey [32]byte
|
|
_, err := io.ReadFull(rand.Reader, newTicketKey[:])
|
|
return newTicketKey, err
|
|
}
|
|
|
|
// STEKProvider is a type that can provide session ticket ephemeral
|
|
// keys (STEKs).
|
|
type STEKProvider interface {
|
|
// Initialize provides the STEK configuration to the STEK
|
|
// module so that it can obtain and manage keys accordingly.
|
|
// It returns the initial key(s) to use. Implementations can
|
|
// rely on Next() being called if Initialize() returns
|
|
// without error, so that it may know when it is done.
|
|
Initialize(config *SessionTicketService) ([][32]byte, error)
|
|
|
|
// Next returns the channel through which the next session
|
|
// ticket keys will be transmitted until doneChan is closed.
|
|
// Keys should be sent on keysChan as they are updated.
|
|
// When doneChan is closed, any resources allocated in
|
|
// Initialize() must be cleaned up.
|
|
Next(doneChan <-chan struct{}) (keysChan <-chan [][32]byte)
|
|
}
|
|
|
|
const (
|
|
defaultSTEKRotationInterval = 12 * time.Hour
|
|
defaultMaxSTEKs = 4
|
|
)
|