namespace RpgRoller.Tests; public sealed class ServiceRollHelperTests { private sealed class FixedDiceRoller(IEnumerable values) : IDiceRoller { public int Roll(int sides) { var next = m_Values.Count > 0 ? m_Values.Dequeue() : 1; return Math.Clamp(next, 1, sides); } private readonly Queue m_Values = new(values); } [Fact] public void RollBreakdownFormatter_FormatsStandardAndRolemasterOpenEndedBreakdowns() { Assert.Equal("4+5+2=11", RollBreakdownFormatter.BuildBreakdown([4, 5], 2, 11)); Assert.Equal("0=0", RollBreakdownFormatter.BuildBreakdown([], 0, 0)); Assert.Equal("97+96+45+85=323", RollBreakdownFormatter.BuildRolemasterOpenEndedBreakdown(97, [96, 45], false, 85, 323)); Assert.Equal("(05) -97 -100 -12 +85 = -124", RollBreakdownFormatter.BuildRolemasterOpenEndedBreakdown(5, [97, 100, 12], true, 85, -124)); Assert.Equal("68+10=78; retry(+5): 42+10=52; final=57", RollBreakdownFormatter.BuildRolemasterRetryBreakdown("68+10=78", 5, "42+10=52", 57)); Assert.Equal("05", RollBreakdownFormatter.FormatRolemasterTriggerRoll(5)); } [Fact] public void CampaignLogSummaryBuilder_ExtractsExpressionsAndBuildsBadgesAndSummaries() { var d6Dice = new[] { new RollDieResult(6, true, false, true, false, false), new RollDieResult(1, false, true, true, false, false) }; var rolemasterDice = new[] { new RollDieResult(5, false, false, false, false, false, 1, RollDieKinds.RolemasterOpenEndedInitial), new RollDieResult(97, false, false, false, false, false, 2, RollDieKinds.RolemasterOpenEndedLowSubtract, -97), new RollDieResult(100, false, false, false, false, false, 3, RollDieKinds.RolemasterOpenEndedLowSubtract, -100) }; var retryDice = new[] { new RollDieResult(68, false, false, false, false, false, 1, RollDieKinds.RolemasterOpenEndedInitial, 68, 1), new RollDieResult(42, false, false, false, false, false, 1, RollDieKinds.RolemasterOpenEndedInitial, 42, 2) }; const string retryBreakdown = "68+10=78; retry(+5): 42+10=52; final=57"; Assert.Equal("1d20+5", CampaignLogSummaryBuilder.ExtractCustomRollExpression("1d20+5 => 20+5=25", " => ")); Assert.Null(CampaignLogSummaryBuilder.ExtractCustomRollExpression("20+5=25", " => ")); Assert.Equal("6, 1", CampaignLogSummaryBuilder.BuildCompactLogSummary(d6Dice)); Assert.Equal("(05) -97 -100 | open-ended low", CampaignLogSummaryBuilder.BuildCompactLogSummary(rolemasterDice)); Assert.Equal("68 | open-ended | retry +5", CampaignLogSummaryBuilder.BuildCompactLogSummary(retryDice, retryBreakdown)); Assert.Equal("No detail available.", CampaignLogSummaryBuilder.BuildCompactLogSummary([])); Assert.Equal(["w6", "w1"], Assert.IsType(CampaignLogSummaryBuilder.BuildCompactLogEventBadges(RulesetKind.D6, null, d6Dice))); Assert.Equal(["n20"], Assert.IsType(CampaignLogSummaryBuilder.BuildCompactLogEventBadges(RulesetKind.Dnd5e, "1d20+5", [new(20, false, false, false, false, false)]))); Assert.Equal(["rf", "r100"], Assert.IsType(CampaignLogSummaryBuilder.BuildCompactLogEventBadges(RulesetKind.Rolemaster, "d100!+85", rolemasterDice))); Assert.Equal(["rs5"], Assert.IsType(CampaignLogSummaryBuilder.BuildCompactLogEventBadges(RulesetKind.Rolemaster, "d100!+10", retryDice, retryBreakdown))); } [Fact] public void RollEngine_DelegatesToRulesetSpecificEngines() { var engine = new RollEngine(new(new FixedDiceRoller([7, 10])), new(new FixedDiceRoller([6, 4, 2])), new(new FixedDiceRoller([97, 96, 45]))); var d6Roll = engine.Roll(RulesetKind.D6, new(2, 6, 1, "2D+1"), 1, true, null); Assert.Equal(13, d6Roll.Total); Assert.Equal("6+4+2+1=13", d6Roll.Breakdown); var standardRoll = engine.Roll(RulesetKind.Dnd5e, new(2, 10, 3, "2d10+3"), 0, false, null); Assert.Equal(20, standardRoll.Total); Assert.Equal("7+10+3=20", standardRoll.Breakdown); var rolemasterRoll = engine.Roll(RulesetKind.Rolemaster, new(1, 100, 85, "d100!+85", DiceExpressionKind.RolemasterOpenEndedPercentile), 0, false, 5); Assert.Equal(323, rolemasterRoll.Total); Assert.Equal("97+96+45+85=323", rolemasterRoll.Breakdown); } }