package policy import ( "errors" "fmt" "net/http" "net/netip" "regexp" "strings" "github.com/TecharoHQ/anubis/internal" "github.com/TecharoHQ/anubis/lib/policy/checker" "github.com/gaissmai/bart" ) var ( ErrMisconfiguration = errors.New("[unexpected] policy: administrator misconfiguration") ) type staticHashChecker struct { hash string } func (staticHashChecker) Check(r *http.Request) (bool, error) { return true, nil } func (s staticHashChecker) Hash() string { return s.hash } func NewStaticHashChecker(hashable string) checker.Impl { return staticHashChecker{hash: internal.FastHash(hashable)} } type RemoteAddrChecker struct { prefixTable *bart.Lite hash string } func NewRemoteAddrChecker(cidrs []string) (checker.Impl, error) { table := new(bart.Lite) for _, cidr := range cidrs { prefix, err := netip.ParsePrefix(cidr) if err != nil { return nil, fmt.Errorf("%w: range %s not parsing: %w", ErrMisconfiguration, cidr, err) } table.Insert(prefix) } return &RemoteAddrChecker{ prefixTable: table, hash: internal.FastHash(strings.Join(cidrs, ",")), }, nil } func (rac *RemoteAddrChecker) Check(r *http.Request) (bool, error) { host := r.Header.Get("X-Real-Ip") if host == "" { return false, fmt.Errorf("%w: header X-Real-Ip is not set", ErrMisconfiguration) } addr, err := netip.ParseAddr(host) if err != nil { return false, fmt.Errorf("%w: %s is not an IP address: %w", ErrMisconfiguration, host, err) } return rac.prefixTable.Contains(addr), nil } func (rac *RemoteAddrChecker) Hash() string { return rac.hash } type HeaderMatchesChecker struct { header string regexp *regexp.Regexp hash string } func NewUserAgentChecker(rexStr string) (checker.Impl, error) { return NewHeaderMatchesChecker("User-Agent", rexStr) } func NewHeaderMatchesChecker(header, rexStr string) (checker.Impl, error) { rex, err := regexp.Compile(strings.TrimSpace(rexStr)) if err != nil { return nil, fmt.Errorf("%w: regex %s failed parse: %w", ErrMisconfiguration, rexStr, err) } return &HeaderMatchesChecker{strings.TrimSpace(header), rex, internal.FastHash(header + ": " + rexStr)}, nil } func (hmc *HeaderMatchesChecker) Check(r *http.Request) (bool, error) { if hmc.regexp.MatchString(r.Header.Get(hmc.header)) { return true, nil } return false, nil } func (hmc *HeaderMatchesChecker) Hash() string { return hmc.hash } type PathChecker struct { regexp *regexp.Regexp hash string } func NewPathChecker(rexStr string) (checker.Impl, error) { rex, err := regexp.Compile(strings.TrimSpace(rexStr)) if err != nil { return nil, fmt.Errorf("%w: regex %s failed parse: %w", ErrMisconfiguration, rexStr, err) } return &PathChecker{rex, internal.FastHash(rexStr)}, nil } func (pc *PathChecker) Check(r *http.Request) (bool, error) { if pc.regexp.MatchString(r.URL.Path) { return true, nil } return false, nil } func (pc *PathChecker) Hash() string { return pc.hash } func NewHeaderExistsChecker(key string) checker.Impl { return headerExistsChecker{strings.TrimSpace(key)} } type headerExistsChecker struct { header string } func (hec headerExistsChecker) Check(r *http.Request) (bool, error) { if r.Header.Get(hec.header) != "" { return true, nil } return false, nil } func (hec headerExistsChecker) Hash() string { return internal.FastHash(hec.header) } func NewHeadersChecker(headermap map[string]string) (checker.Impl, error) { var result checker.List var errs []error for key, rexStr := range headermap { if rexStr == ".*" { result = append(result, headerExistsChecker{strings.TrimSpace(key)}) continue } rex, err := regexp.Compile(strings.TrimSpace(rexStr)) if err != nil { errs = append(errs, fmt.Errorf("while compiling header %s regex %s: %w", key, rexStr, err)) continue } result = append(result, &HeaderMatchesChecker{key, rex, internal.FastHash(key + ": " + rexStr)}) } if len(errs) != 0 { return nil, errors.Join(errs...) } return result, nil }