feat(packs): rewrite PackOpenService against per-pack draw table
Sampler is now driven by PackDrawTable: roll DrawTier per slot by cumulative slot-rate weights, then pick a card within tier by per-card weights renormalized within the tier. Rate-less Guaranteed-Leader-Card rows draw uniform over (pool minus owned), falling back to the full pool when all are owned. Bonus slot fires once at the end of a 10-pack open when HasBonusSlot is set. Slot 8 falls back to the general slot's per-card weights for the rolled tier when slot-8 has only a rarity-level rate quoted (the common shape on normal packs). PackController.Open loads the draw table + viewer owned card ids and passes them to the sampler; the category-based forced-Legendary slot-8 override is gone. ICardFoilLookup replaces ICardPoolProvider for the foil-twin heuristic. Drops the test-fixture pack-draw seed overlay so the production seed flows through the importer tests; controller tests that fabricate their own card sets now call factory.SeedPackDrawTableAsync(...) to install a matching stub draw table. WeightedPick helper handles the cumulative-band roll for both stages. Five sampler tests + four WeightedPick tests + five importer/repo tests; full suite is 653/653 green. Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
This commit is contained in:
@@ -20,14 +20,17 @@ public class PackDrawTableImporterTests
|
||||
|
||||
await new PackDrawTableImporter().ImportAsync(db, SeedDir);
|
||||
|
||||
Assert.That(await db.PackDrawConfigs.CountAsync(), Is.EqualTo(2));
|
||||
Assert.That(await db.PackDrawSlotRates.CountAsync(), Is.EqualTo(8));
|
||||
Assert.That(await db.PackDrawCardWeights.CountAsync(), Is.EqualTo(5));
|
||||
// Production seed is the source of truth in test output (no test-fixture overlay).
|
||||
Assert.That(await db.PackDrawConfigs.CountAsync(), Is.GreaterThanOrEqualTo(200));
|
||||
Assert.That(await db.PackDrawSlotRates.CountAsync(), Is.GreaterThanOrEqualTo(1000));
|
||||
Assert.That(await db.PackDrawCardWeights.CountAsync(), Is.GreaterThanOrEqualTo(50_000));
|
||||
|
||||
// 98001 is a Guaranteed-Leader-Card bundle — bonus slot must contain rate-less
|
||||
// Special-tier leader rows.
|
||||
var bonus = await db.PackDrawCardWeights
|
||||
.Where(w => w.PackId == 98001 && w.Slot == DrawSlot.Bonus)
|
||||
.ToListAsync();
|
||||
Assert.That(bonus.Count, Is.EqualTo(2));
|
||||
Assert.That(bonus.Count, Is.GreaterThan(0));
|
||||
Assert.That(bonus.All(w => w.RatePct == null && w.IsLeader && w.Tier == DrawTier.Special), Is.True);
|
||||
}
|
||||
|
||||
|
||||
Reference in New Issue
Block a user