@littlepartytime/sdk 1.0.0

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 (41) hide show
  1. package/dist/index.d.ts +3 -0
  2. package/dist/index.d.ts.map +1 -0
  3. package/dist/index.js +3 -0
  4. package/dist/index.js.map +1 -0
  5. package/dist/interfaces.d.ts +34 -0
  6. package/dist/interfaces.d.ts.map +1 -0
  7. package/dist/interfaces.js +3 -0
  8. package/dist/interfaces.js.map +1 -0
  9. package/dist/testing/__tests__/game-simulator.test.d.ts +2 -0
  10. package/dist/testing/__tests__/game-simulator.test.d.ts.map +1 -0
  11. package/dist/testing/__tests__/game-simulator.test.js +81 -0
  12. package/dist/testing/__tests__/game-simulator.test.js.map +1 -0
  13. package/dist/testing/__tests__/game-tester.test.d.ts +2 -0
  14. package/dist/testing/__tests__/game-tester.test.d.ts.map +1 -0
  15. package/dist/testing/__tests__/game-tester.test.js +91 -0
  16. package/dist/testing/__tests__/game-tester.test.js.map +1 -0
  17. package/dist/testing/__tests__/mock-players.test.d.ts +2 -0
  18. package/dist/testing/__tests__/mock-players.test.d.ts.map +1 -0
  19. package/dist/testing/__tests__/mock-players.test.js +39 -0
  20. package/dist/testing/__tests__/mock-players.test.js.map +1 -0
  21. package/dist/testing/game-simulator.d.ts +29 -0
  22. package/dist/testing/game-simulator.d.ts.map +1 -0
  23. package/dist/testing/game-simulator.js +52 -0
  24. package/dist/testing/game-simulator.js.map +1 -0
  25. package/dist/testing/game-tester.d.ts +16 -0
  26. package/dist/testing/game-tester.d.ts.map +1 -0
  27. package/dist/testing/game-tester.js +42 -0
  28. package/dist/testing/game-tester.js.map +1 -0
  29. package/dist/testing/index.d.ts +5 -0
  30. package/dist/testing/index.d.ts.map +1 -0
  31. package/dist/testing/index.js +10 -0
  32. package/dist/testing/index.js.map +1 -0
  33. package/dist/testing/mock-players.d.ts +3 -0
  34. package/dist/testing/mock-players.d.ts.map +1 -0
  35. package/dist/testing/mock-players.js +13 -0
  36. package/dist/testing/mock-players.js.map +1 -0
  37. package/dist/types.d.ts +28 -0
  38. package/dist/types.d.ts.map +1 -0
  39. package/dist/types.js +3 -0
  40. package/dist/types.js.map +1 -0
  41. package/package.json +37 -0
@@ -0,0 +1,3 @@
1
+ export type { Player, GameAction, GameResult, Platform } from './types';
2
+ export type { PlayerState, GameState, GameConfig, GameEngine, GameRendererProps } from './interfaces';
3
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,YAAY,EAAE,MAAM,EAAE,UAAU,EAAE,UAAU,EAAE,QAAQ,EAAE,MAAM,SAAS,CAAC;AACxE,YAAY,EAAE,WAAW,EAAE,SAAS,EAAE,UAAU,EAAE,UAAU,EAAE,iBAAiB,EAAE,MAAM,cAAc,CAAC"}
package/dist/index.js ADDED
@@ -0,0 +1,3 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":""}
@@ -0,0 +1,34 @@
1
+ import type { Player, GameAction, GameResult, Platform } from './types';
2
+ export interface PlayerState {
3
+ id: string;
4
+ [key: string]: unknown;
5
+ }
6
+ export interface GameState {
7
+ phase: string;
8
+ players: PlayerState[];
9
+ data: Record<string, unknown>;
10
+ }
11
+ export interface GameConfig {
12
+ id: string;
13
+ name: string;
14
+ description: string;
15
+ coverImage: string;
16
+ minPlayers: number;
17
+ maxPlayers: number;
18
+ tags: string[];
19
+ version: string;
20
+ sdkVersion: string;
21
+ price?: number;
22
+ }
23
+ export interface GameEngine {
24
+ init(players: Player[], options?: Record<string, unknown>): GameState;
25
+ handleAction(state: GameState, playerId: string, action: GameAction): GameState;
26
+ isGameOver(state: GameState): boolean;
27
+ getResult(state: GameState): GameResult;
28
+ getPlayerView(state: GameState, playerId: string): Partial<GameState>;
29
+ }
30
+ export interface GameRendererProps {
31
+ platform: Platform;
32
+ state: Partial<GameState>;
33
+ }
34
+ //# sourceMappingURL=interfaces.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"interfaces.d.ts","sourceRoot":"","sources":["../src/interfaces.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,MAAM,EAAE,UAAU,EAAE,UAAU,EAAE,QAAQ,EAAE,MAAM,SAAS,CAAC;AAExE,MAAM,WAAW,WAAW;IAC1B,EAAE,EAAE,MAAM,CAAC;IACX,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC;CACxB;AAED,MAAM,WAAW,SAAS;IACxB,KAAK,EAAE,MAAM,CAAC;IACd,OAAO,EAAE,WAAW,EAAE,CAAC;IACvB,IAAI,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;CAC/B;AAED,MAAM,WAAW,UAAU;IACzB,EAAE,EAAE,MAAM,CAAC;IACX,IAAI,EAAE,MAAM,CAAC;IACb,WAAW,EAAE,MAAM,CAAC;IACpB,UAAU,EAAE,MAAM,CAAC;IACnB,UAAU,EAAE,MAAM,CAAC;IACnB,UAAU,EAAE,MAAM,CAAC;IACnB,IAAI,EAAE,MAAM,EAAE,CAAC;IACf,OAAO,EAAE,MAAM,CAAC;IAChB,UAAU,EAAE,MAAM,CAAC;IACnB,KAAK,CAAC,EAAE,MAAM,CAAC;CAChB;AAED,MAAM,WAAW,UAAU;IACzB,IAAI,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE,OAAO,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,GAAG,SAAS,CAAC;IACtE,YAAY,CAAC,KAAK,EAAE,SAAS,EAAE,QAAQ,EAAE,MAAM,EAAE,MAAM,EAAE,UAAU,GAAG,SAAS,CAAC;IAChF,UAAU,CAAC,KAAK,EAAE,SAAS,GAAG,OAAO,CAAC;IACtC,SAAS,CAAC,KAAK,EAAE,SAAS,GAAG,UAAU,CAAC;IACxC,aAAa,CAAC,KAAK,EAAE,SAAS,EAAE,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,SAAS,CAAC,CAAC;CACvE;AAED,MAAM,WAAW,iBAAiB;IAChC,QAAQ,EAAE,QAAQ,CAAC;IACnB,KAAK,EAAE,OAAO,CAAC,SAAS,CAAC,CAAC;CAC3B"}
@@ -0,0 +1,3 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ //# sourceMappingURL=interfaces.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"interfaces.js","sourceRoot":"","sources":["../src/interfaces.ts"],"names":[],"mappings":""}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=game-simulator.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"game-simulator.test.d.ts","sourceRoot":"","sources":["../../../src/testing/__tests__/game-simulator.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,81 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const vitest_1 = require("vitest");
4
+ const index_1 = require("../index");
5
+ const counterEngine = {
6
+ init(players) {
7
+ return { phase: 'playing', players: players.map(p => ({ id: p.id })), data: { count: 0, currentPlayerIndex: 0 } };
8
+ },
9
+ handleAction(state, playerId, action) {
10
+ const data = state.data;
11
+ if (action.type !== 'INCREMENT')
12
+ return state;
13
+ const currentPlayer = state.players[data.currentPlayerIndex];
14
+ if (currentPlayer.id !== playerId)
15
+ return state;
16
+ const newCount = data.count + 1;
17
+ if (newCount >= 3) {
18
+ return { ...state, phase: 'ended', data: { count: newCount, currentPlayerIndex: data.currentPlayerIndex, winnerId: playerId } };
19
+ }
20
+ return { ...state, data: { count: newCount, currentPlayerIndex: (data.currentPlayerIndex + 1) % state.players.length } };
21
+ },
22
+ isGameOver(state) { return state.phase === 'ended'; },
23
+ getResult(state) {
24
+ const data = state.data;
25
+ return { rankings: state.players.map(p => ({ playerId: p.id, rank: p.id === data.winnerId ? 1 : 2, score: p.id === data.winnerId ? 10 : 0, isWinner: p.id === data.winnerId })) };
26
+ },
27
+ getPlayerView(state, _playerId) {
28
+ if (state.phase !== 'ended') {
29
+ const { winnerId, ...rest } = state.data;
30
+ return { ...state, data: rest };
31
+ }
32
+ return state;
33
+ },
34
+ };
35
+ (0, vitest_1.describe)('GameSimulator', () => {
36
+ (0, vitest_1.it)('should initialize with player count', () => {
37
+ const sim = new index_1.GameSimulator(counterEngine, { playerCount: 3 });
38
+ sim.start();
39
+ (0, vitest_1.expect)(sim.players).toHaveLength(3);
40
+ (0, vitest_1.expect)(sim.state.phase).toBe('playing');
41
+ });
42
+ (0, vitest_1.it)('should act by player index', () => {
43
+ const sim = new index_1.GameSimulator(counterEngine, { playerCount: 2 });
44
+ sim.start();
45
+ sim.act(0, { type: 'INCREMENT' });
46
+ (0, vitest_1.expect)(sim.state.data.count).toBe(1);
47
+ });
48
+ (0, vitest_1.it)('should return filtered view per player', () => {
49
+ const sim = new index_1.GameSimulator(counterEngine, { playerCount: 2 });
50
+ sim.start();
51
+ const view = sim.getView(0);
52
+ (0, vitest_1.expect)(view.data).not.toHaveProperty('winnerId');
53
+ });
54
+ (0, vitest_1.it)('should track action log', () => {
55
+ const sim = new index_1.GameSimulator(counterEngine, { playerCount: 2 });
56
+ sim.start();
57
+ sim.act(0, { type: 'INCREMENT' });
58
+ sim.act(1, { type: 'INCREMENT' });
59
+ (0, vitest_1.expect)(sim.actionLog).toHaveLength(2);
60
+ (0, vitest_1.expect)(sim.actionLog[0].playerIndex).toBe(0);
61
+ (0, vitest_1.expect)(sim.actionLog[0].action.type).toBe('INCREMENT');
62
+ (0, vitest_1.expect)(sim.actionLog[1].playerIndex).toBe(1);
63
+ });
64
+ (0, vitest_1.it)('should detect game over and get result', () => {
65
+ const sim = new index_1.GameSimulator(counterEngine, { playerCount: 2 });
66
+ sim.start();
67
+ sim.act(0, { type: 'INCREMENT' });
68
+ sim.act(1, { type: 'INCREMENT' });
69
+ sim.act(0, { type: 'INCREMENT' });
70
+ (0, vitest_1.expect)(sim.isGameOver()).toBe(true);
71
+ (0, vitest_1.expect)(sim.getResult().rankings.find((r) => r.isWinner)?.playerId).toBe(sim.players[0].id);
72
+ });
73
+ (0, vitest_1.it)('should expose currentTurn based on currentPlayerIndex', () => {
74
+ const sim = new index_1.GameSimulator(counterEngine, { playerCount: 2 });
75
+ sim.start();
76
+ (0, vitest_1.expect)(sim.currentTurn).toBe(0);
77
+ sim.act(0, { type: 'INCREMENT' });
78
+ (0, vitest_1.expect)(sim.currentTurn).toBe(1);
79
+ });
80
+ });
81
+ //# sourceMappingURL=game-simulator.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"game-simulator.test.js","sourceRoot":"","sources":["../../../src/testing/__tests__/game-simulator.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=game-tester.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"game-tester.test.d.ts","sourceRoot":"","sources":["../../../src/testing/__tests__/game-tester.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,91 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const vitest_1 = require("vitest");
4
+ const index_1 = require("../index");
5
+ // Minimal counter engine for testing GameTester
6
+ const counterEngine = {
7
+ init(players) {
8
+ return {
9
+ phase: 'playing',
10
+ players: players.map(p => ({ id: p.id })),
11
+ data: { count: 0, currentPlayerIndex: 0 },
12
+ };
13
+ },
14
+ handleAction(state, playerId, action) {
15
+ const data = state.data;
16
+ if (action.type !== 'INCREMENT')
17
+ return state;
18
+ const currentPlayer = state.players[data.currentPlayerIndex];
19
+ if (currentPlayer.id !== playerId)
20
+ return state;
21
+ const newCount = data.count + 1;
22
+ if (newCount >= 3) {
23
+ return { ...state, phase: 'ended', data: { count: newCount, currentPlayerIndex: data.currentPlayerIndex, winnerId: playerId } };
24
+ }
25
+ return { ...state, data: { count: newCount, currentPlayerIndex: (data.currentPlayerIndex + 1) % state.players.length } };
26
+ },
27
+ isGameOver(state) { return state.phase === 'ended'; },
28
+ getResult(state) {
29
+ const data = state.data;
30
+ return { rankings: state.players.map(p => ({ playerId: p.id, rank: p.id === data.winnerId ? 1 : 2, score: p.id === data.winnerId ? 10 : 0, isWinner: p.id === data.winnerId })) };
31
+ },
32
+ getPlayerView(state, _playerId) {
33
+ if (state.phase !== 'ended') {
34
+ const { winnerId, ...rest } = state.data;
35
+ return { ...state, data: rest };
36
+ }
37
+ return state;
38
+ },
39
+ };
40
+ (0, vitest_1.describe)('GameTester', () => {
41
+ (0, vitest_1.it)('should initialize with the engine', () => {
42
+ const tester = new index_1.GameTester(counterEngine);
43
+ tester.init((0, index_1.createMockPlayers)(2));
44
+ (0, vitest_1.expect)(tester.state.phase).toBe('playing');
45
+ (0, vitest_1.expect)(tester.phase).toBe('playing');
46
+ (0, vitest_1.expect)(tester.playerStates).toHaveLength(2);
47
+ });
48
+ (0, vitest_1.it)('should apply actions and update state', () => {
49
+ const tester = new index_1.GameTester(counterEngine);
50
+ tester.init((0, index_1.createMockPlayers)(2));
51
+ tester.act('player-1', { type: 'INCREMENT' });
52
+ (0, vitest_1.expect)(tester.state.data.count).toBe(1);
53
+ });
54
+ (0, vitest_1.it)('should ignore invalid actions (wrong player turn)', () => {
55
+ const tester = new index_1.GameTester(counterEngine);
56
+ tester.init((0, index_1.createMockPlayers)(2));
57
+ const before = tester.state;
58
+ tester.act('player-2', { type: 'INCREMENT' });
59
+ (0, vitest_1.expect)(tester.state).toBe(before);
60
+ });
61
+ (0, vitest_1.it)('should detect game over', () => {
62
+ const tester = new index_1.GameTester(counterEngine);
63
+ tester.init((0, index_1.createMockPlayers)(2));
64
+ (0, vitest_1.expect)(tester.isGameOver()).toBe(false);
65
+ tester.act('player-1', { type: 'INCREMENT' });
66
+ tester.act('player-2', { type: 'INCREMENT' });
67
+ tester.act('player-1', { type: 'INCREMENT' });
68
+ (0, vitest_1.expect)(tester.isGameOver()).toBe(true);
69
+ });
70
+ (0, vitest_1.it)('should return game result', () => {
71
+ const tester = new index_1.GameTester(counterEngine);
72
+ tester.init((0, index_1.createMockPlayers)(2));
73
+ tester.act('player-1', { type: 'INCREMENT' });
74
+ tester.act('player-2', { type: 'INCREMENT' });
75
+ tester.act('player-1', { type: 'INCREMENT' });
76
+ const result = tester.getResult();
77
+ (0, vitest_1.expect)(result.rankings.find((r) => r.isWinner)?.playerId).toBe('player-1');
78
+ });
79
+ (0, vitest_1.it)('should return filtered player view', () => {
80
+ const tester = new index_1.GameTester(counterEngine);
81
+ tester.init((0, index_1.createMockPlayers)(2));
82
+ tester.act('player-1', { type: 'INCREMENT' });
83
+ const view = tester.getPlayerView('player-2');
84
+ (0, vitest_1.expect)(view.data).not.toHaveProperty('winnerId');
85
+ });
86
+ (0, vitest_1.it)('should throw if act is called before init', () => {
87
+ const tester = new index_1.GameTester(counterEngine);
88
+ (0, vitest_1.expect)(() => tester.act('p1', { type: 'INCREMENT' })).toThrow();
89
+ });
90
+ });
91
+ //# sourceMappingURL=game-tester.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"game-tester.test.js","sourceRoot":"","sources":["../../../src/testing/__tests__/game-tester.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=mock-players.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mock-players.test.d.ts","sourceRoot":"","sources":["../../../src/testing/__tests__/mock-players.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,39 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const vitest_1 = require("vitest");
4
+ const index_1 = require("../index");
5
+ (0, vitest_1.describe)('createMockPlayers', () => {
6
+ (0, vitest_1.it)('should create the specified number of players', () => {
7
+ const players = (0, index_1.createMockPlayers)(3);
8
+ (0, vitest_1.expect)(players).toHaveLength(3);
9
+ });
10
+ (0, vitest_1.it)('should make the first player the host', () => {
11
+ const players = (0, index_1.createMockPlayers)(2);
12
+ (0, vitest_1.expect)(players[0].isHost).toBe(true);
13
+ (0, vitest_1.expect)(players[1].isHost).toBe(false);
14
+ });
15
+ (0, vitest_1.it)('should generate unique ids', () => {
16
+ const players = (0, index_1.createMockPlayers)(4);
17
+ const ids = players.map(p => p.id);
18
+ (0, vitest_1.expect)(new Set(ids).size).toBe(4);
19
+ });
20
+ (0, vitest_1.it)('should generate default nicknames', () => {
21
+ const players = (0, index_1.createMockPlayers)(2);
22
+ (0, vitest_1.expect)(players[0].nickname).toBe('Player 1');
23
+ (0, vitest_1.expect)(players[1].nickname).toBe('Player 2');
24
+ });
25
+ (0, vitest_1.it)('should accept overrides', () => {
26
+ const players = (0, index_1.createMockPlayers)(2, [
27
+ { nickname: 'Alice' },
28
+ { nickname: 'Bob', isHost: true },
29
+ ]);
30
+ (0, vitest_1.expect)(players[0].nickname).toBe('Alice');
31
+ (0, vitest_1.expect)(players[1].nickname).toBe('Bob');
32
+ (0, vitest_1.expect)(players[1].isHost).toBe(true);
33
+ });
34
+ (0, vitest_1.it)('should set avatarUrl to null by default', () => {
35
+ const players = (0, index_1.createMockPlayers)(1);
36
+ (0, vitest_1.expect)(players[0].avatarUrl).toBeNull();
37
+ });
38
+ });
39
+ //# sourceMappingURL=mock-players.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mock-players.test.js","sourceRoot":"","sources":["../../../src/testing/__tests__/mock-players.test.ts"],"names":[],"mappings":";;AAAA,mCAA8C;AAC9C,oCAA6C;AAE7C,IAAA,iBAAQ,EAAC,mBAAmB,EAAE,GAAG,EAAE;IACjC,IAAA,WAAE,EAAC,+CAA+C,EAAE,GAAG,EAAE;QACvD,MAAM,OAAO,GAAG,IAAA,yBAAiB,EAAC,CAAC,CAAC,CAAC;QACrC,IAAA,eAAM,EAAC,OAAO,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC;IAClC,CAAC,CAAC,CAAC;IAEH,IAAA,WAAE,EAAC,uCAAuC,EAAE,GAAG,EAAE;QAC/C,MAAM,OAAO,GAAG,IAAA,yBAAiB,EAAC,CAAC,CAAC,CAAC;QACrC,IAAA,eAAM,EAAC,OAAO,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACrC,IAAA,eAAM,EAAC,OAAO,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IACxC,CAAC,CAAC,CAAC;IAEH,IAAA,WAAE,EAAC,4BAA4B,EAAE,GAAG,EAAE;QACpC,MAAM,OAAO,GAAG,IAAA,yBAAiB,EAAC,CAAC,CAAC,CAAC;QACrC,MAAM,GAAG,GAAG,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;QACnC,IAAA,eAAM,EAAC,IAAI,GAAG,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IACpC,CAAC,CAAC,CAAC;IAEH,IAAA,WAAE,EAAC,mCAAmC,EAAE,GAAG,EAAE;QAC3C,MAAM,OAAO,GAAG,IAAA,yBAAiB,EAAC,CAAC,CAAC,CAAC;QACrC,IAAA,eAAM,EAAC,OAAO,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QAC7C,IAAA,eAAM,EAAC,OAAO,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IAC/C,CAAC,CAAC,CAAC;IAEH,IAAA,WAAE,EAAC,yBAAyB,EAAE,GAAG,EAAE;QACjC,MAAM,OAAO,GAAG,IAAA,yBAAiB,EAAC,CAAC,EAAE;YACnC,EAAE,QAAQ,EAAE,OAAO,EAAE;YACrB,EAAE,QAAQ,EAAE,KAAK,EAAE,MAAM,EAAE,IAAI,EAAE;SAClC,CAAC,CAAC;QACH,IAAA,eAAM,EAAC,OAAO,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAC1C,IAAA,eAAM,EAAC,OAAO,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACxC,IAAA,eAAM,EAAC,OAAO,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACvC,CAAC,CAAC,CAAC;IAEH,IAAA,WAAE,EAAC,yCAAyC,EAAE,GAAG,EAAE;QACjD,MAAM,OAAO,GAAG,IAAA,yBAAiB,EAAC,CAAC,CAAC,CAAC;QACrC,IAAA,eAAM,EAAC,OAAO,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,QAAQ,EAAE,CAAC;IAC1C,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
@@ -0,0 +1,29 @@
1
+ import type { GameEngine, GameState } from '../interfaces';
2
+ import type { Player, GameAction, GameResult } from '../types';
3
+ export interface ActionLogEntry {
4
+ playerIndex: number;
5
+ playerId: string;
6
+ action: GameAction;
7
+ stateBefore: GameState;
8
+ stateAfter: GameState;
9
+ }
10
+ export declare class GameSimulator {
11
+ private engine;
12
+ private playerCount;
13
+ private _players;
14
+ private _state;
15
+ private _actionLog;
16
+ constructor(engine: GameEngine, options: {
17
+ playerCount: number;
18
+ });
19
+ get players(): Player[];
20
+ get state(): GameState;
21
+ get actionLog(): ActionLogEntry[];
22
+ get currentTurn(): number;
23
+ start(options?: Record<string, unknown>): void;
24
+ act(playerIndex: number, action: GameAction): void;
25
+ getView(playerIndex: number): Partial<GameState>;
26
+ isGameOver(): boolean;
27
+ getResult(): GameResult;
28
+ }
29
+ //# sourceMappingURL=game-simulator.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"game-simulator.d.ts","sourceRoot":"","sources":["../../src/testing/game-simulator.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,UAAU,EAAE,SAAS,EAAE,MAAM,eAAe,CAAC;AAC3D,OAAO,KAAK,EAAE,MAAM,EAAE,UAAU,EAAE,UAAU,EAAE,MAAM,UAAU,CAAC;AAG/D,MAAM,WAAW,cAAc;IAC7B,WAAW,EAAE,MAAM,CAAC;IACpB,QAAQ,EAAE,MAAM,CAAC;IACjB,MAAM,EAAE,UAAU,CAAC;IACnB,WAAW,EAAE,SAAS,CAAC;IACvB,UAAU,EAAE,SAAS,CAAC;CACvB;AAED,qBAAa,aAAa;IACxB,OAAO,CAAC,MAAM,CAAa;IAC3B,OAAO,CAAC,WAAW,CAAS;IAC5B,OAAO,CAAC,QAAQ,CAAgB;IAChC,OAAO,CAAC,MAAM,CAA0B;IACxC,OAAO,CAAC,UAAU,CAAwB;gBAE9B,MAAM,EAAE,UAAU,EAAE,OAAO,EAAE;QAAE,WAAW,EAAE,MAAM,CAAA;KAAE;IAKhE,IAAI,OAAO,IAAI,MAAM,EAAE,CAA0B;IAEjD,IAAI,KAAK,IAAI,SAAS,CAGrB;IAED,IAAI,SAAS,IAAI,cAAc,EAAE,CAA4B;IAE7D,IAAI,WAAW,IAAI,MAAM,CAGxB;IAED,KAAK,CAAC,OAAO,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,GAAG,IAAI;IAM9C,GAAG,CAAC,WAAW,EAAE,MAAM,EAAE,MAAM,EAAE,UAAU,GAAG,IAAI;IAWlD,OAAO,CAAC,WAAW,EAAE,MAAM,GAAG,OAAO,CAAC,SAAS,CAAC;IAOhD,UAAU,IAAI,OAAO;IAErB,SAAS,IAAI,UAAU;CACxB"}
@@ -0,0 +1,52 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.GameSimulator = void 0;
4
+ const mock_players_1 = require("./mock-players");
5
+ class GameSimulator {
6
+ engine;
7
+ playerCount;
8
+ _players = [];
9
+ _state = null;
10
+ _actionLog = [];
11
+ constructor(engine, options) {
12
+ this.engine = engine;
13
+ this.playerCount = options.playerCount;
14
+ }
15
+ get players() { return this._players; }
16
+ get state() {
17
+ if (!this._state)
18
+ throw new Error('GameSimulator: call start() first');
19
+ return this._state;
20
+ }
21
+ get actionLog() { return this._actionLog; }
22
+ get currentTurn() {
23
+ const data = this.state.data;
24
+ return data.currentPlayerIndex ?? 0;
25
+ }
26
+ start(options) {
27
+ this._players = (0, mock_players_1.createMockPlayers)(this.playerCount);
28
+ this._state = this.engine.init(this._players, options);
29
+ this._actionLog = [];
30
+ }
31
+ act(playerIndex, action) {
32
+ if (!this._state)
33
+ throw new Error('GameSimulator: call start() first');
34
+ if (playerIndex < 0 || playerIndex >= this._players.length) {
35
+ throw new Error(`GameSimulator: invalid playerIndex ${playerIndex}`);
36
+ }
37
+ const playerId = this._players[playerIndex].id;
38
+ const stateBefore = this._state;
39
+ this._state = this.engine.handleAction(this._state, playerId, action);
40
+ this._actionLog.push({ playerIndex, playerId, action, stateBefore, stateAfter: this._state });
41
+ }
42
+ getView(playerIndex) {
43
+ if (playerIndex < 0 || playerIndex >= this._players.length) {
44
+ throw new Error(`GameSimulator: invalid playerIndex ${playerIndex}`);
45
+ }
46
+ return this.engine.getPlayerView(this.state, this._players[playerIndex].id);
47
+ }
48
+ isGameOver() { return this.engine.isGameOver(this.state); }
49
+ getResult() { return this.engine.getResult(this.state); }
50
+ }
51
+ exports.GameSimulator = GameSimulator;
52
+ //# sourceMappingURL=game-simulator.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"game-simulator.js","sourceRoot":"","sources":["../../src/testing/game-simulator.ts"],"names":[],"mappings":";;;AAEA,iDAAmD;AAUnD,MAAa,aAAa;IAChB,MAAM,CAAa;IACnB,WAAW,CAAS;IACpB,QAAQ,GAAa,EAAE,CAAC;IACxB,MAAM,GAAqB,IAAI,CAAC;IAChC,UAAU,GAAqB,EAAE,CAAC;IAE1C,YAAY,MAAkB,EAAE,OAAgC;QAC9D,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC;QACrB,IAAI,CAAC,WAAW,GAAG,OAAO,CAAC,WAAW,CAAC;IACzC,CAAC;IAED,IAAI,OAAO,KAAe,OAAO,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC;IAEjD,IAAI,KAAK;QACP,IAAI,CAAC,IAAI,CAAC,MAAM;YAAE,MAAM,IAAI,KAAK,CAAC,mCAAmC,CAAC,CAAC;QACvE,OAAO,IAAI,CAAC,MAAM,CAAC;IACrB,CAAC;IAED,IAAI,SAAS,KAAuB,OAAO,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC;IAE7D,IAAI,WAAW;QACb,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,IAAuC,CAAC;QAChE,OAAO,IAAI,CAAC,kBAAkB,IAAI,CAAC,CAAC;IACtC,CAAC;IAED,KAAK,CAAC,OAAiC;QACrC,IAAI,CAAC,QAAQ,GAAG,IAAA,gCAAiB,EAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QACpD,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;QACvD,IAAI,CAAC,UAAU,GAAG,EAAE,CAAC;IACvB,CAAC;IAED,GAAG,CAAC,WAAmB,EAAE,MAAkB;QACzC,IAAI,CAAC,IAAI,CAAC,MAAM;YAAE,MAAM,IAAI,KAAK,CAAC,mCAAmC,CAAC,CAAC;QACvE,IAAI,WAAW,GAAG,CAAC,IAAI,WAAW,IAAI,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC;YAC3D,MAAM,IAAI,KAAK,CAAC,sCAAsC,WAAW,EAAE,CAAC,CAAC;QACvE,CAAC;QACD,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,CAAC,EAAE,CAAC;QAC/C,MAAM,WAAW,GAAG,IAAI,CAAC,MAAM,CAAC;QAChC,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,IAAI,CAAC,MAAM,EAAE,QAAQ,EAAE,MAAM,CAAC,CAAC;QACtE,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,EAAE,WAAW,EAAE,QAAQ,EAAE,MAAM,EAAE,WAAW,EAAE,UAAU,EAAE,IAAI,CAAC,MAAM,EAAE,CAAC,CAAC;IAChG,CAAC;IAED,OAAO,CAAC,WAAmB;QACzB,IAAI,WAAW,GAAG,CAAC,IAAI,WAAW,IAAI,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC;YAC3D,MAAM,IAAI,KAAK,CAAC,sCAAsC,WAAW,EAAE,CAAC,CAAC;QACvE,CAAC;QACD,OAAO,IAAI,CAAC,MAAM,CAAC,aAAa,CAAC,IAAI,CAAC,KAAK,EAAE,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,CAAC,EAAE,CAAC,CAAC;IAC9E,CAAC;IAED,UAAU,KAAc,OAAO,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;IAEpE,SAAS,KAAiB,OAAO,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;CACtE;AArDD,sCAqDC"}
@@ -0,0 +1,16 @@
1
+ import type { GameEngine, GameState } from '../interfaces';
2
+ import type { Player, GameAction, GameResult } from '../types';
3
+ export declare class GameTester {
4
+ private engine;
5
+ private _state;
6
+ constructor(engine: GameEngine);
7
+ get state(): GameState;
8
+ get phase(): string;
9
+ get playerStates(): import("../interfaces").PlayerState[];
10
+ init(players: Player[], options?: Record<string, unknown>): GameState;
11
+ act(playerId: string, action: GameAction): GameState;
12
+ isGameOver(): boolean;
13
+ getResult(): GameResult;
14
+ getPlayerView(playerId: string): Partial<GameState>;
15
+ }
16
+ //# sourceMappingURL=game-tester.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"game-tester.d.ts","sourceRoot":"","sources":["../../src/testing/game-tester.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,UAAU,EAAE,SAAS,EAAE,MAAM,eAAe,CAAC;AAC3D,OAAO,KAAK,EAAE,MAAM,EAAE,UAAU,EAAE,UAAU,EAAE,MAAM,UAAU,CAAC;AAE/D,qBAAa,UAAU;IACrB,OAAO,CAAC,MAAM,CAAa;IAC3B,OAAO,CAAC,MAAM,CAA0B;gBAE5B,MAAM,EAAE,UAAU;IAI9B,IAAI,KAAK,IAAI,SAAS,CAGrB;IAED,IAAI,KAAK,IAAI,MAAM,CAElB;IAED,IAAI,YAAY,0CAEf;IAED,IAAI,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE,OAAO,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,GAAG,SAAS;IAKrE,GAAG,CAAC,QAAQ,EAAE,MAAM,EAAE,MAAM,EAAE,UAAU,GAAG,SAAS;IAMpD,UAAU,IAAI,OAAO;IAIrB,SAAS,IAAI,UAAU;IAIvB,aAAa,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,SAAS,CAAC;CAGpD"}
@@ -0,0 +1,42 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.GameTester = void 0;
4
+ class GameTester {
5
+ engine;
6
+ _state = null;
7
+ constructor(engine) {
8
+ this.engine = engine;
9
+ }
10
+ get state() {
11
+ if (!this._state)
12
+ throw new Error('GameTester: call init() before accessing state');
13
+ return this._state;
14
+ }
15
+ get phase() {
16
+ return this.state.phase;
17
+ }
18
+ get playerStates() {
19
+ return this.state.players;
20
+ }
21
+ init(players, options) {
22
+ this._state = this.engine.init(players, options);
23
+ return this._state;
24
+ }
25
+ act(playerId, action) {
26
+ if (!this._state)
27
+ throw new Error('GameTester: call init() before act()');
28
+ this._state = this.engine.handleAction(this._state, playerId, action);
29
+ return this._state;
30
+ }
31
+ isGameOver() {
32
+ return this.engine.isGameOver(this.state);
33
+ }
34
+ getResult() {
35
+ return this.engine.getResult(this.state);
36
+ }
37
+ getPlayerView(playerId) {
38
+ return this.engine.getPlayerView(this.state, playerId);
39
+ }
40
+ }
41
+ exports.GameTester = GameTester;
42
+ //# sourceMappingURL=game-tester.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"game-tester.js","sourceRoot":"","sources":["../../src/testing/game-tester.ts"],"names":[],"mappings":";;;AAGA,MAAa,UAAU;IACb,MAAM,CAAa;IACnB,MAAM,GAAqB,IAAI,CAAC;IAExC,YAAY,MAAkB;QAC5B,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC;IACvB,CAAC;IAED,IAAI,KAAK;QACP,IAAI,CAAC,IAAI,CAAC,MAAM;YAAE,MAAM,IAAI,KAAK,CAAC,gDAAgD,CAAC,CAAC;QACpF,OAAO,IAAI,CAAC,MAAM,CAAC;IACrB,CAAC;IAED,IAAI,KAAK;QACP,OAAO,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC;IAC1B,CAAC;IAED,IAAI,YAAY;QACd,OAAO,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC;IAC5B,CAAC;IAED,IAAI,CAAC,OAAiB,EAAE,OAAiC;QACvD,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;QACjD,OAAO,IAAI,CAAC,MAAM,CAAC;IACrB,CAAC;IAED,GAAG,CAAC,QAAgB,EAAE,MAAkB;QACtC,IAAI,CAAC,IAAI,CAAC,MAAM;YAAE,MAAM,IAAI,KAAK,CAAC,sCAAsC,CAAC,CAAC;QAC1E,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,IAAI,CAAC,MAAM,EAAE,QAAQ,EAAE,MAAM,CAAC,CAAC;QACtE,OAAO,IAAI,CAAC,MAAM,CAAC;IACrB,CAAC;IAED,UAAU;QACR,OAAO,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAC5C,CAAC;IAED,SAAS;QACP,OAAO,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAC3C,CAAC;IAED,aAAa,CAAC,QAAgB;QAC5B,OAAO,IAAI,CAAC,MAAM,CAAC,aAAa,CAAC,IAAI,CAAC,KAAK,EAAE,QAAQ,CAAC,CAAC;IACzD,CAAC;CACF;AA3CD,gCA2CC"}
@@ -0,0 +1,5 @@
1
+ export { createMockPlayers } from './mock-players';
2
+ export { GameTester } from './game-tester';
3
+ export { GameSimulator } from './game-simulator';
4
+ export type { ActionLogEntry } from './game-simulator';
5
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/testing/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,iBAAiB,EAAE,MAAM,gBAAgB,CAAC;AACnD,OAAO,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAC3C,OAAO,EAAE,aAAa,EAAE,MAAM,kBAAkB,CAAC;AACjD,YAAY,EAAE,cAAc,EAAE,MAAM,kBAAkB,CAAC"}
@@ -0,0 +1,10 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.GameSimulator = exports.GameTester = exports.createMockPlayers = void 0;
4
+ var mock_players_1 = require("./mock-players");
5
+ Object.defineProperty(exports, "createMockPlayers", { enumerable: true, get: function () { return mock_players_1.createMockPlayers; } });
6
+ var game_tester_1 = require("./game-tester");
7
+ Object.defineProperty(exports, "GameTester", { enumerable: true, get: function () { return game_tester_1.GameTester; } });
8
+ var game_simulator_1 = require("./game-simulator");
9
+ Object.defineProperty(exports, "GameSimulator", { enumerable: true, get: function () { return game_simulator_1.GameSimulator; } });
10
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/testing/index.ts"],"names":[],"mappings":";;;AAAA,+CAAmD;AAA1C,iHAAA,iBAAiB,OAAA;AAC1B,6CAA2C;AAAlC,yGAAA,UAAU,OAAA;AACnB,mDAAiD;AAAxC,+GAAA,aAAa,OAAA"}
@@ -0,0 +1,3 @@
1
+ import type { Player } from '../types';
2
+ export declare function createMockPlayers(count: number, overrides?: Partial<Player>[]): Player[];
3
+ //# sourceMappingURL=mock-players.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mock-players.d.ts","sourceRoot":"","sources":["../../src/testing/mock-players.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,UAAU,CAAC;AAEvC,wBAAgB,iBAAiB,CAC/B,KAAK,EAAE,MAAM,EACb,SAAS,CAAC,EAAE,OAAO,CAAC,MAAM,CAAC,EAAE,GAC5B,MAAM,EAAE,CAQV"}
@@ -0,0 +1,13 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.createMockPlayers = createMockPlayers;
4
+ function createMockPlayers(count, overrides) {
5
+ return Array.from({ length: count }, (_, i) => ({
6
+ id: `player-${i + 1}`,
7
+ nickname: `Player ${i + 1}`,
8
+ avatarUrl: null,
9
+ isHost: i === 0,
10
+ ...overrides?.[i],
11
+ }));
12
+ }
13
+ //# sourceMappingURL=mock-players.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mock-players.js","sourceRoot":"","sources":["../../src/testing/mock-players.ts"],"names":[],"mappings":";;AAEA,8CAWC;AAXD,SAAgB,iBAAiB,CAC/B,KAAa,EACb,SAA6B;IAE7B,OAAO,KAAK,CAAC,IAAI,CAAC,EAAE,MAAM,EAAE,KAAK,EAAE,EAAE,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC;QAC9C,EAAE,EAAE,UAAU,CAAC,GAAG,CAAC,EAAE;QACrB,QAAQ,EAAE,UAAU,CAAC,GAAG,CAAC,EAAE;QAC3B,SAAS,EAAE,IAAI;QACf,MAAM,EAAE,CAAC,KAAK,CAAC;QACf,GAAG,SAAS,EAAE,CAAC,CAAC,CAAC;KAClB,CAAC,CAAC,CAAC;AACN,CAAC"}
@@ -0,0 +1,28 @@
1
+ export interface Player {
2
+ id: string;
3
+ nickname: string;
4
+ avatarUrl: string | null;
5
+ isHost: boolean;
6
+ }
7
+ export interface GameAction {
8
+ type: string;
9
+ payload?: Record<string, unknown>;
10
+ }
11
+ export interface GameResult {
12
+ rankings: {
13
+ playerId: string;
14
+ rank: number;
15
+ score: number;
16
+ isWinner: boolean;
17
+ }[];
18
+ data?: Record<string, unknown>;
19
+ }
20
+ export interface Platform {
21
+ getPlayers(): Player[];
22
+ getLocalPlayer(): Player;
23
+ send(action: GameAction): void;
24
+ on(event: string, handler: (...args: unknown[]) => void): void;
25
+ off(event: string, handler: (...args: unknown[]) => void): void;
26
+ reportResult(result: GameResult): void;
27
+ }
28
+ //# sourceMappingURL=types.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../src/types.ts"],"names":[],"mappings":"AAAA,MAAM,WAAW,MAAM;IACrB,EAAE,EAAE,MAAM,CAAC;IACX,QAAQ,EAAE,MAAM,CAAC;IACjB,SAAS,EAAE,MAAM,GAAG,IAAI,CAAC;IACzB,MAAM,EAAE,OAAO,CAAC;CACjB;AAED,MAAM,WAAW,UAAU;IACzB,IAAI,EAAE,MAAM,CAAC;IACb,OAAO,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;CACnC;AAED,MAAM,WAAW,UAAU;IACzB,QAAQ,EAAE;QACR,QAAQ,EAAE,MAAM,CAAC;QACjB,IAAI,EAAE,MAAM,CAAC;QACb,KAAK,EAAE,MAAM,CAAC;QACd,QAAQ,EAAE,OAAO,CAAC;KACnB,EAAE,CAAC;IACJ,IAAI,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;CAChC;AAED,MAAM,WAAW,QAAQ;IACvB,UAAU,IAAI,MAAM,EAAE,CAAC;IACvB,cAAc,IAAI,MAAM,CAAC;IACzB,IAAI,CAAC,MAAM,EAAE,UAAU,GAAG,IAAI,CAAC;IAC/B,EAAE,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,EAAE,CAAC,GAAG,IAAI,EAAE,OAAO,EAAE,KAAK,IAAI,GAAG,IAAI,CAAC;IAC/D,GAAG,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,EAAE,CAAC,GAAG,IAAI,EAAE,OAAO,EAAE,KAAK,IAAI,GAAG,IAAI,CAAC;IAChE,YAAY,CAAC,MAAM,EAAE,UAAU,GAAG,IAAI,CAAC;CACxC"}
package/dist/types.js ADDED
@@ -0,0 +1,3 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../src/types.ts"],"names":[],"mappings":""}
package/package.json ADDED
@@ -0,0 +1,37 @@
1
+ {
2
+ "name": "@littlepartytime/sdk",
3
+ "version": "1.0.0",
4
+ "description": "Game SDK for Little Party Time platform - type definitions and testing utilities",
5
+ "main": "./dist/index.js",
6
+ "types": "./dist/index.d.ts",
7
+ "exports": {
8
+ ".": {
9
+ "types": "./dist/index.d.ts",
10
+ "require": "./dist/index.js",
11
+ "import": "./dist/index.js"
12
+ },
13
+ "./testing": {
14
+ "types": "./dist/testing/index.d.ts",
15
+ "require": "./dist/testing/index.js",
16
+ "import": "./dist/testing/index.js"
17
+ }
18
+ },
19
+ "scripts": {
20
+ "build": "tsc",
21
+ "dev": "tsc --watch",
22
+ "test": "vitest run",
23
+ "test:watch": "vitest"
24
+ },
25
+ "files": ["dist"],
26
+ "keywords": ["boardgame", "sdk", "game-engine", "littlepartytime"],
27
+ "license": "MIT",
28
+ "repository": {
29
+ "type": "git",
30
+ "url": "https://github.com/chesterli710/littlepartytime-sdk.git",
31
+ "directory": "packages/sdk"
32
+ },
33
+ "devDependencies": {
34
+ "typescript": "^5",
35
+ "vitest": "^3"
36
+ }
37
+ }