mirror of
https://github.com/caddyserver/caddy.git
synced 2026-05-21 06:16:31 -04:00
rewrite: Implement uri query operations (#6120)
* Implemented basic uri query operations * Added support for query operations block * Applied Replacer on all query keys and values * Implemented rename query key opration * Rewrite struct: Changed QueryOperations field to Query and comments cleanup * Cleaned up comments, changed the order of operations and added more tests * Changed order of fields in queryOps struct to match the operations order
This commit is contained in:
@@ -98,7 +98,7 @@ func parseCaddyfileURI(h httpcaddyfile.Helper) (caddyhttp.MiddlewareHandler, err
|
||||
h.Next() // consume directive name
|
||||
|
||||
args := h.RemainingArgs()
|
||||
if len(args) < 2 {
|
||||
if len(args) < 1 {
|
||||
return nil, h.ArgErr()
|
||||
}
|
||||
|
||||
@@ -158,12 +158,70 @@ func parseCaddyfileURI(h httpcaddyfile.Helper) (caddyhttp.MiddlewareHandler, err
|
||||
Replace: replace,
|
||||
})
|
||||
|
||||
case "query":
|
||||
if len(args) > 4 {
|
||||
return nil, h.ArgErr()
|
||||
}
|
||||
rewr.Query = &queryOps{}
|
||||
var hasArgs bool
|
||||
if len(args) > 1 {
|
||||
hasArgs = true
|
||||
err := applyQueryOps(h, rewr.Query, args[1:])
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
}
|
||||
|
||||
for h.NextBlock(0) {
|
||||
if hasArgs {
|
||||
return nil, h.Err("Cannot specify uri query rewrites in both argument and block")
|
||||
}
|
||||
queryArgs := []string{h.Val()}
|
||||
queryArgs = append(queryArgs, h.RemainingArgs()...)
|
||||
err := applyQueryOps(h, rewr.Query, queryArgs)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
}
|
||||
|
||||
default:
|
||||
return nil, h.Errf("unrecognized URI manipulation '%s'", args[0])
|
||||
}
|
||||
return rewr, nil
|
||||
}
|
||||
|
||||
func applyQueryOps(h httpcaddyfile.Helper, qo *queryOps, args []string) error {
|
||||
key := args[0]
|
||||
switch {
|
||||
case strings.HasPrefix(key, "-"):
|
||||
if len(args) != 1 {
|
||||
return h.ArgErr()
|
||||
}
|
||||
qo.Delete = append(qo.Delete, strings.TrimLeft(key, "-"))
|
||||
|
||||
case strings.HasPrefix(key, "+"):
|
||||
if len(args) != 2 {
|
||||
return h.ArgErr()
|
||||
}
|
||||
param := strings.TrimLeft(key, "+")
|
||||
qo.Add = append(qo.Add, queryOpsArguments{Key: param, Val: args[1]})
|
||||
|
||||
case strings.Contains(key, ">"):
|
||||
if len(args) != 1 {
|
||||
return h.ArgErr()
|
||||
}
|
||||
renameValKey := strings.Split(key, ">")
|
||||
qo.Rename = append(qo.Rename, queryOpsArguments{Key: renameValKey[0], Val: renameValKey[1]})
|
||||
|
||||
default:
|
||||
if len(args) != 2 {
|
||||
return h.ArgErr()
|
||||
}
|
||||
qo.Set = append(qo.Set, queryOpsArguments{Key: key, Val: args[1]})
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
// parseCaddyfileHandlePath parses the handle_path directive. Syntax:
|
||||
//
|
||||
// handle_path [<matcher>] {
|
||||
|
||||
@@ -89,6 +89,9 @@ type Rewrite struct {
|
||||
// Performs regular expression replacements on the URI path.
|
||||
PathRegexp []*regexReplacer `json:"path_regexp,omitempty"`
|
||||
|
||||
// Mutates the query string of the URI.
|
||||
Query *queryOps `json:"query,omitempty"`
|
||||
|
||||
logger *zap.Logger
|
||||
}
|
||||
|
||||
@@ -269,6 +272,11 @@ func (rewr Rewrite) Rewrite(r *http.Request, repl *caddy.Replacer) bool {
|
||||
rep.do(r, repl)
|
||||
}
|
||||
|
||||
// apply query operations
|
||||
if rewr.Query != nil {
|
||||
rewr.Query.do(r, repl)
|
||||
}
|
||||
|
||||
// update the encoded copy of the URI
|
||||
r.RequestURI = r.URL.RequestURI()
|
||||
|
||||
@@ -470,5 +478,73 @@ func changePath(req *http.Request, newVal func(pathOrRawPath string) string) {
|
||||
}
|
||||
}
|
||||
|
||||
// queryOps describes the operations to perform on query keys: add, set, rename and delete.
|
||||
type queryOps struct {
|
||||
// Renames a query key from Key to Val, without affecting the value.
|
||||
Rename []queryOpsArguments `json:"rename,omitempty"`
|
||||
|
||||
// Sets query parameters; overwrites a query key with the given value.
|
||||
Set []queryOpsArguments `json:"set,omitempty"`
|
||||
|
||||
// Adds query parameters; does not overwrite an existing query field,
|
||||
// and only appends an additional value for that key if any already exist.
|
||||
Add []queryOpsArguments `json:"add,omitempty"`
|
||||
|
||||
// Deletes a given query key by name.
|
||||
Delete []string `json:"delete,omitempty"`
|
||||
}
|
||||
|
||||
func (q *queryOps) do(r *http.Request, repl *caddy.Replacer) {
|
||||
query := r.URL.Query()
|
||||
|
||||
for _, renameParam := range q.Rename {
|
||||
key := repl.ReplaceAll(renameParam.Key, "")
|
||||
val := repl.ReplaceAll(renameParam.Val, "")
|
||||
if key == "" || val == "" {
|
||||
continue
|
||||
}
|
||||
query[val] = query[key]
|
||||
delete(query, key)
|
||||
}
|
||||
|
||||
for _, setParam := range q.Set {
|
||||
key := repl.ReplaceAll(setParam.Key, "")
|
||||
if key == "" {
|
||||
continue
|
||||
}
|
||||
val := repl.ReplaceAll(setParam.Val, "")
|
||||
query[key] = []string{val}
|
||||
}
|
||||
|
||||
for _, addParam := range q.Add {
|
||||
key := repl.ReplaceAll(addParam.Key, "")
|
||||
if key == "" {
|
||||
continue
|
||||
}
|
||||
val := repl.ReplaceAll(addParam.Val, "")
|
||||
query[key] = append(query[key], val)
|
||||
}
|
||||
|
||||
for _, deleteParam := range q.Delete {
|
||||
param := repl.ReplaceAll(deleteParam, "")
|
||||
if param == "" {
|
||||
continue
|
||||
}
|
||||
delete(query, param)
|
||||
}
|
||||
|
||||
r.URL.RawQuery = query.Encode()
|
||||
}
|
||||
|
||||
type queryOpsArguments struct {
|
||||
// A key in the query string. Note that query string keys may appear multiple times.
|
||||
Key string `json:"key,omitempty"`
|
||||
|
||||
// The value for the given operation; for add and set, this is
|
||||
// simply the value of the query, and for rename this is the
|
||||
// query key to rename to.
|
||||
Val string `json:"val,omitempty"`
|
||||
}
|
||||
|
||||
// Interface guard
|
||||
var _ caddyhttp.MiddlewareHandler = (*Rewrite)(nil)
|
||||
|
||||
Reference in New Issue
Block a user