2019-08-22 22:38:37 +03:00
|
|
|
// 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 httpcaddyfile
|
|
|
|
|
|
|
|
import (
|
|
|
|
"strconv"
|
|
|
|
|
2019-09-19 21:42:36 +03:00
|
|
|
"github.com/caddyserver/caddy/v2"
|
2019-08-22 22:38:37 +03:00
|
|
|
"github.com/caddyserver/caddy/v2/caddyconfig/caddyfile"
|
2020-03-18 06:00:45 +03:00
|
|
|
"github.com/caddyserver/caddy/v2/modules/caddytls"
|
2020-07-31 00:18:14 +03:00
|
|
|
"github.com/mholt/acmez/acme"
|
2019-08-22 22:38:37 +03:00
|
|
|
)
|
|
|
|
|
2020-05-12 00:00:35 +03:00
|
|
|
func init() {
|
|
|
|
RegisterGlobalOption("debug", parseOptTrue)
|
|
|
|
RegisterGlobalOption("http_port", parseOptHTTPPort)
|
|
|
|
RegisterGlobalOption("https_port", parseOptHTTPSPort)
|
|
|
|
RegisterGlobalOption("default_sni", parseOptSingleString)
|
|
|
|
RegisterGlobalOption("order", parseOptOrder)
|
|
|
|
RegisterGlobalOption("experimental_http3", parseOptTrue)
|
|
|
|
RegisterGlobalOption("storage", parseOptStorage)
|
|
|
|
RegisterGlobalOption("acme_ca", parseOptSingleString)
|
|
|
|
RegisterGlobalOption("acme_ca_root", parseOptSingleString)
|
2020-06-12 22:37:56 +03:00
|
|
|
RegisterGlobalOption("acme_dns", parseOptSingleString)
|
|
|
|
RegisterGlobalOption("acme_eab", parseOptACMEEAB)
|
2020-05-12 00:00:35 +03:00
|
|
|
RegisterGlobalOption("email", parseOptSingleString)
|
|
|
|
RegisterGlobalOption("admin", parseOptAdmin)
|
|
|
|
RegisterGlobalOption("on_demand_tls", parseOptOnDemand)
|
|
|
|
RegisterGlobalOption("local_certs", parseOptTrue)
|
|
|
|
RegisterGlobalOption("key_type", parseOptSingleString)
|
2020-05-20 01:59:51 +03:00
|
|
|
RegisterGlobalOption("auto_https", parseOptAutoHTTPS)
|
2020-05-12 00:00:35 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
func parseOptTrue(d *caddyfile.Dispenser) (interface{}, error) {
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func parseOptHTTPPort(d *caddyfile.Dispenser) (interface{}, error) {
|
2019-08-22 22:38:37 +03:00
|
|
|
var httpPort int
|
|
|
|
for d.Next() {
|
|
|
|
var httpPortStr string
|
|
|
|
if !d.AllArgs(&httpPortStr) {
|
|
|
|
return 0, d.ArgErr()
|
|
|
|
}
|
|
|
|
var err error
|
|
|
|
httpPort, err = strconv.Atoi(httpPortStr)
|
|
|
|
if err != nil {
|
|
|
|
return 0, d.Errf("converting port '%s' to integer value: %v", httpPortStr, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return httpPort, nil
|
|
|
|
}
|
|
|
|
|
2020-05-12 00:00:35 +03:00
|
|
|
func parseOptHTTPSPort(d *caddyfile.Dispenser) (interface{}, error) {
|
2019-08-22 22:38:37 +03:00
|
|
|
var httpsPort int
|
|
|
|
for d.Next() {
|
|
|
|
var httpsPortStr string
|
|
|
|
if !d.AllArgs(&httpsPortStr) {
|
|
|
|
return 0, d.ArgErr()
|
|
|
|
}
|
|
|
|
var err error
|
|
|
|
httpsPort, err = strconv.Atoi(httpsPortStr)
|
|
|
|
if err != nil {
|
|
|
|
return 0, d.Errf("converting port '%s' to integer value: %v", httpsPortStr, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return httpsPort, nil
|
|
|
|
}
|
2019-08-22 23:26:33 +03:00
|
|
|
|
2020-05-12 00:00:35 +03:00
|
|
|
func parseOptOrder(d *caddyfile.Dispenser) (interface{}, error) {
|
2020-01-16 22:09:54 +03:00
|
|
|
newOrder := directiveOrder
|
|
|
|
|
|
|
|
for d.Next() {
|
|
|
|
// get directive name
|
|
|
|
if !d.Next() {
|
|
|
|
return nil, d.ArgErr()
|
|
|
|
}
|
|
|
|
dirName := d.Val()
|
|
|
|
if _, ok := registeredDirectives[dirName]; !ok {
|
2020-03-18 06:00:45 +03:00
|
|
|
return nil, d.Errf("%s is not a registered directive", dirName)
|
2020-01-16 22:09:54 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// get positional token
|
|
|
|
if !d.Next() {
|
|
|
|
return nil, d.ArgErr()
|
|
|
|
}
|
|
|
|
pos := d.Val()
|
|
|
|
|
|
|
|
// if directive exists, first remove it
|
|
|
|
for i, d := range newOrder {
|
|
|
|
if d == dirName {
|
|
|
|
newOrder = append(newOrder[:i], newOrder[i+1:]...)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// act on the positional
|
|
|
|
switch pos {
|
|
|
|
case "first":
|
|
|
|
newOrder = append([]string{dirName}, newOrder...)
|
|
|
|
if d.NextArg() {
|
|
|
|
return nil, d.ArgErr()
|
|
|
|
}
|
|
|
|
directiveOrder = newOrder
|
|
|
|
return newOrder, nil
|
|
|
|
case "last":
|
|
|
|
newOrder = append(newOrder, dirName)
|
|
|
|
if d.NextArg() {
|
|
|
|
return nil, d.ArgErr()
|
|
|
|
}
|
|
|
|
directiveOrder = newOrder
|
|
|
|
return newOrder, nil
|
|
|
|
case "before":
|
|
|
|
case "after":
|
|
|
|
default:
|
2020-03-18 06:00:45 +03:00
|
|
|
return nil, d.Errf("unknown positional '%s'", pos)
|
2020-01-16 22:09:54 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// get name of other directive
|
|
|
|
if !d.NextArg() {
|
|
|
|
return nil, d.ArgErr()
|
|
|
|
}
|
|
|
|
otherDir := d.Val()
|
2019-08-22 23:26:33 +03:00
|
|
|
if d.NextArg() {
|
|
|
|
return nil, d.ArgErr()
|
|
|
|
}
|
2020-01-16 22:09:54 +03:00
|
|
|
|
|
|
|
// insert directive into proper position
|
|
|
|
for i, d := range newOrder {
|
|
|
|
if d == otherDir {
|
|
|
|
if pos == "before" {
|
|
|
|
newOrder = append(newOrder[:i], append([]string{dirName}, newOrder[i:]...)...)
|
|
|
|
} else if pos == "after" {
|
|
|
|
newOrder = append(newOrder[:i+1], append([]string{dirName}, newOrder[i+1:]...)...)
|
|
|
|
}
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
2019-08-22 23:26:33 +03:00
|
|
|
}
|
2020-01-16 22:09:54 +03:00
|
|
|
|
|
|
|
directiveOrder = newOrder
|
|
|
|
|
|
|
|
return newOrder, nil
|
2019-08-22 23:26:33 +03:00
|
|
|
}
|
2019-09-19 21:42:36 +03:00
|
|
|
|
2020-05-12 00:00:35 +03:00
|
|
|
func parseOptStorage(d *caddyfile.Dispenser) (interface{}, error) {
|
2020-05-01 18:34:32 +03:00
|
|
|
if !d.Next() { // consume option name
|
2019-09-19 21:42:36 +03:00
|
|
|
return nil, d.ArgErr()
|
|
|
|
}
|
2020-05-01 18:34:32 +03:00
|
|
|
if !d.Next() { // get storage module name
|
2019-09-19 21:42:36 +03:00
|
|
|
return nil, d.ArgErr()
|
|
|
|
}
|
2020-05-01 18:34:32 +03:00
|
|
|
modName := d.Val()
|
2019-09-19 21:42:36 +03:00
|
|
|
mod, err := caddy.GetModule("caddy.storage." + modName)
|
|
|
|
if err != nil {
|
2020-03-18 06:00:45 +03:00
|
|
|
return nil, d.Errf("getting storage module '%s': %v", modName, err)
|
2019-09-19 21:42:36 +03:00
|
|
|
}
|
|
|
|
unm, ok := mod.New().(caddyfile.Unmarshaler)
|
|
|
|
if !ok {
|
2020-03-18 06:00:45 +03:00
|
|
|
return nil, d.Errf("storage module '%s' is not a Caddyfile unmarshaler", mod.ID)
|
2019-09-19 21:42:36 +03:00
|
|
|
}
|
2020-02-14 21:00:16 +03:00
|
|
|
err = unm.UnmarshalCaddyfile(d.NewFromNextSegment())
|
2019-09-19 21:42:36 +03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
storage, ok := unm.(caddy.StorageConverter)
|
|
|
|
if !ok {
|
2020-03-18 06:00:45 +03:00
|
|
|
return nil, d.Errf("module %s is not a StorageConverter", mod.ID)
|
2019-09-19 21:42:36 +03:00
|
|
|
}
|
|
|
|
return storage, nil
|
|
|
|
}
|
2019-09-30 18:11:30 +03:00
|
|
|
|
2020-06-12 22:37:56 +03:00
|
|
|
func parseOptACMEEAB(d *caddyfile.Dispenser) (interface{}, error) {
|
2020-07-31 00:18:14 +03:00
|
|
|
eab := new(acme.EAB)
|
2020-06-12 22:37:56 +03:00
|
|
|
for d.Next() {
|
|
|
|
if d.NextArg() {
|
|
|
|
return nil, d.ArgErr()
|
|
|
|
}
|
|
|
|
for nesting := d.Nesting(); d.NextBlock(nesting); {
|
|
|
|
switch d.Val() {
|
|
|
|
case "key_id":
|
|
|
|
if !d.NextArg() {
|
|
|
|
return nil, d.ArgErr()
|
|
|
|
}
|
|
|
|
eab.KeyID = d.Val()
|
|
|
|
|
2020-07-31 00:18:14 +03:00
|
|
|
case "mac_key":
|
2020-06-12 22:37:56 +03:00
|
|
|
if !d.NextArg() {
|
|
|
|
return nil, d.ArgErr()
|
|
|
|
}
|
2020-07-31 00:18:14 +03:00
|
|
|
eab.MACKey = d.Val()
|
2020-06-12 22:37:56 +03:00
|
|
|
|
|
|
|
default:
|
|
|
|
return nil, d.Errf("unrecognized parameter '%s'", d.Val())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return eab, nil
|
|
|
|
}
|
|
|
|
|
2020-05-12 00:00:35 +03:00
|
|
|
func parseOptSingleString(d *caddyfile.Dispenser) (interface{}, error) {
|
2019-09-30 18:11:30 +03:00
|
|
|
d.Next() // consume parameter name
|
|
|
|
if !d.Next() {
|
|
|
|
return "", d.ArgErr()
|
|
|
|
}
|
|
|
|
val := d.Val()
|
|
|
|
if d.Next() {
|
|
|
|
return "", d.ArgErr()
|
|
|
|
}
|
|
|
|
return val, nil
|
|
|
|
}
|
2019-10-31 00:12:42 +03:00
|
|
|
|
2020-05-12 00:00:35 +03:00
|
|
|
func parseOptAdmin(d *caddyfile.Dispenser) (interface{}, error) {
|
2019-10-31 00:12:42 +03:00
|
|
|
if d.Next() {
|
|
|
|
var listenAddress string
|
2020-03-10 17:25:26 +03:00
|
|
|
if !d.AllArgs(&listenAddress) {
|
|
|
|
return "", d.ArgErr()
|
|
|
|
}
|
2019-10-31 00:12:42 +03:00
|
|
|
if listenAddress == "" {
|
|
|
|
listenAddress = caddy.DefaultAdminListen
|
|
|
|
}
|
|
|
|
return listenAddress, nil
|
|
|
|
}
|
|
|
|
return "", nil
|
|
|
|
}
|
2020-03-18 06:00:45 +03:00
|
|
|
|
2020-05-12 00:00:35 +03:00
|
|
|
func parseOptOnDemand(d *caddyfile.Dispenser) (interface{}, error) {
|
2020-03-18 06:00:45 +03:00
|
|
|
var ond *caddytls.OnDemandConfig
|
|
|
|
for d.Next() {
|
|
|
|
if d.NextArg() {
|
|
|
|
return nil, d.ArgErr()
|
|
|
|
}
|
|
|
|
for nesting := d.Nesting(); d.NextBlock(nesting); {
|
|
|
|
switch d.Val() {
|
|
|
|
case "ask":
|
|
|
|
if !d.NextArg() {
|
|
|
|
return nil, d.ArgErr()
|
|
|
|
}
|
|
|
|
if ond == nil {
|
|
|
|
ond = new(caddytls.OnDemandConfig)
|
|
|
|
}
|
|
|
|
ond.Ask = d.Val()
|
|
|
|
|
|
|
|
case "interval":
|
|
|
|
if !d.NextArg() {
|
|
|
|
return nil, d.ArgErr()
|
|
|
|
}
|
2020-05-12 01:41:11 +03:00
|
|
|
dur, err := caddy.ParseDuration(d.Val())
|
2020-03-18 06:00:45 +03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if ond == nil {
|
|
|
|
ond = new(caddytls.OnDemandConfig)
|
|
|
|
}
|
|
|
|
if ond.RateLimit == nil {
|
|
|
|
ond.RateLimit = new(caddytls.RateLimit)
|
|
|
|
}
|
|
|
|
ond.RateLimit.Interval = caddy.Duration(dur)
|
|
|
|
|
|
|
|
case "burst":
|
|
|
|
if !d.NextArg() {
|
|
|
|
return nil, d.ArgErr()
|
|
|
|
}
|
|
|
|
burst, err := strconv.Atoi(d.Val())
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if ond == nil {
|
|
|
|
ond = new(caddytls.OnDemandConfig)
|
|
|
|
}
|
|
|
|
if ond.RateLimit == nil {
|
|
|
|
ond.RateLimit = new(caddytls.RateLimit)
|
|
|
|
}
|
|
|
|
ond.RateLimit.Burst = burst
|
|
|
|
|
|
|
|
default:
|
|
|
|
return nil, d.Errf("unrecognized parameter '%s'", d.Val())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ond == nil {
|
|
|
|
return nil, d.Err("expected at least one config parameter for on_demand_tls")
|
|
|
|
}
|
|
|
|
return ond, nil
|
|
|
|
}
|
2020-05-20 01:59:51 +03:00
|
|
|
|
|
|
|
func parseOptAutoHTTPS(d *caddyfile.Dispenser) (interface{}, error) {
|
|
|
|
d.Next() // consume parameter name
|
|
|
|
if !d.Next() {
|
|
|
|
return "", d.ArgErr()
|
|
|
|
}
|
|
|
|
val := d.Val()
|
|
|
|
if d.Next() {
|
|
|
|
return "", d.ArgErr()
|
|
|
|
}
|
|
|
|
if val != "off" && val != "disable_redirects" {
|
|
|
|
return "", d.Errf("auto_https must be either 'off' or 'disable_redirects'")
|
|
|
|
}
|
|
|
|
return val, nil
|
|
|
|
}
|