Skip to content

Commit

Permalink
Add OCSP and CRL support to certificate verify
Browse files Browse the repository at this point in the history
Add args and functionality to certificate verify to check a CRL
and OCSP for a certificate based on the extensions. Users can pass
flags to enable verification of each (CRL, OCSP). The command will try
and get the CRL and OCSP server from the certifiacate and verify the
certificate against each.

I also moved functions from the crl command into internal/crlutil
package so they can be re-used with the certificate verify command.

Implements #845
  • Loading branch information
redrac committed May 11, 2024
1 parent e5ab833 commit c369d92
Show file tree
Hide file tree
Showing 5 changed files with 583 additions and 305 deletions.
272 changes: 270 additions & 2 deletions command/certificate/verify.go
Original file line number Diff line number Diff line change
@@ -1,15 +1,22 @@
package certificate

import (
"bytes"
"crypto/tls"
"crypto/x509"
"encoding/pem"
"fmt"
"io"
"net/http"
"os"

"github.com/pkg/errors"
"github.com/smallstep/cli/flags"
"github.com/smallstep/cli/internal/crlutil"
"github.com/urfave/cli"
"go.step.sm/cli-utils/errs"
"go.step.sm/crypto/x509util"
"golang.org/x/crypto/ocsp"
)

func verifyCommand() cli.Command {
Expand All @@ -18,7 +25,10 @@ func verifyCommand() cli.Command {
Action: cli.ActionFunc(verifyAction),
Usage: `verify a certificate`,
UsageText: `**step certificate verify** <crt-file> [**--host**=<host>]
[**--roots**=<root-bundle>] [**--servername**=<servername>]`,
[**--roots**=<root-bundle>] [**--servername**=<servername>]
[**--issuing-ca**=<ca-cert-file>] [**--verify-verbose**]
[**--verify-ocsp**]] [**--ocsp-endpoint**]=url
[**--verify-crl**] [**--crl-endpoint**]=url`,
Description: `**step certificate verify** executes the certificate path
validation algorithm for x.509 certificates defined in RFC 5280. If the
certificate is valid this command will return '0'. If validation fails, or if
Expand Down Expand Up @@ -65,7 +75,24 @@ Verify a certificate using a custom directory of root certificates for path vali
'''
$ step certificate verify ./certificate.crt --roots ./root-certificates/
'''
`,
Verify a certificate including OCSP and CRL using CRL and OCSP defined in the certificate
'''
$ step certificate verify ./certificate.crt --verify-crl --verify-ocsp
'''
Verify a certificate including OCSP and specifying an OCSP server
'''
$ step certificate verify ./certificate.crt --verify-ocsp --ocsp-endpoint http://acme.com/ocsp
'''
Verify a certificate including CRL and specificing a CRL server and providing the issuing CA certificate
'''
$ step certificate verify ./certificate.crt --issuing-ca ./issuing_ca.pem --verify-crl --crl-endpoint http://acme.com/crl
'''`,
Flags: []cli.Flag{
cli.StringFlag{
Name: "host",
Expand All @@ -87,7 +114,32 @@ authenticity of the remote server.
**directory**
: Relative or full path to a directory. Every PEM encoded certificate from each file in the directory will be used for path validation.`,
},
cli.StringFlag{
Name: "issuing-ca",
Usage: `The certificate issuer CA <file> needed to communicate with OCSP and verify a CRL. By default the issuing CA will be taken from the cert Issuing Certificate URL extension.`,
},
cli.BoolFlag{
Name: "verify-ocsp",
Usage: "Verify the certificate against it's OCSP.",
},
cli.StringFlag{
Name: "ocsp-endpoint",
Usage: `The OCSP endpoint to use. If not provided step will attempt to check it against the certificate's OCSPServer AIA extension endpoints.`,
},
cli.BoolFlag{
Name: "verify-crl",
Usage: "Verify the certificate against it's CRL.",
},
cli.StringFlag{
Name: "crl-endpoint",
Usage: "The CRL endpoint to use. If not provided step will attempt to check it against the certificate's CRLDistributionPoints extension endpoints.",
},
cli.BoolFlag{
Name: "verify-verbose",
Usage: "Print result of certificate verification to stdout on success",
},
flags.ServerName,
flags.Insecure,
},
}
}
Expand All @@ -102,9 +154,18 @@ func verifyAction(ctx *cli.Context) error {
host = ctx.String("host")
serverName = ctx.String("servername")
roots = ctx.String("roots")
verifyOCSP = ctx.Bool("verify-ocsp")
ocspEndpoint = ctx.String("ocsp-endpoint")
verifyCRL = ctx.Bool("verify-crl")
crlEndpoint = ctx.String("crl-endpoint")
verbose = ctx.Bool("verify-verbose")
issuerFile = ctx.String("issuing-ca")
insecure = ctx.Bool("insecure")
intermediatePool = x509.NewCertPool()
rootPool *x509.CertPool
cert *x509.Certificate
issuer *x509.Certificate
httpClient *http.Client
)

switch addr, isURL, err := trimURL(crtFile); {
Expand Down Expand Up @@ -180,5 +241,212 @@ func verifyAction(ctx *cli.Context) error {
return errors.Wrapf(err, "failed to verify certificate")
}

verboseMSG := "certificate validated against roots\n"
if host != "" {
verboseMSG += "certificate host name validated\n"
}

switch {
case (verifyCRL || verifyOCSP) && roots != "":
tlsConfig := &tls.Config{

Check failure on line 251 in command/certificate/verify.go

View workflow job for this annotation

GitHub Actions / ci / lint / lint

G402: TLS MinVersion too low. (gosec)
RootCAs: rootPool,
}

transport := &http.Transport{
TLSClientConfig: tlsConfig,
}

httpClient = &http.Client{
Transport: transport,
}
case verifyCRL || verifyOCSP:
httpClient = &http.Client{}
default:
}

switch {
case (verifyCRL || verifyOCSP) && issuerFile == "":
if len(cert.IssuingCertificateURL) == 0 && issuerFile == "" {
return errors.Errorf("could not get the issuing CA from the cert and no issuing CA certificate provided")
}

resp, err := httpClient.Get(cert.IssuingCertificateURL[0])
if err != nil {
return errors.Errorf("failed to download the issuing CA")
}
defer resp.Body.Close()

body, err := io.ReadAll(resp.Body)
if err != nil {
return errors.Errorf("failed to read the response body from the issuing CA url")
}

issuer, err = x509.ParseCertificate(body)
if err != nil {
return errors.Errorf("failed to parse the issuing CA")
}
case issuerFile != "":
issuerCertPEM, err := os.ReadFile(issuerFile)
if err != nil {
return errors.Errorf("unable to load the issuing CA certificate file")
}

issuerBlock, _ := pem.Decode(issuerCertPEM)
if issuerBlock == nil || issuerBlock.Type != "CERTIFICATE" {
return errors.Errorf("failed to decode the issuing CA certificate")
}

issuer, err = x509.ParseCertificate(issuerBlock.Bytes)
if err != nil {
return errors.Errorf("failed to parse the issuing CA certificate")
}
default:
}

if verifyCRL {
var endpoints []string
switch {
case crlEndpoint != "":
endpoints = []string{crlEndpoint}
case len(cert.CRLDistributionPoints) == 0:
return errors.Errorf("CRL distribution endpoint not found in certificate")
default:
endpoints = cert.CRLDistributionPoints
}

crlVerified := false
for _, endpoint := range endpoints {
respReceived, err := VerifyCRLEndpoint(endpoint, cert, issuer, httpClient, insecure)
switch {
case err == nil:
verboseMSG += fmt.Sprintf("certificate not revoked in CRL %s\n", endpoint)
crlVerified = true
break

Check failure on line 324 in command/certificate/verify.go

View workflow job for this annotation

GitHub Actions / ci / lint / lint

S1023: redundant break statement (gosimple)
case respReceived:
return err
case verbose:
fmt.Println(err)
default:
}
}

if !crlVerified {
return errors.Errorf("could not verify certificate against OCSP server(s)")
}
}

if verifyOCSP {
var endpoints []string
switch {
case ocspEndpoint != "":
endpoints = []string{ocspEndpoint}
case len(cert.OCSPServer) == 0:
return errors.Errorf("no OCSP AIA extension found")
default:
endpoints = cert.OCSPServer
}

ocspVerified := false
for _, endpoint := range endpoints {
respReceived, err := VerifyOCSPEndpoint(endpoint, cert, issuer, httpClient)
switch {
case err == nil:
verboseMSG += fmt.Sprintf("certificate status is good according OCSP %s\n", endpoint)
ocspVerified = true
break

Check failure on line 356 in command/certificate/verify.go

View workflow job for this annotation

GitHub Actions / ci / lint / lint

S1023: redundant break statement (gosimple)
case respReceived:
return err
case verbose:
fmt.Println(err)
default:
}
}

if !ocspVerified {
return errors.Errorf("could not verify certificate against OCSP server(s)")
}
}

if verbose {
fmt.Println(verboseMSG + "certficiate is valid")
}
return nil
}

func VerifyOCSPEndpoint(endpoint string, cert *x509.Certificate, issuer *x509.Certificate, httpClient *http.Client) (bool, error) {

Check failure on line 376 in command/certificate/verify.go

View workflow job for this annotation

GitHub Actions / ci / lint / lint

paramTypeCombine: func(endpoint string, cert *x509.Certificate, issuer *x509.Certificate, httpClient *http.Client) (bool, error) could be replaced with func(endpoint string, cert, issuer *x509.Certificate, httpClient *http.Client) (bool, error) (gocritic)
req, err := ocsp.CreateRequest(cert, issuer, nil)
if err != nil {
return false, errors.Errorf("error creating OCSP request")
}

httpReq, err := http.NewRequest(http.MethodPost, endpoint, bytes.NewReader(req))
if err != nil {
return false, errors.Errorf("error contacting OCSP server: %s", endpoint)
}
httpReq.Header.Add("Content-Type", "application/ocsp-request")
httpResp, err := httpClient.Do(httpReq)
if err != nil {
return false, errors.Errorf("error contacting OCSP server: %s", endpoint)
}
defer httpResp.Body.Close()
respBytes, err := io.ReadAll(httpResp.Body)
if err != nil {
return false, errors.Errorf("error reading response from OCSP server: %s", endpoint)
}

resp, err := ocsp.ParseResponse(respBytes, issuer)
if err != nil {
return false, errors.Errorf("error parsing response from OCSP server: %s", endpoint)
}

switch resp.Status {
case ocsp.Revoked:
return true, errors.Errorf("certificate has been revoked according to OCSP %s", endpoint)
case ocsp.Good:
return true, nil
default:
return true, errors.Errorf("certificate status is unknown according to OCSP %s", endpoint)
}
}

func VerifyCRLEndpoint(endpoint string, cert *x509.Certificate, issuer *x509.Certificate, httpClient *http.Client, insecure bool) (bool, error) {

Check failure on line 412 in command/certificate/verify.go

View workflow job for this annotation

GitHub Actions / ci / lint / lint

paramTypeCombine: func(endpoint string, cert *x509.Certificate, issuer *x509.Certificate, httpClient *http.Client, insecure bool) (bool, error) could be replaced with func(endpoint string, cert, issuer *x509.Certificate, httpClient *http.Client, insecure bool) (bool, error) (gocritic)
resp, err := httpClient.Get(endpoint)
if err != nil {
return false, errors.Wrap(err, "error downloading crl")
}
defer resp.Body.Close()

if resp.StatusCode >= 400 {
return false, errors.Errorf("error downloading crl: status code %d", resp.StatusCode)
}

b, err := io.ReadAll(resp.Body)
if err != nil {
return false, errors.Wrap(err, "error downloading crl")
}

crl, err := x509.ParseRevocationList(b)
if err != nil {
return false, errors.Wrap(err, "error parsing crl")
}

crlJSON, err := crlutil.ParseCRL(b)
if err != nil {
return false, errors.Wrap(err, "error parsing crl into json")
}

if issuer != nil && !insecure {
err = crl.CheckSignatureFrom(issuer)
if err != nil {
return false, errors.Wrap(err, "error validating the CRL against the CA issuer")
}
}

for _, revoked := range crlJSON.RevokedCertificates {
if cert.SerialNumber.String() == revoked.SerialNumber {
return true, errors.Errorf("certificate marked as revoked in CRL %s", endpoint)
}
}

return true, nil
}
Loading

0 comments on commit c369d92

Please sign in to comment.