1
0
mirror of https://github.com/containous/traefik.git synced 2025-01-03 01:17:53 +03:00

Fix WASM settings

This commit is contained in:
Julien Salleyron 2024-12-06 16:38:05 +01:00 committed by GitHub
parent 47b4df71bf
commit c8b0285c91
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
6 changed files with 183 additions and 4 deletions

View File

@ -0,0 +1,52 @@
package main
import (
"encoding/json"
"fmt"
"net/http"
"os"
"github.com/http-wasm/http-wasm-guest-tinygo/handler"
"github.com/http-wasm/http-wasm-guest-tinygo/handler/api"
)
type config struct {
File string `json:"file"`
Envs []string `json:"envs"`
}
var cfg config
// Built with
// tinygo build -o plugin.wasm -scheduler=none --no-debug -target=wasi ./demo.go
func main() {
err := json.Unmarshal(handler.Host.GetConfig(), &cfg)
if err != nil {
handler.Host.Log(api.LogLevelError, fmt.Sprintf("Could not load config %v", err))
os.Exit(1)
}
handler.HandleRequestFn = handleRequest
}
func handleRequest(req api.Request, resp api.Response) (next bool, reqCtx uint32) {
var bodyContent []byte
if cfg.File != "" {
var err error
bodyContent, err = os.ReadFile(cfg.File)
if err != nil {
resp.SetStatusCode(http.StatusInternalServerError)
resp.Body().Write([]byte(fmt.Sprintf("error reading file %q: %w", cfg.File, err.Error())))
return false, 0
}
}
if len(cfg.Envs) > 0 {
for _, env := range cfg.Envs {
bodyContent = append(bodyContent, []byte(os.Getenv(env)+"\n")...)
}
}
resp.Body().Write(bodyContent)
return false, 0
}

View File

@ -0,0 +1,5 @@
module withoutsocket
go 1.23.0
require github.com/http-wasm/http-wasm-guest-tinygo v0.4.0

View File

@ -0,0 +1,2 @@
github.com/http-wasm/http-wasm-guest-tinygo v0.4.0 h1:sWd1hqOL8LF3DVRPXloVELTQItibKtDCtVSA4UfMf4Y=
github.com/http-wasm/http-wasm-guest-tinygo v0.4.0/go.mod h1:zcKr7h/t5ha2ZWIMwV4iOqhfC/qno/tNPYgybVkn/MQ=

Binary file not shown.

View File

@ -83,7 +83,7 @@ func (b *wasmMiddlewareBuilder) buildMiddleware(ctx context.Context, next http.H
config := wazero.NewModuleConfig().WithSysWalltime().WithStartFunctions("_start", "_initialize") config := wazero.NewModuleConfig().WithSysWalltime().WithStartFunctions("_start", "_initialize")
for _, env := range b.settings.Envs { for _, env := range b.settings.Envs {
config.WithEnv(env, os.Getenv(env)) config = config.WithEnv(env, os.Getenv(env))
} }
if len(b.settings.Mounts) > 0 { if len(b.settings.Mounts) > 0 {
@ -97,14 +97,14 @@ func (b *wasmMiddlewareBuilder) buildMiddleware(ctx context.Context, next http.H
parts := strings.Split(prefix, ":") parts := strings.Split(prefix, ":")
switch { switch {
case len(parts) == 1: case len(parts) == 1:
withDir(parts[0], parts[0]) fsConfig = withDir(parts[0], parts[0])
case len(parts) == 2: case len(parts) == 2:
withDir(parts[0], parts[1]) fsConfig = withDir(parts[0], parts[1])
default: default:
return nil, nil, fmt.Errorf("invalid directory %q", mount) return nil, nil, fmt.Errorf("invalid directory %q", mount)
} }
} }
config.WithFSConfig(fsConfig) config = config.WithFSConfig(fsConfig)
} }
opts := []handler.Option{ opts := []handler.Option{

View File

@ -0,0 +1,120 @@
package plugins
import (
"context"
"net/http"
"net/http/httptest"
"os"
"path"
"reflect"
"testing"
"github.com/rs/zerolog"
"github.com/rs/zerolog/log"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
"github.com/tetratelabs/wazero"
)
func TestSettingsWithoutSocket(t *testing.T) {
cache := wazero.NewCompilationCache()
zerolog.SetGlobalLevel(zerolog.DebugLevel)
ctx := log.Logger.WithContext(context.Background())
t.Setenv("PLUGIN_TEST", "MY-TEST")
t.Setenv("PLUGIN_TEST_B", "MY-TEST_B")
testCases := []struct {
desc string
getSettings func(t *testing.T) (Settings, map[string]interface{})
expected string
}{
{
desc: "mounts path",
getSettings: func(t *testing.T) (Settings, map[string]interface{}) {
t.Helper()
tempDir := t.TempDir()
filePath := path.Join(tempDir, "hello.txt")
err := os.WriteFile(filePath, []byte("content_test"), 0o644)
require.NoError(t, err)
return Settings{Mounts: []string{
tempDir,
}}, map[string]interface{}{
"file": filePath,
}
},
expected: "content_test",
},
{
desc: "mounts src to dest",
getSettings: func(t *testing.T) (Settings, map[string]interface{}) {
t.Helper()
tempDir := t.TempDir()
filePath := path.Join(tempDir, "hello.txt")
err := os.WriteFile(filePath, []byte("content_test"), 0o644)
require.NoError(t, err)
return Settings{Mounts: []string{
tempDir + ":/tmp",
}}, map[string]interface{}{
"file": "/tmp/hello.txt",
}
},
expected: "content_test",
},
{
desc: "one env",
getSettings: func(t *testing.T) (Settings, map[string]interface{}) {
t.Helper()
envs := []string{"PLUGIN_TEST"}
return Settings{Envs: envs}, map[string]interface{}{
"envs": envs,
}
},
expected: "MY-TEST\n",
},
{
desc: "two env",
getSettings: func(t *testing.T) (Settings, map[string]interface{}) {
t.Helper()
envs := []string{"PLUGIN_TEST", "PLUGIN_TEST_B"}
return Settings{Envs: envs}, map[string]interface{}{
"envs": envs,
}
},
expected: "MY-TEST\nMY-TEST_B\n",
},
}
for _, test := range testCases {
t.Run(test.desc, func(t *testing.T) {
t.Parallel()
settings, config := test.getSettings(t)
builder := &wasmMiddlewareBuilder{path: "./fixtures/withoutsocket/plugin.wasm", cache: cache, settings: settings}
cfg := reflect.ValueOf(config)
m, applyCtx, err := builder.buildMiddleware(ctx, http.HandlerFunc(func(rw http.ResponseWriter, req *http.Request) {
rw.WriteHeader(http.StatusTeapot)
}), cfg, "test")
require.NoError(t, err)
rw := httptest.NewRecorder()
req := httptest.NewRequestWithContext(applyCtx(ctx), "GET", "/", http.NoBody)
m.ServeHTTP(rw, req)
assert.Equal(t, http.StatusOK, rw.Code)
assert.Equal(t, test.expected, rw.Body.String())
})
}
}