2022-03-02 19:08:36 +01:00
|
|
|
// Copyright 2020 Matthew Holt and The Caddy Authors
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
package caddypki
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/caddyserver/caddy/v2"
|
|
|
|
"go.uber.org/zap"
|
|
|
|
)
|
|
|
|
|
|
|
|
func init() {
|
2022-03-02 21:00:37 +01:00
|
|
|
caddy.RegisterModule(adminAPI{})
|
2022-03-02 19:08:36 +01:00
|
|
|
}
|
|
|
|
|
2022-03-02 21:00:37 +01:00
|
|
|
// adminAPI is a module that serves PKI endpoints to retrieve
|
2022-03-02 19:08:36 +01:00
|
|
|
// information about the CAs being managed by Caddy.
|
2022-03-02 21:00:37 +01:00
|
|
|
type adminAPI struct {
|
2022-03-02 19:08:36 +01:00
|
|
|
ctx caddy.Context
|
|
|
|
log *zap.Logger
|
|
|
|
pkiApp *PKI
|
|
|
|
}
|
|
|
|
|
|
|
|
// CaddyModule returns the Caddy module information.
|
2022-03-02 21:00:37 +01:00
|
|
|
func (adminAPI) CaddyModule() caddy.ModuleInfo {
|
2022-03-02 19:08:36 +01:00
|
|
|
return caddy.ModuleInfo{
|
|
|
|
ID: "admin.api.pki",
|
2022-03-02 21:00:37 +01:00
|
|
|
New: func() caddy.Module { return new(adminAPI) },
|
2022-03-02 19:08:36 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-02 21:00:37 +01:00
|
|
|
// Provision sets up the adminAPI module.
|
|
|
|
func (a *adminAPI) Provision(ctx caddy.Context) error {
|
2022-03-02 19:08:36 +01:00
|
|
|
a.ctx = ctx
|
|
|
|
a.log = ctx.Logger(a)
|
|
|
|
|
|
|
|
// First check if the PKI app was configured, because
|
|
|
|
// a.ctx.App() has the side effect of instantiating
|
|
|
|
// and provisioning an app even if it wasn't configured.
|
|
|
|
pkiAppConfigured := a.ctx.AppIsConfigured("pki")
|
|
|
|
if !pkiAppConfigured {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Load the PKI app, so we can query it for information.
|
|
|
|
appModule, err := a.ctx.App("pki")
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
a.pkiApp = appModule.(*PKI)
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Routes returns the admin routes for the PKI app.
|
2022-03-02 21:00:37 +01:00
|
|
|
func (a *adminAPI) Routes() []caddy.AdminRoute {
|
2022-03-02 19:08:36 +01:00
|
|
|
return []caddy.AdminRoute{
|
|
|
|
{
|
2022-03-02 21:00:37 +01:00
|
|
|
Pattern: adminPKIEndpointBase,
|
|
|
|
Handler: caddy.AdminHandlerFunc(a.handleAPIEndpoints),
|
2022-03-02 19:08:36 +01:00
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-02 21:00:37 +01:00
|
|
|
// handleAPIEndpoints routes API requests within adminPKIEndpointBase.
|
|
|
|
func (a *adminAPI) handleAPIEndpoints(w http.ResponseWriter, r *http.Request) error {
|
|
|
|
uri := strings.TrimPrefix(r.URL.Path, "/pki/")
|
|
|
|
parts := strings.Split(uri, "/")
|
|
|
|
switch {
|
|
|
|
case len(parts) == 2 && parts[0] == "ca" && parts[1] != "":
|
|
|
|
return a.handleCAInfo(w, r)
|
|
|
|
case len(parts) == 3 && parts[0] == "ca" && parts[1] != "" && parts[2] == "certificates":
|
|
|
|
return a.handleCACerts(w, r)
|
|
|
|
}
|
|
|
|
return caddy.APIError{
|
|
|
|
HTTPStatus: http.StatusNotFound,
|
|
|
|
Err: fmt.Errorf("resource not found: %v", r.URL.Path),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-13 07:38:11 +01:00
|
|
|
// handleCAInfo returns information about a particular
|
2022-03-02 21:00:37 +01:00
|
|
|
// CA by its ID. If the CA ID is the default, then the CA will be
|
2022-03-02 19:08:36 +01:00
|
|
|
// provisioned if it has not already been. Other CA IDs will return an
|
|
|
|
// error if they have not been previously provisioned.
|
2022-03-02 21:00:37 +01:00
|
|
|
func (a *adminAPI) handleCAInfo(w http.ResponseWriter, r *http.Request) error {
|
2022-03-02 19:08:36 +01:00
|
|
|
if r.Method != http.MethodGet {
|
|
|
|
return caddy.APIError{
|
|
|
|
HTTPStatus: http.StatusMethodNotAllowed,
|
2022-03-02 21:00:37 +01:00
|
|
|
Err: fmt.Errorf("method not allowed: %v", r.Method),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ca, err := a.getCAFromAPIRequestPath(r)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
rootCert, interCert, err := rootAndIntermediatePEM(ca)
|
|
|
|
if err != nil {
|
|
|
|
return caddy.APIError{
|
|
|
|
HTTPStatus: http.StatusInternalServerError,
|
|
|
|
Err: fmt.Errorf("failed to get root and intermediate cert for CA %s: %v", ca.ID, err),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
repl := ca.newReplacer()
|
|
|
|
|
|
|
|
response := caInfo{
|
|
|
|
ID: ca.ID,
|
|
|
|
Name: ca.Name,
|
|
|
|
RootCN: repl.ReplaceAll(ca.RootCommonName, ""),
|
|
|
|
IntermediateCN: repl.ReplaceAll(ca.IntermediateCommonName, ""),
|
|
|
|
RootCert: string(rootCert),
|
|
|
|
IntermediateCert: string(interCert),
|
|
|
|
}
|
|
|
|
|
|
|
|
encoded, err := json.Marshal(response)
|
|
|
|
if err != nil {
|
|
|
|
return caddy.APIError{
|
|
|
|
HTTPStatus: http.StatusInternalServerError,
|
|
|
|
Err: err,
|
2022-03-02 19:08:36 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
w.Header().Set("Content-Type", "application/json")
|
2022-03-02 21:38:05 +01:00
|
|
|
_, _ = w.Write(encoded)
|
2022-03-02 19:08:36 +01:00
|
|
|
|
2022-03-02 21:00:37 +01:00
|
|
|
return nil
|
|
|
|
}
|
2022-03-02 19:08:36 +01:00
|
|
|
|
2022-03-13 07:38:11 +01:00
|
|
|
// handleCACerts returns the certificate chain for a particular
|
2022-03-02 21:00:37 +01:00
|
|
|
// CA by its ID. If the CA ID is the default, then the CA will be
|
|
|
|
// provisioned if it has not already been. Other CA IDs will return an
|
|
|
|
// error if they have not been previously provisioned.
|
|
|
|
func (a *adminAPI) handleCACerts(w http.ResponseWriter, r *http.Request) error {
|
|
|
|
if r.Method != http.MethodGet {
|
2022-03-02 19:08:36 +01:00
|
|
|
return caddy.APIError{
|
2022-03-02 21:00:37 +01:00
|
|
|
HTTPStatus: http.StatusMethodNotAllowed,
|
|
|
|
Err: fmt.Errorf("method not allowed: %v", r.Method),
|
2022-03-02 19:08:36 +01:00
|
|
|
}
|
|
|
|
}
|
2022-03-02 21:00:37 +01:00
|
|
|
|
|
|
|
ca, err := a.getCAFromAPIRequestPath(r)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
rootCert, interCert, err := rootAndIntermediatePEM(ca)
|
|
|
|
if err != nil {
|
2022-03-02 19:08:36 +01:00
|
|
|
return caddy.APIError{
|
2022-03-02 21:00:37 +01:00
|
|
|
HTTPStatus: http.StatusInternalServerError,
|
|
|
|
Err: fmt.Errorf("failed to get root and intermediate cert for CA %s: %v", ca.ID, err),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
w.Header().Set("Content-Type", "application/pem-certificate-chain")
|
|
|
|
_, err = w.Write(interCert)
|
|
|
|
if err == nil {
|
2022-03-02 21:38:05 +01:00
|
|
|
_, _ = w.Write(rootCert)
|
2022-03-02 21:00:37 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *adminAPI) getCAFromAPIRequestPath(r *http.Request) (*CA, error) {
|
|
|
|
// Grab the CA ID from the request path, it should be the 4th segment (/pki/ca/<ca>)
|
|
|
|
id := strings.Split(r.URL.Path, "/")[3]
|
|
|
|
if id == "" {
|
|
|
|
return nil, caddy.APIError{
|
2022-03-02 19:08:36 +01:00
|
|
|
HTTPStatus: http.StatusBadRequest,
|
2022-03-02 21:00:37 +01:00
|
|
|
Err: fmt.Errorf("missing CA in path"),
|
2022-03-02 19:08:36 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Find the CA by ID, if PKI is configured
|
|
|
|
var ca *CA
|
2022-03-02 21:00:37 +01:00
|
|
|
var ok bool
|
2022-03-02 19:08:36 +01:00
|
|
|
if a.pkiApp != nil {
|
|
|
|
ca, ok = a.pkiApp.CAs[id]
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we didn't find the CA, and PKI is not configured
|
|
|
|
// then we'll either error out if the CA ID is not the
|
|
|
|
// default. If the CA ID is the default, then we'll
|
|
|
|
// provision it, because the user probably aims to
|
|
|
|
// change their config to enable PKI immediately after
|
|
|
|
// if they actually requested the local CA ID.
|
|
|
|
if !ok {
|
|
|
|
if id != DefaultCAID {
|
2022-03-02 21:00:37 +01:00
|
|
|
return nil, caddy.APIError{
|
2022-03-02 19:08:36 +01:00
|
|
|
HTTPStatus: http.StatusNotFound,
|
|
|
|
Err: fmt.Errorf("no certificate authority configured with id: %s", id),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Provision the default CA, which generates and stores a root
|
|
|
|
// certificate in storage, if one doesn't already exist.
|
|
|
|
ca = new(CA)
|
|
|
|
err := ca.Provision(a.ctx, id, a.log)
|
|
|
|
if err != nil {
|
2022-03-02 21:00:37 +01:00
|
|
|
return nil, caddy.APIError{
|
2022-03-02 19:08:36 +01:00
|
|
|
HTTPStatus: http.StatusInternalServerError,
|
|
|
|
Err: fmt.Errorf("failed to provision CA %s, %w", id, err),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-02 21:00:37 +01:00
|
|
|
return ca, nil
|
|
|
|
}
|
2022-03-02 19:08:36 +01:00
|
|
|
|
2022-03-02 21:00:37 +01:00
|
|
|
func rootAndIntermediatePEM(ca *CA) (root, inter []byte, err error) {
|
|
|
|
root, err = pemEncodeCert(ca.RootCertificate().Raw)
|
|
|
|
if err != nil {
|
|
|
|
return
|
2022-03-02 19:08:36 +01:00
|
|
|
}
|
2022-03-02 21:00:37 +01:00
|
|
|
inter, err = pemEncodeCert(ca.IntermediateCertificate().Raw)
|
2022-03-02 19:08:36 +01:00
|
|
|
if err != nil {
|
2022-03-02 21:00:37 +01:00
|
|
|
return
|
2022-03-02 19:08:36 +01:00
|
|
|
}
|
2022-03-02 21:00:37 +01:00
|
|
|
return
|
2022-03-02 19:08:36 +01:00
|
|
|
}
|
|
|
|
|
2022-03-02 21:00:37 +01:00
|
|
|
// caInfo is the response structure for the CA info API endpoint.
|
|
|
|
type caInfo struct {
|
|
|
|
ID string `json:"id"`
|
|
|
|
Name string `json:"name"`
|
|
|
|
RootCN string `json:"root_common_name"`
|
|
|
|
IntermediateCN string `json:"intermediate_common_name"`
|
|
|
|
RootCert string `json:"root_certificate"`
|
|
|
|
IntermediateCert string `json:"intermediate_certificate"`
|
2022-03-02 19:08:36 +01:00
|
|
|
}
|
|
|
|
|
2022-03-02 21:00:37 +01:00
|
|
|
// adminPKIEndpointBase is the base admin endpoint under which all PKI admin endpoints exist.
|
|
|
|
const adminPKIEndpointBase = "/pki/"
|
2022-03-02 19:08:36 +01:00
|
|
|
|
|
|
|
// Interface guards
|
|
|
|
var (
|
2022-03-02 21:00:37 +01:00
|
|
|
_ caddy.AdminRouter = (*adminAPI)(nil)
|
|
|
|
_ caddy.Provisioner = (*adminAPI)(nil)
|
2022-03-02 19:08:36 +01:00
|
|
|
)
|