An ACME-based certificate authority, written in Go.
Go to file
Richard Barnes 95bbaecb59 Add vim temp files to .gitignore 2015-03-21 18:26:26 -05:00
analysis Separate imports from the standard library 2015-03-12 12:29:21 +01:00
ca Merge branch 'spec-update' into policy 2015-03-17 11:50:42 -04:00
cmd Migration of CA to individual binary 2015-03-21 18:26:00 -05:00
core Rebase 'lint-errcheck-fixes' of git://github.com/mvdan/boulder to letsencrypt/master 2015-03-20 18:01:03 -07:00
jose Re-fixing 'go vet' errors 2015-03-14 12:21:22 -04:00
log Returning after log.Fatalf is redundant 2015-03-12 12:40:09 +01:00
policy Fixup: missing "," 2015-03-20 13:28:09 -07:00
ra Rebase 'lint-errcheck-fixes' of git://github.com/mvdan/boulder to letsencrypt/master 2015-03-20 18:01:03 -07:00
rpc Rebase 'lint-errcheck-fixes' of git://github.com/mvdan/boulder to letsencrypt/master 2015-03-20 18:01:03 -07:00
sa Rebase 'lint-errcheck-fixes' of git://github.com/mvdan/boulder to letsencrypt/master 2015-03-20 18:01:03 -07:00
test Addressing JCJ comments 2015-03-17 10:17:21 -04:00
va Rebase 'lint-errcheck-fixes' of git://github.com/mvdan/boulder to letsencrypt/master 2015-03-20 18:01:03 -07:00
wfe Rebase 'lint-errcheck-fixes' of git://github.com/mvdan/boulder to letsencrypt/master 2015-03-20 18:01:03 -07:00
.gitignore Add vim temp files to .gitignore 2015-03-21 18:26:26 -05:00
.travis.yml Fixing issues from jsha review 2015-03-13 12:31:41 -07:00
Dockerfile Update per spec 2015-03-20 15:37:53 -07:00
LICENSE.txt Add LICENSE.txt 2015-01-28 23:06:59 +00:00
README.md Gratuitous commit to re-start Travis 2015-03-05 18:26:46 -05:00

README.md

Boulder - An ACME CA

This is an initial implementation of an ACME-based CA. The ACME protocol allows the CA to automatically verify that an applicant for a certificate actually controls an identifier, and allows domain holders to issue and revoke certificates for their domains.

Build Status Coverage Status Docker Repository on Quay.io

Docker

Boulder is available as a Docker image from Quay.io. The entrypoint is the Boulder main method; you can load and run it using in monolithic mode (without AMQP) like:

docker run -p 4000:4000 quay.io/letsencrypt/boulder monolithic

To run a single module, specifying the AMQP server, you might use something more like:

docker run -p 4000:4000 quay.io/letsencrypt/boulder --amqp 'amqp://guest:guest@amqp-server:15672' wfe

Quickstart

> go build github.com/letsencrypt/boulder/boulder-start
> ./boulder-start monolithic # without AMQP
> ./boulder-start monolithic-amqp # with AMQP

The "restify" branch of node-acme has a client that works with this server (npm install node-acme && node node-acme/demo.js).

> git clone https://github.com/letsencrypt/node-acme.git
> cd node-acme
> git branch -f restify origin/restify && git checkout restify
> cd ..
> npm install node-acme
> node node-acme/demo.js

Component Model

The CA is divided into the following main components:

  1. Web Front End
  2. Registration Authority
  3. Validation Authority
  4. Certificate Authority
  5. Storage Authority

This component model lets us separate the function of the CA by security context. The Web Front End and Validation Authority need access to the Internet, which puts them at greater risk of compromise. The Registration Authority can live without Internet connectivity, but still needs to talk to the Web Front End and Validation Authority. The Certificate Authority need only receive instructions from the Registration Authority.


client <--ACME--> WFE ---+
  .                      |
  .                      +--- RA --- CA
  .                      |
client <-checks->  VA ---+

In Boulder, these components are represented by Go interfaces. This allows us to have two operational modes: Consolidated and distributed. In consolidated mode, the objects representing the different components interact directly, through function calls. In distributed mode, each component runs in a separate process (possibly on a separate machine), and sees the other components' methods by way of a messaging layer.

Internally, the logic of the system is based around two types of objects, authorizations and certificates, mapping directly to the resources of the same name in ACME.

Requests from ACME clients result in new objects and changes objects. The Storage Authority maintains persistent copies of the current set of objects.

Objects are also passed from one component to another on change events. For example, when a client provides a successful response to a validation challenge, it results in a change to the corresponding validation object. The Validation Authority forward the new validation object to the Storage Authority for storage, and to the Registration Authority for any updates to a related Authorization object.

Boulder supports distributed operation using AMQP as a message bus (e.g., via RabbitMQ). For components that you want to be remote, it is necessary to instantiate a "client" and "server" for that component. The client implements the component's Go interface, while the server has the actual logic for the component. More details in amqp-rpc.go.

Files

  • interfaces.go - Interfaces to the components, implemented in:
    • web-front-end.go
    • registration-authority.go
    • validation-authority.go
    • certificate-authority.go
    • storage-authority.go
  • amqp-rpc.go - A lightweight RPC framework overlaid on AMQP
    • rpc-wrappers.go - RPC wrappers for the various component type
  • objects.go - Objects that are passed between components
  • util.go - Miscellaneous utility methods
  • boulder_test.go - Unit tests

Dependencies:

ACME Processing

Client -> WebFE:  challengeRequest
WebFE -> RA:      NewAuthorization(AuthorizationRequest)
RA -> RA:         [ select challenges ]
RA -> RA:         [ create Validations with challenges ]
RA -> RA:         [ create Authorization with Validations ]
RA -> SA:         Update(Authorization.ID, Authorization)
RA -> WebFE:      Authorization
WebFE -> WebFE:   [ create challenge from Authorization ]
WebFE -> WebFE:   [ generate nonce and add ]
WebFE -> Client:  challenge

----------

Client -> WebFE:  authorizationRequest
WebFE -> WebFE:   [ look up authorization based on nonce ]
WebFE -> WebFE:   [ verify authorization signature ]
WebFE -> RA:      UpdateAuthorization(Authorization)
RA -> RA:         [ add responses to authorization ]
RA -> SA:         Update(Authorization.ID, Authorization)
WebFE -> VA:      UpdateValidations(Authorization)
WebFE -> Client:  defer(authorizationID)

VA -> SA:         Update(Authorization.ID, Authorization)
VA -> RA:         OnValidationUpdate(Authorization)
RA -> RA:         [ check that validation sufficient ]
RA -> RA:         [ finalize authorization ]
RA -> SA:         Update(Authorization.ID, Authorization)
RA -> WebFE:      OnAuthorizationUpdate(Authorization)
Client -> WebFE:  statusRequest
WebFE -> Client:  error / authorization

----------

Client -> WebFE:  certificateRequest
WebFE -> WebFE:   [ verify authorization signature ]
WebFE -> RA:      NewCertificate(CertificateRequest)
RA -> RA:         [ verify CSR signature ]
RA -> RA:         [ verify authorization to issue ]
RA -> RA:         [ select CA based on issuer ]
RA -> CA:         IssueCertificate(CertificateRequest)
CA -> RA:         Certificate
RA -> CA:         [ look up ancillary data ]
RA -> WebFE:      AcmeCertificate
WebFE -> Client:  certificate

----------

Client -> WebFE:  revocationRequest
WebFE -> WebFE:   [ verify authorization signature ]
WebFE -> RA:      RevokeCertificate(RevocationRequest)
RA -> RA:         [ verify authorization ]
RA -> CA:         RevokeCertificate(Certificate)
CA -> RA:         RevocationResult
RA -> WebFE:      RevocationResult
WebFE -> Client:  revocation

TODO

  • Ensure that distributed mode works with multiple processes
  • Add message signing and verification to the AMQP message layer
  • Add monitoring / syslog
  • Factor out policy layer (e.g., selection of challenges)
  • Add persistent storage