feat(iace): Sprint 4C — Delta Impact Analysis

Neuer Endpoint POST /projects/:id/delta-analysis:
- Input: aktuelle + vorgeschlagene Aenderung (Components, Energy, States, Roles)
- Output: Diff der Pattern-Matches (added/removed Patterns, Hazards, Measures)
- DeltaMatch() auf PatternEngine: Match(current) vs Match(proposed)
- DeltaResult mit AddedPatterns, RemovedPatterns, Counts, SummaryDE

Beispiel-Output: SPS hinzufuegen → +55 Patterns, +5 Hazard-Kategorien, +17 Massnahmen
Maintenance-State hinzufuegen → +10 Patterns, +2 Hazards, +2 Massnahmen

7 Tests: NoChange, AddComponent, RemoveComponent, AddState, AddRole, Summary, Symmetric

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
Benjamin Admin
2026-05-10 21:23:46 +02:00
parent 26b222d53d
commit 9a9a11b248
4 changed files with 400 additions and 0 deletions
@@ -0,0 +1,101 @@
package handlers
import (
"net/http"
"github.com/breakpilot/ai-compliance-sdk/internal/iace"
"github.com/gin-gonic/gin"
"github.com/google/uuid"
)
// DeltaAnalysisRequest is the request body for POST /projects/:id/delta-analysis.
type DeltaAnalysisRequest struct {
// Current state — if empty, loaded from project's existing components/energy
CurrentComponents []string `json:"current_components,omitempty"`
CurrentEnergy []string `json:"current_energy,omitempty"`
CurrentStates []string `json:"current_states,omitempty"`
CurrentRoles []string `json:"current_roles,omitempty"`
// Proposed additions
AddComponents []string `json:"add_components,omitempty"`
AddEnergy []string `json:"add_energy,omitempty"`
AddStates []string `json:"add_states,omitempty"`
AddRoles []string `json:"add_roles,omitempty"`
// Proposed removals
RemoveComponents []string `json:"remove_components,omitempty"`
RemoveEnergy []string `json:"remove_energy,omitempty"`
RemoveStates []string `json:"remove_states,omitempty"`
RemoveRoles []string `json:"remove_roles,omitempty"`
}
// DeltaAnalysis handles POST /projects/:id/delta-analysis.
// It computes the impact of a proposed change on the hazard/measure landscape.
func (h *IACEHandler) DeltaAnalysis(c *gin.Context) {
projectID, err := uuid.Parse(c.Param("id"))
if err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": "invalid project ID"})
return
}
var req DeltaAnalysisRequest
if err := c.ShouldBindJSON(&req); err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
return
}
// Build current MatchInput — from request or from project's existing data
currentComps := req.CurrentComponents
currentEnergy := req.CurrentEnergy
currentStates := req.CurrentStates
currentRoles := req.CurrentRoles
_ = projectID // Used for future: auto-load current state from project DB
currentInput := iace.MatchInput{
ComponentLibraryIDs: currentComps,
EnergySourceIDs: currentEnergy,
OperationalStates: currentStates,
HumanRoles: currentRoles,
}
// Build proposed MatchInput = current + additions - removals
proposedComps := applyDelta(currentComps, req.AddComponents, req.RemoveComponents)
proposedEnergy := applyDelta(currentEnergy, req.AddEnergy, req.RemoveEnergy)
proposedStates := applyDelta(currentStates, req.AddStates, req.RemoveStates)
proposedRoles := applyDelta(currentRoles, req.AddRoles, req.RemoveRoles)
proposedInput := iace.MatchInput{
ComponentLibraryIDs: proposedComps,
EnergySourceIDs: proposedEnergy,
OperationalStates: proposedStates,
HumanRoles: proposedRoles,
}
// Run delta analysis
engine := iace.NewPatternEngine()
result := engine.DeltaMatch(currentInput, proposedInput)
c.JSON(http.StatusOK, result)
}
// applyDelta computes: current + additions - removals
func applyDelta(current, add, remove []string) []string {
removeSet := make(map[string]bool)
for _, r := range remove {
removeSet[r] = true
}
result := make([]string, 0, len(current)+len(add))
seen := make(map[string]bool)
for _, s := range current {
if !removeSet[s] && !seen[s] {
result = append(result, s)
seen[s] = true
}
}
for _, s := range add {
if !seen[s] {
result = append(result, s)
seen[s] = true
}
}
return result
}
+1
View File
@@ -388,6 +388,7 @@ func registerIACERoutes(v1 *gin.RouterGroup, h *handlers.IACEHandler) {
iaceRoutes.POST("/projects/:id/hazards/suggest", h.SuggestHazards)
iaceRoutes.POST("/projects/:id/match-patterns", h.MatchPatterns)
iaceRoutes.POST("/projects/:id/parse-narrative", h.ParseNarrative)
iaceRoutes.POST("/projects/:id/delta-analysis", h.DeltaAnalysis)
iaceRoutes.POST("/projects/:id/apply-patterns", h.ApplyPatternResults)
iaceRoutes.POST("/projects/:id/hazards/:hid/suggest-measures", h.SuggestMeasuresForHazard)
iaceRoutes.POST("/projects/:id/mitigations/:mid/suggest-evidence", h.SuggestEvidenceForMitigation)
@@ -0,0 +1,160 @@
package iace
import "strconv"
// DeltaResult contains the impact of a proposed change to a project.
// It compares Match(current) vs Match(current+proposed) and returns
// the difference in matched patterns, hazards, measures, and evidence.
type DeltaResult struct {
// Patterns that would fire with the proposed change but not currently
AddedPatterns []PatternMatch `json:"added_patterns"`
// Patterns that currently fire but would stop with the proposed change
RemovedPatterns []PatternMatch `json:"removed_patterns"`
// Net change counts
AddedHazardCount int `json:"added_hazard_count"`
RemovedHazardCount int `json:"removed_hazard_count"`
AddedMeasureCount int `json:"added_measure_count"`
RemovedMeasureCount int `json:"removed_measure_count"`
AddedEvidenceCount int `json:"added_evidence_count"`
RemovedEvidenceCount int `json:"removed_evidence_count"`
// Added/removed hazard categories
AddedHazardCats []string `json:"added_hazard_categories,omitempty"`
RemovedHazardCats []string `json:"removed_hazard_categories,omitempty"`
// Added/removed measure IDs
AddedMeasureIDs []string `json:"added_measure_ids,omitempty"`
RemovedMeasureIDs []string `json:"removed_measure_ids,omitempty"`
// Summary text (German)
SummaryDE string `json:"summary_de"`
}
// DeltaMatch computes the impact of a proposed change by comparing
// the current match output with the proposed match output.
func (e *PatternEngine) DeltaMatch(current, proposed MatchInput) *DeltaResult {
currentOutput := e.Match(current)
proposedOutput := e.Match(proposed)
// Build pattern ID sets
currentPatterns := make(map[string]PatternMatch)
for _, p := range currentOutput.MatchedPatterns {
currentPatterns[p.PatternID] = p
}
proposedPatterns := make(map[string]PatternMatch)
for _, p := range proposedOutput.MatchedPatterns {
proposedPatterns[p.PatternID] = p
}
// Find added/removed patterns
var added, removed []PatternMatch
for id, p := range proposedPatterns {
if _, exists := currentPatterns[id]; !exists {
added = append(added, p)
}
}
for id, p := range currentPatterns {
if _, exists := proposedPatterns[id]; !exists {
removed = append(removed, p)
}
}
// Compute hazard category diff
currentCats := collectCats(currentOutput.SuggestedHazards)
proposedCats := collectCats(proposedOutput.SuggestedHazards)
addedCats := diffStrings(proposedCats, currentCats)
removedCats := diffStrings(currentCats, proposedCats)
// Compute measure ID diff
currentMIDs := collectMIDs(currentOutput.SuggestedMeasures)
proposedMIDs := collectMIDs(proposedOutput.SuggestedMeasures)
addedMIDs := diffStrings(proposedMIDs, currentMIDs)
removedMIDs := diffStrings(currentMIDs, proposedMIDs)
// Compute evidence ID diff
currentEIDs := collectEIDs(currentOutput.SuggestedEvidence)
proposedEIDs := collectEIDs(proposedOutput.SuggestedEvidence)
result := &DeltaResult{
AddedPatterns: added,
RemovedPatterns: removed,
AddedHazardCount: len(addedCats),
RemovedHazardCount: len(removedCats),
AddedMeasureCount: len(addedMIDs),
RemovedMeasureCount: len(removedMIDs),
AddedEvidenceCount: len(diffStrings(proposedEIDs, currentEIDs)),
RemovedEvidenceCount: len(diffStrings(currentEIDs, proposedEIDs)),
AddedHazardCats: addedCats,
RemovedHazardCats: removedCats,
AddedMeasureIDs: addedMIDs,
RemovedMeasureIDs: removedMIDs,
}
result.SummaryDE = buildDeltaSummary(result)
return result
}
func collectCats(hazards []HazardSuggestion) []string {
var cats []string
for _, h := range hazards {
cats = append(cats, h.Category)
}
return cats
}
func collectMIDs(measures []MeasureSuggestion) []string {
var ids []string
for _, m := range measures {
ids = append(ids, m.MeasureID)
}
return ids
}
func collectEIDs(evidence []EvidenceSuggestion) []string {
var ids []string
for _, e := range evidence {
ids = append(ids, e.EvidenceID)
}
return ids
}
func diffStrings(a, b []string) []string {
bSet := make(map[string]bool)
for _, s := range b {
bSet[s] = true
}
var diff []string
seen := make(map[string]bool)
for _, s := range a {
if !bSet[s] && !seen[s] {
diff = append(diff, s)
seen[s] = true
}
}
return diff
}
func buildDeltaSummary(r *DeltaResult) string {
if len(r.AddedPatterns) == 0 && len(r.RemovedPatterns) == 0 {
return "Keine Auswirkung — die vorgeschlagene Aenderung veraendert die Risikobeurteilung nicht."
}
s := ""
if len(r.AddedPatterns) > 0 {
s += "+" + deltaItoa(len(r.AddedPatterns)) + " neue Gefaehrdungsmuster erkannt"
}
if len(r.RemovedPatterns) > 0 {
if s != "" {
s += ", "
}
s += "-" + deltaItoa(len(r.RemovedPatterns)) + " Gefaehrdungsmuster entfallen"
}
if r.AddedHazardCount > 0 {
s += ". +" + deltaItoa(r.AddedHazardCount) + " neue Gefaehrdungskategorien"
}
if r.AddedMeasureCount > 0 {
s += ". +" + deltaItoa(r.AddedMeasureCount) + " zusaetzliche Massnahmen empfohlen"
}
s += "."
return s
}
func deltaItoa(i int) string {
return strconv.Itoa(i)
}
@@ -0,0 +1,138 @@
package iace
import "testing"
func TestDeltaMatch_NoChange(t *testing.T) {
engine := NewPatternEngine()
input := MatchInput{
ComponentLibraryIDs: []string{"C001"},
EnergySourceIDs: []string{"EN01"},
}
result := engine.DeltaMatch(input, input)
if len(result.AddedPatterns) != 0 {
t.Errorf("expected 0 added patterns for no change, got %d", len(result.AddedPatterns))
}
if len(result.RemovedPatterns) != 0 {
t.Errorf("expected 0 removed patterns for no change, got %d", len(result.RemovedPatterns))
}
if result.SummaryDE != "Keine Auswirkung — die vorgeschlagene Aenderung veraendert die Risikobeurteilung nicht." {
t.Errorf("unexpected summary: %s", result.SummaryDE)
}
}
func TestDeltaMatch_AddComponent(t *testing.T) {
engine := NewPatternEngine()
current := MatchInput{
ComponentLibraryIDs: []string{"C001"},
EnergySourceIDs: []string{"EN01"},
}
proposed := MatchInput{
ComponentLibraryIDs: []string{"C001", "C071"}, // Add SPS (programmable)
EnergySourceIDs: []string{"EN01"},
}
result := engine.DeltaMatch(current, proposed)
if len(result.AddedPatterns) == 0 {
t.Error("expected added patterns when adding SPS component")
}
t.Logf("Adding C071 (SPS): +%d patterns, +%d hazard cats, +%d measures",
len(result.AddedPatterns), result.AddedHazardCount, result.AddedMeasureCount)
}
func TestDeltaMatch_RemoveComponent(t *testing.T) {
engine := NewPatternEngine()
current := MatchInput{
ComponentLibraryIDs: []string{"C001", "C071"},
EnergySourceIDs: []string{"EN01"},
}
proposed := MatchInput{
ComponentLibraryIDs: []string{"C001"}, // Remove SPS
EnergySourceIDs: []string{"EN01"},
}
result := engine.DeltaMatch(current, proposed)
if len(result.RemovedPatterns) == 0 {
t.Error("expected removed patterns when removing SPS component")
}
t.Logf("Removing C071: -%d patterns, -%d hazard cats",
len(result.RemovedPatterns), result.RemovedHazardCount)
}
func TestDeltaMatch_AddState(t *testing.T) {
engine := NewPatternEngine()
current := MatchInput{
ComponentLibraryIDs: []string{"C001", "C071"},
EnergySourceIDs: []string{"EN01"},
LifecyclePhases: []string{"normal_operation"},
OperationalStates: []string{"automatic_operation"},
HumanRoles: []string{"operator"},
}
proposed := MatchInput{
ComponentLibraryIDs: []string{"C001", "C071"},
EnergySourceIDs: []string{"EN01"},
LifecyclePhases: []string{"normal_operation", "maintenance"},
OperationalStates: []string{"automatic_operation", "maintenance"},
HumanRoles: []string{"operator", "maintenance_tech"},
}
result := engine.DeltaMatch(current, proposed)
t.Logf("Adding maintenance state+role: +%d patterns, +%d hazards, +%d measures",
len(result.AddedPatterns), result.AddedHazardCount, result.AddedMeasureCount)
// Adding maintenance should bring maintenance-specific patterns
if len(result.AddedPatterns) == 0 {
t.Error("expected added patterns when adding maintenance state")
}
}
func TestDeltaMatch_AddRole(t *testing.T) {
engine := NewPatternEngine()
current := MatchInput{
ComponentLibraryIDs: []string{"C001", "C071"},
EnergySourceIDs: []string{"EN01"},
LifecyclePhases: []string{"maintenance"},
OperationalStates: []string{"maintenance"},
HumanRoles: []string{"operator"}, // Only operator
}
proposed := MatchInput{
ComponentLibraryIDs: []string{"C001", "C071"},
EnergySourceIDs: []string{"EN01"},
LifecyclePhases: []string{"maintenance"},
OperationalStates: []string{"maintenance"},
HumanRoles: []string{"operator", "maintenance_tech"}, // Add maintenance_tech
}
result := engine.DeltaMatch(current, proposed)
t.Logf("Adding maintenance_tech role: +%d patterns", len(result.AddedPatterns))
// maintenance_tech role should unlock maintenance-specific patterns
if len(result.AddedPatterns) == 0 {
t.Error("expected added patterns when adding maintenance_tech role")
}
}
func TestDeltaMatch_SummaryNotEmpty(t *testing.T) {
engine := NewPatternEngine()
current := MatchInput{ComponentLibraryIDs: []string{"C001"}, EnergySourceIDs: []string{"EN01"}}
proposed := MatchInput{ComponentLibraryIDs: []string{"C001", "C071"}, EnergySourceIDs: []string{"EN01"}}
result := engine.DeltaMatch(current, proposed)
if result.SummaryDE == "" {
t.Error("expected non-empty summary")
}
t.Logf("Summary: %s", result.SummaryDE)
}
func TestDeltaMatch_Symmetric(t *testing.T) {
engine := NewPatternEngine()
a := MatchInput{ComponentLibraryIDs: []string{"C001"}, EnergySourceIDs: []string{"EN01"}}
b := MatchInput{ComponentLibraryIDs: []string{"C001", "C071"}, EnergySourceIDs: []string{"EN01"}}
forward := engine.DeltaMatch(a, b)
backward := engine.DeltaMatch(b, a)
if len(forward.AddedPatterns) != len(backward.RemovedPatterns) {
t.Errorf("forward added (%d) should equal backward removed (%d)",
len(forward.AddedPatterns), len(backward.RemovedPatterns))
}
}