2018-11-14 10:18:03 +01:00
|
|
|
package passtlsclientcert
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"crypto/x509"
|
|
|
|
"crypto/x509/pkix"
|
|
|
|
"encoding/pem"
|
|
|
|
"fmt"
|
2019-03-04 16:40:05 +01:00
|
|
|
"io"
|
2018-11-14 10:18:03 +01:00
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
"strings"
|
|
|
|
|
2019-08-03 03:58:23 +02:00
|
|
|
"github.com/containous/traefik/v2/pkg/config/dynamic"
|
|
|
|
"github.com/containous/traefik/v2/pkg/log"
|
|
|
|
"github.com/containous/traefik/v2/pkg/middlewares"
|
|
|
|
"github.com/containous/traefik/v2/pkg/tracing"
|
2018-11-14 10:18:03 +01:00
|
|
|
"github.com/opentracing/opentracing-go/ext"
|
|
|
|
"github.com/sirupsen/logrus"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2019-01-09 11:28:04 +01:00
|
|
|
xForwardedTLSClientCert = "X-Forwarded-Tls-Client-Cert"
|
2019-02-26 05:50:07 -08:00
|
|
|
xForwardedTLSClientCertInfo = "X-Forwarded-Tls-Client-Cert-Info"
|
2019-01-09 11:28:04 +01:00
|
|
|
typeName = "PassClientTLSCert"
|
2018-11-14 10:18:03 +01:00
|
|
|
)
|
|
|
|
|
2019-01-09 11:28:04 +01:00
|
|
|
var attributeTypeNames = map[string]string{
|
|
|
|
"0.9.2342.19200300.100.1.25": "DC", // Domain component OID - RFC 2247
|
|
|
|
}
|
|
|
|
|
|
|
|
// DistinguishedNameOptions is a struct for specifying the configuration for the distinguished name info.
|
|
|
|
type DistinguishedNameOptions struct {
|
|
|
|
CommonName bool
|
|
|
|
CountryName bool
|
|
|
|
DomainComponent bool
|
|
|
|
LocalityName bool
|
|
|
|
OrganizationName bool
|
|
|
|
SerialNumber bool
|
|
|
|
StateOrProvinceName bool
|
|
|
|
}
|
|
|
|
|
2019-07-10 09:26:04 +02:00
|
|
|
func newDistinguishedNameOptions(info *dynamic.TLSCLientCertificateDNInfo) *DistinguishedNameOptions {
|
2019-01-09 11:28:04 +01:00
|
|
|
if info == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return &DistinguishedNameOptions{
|
|
|
|
CommonName: info.CommonName,
|
|
|
|
CountryName: info.Country,
|
|
|
|
DomainComponent: info.DomainComponent,
|
|
|
|
LocalityName: info.Locality,
|
|
|
|
OrganizationName: info.Organization,
|
|
|
|
SerialNumber: info.SerialNumber,
|
|
|
|
StateOrProvinceName: info.Province,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-14 10:18:03 +01:00
|
|
|
// passTLSClientCert is a middleware that helps setup a few tls info features.
|
|
|
|
type passTLSClientCert struct {
|
2019-01-09 11:28:04 +01:00
|
|
|
next http.Handler
|
|
|
|
name string
|
|
|
|
pem bool // pass the sanitized pem to the backend in a specific header
|
|
|
|
info *tlsClientCertificateInfo // pass selected information from the client certificate
|
2018-11-14 10:18:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// New constructs a new PassTLSClientCert instance from supplied frontend header struct.
|
2019-07-10 09:26:04 +02:00
|
|
|
func New(ctx context.Context, next http.Handler, config dynamic.PassTLSClientCert, name string) (http.Handler, error) {
|
2018-11-14 10:18:03 +01:00
|
|
|
middlewares.GetLogger(ctx, name, typeName).Debug("Creating middleware")
|
|
|
|
|
|
|
|
return &passTLSClientCert{
|
2019-01-09 11:28:04 +01:00
|
|
|
next: next,
|
|
|
|
name: name,
|
|
|
|
pem: config.PEM,
|
|
|
|
info: newTLSClientInfo(config.Info),
|
2018-11-14 10:18:03 +01:00
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2019-01-09 11:28:04 +01:00
|
|
|
// tlsClientCertificateInfo is a struct for specifying the configuration for the passTLSClientCert middleware.
|
|
|
|
type tlsClientCertificateInfo struct {
|
2018-11-14 10:18:03 +01:00
|
|
|
notAfter bool
|
|
|
|
notBefore bool
|
|
|
|
sans bool
|
2019-01-09 11:28:04 +01:00
|
|
|
subject *DistinguishedNameOptions
|
|
|
|
issuer *DistinguishedNameOptions
|
2018-11-14 10:18:03 +01:00
|
|
|
}
|
|
|
|
|
2019-07-10 09:26:04 +02:00
|
|
|
func newTLSClientInfo(info *dynamic.TLSClientCertificateInfo) *tlsClientCertificateInfo {
|
2019-01-09 11:28:04 +01:00
|
|
|
if info == nil {
|
2018-11-14 10:18:03 +01:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-01-09 11:28:04 +01:00
|
|
|
return &tlsClientCertificateInfo{
|
|
|
|
issuer: newDistinguishedNameOptions(info.Issuer),
|
|
|
|
notAfter: info.NotAfter,
|
|
|
|
notBefore: info.NotBefore,
|
|
|
|
subject: newDistinguishedNameOptions(info.Subject),
|
|
|
|
sans: info.Sans,
|
2018-11-14 10:18:03 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *passTLSClientCert) GetTracingInformation() (string, ext.SpanKindEnum) {
|
|
|
|
return p.name, tracing.SpanKindNoneEnum
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *passTLSClientCert) ServeHTTP(rw http.ResponseWriter, req *http.Request) {
|
|
|
|
logger := middlewares.GetLogger(req.Context(), p.name, typeName)
|
|
|
|
p.modifyRequestHeaders(logger, req)
|
|
|
|
p.next.ServeHTTP(rw, req)
|
|
|
|
}
|
2019-01-09 11:28:04 +01:00
|
|
|
func getDNInfo(prefix string, options *DistinguishedNameOptions, cs *pkix.Name) string {
|
|
|
|
if options == nil {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
content := &strings.Builder{}
|
|
|
|
|
|
|
|
// Manage non standard attributes
|
|
|
|
for _, name := range cs.Names {
|
|
|
|
// Domain Component - RFC 2247
|
|
|
|
if options.DomainComponent && attributeTypeNames[name.Type.String()] == "DC" {
|
|
|
|
content.WriteString(fmt.Sprintf("DC=%s,", name.Value))
|
|
|
|
}
|
|
|
|
}
|
2018-11-14 10:18:03 +01:00
|
|
|
|
2019-01-09 11:28:04 +01:00
|
|
|
if options.CountryName {
|
|
|
|
writeParts(content, cs.Country, "C")
|
|
|
|
}
|
2018-11-14 10:18:03 +01:00
|
|
|
|
2019-01-09 11:28:04 +01:00
|
|
|
if options.StateOrProvinceName {
|
|
|
|
writeParts(content, cs.Province, "ST")
|
|
|
|
}
|
2018-11-14 10:18:03 +01:00
|
|
|
|
2019-01-09 11:28:04 +01:00
|
|
|
if options.LocalityName {
|
|
|
|
writeParts(content, cs.Locality, "L")
|
|
|
|
}
|
2018-11-14 10:18:03 +01:00
|
|
|
|
2019-01-09 11:28:04 +01:00
|
|
|
if options.OrganizationName {
|
|
|
|
writeParts(content, cs.Organization, "O")
|
|
|
|
}
|
2018-11-14 10:18:03 +01:00
|
|
|
|
2019-01-09 11:28:04 +01:00
|
|
|
if options.SerialNumber {
|
|
|
|
writePart(content, cs.SerialNumber, "SN")
|
|
|
|
}
|
2018-11-14 10:18:03 +01:00
|
|
|
|
2019-01-09 11:28:04 +01:00
|
|
|
if options.CommonName {
|
|
|
|
writePart(content, cs.CommonName, "CN")
|
|
|
|
}
|
2018-11-14 10:18:03 +01:00
|
|
|
|
2019-01-09 11:28:04 +01:00
|
|
|
if content.Len() > 0 {
|
|
|
|
return prefix + `="` + strings.TrimSuffix(content.String(), ",") + `"`
|
|
|
|
}
|
2018-11-14 10:18:03 +01:00
|
|
|
|
2019-01-09 11:28:04 +01:00
|
|
|
return ""
|
|
|
|
}
|
2018-11-14 10:18:03 +01:00
|
|
|
|
2019-03-04 16:40:05 +01:00
|
|
|
func writeParts(content io.StringWriter, entries []string, prefix string) {
|
2019-01-09 11:28:04 +01:00
|
|
|
for _, entry := range entries {
|
|
|
|
writePart(content, entry, prefix)
|
2018-11-14 10:18:03 +01:00
|
|
|
}
|
2019-01-09 11:28:04 +01:00
|
|
|
}
|
2018-11-14 10:18:03 +01:00
|
|
|
|
2019-03-04 16:40:05 +01:00
|
|
|
func writePart(content io.StringWriter, entry string, prefix string) {
|
2019-01-09 11:28:04 +01:00
|
|
|
if len(entry) > 0 {
|
2019-03-04 16:40:05 +01:00
|
|
|
_, err := content.WriteString(fmt.Sprintf("%s=%s,", prefix, entry))
|
|
|
|
if err != nil {
|
|
|
|
log.Error(err)
|
|
|
|
}
|
2019-01-09 11:28:04 +01:00
|
|
|
}
|
2018-11-14 10:18:03 +01:00
|
|
|
}
|
|
|
|
|
2019-01-09 11:28:04 +01:00
|
|
|
// getXForwardedTLSClientCertInfo Build a string with the wanted client certificates information
|
2018-11-14 10:18:03 +01:00
|
|
|
// like Subject="C=%s,ST=%s,L=%s,O=%s,CN=%s",NB=%d,NA=%d,SAN=%s;
|
2019-01-09 11:28:04 +01:00
|
|
|
func (p *passTLSClientCert) getXForwardedTLSClientCertInfo(certs []*x509.Certificate) string {
|
2018-11-14 10:18:03 +01:00
|
|
|
var headerValues []string
|
|
|
|
|
|
|
|
for _, peerCert := range certs {
|
|
|
|
var values []string
|
|
|
|
var sans string
|
|
|
|
var nb string
|
|
|
|
var na string
|
|
|
|
|
2019-01-09 11:28:04 +01:00
|
|
|
if p.info != nil {
|
|
|
|
subject := getDNInfo("Subject", p.info.subject, &peerCert.Subject)
|
|
|
|
if len(subject) > 0 {
|
|
|
|
values = append(values, subject)
|
|
|
|
}
|
|
|
|
|
|
|
|
issuer := getDNInfo("Issuer", p.info.issuer, &peerCert.Issuer)
|
|
|
|
if len(issuer) > 0 {
|
|
|
|
values = append(values, issuer)
|
|
|
|
}
|
2018-11-14 10:18:03 +01:00
|
|
|
}
|
|
|
|
|
2019-01-09 11:28:04 +01:00
|
|
|
ci := p.info
|
2018-11-14 10:18:03 +01:00
|
|
|
if ci != nil {
|
|
|
|
if ci.notBefore {
|
|
|
|
nb = fmt.Sprintf("NB=%d", uint64(peerCert.NotBefore.Unix()))
|
|
|
|
values = append(values, nb)
|
|
|
|
}
|
|
|
|
if ci.notAfter {
|
|
|
|
na = fmt.Sprintf("NA=%d", uint64(peerCert.NotAfter.Unix()))
|
|
|
|
values = append(values, na)
|
|
|
|
}
|
|
|
|
|
|
|
|
if ci.sans {
|
|
|
|
sans = fmt.Sprintf("SAN=%s", strings.Join(getSANs(peerCert), ","))
|
|
|
|
values = append(values, sans)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
value := strings.Join(values, ",")
|
|
|
|
headerValues = append(headerValues, value)
|
|
|
|
}
|
|
|
|
|
|
|
|
return strings.Join(headerValues, ";")
|
|
|
|
}
|
|
|
|
|
|
|
|
// modifyRequestHeaders set the wanted headers with the certificates information.
|
|
|
|
func (p *passTLSClientCert) modifyRequestHeaders(logger logrus.FieldLogger, r *http.Request) {
|
|
|
|
if p.pem {
|
|
|
|
if r.TLS != nil && len(r.TLS.PeerCertificates) > 0 {
|
|
|
|
r.Header.Set(xForwardedTLSClientCert, getXForwardedTLSClientCert(logger, r.TLS.PeerCertificates))
|
|
|
|
} else {
|
2019-07-12 17:50:04 +02:00
|
|
|
logger.Warn("Tried to extract a certificate on a request without mutual TLS")
|
2018-11-14 10:18:03 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-09 11:28:04 +01:00
|
|
|
if p.info != nil {
|
2018-11-14 10:18:03 +01:00
|
|
|
if r.TLS != nil && len(r.TLS.PeerCertificates) > 0 {
|
2019-01-09 11:28:04 +01:00
|
|
|
headerContent := p.getXForwardedTLSClientCertInfo(r.TLS.PeerCertificates)
|
|
|
|
r.Header.Set(xForwardedTLSClientCertInfo, url.QueryEscape(headerContent))
|
2018-11-14 10:18:03 +01:00
|
|
|
} else {
|
2019-07-12 17:50:04 +02:00
|
|
|
logger.Warn("Tried to extract a certificate on a request without mutual TLS")
|
2018-11-14 10:18:03 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// sanitize As we pass the raw certificates, remove the useless data and make it http request compliant.
|
|
|
|
func sanitize(cert []byte) string {
|
|
|
|
s := string(cert)
|
|
|
|
r := strings.NewReplacer("-----BEGIN CERTIFICATE-----", "",
|
|
|
|
"-----END CERTIFICATE-----", "",
|
|
|
|
"\n", "")
|
|
|
|
cleaned := r.Replace(s)
|
|
|
|
|
|
|
|
return url.QueryEscape(cleaned)
|
|
|
|
}
|
|
|
|
|
|
|
|
// extractCertificate extract the certificate from the request.
|
|
|
|
func extractCertificate(logger logrus.FieldLogger, cert *x509.Certificate) string {
|
|
|
|
b := pem.Block{Type: "CERTIFICATE", Bytes: cert.Raw}
|
|
|
|
certPEM := pem.EncodeToMemory(&b)
|
|
|
|
if certPEM == nil {
|
|
|
|
logger.Error("Cannot extract the certificate content")
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
return sanitize(certPEM)
|
|
|
|
}
|
|
|
|
|
|
|
|
// getXForwardedTLSClientCert Build a string with the client certificates.
|
|
|
|
func getXForwardedTLSClientCert(logger logrus.FieldLogger, certs []*x509.Certificate) string {
|
|
|
|
var headerValues []string
|
|
|
|
|
|
|
|
for _, peerCert := range certs {
|
|
|
|
headerValues = append(headerValues, extractCertificate(logger, peerCert))
|
|
|
|
}
|
|
|
|
|
|
|
|
return strings.Join(headerValues, ",")
|
|
|
|
}
|
|
|
|
|
|
|
|
// getSANs get the Subject Alternate Name values.
|
|
|
|
func getSANs(cert *x509.Certificate) []string {
|
|
|
|
var sans []string
|
|
|
|
if cert == nil {
|
|
|
|
return sans
|
|
|
|
}
|
|
|
|
|
2019-02-05 17:10:03 +01:00
|
|
|
sans = append(sans, cert.DNSNames...)
|
|
|
|
sans = append(sans, cert.EmailAddresses...)
|
2018-11-14 10:18:03 +01:00
|
|
|
|
|
|
|
var ips []string
|
|
|
|
for _, ip := range cert.IPAddresses {
|
|
|
|
ips = append(ips, ip.String())
|
|
|
|
}
|
|
|
|
sans = append(sans, ips...)
|
|
|
|
|
|
|
|
var uris []string
|
|
|
|
for _, uri := range cert.URIs {
|
|
|
|
uris = append(uris, uri.String())
|
|
|
|
}
|
|
|
|
|
|
|
|
return append(sans, uris...)
|
|
|
|
}
|