315 lines
10 KiB
C#
315 lines
10 KiB
C#
using ZB.MOM.WW.ScadaBridge.Commons.Entities.Templates;
|
|
using ZB.MOM.WW.ScadaBridge.Commons.Types.Enums;
|
|
|
|
namespace ZB.MOM.WW.ScadaBridge.TemplateEngine.Tests;
|
|
|
|
public class LockEnforcerTests
|
|
{
|
|
// ========================================================================
|
|
// WP-8: Override Granularity
|
|
// ========================================================================
|
|
|
|
[Fact]
|
|
public void ValidateAttributeOverride_LockedAttribute_ReturnsError()
|
|
{
|
|
var original = new TemplateAttribute("Speed")
|
|
{
|
|
DataType = DataType.Float, IsLocked = true, Value = "0"
|
|
};
|
|
var proposed = new TemplateAttribute("Speed")
|
|
{
|
|
DataType = DataType.Float, IsLocked = true, Value = "100"
|
|
};
|
|
|
|
var result = LockEnforcer.ValidateAttributeOverride(original, proposed);
|
|
|
|
Assert.NotNull(result);
|
|
Assert.Contains("locked", result);
|
|
}
|
|
|
|
[Fact]
|
|
public void ValidateAttributeOverride_DataTypeChanged_ReturnsError()
|
|
{
|
|
var original = new TemplateAttribute("Speed")
|
|
{
|
|
DataType = DataType.Float, IsLocked = false
|
|
};
|
|
var proposed = new TemplateAttribute("Speed")
|
|
{
|
|
DataType = DataType.String, IsLocked = false // DataType changed!
|
|
};
|
|
|
|
var result = LockEnforcer.ValidateAttributeOverride(original, proposed);
|
|
|
|
Assert.NotNull(result);
|
|
Assert.Contains("DataType", result);
|
|
}
|
|
|
|
[Fact]
|
|
public void ValidateAttributeOverride_DataSourceReferenceChanged_ReturnsError()
|
|
{
|
|
var original = new TemplateAttribute("Speed")
|
|
{
|
|
DataType = DataType.Float, IsLocked = false, DataSourceReference = "tag1"
|
|
};
|
|
var proposed = new TemplateAttribute("Speed")
|
|
{
|
|
DataType = DataType.Float, IsLocked = false, DataSourceReference = "tag2" // Changed!
|
|
};
|
|
|
|
var result = LockEnforcer.ValidateAttributeOverride(original, proposed);
|
|
|
|
Assert.NotNull(result);
|
|
Assert.Contains("DataSourceReference", result);
|
|
}
|
|
|
|
[Fact]
|
|
public void ValidateAttributeOverride_ValueAndDescriptionChanged_ReturnsNull()
|
|
{
|
|
var original = new TemplateAttribute("Speed")
|
|
{
|
|
DataType = DataType.Float, IsLocked = false, Value = "0", Description = "old"
|
|
};
|
|
var proposed = new TemplateAttribute("Speed")
|
|
{
|
|
DataType = DataType.Float, IsLocked = false, Value = "100", Description = "new"
|
|
};
|
|
|
|
var result = LockEnforcer.ValidateAttributeOverride(original, proposed);
|
|
|
|
Assert.Null(result);
|
|
}
|
|
|
|
[Fact]
|
|
public void ValidateAttributeOverride_ElementDataTypeChanged_ReturnsError()
|
|
{
|
|
// MV-10 review fix: ElementDataType is the element scalar type of a List
|
|
// attribute and is fixed by the defining level, exactly like DataType.
|
|
// TemplateService.UpdateAttributeAsync never copies it onto the persisted
|
|
// row, so a mismatch must be rejected before the Value (validated against
|
|
// the real element type) is persisted against the wrong type.
|
|
var original = new TemplateAttribute("Tags")
|
|
{
|
|
DataType = DataType.List, ElementDataType = DataType.Int32, IsLocked = false
|
|
};
|
|
var proposed = new TemplateAttribute("Tags")
|
|
{
|
|
DataType = DataType.List, ElementDataType = DataType.String, IsLocked = false // changed!
|
|
};
|
|
|
|
var result = LockEnforcer.ValidateAttributeOverride(original, proposed);
|
|
|
|
Assert.NotNull(result);
|
|
Assert.Contains("ElementDataType", result);
|
|
}
|
|
|
|
[Fact]
|
|
public void ValidateAttributeOverride_ElementDataTypeMatches_ReturnsNull()
|
|
{
|
|
var original = new TemplateAttribute("Tags")
|
|
{
|
|
DataType = DataType.List, ElementDataType = DataType.Int32, IsLocked = false, Value = "[1]"
|
|
};
|
|
var proposed = new TemplateAttribute("Tags")
|
|
{
|
|
DataType = DataType.List, ElementDataType = DataType.Int32, IsLocked = false, Value = "[1,2]"
|
|
};
|
|
|
|
var result = LockEnforcer.ValidateAttributeOverride(original, proposed);
|
|
|
|
Assert.Null(result);
|
|
}
|
|
|
|
[Fact]
|
|
public void ValidateAttributeOverride_ElementDataTypeBothNull_ReturnsNull()
|
|
{
|
|
// Scalar attributes carry no element type on either side — not a change.
|
|
var original = new TemplateAttribute("Speed")
|
|
{
|
|
DataType = DataType.Float, ElementDataType = null, IsLocked = false, Value = "0"
|
|
};
|
|
var proposed = new TemplateAttribute("Speed")
|
|
{
|
|
DataType = DataType.Float, ElementDataType = null, IsLocked = false, Value = "100"
|
|
};
|
|
|
|
var result = LockEnforcer.ValidateAttributeOverride(original, proposed);
|
|
|
|
Assert.Null(result);
|
|
}
|
|
|
|
[Fact]
|
|
public void ValidateAlarmOverride_LockedAlarm_ReturnsError()
|
|
{
|
|
var original = new TemplateAlarm("HighTemp")
|
|
{
|
|
TriggerType = AlarmTriggerType.ValueMatch, IsLocked = true, PriorityLevel = 500
|
|
};
|
|
var proposed = new TemplateAlarm("HighTemp")
|
|
{
|
|
TriggerType = AlarmTriggerType.ValueMatch, IsLocked = true, PriorityLevel = 600
|
|
};
|
|
|
|
var result = LockEnforcer.ValidateAlarmOverride(original, proposed);
|
|
|
|
Assert.NotNull(result);
|
|
Assert.Contains("locked", result);
|
|
}
|
|
|
|
[Fact]
|
|
public void ValidateAlarmOverride_TriggerTypeChanged_ReturnsError()
|
|
{
|
|
var original = new TemplateAlarm("HighTemp")
|
|
{
|
|
TriggerType = AlarmTriggerType.ValueMatch, IsLocked = false
|
|
};
|
|
var proposed = new TemplateAlarm("HighTemp")
|
|
{
|
|
TriggerType = AlarmTriggerType.RangeViolation, IsLocked = false // Changed!
|
|
};
|
|
|
|
var result = LockEnforcer.ValidateAlarmOverride(original, proposed);
|
|
|
|
Assert.NotNull(result);
|
|
Assert.Contains("TriggerType", result);
|
|
}
|
|
|
|
[Fact]
|
|
public void ValidateAlarmOverride_OverridableFieldsChanged_ReturnsNull()
|
|
{
|
|
var original = new TemplateAlarm("HighTemp")
|
|
{
|
|
TriggerType = AlarmTriggerType.ValueMatch, IsLocked = false,
|
|
PriorityLevel = 500, Description = "old"
|
|
};
|
|
var proposed = new TemplateAlarm("HighTemp")
|
|
{
|
|
TriggerType = AlarmTriggerType.ValueMatch, IsLocked = false,
|
|
PriorityLevel = 700, Description = "new", TriggerConfiguration = """{"value": 100}"""
|
|
};
|
|
|
|
var result = LockEnforcer.ValidateAlarmOverride(original, proposed);
|
|
|
|
Assert.Null(result);
|
|
}
|
|
|
|
[Fact]
|
|
public void ValidateScriptOverride_LockedScript_ReturnsError()
|
|
{
|
|
var original = new TemplateScript("OnStart", "code") { IsLocked = true };
|
|
var proposed = new TemplateScript("OnStart", "new code") { IsLocked = true };
|
|
|
|
var result = LockEnforcer.ValidateScriptOverride(original, proposed);
|
|
|
|
Assert.NotNull(result);
|
|
Assert.Contains("locked", result);
|
|
}
|
|
|
|
[Fact]
|
|
public void ValidateScriptOverride_NameChanged_ReturnsError()
|
|
{
|
|
var original = new TemplateScript("OnStart", "code") { IsLocked = false };
|
|
var proposed = new TemplateScript("OnStop", "code") { IsLocked = false }; // Name changed!
|
|
|
|
var result = LockEnforcer.ValidateScriptOverride(original, proposed);
|
|
|
|
Assert.NotNull(result);
|
|
Assert.Contains("Name", result);
|
|
}
|
|
|
|
[Fact]
|
|
public void ValidateScriptOverride_OverridableFieldsChanged_ReturnsNull()
|
|
{
|
|
var original = new TemplateScript("OnStart", "old code") { IsLocked = false };
|
|
var proposed = new TemplateScript("OnStart", "new code")
|
|
{
|
|
IsLocked = false,
|
|
TriggerType = "Timer",
|
|
MinTimeBetweenRuns = TimeSpan.FromSeconds(30)
|
|
};
|
|
|
|
var result = LockEnforcer.ValidateScriptOverride(original, proposed);
|
|
|
|
Assert.Null(result);
|
|
}
|
|
|
|
// ========================================================================
|
|
// WP-9: Locking Rules
|
|
// ========================================================================
|
|
|
|
[Fact]
|
|
public void ValidateLockChange_UnlockLocked_ReturnsError()
|
|
{
|
|
var result = LockEnforcer.ValidateLockChange(true, false, "Speed");
|
|
|
|
Assert.NotNull(result);
|
|
Assert.Contains("cannot be unlocked", result);
|
|
}
|
|
|
|
[Fact]
|
|
public void ValidateLockChange_LockUnlocked_ReturnsNull()
|
|
{
|
|
var result = LockEnforcer.ValidateLockChange(false, true, "Speed");
|
|
|
|
Assert.Null(result);
|
|
}
|
|
|
|
[Fact]
|
|
public void ValidateLockChange_KeepLocked_ReturnsNull()
|
|
{
|
|
var result = LockEnforcer.ValidateLockChange(true, true, "Speed");
|
|
|
|
Assert.Null(result);
|
|
}
|
|
|
|
[Fact]
|
|
public void ValidateLockChange_KeepUnlocked_ReturnsNull()
|
|
{
|
|
var result = LockEnforcer.ValidateLockChange(false, false, "Speed");
|
|
|
|
Assert.Null(result);
|
|
}
|
|
|
|
// ========================================================================
|
|
// TemplateEngine-022: LockedInDerived one-way ratchet
|
|
// ========================================================================
|
|
|
|
[Fact]
|
|
public void ValidateLockedInDerivedChange_ClearLocked_ReturnsError()
|
|
{
|
|
// Once a base template marks a member LockedInDerived, the flag may
|
|
// not be cleared — derived overrides previously blocked would
|
|
// otherwise become retroactively legal.
|
|
var result = LockEnforcer.ValidateLockedInDerivedChange(true, false, "Speed");
|
|
|
|
Assert.NotNull(result);
|
|
Assert.Contains("locked-in-derived", result);
|
|
Assert.Contains("cannot be cleared", result);
|
|
}
|
|
|
|
[Fact]
|
|
public void ValidateLockedInDerivedChange_LockUnlocked_ReturnsNull()
|
|
{
|
|
// Setting the flag from false→true is the normal direction.
|
|
var result = LockEnforcer.ValidateLockedInDerivedChange(false, true, "Speed");
|
|
|
|
Assert.Null(result);
|
|
}
|
|
|
|
[Fact]
|
|
public void ValidateLockedInDerivedChange_KeepLocked_ReturnsNull()
|
|
{
|
|
var result = LockEnforcer.ValidateLockedInDerivedChange(true, true, "Speed");
|
|
|
|
Assert.Null(result);
|
|
}
|
|
|
|
[Fact]
|
|
public void ValidateLockedInDerivedChange_KeepUnlocked_ReturnsNull()
|
|
{
|
|
var result = LockEnforcer.ValidateLockedInDerivedChange(false, false, "Speed");
|
|
|
|
Assert.Null(result);
|
|
}
|
|
}
|