fix: preferred chain support. (#1298)

This commit is contained in:
Ludovic Fernandez 2020-11-21 20:24:11 +01:00 committed by GitHub
parent 2029375e04
commit a7387202e4
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
6 changed files with 111 additions and 92 deletions

View file

@ -20,19 +20,82 @@ type CertificateService service
// Get Returns the certificate and the issuer certificate. // Get Returns the certificate and the issuer certificate.
// 'bundle' is only applied if the issuer is provided by the 'up' link. // 'bundle' is only applied if the issuer is provided by the 'up' link.
func (c *CertificateService) Get(certURL string, bundle bool) ([]byte, []byte, error) { func (c *CertificateService) Get(certURL string, bundle bool) ([]byte, []byte, error) {
cert, up, err := c.get(certURL) cert, _, err := c.get(certURL, bundle)
if err != nil { if err != nil {
return nil, nil, err return nil, nil, err
} }
return cert.Cert, cert.Issuer, nil
}
// GetAll the certificates and the alternate certificates.
// bundle' is only applied if the issuer is provided by the 'up' link.
func (c *CertificateService) GetAll(certURL string, bundle bool) (map[string]*acme.RawCertificate, error) {
cert, headers, err := c.get(certURL, bundle)
if err != nil {
return nil, err
}
certs := map[string]*acme.RawCertificate{certURL: cert}
// URLs of "alternate" link relation
// - https://tools.ietf.org/html/rfc8555#section-7.4.2
alts := getLinks(headers, "alternate")
for _, alt := range alts {
altCert, _, err := c.get(alt, bundle)
if err != nil {
return nil, err
}
certs[alt] = altCert
}
return certs, nil
}
// Revoke Revokes a certificate.
func (c *CertificateService) Revoke(req acme.RevokeCertMessage) error {
_, err := c.core.post(c.core.GetDirectory().RevokeCertURL, req, nil)
return err
}
// get Returns the certificate and the "up" link.
func (c *CertificateService) get(certURL string, bundle bool) (*acme.RawCertificate, http.Header, error) {
if len(certURL) == 0 {
return nil, nil, errors.New("certificate[get]: empty URL")
}
resp, err := c.core.postAsGet(certURL, nil)
if err != nil {
return nil, nil, err
}
data, err := ioutil.ReadAll(http.MaxBytesReader(nil, resp.Body, maxBodySize))
if err != nil {
return nil, resp.Header, err
}
cert := c.getCertificateChain(data, resp.Header, bundle, certURL)
return cert, resp.Header, err
}
// getCertificateChain Returns the certificate and the issuer certificate.
func (c *CertificateService) getCertificateChain(cert []byte, headers http.Header, bundle bool, certURL string) *acme.RawCertificate {
// Get issuerCert from bundled response from Let's Encrypt // Get issuerCert from bundled response from Let's Encrypt
// See https://community.letsencrypt.org/t/acme-v2-no-up-link-in-response/64962 // See https://community.letsencrypt.org/t/acme-v2-no-up-link-in-response/64962
_, issuer := pem.Decode(cert) _, issuer := pem.Decode(cert)
if issuer != nil { if issuer != nil {
return cert, issuer, nil return &acme.RawCertificate{Cert: cert, Issuer: issuer}
} }
issuer, err = c.getIssuerFromLink(up) // The issuer certificate link may be supplied via an "up" link
// in the response headers of a new certificate.
// See https://tools.ietf.org/html/rfc8555#section-7.4.2
up := getLink(headers, "up")
issuer, err := c.getIssuerFromLink(up)
if err != nil { if err != nil {
// If we fail to acquire the issuer cert, return the issued certificate - do not fail. // If we fail to acquire the issuer cert, return the issued certificate - do not fail.
log.Warnf("acme: Could not bundle issuer certificate [%s]: %v", certURL, err) log.Warnf("acme: Could not bundle issuer certificate [%s]: %v", certURL, err)
@ -44,37 +107,7 @@ func (c *CertificateService) Get(certURL string, bundle bool) ([]byte, []byte, e
} }
} }
return cert, issuer, nil return &acme.RawCertificate{Cert: cert, Issuer: issuer}
}
// Revoke Revokes a certificate.
func (c *CertificateService) Revoke(req acme.RevokeCertMessage) error {
_, err := c.core.post(c.core.GetDirectory().RevokeCertURL, req, nil)
return err
}
// get Returns the certificate and the "up" link.
func (c *CertificateService) get(certURL string) ([]byte, string, error) {
if len(certURL) == 0 {
return nil, "", errors.New("certificate[get]: empty URL")
}
resp, err := c.core.postAsGet(certURL, nil)
if err != nil {
return nil, "", err
}
cert, err := ioutil.ReadAll(http.MaxBytesReader(nil, resp.Body, maxBodySize))
if err != nil {
return nil, "", err
}
// The issuer certificate link may be supplied via an "up" link
// in the response headers of a new certificate.
// See https://tools.ietf.org/html/rfc8555#section-7.4.2
up := getLink(resp.Header, "up")
return cert, up, err
} }
// getIssuerFromLink requests the issuer certificate. // getIssuerFromLink requests the issuer certificate.
@ -85,15 +118,15 @@ func (c *CertificateService) getIssuerFromLink(up string) ([]byte, error) {
log.Infof("acme: Requesting issuer cert from %s", up) log.Infof("acme: Requesting issuer cert from %s", up)
cert, _, err := c.get(up) cert, _, err := c.get(up, false)
if err != nil { if err != nil {
return nil, err return nil, err
} }
_, err = x509.ParseCertificate(cert) _, err = x509.ParseCertificate(cert.Cert)
if err != nil { if err != nil {
return nil, err return nil, err
} }
return certcrypto.PEMEncode(certcrypto.DERCertificateBytes(cert)), nil return certcrypto.PEMEncode(certcrypto.DERCertificateBytes(cert.Cert)), nil
} }

View file

@ -25,9 +25,8 @@ func (o *OrderService) New(domains []string) (acme.ExtendedOrder, error) {
} }
return acme.ExtendedOrder{ return acme.ExtendedOrder{
Order: order, Order: order,
Location: resp.Header.Get("Location"), Location: resp.Header.Get("Location"),
AlternateChainLinks: getLinks(resp.Header, "alternate"),
}, nil }, nil
} }
@ -38,15 +37,12 @@ func (o *OrderService) Get(orderURL string) (acme.ExtendedOrder, error) {
} }
var order acme.Order var order acme.Order
resp, err := o.core.postAsGet(orderURL, &order) _, err := o.core.postAsGet(orderURL, &order)
if err != nil { if err != nil {
return acme.ExtendedOrder{}, err return acme.ExtendedOrder{}, err
} }
return acme.ExtendedOrder{ return acme.ExtendedOrder{Order: order}, nil
Order: order,
AlternateChainLinks: getLinks(resp.Header, "alternate"),
}, nil
} }
// UpdateForCSR Updates an order for a CSR. // UpdateForCSR Updates an order for a CSR.
@ -56,7 +52,7 @@ func (o *OrderService) UpdateForCSR(orderURL string, csr []byte) (acme.ExtendedO
} }
var order acme.Order var order acme.Order
resp, err := o.core.post(orderURL, csrMsg, &order) _, err := o.core.post(orderURL, csrMsg, &order)
if err != nil { if err != nil {
return acme.ExtendedOrder{}, err return acme.ExtendedOrder{}, err
} }
@ -65,8 +61,5 @@ func (o *OrderService) UpdateForCSR(orderURL string, csr []byte) (acme.ExtendedO
return acme.ExtendedOrder{}, order.Error return acme.ExtendedOrder{}, order.Error
} }
return acme.ExtendedOrder{ return acme.ExtendedOrder{Order: order}, nil
Order: order,
AlternateChainLinks: getLinks(resp.Header, "alternate"),
}, nil
} }

View file

@ -41,10 +41,6 @@ func TestOrderService_New(t *testing.T) {
return return
} }
w.Header().Add("Link", `<https://example.com/acme/cert/1>;rel="alternate"`)
w.Header().Add("Link", `<https://example.com/acme/cert/2>;title="foo";rel="alternate"`)
w.Header().Add("Link", `<https://example.com/acme/cert/3>;title="foo";rel="alternate", <https://example.com/acme/cert/4>;rel="alternate"`)
err = tester.WriteJSONResponse(w, acme.Order{ err = tester.WriteJSONResponse(w, acme.Order{
Status: acme.StatusValid, Status: acme.StatusValid,
Identifiers: order.Identifiers, Identifiers: order.Identifiers,
@ -66,12 +62,6 @@ func TestOrderService_New(t *testing.T) {
Status: "valid", Status: "valid",
Identifiers: []acme.Identifier{{Type: "dns", Value: "example.com"}}, Identifiers: []acme.Identifier{{Type: "dns", Value: "example.com"}},
}, },
AlternateChainLinks: []string{
"https://example.com/acme/cert/1",
"https://example.com/acme/cert/2",
"https://example.com/acme/cert/3",
"https://example.com/acme/cert/4",
},
} }
assert.Equal(t, expected, order) assert.Equal(t, expected, order)
} }

View file

@ -106,13 +106,9 @@ type Account struct {
// ExtendedOrder a extended Order. // ExtendedOrder a extended Order.
type ExtendedOrder struct { type ExtendedOrder struct {
Order Order
// The order URL, contains the value of the response header `Location` // The order URL, contains the value of the response header `Location`
Location string `json:"-"` Location string `json:"-"`
// AlternateChainLinks (optional, array of string):
// URLs of "alternate" link relation
// - https://tools.ietf.org/html/rfc8555#section-7.4.2
AlternateChainLinks []string `json:"-"`
} }
// Order the ACME order Object. // Order the ACME order Object.
@ -287,3 +283,9 @@ type RevokeCertMessage struct {
// The problem document detail SHOULD indicate which reasonCodes are allowed. // The problem document detail SHOULD indicate which reasonCodes are allowed.
Reason *uint `json:"reason,omitempty"` Reason *uint `json:"reason,omitempty"`
} }
// RawCertificate raw data of a certificate.
type RawCertificate struct {
Cert []byte
Issuer []byte
}

View file

@ -307,29 +307,25 @@ func (c *Certifier) checkResponse(order acme.ExtendedOrder, certRes *Resource, b
return valid, err return valid, err
} }
links := append([]string{order.Certificate}, order.AlternateChainLinks...) certs, err := c.core.Certificates.GetAll(order.Certificate, bundle)
if err != nil {
return false, err
}
for i, link := range links { // Set the default certificate
cert, issuer, err := c.core.Certificates.Get(link, bundle) certRes.IssuerCertificate = certs[order.Certificate].Issuer
if err != nil { certRes.Certificate = certs[order.Certificate].Cert
return false, err certRes.CertURL = order.Certificate
} certRes.CertStableURL = order.Certificate
// Set the default certificate if preferredChain == "" {
if i == 0 { log.Infof("[%s] Server responded with a certificate.", certRes.Domain)
certRes.IssuerCertificate = issuer
certRes.Certificate = cert
certRes.CertURL = link
certRes.CertStableURL = link
}
if preferredChain == "" { return true, nil
log.Infof("[%s] Server responded with a certificate.", certRes.Domain) }
return true, nil for link, cert := range certs {
} ok, err := hasPreferredChain(cert.Issuer, preferredChain)
ok, err := hasPreferredChain(issuer, preferredChain)
if err != nil { if err != nil {
return false, err return false, err
} }
@ -337,8 +333,8 @@ func (c *Certifier) checkResponse(order acme.ExtendedOrder, certRes *Resource, b
if ok { if ok {
log.Infof("[%s] Server responded with a certificate for the preferred certificate chains %q.", certRes.Domain, preferredChain) log.Infof("[%s] Server responded with a certificate for the preferred certificate chains %q.", certRes.Domain, preferredChain)
certRes.IssuerCertificate = issuer certRes.IssuerCertificate = cert.Issuer
certRes.Certificate = cert certRes.Certificate = cert.Cert
certRes.CertURL = link certRes.CertURL = link
certRes.CertStableURL = link certRes.CertStableURL = link

View file

@ -4,6 +4,7 @@ import (
"crypto/rand" "crypto/rand"
"crypto/rsa" "crypto/rsa"
"encoding/pem" "encoding/pem"
"fmt"
"net/http" "net/http"
"testing" "testing"
@ -289,12 +290,15 @@ func Test_checkResponse_alternate(t *testing.T) {
defer tearDown() defer tearDown()
mux.HandleFunc("/certificate", func(w http.ResponseWriter, _ *http.Request) { mux.HandleFunc("/certificate", func(w http.ResponseWriter, _ *http.Request) {
w.Header().Add("Link", fmt.Sprintf(`<%s/certificate/1>;title="foo";rel="alternate"`, apiURL))
_, err := w.Write([]byte(certResponseMock)) _, err := w.Write([]byte(certResponseMock))
if err != nil { if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError) http.Error(w, err.Error(), http.StatusInternalServerError)
} }
}) })
mux.HandleFunc("/certificate2", func(w http.ResponseWriter, _ *http.Request) {
mux.HandleFunc("/certificate/1", func(w http.ResponseWriter, _ *http.Request) {
_, err := w.Write([]byte(certResponseMock2)) _, err := w.Write([]byte(certResponseMock2))
if err != nil { if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError) http.Error(w, err.Error(), http.StatusInternalServerError)
@ -314,9 +318,10 @@ func Test_checkResponse_alternate(t *testing.T) {
Status: acme.StatusValid, Status: acme.StatusValid,
Certificate: apiURL + "/certificate", Certificate: apiURL + "/certificate",
}, },
AlternateChainLinks: []string{apiURL + "/certificate2"},
} }
certRes := &Resource{} certRes := &Resource{
Domain: "example.com",
}
bundle := false bundle := false
valid, err := certifier.checkResponse(order, certRes, bundle, "DST Root CA X3") valid, err := certifier.checkResponse(order, certRes, bundle, "DST Root CA X3")
@ -324,9 +329,9 @@ func Test_checkResponse_alternate(t *testing.T) {
assert.True(t, valid) assert.True(t, valid)
assert.NotNil(t, certRes) assert.NotNil(t, certRes)
assert.Equal(t, "", certRes.Domain) assert.Equal(t, "example.com", certRes.Domain)
assert.Contains(t, certRes.CertStableURL, "/certificate2") assert.Contains(t, certRes.CertStableURL, "/certificate/1")
assert.Contains(t, certRes.CertURL, "/certificate2") assert.Contains(t, certRes.CertURL, "/certificate/1")
assert.Nil(t, certRes.CSR) assert.Nil(t, certRes.CSR)
assert.Nil(t, certRes.PrivateKey) assert.Nil(t, certRes.PrivateKey)
assert.Equal(t, certResponseMock2, string(certRes.Certificate), "Certificate") assert.Equal(t, certResponseMock2, string(certRes.Certificate), "Certificate")