mirror of
				https://github.com/caddyserver/caddy.git
				synced 2025-10-25 07:49:19 -04:00 
			
		
		
		
	* caddytls: Make on-demand 'ask' permission modular This makes the 'ask' endpoint a module, which means that developers can write custom plugins for granting permission for on-demand certificates. Kicking myself that we didn't do it this way at the beginning, but who coulda known... * Lint * Error on conflicting config * Fix bad merge --------- Co-authored-by: Francis Lavoie <lavofr@gmail.com>
		
			
				
	
	
		
			193 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			193 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2015 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 caddytls
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"crypto/tls"
 | |
| 	"encoding/json"
 | |
| 	"errors"
 | |
| 	"fmt"
 | |
| 	"net/http"
 | |
| 	"net/url"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/caddyserver/certmagic"
 | |
| 	"go.uber.org/zap"
 | |
| 
 | |
| 	"github.com/caddyserver/caddy/v2"
 | |
| )
 | |
| 
 | |
| func init() {
 | |
| 	caddy.RegisterModule(PermissionByHTTP{})
 | |
| }
 | |
| 
 | |
| // OnDemandConfig configures on-demand TLS, for obtaining
 | |
| // needed certificates at handshake-time. Because this
 | |
| // feature can easily be abused, you should use this to
 | |
| // establish rate limits and/or an internal endpoint that
 | |
| // Caddy can "ask" if it should be allowed to manage
 | |
| // certificates for a given hostname.
 | |
| type OnDemandConfig struct {
 | |
| 	// DEPRECATED. WILL BE REMOVED SOON. Use 'permission' instead.
 | |
| 	Ask string `json:"ask,omitempty"`
 | |
| 
 | |
| 	// REQUIRED. A module that will determine whether a
 | |
| 	// certificate is allowed to be loaded from storage
 | |
| 	// or obtained from an issuer on demand.
 | |
| 	PermissionRaw json.RawMessage `json:"permission,omitempty" caddy:"namespace=tls.permission inline_key=module"`
 | |
| 	permission    OnDemandPermission
 | |
| 
 | |
| 	// DEPRECATED. An optional rate limit to throttle
 | |
| 	// the checking of storage and the issuance of
 | |
| 	// certificates from handshakes if not already in
 | |
| 	// storage. WILL BE REMOVED IN A FUTURE RELEASE.
 | |
| 	RateLimit *RateLimit `json:"rate_limit,omitempty"`
 | |
| }
 | |
| 
 | |
| // DEPRECATED. WILL LIKELY BE REMOVED SOON.
 | |
| // Instead of using this rate limiter, use a proper tool such as a
 | |
| // level 3 or 4 firewall and/or a permission module to apply rate limits.
 | |
| type RateLimit struct {
 | |
| 	// A duration value. Storage may be checked and a certificate may be
 | |
| 	// obtained 'burst' times during this interval.
 | |
| 	Interval caddy.Duration `json:"interval,omitempty"`
 | |
| 
 | |
| 	// How many times during an interval storage can be checked or a
 | |
| 	// certificate can be obtained.
 | |
| 	Burst int `json:"burst,omitempty"`
 | |
| }
 | |
| 
 | |
| // OnDemandPermission is a type that can give permission for
 | |
| // whether a certificate should be allowed to be obtained or
 | |
| // loaded from storage on-demand.
 | |
| // EXPERIMENTAL: This API is experimental and subject to change.
 | |
| type OnDemandPermission interface {
 | |
| 	// CertificateAllowed returns nil if a certificate for the given
 | |
| 	// name is allowed to be either obtained from an issuer or loaded
 | |
| 	// from storage on-demand.
 | |
| 	//
 | |
| 	// The context passed in has the associated *tls.ClientHelloInfo
 | |
| 	// value available at the certmagic.ClientHelloInfoCtxKey key.
 | |
| 	//
 | |
| 	// In the worst case, this function may be called as frequently
 | |
| 	// as every TLS handshake, so it should return as quick as possible
 | |
| 	// to reduce latency. In the normal case, this function is only
 | |
| 	// called when a certificate is needed that is not already loaded
 | |
| 	// into memory ready to serve.
 | |
| 	CertificateAllowed(ctx context.Context, name string) error
 | |
| }
 | |
| 
 | |
| // PermissionByHTTP determines permission for a TLS certificate by
 | |
| // making a request to an HTTP endpoint.
 | |
| type PermissionByHTTP struct {
 | |
| 	// The endpoint to access. It should be a full URL.
 | |
| 	// A query string parameter "domain" will be added to it,
 | |
| 	// containing the domain (or IP) for the desired certificate,
 | |
| 	// like so: `?domain=example.com`. Generally, this endpoint
 | |
| 	// is not exposed publicly to avoid a minor information leak
 | |
| 	// (which domains are serviced by your application).
 | |
| 	//
 | |
| 	// The endpoint must return a 200 OK status if a certificate
 | |
| 	// is allowed; anything else will cause it to be denied.
 | |
| 	// Redirects are not followed.
 | |
| 	Endpoint string `json:"endpoint"`
 | |
| 
 | |
| 	logger   *zap.Logger
 | |
| 	replacer *caddy.Replacer
 | |
| }
 | |
| 
 | |
| // CaddyModule returns the Caddy module information.
 | |
| func (PermissionByHTTP) CaddyModule() caddy.ModuleInfo {
 | |
| 	return caddy.ModuleInfo{
 | |
| 		ID:  "tls.permission.http",
 | |
| 		New: func() caddy.Module { return new(PermissionByHTTP) },
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (p *PermissionByHTTP) Provision(ctx caddy.Context) error {
 | |
| 	p.logger = ctx.Logger()
 | |
| 	p.replacer = caddy.NewReplacer()
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (p PermissionByHTTP) CertificateAllowed(ctx context.Context, name string) error {
 | |
| 	// run replacer on endpoint URL (for environment variables) -- return errors to prevent surprises (#5036)
 | |
| 	askEndpoint, err := p.replacer.ReplaceOrErr(p.Endpoint, true, true)
 | |
| 	if err != nil {
 | |
| 		return fmt.Errorf("preparing 'ask' endpoint: %v", err)
 | |
| 	}
 | |
| 
 | |
| 	askURL, err := url.Parse(askEndpoint)
 | |
| 	if err != nil {
 | |
| 		return fmt.Errorf("parsing ask URL: %v", err)
 | |
| 	}
 | |
| 	qs := askURL.Query()
 | |
| 	qs.Set("domain", name)
 | |
| 	askURL.RawQuery = qs.Encode()
 | |
| 	askURLString := askURL.String()
 | |
| 
 | |
| 	var remote string
 | |
| 	if chi, ok := ctx.Value(certmagic.ClientHelloInfoCtxKey).(*tls.ClientHelloInfo); ok && chi != nil {
 | |
| 		remote = chi.Conn.RemoteAddr().String()
 | |
| 	}
 | |
| 
 | |
| 	p.logger.Debug("asking permission endpoint",
 | |
| 		zap.String("remote", remote),
 | |
| 		zap.String("domain", name),
 | |
| 		zap.String("url", askURLString))
 | |
| 
 | |
| 	resp, err := onDemandAskClient.Get(askURLString)
 | |
| 	if err != nil {
 | |
| 		return fmt.Errorf("checking %v to determine if certificate for hostname '%s' should be allowed: %v",
 | |
| 			askEndpoint, name, err)
 | |
| 	}
 | |
| 	resp.Body.Close()
 | |
| 
 | |
| 	p.logger.Debug("response from permission endpoint",
 | |
| 		zap.String("remote", remote),
 | |
| 		zap.String("domain", name),
 | |
| 		zap.String("url", askURLString),
 | |
| 		zap.Int("status", resp.StatusCode))
 | |
| 
 | |
| 	if resp.StatusCode < 200 || resp.StatusCode > 299 {
 | |
| 		return fmt.Errorf("%s: %w %s - non-2xx status code %d", name, ErrPermissionDenied, askEndpoint, resp.StatusCode)
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // ErrPermissionDenied is an error that should be wrapped or returned when the
 | |
| // configured permission module does not allow a certificate to be issued,
 | |
| // to distinguish that from other errors such as connection failure.
 | |
| var ErrPermissionDenied = errors.New("certificate not allowed by permission module")
 | |
| 
 | |
| // These perpetual values are used for on-demand TLS.
 | |
| var (
 | |
| 	onDemandRateLimiter = certmagic.NewRateLimiter(0, 0)
 | |
| 	onDemandAskClient   = &http.Client{
 | |
| 		Timeout: 10 * time.Second,
 | |
| 		CheckRedirect: func(req *http.Request, via []*http.Request) error {
 | |
| 			return fmt.Errorf("following http redirects is not allowed")
 | |
| 		},
 | |
| 	}
 | |
| )
 | |
| 
 | |
| // Interface guards
 | |
| var (
 | |
| 	_ OnDemandPermission = (*PermissionByHTTP)(nil)
 | |
| 	_ caddy.Provisioner  = (*PermissionByHTTP)(nil)
 | |
| )
 |