mirror of
				https://github.com/caddyserver/caddy.git
				synced 2025-10-31 18:47:20 -04:00 
			
		
		
		
	* admin: Refactor /load endpoint out of caddy package This eliminates the caddy package's dependency on the caddyconfig package, which helps prevent import cycles. * v2: adapter: register config adapters as Caddy modules * v2: adapter: simplify adapter registration as adapters and modules * v2: adapter: let RegisterAdapter be in charge of registering adapters as modules * v2: adapter: remove underscrores placeholders * v2: adapter: explicitly ignore the error of writing response of writing warnings back to client * Implicitly wrap config adapters as modules Co-authored-by: Matthew Holt <mholt@users.noreply.github.com>
		
			
				
	
	
		
			137 lines
		
	
	
		
			4.3 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			137 lines
		
	
	
		
			4.3 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // 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 caddyconfig
 | |
| 
 | |
| import (
 | |
| 	"encoding/json"
 | |
| 	"fmt"
 | |
| 
 | |
| 	"github.com/caddyserver/caddy/v2"
 | |
| )
 | |
| 
 | |
| // Adapter is a type which can adapt a configuration to Caddy JSON.
 | |
| // It returns the results and any warnings, or an error.
 | |
| type Adapter interface {
 | |
| 	Adapt(body []byte, options map[string]interface{}) ([]byte, []Warning, error)
 | |
| }
 | |
| 
 | |
| // Warning represents a warning or notice related to conversion.
 | |
| type Warning struct {
 | |
| 	File      string `json:"file,omitempty"`
 | |
| 	Line      int    `json:"line,omitempty"`
 | |
| 	Directive string `json:"directive,omitempty"`
 | |
| 	Message   string `json:"message,omitempty"`
 | |
| }
 | |
| 
 | |
| // JSON encodes val as JSON, returning it as a json.RawMessage. Any
 | |
| // marshaling errors (which are highly unlikely with correct code)
 | |
| // are converted to warnings. This is convenient when filling config
 | |
| // structs that require a json.RawMessage, without having to worry
 | |
| // about errors.
 | |
| func JSON(val interface{}, warnings *[]Warning) json.RawMessage {
 | |
| 	b, err := json.Marshal(val)
 | |
| 	if err != nil {
 | |
| 		if warnings != nil {
 | |
| 			*warnings = append(*warnings, Warning{Message: err.Error()})
 | |
| 		}
 | |
| 		return nil
 | |
| 	}
 | |
| 	return b
 | |
| }
 | |
| 
 | |
| // JSONModuleObject is like JSON, except it marshals val into a JSON object
 | |
| // and then adds a key to that object named fieldName with the value fieldVal.
 | |
| // This is useful for JSON-encoding module values where the module name has to
 | |
| // be described within the object by a certain key; for example,
 | |
| // "responder": "file_server" for a file server HTTP responder. The val must
 | |
| // encode into a map[string]interface{} (i.e. it must be a struct or map),
 | |
| // and any errors are converted into warnings, so this can be conveniently
 | |
| // used when filling a struct. For correct code, there should be no errors.
 | |
| func JSONModuleObject(val interface{}, fieldName, fieldVal string, warnings *[]Warning) json.RawMessage {
 | |
| 	// encode to a JSON object first
 | |
| 	enc, err := json.Marshal(val)
 | |
| 	if err != nil {
 | |
| 		if warnings != nil {
 | |
| 			*warnings = append(*warnings, Warning{Message: err.Error()})
 | |
| 		}
 | |
| 		return nil
 | |
| 	}
 | |
| 
 | |
| 	// then decode the object
 | |
| 	var tmp map[string]interface{}
 | |
| 	err = json.Unmarshal(enc, &tmp)
 | |
| 	if err != nil {
 | |
| 		if warnings != nil {
 | |
| 			*warnings = append(*warnings, Warning{Message: err.Error()})
 | |
| 		}
 | |
| 		return nil
 | |
| 	}
 | |
| 
 | |
| 	// so we can easily add the module's field with its appointed value
 | |
| 	tmp[fieldName] = fieldVal
 | |
| 
 | |
| 	// then re-marshal as JSON
 | |
| 	result, err := json.Marshal(tmp)
 | |
| 	if err != nil {
 | |
| 		if warnings != nil {
 | |
| 			*warnings = append(*warnings, Warning{Message: err.Error()})
 | |
| 		}
 | |
| 		return nil
 | |
| 	}
 | |
| 
 | |
| 	return result
 | |
| }
 | |
| 
 | |
| // JSONIndent is used to JSON-marshal the final resulting Caddy
 | |
| // configuration in a consistent, human-readable way.
 | |
| func JSONIndent(val interface{}) ([]byte, error) {
 | |
| 	return json.MarshalIndent(val, "", "\t")
 | |
| }
 | |
| 
 | |
| // RegisterAdapter registers a config adapter with the given name.
 | |
| // This should usually be done at init-time.
 | |
| func RegisterAdapter(name string, adapter Adapter) error {
 | |
| 	if _, ok := configAdapters[name]; ok {
 | |
| 		return fmt.Errorf("%s: already registered", name)
 | |
| 	}
 | |
| 	configAdapters[name] = adapter
 | |
| 	return caddy.RegisterModule(adapterModule{name, adapter})
 | |
| }
 | |
| 
 | |
| // GetAdapter returns the adapter with the given name,
 | |
| // or nil if one with that name is not registered.
 | |
| func GetAdapter(name string) Adapter {
 | |
| 	return configAdapters[name]
 | |
| }
 | |
| 
 | |
| // adapterModule is a wrapper type that can turn any config
 | |
| // adapter into a Caddy module, which has the benefit of being
 | |
| // counted with other modules, even though they do not
 | |
| // technically extend the Caddy configuration structure.
 | |
| // See caddyserver/caddy#3132.
 | |
| type adapterModule struct {
 | |
| 	name string
 | |
| 	Adapter
 | |
| }
 | |
| 
 | |
| func (am adapterModule) CaddyModule() caddy.ModuleInfo {
 | |
| 	return caddy.ModuleInfo{
 | |
| 		ID:  caddy.ModuleID("caddy.adapters." + am.name),
 | |
| 		New: func() caddy.Module { return am },
 | |
| 	}
 | |
| }
 | |
| 
 | |
| var configAdapters = make(map[string]Adapter)
 |