| 
									
										
										
										
											2019-03-26 12:00:54 -06:00
										 |  |  | package caddy2 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2019-03-26 15:45:51 -06:00
										 |  |  | 	"encoding/json" | 
					
						
							| 
									
										
										
										
											2019-03-26 12:00:54 -06:00
										 |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"sort" | 
					
						
							|  |  |  | 	"strings" | 
					
						
							|  |  |  | 	"sync" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-25 13:54:48 -06:00
										 |  |  | // Module represents a Caddy module. | 
					
						
							| 
									
										
										
										
											2019-03-26 12:00:54 -06:00
										 |  |  | type Module struct { | 
					
						
							| 
									
										
										
										
											2019-04-26 12:35:39 -06:00
										 |  |  | 	// 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{}, error) | 
					
						
							| 
									
										
										
										
											2019-03-26 12:00:54 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (m Module) String() string { return m.Name } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-26 12:35:39 -06:00
										 |  |  | // RegisterModule registers a module. Modules must call | 
					
						
							|  |  |  | // this function in the init phase of runtime. | 
					
						
							| 
									
										
										
										
											2019-03-26 12:00:54 -06:00
										 |  |  | func RegisterModule(mod Module) error { | 
					
						
							| 
									
										
										
										
											2019-04-25 13:54:48 -06:00
										 |  |  | 	if mod.Name == "caddy" { | 
					
						
							|  |  |  | 		return fmt.Errorf("modules cannot be named 'caddy'") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-26 12:00:54 -06:00
										 |  |  | 	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 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-26 12:35:39 -06:00
										 |  |  | // GetModule returns a module by its full name. | 
					
						
							| 
									
										
										
										
											2019-03-26 12:00:54 -06:00
										 |  |  | 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", | 
					
						
							| 
									
										
										
										
											2019-04-25 13:54:48 -06:00
										 |  |  | // "foo.loo", but not "bar", "foo.bar.loo", etc. An empty scope | 
					
						
							| 
									
										
										
										
											2019-03-26 12:00:54 -06:00
										 |  |  | // 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 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-25 13:54:48 -06:00
										 |  |  | // getModuleNameInline loads the string value from raw of moduleNameKey, | 
					
						
							|  |  |  | // where raw must be a JSON encoding of a map. | 
					
						
							|  |  |  | func getModuleNameInline(moduleNameKey string, raw json.RawMessage) (string, error) { | 
					
						
							|  |  |  | 	var tmp map[string]interface{} | 
					
						
							|  |  |  | 	err := json.Unmarshal(raw, &tmp) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return "", err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	moduleName, ok := tmp[moduleNameKey].(string) | 
					
						
							|  |  |  | 	if !ok || moduleName == "" { | 
					
						
							|  |  |  | 		return "", fmt.Errorf("module name not specified with key '%s' in %+v", moduleNameKey, tmp) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return moduleName, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Provisioner is implemented by modules which may need to perform | 
					
						
							|  |  |  | // some additional "setup" steps immediately after being loaded. | 
					
						
							| 
									
										
										
										
											2019-05-17 08:48:12 -06:00
										 |  |  | // 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. | 
					
						
							| 
									
										
										
										
											2019-04-25 13:54:48 -06:00
										 |  |  | type Provisioner interface { | 
					
						
							| 
									
										
										
										
											2019-05-16 16:05:38 -06:00
										 |  |  | 	Provision(Context) error | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-17 08:48:12 -06:00
										 |  |  | // 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. | 
					
						
							| 
									
										
										
										
											2019-05-16 16:05:38 -06:00
										 |  |  | type CleanerUpper interface { | 
					
						
							|  |  |  | 	Cleanup() error | 
					
						
							| 
									
										
										
										
											2019-04-25 13:54:48 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-26 12:00:54 -06:00
										 |  |  | var ( | 
					
						
							|  |  |  | 	modules   = make(map[string]Module) | 
					
						
							|  |  |  | 	modulesMu sync.Mutex | 
					
						
							|  |  |  | ) |