2021-05-18 07:36:33 +00:00
|
|
|
package aicra
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
"net/http/httptest"
|
|
|
|
"strings"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"git.xdrm.io/go/aicra/api"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestWith(t *testing.T) {
|
|
|
|
builder := &Builder{}
|
|
|
|
if err := addBuiltinTypes(builder); err != nil {
|
|
|
|
t.Fatalf("unexpected error <%v>", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// build @n middlewares that take data from context and increment it
|
|
|
|
n := 1024
|
|
|
|
|
|
|
|
type ckey int
|
|
|
|
const key ckey = 0
|
|
|
|
|
|
|
|
middleware := func(next http.HandlerFunc) http.HandlerFunc {
|
|
|
|
return func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
newr := r
|
|
|
|
|
|
|
|
// first time -> store 1
|
|
|
|
value := r.Context().Value(key)
|
|
|
|
if value == nil {
|
|
|
|
newr = r.WithContext(context.WithValue(r.Context(), key, int(1)))
|
|
|
|
next(w, newr)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// get value and increment
|
|
|
|
cast, ok := value.(int)
|
|
|
|
if !ok {
|
|
|
|
t.Fatalf("value is not an int")
|
|
|
|
}
|
|
|
|
cast++
|
|
|
|
newr = r.WithContext(context.WithValue(r.Context(), key, cast))
|
|
|
|
next(w, newr)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// add middleware @n times
|
|
|
|
for i := 0; i < n; i++ {
|
|
|
|
builder.With(middleware)
|
|
|
|
}
|
|
|
|
|
|
|
|
config := strings.NewReader(`[ { "method": "GET", "path": "/path", "scope": [[]], "info": "info", "in": {}, "out": {} } ]`)
|
|
|
|
err := builder.Setup(config)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("setup: unexpected error <%v>", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
pathHandler := func(ctx api.Ctx) (*struct{}, api.Err) {
|
|
|
|
// write value from middlewares into response
|
|
|
|
value := ctx.Req.Context().Value(key)
|
|
|
|
if value == nil {
|
|
|
|
t.Fatalf("nothing found in context")
|
|
|
|
}
|
|
|
|
cast, ok := value.(int)
|
|
|
|
if !ok {
|
|
|
|
t.Fatalf("cannot cast context data to int")
|
|
|
|
}
|
|
|
|
// write to response
|
|
|
|
ctx.Res.Write([]byte(fmt.Sprintf("#%d#", cast)))
|
|
|
|
|
|
|
|
return nil, api.ErrSuccess
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := builder.Bind(http.MethodGet, "/path", pathHandler); err != nil {
|
|
|
|
t.Fatalf("bind: unexpected error <%v>", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
handler, err := builder.Build()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("build: unexpected error <%v>", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
response := httptest.NewRecorder()
|
|
|
|
request := httptest.NewRequest(http.MethodGet, "/path", &bytes.Buffer{})
|
|
|
|
|
|
|
|
// test request
|
|
|
|
handler.ServeHTTP(response, request)
|
|
|
|
if response.Body == nil {
|
|
|
|
t.Fatalf("response has no body")
|
|
|
|
}
|
|
|
|
token := fmt.Sprintf("#%d#", n)
|
|
|
|
if !strings.Contains(response.Body.String(), token) {
|
|
|
|
t.Fatalf("expected '%s' to be in response <%s>", token, response.Body.String())
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2021-05-18 07:59:49 +00:00
|
|
|
|
|
|
|
func TestWithAuth(t *testing.T) {
|
|
|
|
|
|
|
|
tt := []struct {
|
|
|
|
name string
|
|
|
|
manifest string
|
|
|
|
permissions []string
|
|
|
|
granted bool
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "provide only requirement A",
|
|
|
|
manifest: `[ { "method": "GET", "path": "/path", "scope": [["A"]], "info": "info", "in": {}, "out": {} } ]`,
|
|
|
|
permissions: []string{"A"},
|
|
|
|
granted: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "missing requirement",
|
|
|
|
manifest: `[ { "method": "GET", "path": "/path", "scope": [["A"]], "info": "info", "in": {}, "out": {} } ]`,
|
|
|
|
permissions: []string{},
|
|
|
|
granted: false,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "missing requirements",
|
|
|
|
manifest: `[ { "method": "GET", "path": "/path", "scope": [["A", "B"]], "info": "info", "in": {}, "out": {} } ]`,
|
|
|
|
permissions: []string{},
|
|
|
|
granted: false,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "missing some requirements",
|
|
|
|
manifest: `[ { "method": "GET", "path": "/path", "scope": [["A", "B"]], "info": "info", "in": {}, "out": {} } ]`,
|
|
|
|
permissions: []string{"A"},
|
|
|
|
granted: false,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "provide requirements",
|
|
|
|
manifest: `[ { "method": "GET", "path": "/path", "scope": [["A", "B"]], "info": "info", "in": {}, "out": {} } ]`,
|
|
|
|
permissions: []string{"A", "B"},
|
|
|
|
granted: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "missing OR requirements",
|
|
|
|
manifest: `[ { "method": "GET", "path": "/path", "scope": [["A"], ["B"]], "info": "info", "in": {}, "out": {} } ]`,
|
|
|
|
permissions: []string{"C"},
|
|
|
|
granted: false,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "provide 1 OR requirement",
|
|
|
|
manifest: `[ { "method": "GET", "path": "/path", "scope": [["A"], ["B"]], "info": "info", "in": {}, "out": {} } ]`,
|
|
|
|
permissions: []string{"A"},
|
|
|
|
granted: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "provide both OR requirements",
|
|
|
|
manifest: `[ { "method": "GET", "path": "/path", "scope": [["A"], ["B"]], "info": "info", "in": {}, "out": {} } ]`,
|
|
|
|
permissions: []string{"A", "B"},
|
|
|
|
granted: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "missing composite OR requirements",
|
|
|
|
manifest: `[ { "method": "GET", "path": "/path", "scope": [["A", "B"], ["C", "D"]], "info": "info", "in": {}, "out": {} } ]`,
|
|
|
|
permissions: []string{},
|
|
|
|
granted: false,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "missing partial composite OR requirements",
|
|
|
|
manifest: `[ { "method": "GET", "path": "/path", "scope": [["A", "B"], ["C", "D"]], "info": "info", "in": {}, "out": {} } ]`,
|
|
|
|
permissions: []string{"A", "C"},
|
|
|
|
granted: false,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "provide 1 composite OR requirement",
|
|
|
|
manifest: `[ { "method": "GET", "path": "/path", "scope": [["A", "B"], ["C", "D"]], "info": "info", "in": {}, "out": {} } ]`,
|
|
|
|
permissions: []string{"A", "B", "C"},
|
|
|
|
granted: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "provide both composite OR requirements",
|
|
|
|
manifest: `[ { "method": "GET", "path": "/path", "scope": [["A", "B"], ["C", "D"]], "info": "info", "in": {}, "out": {} } ]`,
|
|
|
|
permissions: []string{"A", "B", "C", "D"},
|
|
|
|
granted: true,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tc := range tt {
|
|
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
|
|
builder := &Builder{}
|
|
|
|
if err := addBuiltinTypes(builder); err != nil {
|
|
|
|
t.Fatalf("unexpected error <%v>", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// tester middleware (last executed)
|
|
|
|
builder.WithAuth(func(next api.AuthHandlerFunc) api.AuthHandlerFunc {
|
|
|
|
return func(a api.Auth, w http.ResponseWriter, r *http.Request) {
|
|
|
|
if a.Granted() == tc.granted {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if a.Granted() {
|
|
|
|
t.Fatalf("unexpected granted auth")
|
|
|
|
} else {
|
|
|
|
t.Fatalf("expected granted auth")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
builder.WithAuth(func(next api.AuthHandlerFunc) api.AuthHandlerFunc {
|
|
|
|
return func(a api.Auth, w http.ResponseWriter, r *http.Request) {
|
|
|
|
a.Active = tc.permissions
|
|
|
|
next(a, w, r)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
err := builder.Setup(strings.NewReader(tc.manifest))
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("setup: unexpected error <%v>", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
pathHandler := func(ctx api.Ctx) (*struct{}, api.Err) {
|
|
|
|
return nil, api.ErrNotImplemented
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := builder.Bind(http.MethodGet, "/path", pathHandler); err != nil {
|
|
|
|
t.Fatalf("bind: unexpected error <%v>", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
handler, err := builder.Build()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("build: unexpected error <%v>", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
response := httptest.NewRecorder()
|
|
|
|
request := httptest.NewRequest(http.MethodGet, "/path", &bytes.Buffer{})
|
|
|
|
|
|
|
|
// test request
|
|
|
|
handler.ServeHTTP(response, request)
|
|
|
|
if response.Body == nil {
|
|
|
|
t.Fatalf("response has no body")
|
|
|
|
}
|
|
|
|
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|