mirror of
https://github.com/caddyserver/caddy.git
synced 2024-11-25 17:56:34 +08:00
f9cba03d25
This allows you to have multiple redir directives conditioned solely upon if statements, without regard to path.
203 lines
4.8 KiB
Go
203 lines
4.8 KiB
Go
// Copyright 2015 Light Code Labs, LLC
|
|
//
|
|
// 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 httpserver
|
|
|
|
import (
|
|
"fmt"
|
|
"net/http"
|
|
"regexp"
|
|
"strings"
|
|
|
|
"github.com/mholt/caddy"
|
|
)
|
|
|
|
// SetupIfMatcher parses `if` or `if_op` in the current dispenser block.
|
|
// It returns a RequestMatcher and an error if any.
|
|
func SetupIfMatcher(controller *caddy.Controller) (RequestMatcher, error) {
|
|
var c = controller.Dispenser // copy the dispenser
|
|
var matcher IfMatcher
|
|
for c.NextBlock() {
|
|
switch c.Val() {
|
|
case "if":
|
|
args1 := c.RemainingArgs()
|
|
if len(args1) != 3 {
|
|
return matcher, c.ArgErr()
|
|
}
|
|
ifc, err := newIfCond(args1[0], args1[1], args1[2])
|
|
if err != nil {
|
|
return matcher, err
|
|
}
|
|
matcher.ifs = append(matcher.ifs, ifc)
|
|
matcher.Enabled = true
|
|
case "if_op":
|
|
if !c.NextArg() {
|
|
return matcher, c.ArgErr()
|
|
}
|
|
switch c.Val() {
|
|
case "and":
|
|
matcher.isOr = false
|
|
case "or":
|
|
matcher.isOr = true
|
|
default:
|
|
return matcher, c.ArgErr()
|
|
}
|
|
}
|
|
}
|
|
return matcher, nil
|
|
}
|
|
|
|
// operators
|
|
const (
|
|
isOp = "is"
|
|
notOp = "not"
|
|
hasOp = "has"
|
|
startsWithOp = "starts_with"
|
|
endsWithOp = "ends_with"
|
|
matchOp = "match"
|
|
)
|
|
|
|
// ifCondition is a 'if' condition.
|
|
type ifFunc func(a, b string) bool
|
|
|
|
// ifCond is statement for a IfMatcher condition.
|
|
type ifCond struct {
|
|
a string
|
|
op string
|
|
b string
|
|
neg bool
|
|
rex *regexp.Regexp
|
|
f ifFunc
|
|
}
|
|
|
|
// newIfCond creates a new If condition.
|
|
func newIfCond(a, op, b string) (ifCond, error) {
|
|
i := ifCond{a: a, op: op, b: b}
|
|
if strings.HasPrefix(op, "not_") {
|
|
i.neg = true
|
|
i.op = op[4:]
|
|
}
|
|
|
|
switch i.op {
|
|
case isOp:
|
|
// It checks for equality.
|
|
i.f = i.isFunc
|
|
case notOp:
|
|
// It checks for inequality.
|
|
i.f = i.notFunc
|
|
case hasOp:
|
|
// It checks if b is a substring of a.
|
|
i.f = strings.Contains
|
|
case startsWithOp:
|
|
// It checks if b is a prefix of a.
|
|
i.f = strings.HasPrefix
|
|
case endsWithOp:
|
|
// It checks if b is a suffix of a.
|
|
i.f = strings.HasSuffix
|
|
case matchOp:
|
|
// It does regexp matching of a against pattern in b and returns if they match.
|
|
var err error
|
|
if i.rex, err = regexp.Compile(i.b); err != nil {
|
|
return ifCond{}, fmt.Errorf("Invalid regular expression: '%s', %v", i.b, err)
|
|
}
|
|
i.f = i.matchFunc
|
|
default:
|
|
return ifCond{}, fmt.Errorf("Invalid operator %v", i.op)
|
|
}
|
|
|
|
return i, nil
|
|
}
|
|
|
|
// isFunc is condition for Is operator.
|
|
func (i ifCond) isFunc(a, b string) bool {
|
|
return a == b
|
|
}
|
|
|
|
// notFunc is condition for Not operator.
|
|
func (i ifCond) notFunc(a, b string) bool {
|
|
return a != b
|
|
}
|
|
|
|
// matchFunc is condition for Match operator.
|
|
func (i ifCond) matchFunc(a, b string) bool {
|
|
return i.rex.MatchString(a)
|
|
}
|
|
|
|
// True returns true if the condition is true and false otherwise.
|
|
// If r is not nil, it replaces placeholders before comparison.
|
|
func (i ifCond) True(r *http.Request) bool {
|
|
if i.f != nil {
|
|
a, b := i.a, i.b
|
|
if r != nil {
|
|
replacer := NewReplacer(r, nil, "")
|
|
a = replacer.Replace(i.a)
|
|
if i.op != matchOp {
|
|
b = replacer.Replace(i.b)
|
|
}
|
|
}
|
|
if i.neg {
|
|
return !i.f(a, b)
|
|
}
|
|
return i.f(a, b)
|
|
}
|
|
return i.neg // false if not negated, true otherwise
|
|
}
|
|
|
|
// IfMatcher is a RequestMatcher for 'if' conditions.
|
|
type IfMatcher struct {
|
|
Enabled bool // if true, matcher has been configured; otherwise it's no-op
|
|
ifs []ifCond // list of If
|
|
isOr bool // if true, conditions are 'or' instead of 'and'
|
|
}
|
|
|
|
// Match satisfies RequestMatcher interface.
|
|
// It returns true if the conditions in m are true.
|
|
func (m IfMatcher) Match(r *http.Request) bool {
|
|
if m.isOr {
|
|
return m.Or(r)
|
|
}
|
|
return m.And(r)
|
|
}
|
|
|
|
// And returns true if all conditions in m are true.
|
|
func (m IfMatcher) And(r *http.Request) bool {
|
|
for _, i := range m.ifs {
|
|
if !i.True(r) {
|
|
return false
|
|
}
|
|
}
|
|
return true
|
|
}
|
|
|
|
// Or returns true if any of the conditions in m is true.
|
|
func (m IfMatcher) Or(r *http.Request) bool {
|
|
for _, i := range m.ifs {
|
|
if i.True(r) {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
// IfMatcherKeyword checks if the next value in the dispenser is a keyword for 'if' config block.
|
|
// If true, remaining arguments in the dispinser are cleard to keep the dispenser valid for use.
|
|
func IfMatcherKeyword(c *caddy.Controller) bool {
|
|
if c.Val() == "if" || c.Val() == "if_op" {
|
|
// clear remaining args
|
|
c.RemainingArgs()
|
|
return true
|
|
}
|
|
return false
|
|
}
|