mirror of
https://github.com/traefik/traefik.git
synced 2025-09-20 21:31:14 +02:00
342 lines
9.1 KiB
Go
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)
|
|
}
|
|
})
|
|
}
|
|
}
|