mirror of
				https://github.com/traefik/traefik.git
				synced 2025-10-31 00:11:38 +01: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)
 | |
| 			}
 | |
| 		})
 | |
| 	}
 | |
| }
 |