Some minor updates, and get rid of OnLoad/OnUnload

This commit is contained in:
Matthew Holt 2019-05-16 11:46:17 -06:00
parent f9d93ead4e
commit ff5b4639d5
7 changed files with 20 additions and 107 deletions

View file

@ -100,33 +100,6 @@ func Run(cfg *Config) error {
} }
}() }()
// OnLoad
err = func() error {
for modName, instances := range moduleInstances {
mod, err := GetModule(modName)
if err != nil {
return err
}
if mod.OnLoad != nil {
var priorState interface{}
if oldCfg != nil {
priorState = oldCfg.moduleStates[modName]
}
modState, err := mod.OnLoad(instances, priorState)
if err != nil {
return fmt.Errorf("module OnLoad: %s: %v", modName, err)
}
if modState != nil {
cfg.moduleStates[modName] = modState
}
}
}
return nil
}()
if err != nil {
return err
}
// Start // Start
err = func() error { err = func() error {
h := Handle{cfg} h := Handle{cfg}
@ -158,31 +131,6 @@ func Run(cfg *Config) error {
} }
} }
// OnUnload
err = func() error {
for modName := range oldModuleInstances {
mod, err := GetModule(modName)
if err != nil {
return err
}
if mod.OnUnload != nil {
var unloadingState interface{}
if oldCfg != nil {
unloadingState = oldCfg.moduleStates[modName]
}
err := mod.OnUnload(unloadingState)
if err != nil {
log.Printf("[ERROR] module OnUnload: %s: %v", modName, err)
continue
}
}
}
return nil
}()
if err != nil {
return err
}
// shut down listeners that are no longer being used // shut down listeners that are no longer being used
err = func() error { err = func() error {
listenersMu.Lock() listenersMu.Lock()
@ -240,7 +188,7 @@ type Handle struct {
// App returns the configured app named name. If no app with // App returns the configured app named name. If no app with
// that name is currently configured, a new empty one will be // that name is currently configured, a new empty one will be
// instantiated. (The app module must still be plugged in.) // instantiated. (The app module must still be registered.)
func (h Handle) App(name string) (interface{}, error) { func (h Handle) App(name string) (interface{}, error) {
if app, ok := h.current.apps[name]; ok { if app, ok := h.current.apps[name]; ok {
return app, nil return app, nil

View file

@ -22,27 +22,6 @@ type Module struct {
// It must return a pointer; if not, it // It must return a pointer; if not, it
// is converted into one. // is converted into one.
New func() (interface{}, error) New func() (interface{}, error)
// OnLoad is invoked after all module
// instances ave been loaded. It receives
// pointers to each instance of this
// module, and any state from a previous
// running configuration, which may be
// nil.
//
// If this module is to carry "global"
// state between all instances through
// reloads, you might find it helpful
// to return it.
// TODO: Is this really better/safer than a global variable?
OnLoad func(instances []interface{}, priorState interface{}) (newState interface{}, err error)
// OnUnload is called after all module
// instances have been stopped, possibly
// in favor of a new configuration. It
// receives the state given by OnLoad (if
// any).
OnUnload func(state interface{}) error
} }
func (m Module) String() string { return m.Name } func (m Module) String() string { return m.Name }
@ -53,6 +32,9 @@ func RegisterModule(mod Module) error {
if mod.Name == "caddy" { if mod.Name == "caddy" {
return fmt.Errorf("modules cannot be named 'caddy'") return fmt.Errorf("modules cannot be named 'caddy'")
} }
if strings.HasPrefix(mod.Name, "caddy.") {
return fmt.Errorf("modules cannot be namespaced in 'caddy'")
}
modulesMu.Lock() modulesMu.Lock()
defer modulesMu.Unlock() defer modulesMu.Unlock()

View file

@ -13,24 +13,6 @@ func init() {
caddy2.RegisterModule(caddy2.Module{ caddy2.RegisterModule(caddy2.Module{
Name: "http.middleware.log", Name: "http.middleware.log",
New: func() (interface{}, error) { return new(Log), nil }, New: func() (interface{}, error) { return new(Log), nil },
// TODO: Examples of OnLoad and OnUnload.
OnLoad: func(instances []interface{}, priorState interface{}) (interface{}, error) {
var counter int
if priorState != nil {
counter = priorState.(int)
}
counter++
for _, inst := range instances {
logInst := inst.(*Log)
logInst.counter = counter
}
log.Println("State is now:", counter)
return counter, nil
},
OnUnload: func(state interface{}) error {
log.Println("Closing log files, state:", state)
return nil
},
}) })
} }

View file

@ -233,14 +233,14 @@ func (mre *matchRegexp) match(input string, repl *Replacer, scope string) bool {
// save all capture groups, first by index // save all capture groups, first by index
for i, match := range matches { for i, match := range matches {
key := fmt.Sprintf("http.matchers.%s.%s.%d", scope, mre.Name, i) key := fmt.Sprintf("matchers.%s.%s.%d", scope, mre.Name, i)
repl.Map(key, match) repl.Map(key, match)
} }
// then by name // then by name
for i, name := range mre.compiled.SubexpNames() { for i, name := range mre.compiled.SubexpNames() {
if i != 0 && name != "" { if i != 0 && name != "" {
key := fmt.Sprintf("http.matchers.%s.%s.%s", scope, mre.Name, name) key := fmt.Sprintf("matchers.%s.%s.%s", scope, mre.Name, name)
repl.Map(key, matches[i]) repl.Map(key, matches[i])
} }
} }

View file

@ -217,10 +217,10 @@ func TestPathREMatcher(t *testing.T) {
} }
for key, expectVal := range tc.expectRepl { for key, expectVal := range tc.expectRepl {
placeholder := fmt.Sprintf("{http.matchers.path_regexp.%s}", key) placeholder := fmt.Sprintf("{matchers.path_regexp.%s}", key)
actualVal := repl.Replace(placeholder, "<empty>") actualVal := repl.Replace(placeholder, "<empty>")
if actualVal != expectVal { if actualVal != expectVal {
t.Errorf("Test %d [%v]: Expected placeholder {http.matchers.path_regexp.%s} to be '%s' but got '%s'", t.Errorf("Test %d [%v]: Expected placeholder {matchers.path_regexp.%s} to be '%s' but got '%s'",
i, tc.match.Pattern, key, expectVal, actualVal) i, tc.match.Pattern, key, expectVal, actualVal)
continue continue
} }
@ -334,10 +334,10 @@ func TestHeaderREMatcher(t *testing.T) {
} }
for key, expectVal := range tc.expectRepl { for key, expectVal := range tc.expectRepl {
placeholder := fmt.Sprintf("{http.matchers.header_regexp.%s}", key) placeholder := fmt.Sprintf("{matchers.header_regexp.%s}", key)
actualVal := repl.Replace(placeholder, "<empty>") actualVal := repl.Replace(placeholder, "<empty>")
if actualVal != expectVal { if actualVal != expectVal {
t.Errorf("Test %d [%v]: Expected placeholder {http.matchers.header_regexp.%s} to be '%s' but got '%s'", t.Errorf("Test %d [%v]: Expected placeholder {matchers.header_regexp.%s} to be '%s' but got '%s'",
i, tc.match, key, expectVal, actualVal) i, tc.match, key, expectVal, actualVal)
continue continue
} }

View file

@ -93,19 +93,20 @@ func (r *Replacer) defaults() map[string]string {
m["request.uri"] = r.req.URL.RequestURI() m["request.uri"] = r.req.URL.RequestURI()
m["request.uri.path"] = r.req.URL.Path m["request.uri.path"] = r.req.URL.Path
// TODO: why should header fields, cookies, and query params get special treatment like this?
// maybe they should be scoped by words like "request.header." just like everything else.
for field, vals := range r.req.Header { for field, vals := range r.req.Header {
m[">"+strings.ToLower(field)] = strings.Join(vals, ",") m["request.header."+strings.ToLower(field)] = strings.Join(vals, ",")
}
for field, vals := range r.resp.Header() {
m["<"+strings.ToLower(field)] = strings.Join(vals, ",")
} }
for _, cookie := range r.req.Cookies() { for _, cookie := range r.req.Cookies() {
m["~"+cookie.Name] = cookie.Value m["request.cookie."+cookie.Name] = cookie.Value
} }
for param, vals := range r.req.URL.Query() { for param, vals := range r.req.URL.Query() {
m["?"+param] = strings.Join(vals, ",") m["request.uri.query."+param] = strings.Join(vals, ",")
}
}
if r.resp != nil {
for field, vals := range r.resp.Header() {
m["response.header."+strings.ToLower(field)] = strings.Join(vals, ",")
} }
} }

View file

@ -25,7 +25,7 @@ type Static struct {
func (s Static) ServeHTTP(w http.ResponseWriter, r *http.Request) error { func (s Static) ServeHTTP(w http.ResponseWriter, r *http.Request) error {
repl := r.Context().Value(ReplacerCtxKey).(*Replacer) repl := r.Context().Value(ReplacerCtxKey).(*Replacer)
// close the connection // close the connection after responding
r.Close = s.Close r.Close = s.Close
// set all headers, with replacements // set all headers, with replacements