mirror of
https://github.com/yusing/godoxy.git
synced 2025-05-20 04:42:33 +02:00
[BREAKING] added entrypoint middleware support and config, config schema update
This commit is contained in:
parent
3af3a88f66
commit
1c1ba1b55e
9 changed files with 274 additions and 74 deletions
|
@ -137,14 +137,13 @@ func main() {
|
||||||
HTTPAddr: common.ProxyHTTPAddr,
|
HTTPAddr: common.ProxyHTTPAddr,
|
||||||
HTTPSAddr: common.ProxyHTTPSAddr,
|
HTTPSAddr: common.ProxyHTTPSAddr,
|
||||||
Handler: http.HandlerFunc(entrypoint.Handler),
|
Handler: http.HandlerFunc(entrypoint.Handler),
|
||||||
RedirectToHTTPS: config.Value().RedirectToHTTPS,
|
RedirectToHTTPS: config.Value().Entrypoint.RedirectToHTTPS,
|
||||||
})
|
})
|
||||||
server.StartServer(server.Options{
|
server.StartServer(server.Options{
|
||||||
Name: "api",
|
Name: "api",
|
||||||
CertProvider: autocert,
|
CertProvider: autocert,
|
||||||
HTTPAddr: common.APIHTTPAddr,
|
HTTPAddr: common.APIHTTPAddr,
|
||||||
Handler: api.NewHandler(),
|
Handler: api.NewHandler(),
|
||||||
RedirectToHTTPS: config.Value().RedirectToHTTPS,
|
|
||||||
})
|
})
|
||||||
|
|
||||||
if common.PrometheusEnabled {
|
if common.PrometheusEnabled {
|
||||||
|
@ -153,7 +152,6 @@ func main() {
|
||||||
CertProvider: autocert,
|
CertProvider: autocert,
|
||||||
HTTPAddr: common.MetricsHTTPAddr,
|
HTTPAddr: common.MetricsHTTPAddr,
|
||||||
Handler: metrics.NewHandler(),
|
Handler: metrics.NewHandler(),
|
||||||
RedirectToHTTPS: config.Value().RedirectToHTTPS,
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -20,6 +20,20 @@
|
||||||
#
|
#
|
||||||
# 3. other providers, check docs/dns_providers.md for more
|
# 3. other providers, check docs/dns_providers.md for more
|
||||||
|
|
||||||
|
entrypoint:
|
||||||
|
# global setting redirect http requests to https (if https available, otherwise this will be ignored)
|
||||||
|
# proxy.<alias>.middlewares.redirect_http will override this
|
||||||
|
#
|
||||||
|
redirect_to_https: false
|
||||||
|
middlewares:
|
||||||
|
- use: CIDRWhitelist
|
||||||
|
allow:
|
||||||
|
- "127.0.0.1"
|
||||||
|
- "10.0.0.0/8"
|
||||||
|
- "192.168.0.0/16"
|
||||||
|
status: 403
|
||||||
|
message: "Forbidden"
|
||||||
|
|
||||||
providers:
|
providers:
|
||||||
# include files are standalone yaml files under `config/` directory
|
# include files are standalone yaml files under `config/` directory
|
||||||
#
|
#
|
||||||
|
@ -41,6 +55,28 @@ providers:
|
||||||
#
|
#
|
||||||
# remote-1: tcp://10.0.2.1:2375
|
# remote-1: tcp://10.0.2.1:2375
|
||||||
# remote-2: ssh://root:1234@10.0.2.2
|
# remote-2: ssh://root:1234@10.0.2.2
|
||||||
|
|
||||||
|
# notification providers (notify when service health changes)
|
||||||
|
#
|
||||||
|
# notification:
|
||||||
|
# - name: gotify
|
||||||
|
# provider: gotify
|
||||||
|
# url: https://gotify.domain.tld
|
||||||
|
# token: abcd
|
||||||
|
# - name: discord
|
||||||
|
# provider: webhook
|
||||||
|
# url: https://discord.com/api/webhooks/...
|
||||||
|
# template: discord
|
||||||
|
# # payload: | # discord template implies the following
|
||||||
|
# # {
|
||||||
|
# # "embeds": [
|
||||||
|
# # {
|
||||||
|
# # "title": $title,
|
||||||
|
# # "fields": $fields,
|
||||||
|
# # "color": "$color"
|
||||||
|
# # }
|
||||||
|
# # ]
|
||||||
|
# # }
|
||||||
# if match_domains not defined
|
# if match_domains not defined
|
||||||
# any host = alias+[any domain] will match
|
# any host = alias+[any domain] will match
|
||||||
# i.e. https://app1.y.z will match alias app1 for any domain y.z
|
# i.e. https://app1.y.z will match alias app1 for any domain y.z
|
||||||
|
@ -68,8 +104,3 @@ homepage:
|
||||||
# timeout for shutdown (in seconds)
|
# timeout for shutdown (in seconds)
|
||||||
#
|
#
|
||||||
timeout_shutdown: 5
|
timeout_shutdown: 5
|
||||||
|
|
||||||
# global setting redirect http requests to https (if https available, otherwise this will be ignored)
|
|
||||||
# proxy.<alias>.middlewares.redirect_http will override this
|
|
||||||
#
|
|
||||||
redirect_to_https: false
|
|
||||||
|
|
|
@ -173,6 +173,7 @@ func (cfg *Config) load() E.Error {
|
||||||
|
|
||||||
// errors are non fatal below
|
// errors are non fatal below
|
||||||
errs := E.NewBuilder(errMsg)
|
errs := E.NewBuilder(errMsg)
|
||||||
|
errs.Add(entrypoint.SetMiddlewares(model.Entrypoint.Middlewares))
|
||||||
errs.Add(cfg.initNotification(model.Providers.Notification))
|
errs.Add(cfg.initNotification(model.Providers.Notification))
|
||||||
errs.Add(cfg.initAutoCert(&model.AutoCert))
|
errs.Add(cfg.initAutoCert(&model.AutoCert))
|
||||||
errs.Add(cfg.loadRouteProviders(&model.Providers))
|
errs.Add(cfg.loadRouteProviders(&model.Providers))
|
||||||
|
|
|
@ -2,19 +2,22 @@ package types
|
||||||
|
|
||||||
type (
|
type (
|
||||||
Config struct {
|
Config struct {
|
||||||
Providers Providers `json:"providers" yaml:",flow"`
|
|
||||||
AutoCert AutoCertConfig `json:"autocert" yaml:",flow"`
|
AutoCert AutoCertConfig `json:"autocert" yaml:",flow"`
|
||||||
ExplicitOnly bool `json:"explicit_only" yaml:"explicit_only"`
|
Entrypoint Entrypoint `json:"entrypoint" yaml:",flow"`
|
||||||
|
Providers Providers `json:"providers" yaml:",flow"`
|
||||||
MatchDomains []string `json:"match_domains" yaml:"match_domains"`
|
MatchDomains []string `json:"match_domains" yaml:"match_domains"`
|
||||||
Homepage HomepageConfig `json:"homepage" yaml:"homepage"`
|
Homepage HomepageConfig `json:"homepage" yaml:"homepage"`
|
||||||
TimeoutShutdown int `json:"timeout_shutdown" yaml:"timeout_shutdown"`
|
TimeoutShutdown int `json:"timeout_shutdown" yaml:"timeout_shutdown"`
|
||||||
RedirectToHTTPS bool `json:"redirect_to_https" yaml:"redirect_to_https"`
|
|
||||||
}
|
}
|
||||||
Providers struct {
|
Providers struct {
|
||||||
Files []string `json:"include" yaml:"include"`
|
Files []string `json:"include" yaml:"include"`
|
||||||
Docker map[string]string `json:"docker" yaml:"docker"`
|
Docker map[string]string `json:"docker" yaml:"docker"`
|
||||||
Notification []NotificationConfig `json:"notification" yaml:"notification"`
|
Notification []NotificationConfig `json:"notification" yaml:"notification"`
|
||||||
}
|
}
|
||||||
|
Entrypoint struct {
|
||||||
|
RedirectToHTTPS bool `json:"redirect_to_https" yaml:"redirect_to_https"`
|
||||||
|
Middlewares []map[string]any
|
||||||
|
}
|
||||||
NotificationConfig map[string]any
|
NotificationConfig map[string]any
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -24,6 +27,8 @@ func DefaultConfig() *Config {
|
||||||
Homepage: HomepageConfig{
|
Homepage: HomepageConfig{
|
||||||
UseDefaultCategories: true,
|
UseDefaultCategories: true,
|
||||||
},
|
},
|
||||||
|
Entrypoint: Entrypoint{
|
||||||
RedirectToHTTPS: false,
|
RedirectToHTTPS: false,
|
||||||
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,6 +5,7 @@ import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"net/http"
|
"net/http"
|
||||||
"strings"
|
"strings"
|
||||||
|
"sync"
|
||||||
|
|
||||||
"github.com/yusing/go-proxy/internal/net/http/middleware"
|
"github.com/yusing/go-proxy/internal/net/http/middleware"
|
||||||
"github.com/yusing/go-proxy/internal/net/http/middleware/errorpage"
|
"github.com/yusing/go-proxy/internal/net/http/middleware/errorpage"
|
||||||
|
@ -14,6 +15,11 @@ import (
|
||||||
|
|
||||||
var findRouteFunc = findRouteAnyDomain
|
var findRouteFunc = findRouteAnyDomain
|
||||||
|
|
||||||
|
var (
|
||||||
|
epMiddleware *middleware.Middleware
|
||||||
|
epMiddlewareMu sync.Mutex
|
||||||
|
)
|
||||||
|
|
||||||
func SetFindRouteDomains(domains []string) {
|
func SetFindRouteDomains(domains []string) {
|
||||||
if len(domains) == 0 {
|
if len(domains) == 0 {
|
||||||
findRouteFunc = findRouteAnyDomain
|
findRouteFunc = findRouteAnyDomain
|
||||||
|
@ -22,9 +28,25 @@ func SetFindRouteDomains(domains []string) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func SetMiddlewares(mws []map[string]any) error {
|
||||||
|
epMiddlewareMu.Lock()
|
||||||
|
defer epMiddlewareMu.Unlock()
|
||||||
|
|
||||||
|
mid, err := middleware.BuildMiddlewareFromChainRaw("entrypoint", mws)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
epMiddleware = mid
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
func Handler(w http.ResponseWriter, r *http.Request) {
|
func Handler(w http.ResponseWriter, r *http.Request) {
|
||||||
mux, err := findRouteFunc(r.Host)
|
mux, err := findRouteFunc(r.Host)
|
||||||
if err == nil {
|
if err == nil {
|
||||||
|
if epMiddleware != nil {
|
||||||
|
epMiddleware.ServeHTTP(mux.ServeHTTP, w, r)
|
||||||
|
return
|
||||||
|
}
|
||||||
mux.ServeHTTP(w, r)
|
mux.ServeHTTP(w, r)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
|
@ -23,7 +23,7 @@ type (
|
||||||
|
|
||||||
BeforeFunc func(next http.HandlerFunc, w ResponseWriter, r *Request)
|
BeforeFunc func(next http.HandlerFunc, w ResponseWriter, r *Request)
|
||||||
RewriteFunc func(req *Request)
|
RewriteFunc func(req *Request)
|
||||||
ModifyResponseFunc func(resp *Response) error
|
ModifyResponseFunc = gphttp.ModifyResponseFunc
|
||||||
CloneWithOptFunc func(opts OptionsRaw) (*Middleware, E.Error)
|
CloneWithOptFunc func(opts OptionsRaw) (*Middleware, E.Error)
|
||||||
|
|
||||||
OptionsRaw = map[string]any
|
OptionsRaw = map[string]any
|
||||||
|
@ -114,6 +114,17 @@ func (m *Middleware) ModifyResponse(resp *Response) error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (m *Middleware) ServeHTTP(next http.HandlerFunc, w ResponseWriter, r *Request) {
|
||||||
|
if m.modifyResponse != nil {
|
||||||
|
w = gphttp.NewModifyResponseWriter(w, r, m.modifyResponse)
|
||||||
|
}
|
||||||
|
if m.before != nil {
|
||||||
|
m.before(next, w, r)
|
||||||
|
} else {
|
||||||
|
next(w, r)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// TODO: check conflict or duplicates.
|
// TODO: check conflict or duplicates.
|
||||||
func createMiddlewares(middlewaresMap map[string]OptionsRaw) ([]*Middleware, E.Error) {
|
func createMiddlewares(middlewaresMap map[string]OptionsRaw) ([]*Middleware, E.Error) {
|
||||||
middlewares := make([]*Middleware, 0, len(middlewaresMap))
|
middlewares := make([]*Middleware, 0, len(middlewaresMap))
|
||||||
|
|
|
@ -11,13 +11,20 @@ import (
|
||||||
"gopkg.in/yaml.v3"
|
"gopkg.in/yaml.v3"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var ErrMissingMiddlewareUse = E.New("missing middleware 'use' field")
|
||||||
|
|
||||||
func BuildMiddlewaresFromComposeFile(filePath string, eb *E.Builder) map[string]*Middleware {
|
func BuildMiddlewaresFromComposeFile(filePath string, eb *E.Builder) map[string]*Middleware {
|
||||||
fileContent, err := os.ReadFile(filePath)
|
fileContent, err := os.ReadFile(filePath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
eb.Add(err)
|
eb.Add(err)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
return BuildMiddlewaresFromYAML(path.Base(filePath), fileContent, eb)
|
mids := BuildMiddlewaresFromYAML(path.Base(filePath), fileContent, eb)
|
||||||
|
results := make(map[string]*Middleware, len(mids))
|
||||||
|
for k, v := range mids {
|
||||||
|
results[k+"@file"] = v
|
||||||
|
}
|
||||||
|
return results
|
||||||
}
|
}
|
||||||
|
|
||||||
func BuildMiddlewaresFromYAML(source string, data []byte, eb *E.Builder) map[string]*Middleware {
|
func BuildMiddlewaresFromYAML(source string, data []byte, eb *E.Builder) map[string]*Middleware {
|
||||||
|
@ -29,11 +36,22 @@ func BuildMiddlewaresFromYAML(source string, data []byte, eb *E.Builder) map[str
|
||||||
}
|
}
|
||||||
middlewares := make(map[string]*Middleware)
|
middlewares := make(map[string]*Middleware)
|
||||||
for name, defs := range rawMap {
|
for name, defs := range rawMap {
|
||||||
|
chain, err := BuildMiddlewareFromChainRaw(name, defs)
|
||||||
|
if err != nil {
|
||||||
|
eb.Add(err.Subject(source))
|
||||||
|
} else {
|
||||||
|
middlewares[name] = chain
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return middlewares
|
||||||
|
}
|
||||||
|
|
||||||
|
func BuildMiddlewareFromChainRaw(name string, defs []map[string]any) (*Middleware, E.Error) {
|
||||||
chainErr := E.NewBuilder("")
|
chainErr := E.NewBuilder("")
|
||||||
chain := make([]*Middleware, 0, len(defs))
|
chain := make([]*Middleware, 0, len(defs))
|
||||||
for i, def := range defs {
|
for i, def := range defs {
|
||||||
if def["use"] == nil || def["use"] == "" {
|
if def["use"] == nil || def["use"] == "" {
|
||||||
chainErr.Addf("item %d: missing field 'use'", i)
|
chainErr.Add(ErrMissingMiddlewareUse.Subjectf("%s[%d]", name, i))
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
baseName := def["use"].(string)
|
baseName := def["use"].(string)
|
||||||
|
@ -52,13 +70,11 @@ func BuildMiddlewaresFromYAML(source string, data []byte, eb *E.Builder) map[str
|
||||||
chain = append(chain, m)
|
chain = append(chain, m)
|
||||||
}
|
}
|
||||||
if chainErr.HasError() {
|
if chainErr.HasError() {
|
||||||
eb.Add(chainErr.Error().Subject(source))
|
return nil, chainErr.Error()
|
||||||
} else {
|
} else {
|
||||||
middlewares[name+"@file"] = BuildMiddlewareFromChain(name, chain)
|
return BuildMiddlewareFromChain(name, chain), nil
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return middlewares
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO: check conflict or duplicates.
|
// TODO: check conflict or duplicates.
|
||||||
func BuildMiddlewareFromChain(name string, chain []*Middleware) *Middleware {
|
func BuildMiddlewareFromChain(name string, chain []*Middleware) *Middleware {
|
||||||
|
|
|
@ -6,6 +6,7 @@ import (
|
||||||
"errors"
|
"errors"
|
||||||
"io"
|
"io"
|
||||||
"log"
|
"log"
|
||||||
|
"net"
|
||||||
"net/http"
|
"net/http"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
@ -57,7 +58,11 @@ func NewServer(opt Options) (s *Server) {
|
||||||
}
|
}
|
||||||
|
|
||||||
if certAvailable && opt.RedirectToHTTPS && opt.HTTPSAddr != "" {
|
if certAvailable && opt.RedirectToHTTPS && opt.HTTPSAddr != "" {
|
||||||
httpHandler = redirectToTLSHandler(opt.HTTPSAddr)
|
_, port, err := net.SplitHostPort(opt.HTTPSAddr)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
httpHandler = redirectToTLSHandler(port)
|
||||||
} else {
|
} else {
|
||||||
httpHandler = opt.Handler
|
httpHandler = opt.Handler
|
||||||
}
|
}
|
||||||
|
@ -151,7 +156,7 @@ func (s *Server) handleErr(scheme string, err error) {
|
||||||
func redirectToTLSHandler(port string) http.HandlerFunc {
|
func redirectToTLSHandler(port string) http.HandlerFunc {
|
||||||
return func(w http.ResponseWriter, r *http.Request) {
|
return func(w http.ResponseWriter, r *http.Request) {
|
||||||
r.URL.Scheme = "https"
|
r.URL.Scheme = "https"
|
||||||
r.URL.Host = r.URL.Hostname() + port
|
r.URL.Host = r.URL.Hostname() + ":" + port
|
||||||
|
|
||||||
var redirectCode int
|
var redirectCode int
|
||||||
if r.Method == http.MethodGet {
|
if r.Method == http.MethodGet {
|
||||||
|
|
|
@ -303,14 +303,36 @@
|
||||||
},
|
},
|
||||||
"notification": {
|
"notification": {
|
||||||
"description": "Notification provider configuration",
|
"description": "Notification provider configuration",
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
"type": "object",
|
"type": "object",
|
||||||
"additionalProperties": false,
|
"required": [
|
||||||
|
"name",
|
||||||
|
"provider"
|
||||||
|
],
|
||||||
"properties": {
|
"properties": {
|
||||||
"gotify": {
|
"name": {
|
||||||
|
"type": "string",
|
||||||
|
"description": "Notifier name"
|
||||||
|
},
|
||||||
|
"provider": {
|
||||||
|
"description": "Notifier provider",
|
||||||
|
"type": "string",
|
||||||
|
"enum": [
|
||||||
|
"gotify",
|
||||||
|
"webhook"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"oneOf": [
|
||||||
|
{
|
||||||
"description": "Gotify configuration",
|
"description": "Gotify configuration",
|
||||||
"type": "object",
|
|
||||||
"additionalProperties": false,
|
"additionalProperties": false,
|
||||||
"properties": {
|
"properties": {
|
||||||
|
"name": {},
|
||||||
|
"provider": {
|
||||||
|
"const": "gotify"
|
||||||
|
},
|
||||||
"url": {
|
"url": {
|
||||||
"description": "Gotify URL",
|
"description": "Gotify URL",
|
||||||
"type": "string"
|
"type": "string"
|
||||||
|
@ -324,7 +346,62 @@
|
||||||
"url",
|
"url",
|
||||||
"token"
|
"token"
|
||||||
]
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"description": "Webhook configuration",
|
||||||
|
"additionalProperties": false,
|
||||||
|
"properties": {
|
||||||
|
"name": {},
|
||||||
|
"provider": {
|
||||||
|
"const": "webhook"
|
||||||
|
},
|
||||||
|
"url": {
|
||||||
|
"description": "Webhook URL",
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
"token": {
|
||||||
|
"description": "Webhook bearer token",
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
"template": {
|
||||||
|
"description": "Webhook template",
|
||||||
|
"type": "string",
|
||||||
|
"enum": [
|
||||||
|
"discord"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"payload": {
|
||||||
|
"description": "Webhook payload",
|
||||||
|
"type": "string",
|
||||||
|
"format": "json"
|
||||||
|
},
|
||||||
|
"method": {
|
||||||
|
"description": "Webhook request method",
|
||||||
|
"type": "string",
|
||||||
|
"enum": [
|
||||||
|
"GET",
|
||||||
|
"POST",
|
||||||
|
"PUT"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"mime_type": {
|
||||||
|
"description": "Webhook NIME type",
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
"color_mode": {
|
||||||
|
"description": "Webhook color mode",
|
||||||
|
"type": "string",
|
||||||
|
"enum": [
|
||||||
|
"hex",
|
||||||
|
"dec"
|
||||||
|
]
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
"required": [
|
||||||
|
"url"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -337,14 +414,48 @@
|
||||||
},
|
},
|
||||||
"minItems": 1
|
"minItems": 1
|
||||||
},
|
},
|
||||||
|
"homepage": {
|
||||||
|
"title": "Homepage configuration",
|
||||||
|
"type": "object",
|
||||||
|
"additionalProperties": false,
|
||||||
|
"properties": {
|
||||||
|
"use_default_categories": {
|
||||||
|
"title": "Use default categories",
|
||||||
|
"type": "boolean"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"entrypoint": {
|
||||||
|
"title": "Entrypoint configuration",
|
||||||
|
"type": "object",
|
||||||
|
"additionalProperties": false,
|
||||||
|
"properties": {
|
||||||
|
"redirect_to_https": {
|
||||||
|
"title": "Redirect to HTTPS on HTTP requests",
|
||||||
|
"type": "boolean"
|
||||||
|
},
|
||||||
|
"middlewares": {
|
||||||
|
"title": "Entrypoint middlewares",
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"type": "object",
|
||||||
|
"required": [
|
||||||
|
"use"
|
||||||
|
],
|
||||||
|
"properties": {
|
||||||
|
"use": {
|
||||||
|
"type": "string",
|
||||||
|
"description": "Middleware to use"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
"timeout_shutdown": {
|
"timeout_shutdown": {
|
||||||
"title": "Shutdown timeout (in seconds)",
|
"title": "Shutdown timeout (in seconds)",
|
||||||
"type": "integer",
|
"type": "integer",
|
||||||
"minimum": 0
|
"minimum": 0
|
||||||
},
|
|
||||||
"redirect_to_https": {
|
|
||||||
"title": "Redirect to HTTPS on HTTP requests",
|
|
||||||
"type": "boolean"
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"additionalProperties": false,
|
"additionalProperties": false,
|
||||||
|
|
Loading…
Add table
Reference in a new issue