2018-02-17 08:29:53 +03:00
|
|
|
package handshake
|
|
|
|
|
|
|
|
import (
|
|
|
|
"net"
|
|
|
|
|
|
|
|
"github.com/bifurcation/mint"
|
|
|
|
"github.com/lucas-clemente/quic-go/internal/utils"
|
|
|
|
)
|
|
|
|
|
2018-03-26 07:37:41 +03:00
|
|
|
// A CookieHandler generates and validates cookies.
|
|
|
|
// The cookie is sent in the TLS Retry.
|
|
|
|
// By including the cookie in its ClientHello, a client can proof ownership of its source address.
|
2018-02-17 08:29:53 +03:00
|
|
|
type CookieHandler struct {
|
|
|
|
callback func(net.Addr, *Cookie) bool
|
|
|
|
|
|
|
|
cookieGenerator *CookieGenerator
|
|
|
|
}
|
|
|
|
|
|
|
|
var _ mint.CookieHandler = &CookieHandler{}
|
|
|
|
|
2018-03-26 07:37:41 +03:00
|
|
|
// NewCookieHandler creates a new CookieHandler.
|
2018-02-17 08:29:53 +03:00
|
|
|
func NewCookieHandler(callback func(net.Addr, *Cookie) bool) (*CookieHandler, error) {
|
|
|
|
cookieGenerator, err := NewCookieGenerator()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &CookieHandler{
|
|
|
|
callback: callback,
|
|
|
|
cookieGenerator: cookieGenerator,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2018-03-26 07:37:41 +03:00
|
|
|
// Generate a new cookie for a mint connection.
|
2018-02-17 08:29:53 +03:00
|
|
|
func (h *CookieHandler) Generate(conn *mint.Conn) ([]byte, error) {
|
|
|
|
if h.callback(conn.RemoteAddr(), nil) {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
return h.cookieGenerator.NewToken(conn.RemoteAddr())
|
|
|
|
}
|
|
|
|
|
2018-03-26 07:37:41 +03:00
|
|
|
// Validate a cookie.
|
2018-02-17 08:29:53 +03:00
|
|
|
func (h *CookieHandler) Validate(conn *mint.Conn, token []byte) bool {
|
|
|
|
data, err := h.cookieGenerator.DecodeToken(token)
|
|
|
|
if err != nil {
|
|
|
|
utils.Debugf("Couldn't decode cookie from %s: %s", conn.RemoteAddr(), err.Error())
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return h.callback(conn.RemoteAddr(), data)
|
|
|
|
}
|