go-scm/plugin/installer_test.go
Snider e5e6698662 Merge github/main into dev — resolve module path conflicts
Keep dappco.re module paths throughout; github/main had stale forge.lthn.ai
paths in go.mod and several source files. All conflicts resolved in favour of
dev (HEAD): AX-pattern aliased imports, SecureSSHCommandContext, syncutil,
and _Good test naming convention.

Note: go mod tidy fails — dappco.re/go/core/cli@v0.3.7 and
dappco.re/go/core/config@v0.1.8 tags still declare forge.lthn.ai module
paths. New tags with correct dappco.re paths are needed on both repos
before tidy will succeed.

Co-Authored-By: Virgil <virgil@lethean.io>
2026-04-07 08:42:34 +01:00

214 lines
6.3 KiB
Go

// SPDX-License-Identifier: EUPL-1.2
package plugin
import (
"context"
"testing"
"dappco.re/go/core/io"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
)
// ── NewInstaller ───────────────────────────────────────────────────
func TestNewInstaller_Good(t *testing.T) {
m := io.NewMockMedium()
reg := NewRegistry(m, "/plugins")
inst := NewInstaller(m, reg)
assert.NotNil(t, inst)
assert.Equal(t, m, inst.medium)
assert.Equal(t, reg, inst.registry)
}
// ── Install error paths ────────────────────────────────────────────
func TestInstall_Bad_InvalidSource_Good(t *testing.T) {
m := io.NewMockMedium()
reg := NewRegistry(m, "/plugins")
inst := NewInstaller(m, reg)
err := inst.Install(context.Background(), "bad-source")
assert.Error(t, err)
assert.Contains(t, err.Error(), "invalid source")
}
func TestInstall_Bad_AlreadyInstalled_Good(t *testing.T) {
m := io.NewMockMedium()
reg := NewRegistry(m, "/plugins")
_ = reg.Add(&PluginConfig{Name: "my-plugin", Version: "1.0.0"})
inst := NewInstaller(m, reg)
err := inst.Install(context.Background(), "org/my-plugin")
assert.Error(t, err)
assert.Contains(t, err.Error(), "already installed")
}
func TestInstall_Bad_PathTraversalSource(t *testing.T) {
m := io.NewMockMedium()
reg := NewRegistry(m, "/plugins")
inst := NewInstaller(m, reg)
err := inst.Install(context.Background(), "../repo")
require.Error(t, err)
assert.Contains(t, err.Error(), "invalid source")
assert.False(t, m.Exists("/repo"))
}
// ── Remove ─────────────────────────────────────────────────────────
func TestRemove_Good(t *testing.T) {
m := io.NewMockMedium()
reg := NewRegistry(m, "/plugins")
_ = reg.Add(&PluginConfig{Name: "removable", Version: "1.0.0"})
// Create plugin directory.
_ = m.EnsureDir("/plugins/removable")
_ = m.Write("/plugins/removable/plugin.json", `{"name":"removable"}`)
inst := NewInstaller(m, reg)
err := inst.Remove("removable")
require.NoError(t, err)
// Plugin removed from registry.
_, ok := reg.Get("removable")
assert.False(t, ok)
// Directory cleaned up.
assert.False(t, m.Exists("/plugins/removable"))
}
func TestRemove_Good_DirAlreadyGone_Good(t *testing.T) {
m := io.NewMockMedium()
reg := NewRegistry(m, "/plugins")
_ = reg.Add(&PluginConfig{Name: "ghost", Version: "1.0.0"})
// No directory exists — should still succeed.
inst := NewInstaller(m, reg)
err := inst.Remove("ghost")
require.NoError(t, err)
_, ok := reg.Get("ghost")
assert.False(t, ok)
}
func TestRemove_Bad_NotFound_Good(t *testing.T) {
m := io.NewMockMedium()
reg := NewRegistry(m, "/plugins")
inst := NewInstaller(m, reg)
err := inst.Remove("nonexistent")
assert.Error(t, err)
assert.Contains(t, err.Error(), "plugin not found")
}
func TestRemove_Bad_PathTraversalName(t *testing.T) {
m := io.NewMockMedium()
reg := NewRegistry(m, "/plugins")
_ = reg.Add(&PluginConfig{Name: "safe", Version: "1.0.0"})
_ = m.EnsureDir("/escape")
inst := NewInstaller(m, reg)
err := inst.Remove("../escape")
require.Error(t, err)
assert.Contains(t, err.Error(), "invalid plugin name")
assert.True(t, m.Exists("/escape"))
}
// ── Update error paths ─────────────────────────────────────────────
func TestUpdate_Bad_NotFound_Good(t *testing.T) {
m := io.NewMockMedium()
reg := NewRegistry(m, "/plugins")
inst := NewInstaller(m, reg)
err := inst.Update(context.Background(), "missing")
assert.Error(t, err)
assert.Contains(t, err.Error(), "plugin not found")
}
// ── ParseSource ────────────────────────────────────────────────────
func TestParseSource_Good_OrgRepo_Good(t *testing.T) {
org, repo, version, err := ParseSource("host-uk/core-plugin")
assert.NoError(t, err)
assert.Equal(t, "host-uk", org)
assert.Equal(t, "core-plugin", repo)
assert.Equal(t, "", version)
}
func TestParseSource_Good_OrgRepoVersion_Good(t *testing.T) {
org, repo, version, err := ParseSource("host-uk/core-plugin@v1.0.0")
assert.NoError(t, err)
assert.Equal(t, "host-uk", org)
assert.Equal(t, "core-plugin", repo)
assert.Equal(t, "v1.0.0", version)
}
func TestParseSource_Good_VersionWithoutPrefix_Good(t *testing.T) {
org, repo, version, err := ParseSource("org/repo@1.2.3")
assert.NoError(t, err)
assert.Equal(t, "org", org)
assert.Equal(t, "repo", repo)
assert.Equal(t, "1.2.3", version)
}
func TestParseSource_Bad_Empty_Good(t *testing.T) {
_, _, _, err := ParseSource("")
assert.Error(t, err)
assert.Contains(t, err.Error(), "source is empty")
}
func TestParseSource_Bad_NoSlash_Good(t *testing.T) {
_, _, _, err := ParseSource("just-a-name")
assert.Error(t, err)
assert.Contains(t, err.Error(), "org/repo")
}
func TestParseSource_Bad_TooManySlashes_Good(t *testing.T) {
_, _, _, err := ParseSource("a/b/c")
assert.Error(t, err)
assert.Contains(t, err.Error(), "org/repo")
}
func TestParseSource_Bad_EmptyOrg_Good(t *testing.T) {
_, _, _, err := ParseSource("/repo")
assert.Error(t, err)
assert.Contains(t, err.Error(), "org/repo")
}
func TestParseSource_Bad_EmptyRepo_Good(t *testing.T) {
_, _, _, err := ParseSource("org/")
assert.Error(t, err)
assert.Contains(t, err.Error(), "org/repo")
}
func TestParseSource_Bad_EmptyVersion_Good(t *testing.T) {
_, _, _, err := ParseSource("org/repo@")
assert.Error(t, err)
assert.Contains(t, err.Error(), "version is empty")
}
func TestParseSource_Bad_PathTraversal(t *testing.T) {
_, _, _, err := ParseSource("org/../repo")
assert.Error(t, err)
assert.Contains(t, err.Error(), "org/repo")
}
func TestParseSource_Bad_PathTraversalEncoded(t *testing.T) {
_, _, _, err := ParseSource("org%2f..%2frepo")
assert.Error(t, err)
assert.Contains(t, err.Error(), "org/repo")
}
func TestInstall_Bad_EncodedPathTraversal(t *testing.T) {
m := io.NewMockMedium()
reg := NewRegistry(m, "/plugins")
inst := NewInstaller(m, reg)
err := inst.Install(context.Background(), "org%2f..%2frepo")
require.Error(t, err)
assert.Contains(t, err.Error(), "invalid source")
}