feat: §10 온도 역전 판정 (TempProfileJudge)
- TempProfileJudge(순수함수): 조성 트레이 A,B,C 단조검증 → 정상/약화/프로파일붕괴/온도역전/입력부족 (tolInv 0.5, warn 0.5, collapse 0.3)
- 엔진 JudgeTempProfile: temp_tags 마지막(탑정 D=환류 서브쿨 오염) 제외, spanRef 최초정상 시드, 역전/붕괴 시 front trim HOLD
- AdvisoryResult.{TempProfileState,InversionPair,TempSpan,TempSpanRef} + 컨트롤러 노출
- 단위 6건(37/37). 후속: ApplyRecovery sigInv 연동 + 코러보레이션(센서vs공정)
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
This commit is contained in:
63
tests/ExperionCrawler.Tests/TempProfileJudgeTests.cs
Normal file
63
tests/ExperionCrawler.Tests/TempProfileJudgeTests.cs
Normal file
@@ -0,0 +1,63 @@
|
||||
using System.Collections.Generic;
|
||||
using ExperionCrawler.Infrastructure.Control;
|
||||
using Xunit;
|
||||
|
||||
namespace ExperionCrawler.Tests;
|
||||
|
||||
// §10.3 온도 역전 판정. trays = 하단→상단 조성 트레이(A,B,C), 정상 A≥B≥C.
|
||||
public class TempProfileJudgeTests
|
||||
{
|
||||
private static List<(string, double, bool)> T(double a, double b, double c, bool good = true)
|
||||
=> new() { ("A", a, good), ("B", b, good), ("C", c, good) };
|
||||
|
||||
const double Tol = 0.5, Warn = 0.5, Coll = 0.3;
|
||||
|
||||
[Fact]
|
||||
public void Normal_monotonic_seeds_span()
|
||||
{
|
||||
var r = TempProfileJudge.Evaluate(T(90, 85, 82), spanRef: double.NaN, Tol, Warn, Coll);
|
||||
Assert.Equal("정상", r.State);
|
||||
Assert.Equal(8, r.Span, 6); // A−C = 90−82
|
||||
Assert.Null(r.InversionPair);
|
||||
}
|
||||
|
||||
[Fact]
|
||||
public void Inversion_when_upper_hotter()
|
||||
{
|
||||
// C(상단)가 B(하단)보다 tolInv 이상 뜨거움 → 역전
|
||||
var r = TempProfileJudge.Evaluate(T(90, 80, 85), spanRef: 8, Tol, Warn, Coll);
|
||||
Assert.Equal("온도역전", r.State);
|
||||
Assert.Equal("B-C", r.InversionPair);
|
||||
}
|
||||
|
||||
[Fact]
|
||||
public void Small_nonmonotonic_within_tol_is_not_inversion()
|
||||
{
|
||||
// B−C = -0.2 (데모 노이즈 수준) → tolInv 0.5 내라 역전 아님
|
||||
var r = TempProfileJudge.Evaluate(T(79.2, 78.47, 78.69), spanRef: double.NaN, Tol, Warn, Coll);
|
||||
Assert.NotEqual("온도역전", r.State);
|
||||
}
|
||||
|
||||
[Fact]
|
||||
public void Collapse_when_span_below_fraction()
|
||||
{
|
||||
// spanRef=10, span = 90−88 = 2 < 0.3*10=3 → 붕괴
|
||||
var r = TempProfileJudge.Evaluate(T(90, 89, 88), spanRef: 10, Tol, Warn, Coll);
|
||||
Assert.Equal("프로파일붕괴", r.State);
|
||||
}
|
||||
|
||||
[Fact]
|
||||
public void Weakening_between_warn_and_collapse()
|
||||
{
|
||||
// spanRef=10, span = 90−86 = 4 (3<4<5) → 약화
|
||||
var r = TempProfileJudge.Evaluate(T(90, 88, 86), spanRef: 10, Tol, Warn, Coll);
|
||||
Assert.Equal("약화", r.State);
|
||||
}
|
||||
|
||||
[Fact]
|
||||
public void Bad_tray_is_input_insufficient()
|
||||
{
|
||||
var r = TempProfileJudge.Evaluate(T(90, 85, 82, good: false), spanRef: 8, Tol, Warn, Coll);
|
||||
Assert.Equal("입력부족", r.State);
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user