mirror of
https://github.com/router-for-me/CLIProxyAPI.git
synced 2026-06-22 06:57:37 +08:00
- Removed `examples/plugin/main.go` and `internal/pluginhost/loader_plugin.go` after migrating to a more modular system. - Introduced `streamBridge` in `internal/pluginhost/stream_bridge.go` for efficient stream handling and communication. - Added examples of `thinking` plugins written in both Rust and Go under `examples/plugin/thinking`. - Enhanced test coverage for plugin host system changes, including stream chunk translation and thinking logic. - Improved API compatibility and ensured backward-compatible upgrades for plugin execution.
196 lines
5.9 KiB
Go
196 lines
5.9 KiB
Go
package pluginhost
|
|
|
|
import (
|
|
"os"
|
|
"path/filepath"
|
|
"runtime"
|
|
"strings"
|
|
"testing"
|
|
)
|
|
|
|
func TestCandidateDirs(t *testing.T) {
|
|
got := candidateDirs("plugins", "darwin", "arm64", "v3")
|
|
want := []string{
|
|
filepath.Join("plugins", "darwin", "arm64-v3"),
|
|
filepath.Join("plugins", "darwin", "arm64"),
|
|
"plugins",
|
|
}
|
|
if len(got) != len(want) {
|
|
t.Fatalf("len(candidateDirs) = %d, want %d", len(got), len(want))
|
|
}
|
|
for index := range want {
|
|
if got[index] != want[index] {
|
|
t.Fatalf("candidateDirs[%d] = %q, want %q", index, got[index], want[index])
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestCandidateDirsOmitsEmptyVariant(t *testing.T) {
|
|
got := candidateDirs("plugins", "linux", "arm64", "")
|
|
want := []string{
|
|
filepath.Join("plugins", "linux", "arm64"),
|
|
"plugins",
|
|
}
|
|
if len(got) != len(want) {
|
|
t.Fatalf("len(candidateDirs) = %d, want %d", len(got), len(want))
|
|
}
|
|
for index := range want {
|
|
if got[index] != want[index] {
|
|
t.Fatalf("candidateDirs[%d] = %q, want %q", index, got[index], want[index])
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestPluginExtensionForPlatform(t *testing.T) {
|
|
cases := []struct {
|
|
goos string
|
|
want string
|
|
}{
|
|
{goos: "linux", want: ".so"},
|
|
{goos: "freebsd", want: ".so"},
|
|
{goos: "darwin", want: ".dylib"},
|
|
{goos: "windows", want: ".dll"},
|
|
}
|
|
|
|
for _, tc := range cases {
|
|
if got := pluginExtension(tc.goos); got != tc.want {
|
|
t.Fatalf("pluginExtension(%q) = %q, want %q", tc.goos, got, tc.want)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestPluginIDFromDynamicLibraryPath(t *testing.T) {
|
|
cases := map[string]string{
|
|
"plugins/example.so": "example",
|
|
"plugins/example.dylib": "example",
|
|
"plugins/example.dll": "example",
|
|
"plugins/example.custom": "example.custom",
|
|
}
|
|
|
|
for path, want := range cases {
|
|
if got := pluginIDFromPath(path); got != want {
|
|
t.Fatalf("pluginIDFromPath(%q) = %q, want %q", path, got, want)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestSelectPluginFilesFiltersInvalidIDAndDeduplicatesByID(t *testing.T) {
|
|
root := t.TempDir()
|
|
archDir := filepath.Join(root, runtime.GOOS, runtime.GOARCH)
|
|
if errMkdirAll := os.MkdirAll(archDir, 0o755); errMkdirAll != nil {
|
|
t.Fatalf("MkdirAll() error = %v", errMkdirAll)
|
|
}
|
|
|
|
extension := pluginExtension(runtime.GOOS)
|
|
paths := []string{
|
|
filepath.Join(root, "sample"+extension),
|
|
filepath.Join(archDir, "sample"+extension),
|
|
filepath.Join(archDir, "bad name"+extension),
|
|
filepath.Join(archDir, "-bad"+extension),
|
|
filepath.Join(archDir, "another"+strings.ToUpper(extension)),
|
|
filepath.Join(archDir, "ignored.txt"),
|
|
}
|
|
for _, path := range paths {
|
|
if errWriteFile := os.WriteFile(path, []byte("x"), 0o644); errWriteFile != nil {
|
|
t.Fatalf("WriteFile(%s) error = %v", path, errWriteFile)
|
|
}
|
|
}
|
|
if errMkdir := os.Mkdir(filepath.Join(archDir, "dir"+extension), 0o755); errMkdir != nil {
|
|
t.Fatalf("Mkdir() error = %v", errMkdir)
|
|
}
|
|
|
|
files, errSelect := selectPluginFiles(root)
|
|
if errSelect != nil {
|
|
t.Fatalf("selectPluginFiles() error = %v", errSelect)
|
|
}
|
|
|
|
want := []pluginFile{
|
|
{ID: "another", Path: filepath.Join(archDir, "another"+strings.ToUpper(extension))},
|
|
{ID: "sample", Path: filepath.Join(archDir, "sample"+extension)},
|
|
}
|
|
if len(files) != len(want) {
|
|
t.Fatalf("selectPluginFiles() = %v, want %v", files, want)
|
|
}
|
|
for index := range want {
|
|
if files[index] != want[index] {
|
|
t.Fatalf("selectPluginFiles()[%d] = %v, want %v", index, files[index], want[index])
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestSelectPluginFilesPrefersPlatformDirOverRootFallback(t *testing.T) {
|
|
root := t.TempDir()
|
|
archDir := filepath.Join(root, runtime.GOOS, runtime.GOARCH)
|
|
if errMkdirAll := os.MkdirAll(archDir, 0o755); errMkdirAll != nil {
|
|
t.Fatalf("MkdirAll() error = %v", errMkdirAll)
|
|
}
|
|
|
|
extension := pluginExtension(runtime.GOOS)
|
|
platformPath := filepath.Join(archDir, "alpha"+extension)
|
|
rootPath := filepath.Join(root, "alpha"+extension)
|
|
for _, path := range []string{rootPath, platformPath} {
|
|
if errWriteFile := os.WriteFile(path, []byte("x"), 0o644); errWriteFile != nil {
|
|
t.Fatalf("WriteFile(%s) error = %v", path, errWriteFile)
|
|
}
|
|
}
|
|
|
|
files, errSelect := selectPluginFiles(root)
|
|
if errSelect != nil {
|
|
t.Fatalf("selectPluginFiles() error = %v", errSelect)
|
|
}
|
|
if len(files) != 1 {
|
|
t.Fatalf("selectPluginFiles() = %v, want exactly one alpha plugin", files)
|
|
}
|
|
if files[0] != (pluginFile{ID: "alpha", Path: platformPath}) {
|
|
t.Fatalf("selectPluginFiles()[0] = %v, want platform plugin %s", files[0], platformPath)
|
|
}
|
|
}
|
|
|
|
func TestDiscoverPluginFilesReturnsSelectedPluginFiles(t *testing.T) {
|
|
root := makePluginDir(t, "alpha")
|
|
|
|
files, errDiscover := DiscoverPluginFiles(root)
|
|
if errDiscover != nil {
|
|
t.Fatalf("DiscoverPluginFiles() error = %v", errDiscover)
|
|
}
|
|
|
|
if len(files) != 1 || files[0].ID != "alpha" || files[0].Path == "" {
|
|
t.Fatalf("DiscoverPluginFiles() = %#v, want alpha file", files)
|
|
}
|
|
}
|
|
|
|
func TestSelectPluginFilesPrefersCPUVariantOverGenericArchDir(t *testing.T) {
|
|
variant := cpuVariant()
|
|
if variant == "" {
|
|
t.Skip("current GOARCH has no plugin CPU variant")
|
|
}
|
|
root := t.TempDir()
|
|
archDir := filepath.Join(root, runtime.GOOS, runtime.GOARCH)
|
|
variantDir := filepath.Join(root, runtime.GOOS, runtime.GOARCH+"-"+variant)
|
|
for _, dir := range []string{archDir, variantDir} {
|
|
if errMkdirAll := os.MkdirAll(dir, 0o755); errMkdirAll != nil {
|
|
t.Fatalf("MkdirAll(%s) error = %v", dir, errMkdirAll)
|
|
}
|
|
}
|
|
|
|
extension := pluginExtension(runtime.GOOS)
|
|
genericPath := filepath.Join(archDir, "alpha"+extension)
|
|
variantPath := filepath.Join(variantDir, "alpha"+extension)
|
|
for _, path := range []string{genericPath, variantPath} {
|
|
if errWriteFile := os.WriteFile(path, []byte("x"), 0o644); errWriteFile != nil {
|
|
t.Fatalf("WriteFile(%s) error = %v", path, errWriteFile)
|
|
}
|
|
}
|
|
|
|
files, errSelect := selectPluginFiles(root)
|
|
if errSelect != nil {
|
|
t.Fatalf("selectPluginFiles() error = %v", errSelect)
|
|
}
|
|
if len(files) != 1 {
|
|
t.Fatalf("selectPluginFiles() = %v, want exactly one alpha plugin", files)
|
|
}
|
|
if files[0] != (pluginFile{ID: "alpha", Path: variantPath}) {
|
|
t.Fatalf("selectPluginFiles()[0] = %v, want CPU variant plugin %s", files[0], variantPath)
|
|
}
|
|
}
|