traefik/pkg/plugins/manager_test.go
2025-09-16 16:16:07 +02:00

342 lines
9.1 KiB
Go

package plugins
import (
zipa "archive/zip"
"context"
"encoding/json"
"os"
"path/filepath"
"testing"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
"gopkg.in/yaml.v3"
)
// mockDownloader is a test implementation of PluginDownloader
type mockDownloader struct {
downloadFunc func(ctx context.Context, pName, pVersion string) (string, error)
checkFunc func(ctx context.Context, pName, pVersion, hash string) error
}
func (m *mockDownloader) Download(ctx context.Context, pName, pVersion string) (string, error) {
if m.downloadFunc != nil {
return m.downloadFunc(ctx, pName, pVersion)
}
return "mockhash", nil
}
func (m *mockDownloader) Check(ctx context.Context, pName, pVersion, hash string) error {
if m.checkFunc != nil {
return m.checkFunc(ctx, pName, pVersion, hash)
}
return nil
}
func TestPluginManager_ReadManifest(t *testing.T) {
tempDir := t.TempDir()
opts := ManagerOptions{Output: tempDir}
downloader := &mockDownloader{}
manager, err := NewManager(downloader, opts)
require.NoError(t, err)
moduleName := "github.com/test/plugin"
pluginPath := filepath.Join(manager.goPath, "src", moduleName)
err = os.MkdirAll(pluginPath, 0o755)
require.NoError(t, err)
manifest := &Manifest{
DisplayName: "Test Plugin",
Type: "middleware",
Import: "github.com/test/plugin",
Summary: "A test plugin",
TestData: map[string]interface{}{
"test": "data",
},
}
manifestPath := filepath.Join(pluginPath, pluginManifest)
manifestData, err := yaml.Marshal(manifest)
require.NoError(t, err)
err = os.WriteFile(manifestPath, manifestData, 0o644)
require.NoError(t, err)
readManifest, err := manager.ReadManifest(moduleName)
require.NoError(t, err)
assert.Equal(t, manifest.DisplayName, readManifest.DisplayName)
assert.Equal(t, manifest.Type, readManifest.Type)
assert.Equal(t, manifest.Import, readManifest.Import)
assert.Equal(t, manifest.Summary, readManifest.Summary)
}
func TestPluginManager_ReadManifest_NotFound(t *testing.T) {
tempDir := t.TempDir()
opts := ManagerOptions{Output: tempDir}
downloader := &mockDownloader{}
manager, err := NewManager(downloader, opts)
require.NoError(t, err)
_, err = manager.ReadManifest("nonexistent/plugin")
assert.Error(t, err)
}
func TestPluginManager_CleanArchives(t *testing.T) {
tempDir := t.TempDir()
opts := ManagerOptions{Output: tempDir}
downloader := &mockDownloader{}
manager, err := NewManager(downloader, opts)
require.NoError(t, err)
testPlugin1 := "test/plugin1"
testPlugin2 := "test/plugin2"
archive1Dir := filepath.Join(manager.archives, "test", "plugin1")
archive2Dir := filepath.Join(manager.archives, "test", "plugin2")
err = os.MkdirAll(archive1Dir, 0o755)
require.NoError(t, err)
err = os.MkdirAll(archive2Dir, 0o755)
require.NoError(t, err)
archive1Old := filepath.Join(archive1Dir, "v1.0.0.zip")
archive1New := filepath.Join(archive1Dir, "v2.0.0.zip")
archive2 := filepath.Join(archive2Dir, "v1.0.0.zip")
err = os.WriteFile(archive1Old, []byte("old archive"), 0o644)
require.NoError(t, err)
err = os.WriteFile(archive1New, []byte("new archive"), 0o644)
require.NoError(t, err)
err = os.WriteFile(archive2, []byte("archive 2"), 0o644)
require.NoError(t, err)
state := map[string]string{
testPlugin1: "v1.0.0",
testPlugin2: "v1.0.0",
}
stateData, err := json.MarshalIndent(state, "", " ")
require.NoError(t, err)
err = os.WriteFile(manager.stateFile, stateData, 0o600)
require.NoError(t, err)
currentPlugins := map[string]Descriptor{
"plugin1": {
ModuleName: testPlugin1,
Version: "v2.0.0",
},
"plugin2": {
ModuleName: testPlugin2,
Version: "v1.0.0",
},
}
err = manager.CleanArchives(currentPlugins)
require.NoError(t, err)
assert.NoFileExists(t, archive1Old)
assert.FileExists(t, archive1New)
assert.FileExists(t, archive2)
}
func TestPluginManager_WriteState(t *testing.T) {
tempDir := t.TempDir()
opts := ManagerOptions{Output: tempDir}
downloader := &mockDownloader{}
manager, err := NewManager(downloader, opts)
require.NoError(t, err)
plugins := map[string]Descriptor{
"plugin1": {
ModuleName: "test/plugin1",
Version: "v1.0.0",
},
"plugin2": {
ModuleName: "test/plugin2",
Version: "v2.0.0",
},
}
err = manager.WriteState(plugins)
require.NoError(t, err)
assert.FileExists(t, manager.stateFile)
data, err := os.ReadFile(manager.stateFile)
require.NoError(t, err)
var state map[string]string
err = json.Unmarshal(data, &state)
require.NoError(t, err)
expectedState := map[string]string{
"test/plugin1": "v1.0.0",
"test/plugin2": "v2.0.0",
}
assert.Equal(t, expectedState, state)
}
func TestPluginManager_ResetAll(t *testing.T) {
tempDir := t.TempDir()
opts := ManagerOptions{Output: tempDir}
downloader := &mockDownloader{}
manager, err := NewManager(downloader, opts)
require.NoError(t, err)
testFile := filepath.Join(manager.GoPath(), "test.txt")
err = os.WriteFile(testFile, []byte("test"), 0o644)
require.NoError(t, err)
archiveFile := filepath.Join(manager.archives, "test.zip")
err = os.WriteFile(archiveFile, []byte("archive"), 0o644)
require.NoError(t, err)
err = manager.ResetAll()
require.NoError(t, err)
assert.DirExists(t, manager.archives)
assert.NoFileExists(t, testFile)
assert.NoFileExists(t, archiveFile)
}
func TestPluginManager_InstallPlugin(t *testing.T) {
tests := []struct {
name string
plugin Descriptor
downloadFunc func(ctx context.Context, pName, pVersion string) (string, error)
checkFunc func(ctx context.Context, pName, pVersion, hash string) error
setupArchive func(t *testing.T, archivePath string)
expectError bool
errorMsg string
}{
{
name: "successful installation",
plugin: Descriptor{
ModuleName: "github.com/test/plugin",
Version: "v1.0.0",
Hash: "expected-hash",
},
downloadFunc: func(ctx context.Context, pName, pVersion string) (string, error) {
return "expected-hash", nil
},
checkFunc: func(ctx context.Context, pName, pVersion, hash string) error {
return nil
},
setupArchive: func(t *testing.T, archivePath string) {
t.Helper()
// Create a valid zip archive
err := os.MkdirAll(filepath.Dir(archivePath), 0o755)
require.NoError(t, err)
file, err := os.Create(archivePath)
require.NoError(t, err)
defer file.Close()
// Write a minimal zip file with a test file
writer := zipa.NewWriter(file)
defer writer.Close()
fileWriter, err := writer.Create("test-module-v1.0.0/main.go")
require.NoError(t, err)
_, err = fileWriter.Write([]byte("package main\n\nfunc main() {}\n"))
require.NoError(t, err)
},
expectError: false,
},
{
name: "download error",
plugin: Descriptor{
ModuleName: "github.com/test/plugin",
Version: "v1.0.0",
},
downloadFunc: func(ctx context.Context, pName, pVersion string) (string, error) {
return "", assert.AnError
},
expectError: true,
errorMsg: "unable to download plugin",
},
{
name: "check error",
plugin: Descriptor{
ModuleName: "github.com/test/plugin",
Version: "v1.0.0",
Hash: "expected-hash",
},
downloadFunc: func(ctx context.Context, pName, pVersion string) (string, error) {
return "actual-hash", nil
},
checkFunc: func(ctx context.Context, pName, pVersion, hash string) error {
return assert.AnError
},
expectError: true,
errorMsg: "invalid hash for plugin",
},
{
name: "unzip error - invalid archive",
plugin: Descriptor{
ModuleName: "github.com/test/plugin",
Version: "v1.0.0",
},
downloadFunc: func(ctx context.Context, pName, pVersion string) (string, error) {
return "test-hash", nil
},
checkFunc: func(ctx context.Context, pName, pVersion, hash string) error {
return nil
},
setupArchive: func(t *testing.T, archivePath string) {
t.Helper()
// Create an invalid zip archive
err := os.MkdirAll(filepath.Dir(archivePath), 0o755)
require.NoError(t, err)
err = os.WriteFile(archivePath, []byte("invalid zip content"), 0o644)
require.NoError(t, err)
},
expectError: true,
errorMsg: "unable to unzip plugin",
},
}
for _, test := range tests {
t.Run(test.name, func(t *testing.T) {
tempDir := t.TempDir()
opts := ManagerOptions{Output: tempDir}
downloader := &mockDownloader{
downloadFunc: test.downloadFunc,
checkFunc: test.checkFunc,
}
manager, err := NewManager(downloader, opts)
require.NoError(t, err)
// Setup archive if needed
if test.setupArchive != nil {
archivePath := filepath.Join(manager.archives,
filepath.FromSlash(test.plugin.ModuleName),
test.plugin.Version+".zip")
test.setupArchive(t, archivePath)
}
ctx := t.Context()
err = manager.InstallPlugin(ctx, test.plugin)
if test.expectError {
assert.Error(t, err)
if test.errorMsg != "" {
assert.Contains(t, err.Error(), test.errorMsg)
}
} else {
assert.NoError(t, err)
// Verify that plugin sources were extracted
sourcePath := filepath.Join(manager.sources, filepath.FromSlash(test.plugin.ModuleName))
assert.DirExists(t, sourcePath)
}
})
}
}