@omnitronix/happy-panda-game-engine 0.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (71) hide show
  1. package/README.md +212 -0
  2. package/dist/__tests__/bonus-sequence.test.js +337 -0
  3. package/dist/__tests__/bonus-sequence.test.js.map +1 -0
  4. package/dist/__tests__/cherry-frequency.test.js +128 -0
  5. package/dist/__tests__/cherry-frequency.test.js.map +1 -0
  6. package/dist/__tests__/counter-manager.test.js +316 -0
  7. package/dist/__tests__/counter-manager.test.js.map +1 -0
  8. package/dist/__tests__/cpp-parity.test.js +368 -0
  9. package/dist/__tests__/cpp-parity.test.js.map +1 -0
  10. package/dist/__tests__/fixtures/cpp-parity-vectors.json +438 -0
  11. package/dist/__tests__/happy-panda-engine.test.js +367 -0
  12. package/dist/__tests__/happy-panda-engine.test.js.map +1 -0
  13. package/dist/__tests__/jackpot-manager.test.js +313 -0
  14. package/dist/__tests__/jackpot-manager.test.js.map +1 -0
  15. package/dist/__tests__/jackpot-trigger-trace.test.js +146 -0
  16. package/dist/__tests__/jackpot-trigger-trace.test.js.map +1 -0
  17. package/dist/__tests__/rtp-1million.test.js +156 -0
  18. package/dist/__tests__/rtp-1million.test.js.map +1 -0
  19. package/dist/__tests__/rtp-analysis.test.js +138 -0
  20. package/dist/__tests__/rtp-analysis.test.js.map +1 -0
  21. package/dist/__tests__/rtp-diagnostic.test.js +126 -0
  22. package/dist/__tests__/rtp-diagnostic.test.js.map +1 -0
  23. package/dist/__tests__/rtp-simulation.test.js +409 -0
  24. package/dist/__tests__/rtp-simulation.test.js.map +1 -0
  25. package/dist/__tests__/special-wins.test.js +179 -0
  26. package/dist/__tests__/special-wins.test.js.map +1 -0
  27. package/dist/__tests__/spin-generator.test.js +250 -0
  28. package/dist/__tests__/spin-generator.test.js.map +1 -0
  29. package/dist/__tests__/spin-handler.test.js +210 -0
  30. package/dist/__tests__/spin-handler.test.js.map +1 -0
  31. package/dist/__tests__/symbol-distribution.test.js +119 -0
  32. package/dist/__tests__/symbol-distribution.test.js.map +1 -0
  33. package/dist/__tests__/weighted-random.test.js +165 -0
  34. package/dist/__tests__/weighted-random.test.js.map +1 -0
  35. package/dist/__tests__/win-evaluator.test.js +254 -0
  36. package/dist/__tests__/win-evaluator.test.js.map +1 -0
  37. package/dist/config/happy-panda.config.js +714 -0
  38. package/dist/config/happy-panda.config.js.map +1 -0
  39. package/dist/config/index.js +21 -0
  40. package/dist/config/index.js.map +1 -0
  41. package/dist/domain/index.js +21 -0
  42. package/dist/domain/index.js.map +1 -0
  43. package/dist/domain/types.js +28 -0
  44. package/dist/domain/types.js.map +1 -0
  45. package/dist/engine/happy-panda-engine.js +197 -0
  46. package/dist/engine/happy-panda-engine.js.map +1 -0
  47. package/dist/engine/index.js +21 -0
  48. package/dist/engine/index.js.map +1 -0
  49. package/dist/index.js +34 -0
  50. package/dist/index.js.map +1 -0
  51. package/dist/logic/handlers/index.js +21 -0
  52. package/dist/logic/handlers/index.js.map +1 -0
  53. package/dist/logic/handlers/spin-handler.js +256 -0
  54. package/dist/logic/handlers/spin-handler.js.map +1 -0
  55. package/dist/logic/index.js +22 -0
  56. package/dist/logic/index.js.map +1 -0
  57. package/dist/logic/services/counter-manager.js +265 -0
  58. package/dist/logic/services/counter-manager.js.map +1 -0
  59. package/dist/logic/services/index.js +23 -0
  60. package/dist/logic/services/index.js.map +1 -0
  61. package/dist/logic/services/jackpot-manager.js +142 -0
  62. package/dist/logic/services/jackpot-manager.js.map +1 -0
  63. package/dist/logic/services/win-evaluator.js +470 -0
  64. package/dist/logic/services/win-evaluator.js.map +1 -0
  65. package/dist/rng/index.js +22 -0
  66. package/dist/rng/index.js.map +1 -0
  67. package/dist/rng/spin-generator.js +341 -0
  68. package/dist/rng/spin-generator.js.map +1 -0
  69. package/dist/rng/weighted-random.js +58 -0
  70. package/dist/rng/weighted-random.js.map +1 -0
  71. package/package.json +49 -0
@@ -0,0 +1,250 @@
1
+ "use strict";
2
+ /**
3
+ * Spin Generator Tests
4
+ *
5
+ * Tests for grid generation with symbol limitation system.
6
+ */
7
+ Object.defineProperty(exports, "__esModule", { value: true });
8
+ const spin_generator_1 = require("../rng/spin-generator");
9
+ const happy_panda_config_1 = require("../config/happy-panda.config");
10
+ const types_1 = require("../domain/types");
11
+ // Deterministic mock RNG
12
+ function createMockRng(values = []) {
13
+ let index = 0;
14
+ return {
15
+ async random(max) {
16
+ const val = values[index % Math.max(values.length, 1)] || 0;
17
+ index++;
18
+ return val % max;
19
+ },
20
+ async randomBatch(count, max) {
21
+ const results = [];
22
+ for (let i = 0; i < count; i++) {
23
+ const val = values[index % Math.max(values.length, 1)] || 0;
24
+ index++;
25
+ results.push(val % max);
26
+ }
27
+ return results;
28
+ },
29
+ };
30
+ }
31
+ describe('Spin Generator', () => {
32
+ describe('generateGrid - Basic Functionality', () => {
33
+ it('should generate a 3x3 grid', async () => {
34
+ const rng = createMockRng([0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15]);
35
+ const grid = await (0, spin_generator_1.generateGrid)(happy_panda_config_1.SpinType.PAID_SPIN, types_1.GameDirection.SINGLE, rng);
36
+ expect(grid).toHaveLength(happy_panda_config_1.GRID.REELS);
37
+ grid.forEach(reel => {
38
+ expect(reel).toHaveLength(happy_panda_config_1.GRID.ROWS);
39
+ });
40
+ });
41
+ it('should generate valid symbols', async () => {
42
+ const rng = createMockRng([0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15]);
43
+ const grid = await (0, spin_generator_1.generateGrid)(happy_panda_config_1.SpinType.PAID_SPIN, types_1.GameDirection.SINGLE, rng);
44
+ grid.forEach(reel => {
45
+ reel.forEach(symbol => {
46
+ expect(symbol).toBeGreaterThanOrEqual(happy_panda_config_1.Symbol.SUPER_SEVEN);
47
+ expect(symbol).toBeLessThanOrEqual(happy_panda_config_1.Symbol.COLOR_BLUE);
48
+ });
49
+ });
50
+ });
51
+ it('should work with BOTH game direction', async () => {
52
+ const rng = createMockRng([0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15]);
53
+ const grid = await (0, spin_generator_1.generateGrid)(happy_panda_config_1.SpinType.PAID_SPIN, types_1.GameDirection.BOTH, rng);
54
+ expect(grid).toHaveLength(3);
55
+ expect(grid[0]).toHaveLength(3);
56
+ });
57
+ });
58
+ describe('generateGrid - Spin Types', () => {
59
+ it('should generate grid for PAID_SPIN', async () => {
60
+ const rng = createMockRng([0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10]);
61
+ const grid = await (0, spin_generator_1.generateGrid)(happy_panda_config_1.SpinType.PAID_SPIN, types_1.GameDirection.SINGLE, rng);
62
+ expect(grid).toBeDefined();
63
+ expect(grid).toHaveLength(3);
64
+ });
65
+ it('should generate grid for BONUS_JACKPOT', async () => {
66
+ const rng = createMockRng([0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10]);
67
+ const grid = await (0, spin_generator_1.generateGrid)(happy_panda_config_1.SpinType.BONUS_JACKPOT, types_1.GameDirection.SINGLE, rng);
68
+ expect(grid).toBeDefined();
69
+ expect(grid).toHaveLength(3);
70
+ });
71
+ it('should generate grid for BONUS_CHERRY', async () => {
72
+ const rng = createMockRng([0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10]);
73
+ const grid = await (0, spin_generator_1.generateGrid)(happy_panda_config_1.SpinType.BONUS_CHERRY, types_1.GameDirection.SINGLE, rng);
74
+ expect(grid).toBeDefined();
75
+ expect(grid).toHaveLength(3);
76
+ });
77
+ it('should generate grid for BONUS_BELL', async () => {
78
+ const rng = createMockRng([0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10]);
79
+ const grid = await (0, spin_generator_1.generateGrid)(happy_panda_config_1.SpinType.BONUS_BELL, types_1.GameDirection.SINGLE, rng);
80
+ expect(grid).toBeDefined();
81
+ expect(grid).toHaveLength(3);
82
+ });
83
+ it('should generate grid for BONUS_BAR1', async () => {
84
+ const rng = createMockRng([0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10]);
85
+ const grid = await (0, spin_generator_1.generateGrid)(happy_panda_config_1.SpinType.BONUS_BAR1, types_1.GameDirection.SINGLE, rng);
86
+ expect(grid).toBeDefined();
87
+ expect(grid).toHaveLength(3);
88
+ });
89
+ it('should generate grid for RESPIN_CHERRY', async () => {
90
+ const rng = createMockRng([0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10]);
91
+ const grid = await (0, spin_generator_1.generateGrid)(happy_panda_config_1.SpinType.RESPIN_CHERRY, types_1.GameDirection.SINGLE, rng);
92
+ expect(grid).toBeDefined();
93
+ expect(grid).toHaveLength(3);
94
+ });
95
+ it('should throw for unknown spin type', async () => {
96
+ const rng = createMockRng([0, 1, 2, 3, 4, 5]);
97
+ await expect((0, spin_generator_1.generateGrid)(999, types_1.GameDirection.SINGLE, rng)).rejects.toThrow('Unknown spin type');
98
+ });
99
+ });
100
+ describe('generateGrid - Center Symbol for Respin', () => {
101
+ it('should preserve center symbol during respin', async () => {
102
+ const rng = createMockRng([0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10]);
103
+ const centerSymbol = happy_panda_config_1.Symbol.CHERRY;
104
+ const grid = await (0, spin_generator_1.generateGrid)(happy_panda_config_1.SpinType.RESPIN_CHERRY, types_1.GameDirection.SINGLE, rng, centerSymbol);
105
+ expect(grid[1][1]).toBe(centerSymbol);
106
+ });
107
+ it('should preserve Super Cherry as center symbol', async () => {
108
+ const rng = createMockRng([0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10]);
109
+ const centerSymbol = happy_panda_config_1.Symbol.SUPER_CHERRY;
110
+ const grid = await (0, spin_generator_1.generateGrid)(happy_panda_config_1.SpinType.RESPIN_CHERRY, types_1.GameDirection.SINGLE, rng, centerSymbol);
111
+ expect(grid[1][1]).toBe(centerSymbol);
112
+ });
113
+ });
114
+ describe('generateGrid - Symbol Limitation', () => {
115
+ it('should not exceed maximum symbol counts', async () => {
116
+ // Run multiple generations and check no symbol appears more than 9 times
117
+ const rng = createMockRng([0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15]);
118
+ for (let i = 0; i < 10; i++) {
119
+ const grid = await (0, spin_generator_1.generateGrid)(happy_panda_config_1.SpinType.PAID_SPIN, types_1.GameDirection.SINGLE, rng);
120
+ // Count each symbol
121
+ const counts = new Map();
122
+ grid.forEach(reel => {
123
+ reel.forEach(symbol => {
124
+ counts.set(symbol, (counts.get(symbol) || 0) + 1);
125
+ });
126
+ });
127
+ // No symbol should appear more than 9 times (all positions)
128
+ counts.forEach((count, _symbol) => {
129
+ expect(count).toBeLessThanOrEqual(happy_panda_config_1.GRID.POSITIONS);
130
+ });
131
+ }
132
+ });
133
+ it('should fill all 9 positions', async () => {
134
+ const rng = createMockRng([0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10]);
135
+ const grid = await (0, spin_generator_1.generateGrid)(happy_panda_config_1.SpinType.PAID_SPIN, types_1.GameDirection.SINGLE, rng);
136
+ let filledCount = 0;
137
+ grid.forEach(reel => {
138
+ reel.forEach(symbol => {
139
+ if (symbol !== happy_panda_config_1.Symbol.NONE) {
140
+ filledCount++;
141
+ }
142
+ });
143
+ });
144
+ expect(filledCount).toBe(9);
145
+ });
146
+ });
147
+ describe('generateGrid - Forced 3x3 Wins', () => {
148
+ it('should generate filled grid when 3x3 triggered', async () => {
149
+ // RNG values that would trigger 3x3 (first value = 0 for yes)
150
+ const rng = createMockRng([0, 0, 0, 0, 0, 0, 0, 0, 0, 0]);
151
+ const grid = await (0, spin_generator_1.generateGrid)(happy_panda_config_1.SpinType.PAID_SPIN, types_1.GameDirection.SINGLE, rng);
152
+ // If 3x3 triggered, all symbols should be the same
153
+ const firstSymbol = grid[0][0];
154
+ let allSame = true;
155
+ grid.forEach(reel => {
156
+ reel.forEach(symbol => {
157
+ if (symbol !== firstSymbol) {
158
+ allSame = false;
159
+ }
160
+ });
161
+ });
162
+ // Either all same (3x3) or properly distributed
163
+ expect(grid).toHaveLength(3);
164
+ });
165
+ });
166
+ describe('generateCherryPieceGrid', () => {
167
+ it('should generate grid with all cherry pieces', () => {
168
+ const grid = (0, spin_generator_1.generateCherryPieceGrid)();
169
+ expect(grid[0][0]).toBe(happy_panda_config_1.Symbol.CHERRY_00);
170
+ expect(grid[0][1]).toBe(happy_panda_config_1.Symbol.CHERRY_01);
171
+ expect(grid[0][2]).toBe(happy_panda_config_1.Symbol.CHERRY_02);
172
+ expect(grid[1][0]).toBe(happy_panda_config_1.Symbol.CHERRY_10);
173
+ expect(grid[1][1]).toBe(happy_panda_config_1.Symbol.CHERRY_11);
174
+ expect(grid[1][2]).toBe(happy_panda_config_1.Symbol.CHERRY_12);
175
+ expect(grid[2][0]).toBe(happy_panda_config_1.Symbol.CHERRY_20);
176
+ expect(grid[2][1]).toBe(happy_panda_config_1.Symbol.CHERRY_21);
177
+ expect(grid[2][2]).toBe(happy_panda_config_1.Symbol.CHERRY_22);
178
+ });
179
+ it('should have 9 cherry piece symbols', () => {
180
+ const grid = (0, spin_generator_1.generateCherryPieceGrid)();
181
+ let pieceCount = 0;
182
+ const cherryPieces = [
183
+ happy_panda_config_1.Symbol.CHERRY_00, happy_panda_config_1.Symbol.CHERRY_01, happy_panda_config_1.Symbol.CHERRY_02,
184
+ happy_panda_config_1.Symbol.CHERRY_10, happy_panda_config_1.Symbol.CHERRY_11, happy_panda_config_1.Symbol.CHERRY_12,
185
+ happy_panda_config_1.Symbol.CHERRY_20, happy_panda_config_1.Symbol.CHERRY_21, happy_panda_config_1.Symbol.CHERRY_22,
186
+ ];
187
+ grid.forEach(reel => {
188
+ reel.forEach(symbol => {
189
+ if (cherryPieces.includes(symbol)) {
190
+ pieceCount++;
191
+ }
192
+ });
193
+ });
194
+ expect(pieceCount).toBe(9);
195
+ });
196
+ });
197
+ describe('isCompleteCherryPiece', () => {
198
+ it('should return true for complete cherry piece grid', () => {
199
+ const grid = (0, spin_generator_1.generateCherryPieceGrid)();
200
+ expect((0, spin_generator_1.isCompleteCherryPiece)(grid)).toBe(true);
201
+ });
202
+ it('should return false for non-cherry piece grid', () => {
203
+ const grid = [
204
+ [happy_panda_config_1.Symbol.CHERRY, happy_panda_config_1.Symbol.CHERRY, happy_panda_config_1.Symbol.CHERRY],
205
+ [happy_panda_config_1.Symbol.CHERRY, happy_panda_config_1.Symbol.CHERRY, happy_panda_config_1.Symbol.CHERRY],
206
+ [happy_panda_config_1.Symbol.CHERRY, happy_panda_config_1.Symbol.CHERRY, happy_panda_config_1.Symbol.CHERRY],
207
+ ];
208
+ expect((0, spin_generator_1.isCompleteCherryPiece)(grid)).toBe(false);
209
+ });
210
+ it('should return false for partial cherry piece grid', () => {
211
+ const grid = [
212
+ [happy_panda_config_1.Symbol.CHERRY_00, happy_panda_config_1.Symbol.CHERRY_01, happy_panda_config_1.Symbol.CHERRY_02],
213
+ [happy_panda_config_1.Symbol.CHERRY_10, happy_panda_config_1.Symbol.CHERRY_11, happy_panda_config_1.Symbol.CHERRY_12],
214
+ [happy_panda_config_1.Symbol.CHERRY_20, happy_panda_config_1.Symbol.CHERRY_21, happy_panda_config_1.Symbol.MELON], // One non-piece
215
+ ];
216
+ expect((0, spin_generator_1.isCompleteCherryPiece)(grid)).toBe(false);
217
+ });
218
+ it('should return false for mixed grid', () => {
219
+ const grid = [
220
+ [happy_panda_config_1.Symbol.CHERRY_00, happy_panda_config_1.Symbol.BAR3, happy_panda_config_1.Symbol.SEVEN],
221
+ [happy_panda_config_1.Symbol.MELON, happy_panda_config_1.Symbol.BELL, happy_panda_config_1.Symbol.ORANGE],
222
+ [happy_panda_config_1.Symbol.PRUNE, happy_panda_config_1.Symbol.BAR2, happy_panda_config_1.Symbol.BAR1],
223
+ ];
224
+ expect((0, spin_generator_1.isCompleteCherryPiece)(grid)).toBe(false);
225
+ });
226
+ });
227
+ describe('generateGrid - Determinism', () => {
228
+ it('should produce same grid with same RNG sequence', async () => {
229
+ const values = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15];
230
+ const rng1 = createMockRng(values);
231
+ const rng2 = createMockRng(values);
232
+ const grid1 = await (0, spin_generator_1.generateGrid)(happy_panda_config_1.SpinType.PAID_SPIN, types_1.GameDirection.SINGLE, rng1);
233
+ const grid2 = await (0, spin_generator_1.generateGrid)(happy_panda_config_1.SpinType.PAID_SPIN, types_1.GameDirection.SINGLE, rng2);
234
+ expect(grid1).toEqual(grid2);
235
+ });
236
+ });
237
+ describe('generateGrid - All Positions Covered', () => {
238
+ it('should generate symbol for every position', async () => {
239
+ const rng = createMockRng([0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15]);
240
+ const grid = await (0, spin_generator_1.generateGrid)(happy_panda_config_1.SpinType.PAID_SPIN, types_1.GameDirection.SINGLE, rng);
241
+ for (let reel = 0; reel < 3; reel++) {
242
+ for (let row = 0; row < 3; row++) {
243
+ expect(grid[reel][row]).toBeDefined();
244
+ expect(grid[reel][row]).not.toBe(happy_panda_config_1.Symbol.NONE);
245
+ }
246
+ }
247
+ });
248
+ });
249
+ });
250
+ //# sourceMappingURL=spin-generator.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"spin-generator.test.js","sourceRoot":"","sources":["../../src/__tests__/spin-generator.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,210 @@
1
+ "use strict";
2
+ /**
3
+ * Spin Handler Integration Tests
4
+ *
5
+ * Tests for the complete spin flow: state creation, spin execution, and transitions.
6
+ */
7
+ Object.defineProperty(exports, "__esModule", { value: true });
8
+ const spin_handler_1 = require("../logic/handlers/spin-handler");
9
+ const happy_panda_config_1 = require("../config/happy-panda.config");
10
+ const types_1 = require("../domain/types");
11
+ // Deterministic mock RNG for predictable tests
12
+ function createMockRng(values = []) {
13
+ let index = 0;
14
+ return {
15
+ async random(max) {
16
+ const val = values[index % values.length] || 0;
17
+ index++;
18
+ return val % max;
19
+ },
20
+ async randomBatch(count, max) {
21
+ const results = [];
22
+ for (let i = 0; i < count; i++) {
23
+ const val = values[index % values.length] || 0;
24
+ index++;
25
+ results.push(val % max);
26
+ }
27
+ return results;
28
+ },
29
+ };
30
+ }
31
+ describe('Spin Handler', () => {
32
+ describe('State Initialization', () => {
33
+ it('should create initial state with correct bet calculation (SINGLE mode)', () => {
34
+ const state = (0, spin_handler_1.createInitialState)(types_1.GameDirection.SINGLE, 1);
35
+ expect(state.gameDirection).toBe(types_1.GameDirection.SINGLE);
36
+ expect(state.betStake).toBe(1);
37
+ expect(state.betGame).toBe(8); // 8 lines * 1 stake
38
+ expect(state.currentSpinType).toBe(happy_panda_config_1.SpinType.PAID_SPIN);
39
+ expect(state.spinsRemaining).toBe(0);
40
+ });
41
+ it('should create initial state with correct bet calculation (BOTH mode)', () => {
42
+ const state = (0, spin_handler_1.createInitialState)(types_1.GameDirection.BOTH, 2);
43
+ expect(state.gameDirection).toBe(types_1.GameDirection.BOTH);
44
+ expect(state.betStake).toBe(2);
45
+ expect(state.betGame).toBe(32); // 16 lines * 2 stake
46
+ });
47
+ it('should initialize counters correctly', () => {
48
+ const state = (0, spin_handler_1.createInitialState)(types_1.GameDirection.SINGLE, 1);
49
+ expect(state.counters.jackpot).toBe(1);
50
+ expect(state.counters.bar1).toBe(1);
51
+ // Cherry and bell have initial values
52
+ expect(state.counters.cherry).toBeGreaterThan(0);
53
+ expect(state.counters.bell).toBeGreaterThan(0);
54
+ });
55
+ it('should initialize jackpots correctly', () => {
56
+ const state = (0, spin_handler_1.createInitialState)(types_1.GameDirection.SINGLE, 1);
57
+ expect(state.jackpots.bonusJackpotValue).toBe(100 * 8); // 100 * betGame
58
+ expect(state.jackpots.poolJackpotValue).toBe(0);
59
+ });
60
+ it('should initialize empty grid', () => {
61
+ const state = (0, spin_handler_1.createInitialState)(types_1.GameDirection.SINGLE, 1);
62
+ expect(state.grid).toHaveLength(3);
63
+ state.grid.forEach(reel => {
64
+ expect(reel).toHaveLength(3);
65
+ reel.forEach(symbol => expect(symbol).toBe(happy_panda_config_1.Symbol.NONE));
66
+ });
67
+ });
68
+ });
69
+ describe('Input Validation', () => {
70
+ it('should throw on invalid game direction', () => {
71
+ expect(() => (0, spin_handler_1.validateBetParams)(5, 1)).toThrow(/Invalid game direction/);
72
+ });
73
+ it('should throw on non-integer bet stake', () => {
74
+ expect(() => (0, spin_handler_1.validateBetParams)(types_1.GameDirection.SINGLE, 1.5)).toThrow(/must be an integer/);
75
+ });
76
+ it('should throw on bet stake below minimum', () => {
77
+ expect(() => (0, spin_handler_1.validateBetParams)(types_1.GameDirection.SINGLE, 0)).toThrow(/must be between/);
78
+ });
79
+ it('should throw on bet stake above maximum', () => {
80
+ expect(() => (0, spin_handler_1.validateBetParams)(types_1.GameDirection.SINGLE, 1000)).toThrow(/must be between/);
81
+ });
82
+ it('should accept valid parameters', () => {
83
+ expect(() => (0, spin_handler_1.validateBetParams)(types_1.GameDirection.SINGLE, spin_handler_1.MIN_BET_STAKE)).not.toThrow();
84
+ expect(() => (0, spin_handler_1.validateBetParams)(types_1.GameDirection.BOTH, spin_handler_1.MAX_BET_STAKE)).not.toThrow();
85
+ });
86
+ });
87
+ describe('Bet Updates', () => {
88
+ it('should update bet correctly', () => {
89
+ const initialState = (0, spin_handler_1.createInitialState)(types_1.GameDirection.SINGLE, 1);
90
+ const newState = (0, spin_handler_1.updateBet)(initialState, types_1.GameDirection.BOTH, 5);
91
+ expect(newState.gameDirection).toBe(types_1.GameDirection.BOTH);
92
+ expect(newState.betStake).toBe(5);
93
+ expect(newState.betGame).toBe(80); // 16 * 5
94
+ });
95
+ it('should throw when changing bet during bonus', () => {
96
+ const state = (0, spin_handler_1.createInitialState)(types_1.GameDirection.SINGLE, 1);
97
+ state.currentSpinType = happy_panda_config_1.SpinType.BONUS_JACKPOT;
98
+ expect(() => (0, spin_handler_1.updateBet)(state, types_1.GameDirection.BOTH, 2)).toThrow(/Cannot change bet/);
99
+ });
100
+ it('should throw when changing bet with pending bonuses', () => {
101
+ const state = (0, spin_handler_1.createInitialState)(types_1.GameDirection.SINGLE, 1);
102
+ state.pendingBonuses.jackpot = 1;
103
+ expect(() => (0, spin_handler_1.updateBet)(state, types_1.GameDirection.BOTH, 2)).toThrow(/Cannot change bet/);
104
+ });
105
+ });
106
+ describe('Spin Execution', () => {
107
+ it('should execute a basic paid spin and generate grid', async () => {
108
+ const state = (0, spin_handler_1.createInitialState)(types_1.GameDirection.SINGLE, 1);
109
+ const rng = createMockRng([0, 1, 2, 3, 4, 5, 6, 7, 8]);
110
+ const result = await (0, spin_handler_1.executeSpin)(state, rng);
111
+ // Grid should be populated (not all NONE)
112
+ expect(result.grid).toBeDefined();
113
+ expect(result.grid).toHaveLength(3);
114
+ // State should be returned
115
+ expect(result.state).toBeDefined();
116
+ expect(result.state.grid).toEqual(result.grid);
117
+ });
118
+ it('should evaluate wins correctly', async () => {
119
+ const state = (0, spin_handler_1.createInitialState)(types_1.GameDirection.SINGLE, 1);
120
+ const rng = createMockRng([0, 1, 2, 3, 4, 5, 6, 7, 8]);
121
+ const result = await (0, spin_handler_1.executeSpin)(state, rng);
122
+ // Wins structure should be present
123
+ expect(result.wins).toBeDefined();
124
+ expect(result.wins.lineWins).toBeDefined();
125
+ expect(result.wins.wallWin).toBeDefined();
126
+ expect(result.wins.scatterWins).toBeDefined();
127
+ expect(result.wins.totalPayout).toBeGreaterThanOrEqual(0);
128
+ });
129
+ it('should update state for next spin', async () => {
130
+ const state = (0, spin_handler_1.createInitialState)(types_1.GameDirection.SINGLE, 1);
131
+ const rng = createMockRng([0, 1, 2, 3, 4, 5, 6, 7, 8]);
132
+ const result = await (0, spin_handler_1.executeSpin)(state, rng);
133
+ // Next spin type should be determined
134
+ expect(result.state.currentSpinType).toBeDefined();
135
+ expect(result.state.nextSpinType).toBeDefined();
136
+ });
137
+ it('should track jackpot and pool jackpot separately', async () => {
138
+ const state = (0, spin_handler_1.createInitialState)(types_1.GameDirection.SINGLE, 1);
139
+ const rng = createMockRng([0, 1, 2, 3, 4, 5, 6, 7, 8]);
140
+ const result = await (0, spin_handler_1.executeSpin)(state, rng);
141
+ expect(result.jackpotWon).toBeGreaterThanOrEqual(0);
142
+ expect(result.poolJackpotWon).toBeGreaterThanOrEqual(0);
143
+ });
144
+ });
145
+ describe('Bonus Flow', () => {
146
+ it('should track bonus triggered flag', async () => {
147
+ const state = (0, spin_handler_1.createInitialState)(types_1.GameDirection.SINGLE, 1);
148
+ const rng = createMockRng([0, 1, 2, 3, 4, 5, 6, 7, 8]);
149
+ const result = await (0, spin_handler_1.executeSpin)(state, rng);
150
+ // bonusTriggered can be null or a SpinType
151
+ expect(result.bonusTriggered === null || typeof result.bonusTriggered === 'number').toBe(true);
152
+ });
153
+ it('should track bonus completion flag', async () => {
154
+ const state = (0, spin_handler_1.createInitialState)(types_1.GameDirection.SINGLE, 1);
155
+ const rng = createMockRng([0, 1, 2, 3, 4, 5, 6, 7, 8]);
156
+ const result = await (0, spin_handler_1.executeSpin)(state, rng);
157
+ expect(typeof result.isBonusComplete).toBe('boolean');
158
+ });
159
+ it('should handle pending bonuses in state', async () => {
160
+ const state = (0, spin_handler_1.createInitialState)(types_1.GameDirection.SINGLE, 1);
161
+ const rng = createMockRng([0, 1, 2, 3, 4, 5, 6, 7, 8]);
162
+ const result = await (0, spin_handler_1.executeSpin)(state, rng);
163
+ expect(result.state.pendingBonuses).toBeDefined();
164
+ expect(result.state.pendingBonuses.jackpot).toBeGreaterThanOrEqual(0);
165
+ expect(result.state.pendingBonuses.cherry).toBeGreaterThanOrEqual(0);
166
+ expect(result.state.pendingBonuses.bell).toBeGreaterThanOrEqual(0);
167
+ expect(result.state.pendingBonuses.bar1).toBeGreaterThanOrEqual(0);
168
+ expect(result.state.pendingBonuses.respin).toBeGreaterThanOrEqual(0);
169
+ });
170
+ });
171
+ describe('State Immutability', () => {
172
+ it('should not mutate original state', async () => {
173
+ const originalState = (0, spin_handler_1.createInitialState)(types_1.GameDirection.SINGLE, 1);
174
+ const originalCounters = { ...originalState.counters };
175
+ const originalPending = { ...originalState.pendingBonuses };
176
+ const originalJackpots = { ...originalState.jackpots };
177
+ const rng = createMockRng([0, 1, 2, 3, 4, 5, 6, 7, 8]);
178
+ await (0, spin_handler_1.executeSpin)(originalState, rng);
179
+ // Original state should be unchanged
180
+ expect(originalState.counters).toEqual(originalCounters);
181
+ expect(originalState.pendingBonuses).toEqual(originalPending);
182
+ expect(originalState.jackpots).toEqual(originalJackpots);
183
+ });
184
+ });
185
+ describe('Accumulated Bonus Wins', () => {
186
+ it('should start with zero accumulated bonus wins', () => {
187
+ const state = (0, spin_handler_1.createInitialState)(types_1.GameDirection.SINGLE, 1);
188
+ expect(state.accumulatedBonusWins).toBe(0);
189
+ });
190
+ it('should track center cherry symbol for respins', () => {
191
+ const state = (0, spin_handler_1.createInitialState)(types_1.GameDirection.SINGLE, 1);
192
+ expect(state.centerCherrySymbol).toBeNull();
193
+ });
194
+ });
195
+ describe('Multiple Spins Sequence', () => {
196
+ it('should handle multiple consecutive spins', async () => {
197
+ let state = (0, spin_handler_1.createInitialState)(types_1.GameDirection.SINGLE, 1);
198
+ const rng = createMockRng([0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15]);
199
+ // Execute 3 spins
200
+ for (let i = 0; i < 3; i++) {
201
+ const result = await (0, spin_handler_1.executeSpin)(state, rng);
202
+ state = result.state;
203
+ // Each spin should return valid grid
204
+ expect(state.grid).toBeDefined();
205
+ expect(state.grid).toHaveLength(3);
206
+ }
207
+ });
208
+ });
209
+ });
210
+ //# sourceMappingURL=spin-handler.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"spin-handler.test.js","sourceRoot":"","sources":["../../src/__tests__/spin-handler.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,119 @@
1
+ "use strict";
2
+ /**
3
+ * Symbol Distribution Diagnostic
4
+ * Checks if our screen generation matches C++ probability expectations.
5
+ */
6
+ Object.defineProperty(exports, "__esModule", { value: true });
7
+ const spin_generator_1 = require("../rng/spin-generator");
8
+ const happy_panda_config_1 = require("../config/happy-panda.config");
9
+ const types_1 = require("../domain/types");
10
+ function createTestRng(seed = 12345) {
11
+ let state = seed;
12
+ function nextInt() {
13
+ state = (state * 1664525 + 1013904223) >>> 0;
14
+ return state;
15
+ }
16
+ return {
17
+ async random(max) {
18
+ return nextInt() % max;
19
+ },
20
+ async randomBatch(count, max) {
21
+ const results = [];
22
+ for (let i = 0; i < count; i++) {
23
+ results.push(nextInt() % max);
24
+ }
25
+ return results;
26
+ },
27
+ };
28
+ }
29
+ describe('Symbol Distribution Diagnostic', () => {
30
+ it('should check scatter rates across spin types', async () => {
31
+ const rng = createTestRng(99);
32
+ const spinsPerType = 5000;
33
+ const results = {};
34
+ for (const spinType of [happy_panda_config_1.SpinType.PAID_SPIN, happy_panda_config_1.SpinType.BONUS_JACKPOT, happy_panda_config_1.SpinType.BONUS_CHERRY, happy_panda_config_1.SpinType.BONUS_BELL, happy_panda_config_1.SpinType.BONUS_BAR1]) {
35
+ const typeName = happy_panda_config_1.SpinType[spinType];
36
+ results[typeName] = { total: spinsPerType, scatter2plus: 0, wall: 0 };
37
+ for (let i = 0; i < spinsPerType; i++) {
38
+ const grid = await (0, spin_generator_1.generateGrid)(spinType, types_1.GameDirection.SINGLE, rng);
39
+ let sevenCount = 0;
40
+ let superSevenCount = 0;
41
+ let allSame = true;
42
+ const firstSymbol = grid[0][0];
43
+ for (let reel = 0; reel < 3; reel++) {
44
+ for (let row = 0; row < 3; row++) {
45
+ if (grid[reel][row] === happy_panda_config_1.Symbol.SEVEN)
46
+ sevenCount++;
47
+ if (grid[reel][row] === happy_panda_config_1.Symbol.SUPER_SEVEN)
48
+ superSevenCount++;
49
+ if (grid[reel][row] !== firstSymbol)
50
+ allSame = false;
51
+ }
52
+ }
53
+ if (sevenCount >= 2 || superSevenCount >= 2)
54
+ results[typeName].scatter2plus++;
55
+ if (allSame)
56
+ results[typeName].wall++;
57
+ }
58
+ }
59
+ console.log('=== SCATTER & WALL RATES BY SPIN TYPE ===\n');
60
+ for (const [typeName, data] of Object.entries(results)) {
61
+ console.log(`${typeName}:`);
62
+ console.log(` Scatter (2+ Seven): ${(data.scatter2plus / data.total * 100).toFixed(2)}%`);
63
+ console.log(` Wall (9 same): ${(data.wall / data.total * 100).toFixed(2)}%`);
64
+ }
65
+ expect(true).toBe(true);
66
+ }, 120000);
67
+ it('should match expected limitation tier distribution', async () => {
68
+ const rng = createTestRng(42);
69
+ const spins = 10000;
70
+ const sevenCounts = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0];
71
+ const superSevenCounts = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0];
72
+ for (let i = 0; i < spins; i++) {
73
+ const grid = await (0, spin_generator_1.generateGrid)(happy_panda_config_1.SpinType.PAID_SPIN, types_1.GameDirection.SINGLE, rng);
74
+ let sevenCount = 0;
75
+ let superSevenCount = 0;
76
+ for (let reel = 0; reel < 3; reel++) {
77
+ for (let row = 0; row < 3; row++) {
78
+ if (grid[reel][row] === happy_panda_config_1.Symbol.SEVEN)
79
+ sevenCount++;
80
+ if (grid[reel][row] === happy_panda_config_1.Symbol.SUPER_SEVEN)
81
+ superSevenCount++;
82
+ }
83
+ }
84
+ sevenCounts[sevenCount]++;
85
+ superSevenCounts[superSevenCount]++;
86
+ }
87
+ console.log('=== SYMBOL DISTRIBUTION ANALYSIS (10k spins) ===\n');
88
+ console.log('Super Seven count distribution:');
89
+ for (let i = 0; i <= 9; i++) {
90
+ if (superSevenCounts[i] > 0) {
91
+ const pct = (superSevenCounts[i] / spins * 100).toFixed(2);
92
+ console.log(` ${i} copies: ${superSevenCounts[i]} (${pct}%)`);
93
+ }
94
+ }
95
+ console.log('\nSeven count distribution:');
96
+ for (let i = 0; i <= 9; i++) {
97
+ if (sevenCounts[i] > 0) {
98
+ const pct = (sevenCounts[i] / spins * 100).toFixed(2);
99
+ console.log(` ${i} copies: ${sevenCounts[i]} (${pct}%)`);
100
+ }
101
+ }
102
+ console.log('\nExpected from C++ limitation weights:');
103
+ console.log(' SEV_S: [9,2,1,1] = 69% max2, 15% max4, 7.7% max7, 7.7% max9');
104
+ console.log(' SEV: [8,3,1,1] = 62% max2, 23% max4, 7.7% max7, 7.7% max9');
105
+ const superSeven2OrLess = superSevenCounts[0] + superSevenCounts[1] + superSevenCounts[2];
106
+ const seven2OrLess = sevenCounts[0] + sevenCounts[1] + sevenCounts[2];
107
+ console.log('\nActual 0-2 copies (should match max2 tier):');
108
+ console.log(` Super Seven: ${(superSeven2OrLess / spins * 100).toFixed(1)}%`);
109
+ console.log(` Seven: ${(seven2OrLess / spins * 100).toFixed(1)}%`);
110
+ const scatter2Plus = sevenCounts.slice(2).reduce((a, b) => a + b, 0);
111
+ const superScatter2Plus = superSevenCounts.slice(2).reduce((a, b) => a + b, 0);
112
+ console.log('\nScatter potential (2+ symbols):');
113
+ console.log(` Super Seven 2+: ${(superScatter2Plus / spins * 100).toFixed(1)}%`);
114
+ console.log(` Seven 2+: ${(scatter2Plus / spins * 100).toFixed(1)}%`);
115
+ console.log(` Either 2+: ${((scatter2Plus + superScatter2Plus - (sevenCounts.slice(2).reduce((a, b, i) => a + Math.min(b, superSevenCounts[i + 2] || 0), 0))) / spins * 100).toFixed(1)}%`);
116
+ expect(true).toBe(true);
117
+ }, 60000);
118
+ });
119
+ //# sourceMappingURL=symbol-distribution.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"symbol-distribution.test.js","sourceRoot":"","sources":["../../src/__tests__/symbol-distribution.test.ts"],"names":[],"mappings":"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"}