mirror of
https://github.com/router-for-me/CLIProxyAPI.git
synced 2026-06-12 09:13:03 +08:00
- Added `resources` field in `management.register` for defining browser-accessible resources. - Updated examples and documentation to reflect resource-based paths under `/v0/resource/plugins/<pluginID>/...`. - Replaced legacy `GET` menu routes with resource-based implementations for consistent plugin behavior. - Enhanced request handling for resource paths, including proper response headers and streamlined test coverage.
346 lines
11 KiB
Go
346 lines
11 KiB
Go
package pluginhost
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
"strings"
|
|
|
|
"github.com/router-for-me/CLIProxyAPI/v7/sdk/pluginapi"
|
|
log "github.com/sirupsen/logrus"
|
|
)
|
|
|
|
const (
|
|
managementBasePath = "/v0/management"
|
|
resourcePluginBasePath = "/v0/resource/plugins"
|
|
legacyPluginRoutePrefix = "/plugins"
|
|
)
|
|
|
|
type managementRouteRecord struct {
|
|
pluginID string
|
|
route pluginapi.ManagementRoute
|
|
}
|
|
|
|
type resourceRouteRecord struct {
|
|
pluginID string
|
|
route pluginapi.ResourceRoute
|
|
}
|
|
|
|
// RegisterManagementRoutes rebuilds the plugin-owned Management API and resource route tables.
|
|
func (h *Host) RegisterManagementRoutes(ctx context.Context, reserved map[string]struct{}) {
|
|
if h == nil {
|
|
return
|
|
}
|
|
|
|
nextRoutes := make(map[string]managementRouteRecord)
|
|
nextResources := make(map[string]resourceRouteRecord)
|
|
for _, record := range h.Snapshot().records {
|
|
plugin := record.plugin.Capabilities.ManagementAPI
|
|
if plugin == nil || h.isPluginFused(record.id) {
|
|
continue
|
|
}
|
|
resp, errRegister := h.callManagementRegistrar(ctx, record, plugin)
|
|
if errRegister != nil {
|
|
log.Warnf("pluginhost: management registrar %s failed: %v", record.id, errRegister)
|
|
continue
|
|
}
|
|
|
|
for _, item := range resp.Routes {
|
|
method, path, okRoute := normalizeManagementRoute(item)
|
|
if !okRoute {
|
|
log.Warnf("pluginhost: plugin %s declared invalid management route %s %s", record.id, item.Method, item.Path)
|
|
continue
|
|
}
|
|
if routeDeclaresLegacyMenuResource(method, item) {
|
|
if !registerResourceRoute(nextResources, record.id, resourceRouteFromManagementRoute(item)) {
|
|
log.Warnf("pluginhost: plugin %s declared invalid resource route %s", record.id, item.Path)
|
|
}
|
|
continue
|
|
}
|
|
key := managementRouteKey(method, path)
|
|
if _, exists := reserved[key]; exists {
|
|
log.Warnf("pluginhost: plugin %s management route %s conflicts with an existing route and was skipped", record.id, key)
|
|
continue
|
|
}
|
|
if _, exists := nextRoutes[key]; exists {
|
|
log.Warnf("pluginhost: plugin %s management route %s conflicts with a higher-priority plugin and was skipped", record.id, key)
|
|
continue
|
|
}
|
|
item.Method = method
|
|
item.Path = path
|
|
nextRoutes[key] = managementRouteRecord{
|
|
pluginID: record.id,
|
|
route: item,
|
|
}
|
|
}
|
|
|
|
for _, item := range resp.Resources {
|
|
if !registerResourceRoute(nextResources, record.id, item) {
|
|
log.Warnf("pluginhost: plugin %s declared invalid resource route %s", record.id, item.Path)
|
|
}
|
|
}
|
|
}
|
|
|
|
h.mu.Lock()
|
|
h.managementRoutes = nextRoutes
|
|
h.resourceRoutes = nextResources
|
|
h.mu.Unlock()
|
|
}
|
|
|
|
func (h *Host) callManagementRegistrar(ctx context.Context, record capabilityRecord, plugin pluginapi.ManagementAPI) (resp pluginapi.ManagementRegistrationResponse, err error) {
|
|
if h == nil || plugin == nil || h.isPluginFused(record.id) {
|
|
return pluginapi.ManagementRegistrationResponse{}, nil
|
|
}
|
|
defer func() {
|
|
if recovered := recover(); recovered != nil {
|
|
h.fusePlugin(record.id, "ManagementAPI.RegisterManagement", recovered)
|
|
resp = pluginapi.ManagementRegistrationResponse{}
|
|
err = fmt.Errorf("management registrar panic: %v", recovered)
|
|
}
|
|
}()
|
|
return plugin.RegisterManagement(ctx, pluginapi.ManagementRegistrationRequest{
|
|
Plugin: record.meta,
|
|
BasePath: managementBasePath,
|
|
ResourceBasePath: resourcePluginBasePath + "/" + record.id,
|
|
})
|
|
}
|
|
|
|
func normalizeManagementRoute(item pluginapi.ManagementRoute) (string, string, bool) {
|
|
if item.Handler == nil {
|
|
return "", "", false
|
|
}
|
|
method := strings.ToUpper(strings.TrimSpace(item.Method))
|
|
if method == "" {
|
|
method = http.MethodGet
|
|
}
|
|
if strings.ContainsAny(method, " \t\r\n") {
|
|
return "", "", false
|
|
}
|
|
|
|
path := strings.TrimSpace(item.Path)
|
|
if path == "" {
|
|
return "", "", false
|
|
}
|
|
if !strings.HasPrefix(path, "/") {
|
|
path = "/" + path
|
|
}
|
|
if strings.HasPrefix(path, managementBasePath+"/") {
|
|
path = strings.TrimPrefix(path, managementBasePath)
|
|
}
|
|
path = strings.TrimRight(path, "/")
|
|
if path == "" {
|
|
return "", "", false
|
|
}
|
|
fullPath := managementBasePath + path
|
|
if !strings.HasPrefix(fullPath, managementBasePath+"/") {
|
|
return "", "", false
|
|
}
|
|
if strings.ContainsAny(fullPath, " \t\r\n") || strings.Contains(fullPath, ":") || strings.Contains(fullPath, "*") {
|
|
return "", "", false
|
|
}
|
|
return method, fullPath, true
|
|
}
|
|
|
|
func routeDeclaresLegacyMenuResource(method string, item pluginapi.ManagementRoute) bool {
|
|
return strings.EqualFold(strings.TrimSpace(method), http.MethodGet) && strings.TrimSpace(item.Menu) != ""
|
|
}
|
|
|
|
func resourceRouteFromManagementRoute(item pluginapi.ManagementRoute) pluginapi.ResourceRoute {
|
|
return pluginapi.ResourceRoute{
|
|
Path: item.Path,
|
|
Menu: item.Menu,
|
|
Description: item.Description,
|
|
Handler: item.Handler,
|
|
}
|
|
}
|
|
|
|
func registerResourceRoute(routes map[string]resourceRouteRecord, pluginID string, item pluginapi.ResourceRoute) bool {
|
|
path, okRoute := normalizeResourceRoute(pluginID, item)
|
|
if !okRoute {
|
|
return false
|
|
}
|
|
key := managementRouteKey(http.MethodGet, path)
|
|
if _, exists := routes[key]; exists {
|
|
log.Warnf("pluginhost: plugin %s resource route %s conflicts with a higher-priority plugin and was skipped", pluginID, key)
|
|
return true
|
|
}
|
|
item.Path = path
|
|
routes[key] = resourceRouteRecord{
|
|
pluginID: pluginID,
|
|
route: item,
|
|
}
|
|
return true
|
|
}
|
|
|
|
func normalizeResourceRoute(pluginID string, item pluginapi.ResourceRoute) (string, bool) {
|
|
if item.Handler == nil {
|
|
return "", false
|
|
}
|
|
pluginID = strings.TrimSpace(pluginID)
|
|
if pluginID == "" {
|
|
return "", false
|
|
}
|
|
|
|
path := strings.TrimSpace(item.Path)
|
|
if path == "" {
|
|
return "", false
|
|
}
|
|
if !strings.HasPrefix(path, "/") {
|
|
path = "/" + path
|
|
}
|
|
|
|
pluginBasePath := resourcePluginBasePath + "/" + pluginID
|
|
if strings.HasPrefix(path, pluginBasePath+"/") {
|
|
path = strings.TrimPrefix(path, pluginBasePath)
|
|
} else if strings.HasPrefix(path, legacyPluginRoutePrefix+"/"+pluginID+"/") {
|
|
path = strings.TrimPrefix(path, legacyPluginRoutePrefix+"/"+pluginID)
|
|
}
|
|
path = strings.TrimRight(path, "/")
|
|
if path == "" {
|
|
return "", false
|
|
}
|
|
|
|
fullPath := pluginBasePath + path
|
|
if !strings.HasPrefix(fullPath, pluginBasePath+"/") {
|
|
return "", false
|
|
}
|
|
if strings.ContainsAny(fullPath, " \t\r\n") || strings.Contains(fullPath, ":") || strings.Contains(fullPath, "*") || strings.Contains(fullPath, "..") {
|
|
return "", false
|
|
}
|
|
return fullPath, true
|
|
}
|
|
|
|
func managementRouteKey(method, path string) string {
|
|
return strings.ToUpper(strings.TrimSpace(method)) + " " + strings.TrimSpace(path)
|
|
}
|
|
|
|
// ServeManagementHTTP dispatches an authenticated Management API request to a plugin route.
|
|
func (h *Host) ServeManagementHTTP(w http.ResponseWriter, r *http.Request) bool {
|
|
if h == nil || w == nil || r == nil || r.URL == nil {
|
|
return false
|
|
}
|
|
key := managementRouteKey(r.Method, r.URL.Path)
|
|
h.mu.Lock()
|
|
record, okRoute := h.managementRoutes[key]
|
|
h.mu.Unlock()
|
|
if !okRoute || record.route.Handler == nil || h.isPluginFused(record.pluginID) {
|
|
return false
|
|
}
|
|
|
|
var body []byte
|
|
if r.Body != nil {
|
|
var errRead error
|
|
body, errRead = io.ReadAll(r.Body)
|
|
if errRead != nil {
|
|
http.Error(w, "failed to read plugin management request body", http.StatusBadRequest)
|
|
return true
|
|
}
|
|
if errClose := r.Body.Close(); errClose != nil {
|
|
log.Warnf("pluginhost: failed to close plugin management request body: %v", errClose)
|
|
}
|
|
}
|
|
r.Body = io.NopCloser(bytes.NewReader(body))
|
|
|
|
resp, errHandle := h.callManagementHandler(r.Context(), record, pluginapi.ManagementRequest{
|
|
Method: r.Method,
|
|
Path: r.URL.Path,
|
|
Headers: cloneHeader(r.Header),
|
|
Query: cloneValues(r.URL.Query()),
|
|
Body: bytes.Clone(body),
|
|
})
|
|
if errHandle != nil {
|
|
log.Warnf("pluginhost: management handler %s failed: %v", record.pluginID, errHandle)
|
|
http.Error(w, "plugin management handler failed", http.StatusBadGateway)
|
|
return true
|
|
}
|
|
|
|
for keyHeader, values := range resp.Headers {
|
|
for _, value := range values {
|
|
w.Header().Add(keyHeader, value)
|
|
}
|
|
}
|
|
statusCode := resp.StatusCode
|
|
if statusCode == 0 {
|
|
statusCode = http.StatusOK
|
|
}
|
|
w.WriteHeader(statusCode)
|
|
if _, errWrite := w.Write(resp.Body); errWrite != nil {
|
|
log.Warnf("pluginhost: failed to write plugin management response: %v", errWrite)
|
|
}
|
|
return true
|
|
}
|
|
|
|
// ServeResourceHTTP dispatches an unauthenticated browser-navigable resource request to a plugin route.
|
|
func (h *Host) ServeResourceHTTP(w http.ResponseWriter, r *http.Request) bool {
|
|
if h == nil || w == nil || r == nil || r.URL == nil {
|
|
return false
|
|
}
|
|
if !strings.EqualFold(r.Method, http.MethodGet) {
|
|
return false
|
|
}
|
|
key := managementRouteKey(http.MethodGet, r.URL.Path)
|
|
h.mu.Lock()
|
|
record, okRoute := h.resourceRoutes[key]
|
|
h.mu.Unlock()
|
|
if !okRoute || record.route.Handler == nil || h.isPluginFused(record.pluginID) {
|
|
return false
|
|
}
|
|
|
|
resp, errHandle := h.callResourceHandler(r.Context(), record, pluginapi.ManagementRequest{
|
|
Method: http.MethodGet,
|
|
Path: r.URL.Path,
|
|
Headers: cloneHeader(r.Header),
|
|
Query: cloneValues(r.URL.Query()),
|
|
})
|
|
if errHandle != nil {
|
|
log.Warnf("pluginhost: resource handler %s failed: %v", record.pluginID, errHandle)
|
|
http.Error(w, "plugin resource handler failed", http.StatusBadGateway)
|
|
return true
|
|
}
|
|
|
|
for keyHeader, values := range resp.Headers {
|
|
for _, value := range values {
|
|
w.Header().Add(keyHeader, value)
|
|
}
|
|
}
|
|
statusCode := resp.StatusCode
|
|
if statusCode == 0 {
|
|
statusCode = http.StatusOK
|
|
}
|
|
w.WriteHeader(statusCode)
|
|
if _, errWrite := w.Write(resp.Body); errWrite != nil {
|
|
log.Warnf("pluginhost: failed to write plugin resource response: %v", errWrite)
|
|
}
|
|
return true
|
|
}
|
|
|
|
func (h *Host) callManagementHandler(ctx context.Context, record managementRouteRecord, req pluginapi.ManagementRequest) (resp pluginapi.ManagementResponse, err error) {
|
|
if h == nil || record.route.Handler == nil || h.isPluginFused(record.pluginID) {
|
|
return pluginapi.ManagementResponse{}, nil
|
|
}
|
|
defer func() {
|
|
if recovered := recover(); recovered != nil {
|
|
h.fusePlugin(record.pluginID, "ManagementHandler.HandleManagement", recovered)
|
|
resp = pluginapi.ManagementResponse{}
|
|
err = fmt.Errorf("management handler panic: %v", recovered)
|
|
}
|
|
}()
|
|
return record.route.Handler.HandleManagement(ctx, req)
|
|
}
|
|
|
|
func (h *Host) callResourceHandler(ctx context.Context, record resourceRouteRecord, req pluginapi.ManagementRequest) (resp pluginapi.ManagementResponse, err error) {
|
|
if h == nil || record.route.Handler == nil || h.isPluginFused(record.pluginID) {
|
|
return pluginapi.ManagementResponse{}, nil
|
|
}
|
|
defer func() {
|
|
if recovered := recover(); recovered != nil {
|
|
h.fusePlugin(record.pluginID, "ResourceHandler.HandleManagement", recovered)
|
|
resp = pluginapi.ManagementResponse{}
|
|
err = fmt.Errorf("resource handler panic: %v", recovered)
|
|
}
|
|
}()
|
|
return record.route.Handler.HandleManagement(ctx, req)
|
|
}
|