Protocol and Caddyscript matchers

* Added matcher to determine what protocol the request is being made by
  - grpc, tls, http
* Added ability to run caddyscript in a matcher to evaluate the http request
* Added TLS field to caddyscript request time
* Added a library to manipulate and compare a new caddyscript time type
* Library for regex in starlark
This commit is contained in:
dev
2019-04-03 11:47:27 -04:00
parent 402f423693
commit 27ecc7f384
9 changed files with 341 additions and 12 deletions
+72
View File
@@ -0,0 +1,72 @@
package caddyscript
import (
"fmt"
"net/http"
"go.starlark.net/starlark"
)
// HTTPRequest represents an http request type in caddyscript.
type HTTPRequest struct{ Req *http.Request }
// AttrNames defines what properties and methods are available on the HTTPRequest type.
func (r HTTPRequest) AttrNames() []string {
return []string{"header", "query", "url", "method", "host", "tls"}
}
func (r HTTPRequest) Freeze() {}
func (r HTTPRequest) Hash() (uint32, error) { return 0, fmt.Errorf("unhashable: HTTPRequest") }
func (r HTTPRequest) String() string { return fmt.Sprint(r.Req) }
func (r HTTPRequest) Type() string { return "HTTPRequest" }
func (r HTTPRequest) Truth() starlark.Bool { return true }
// Header handles returning a header key.
func (r HTTPRequest) Header(thread *starlark.Thread, fn *starlark.Builtin, args starlark.Tuple, kwargs []starlark.Tuple) (starlark.Value, error) {
var key string
err := starlark.UnpackPositionalArgs(fn.Name(), args, kwargs, 1, &key)
if err != nil {
return starlark.None, fmt.Errorf("get request header: %v", err.Error())
}
return starlark.String(r.Req.Header.Get(key)), nil
}
// Attr defines what happens when props or methods are called on the HTTPRequest type.
func (r HTTPRequest) Attr(name string) (starlark.Value, error) {
switch name {
case "tls":
tls := new(starlark.Dict)
tls.SetKey(starlark.String("cipher_suite"), starlark.MakeUint(uint(r.Req.TLS.CipherSuite)))
tls.SetKey(starlark.String("did_resume"), starlark.Bool(r.Req.TLS.DidResume))
tls.SetKey(starlark.String("handshake_complete"), starlark.Bool(r.Req.TLS.HandshakeComplete))
tls.SetKey(starlark.String("negotiated_protocol"), starlark.String(r.Req.TLS.NegotiatedProtocol))
tls.SetKey(starlark.String("negotiated_protocol_is_mutual"), starlark.Bool(r.Req.TLS.NegotiatedProtocolIsMutual))
tls.SetKey(starlark.String("server_name"), starlark.String(r.Req.TLS.ServerName))
tls.SetKey(starlark.String("version"), starlark.String(r.Req.TLS.Version))
return tls, nil
case "header":
b := starlark.NewBuiltin("Header", r.Header)
b = b.BindReceiver(r)
return b, nil
case "query":
qVals := r.Req.URL.Query()
query := starlark.NewDict(len(qVals))
for k, v := range qVals {
query.SetKey(starlark.String(k), starlark.String(v[0]))
}
return query, nil
case "url":
return starlark.String(r.Req.URL.Path), nil
case "method":
return starlark.String(r.Req.Method), nil
case "host":
return starlark.String(r.Req.Host), nil
}
return nil, nil
}
+11
View File
@@ -0,0 +1,11 @@
package caddyscript
import (
"fmt"
"go.starlark.net/starlark"
)
func invalidReciever(v starlark.Value, want string) (starlark.Value, error) {
return starlark.None, fmt.Errorf("invalid receiver: receiver set to type %v, want %v", v.Type(), want)
}
+50
View File
@@ -0,0 +1,50 @@
package caddyscript
import (
"fmt"
"regexp"
"go.starlark.net/starlark"
)
// Regexp represents a regexp type for caddyscript.
type Regexp struct{}
// AttrNames defines what properties and methods are available on the Time type.
func (r Regexp) AttrNames() []string {
return []string{"match_string"}
}
// Attr defines what happens when props or methods are called on the Time type.
func (r Regexp) Attr(name string) (starlark.Value, error) {
switch name {
case "match_string":
b := starlark.NewBuiltin("match_string", r.MatchString)
b = b.BindReceiver(r)
return b, nil
}
return nil, nil
}
// MatchString reports whether the string s contains any match of the regular expression pattern. More complicated queries need to use Compile and the full Regexp interface.
func (r Regexp) MatchString(thread *starlark.Thread, fn *starlark.Builtin, args starlark.Tuple, kwargs []starlark.Tuple) (starlark.Value, error) {
var pattern, match string
err := starlark.UnpackPositionalArgs(fn.Name(), args, kwargs, 2, &pattern, &match)
if err != nil {
return starlark.None, fmt.Errorf("could not unpack args: %v", err.Error())
}
matched, err := regexp.MatchString(pattern, match)
if err != nil {
return starlark.False, fmt.Errorf("matchstring: %v", err.Error())
}
return starlark.Bool(matched), nil
}
func (r Regexp) Freeze() {}
func (r Regexp) Hash() (uint32, error) { return 0, fmt.Errorf("unhashable: Regexp") }
func (r Regexp) String() string { return fmt.Sprint(r) }
func (r Regexp) Type() string { return "Regexp" }
func (r Regexp) Truth() starlark.Bool { return true }
+130
View File
@@ -0,0 +1,130 @@
package caddyscript
import (
"fmt"
ti "time"
"go.starlark.net/starlark"
)
// Time represents a time type for caddyscript.
type Time struct {
value int64 // time since epoch in nanoseconds
}
// AttrNames defines what properties and methods are available on the Time type.
func (r Time) AttrNames() []string {
return []string{"now", "parse", "add", "subtract", "minute", "hour", "day", "value"}
}
// Attr defines what happens when props or methods are called on the Time type.
func (r Time) Attr(name string) (starlark.Value, error) {
switch name {
case "now":
b := starlark.NewBuiltin("now", r.Now)
b = b.BindReceiver(r)
return b, nil
case "parse_duration":
b := starlark.NewBuiltin("parse_duration", r.ParseDuration)
b = b.BindReceiver(r)
return b, nil
case "add":
b := starlark.NewBuiltin("add", r.Add)
b = b.BindReceiver(r)
return b, nil
case "subtract":
b := starlark.NewBuiltin("subtract", r.Subtract)
b = b.BindReceiver(r)
return b, nil
case "minute":
b := starlark.NewBuiltin("minute", r.Minute)
b = b.BindReceiver(r)
return b, nil
case "hour":
b := starlark.NewBuiltin("hour", r.Hour)
b = b.BindReceiver(r)
return b, nil
case "day":
b := starlark.NewBuiltin("day", r.Day)
b = b.BindReceiver(r)
return b, nil
case "value":
return starlark.MakeInt64(r.value), nil
}
return nil, nil
}
func (r Time) Freeze() {}
func (r Time) Hash() (uint32, error) { return 0, fmt.Errorf("unhashable: Time") }
func (r Time) String() string { return fmt.Sprint(r.value) }
func (r Time) Type() string { return "Time" }
func (r Time) Truth() starlark.Bool { return true }
// Hour returns the current hour of a unix timestamp in range [0, 23].
func (r Time) Hour(thread *starlark.Thread, fn *starlark.Builtin, args starlark.Tuple, kwargs []starlark.Tuple) (starlark.Value, error) {
t := ti.Unix(0, r.value)
return starlark.MakeInt(t.Hour()), nil
}
// Minute returns the current minute of the hour for a unix timestamp in range [0, 59].
func (r Time) Minute(thread *starlark.Thread, fn *starlark.Builtin, args starlark.Tuple, kwargs []starlark.Tuple) (starlark.Value, error) {
t := ti.Unix(0, r.value)
return starlark.MakeInt(t.Minute()), nil
}
// Day returns the current day in a week of a unix timestamp... [Sunday = 0...]
func (r Time) Day(thread *starlark.Thread, fn *starlark.Builtin, args starlark.Tuple, kwargs []starlark.Tuple) (starlark.Value, error) {
t := ti.Unix(0, r.value)
return starlark.MakeInt(int(t.Weekday())), nil
}
// Now returns the current time as a unix timestamp.
func (r Time) Now(thread *starlark.Thread, fn *starlark.Builtin, args starlark.Tuple, kwargs []starlark.Tuple) (starlark.Value, error) {
val := ti.Now().UnixNano()
r.value = val
return r, nil
}
// ParseDuration parses a go duration string to a time type.
func (r Time) ParseDuration(thread *starlark.Thread, fn *starlark.Builtin, args starlark.Tuple, kwargs []starlark.Tuple) (starlark.Value, error) {
var dur string
err := starlark.UnpackPositionalArgs(fn.Name(), args, kwargs, 1, &dur)
if err != nil {
return starlark.None, fmt.Errorf("could not unpack args: %v", err.Error())
}
if parsed, err := ti.ParseDuration(dur); err == nil {
val := parsed.Nanoseconds()
r.value = val
return r, nil
}
return starlark.None, fmt.Errorf("time.parse_duration: argument cannot be parsed as a valid go time duration")
}
// Add adds time to a time type.
func (r Time) Add(thread *starlark.Thread, fn *starlark.Builtin, args starlark.Tuple, kwargs []starlark.Tuple) (starlark.Value, error) {
var t Time
err := starlark.UnpackPositionalArgs(fn.Name(), args, kwargs, 1, &t)
if err != nil {
return starlark.None, fmt.Errorf("could not unpack args: %v", err.Error())
}
val := r.value + t.value
r.value = val
return r, nil
}
// Subtract adds time to a time type.
func (r Time) Subtract(thread *starlark.Thread, fn *starlark.Builtin, args starlark.Tuple, kwargs []starlark.Tuple) (starlark.Value, error) {
var t Time
err := starlark.UnpackPositionalArgs(fn.Name(), args, kwargs, 1, &t)
if err != nil {
return starlark.None, fmt.Errorf("could not unpack args: %v", err.Error())
}
val := r.value - t.value
r.value = val
return r, nil
}
+18
View File
@@ -0,0 +1,18 @@
package caddyscript
import (
"net/http"
caddyscript "bitbucket.org/lightcodelabs/caddy2/internal/caddyscript/lib"
"go.starlark.net/starlark"
)
// MatcherEnv sets up the global context for the matcher caddyscript environment.
func MatcherEnv(r *http.Request) starlark.StringDict {
env := make(starlark.StringDict)
env["req"] = caddyscript.HTTPRequest{Req: r}
env["time"] = caddyscript.Time{}
env["regexp"] = caddyscript.Regexp{}
return env
}