components-contrib/middleware/http/wasm/httpwasm_test.go

137 lines
3.7 KiB
Go

/*
Copyright 2023 The Dapr Authors
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package wasm
import (
"bytes"
_ "embed"
"io"
"net/http"
"net/http/httptest"
"testing"
"github.com/http-wasm/http-wasm-host-go/api"
"github.com/stretchr/testify/require"
"github.com/dapr/components-contrib/common/httputils"
"github.com/dapr/components-contrib/metadata"
dapr "github.com/dapr/components-contrib/middleware"
"github.com/dapr/kit/logger"
)
//go:embed internal/testdata/rewrite.wasm
var exampleWasmBin []byte
func Test_NewMiddleWare(t *testing.T) {
l := logger.NewLogger(t.Name())
require.Equal(t, &middleware{logger: l}, NewMiddleware(l))
}
func Test_middleware_log(t *testing.T) {
l := logger.NewLogger(t.Name())
var buf bytes.Buffer
l.SetOutput(&buf)
m := &middleware{logger: l}
message := "alert"
m.Log(t.Context(), api.LogLevelInfo, message)
require.Contains(t, buf.String(), `level=info msg=alert`)
}
func Test_middleware_getHandler(t *testing.T) {
m := &middleware{logger: logger.NewLogger(t.Name())}
ts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
if r.URL.Path == "/example.wasm" {
w.Write(exampleWasmBin)
}
}))
type testCase struct {
name string
metadata metadata.Base
expectedErr string
}
tests := []testCase{
// This just tests the error message prefixes properly. Otherwise, it is
// redundant to Test_middleware_getMetadata
{
name: "requires url metadata",
metadata: metadata.Base{Properties: map[string]string{}},
expectedErr: "wasm: failed to parse metadata: missing url",
},
// This is more than Test_middleware_getMetadata, as it ensures the
// contents are actually wasm.
{
name: "url not wasm",
metadata: metadata.Base{Properties: map[string]string{
"url": "file://example/router.go",
}},
expectedErr: "wasm: error compiling guest: invalid magic number",
},
{
name: "remote wasm url",
metadata: metadata.Base{Properties: map[string]string{
"url": ts.URL + "/example.wasm",
}},
},
{
name: "ok",
metadata: metadata.Base{Properties: map[string]string{
"url": "file://example/router.wasm",
}},
},
}
for _, tt := range tests {
tc := tt
t.Run(tc.name, func(t *testing.T) {
h, err := m.getHandler(t.Context(), dapr.Metadata{Base: tc.metadata})
if tc.expectedErr == "" {
require.NoError(t, err)
require.NotNil(t, h.mw)
} else {
require.EqualError(t, err, tc.expectedErr)
}
})
}
}
func Test_Example(t *testing.T) {
l := logger.NewLogger(t.Name())
var buf bytes.Buffer
l.SetOutput(&buf)
meta := metadata.Base{Properties: map[string]string{
// router.wasm was compiled via the following:
// tinygo build -o router.wasm -scheduler=none --no-debug -target=wasi router.go`
"url": "file://example/router.wasm",
}}
handlerFn, err := NewMiddleware(l).GetHandler(t.Context(), dapr.Metadata{Base: meta})
require.NoError(t, err)
handler := handlerFn(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {}))
r := httptest.NewRequest(http.MethodGet, "/host/hi?name=panda", nil)
w := httptest.NewRecorder()
handler.ServeHTTP(w, r)
require.Equal(t, "/hi?name=panda", httputils.RequestURI(r))
require.Empty(t, buf.String())
}
func Test_ioCloser(t *testing.T) {
var _ io.Closer = &requestHandler{}
}