mirror of
https://github.com/yusing/godoxy.git
synced 2025-05-20 20:52:33 +02:00

* cleanup code for URL type * fix makefile for trace mode * refactor, merge Entry, RawEntry and Route into one. * Implement fileserver. * refactor: rename HTTPRoute to ReverseProxyRoute to avoid confusion * refactor: move metrics logger to middleware package - fix prometheus metrics for load balanced routes - route will now fail when health monitor fail to start * fix extra output of ls-* commands by defer initializaing stuff, speed up start time * add test for path traversal attack, small fix on FileServer.Start method * rename rule.on.bypass to pass * refactor and fixed map-to-map deserialization * updated route loading logic * schemas: add "add_prefix" option to modify_request middleware * updated route JSONMarshalling --------- Co-authored-by: yusing <yusing@6uo.me>
107 lines
2.6 KiB
Go
107 lines
2.6 KiB
Go
package middleware
|
|
|
|
import (
|
|
"fmt"
|
|
"os"
|
|
"path"
|
|
"sort"
|
|
|
|
E "github.com/yusing/go-proxy/internal/error"
|
|
"gopkg.in/yaml.v3"
|
|
)
|
|
|
|
var ErrMissingMiddlewareUse = E.New("missing middleware 'use' field")
|
|
|
|
func BuildMiddlewaresFromComposeFile(filePath string, eb *E.Builder) map[string]*Middleware {
|
|
fileContent, err := os.ReadFile(filePath)
|
|
if err != nil {
|
|
eb.Add(err)
|
|
return nil
|
|
}
|
|
return BuildMiddlewaresFromYAML(path.Base(filePath), fileContent, eb)
|
|
}
|
|
|
|
func BuildMiddlewaresFromYAML(source string, data []byte, eb *E.Builder) map[string]*Middleware {
|
|
var rawMap map[string][]map[string]any
|
|
err := yaml.Unmarshal(data, &rawMap)
|
|
if err != nil {
|
|
eb.Add(err)
|
|
return nil
|
|
}
|
|
middlewares := make(map[string]*Middleware)
|
|
for name, defs := range rawMap {
|
|
chain, err := BuildMiddlewareFromChainRaw(name, defs)
|
|
if err != nil {
|
|
eb.Add(err.Subject(source))
|
|
} else {
|
|
middlewares[name+"@file"] = chain
|
|
}
|
|
}
|
|
return middlewares
|
|
}
|
|
|
|
func compileMiddlewares(middlewaresMap map[string]OptionsRaw) ([]*Middleware, E.Error) {
|
|
middlewares := make([]*Middleware, 0, len(middlewaresMap))
|
|
|
|
errs := E.NewBuilder("middlewares compile error")
|
|
invalidOpts := E.NewBuilder("options compile error")
|
|
|
|
for name, opts := range middlewaresMap {
|
|
m, err := Get(name)
|
|
if err != nil {
|
|
errs.Add(err)
|
|
continue
|
|
}
|
|
|
|
m, err = m.New(opts)
|
|
if err != nil {
|
|
invalidOpts.Add(err.Subject(name))
|
|
continue
|
|
}
|
|
middlewares = append(middlewares, m)
|
|
}
|
|
|
|
if invalidOpts.HasError() {
|
|
errs.Add(invalidOpts.Error())
|
|
}
|
|
sort.Sort(ByPriority(middlewares))
|
|
return middlewares, errs.Error()
|
|
}
|
|
|
|
func BuildMiddlewareFromMap(name string, middlewaresMap map[string]OptionsRaw) (*Middleware, E.Error) {
|
|
compiled, err := compileMiddlewares(middlewaresMap)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return NewMiddlewareChain(name, compiled), nil
|
|
}
|
|
|
|
// TODO: check conflict or duplicates.
|
|
func BuildMiddlewareFromChainRaw(name string, defs []map[string]any) (*Middleware, E.Error) {
|
|
chainErr := E.NewBuilder("")
|
|
chain := make([]*Middleware, 0, len(defs))
|
|
for i, def := range defs {
|
|
if def["use"] == nil || def["use"] == "" {
|
|
chainErr.Add(ErrMissingMiddlewareUse.Subjectf("%s[%d]", name, i))
|
|
continue
|
|
}
|
|
baseName := def["use"].(string)
|
|
base, err := Get(baseName)
|
|
if err != nil {
|
|
chainErr.Add(err.Subjectf("%s[%d]", name, i))
|
|
continue
|
|
}
|
|
delete(def, "use")
|
|
m, err := base.New(def)
|
|
if err != nil {
|
|
chainErr.Add(err.Subjectf("%s[%d]", name, i))
|
|
continue
|
|
}
|
|
m.name = fmt.Sprintf("%s[%d]", name, i)
|
|
chain = append(chain, m)
|
|
}
|
|
if chainErr.HasError() {
|
|
return nil, chainErr.Error()
|
|
}
|
|
return NewMiddlewareChain(name, chain), nil
|
|
}
|