Skip to content

Commit

Permalink
added suggestions from PR
Browse files Browse the repository at this point in the history
  • Loading branch information
Magnus Kaiser committed Feb 28, 2022
1 parent 22ebe86 commit 638cf96
Show file tree
Hide file tree
Showing 2 changed files with 58 additions and 45 deletions.
6 changes: 3 additions & 3 deletions cmd/thanos/receive.go
Original file line number Diff line number Diff line change
Expand Up @@ -191,7 +191,7 @@ func runReceive(
Registry: reg,
Endpoint: conf.endpoint,
TenantHeader: conf.tenantHeader,
TenantAttribute: conf.tenantAttribute,
TenantField: conf.tenantField,
DefaultTenantID: conf.defaultTenantID,
ReplicaHeader: conf.replicaHeader,
ReplicationFactor: conf.replicationFactor,
Expand Down Expand Up @@ -705,7 +705,7 @@ type receiveConfig struct {
refreshInterval *model.Duration
endpoint string
tenantHeader string
tenantAttribute string
tenantField string
tenantLabelName string
defaultTenantID string
replicaHeader string
Expand Down Expand Up @@ -769,7 +769,7 @@ func (rc *receiveConfig) registerFlag(cmd extkingpin.FlagClause) {

cmd.Flag("receive.tenant-header", "HTTP header to determine tenant for write requests.").Default(receive.DefaultTenantHeader).StringVar(&rc.tenantHeader)

cmd.Flag("receive.tenant-certificate-attribute", "Use TLS client certificate's attribute to determine tenant for write requests. Must be one of organization, organizationalUnit or commonName.").Default("").EnumVar(&rc.tenantAttribute, "", "organization", "organizationalUnit", "commonName")
cmd.Flag("receive.tenant-certificate-field", "Use TLS client's certificate field to determine tenant for write requests. Must be one of "+receive.CertificateFieldOrganization+", "+receive.CertificateFieldOrganizationalUnit+" or "+receive.CertificateFieldCommonName+". This setting takes will cause the receive.tenant-header flag value to be ignored.").Default("").EnumVar(&rc.tenantField, "", receive.CertificateFieldOrganization, receive.CertificateFieldOrganizationalUnit, receive.CertificateFieldCommonName)

cmd.Flag("receive.default-tenant-id", "Default tenant ID to use when none is provided via a header.").Default(receive.DefaultTenant).StringVar(&rc.defaultTenantID)

Expand Down
97 changes: 55 additions & 42 deletions pkg/receive/handler.go
Original file line number Diff line number Diff line change
Expand Up @@ -57,6 +57,13 @@ const (
labelError = "error"
)

// Allowed fields in client certificates.
const (
CertificateFieldOrganization = "organization"
CertificateFieldOrganizationalUnit = "organizationalUnit"
CertificateFieldCommonName = "commonName"
)

var (
// errConflict is returned whenever an operation fails due to any conflict-type error.
errConflict = errors.New("conflict")
Expand All @@ -72,7 +79,7 @@ type Options struct {
ListenAddress string
Registry prometheus.Registerer
TenantHeader string
TenantAttribute string
TenantField string
DefaultTenantID string
ReplicaHeader string
Endpoint string
Expand Down Expand Up @@ -340,47 +347,11 @@ func (h *Handler) receiveHTTP(w http.ResponseWriter, r *http.Request) {
tenant = h.options.DefaultTenantID
}

// Checking certs for possible value.
if h.options.TenantAttribute != "" {
if len(r.TLS.PeerCertificates) > 0 {
// First cert is the leaf authenticated against.
cert := r.TLS.PeerCertificates[0]

switch h.options.TenantAttribute {

case "organization":
if len(cert.Subject.Organization) > 0 {
tenant = cert.Subject.Organization[0]
} else {
http.Error(w, "could not get organization attribute from client cert", http.StatusBadRequest)
return
}

case "organizationalUnit":
if len(cert.Subject.OrganizationalUnit) > 0 {
tenant = cert.Subject.OrganizationalUnit[0]
} else {
http.Error(w, "could not get organizationalUnit attribute from client cert", http.StatusBadRequest)
return
}

case "commonName":
if cert.Subject.CommonName != "" {
tenant = cert.Subject.CommonName
} else {
http.Error(w, "could not get commonName attribute from client cert", http.StatusBadRequest)
return
}

default:
// Unknown/unsupported attribute requested, can't continue.
level.Error(h.logger).Log("err", err, "msg", "tls client cert attribute requested is not supported")
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}

} else {
http.Error(w, "could not get required certificate attribute from client cert", http.StatusBadRequest)
if h.options.TenantField != "" {
tenant, err = h.getTenantFromCertificate(r)
if err != nil {
// This must hard fail to ensure hard tenancy when feature is enabled.
http.Error(w, err.Error(), http.StatusBadRequest)
return
}
}
Expand Down Expand Up @@ -845,3 +816,45 @@ func (p *peerGroup) get(ctx context.Context, addr string) (storepb.WriteableStor
p.cache[addr] = client
return client, nil
}

// GetTenantFromCertificate extracts the tenant value from a client's presented certificate. The x509 field to use as
// value can be configured with Options.TenantField. An error is returned when the extraction has not succeeded.
func (h *Handler) getTenantFromCertificate(r *http.Request) (string, error) {
var tenant string

if len(r.TLS.PeerCertificates) == 0 {
return "", errors.New("could not get required certificate field from client cert")
}

// First cert is the leaf authenticated against.
cert := r.TLS.PeerCertificates[0]

switch h.options.TenantField {

case CertificateFieldOrganization:
if len(cert.Subject.Organization) > 0 {
tenant = cert.Subject.Organization[0]
} else {
return "", errors.New("could not get organization field from client cert")
}

case CertificateFieldOrganizationalUnit:
if len(cert.Subject.OrganizationalUnit) > 0 {
tenant = cert.Subject.OrganizationalUnit[0]
} else {
return "", errors.New("could not get organizationalUnit field from client cert")
}

case CertificateFieldCommonName:
if cert.Subject.CommonName != "" {
tenant = cert.Subject.CommonName
} else {
return "", errors.New("could not get commonName field from client cert")
}

default:
return "", errors.New("tls client cert field requested is not supported")
}

return tenant, nil
}

0 comments on commit 638cf96

Please sign in to comment.