mirror of
https://github.com/yusing/godoxy.git
synced 2026-02-24 19:34:53 +01:00
151 lines
2.7 KiB
Go
151 lines
2.7 KiB
Go
package rules
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"os"
|
|
"unicode"
|
|
|
|
gperr "github.com/yusing/goutils/errs"
|
|
)
|
|
|
|
var escapedChars = map[rune]rune{
|
|
'n': '\n',
|
|
't': '\t',
|
|
'r': '\r',
|
|
'\'': '\'',
|
|
'"': '"',
|
|
'\\': '\\',
|
|
' ': ' ',
|
|
}
|
|
|
|
// 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 gperr.Error) {
|
|
buf := bytes.NewBuffer(make([]byte, 0, len(v)))
|
|
|
|
escaped := false
|
|
quote := rune(0)
|
|
brackets := 0
|
|
|
|
var envVar bytes.Buffer
|
|
var missingEnvVars bytes.Buffer
|
|
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 {
|
|
fmt.Fprintf(buf, `\%c`, r)
|
|
}
|
|
escaped = false
|
|
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 {
|
|
envValue, ok := os.LookupEnv(envVar.String())
|
|
if !ok {
|
|
fmt.Fprintf(&missingEnvVars, "%q, ", envVar.String())
|
|
} else {
|
|
buf.WriteString(envValue)
|
|
}
|
|
inEnvVar = false
|
|
} else {
|
|
buf.WriteRune(r)
|
|
}
|
|
case '"', '\'', '`':
|
|
switch {
|
|
case quote == 0 && brackets == 0:
|
|
quote = r
|
|
flush(false)
|
|
case r == quote:
|
|
quote = 0
|
|
flush(true)
|
|
default:
|
|
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 inEnvVar {
|
|
envVar.WriteRune(r)
|
|
} else {
|
|
buf.WriteRune(r)
|
|
}
|
|
}
|
|
}
|
|
|
|
if quote != 0 {
|
|
err = ErrUnterminatedQuotes
|
|
} else if brackets != 0 {
|
|
err = ErrUnterminatedBrackets
|
|
} else {
|
|
flush(false)
|
|
}
|
|
if missingEnvVars.Len() > 0 {
|
|
err = gperr.Join(err, ErrEnvVarNotFound.Subject(missingEnvVars.String()))
|
|
}
|
|
return subject, args, err
|
|
}
|