mirror of
https://github.com/caddyserver/caddy.git
synced 2025-01-20 22:52:58 -05:00
f976451d19
This makes it faster and easier to detect broken configurations, but is a slight performance hit on config loads since we have to re-encode the decoded struct back into JSON without the module name's key
194 lines
5.4 KiB
Go
194 lines
5.4 KiB
Go
package caddy2
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/json"
|
|
"fmt"
|
|
"sort"
|
|
"strings"
|
|
"sync"
|
|
)
|
|
|
|
// Module represents a Caddy module.
|
|
type Module struct {
|
|
// Name is the full name of the module. It
|
|
// must be unique and properly namespaced.
|
|
Name string
|
|
|
|
// New returns a new, empty instance of
|
|
// the module's type. The host module
|
|
// which loads this module will likely
|
|
// invoke methods on the returned value.
|
|
// It must return a pointer; if not, it
|
|
// is converted into one.
|
|
New func() interface{}
|
|
}
|
|
|
|
func (m Module) String() string { return m.Name }
|
|
|
|
// RegisterModule registers a module. Modules must call
|
|
// this function in the init phase of runtime.
|
|
func RegisterModule(mod Module) error {
|
|
if mod.Name == "caddy" {
|
|
return fmt.Errorf("modules cannot be named 'caddy'")
|
|
}
|
|
|
|
modulesMu.Lock()
|
|
defer modulesMu.Unlock()
|
|
|
|
if _, ok := modules[mod.Name]; ok {
|
|
return fmt.Errorf("module already registered: %s", mod.Name)
|
|
}
|
|
modules[mod.Name] = mod
|
|
return nil
|
|
}
|
|
|
|
// GetModule returns a module by its full name.
|
|
func GetModule(name string) (Module, error) {
|
|
modulesMu.Lock()
|
|
defer modulesMu.Unlock()
|
|
|
|
m, ok := modules[name]
|
|
if !ok {
|
|
return Module{}, fmt.Errorf("module not registered: %s", name)
|
|
}
|
|
return m, nil
|
|
}
|
|
|
|
// GetModules returns all modules in the given scope/namespace.
|
|
// For example, a scope of "foo" returns modules named "foo.bar",
|
|
// "foo.loo", but not "bar", "foo.bar.loo", etc. An empty scope
|
|
// returns top-level modules, for example "foo" or "bar". Partial
|
|
// scopes are not matched (i.e. scope "foo.ba" does not match
|
|
// name "foo.bar").
|
|
//
|
|
// Because modules are registered to a map, the returned slice
|
|
// will be sorted to keep it deterministic.
|
|
func GetModules(scope string) []Module {
|
|
modulesMu.Lock()
|
|
defer modulesMu.Unlock()
|
|
|
|
scopeParts := strings.Split(scope, ".")
|
|
|
|
// handle the special case of an empty scope, which
|
|
// should match only the top-level modules
|
|
if len(scopeParts) == 1 && scopeParts[0] == "" {
|
|
scopeParts = []string{}
|
|
}
|
|
|
|
var mods []Module
|
|
iterateModules:
|
|
for name, m := range modules {
|
|
modParts := strings.Split(name, ".")
|
|
|
|
// match only the next level of nesting
|
|
if len(modParts) != len(scopeParts)+1 {
|
|
continue
|
|
}
|
|
|
|
// specified parts must be exact matches
|
|
for i := range scopeParts {
|
|
if modParts[i] != scopeParts[i] {
|
|
continue iterateModules
|
|
}
|
|
}
|
|
|
|
mods = append(mods, m)
|
|
}
|
|
|
|
// make return value deterministic
|
|
sort.Slice(mods, func(i, j int) bool {
|
|
return mods[i].Name < mods[j].Name
|
|
})
|
|
|
|
return mods
|
|
}
|
|
|
|
// Modules returns the names of all registered modules
|
|
// in ascending lexicographical order.
|
|
func Modules() []string {
|
|
modulesMu.Lock()
|
|
defer modulesMu.Unlock()
|
|
|
|
var names []string
|
|
for name := range modules {
|
|
names = append(names, name)
|
|
}
|
|
|
|
sort.Strings(names)
|
|
|
|
return names
|
|
}
|
|
|
|
// getModuleNameInline loads the string value from raw of moduleNameKey,
|
|
// where raw must be a JSON encoding of a map. It returns that value,
|
|
// along with the result of removing that key from raw.
|
|
func getModuleNameInline(moduleNameKey string, raw json.RawMessage) (string, json.RawMessage, error) {
|
|
var tmp map[string]interface{}
|
|
err := json.Unmarshal(raw, &tmp)
|
|
if err != nil {
|
|
return "", nil, err
|
|
}
|
|
|
|
moduleName, ok := tmp[moduleNameKey].(string)
|
|
if !ok || moduleName == "" {
|
|
return "", nil, fmt.Errorf("module name not specified with key '%s' in %+v", moduleNameKey, tmp)
|
|
}
|
|
|
|
// remove key from the object, otherwise decoding it later
|
|
// will yield an error because the struct won't recognize it
|
|
// (this is only needed because we strictly enforce that
|
|
// all keys are recognized when loading modules)
|
|
delete(tmp, moduleNameKey)
|
|
result, err := json.Marshal(tmp)
|
|
if err != nil {
|
|
return "", nil, fmt.Errorf("re-encoding module configuration: %v", err)
|
|
}
|
|
|
|
return moduleName, result, nil
|
|
}
|
|
|
|
// Provisioner is implemented by modules which may need to perform
|
|
// some additional "setup" steps immediately after being loaded.
|
|
// Provisioning should be fast (imperceptible running time). If
|
|
// any side-effects result in the execution of this function (e.g.
|
|
// creating global state, any other allocations which require
|
|
// garbage collection, opening files, starting goroutines etc.),
|
|
// be sure to clean up properly by implementing the CleanerUpper
|
|
// interface to avoid leaking resources.
|
|
type Provisioner interface {
|
|
Provision(Context) error
|
|
}
|
|
|
|
// Validator is implemented by modules which can verify that their
|
|
// configurations are valid. This method will be called after
|
|
// Provision() (if implemented). Validation should always be fast
|
|
// (imperceptible running time) and an error should be returned only
|
|
// if the value's configuration is invalid.
|
|
type Validator interface {
|
|
Validate() error
|
|
}
|
|
|
|
// CleanerUpper is implemented by modules which may have side-effects
|
|
// such as opened files, spawned goroutines, or allocated some sort
|
|
// of non-local state when they were provisioned. This method should
|
|
// deallocate/cleanup those resources to prevent memory leaks. Cleanup
|
|
// should be fast and efficient.
|
|
type CleanerUpper interface {
|
|
Cleanup() error
|
|
}
|
|
|
|
// strictUnmarshalJSON is like json.Unmarshal but returns an error
|
|
// if any of the fields are unrecognized. Useful when decoding
|
|
// module configurations, where you want to be more sure they're
|
|
// correct.
|
|
func strictUnmarshalJSON(data []byte, v interface{}) error {
|
|
dec := json.NewDecoder(bytes.NewReader(data))
|
|
dec.DisallowUnknownFields()
|
|
return dec.Decode(v)
|
|
}
|
|
|
|
var (
|
|
modules = make(map[string]Module)
|
|
modulesMu sync.Mutex
|
|
)
|