| 
									
										
										
										
											2019-06-30 16:07:58 -06: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. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-14 11:58:28 -06:00
										 |  |  | package caddy | 
					
						
							| 
									
										
										
										
											2019-03-26 15:45:51 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2019-11-04 12:05:20 -07:00
										 |  |  | 	"bytes" | 
					
						
							| 
									
										
										
										
											2019-05-16 16:05:38 -06:00
										 |  |  | 	"context" | 
					
						
							| 
									
										
										
										
											2019-03-26 15:45:51 -06:00
										 |  |  | 	"encoding/json" | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							| 
									
										
										
										
											2019-11-04 12:05:20 -07:00
										 |  |  | 	"io" | 
					
						
							| 
									
										
										
										
											2019-12-31 16:56:19 -07:00
										 |  |  | 	"io/ioutil" | 
					
						
							| 
									
										
										
										
											2019-03-26 19:42:52 -06:00
										 |  |  | 	"log" | 
					
						
							| 
									
										
										
										
											2019-11-04 12:05:20 -07:00
										 |  |  | 	"net/http" | 
					
						
							| 
									
										
										
										
											2019-12-31 22:51:55 -07:00
										 |  |  | 	"os" | 
					
						
							| 
									
										
										
										
											2019-11-04 12:05:20 -07:00
										 |  |  | 	"path" | 
					
						
							| 
									
										
										
										
											2019-12-31 22:51:55 -07:00
										 |  |  | 	"path/filepath" | 
					
						
							| 
									
										
										
										
											2019-06-28 19:28:28 -06:00
										 |  |  | 	"runtime/debug" | 
					
						
							| 
									
										
										
										
											2019-11-04 12:05:20 -07:00
										 |  |  | 	"strconv" | 
					
						
							| 
									
										
										
										
											2019-03-26 15:45:51 -06:00
										 |  |  | 	"strings" | 
					
						
							| 
									
										
										
										
											2019-03-26 19:42:52 -06:00
										 |  |  | 	"sync" | 
					
						
							| 
									
										
										
										
											2019-03-26 15:45:51 -06:00
										 |  |  | 	"time" | 
					
						
							| 
									
										
										
										
											2019-04-25 13:54:48 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-06 23:15:25 -07:00
										 |  |  | 	"github.com/caddyserver/certmagic" | 
					
						
							| 
									
										
										
										
											2019-12-31 16:56:19 -07:00
										 |  |  | 	"go.uber.org/zap" | 
					
						
							| 
									
										
										
										
											2019-03-26 15:45:51 -06:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-10 13:36:46 -07:00
										 |  |  | // Config is the top (or beginning) of the Caddy configuration structure. | 
					
						
							|  |  |  | // Caddy config is expressed natively as a JSON document. If you prefer | 
					
						
							|  |  |  | // not to work with JSON directly, there are [many config adapters](/docs/config-adapters) | 
					
						
							|  |  |  | // available that can convert various inputs into Caddy JSON. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // Many parts of this config are extensible through the use of Caddy modules. | 
					
						
							|  |  |  | // Fields which have a json.RawMessage type and which appear as dots (•••) in | 
					
						
							|  |  |  | // the online docs can be fulfilled by modules in a certain module | 
					
						
							|  |  |  | // namespace. The docs show which modules can be used in a given place. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // Whenever a module is used, its name must be given either inline as part of | 
					
						
							|  |  |  | // the module, or as the key to the module's value. The docs will make it clear | 
					
						
							|  |  |  | // which to use. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // Generally, all config settings are optional, as it is Caddy convention to | 
					
						
							|  |  |  | // have good, documented default values. If a parameter is required, the docs | 
					
						
							|  |  |  | // should say so. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // Go programs which are directly building a Config struct value should take | 
					
						
							|  |  |  | // care to populate the JSON-encodable fields of the struct (i.e. the fields | 
					
						
							|  |  |  | // with `json` struct tags) if employing the module lifecycle (e.g. Provision | 
					
						
							|  |  |  | // method calls). | 
					
						
							| 
									
										
										
										
											2019-06-28 19:28:28 -06:00
										 |  |  | type Config struct { | 
					
						
							| 
									
										
										
										
											2019-12-10 13:36:46 -07:00
										 |  |  | 	Admin   *AdminConfig `json:"admin,omitempty"` | 
					
						
							|  |  |  | 	Logging *Logging     `json:"logging,omitempty"` | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// StorageRaw is a storage module that defines how/where Caddy | 
					
						
							| 
									
										
										
										
											2019-12-31 18:31:43 -07:00
										 |  |  | 	// stores assets (such as TLS certificates). The default storage | 
					
						
							|  |  |  | 	// module is `caddy.storage.file_system` (the local file system), | 
					
						
							|  |  |  | 	// and the default path | 
					
						
							|  |  |  | 	// [depends on the OS and environment](/docs/conventions#data-directory). | 
					
						
							| 
									
										
										
										
											2019-12-10 13:36:46 -07:00
										 |  |  | 	StorageRaw json.RawMessage `json:"storage,omitempty" caddy:"namespace=caddy.storage inline_key=module"` | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// AppsRaw are the apps that Caddy will load and run. The | 
					
						
							|  |  |  | 	// app module name is the key, and the app's config is the | 
					
						
							|  |  |  | 	// associated value. | 
					
						
							|  |  |  | 	AppsRaw ModuleMap `json:"apps,omitempty" caddy:"namespace="` | 
					
						
							| 
									
										
										
										
											2019-06-28 19:28:28 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-28 14:39:37 -06:00
										 |  |  | 	apps    map[string]App | 
					
						
							|  |  |  | 	storage certmagic.Storage | 
					
						
							| 
									
										
										
										
											2019-06-28 19:28:28 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	cancelFunc context.CancelFunc | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // App is a thing that Caddy runs. | 
					
						
							|  |  |  | type App interface { | 
					
						
							|  |  |  | 	Start() error | 
					
						
							|  |  |  | 	Stop() error | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-04 12:05:20 -07:00
										 |  |  | // Run runs the given config, replacing any existing config. | 
					
						
							|  |  |  | func Run(cfg *Config) error { | 
					
						
							|  |  |  | 	cfgJSON, err := json.Marshal(cfg) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return Load(cfgJSON, true) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Load loads the given config JSON and runs it only | 
					
						
							|  |  |  | // if it is different from the current config or | 
					
						
							|  |  |  | // forceReload is true. | 
					
						
							|  |  |  | func Load(cfgJSON []byte, forceReload bool) error { | 
					
						
							|  |  |  | 	return changeConfig(http.MethodPost, "/"+rawConfigKey, cfgJSON, forceReload) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // changeConfig changes the current config (rawCfg) according to the | 
					
						
							|  |  |  | // method, traversed via the given path, and uses the given input as | 
					
						
							|  |  |  | // the new value (if applicable; i.e. "DELETE" doesn't have an input). | 
					
						
							|  |  |  | // If the resulting config is the same as the previous, no reload will | 
					
						
							|  |  |  | // occur unless forceReload is true. This function is safe for | 
					
						
							|  |  |  | // concurrent use. | 
					
						
							|  |  |  | func changeConfig(method, path string, input []byte, forceReload bool) error { | 
					
						
							|  |  |  | 	switch method { | 
					
						
							|  |  |  | 	case http.MethodGet, | 
					
						
							|  |  |  | 		http.MethodHead, | 
					
						
							|  |  |  | 		http.MethodOptions, | 
					
						
							|  |  |  | 		http.MethodConnect, | 
					
						
							|  |  |  | 		http.MethodTrace: | 
					
						
							|  |  |  | 		return fmt.Errorf("method not allowed") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 16:05:38 -06:00
										 |  |  | 	currentCfgMu.Lock() | 
					
						
							|  |  |  | 	defer currentCfgMu.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-04 12:05:20 -07:00
										 |  |  | 	err := unsyncedConfigAccess(method, path, input, nil) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// the mutation is complete, so encode the entire config as JSON | 
					
						
							|  |  |  | 	newCfg, err := json.Marshal(rawCfg[rawConfigKey]) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return APIError{ | 
					
						
							|  |  |  | 			Code: http.StatusBadRequest, | 
					
						
							|  |  |  | 			Err:  fmt.Errorf("encoding new config: %v", err), | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// if nothing changed, no need to do a whole reload unless the client forces it | 
					
						
							|  |  |  | 	if !forceReload && bytes.Equal(rawCfgJSON, newCfg) { | 
					
						
							| 
									
										
										
										
											2019-12-15 14:58:01 -05:00
										 |  |  | 		Log().Named("admin.api").Info("config is unchanged") | 
					
						
							| 
									
										
										
										
											2019-11-04 12:05:20 -07:00
										 |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-27 11:49:49 -07:00
										 |  |  | 	// find any IDs in this config and index them | 
					
						
							|  |  |  | 	idx := make(map[string]string) | 
					
						
							|  |  |  | 	err = indexConfigObjects(rawCfg[rawConfigKey], "/"+rawConfigKey, idx) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return APIError{ | 
					
						
							|  |  |  | 			Code: http.StatusInternalServerError, | 
					
						
							|  |  |  | 			Err:  fmt.Errorf("indexing config: %v", err), | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-04 12:05:20 -07:00
										 |  |  | 	// load this new config; if it fails, we need to revert to | 
					
						
							|  |  |  | 	// our old representation of caddy's actual config | 
					
						
							|  |  |  | 	err = unsyncedDecodeAndRun(newCfg) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		if len(rawCfgJSON) > 0 { | 
					
						
							|  |  |  | 			// restore old config state to keep it consistent | 
					
						
							|  |  |  | 			// with what caddy is still running; we need to | 
					
						
							|  |  |  | 			// unmarshal it again because it's likely that | 
					
						
							|  |  |  | 			// pointers deep in our rawCfg map were modified | 
					
						
							|  |  |  | 			var oldCfg interface{} | 
					
						
							|  |  |  | 			err2 := json.Unmarshal(rawCfgJSON, &oldCfg) | 
					
						
							|  |  |  | 			if err2 != nil { | 
					
						
							|  |  |  | 				err = fmt.Errorf("%v; additionally, restoring old config: %v", err, err2) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			rawCfg[rawConfigKey] = oldCfg | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return fmt.Errorf("loading new config: %v", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// success, so update our stored copy of the encoded | 
					
						
							|  |  |  | 	// config to keep it consistent with what caddy is now | 
					
						
							|  |  |  | 	// running (storing an encoded copy is not strictly | 
					
						
							|  |  |  | 	// necessary, but avoids an extra json.Marshal for | 
					
						
							|  |  |  | 	// each config change) | 
					
						
							|  |  |  | 	rawCfgJSON = newCfg | 
					
						
							|  |  |  | 	rawCfgIndex = idx | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // readConfig traverses the current config to path | 
					
						
							|  |  |  | // and writes its JSON encoding to out. | 
					
						
							|  |  |  | func readConfig(path string, out io.Writer) error { | 
					
						
							|  |  |  | 	currentCfgMu.RLock() | 
					
						
							|  |  |  | 	defer currentCfgMu.RUnlock() | 
					
						
							|  |  |  | 	return unsyncedConfigAccess(http.MethodGet, path, nil, out) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-27 13:22:40 +08:00
										 |  |  | // indexConfigObjects recursively searches ptr for object fields named | 
					
						
							| 
									
										
										
										
											2019-11-04 12:05:20 -07:00
										 |  |  | // "@id" and maps that ID value to the full configPath in the index. | 
					
						
							|  |  |  | // This function is NOT safe for concurrent access; obtain a write lock | 
					
						
							|  |  |  | // on currentCfgMu. | 
					
						
							|  |  |  | func indexConfigObjects(ptr interface{}, configPath string, index map[string]string) error { | 
					
						
							|  |  |  | 	switch val := ptr.(type) { | 
					
						
							|  |  |  | 	case map[string]interface{}: | 
					
						
							|  |  |  | 		for k, v := range val { | 
					
						
							|  |  |  | 			if k == idKey { | 
					
						
							|  |  |  | 				switch idVal := v.(type) { | 
					
						
							|  |  |  | 				case string: | 
					
						
							|  |  |  | 					index[idVal] = configPath | 
					
						
							|  |  |  | 				case float64: // all JSON numbers decode as float64 | 
					
						
							|  |  |  | 					index[fmt.Sprintf("%v", idVal)] = configPath | 
					
						
							|  |  |  | 				default: | 
					
						
							|  |  |  | 					return fmt.Errorf("%s: %s field must be a string or number", configPath, idKey) | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				continue | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			// traverse this object property recursively | 
					
						
							|  |  |  | 			err := indexConfigObjects(val[k], path.Join(configPath, k), index) | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				return err | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	case []interface{}: | 
					
						
							|  |  |  | 		// traverse each element of the array recursively | 
					
						
							|  |  |  | 		for i := range val { | 
					
						
							|  |  |  | 			err := indexConfigObjects(val[i], path.Join(configPath, strconv.Itoa(i)), index) | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				return err | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-31 16:56:19 -07:00
										 |  |  | // unsyncedDecodeAndRun removes any meta fields (like @id tags) | 
					
						
							|  |  |  | // from cfgJSON, decodes the result into a *Config, and runs | 
					
						
							|  |  |  | // it as the new config, replacing any other current config. | 
					
						
							|  |  |  | // It does NOT update the raw config state, as this is a | 
					
						
							|  |  |  | // lower-level function; most callers will want to use Load | 
					
						
							|  |  |  | // instead. A write lock on currentCfgMu is required! | 
					
						
							| 
									
										
										
										
											2019-11-04 12:05:20 -07:00
										 |  |  | func unsyncedDecodeAndRun(cfgJSON []byte) error { | 
					
						
							| 
									
										
										
										
											2019-12-31 16:56:19 -07:00
										 |  |  | 	// remove any @id fields from the JSON, which would cause | 
					
						
							|  |  |  | 	// loading to break since the field wouldn't be recognized | 
					
						
							|  |  |  | 	strippedCfgJSON := RemoveMetaFields(cfgJSON) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-04 12:05:20 -07:00
										 |  |  | 	var newCfg *Config | 
					
						
							| 
									
										
										
										
											2019-12-31 16:56:19 -07:00
										 |  |  | 	err := strictUnmarshalJSON(strippedCfgJSON, &newCfg) | 
					
						
							| 
									
										
										
										
											2019-11-04 12:05:20 -07:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-30 09:16:01 -06:00
										 |  |  | 	// run the new config and start all its apps | 
					
						
							| 
									
										
										
										
											2019-11-04 12:05:20 -07:00
										 |  |  | 	err = run(newCfg, true) | 
					
						
							| 
									
										
										
										
											2019-09-30 09:16:01 -06:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// swap old config with the new one | 
					
						
							|  |  |  | 	oldCfg := currentCfg | 
					
						
							|  |  |  | 	currentCfg = newCfg | 
					
						
							| 
									
										
										
										
											2019-05-16 16:05:38 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-30 09:16:01 -06:00
										 |  |  | 	// Stop, Cleanup each old app | 
					
						
							|  |  |  | 	unsyncedStop(oldCfg) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-31 16:56:19 -07:00
										 |  |  | 	// autosave a non-nil config, if not disabled | 
					
						
							|  |  |  | 	if newCfg != nil && | 
					
						
							|  |  |  | 		(newCfg.Admin == nil || | 
					
						
							|  |  |  | 			newCfg.Admin.Config == nil || | 
					
						
							|  |  |  | 			newCfg.Admin.Config.Persist == nil || | 
					
						
							|  |  |  | 			*newCfg.Admin.Config.Persist) { | 
					
						
							| 
									
										
										
										
											2019-12-31 22:51:55 -07:00
										 |  |  | 		dir := filepath.Dir(ConfigAutosavePath) | 
					
						
							|  |  |  | 		err := os.MkdirAll(dir, 0700) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			Log().Error("unable to create folder for config autosave", | 
					
						
							|  |  |  | 				zap.String("dir", dir), | 
					
						
							| 
									
										
										
										
											2019-12-31 16:56:19 -07:00
										 |  |  | 				zap.Error(err)) | 
					
						
							| 
									
										
										
										
											2019-12-31 22:51:55 -07:00
										 |  |  | 		} else { | 
					
						
							|  |  |  | 			err := ioutil.WriteFile(ConfigAutosavePath, cfgJSON, 0600) | 
					
						
							|  |  |  | 			if err == nil { | 
					
						
							|  |  |  | 				Log().Info("autosaved config", zap.String("file", ConfigAutosavePath)) | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				Log().Error("unable to autosave config", | 
					
						
							|  |  |  | 					zap.String("file", ConfigAutosavePath), | 
					
						
							|  |  |  | 					zap.Error(err)) | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2019-12-31 16:56:19 -07:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-30 09:16:01 -06:00
										 |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // run runs newCfg and starts all its apps if | 
					
						
							|  |  |  | // start is true. If any errors happen, cleanup | 
					
						
							|  |  |  | // is performed if any modules were provisioned; | 
					
						
							|  |  |  | // apps that were started already will be stopped, | 
					
						
							|  |  |  | // so this function should not leak resources if | 
					
						
							| 
									
										
										
										
											2019-10-28 14:39:37 -06:00
										 |  |  | // an error is returned. However, if no error is | 
					
						
							|  |  |  | // returned and start == false, you should cancel | 
					
						
							|  |  |  | // the config if you are not going to start it, | 
					
						
							|  |  |  | // so that each provisioned module will be | 
					
						
							|  |  |  | // cleaned up. | 
					
						
							| 
									
										
										
										
											2019-11-04 12:05:20 -07:00
										 |  |  | // | 
					
						
							|  |  |  | // This is a low-level function; most callers | 
					
						
							|  |  |  | // will want to use Run instead, which also | 
					
						
							|  |  |  | // updates the config's raw state. | 
					
						
							| 
									
										
										
										
											2019-09-30 09:16:01 -06:00
										 |  |  | func run(newCfg *Config, start bool) error { | 
					
						
							|  |  |  | 	// because we will need to roll back any state | 
					
						
							|  |  |  | 	// modifications if this function errors, we | 
					
						
							|  |  |  | 	// keep a single error value and scope all | 
					
						
							|  |  |  | 	// sub-operations to their own functions to | 
					
						
							|  |  |  | 	// ensure this error value does not get | 
					
						
							|  |  |  | 	// overridden or missed when it should have | 
					
						
							|  |  |  | 	// been set by a short assignment | 
					
						
							|  |  |  | 	var err error | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-04 12:05:20 -07:00
										 |  |  | 	// start the admin endpoint (and stop any prior one) | 
					
						
							| 
									
										
										
										
											2019-12-12 14:30:22 -07:00
										 |  |  | 	if start { | 
					
						
							|  |  |  | 		err = replaceAdmin(newCfg) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return fmt.Errorf("starting caddy administration endpoint: %v", err) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-11-04 12:05:20 -07:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if newCfg == nil { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-30 09:16:01 -06:00
										 |  |  | 	// prepare the new config for use | 
					
						
							|  |  |  | 	newCfg.apps = make(map[string]App) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// create a context within which to load | 
					
						
							|  |  |  | 	// modules - essentially our new config's | 
					
						
							|  |  |  | 	// execution environment; be sure that | 
					
						
							|  |  |  | 	// cleanup occurs when we return if there | 
					
						
							|  |  |  | 	// was an error; if no error, it will get | 
					
						
							|  |  |  | 	// cleaned up on next config cycle | 
					
						
							|  |  |  | 	ctx, cancel := NewContext(Context{Context: context.Background(), cfg: newCfg}) | 
					
						
							|  |  |  | 	defer func() { | 
					
						
							| 
									
										
										
										
											2019-05-16 16:05:38 -06:00
										 |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2019-10-28 14:39:37 -06:00
										 |  |  | 			// if there were any errors during startup, | 
					
						
							|  |  |  | 			// we should cancel the new context we created | 
					
						
							|  |  |  | 			// since the associated config won't be used; | 
					
						
							|  |  |  | 			// this will cause all modules that were newly | 
					
						
							|  |  |  | 			// provisioned to clean themselves up | 
					
						
							|  |  |  | 			cancel() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// also undo any other state changes we made | 
					
						
							|  |  |  | 			if currentCfg != nil { | 
					
						
							|  |  |  | 				certmagic.Default.Storage = currentCfg.storage | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2019-03-26 19:42:52 -06:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-09-30 09:16:01 -06:00
										 |  |  | 	}() | 
					
						
							|  |  |  | 	newCfg.cancelFunc = cancel // clean up later | 
					
						
							| 
									
										
										
										
											2019-04-25 13:54:48 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-28 14:39:37 -06:00
										 |  |  | 	// set up logging before anything bad happens | 
					
						
							| 
									
										
										
										
											2019-10-29 11:58:29 -06:00
										 |  |  | 	if newCfg.Logging == nil { | 
					
						
							|  |  |  | 		newCfg.Logging = new(Logging) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	err = newCfg.Logging.openLogs(ctx) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							| 
									
										
										
										
											2019-10-28 14:39:37 -06:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// set up global storage and make it CertMagic's default storage, too | 
					
						
							| 
									
										
										
										
											2019-09-30 09:16:01 -06:00
										 |  |  | 	err = func() error { | 
					
						
							|  |  |  | 		if newCfg.StorageRaw != nil { | 
					
						
							| 
									
										
										
										
											2019-12-10 13:36:46 -07:00
										 |  |  | 			val, err := ctx.LoadModule(newCfg, "StorageRaw") | 
					
						
							| 
									
										
										
										
											2019-09-30 09:16:01 -06:00
										 |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				return fmt.Errorf("loading storage module: %v", err) | 
					
						
							| 
									
										
										
										
											2019-05-16 16:05:38 -06:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2019-09-30 09:16:01 -06:00
										 |  |  | 			stor, err := val.(StorageConverter).CertMagicStorage() | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				return fmt.Errorf("creating storage value: %v", err) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			newCfg.storage = stor | 
					
						
							| 
									
										
										
										
											2019-04-08 00:00:14 -06:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-12-10 13:36:46 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-30 09:16:01 -06:00
										 |  |  | 		if newCfg.storage == nil { | 
					
						
							| 
									
										
										
										
											2020-03-31 17:56:36 -06:00
										 |  |  | 			newCfg.storage = DefaultStorage | 
					
						
							| 
									
										
										
										
											2019-09-30 09:16:01 -06:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		certmagic.Default.Storage = newCfg.storage | 
					
						
							| 
									
										
										
										
											2019-04-25 13:54:48 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-30 09:16:01 -06:00
										 |  |  | 		return nil | 
					
						
							|  |  |  | 	}() | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-06 23:15:25 -07:00
										 |  |  | 	// Load and Provision each app and their submodules | 
					
						
							| 
									
										
										
										
											2019-09-30 09:16:01 -06:00
										 |  |  | 	err = func() error { | 
					
						
							| 
									
										
										
										
											2020-03-06 23:15:25 -07:00
										 |  |  | 		for appName := range newCfg.AppsRaw { | 
					
						
							|  |  |  | 			if _, err := ctx.App(appName); err != nil { | 
					
						
							|  |  |  | 				return err | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2019-04-08 00:00:14 -06:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-09-30 09:16:01 -06:00
										 |  |  | 		return nil | 
					
						
							|  |  |  | 	}() | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							| 
									
										
										
										
											2019-04-25 13:54:48 -06:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-30 09:16:01 -06:00
										 |  |  | 	if !start { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-04-25 13:54:48 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-30 09:16:01 -06:00
										 |  |  | 	// Start | 
					
						
							|  |  |  | 	return func() error { | 
					
						
							|  |  |  | 		var started []string | 
					
						
							|  |  |  | 		for name, a := range newCfg.apps { | 
					
						
							|  |  |  | 			err := a.Start() | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				// an app failed to start, so we need to stop | 
					
						
							|  |  |  | 				// all other apps that were already started | 
					
						
							|  |  |  | 				for _, otherAppName := range started { | 
					
						
							|  |  |  | 					err2 := newCfg.apps[otherAppName].Stop() | 
					
						
							|  |  |  | 					if err2 != nil { | 
					
						
							|  |  |  | 						err = fmt.Errorf("%v; additionally, aborting app %s: %v", | 
					
						
							|  |  |  | 							err, otherAppName, err2) | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				return fmt.Errorf("%s app module: start: %v", name, err) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			started = append(started, name) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	}() | 
					
						
							| 
									
										
										
										
											2019-07-12 10:07:11 -06:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-04-02 15:31:02 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-12 10:07:11 -06:00
										 |  |  | // Stop stops running the current configuration. | 
					
						
							|  |  |  | // It is the antithesis of Run(). This function | 
					
						
							|  |  |  | // will log any errors that occur during the | 
					
						
							|  |  |  | // stopping of individual apps and continue to | 
					
						
							| 
									
										
										
										
											2019-11-04 12:05:20 -07:00
										 |  |  | // stop the others. Stop should only be called | 
					
						
							|  |  |  | // if not replacing with a new config. | 
					
						
							| 
									
										
										
										
											2019-07-12 10:07:11 -06:00
										 |  |  | func Stop() error { | 
					
						
							|  |  |  | 	currentCfgMu.Lock() | 
					
						
							|  |  |  | 	defer currentCfgMu.Unlock() | 
					
						
							|  |  |  | 	unsyncedStop(currentCfg) | 
					
						
							|  |  |  | 	currentCfg = nil | 
					
						
							| 
									
										
										
										
											2019-11-04 12:05:20 -07:00
										 |  |  | 	rawCfgJSON = nil | 
					
						
							|  |  |  | 	rawCfgIndex = nil | 
					
						
							|  |  |  | 	rawCfg[rawConfigKey] = nil | 
					
						
							| 
									
										
										
										
											2019-03-26 15:45:51 -06:00
										 |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-04 12:05:20 -07:00
										 |  |  | // unsyncedStop stops cfg from running, but has | 
					
						
							|  |  |  | // no locking around cfg. It is a no-op if cfg is | 
					
						
							|  |  |  | // nil. If any app returns an error when stopping, | 
					
						
							|  |  |  | // it is logged and the function continues stopping | 
					
						
							|  |  |  | // the next app. This function assumes all apps in | 
					
						
							|  |  |  | // cfg were successfully started first. | 
					
						
							| 
									
										
										
										
											2019-09-30 09:16:01 -06:00
										 |  |  | func unsyncedStop(cfg *Config) { | 
					
						
							|  |  |  | 	if cfg == nil { | 
					
						
							| 
									
										
										
										
											2019-07-12 10:07:11 -06:00
										 |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// stop each app | 
					
						
							| 
									
										
										
										
											2019-09-30 09:16:01 -06:00
										 |  |  | 	for name, a := range cfg.apps { | 
					
						
							| 
									
										
										
										
											2019-07-12 10:07:11 -06:00
										 |  |  | 		err := a.Stop() | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			log.Printf("[ERROR] stop %s: %v", name, err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-30 09:16:01 -06:00
										 |  |  | 	// clean up all modules | 
					
						
							|  |  |  | 	cfg.cancelFunc() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-04 12:05:20 -07:00
										 |  |  | // stopAndCleanup calls stop and cleans up anything | 
					
						
							|  |  |  | // else that is expedient. This should only be used | 
					
						
							|  |  |  | // when stopping and not replacing with a new config. | 
					
						
							|  |  |  | func stopAndCleanup() error { | 
					
						
							|  |  |  | 	if err := Stop(); err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	certmagic.CleanUpOwnLocks() | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-30 09:16:01 -06:00
										 |  |  | // Validate loads, provisions, and validates | 
					
						
							|  |  |  | // cfg, but does not start running it. | 
					
						
							|  |  |  | func Validate(cfg *Config) error { | 
					
						
							| 
									
										
										
										
											2019-10-28 14:39:37 -06:00
										 |  |  | 	err := run(cfg, false) | 
					
						
							|  |  |  | 	if err == nil { | 
					
						
							|  |  |  | 		cfg.cancelFunc() // call Cleanup on all modules | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return err | 
					
						
							| 
									
										
										
										
											2019-07-12 10:07:11 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-10 13:36:46 -07:00
										 |  |  | // Duration can be an integer or a string. An integer is | 
					
						
							|  |  |  | // interpreted as nanoseconds. If a string, it is a Go | 
					
						
							|  |  |  | // time.Duration value such as `300ms`, `1.5h`, or `2h45m`; | 
					
						
							|  |  |  | // valid units are `ns`, `us`/`µs`, `ms`, `s`, `m`, and `h`. | 
					
						
							| 
									
										
										
										
											2019-03-26 15:45:51 -06:00
										 |  |  | type Duration time.Duration | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // UnmarshalJSON satisfies json.Unmarshaler. | 
					
						
							| 
									
										
										
										
											2019-03-31 20:41:29 -06:00
										 |  |  | func (d *Duration) UnmarshalJSON(b []byte) error { | 
					
						
							| 
									
										
										
										
											2019-11-04 12:54:46 -07:00
										 |  |  | 	if len(b) == 0 { | 
					
						
							|  |  |  | 		return io.EOF | 
					
						
							| 
									
										
										
										
											2019-03-31 20:41:29 -06:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-11-04 12:54:46 -07:00
										 |  |  | 	var dur time.Duration | 
					
						
							|  |  |  | 	var err error | 
					
						
							|  |  |  | 	if b[0] == byte('"') && b[len(b)-1] == byte('"') { | 
					
						
							|  |  |  | 		dur, err = time.ParseDuration(strings.Trim(string(b), `"`)) | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		err = json.Unmarshal(b, &dur) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	*d = Duration(dur) | 
					
						
							|  |  |  | 	return err | 
					
						
							| 
									
										
										
										
											2019-03-26 15:45:51 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-28 19:28:28 -06:00
										 |  |  | // GoModule returns the build info of this Caddy | 
					
						
							|  |  |  | // build from debug.BuildInfo (requires Go modules). | 
					
						
							|  |  |  | // If no version information is available, a non-nil | 
					
						
							|  |  |  | // value will still be returned, but with an | 
					
						
							|  |  |  | // unknown version. | 
					
						
							| 
									
										
										
										
											2019-08-08 07:59:02 +02:00
										 |  |  | func GoModule() *debug.Module { | 
					
						
							|  |  |  | 	var mod debug.Module | 
					
						
							|  |  |  | 	return goModule(&mod) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // goModule holds the actual implementation of GoModule. | 
					
						
							|  |  |  | // Allocating debug.Module in GoModule() and passing a | 
					
						
							|  |  |  | // reference to goModule enables mid-stack inlining. | 
					
						
							|  |  |  | func goModule(mod *debug.Module) *debug.Module { | 
					
						
							|  |  |  | 	mod.Version = "unknown" | 
					
						
							| 
									
										
										
										
											2019-06-28 19:28:28 -06:00
										 |  |  | 	bi, ok := debug.ReadBuildInfo() | 
					
						
							|  |  |  | 	if ok { | 
					
						
							| 
									
										
										
										
											2019-07-12 10:15:27 -06:00
										 |  |  | 		mod.Path = bi.Main.Path | 
					
						
							| 
									
										
										
										
											2019-06-28 19:28:28 -06:00
										 |  |  | 		// The recommended way to build Caddy involves | 
					
						
							|  |  |  | 		// creating a separate main module, which | 
					
						
							|  |  |  | 		// TODO: track related Go issue: https://github.com/golang/go/issues/29228 | 
					
						
							| 
									
										
										
										
											2019-07-12 10:15:27 -06:00
										 |  |  | 		// once that issue is fixed, we should just be able to use bi.Main... hopefully. | 
					
						
							|  |  |  | 		for _, dep := range bi.Deps { | 
					
						
							| 
									
										
										
										
											2019-12-29 13:16:34 -07:00
										 |  |  | 			if dep.Path == ImportPath { | 
					
						
							| 
									
										
										
										
											2019-08-18 04:14:55 +03:00
										 |  |  | 				return dep | 
					
						
							| 
									
										
										
										
											2019-06-28 19:28:28 -06:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-08-18 04:14:55 +03:00
										 |  |  | 		return &bi.Main | 
					
						
							| 
									
										
										
										
											2019-06-28 19:28:28 -06:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-07-12 10:15:27 -06:00
										 |  |  | 	return mod | 
					
						
							| 
									
										
										
										
											2019-06-28 19:28:28 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-11 20:42:55 -06:00
										 |  |  | // CtxKey is a value type for use with context.WithValue. | 
					
						
							|  |  |  | type CtxKey string | 
					
						
							| 
									
										
										
										
											2019-04-08 00:00:14 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-04 12:05:20 -07:00
										 |  |  | // This group of variables pertains to the current configuration. | 
					
						
							| 
									
										
										
										
											2019-04-08 00:00:14 -06:00
										 |  |  | var ( | 
					
						
							| 
									
										
										
										
											2019-11-04 12:05:20 -07:00
										 |  |  | 	// currentCfgMu protects everything in this var block. | 
					
						
							| 
									
										
										
										
											2019-04-25 13:54:48 -06:00
										 |  |  | 	currentCfgMu sync.RWMutex | 
					
						
							| 
									
										
										
										
											2019-11-04 12:05:20 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// currentCfg is the currently-running configuration. | 
					
						
							|  |  |  | 	currentCfg *Config | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// rawCfg is the current, generic-decoded configuration; | 
					
						
							|  |  |  | 	// we initialize it as a map with one field ("config") | 
					
						
							|  |  |  | 	// to maintain parity with the API endpoint and to avoid | 
					
						
							|  |  |  | 	// the special case of having to access/mutate the variable | 
					
						
							|  |  |  | 	// directly without traversing into it. | 
					
						
							|  |  |  | 	rawCfg = map[string]interface{}{ | 
					
						
							|  |  |  | 		rawConfigKey: nil, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// rawCfgJSON is the JSON-encoded form of rawCfg. Keeping | 
					
						
							|  |  |  | 	// this around avoids an extra Marshal call during changes. | 
					
						
							|  |  |  | 	rawCfgJSON []byte | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// rawCfgIndex is the map of user-assigned ID to expanded | 
					
						
							|  |  |  | 	// path, for converting /id/ paths to /config/ paths. | 
					
						
							|  |  |  | 	rawCfgIndex map[string]string | 
					
						
							| 
									
										
										
										
											2019-04-08 00:00:14 -06:00
										 |  |  | ) | 
					
						
							| 
									
										
										
										
											2019-12-29 13:16:34 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | // ImportPath is the package import path for Caddy core. | 
					
						
							|  |  |  | const ImportPath = "github.com/caddyserver/caddy/v2" |