mirror of
https://github.com/Syngnat/GoNavi.git
synced 2026-06-07 15:09:34 +08:00
- 诊断 SSE 支持空心跳事件,避免无输出时解码失败 - Arthas Tunnel 增加会话过期清理、配置漂移校验和取消兜底 - Provider 合约清理 Base URL 查询参数和片段,避免探测泄露敏感信息 - JVM 变更请求强制校验原因并规范化写入审计字段
166 lines
4.4 KiB
Go
166 lines
4.4 KiB
Go
package jvm
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"strings"
|
|
|
|
"GoNavi-Wails/internal/connection"
|
|
)
|
|
|
|
// BuildChangePreview builds a guarded preview for JVM mutations.
|
|
// It always produces a local before/after baseline and, when writes are still
|
|
// allowed, merges provider preview details on top of that baseline.
|
|
func BuildChangePreview(
|
|
ctx context.Context,
|
|
provider Provider,
|
|
cfg connection.ConnectionConfig,
|
|
req ChangeRequest,
|
|
) (ChangePreview, error) {
|
|
req, err := NormalizeChangeRequest(req)
|
|
if err != nil {
|
|
return ChangePreview{}, err
|
|
}
|
|
|
|
normalized, err := NormalizeConnectionConfig(cfg)
|
|
if err != nil {
|
|
return ChangePreview{}, err
|
|
}
|
|
|
|
resourceID := req.ResourceID
|
|
action := req.Action
|
|
|
|
before := ValueSnapshot{
|
|
ResourceID: resourceID,
|
|
Kind: "resource",
|
|
Format: "json",
|
|
}
|
|
if provider != nil {
|
|
if snapshot, snapshotErr := provider.GetValue(ctx, normalized, resourceID); snapshotErr == nil {
|
|
before = snapshot
|
|
if strings.TrimSpace(before.ResourceID) == "" {
|
|
before.ResourceID = resourceID
|
|
}
|
|
if strings.TrimSpace(before.Format) == "" {
|
|
before.Format = "json"
|
|
}
|
|
}
|
|
}
|
|
|
|
after := before
|
|
after.ResourceID = resourceID
|
|
if req.ExpectedVersion != "" {
|
|
after.Version = req.ExpectedVersion
|
|
}
|
|
if req.Payload != nil {
|
|
after.Value = req.Payload
|
|
}
|
|
|
|
preview := ChangePreview{
|
|
Allowed: true,
|
|
Summary: fmt.Sprintf("%s -> %s", resourceID, action),
|
|
RiskLevel: "medium",
|
|
Before: before,
|
|
After: after,
|
|
}
|
|
|
|
if normalized.JVM.ReadOnly != nil && *normalized.JVM.ReadOnly {
|
|
preview.Allowed = false
|
|
preview.RiskLevel = "high"
|
|
preview.BlockingReason = "当前连接为只读,禁止写入"
|
|
}
|
|
if normalized.JVM.Environment == EnvPROD {
|
|
preview.RequiresConfirmation = true
|
|
if preview.RiskLevel == "" || preview.RiskLevel == "low" {
|
|
preview.RiskLevel = "medium"
|
|
}
|
|
}
|
|
|
|
if !preview.Allowed || provider == nil {
|
|
return preview, nil
|
|
}
|
|
|
|
providerPreview, err := provider.PreviewChange(ctx, normalized, req)
|
|
if err != nil {
|
|
return ChangePreview{}, err
|
|
}
|
|
|
|
if strings.TrimSpace(providerPreview.Summary) != "" {
|
|
preview.Summary = providerPreview.Summary
|
|
}
|
|
if strings.TrimSpace(providerPreview.RiskLevel) != "" {
|
|
preview.RiskLevel = providerPreview.RiskLevel
|
|
}
|
|
if providerPreview.RequiresConfirmation {
|
|
preview.RequiresConfirmation = true
|
|
}
|
|
if !providerPreview.Allowed {
|
|
preview.Allowed = false
|
|
}
|
|
if strings.TrimSpace(providerPreview.BlockingReason) != "" {
|
|
preview.BlockingReason = providerPreview.BlockingReason
|
|
}
|
|
if hasSnapshotOverride(providerPreview.Before) {
|
|
preview.Before = mergeValueSnapshot(preview.Before, providerPreview.Before)
|
|
}
|
|
if hasSnapshotOverride(providerPreview.After) {
|
|
preview.After = mergeValueSnapshot(preview.After, providerPreview.After)
|
|
}
|
|
|
|
return preview, nil
|
|
}
|
|
|
|
func NormalizeChangeRequest(req ChangeRequest) (ChangeRequest, error) {
|
|
normalized := req
|
|
normalized.ProviderMode = strings.ToLower(strings.TrimSpace(normalized.ProviderMode))
|
|
normalized.ResourceID = strings.TrimSpace(normalized.ResourceID)
|
|
normalized.Action = strings.TrimSpace(normalized.Action)
|
|
normalized.Reason = strings.TrimSpace(normalized.Reason)
|
|
normalized.Source = strings.TrimSpace(normalized.Source)
|
|
normalized.ExpectedVersion = strings.TrimSpace(normalized.ExpectedVersion)
|
|
|
|
if normalized.ResourceID == "" {
|
|
return ChangeRequest{}, fmt.Errorf("resource id is required")
|
|
}
|
|
if normalized.Action == "" {
|
|
return ChangeRequest{}, fmt.Errorf("action is required")
|
|
}
|
|
if normalized.Reason == "" {
|
|
return ChangeRequest{}, fmt.Errorf("reason is required")
|
|
}
|
|
|
|
return normalized, nil
|
|
}
|
|
|
|
func hasSnapshotOverride(snapshot ValueSnapshot) bool {
|
|
return strings.TrimSpace(snapshot.ResourceID) != "" ||
|
|
strings.TrimSpace(snapshot.Kind) != "" ||
|
|
strings.TrimSpace(snapshot.Format) != "" ||
|
|
strings.TrimSpace(snapshot.Version) != "" ||
|
|
snapshot.Value != nil ||
|
|
snapshot.Metadata != nil
|
|
}
|
|
|
|
func mergeValueSnapshot(base ValueSnapshot, override ValueSnapshot) ValueSnapshot {
|
|
merged := base
|
|
if strings.TrimSpace(override.ResourceID) != "" {
|
|
merged.ResourceID = override.ResourceID
|
|
}
|
|
if strings.TrimSpace(override.Kind) != "" {
|
|
merged.Kind = override.Kind
|
|
}
|
|
if strings.TrimSpace(override.Format) != "" {
|
|
merged.Format = override.Format
|
|
}
|
|
if strings.TrimSpace(override.Version) != "" {
|
|
merged.Version = override.Version
|
|
}
|
|
if override.Value != nil {
|
|
merged.Value = override.Value
|
|
}
|
|
if override.Metadata != nil {
|
|
merged.Metadata = override.Metadata
|
|
}
|
|
return merged
|
|
}
|