mirror of
https://github.com/yusing/godoxy.git
synced 2026-02-25 20:04:55 +01:00
* chore(deps): update submodule goutils * docs(http): remove default client from README.md * refactor(rules): introduce block DSL, phase-based execution, and flow validation - add block syntax parser/scanner with nested @blocks and elif/else support - restructure rule execution into explicit pre/post phases with phase flags - classify commands by phase and termination behavior - enforce flow semantics (default rule handling, dead-rule detection) - expand HTTP flow coverage with block + YAML parity tests and benches - refresh rules README/spec and update playground/docs integration - Default rules act as fallback handlers that execute only when no matching non-default rule exists in the pre phase - IfElseBlockCommand now returns early when a condition matches with a nil Do block, instead of falling through to else blocks - Add nil check for auth handler to allow requests when no auth is configured * fix(rules): buffer log output before writing to stdout/stderr * refactor(api/rules): remove IsResponseRule field from ParsedRule and related logic * docs(rules): update examples to use block syntax
253 lines
4.6 KiB
Go
253 lines
4.6 KiB
Go
package rules
|
|
|
|
import (
|
|
"strings"
|
|
"unicode"
|
|
|
|
"github.com/yusing/goutils/env"
|
|
gperr "github.com/yusing/goutils/errs"
|
|
)
|
|
|
|
var escapedChars = map[rune]rune{
|
|
'n': '\n',
|
|
't': '\t',
|
|
'r': '\r',
|
|
'\'': '\'',
|
|
'"': '"',
|
|
'\\': '\\',
|
|
' ': ' ',
|
|
}
|
|
|
|
var quoteChars = [256]bool{
|
|
'"': true,
|
|
'\'': true,
|
|
'`': true,
|
|
}
|
|
|
|
func parseSimple(v string) (subject string, args []string, err error, ok bool) {
|
|
brackets := 0
|
|
for i := range len(v) {
|
|
switch v[i] {
|
|
case '\\', '$', '"', '\'', '`', '\t', '\r', '\n':
|
|
return "", nil, nil, false
|
|
case '(':
|
|
brackets++
|
|
case ')':
|
|
if brackets == 0 {
|
|
return "", nil, ErrUnterminatedBrackets, true
|
|
}
|
|
brackets--
|
|
}
|
|
}
|
|
if brackets != 0 {
|
|
return "", nil, ErrUnterminatedBrackets, true
|
|
}
|
|
|
|
i := 0
|
|
for i < len(v) && v[i] == ' ' {
|
|
i++
|
|
}
|
|
if i >= len(v) {
|
|
return "", nil, nil, true
|
|
}
|
|
|
|
start := i
|
|
for i < len(v) && v[i] != ' ' {
|
|
i++
|
|
}
|
|
subject = v[start:i]
|
|
|
|
if i >= len(v) {
|
|
return subject, nil, nil, true
|
|
}
|
|
|
|
argCount := 0
|
|
for j := i; j < len(v); {
|
|
for j < len(v) && v[j] == ' ' {
|
|
j++
|
|
}
|
|
if j >= len(v) {
|
|
break
|
|
}
|
|
argCount++
|
|
for j < len(v) && v[j] != ' ' {
|
|
j++
|
|
}
|
|
}
|
|
if argCount == 0 {
|
|
return subject, nil, nil, true
|
|
}
|
|
args = make([]string, 0, argCount)
|
|
for i < len(v) {
|
|
for i < len(v) && v[i] == ' ' {
|
|
i++
|
|
}
|
|
if i >= len(v) {
|
|
break
|
|
}
|
|
start = i
|
|
for i < len(v) && v[i] != ' ' {
|
|
i++
|
|
}
|
|
args = append(args, v[start:i])
|
|
}
|
|
return subject, args, nil, true
|
|
}
|
|
|
|
// parse expression to subject and args
|
|
// with support for quotes, escaped chars, and env substitution, e.g.
|
|
//
|
|
// error 403 "Forbidden 'foo' 'bar'"
|
|
// error 403 Forbidden\ \"foo\"\ \"bar\".
|
|
// error 403 "Message: ${CLOUDFLARE_API_KEY}"
|
|
func parse(v string) (subject string, args []string, err error) {
|
|
if subject, args, err, ok := parseSimple(v); ok {
|
|
return subject, args, err
|
|
}
|
|
|
|
buf := getStringBuffer(len(v))
|
|
args = make([]string, 0, 4)
|
|
|
|
escaped := false
|
|
quote := rune(0)
|
|
brackets := 0
|
|
|
|
var (
|
|
envVar strings.Builder
|
|
missingEnvVars []string
|
|
)
|
|
inEnvVar := false
|
|
expectingBrace := false
|
|
|
|
flush := func(quoted bool) {
|
|
part := buf.String()
|
|
if !quoted {
|
|
beg := 0
|
|
for i, r := range part {
|
|
if unicode.IsSpace(r) {
|
|
beg = i + 1
|
|
} else {
|
|
break
|
|
}
|
|
}
|
|
if beg == len(part) { // all spaces
|
|
return
|
|
}
|
|
part = part[beg:] // trim leading spaces
|
|
}
|
|
if subject == "" {
|
|
subject = part
|
|
} else {
|
|
args = append(args, part)
|
|
}
|
|
buf.Reset()
|
|
}
|
|
for _, r := range v {
|
|
if escaped {
|
|
if ch, ok := escapedChars[r]; ok {
|
|
buf.WriteRune(ch)
|
|
} else {
|
|
buf.WriteRune('\\')
|
|
buf.WriteRune(r)
|
|
}
|
|
escaped = false
|
|
continue
|
|
}
|
|
if expectingBrace && r != '{' && r != '$' { // not escaped and not env var
|
|
buf.WriteRune('$')
|
|
expectingBrace = false
|
|
}
|
|
if quoteChars[r] {
|
|
switch {
|
|
case quote == 0 && brackets == 0:
|
|
quote = r
|
|
flush(false)
|
|
case r == quote:
|
|
quote = 0
|
|
flush(true)
|
|
default:
|
|
buf.WriteRune(r)
|
|
}
|
|
continue
|
|
}
|
|
switch r {
|
|
case '\\':
|
|
escaped = true
|
|
case '$':
|
|
if expectingBrace { // $$ => $ and continue
|
|
buf.WriteRune('$')
|
|
expectingBrace = false
|
|
} else {
|
|
expectingBrace = true
|
|
}
|
|
case '{':
|
|
if expectingBrace {
|
|
inEnvVar = true
|
|
expectingBrace = false
|
|
envVar.Reset()
|
|
} else {
|
|
buf.WriteRune(r)
|
|
}
|
|
case '}':
|
|
if inEnvVar {
|
|
// NOTE: use env.LookupEnv instead of os.LookupEnv to support environment variable prefixes
|
|
// like ${API_ADDR} will lookup for GODOXY_API_ADDR, GOPROXY_API_ADDR and API_ADDR.
|
|
envValue, ok := env.LookupEnv(envVar.String())
|
|
if !ok {
|
|
missingEnvVars = append(missingEnvVars, envVar.String())
|
|
} else {
|
|
buf.WriteString(envValue)
|
|
}
|
|
inEnvVar = false
|
|
} else {
|
|
buf.WriteRune(r)
|
|
}
|
|
case '(':
|
|
brackets++
|
|
buf.WriteRune(r)
|
|
case ')':
|
|
if brackets == 0 {
|
|
err = ErrUnterminatedBrackets
|
|
return subject, args, err
|
|
}
|
|
brackets--
|
|
buf.WriteRune(r)
|
|
case ' ':
|
|
if quote == 0 {
|
|
flush(false)
|
|
} else {
|
|
buf.WriteRune(r)
|
|
}
|
|
default:
|
|
if expectingBrace { // last was $ but { not matched
|
|
buf.WriteRune('$')
|
|
expectingBrace = false
|
|
}
|
|
if inEnvVar {
|
|
envVar.WriteRune(r)
|
|
} else {
|
|
buf.WriteRune(r)
|
|
}
|
|
}
|
|
}
|
|
|
|
if expectingBrace {
|
|
buf.WriteRune('$')
|
|
}
|
|
|
|
switch {
|
|
case quote != 0:
|
|
err = ErrUnterminatedQuotes
|
|
case brackets != 0:
|
|
err = ErrUnterminatedBrackets
|
|
case inEnvVar:
|
|
err = ErrUnterminatedEnvVar
|
|
default:
|
|
flush(false)
|
|
}
|
|
if len(missingEnvVars) > 0 {
|
|
err = gperr.Join(err, ErrEnvVarNotFound.With(gperr.Multiline().AddStrings(missingEnvVars...)))
|
|
}
|
|
return subject, args, err
|
|
}
|