@1upmonster/duel 0.1.8 → 0.2.2

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 (186) hide show
  1. package/README.md +110 -59
  2. package/dist/admin.d.ts +25 -40
  3. package/dist/admin.js +93 -114
  4. package/dist/encryption.js +1 -2
  5. package/dist/generated/duel/accounts/index.d.ts +10 -0
  6. package/dist/generated/duel/accounts/index.js +11 -0
  7. package/dist/generated/duel/accounts/matchTicket.d.ts +38 -0
  8. package/dist/generated/duel/accounts/matchTicket.js +45 -0
  9. package/dist/generated/duel/accounts/queue.d.ts +42 -0
  10. package/dist/generated/duel/accounts/queue.js +45 -0
  11. package/dist/generated/duel/accounts/tenant.d.ts +41 -0
  12. package/dist/generated/duel/accounts/tenant.js +44 -0
  13. package/dist/generated/duel/errors/duel.d.ts +31 -0
  14. package/dist/generated/duel/errors/duel.js +35 -0
  15. package/dist/generated/duel/errors/index.d.ts +8 -0
  16. package/dist/generated/duel/errors/index.js +9 -0
  17. package/dist/generated/duel/index.d.ts +12 -0
  18. package/dist/generated/duel/index.js +13 -0
  19. package/dist/generated/duel/instructions/cancelTicket.d.ts +45 -0
  20. package/dist/generated/duel/instructions/cancelTicket.js +56 -0
  21. package/dist/generated/duel/instructions/closeTicket.d.ts +45 -0
  22. package/dist/generated/duel/instructions/closeTicket.js +56 -0
  23. package/dist/generated/duel/instructions/commitTickets.d.ts +39 -0
  24. package/dist/generated/duel/instructions/commitTickets.js +50 -0
  25. package/dist/generated/duel/instructions/createTicket.d.ts +48 -0
  26. package/dist/generated/duel/instructions/createTicket.js +63 -0
  27. package/dist/generated/duel/instructions/delegateQueue.d.ts +69 -0
  28. package/dist/generated/duel/instructions/delegateQueue.js +90 -0
  29. package/dist/generated/duel/instructions/delegateTicket.d.ts +69 -0
  30. package/dist/generated/duel/instructions/delegateTicket.js +90 -0
  31. package/dist/generated/duel/instructions/flushMatches.d.ts +37 -0
  32. package/dist/generated/duel/instructions/flushMatches.js +43 -0
  33. package/dist/generated/duel/instructions/index.d.ts +19 -0
  34. package/dist/generated/duel/instructions/index.js +20 -0
  35. package/dist/generated/duel/instructions/initializeQueue.d.ts +48 -0
  36. package/dist/generated/duel/instructions/initializeQueue.js +63 -0
  37. package/dist/generated/duel/instructions/initializeTenant.d.ts +70 -0
  38. package/dist/generated/duel/instructions/initializeTenant.js +67 -0
  39. package/dist/generated/duel/instructions/joinQueue.d.ts +51 -0
  40. package/dist/generated/duel/instructions/joinQueue.js +56 -0
  41. package/dist/generated/duel/instructions/processUndelegation.d.ts +43 -0
  42. package/dist/generated/duel/instructions/processUndelegation.js +49 -0
  43. package/dist/generated/duel/instructions/setupTicketPermission.d.ts +54 -0
  44. package/dist/generated/duel/instructions/setupTicketPermission.js +63 -0
  45. package/dist/generated/duel/programs/duel.d.ts +92 -0
  46. package/dist/generated/duel/programs/duel.js +146 -0
  47. package/dist/generated/duel/programs/index.d.ts +8 -0
  48. package/dist/generated/duel/programs/index.js +9 -0
  49. package/dist/generated/duel/types/accountType.d.ts +25 -0
  50. package/dist/generated/duel/types/accountType.js +25 -0
  51. package/dist/generated/duel/types/index.d.ts +13 -0
  52. package/dist/generated/duel/types/index.js +14 -0
  53. package/dist/generated/duel/types/matchEntry.d.ts +23 -0
  54. package/dist/generated/duel/types/matchEntry.js +18 -0
  55. package/dist/generated/duel/types/matchFound.d.ts +23 -0
  56. package/dist/generated/duel/types/matchFound.js +18 -0
  57. package/dist/generated/duel/types/pendingMatch.d.ts +21 -0
  58. package/dist/generated/duel/types/pendingMatch.js +18 -0
  59. package/dist/generated/duel/types/queueEntry.d.ts +19 -0
  60. package/dist/generated/duel/types/queueEntry.js +18 -0
  61. package/dist/generated/duel/types/ticketStatus.d.ts +40 -0
  62. package/dist/generated/duel/types/ticketStatus.js +25 -0
  63. package/dist/generated/rps-game/accounts/gameSession.d.ts +40 -0
  64. package/dist/generated/rps-game/accounts/gameSession.js +45 -0
  65. package/dist/generated/rps-game/accounts/index.d.ts +9 -0
  66. package/dist/generated/rps-game/accounts/index.js +10 -0
  67. package/dist/generated/rps-game/accounts/playerProfile.d.ts +36 -0
  68. package/dist/generated/rps-game/accounts/playerProfile.js +47 -0
  69. package/dist/generated/rps-game/errors/index.d.ts +8 -0
  70. package/dist/generated/rps-game/errors/index.js +9 -0
  71. package/dist/generated/rps-game/errors/rpsGame.d.ts +25 -0
  72. package/dist/generated/rps-game/errors/rpsGame.js +29 -0
  73. package/dist/generated/rps-game/index.d.ts +12 -0
  74. package/dist/generated/rps-game/index.js +13 -0
  75. package/dist/generated/rps-game/instructions/closePlayer.d.ts +45 -0
  76. package/dist/generated/rps-game/instructions/closePlayer.js +56 -0
  77. package/dist/generated/rps-game/instructions/delegatePda.d.ts +69 -0
  78. package/dist/generated/rps-game/instructions/delegatePda.js +90 -0
  79. package/dist/generated/rps-game/instructions/index.d.ts +16 -0
  80. package/dist/generated/rps-game/instructions/index.js +17 -0
  81. package/dist/generated/rps-game/instructions/initializePlayer.d.ts +48 -0
  82. package/dist/generated/rps-game/instructions/initializePlayer.js +63 -0
  83. package/dist/generated/rps-game/instructions/makeChoice.d.ts +44 -0
  84. package/dist/generated/rps-game/instructions/makeChoice.js +46 -0
  85. package/dist/generated/rps-game/instructions/onMatchFound.d.ts +43 -0
  86. package/dist/generated/rps-game/instructions/onMatchFound.js +45 -0
  87. package/dist/generated/rps-game/instructions/persistResults.d.ts +43 -0
  88. package/dist/generated/rps-game/instructions/persistResults.js +50 -0
  89. package/dist/generated/rps-game/instructions/processUndelegation.d.ts +43 -0
  90. package/dist/generated/rps-game/instructions/processUndelegation.js +49 -0
  91. package/dist/generated/rps-game/instructions/startGame.d.ts +54 -0
  92. package/dist/generated/rps-game/instructions/startGame.js +67 -0
  93. package/dist/generated/rps-game/instructions/startGameWithTicket.d.ts +57 -0
  94. package/dist/generated/rps-game/instructions/startGameWithTicket.js +67 -0
  95. package/dist/generated/rps-game/programs/index.d.ts +8 -0
  96. package/dist/generated/rps-game/programs/index.js +9 -0
  97. package/dist/generated/rps-game/programs/rpsGame.d.ts +78 -0
  98. package/dist/generated/rps-game/programs/rpsGame.js +118 -0
  99. package/dist/generated/rps-game/types/accountType.d.ts +34 -0
  100. package/dist/generated/rps-game/types/accountType.js +25 -0
  101. package/dist/generated/rps-game/types/choice.d.ts +17 -0
  102. package/dist/generated/rps-game/types/choice.js +25 -0
  103. package/dist/generated/rps-game/types/gameResult.d.ts +26 -0
  104. package/dist/generated/rps-game/types/gameResult.js +25 -0
  105. package/dist/generated/rps-game/types/index.d.ts +10 -0
  106. package/dist/generated/rps-game/types/index.js +11 -0
  107. package/dist/index.d.ts +2 -1
  108. package/dist/index.js +3 -2
  109. package/dist/player.d.ts +25 -46
  110. package/dist/player.js +81 -140
  111. package/dist/tee.d.ts +14 -0
  112. package/dist/tee.js +62 -0
  113. package/dist/transaction.d.ts +11 -0
  114. package/dist/transaction.js +50 -0
  115. package/dist/utils.d.ts +4 -4
  116. package/dist/utils.js +23 -8
  117. package/package.json +3 -6
  118. package/src/admin.ts +151 -161
  119. package/src/duel.json +66 -7
  120. package/src/encryption.ts +0 -3
  121. package/src/generated/duel/accounts/index.ts +11 -0
  122. package/src/generated/duel/accounts/matchTicket.ts +77 -0
  123. package/src/generated/duel/accounts/queue.ts +77 -0
  124. package/src/generated/duel/accounts/tenant.ts +76 -0
  125. package/src/generated/duel/errors/duel.ts +46 -0
  126. package/src/generated/duel/errors/index.ts +9 -0
  127. package/src/generated/duel/index.ts +13 -0
  128. package/src/generated/duel/instructions/cancelTicket.ts +100 -0
  129. package/src/generated/duel/instructions/closeTicket.ts +100 -0
  130. package/src/generated/duel/instructions/commitTickets.ts +84 -0
  131. package/src/generated/duel/instructions/createTicket.ts +109 -0
  132. package/src/generated/duel/instructions/delegateQueue.ts +157 -0
  133. package/src/generated/duel/instructions/delegateTicket.ts +157 -0
  134. package/src/generated/duel/instructions/flushMatches.ts +76 -0
  135. package/src/generated/duel/instructions/index.ts +20 -0
  136. package/src/generated/duel/instructions/initializeQueue.ts +109 -0
  137. package/src/generated/duel/instructions/initializeTenant.ts +126 -0
  138. package/src/generated/duel/instructions/joinQueue.ts +106 -0
  139. package/src/generated/duel/instructions/processUndelegation.ts +86 -0
  140. package/src/generated/duel/instructions/setupTicketPermission.ts +115 -0
  141. package/src/generated/duel/programs/duel.ts +108 -0
  142. package/src/generated/duel/programs/index.ts +9 -0
  143. package/src/generated/duel/types/accountType.ts +36 -0
  144. package/src/generated/duel/types/index.ts +14 -0
  145. package/src/generated/duel/types/matchEntry.ts +25 -0
  146. package/src/generated/duel/types/matchFound.ts +25 -0
  147. package/src/generated/duel/types/pendingMatch.ts +25 -0
  148. package/src/generated/duel/types/queueEntry.ts +25 -0
  149. package/src/generated/duel/types/ticketStatus.ts +38 -0
  150. package/src/generated/rps-game/accounts/gameSession.ts +77 -0
  151. package/src/generated/rps-game/accounts/index.ts +10 -0
  152. package/src/generated/rps-game/accounts/playerProfile.ts +80 -0
  153. package/src/generated/rps-game/errors/index.ts +9 -0
  154. package/src/generated/rps-game/errors/rpsGame.ts +40 -0
  155. package/src/generated/rps-game/index.ts +13 -0
  156. package/src/generated/rps-game/instructions/closePlayer.ts +100 -0
  157. package/src/generated/rps-game/instructions/delegatePda.ts +157 -0
  158. package/src/generated/rps-game/instructions/index.ts +17 -0
  159. package/src/generated/rps-game/instructions/initializePlayer.ts +109 -0
  160. package/src/generated/rps-game/instructions/makeChoice.ts +84 -0
  161. package/src/generated/rps-game/instructions/onMatchFound.ts +79 -0
  162. package/src/generated/rps-game/instructions/persistResults.ts +88 -0
  163. package/src/generated/rps-game/instructions/processUndelegation.ts +86 -0
  164. package/src/generated/rps-game/instructions/startGame.ts +118 -0
  165. package/src/generated/rps-game/instructions/startGameWithTicket.ts +121 -0
  166. package/src/generated/rps-game/programs/index.ts +9 -0
  167. package/src/generated/rps-game/programs/rpsGame.ts +95 -0
  168. package/src/generated/rps-game/types/accountType.ts +36 -0
  169. package/src/generated/rps-game/types/choice.ts +25 -0
  170. package/src/generated/rps-game/types/gameResult.ts +37 -0
  171. package/src/generated/rps-game/types/index.ts +11 -0
  172. package/src/index.ts +2 -1
  173. package/src/player.ts +129 -192
  174. package/src/{idl/private_matchmaking.json → rps_game.json} +547 -583
  175. package/src/tee.ts +79 -0
  176. package/src/transaction.ts +90 -0
  177. package/src/utils.ts +35 -20
  178. package/tsconfig.json +2 -2
  179. package/dist/client.d.ts +0 -54
  180. package/dist/client.js +0 -265
  181. package/dist/duel.json +0 -1207
  182. package/dist/private_matchmaking.json +0 -534
  183. package/dist/types.d.ts +0 -635
  184. package/dist/types.js +0 -2
  185. package/src/idl/private_matchmaking.ts +0 -1033
  186. package/src/types.ts +0 -300
package/README.md CHANGED
@@ -1,13 +1,8 @@
1
1
  # @1upmonster/duel
2
2
 
3
- TypeScript SDK for **Duel** - a privacy-focused matchmaking protocol on Solana using MagicBlock's Ephemeral Rollups (TEE).
3
+ TypeScript SDK for **Duel** a privacy-preserving matchmaking protocol on Solana using MagicBlock Ephemeral Rollups (TEE).
4
4
 
5
- ## Features
6
-
7
- - **Privacy-First**: Player ELO and queue status hidden in TEE
8
- - **Client-Side**: Players join queues directly from their wallets (no CPI required)
9
- - **Automatic Matching**: TEE processes matches based on ELO windows
10
- - **Easy Integration**: Simple SDK for both game owners and players
5
+ Built on [`@solana/kit`](https://github.com/anza-xyz/kit). No `@coral-xyz/anchor` or legacy `web3.js` dependency.
11
6
 
12
7
  ## Installation
13
8
 
@@ -17,85 +12,141 @@ npm install @1upmonster/duel
17
12
 
18
13
  ## Quick Start
19
14
 
20
- ### For Game Owners
21
-
22
- Initialize matchmaking infrastructure for your game:
15
+ ### For Game Owners (`MatchmakingAdmin`)
23
16
 
24
17
  ```typescript
25
18
  import { MatchmakingAdmin } from "@1upmonster/duel";
26
- import { AnchorProvider } from "@coral-xyz/anchor";
27
-
28
- const admin = new MatchmakingAdmin(provider);
29
-
30
- // 1. Initialize Tenant for your Game Program
31
- await admin.initializeTenant(gameProgramId, {
32
- authority, // optional, defaults to gameProgramId
33
- eloWindow: 200, // optional, default 100
34
- eloOffset: 40, // optional, default 40
35
- eloDataType: 'u16' // optional, default 'u16' (u8/u16/u32/u64)
19
+ import { createSolanaRpc, createKeyPairSignerFromBytes } from "@solana/kit";
20
+ import * as crypto from "crypto";
21
+
22
+ const rpc = createSolanaRpc("https://api.devnet.solana.com");
23
+ const signer = await createKeyPairSignerFromBytes(/* your keypair bytes */);
24
+ const admin = new MatchmakingAdmin(rpc, signer);
25
+
26
+ // 1. Initialize Tenant stores callback config on L1
27
+ const callbackDiscriminator = Array.from(
28
+ crypto.createHash("sha256").update("global:on_match_found").digest().slice(0, 8)
29
+ );
30
+ await admin.initializeTenant(YOUR_GAME_PROGRAM_ID, {
31
+ eloWindow: 100n, // max ELO diff for a match
32
+ eloOffset: 40, // byte offset of ELO in player account
33
+ eloDataType: "u64", // u8 | u16 | u32 | u64
34
+ callbackProgramId: YOUR_GAME_PROGRAM_ID,
35
+ callbackDiscriminator,
36
36
  });
37
37
 
38
- // 2. Initialize a Matchmaking Queue
39
- const tenantPda = admin.getTenantPda(authority);
38
+ // 2. Create and delegate queue to TEE
39
+ const authority = signer.address;
40
40
  await admin.initializeQueue(authority, tenantPda);
41
-
42
- // 3. Delegate to TEE Validator
43
41
  await admin.delegateQueue(authority, validatorPubkey);
44
- ```
45
-
46
- ### ELO Data Types
47
-
48
- The SDK supports different ELO data types to optimize storage:
49
- - `'u8'`: 0-255 (1 byte)
50
- - `'u16'`: 0-65,535 (2 bytes) **[DEFAULT]**
51
- - `'u32'`: 0-4,294,967,295 (4 bytes)
52
- - `'u64'`: 0-18,446,744,073,709,551,615 (8 bytes)
53
42
 
54
- ### For Players
43
+ // 3. After players have matched, flush opponent tickets + commit all to L1
44
+ await admin.resolveMatches(queuePda, tenantPda, [p1TicketPda, p2TicketPda]);
45
+ ```
55
46
 
56
- Join the private matchmaking queue:
47
+ ### For Players (`MatchmakingPlayer`)
57
48
 
58
49
  ```typescript
59
50
  import { MatchmakingPlayer } from "@1upmonster/duel";
60
-
61
- const player = new MatchmakingPlayer(provider);
62
-
63
- // Join queue (matching happens automatically in TEE)
64
- await player.joinQueue(queuePda, tenantPda, playerProfilePda);
51
+ import { getAuthToken } from "@1upmonster/duel";
52
+ import { createSolanaRpc } from "@solana/kit";
53
+
54
+ const l1Rpc = createSolanaRpc("https://api.devnet.solana.com");
55
+ const player = new MatchmakingPlayer(l1Rpc, signer);
56
+
57
+ // High-level: creates ticket on L1, delegates to TEE, joins queue in one call
58
+ const { token } = await getAuthToken("https://tee.magicblock.app", signer);
59
+ const teeRpc = createSolanaRpc(`https://tee.magicblock.app?token=${token}`);
60
+
61
+ const ticketPda = await player.enterQueue(
62
+ tenantPda,
63
+ queuePda,
64
+ playerProfilePda,
65
+ teeRpc,
66
+ `https://tee.magicblock.app?token=${token}`,
67
+ validatorPubkey, // optional TEE validator
68
+ YOUR_GAME_PROGRAM_ID, // optional: callback fires via Tenant PDA when matched
69
+ );
70
+
71
+ // Poll L1 until the ticket shows Matched status
72
+ const match = await player.pollForMatch(ticketPda);
73
+ // match = { opponent: Address, matchId: bigint }
65
74
  ```
66
75
 
67
76
  ## API Reference
68
77
 
69
78
  ### `MatchmakingAdmin`
70
79
 
71
- **Constructor**
72
- - `new MatchmakingAdmin(provider: AnchorProvider, programId?: PublicKey)`
80
+ **Constructor:** `new MatchmakingAdmin(rpc, signer, programId?)`
73
81
 
74
- **Methods**
75
- - `initializeTenant(authority, tenantProgramId, eloWindow?, eloOffset?)` - Set up game tenant
76
- - `initializeQueue(authority, tenant)` - Create matchmaking queue
77
- - `delegateQueue(authority, validator?)` - Delegate to TEE validator
78
- - `getTenantPda(authority)` - Derive tenant PDA
79
- - `getQueuePda(authority)` - Derive queue PDA
82
+ | Method | Description |
83
+ |---|---|
84
+ | `initializeTenant(tenantProgramId, options?)` | Create Tenant PDA with ELO config and optional callback |
85
+ | `initializeQueue(authority, tenant)` | Create Queue PDA linked to tenant |
86
+ | `delegateQueue(authority, validator?)` | Delegate queue to TEE (makes it private) |
87
+ | `flushMatches(queue, tenant, ticketPdas)` | Update opponent tickets from pending matches |
88
+ | `commitTickets(tenant, ticketPdas)` | Push matched ticket state back to L1 |
89
+ | `resolveMatches(queue, tenant, ticketPdas, settlementDelayMs?)` | High-level: flush + wait + commit |
90
+ | `getQueue(queuePda)` | Fetch queue account |
91
+ | `getQueuePda(authority)` | Derive queue PDA |
92
+ | `getTenantPda(authority)` | Derive tenant PDA |
80
93
 
81
94
  ### `MatchmakingPlayer`
82
95
 
83
- **Constructor**
84
- - `new MatchmakingPlayer(provider: AnchorProvider, programId?: PublicKey)`
96
+ **Constructor:** `new MatchmakingPlayer(rpc, signer, programId?)`
97
+
98
+ | Method | Description |
99
+ |---|---|
100
+ | `enterQueue(tenant, queue, playerData, teeRpc, teeUrlWithToken, validator?, callbackProgram?)` | High-level: create ticket → delegate → join queue |
101
+ | `createTicket(tenant)` | Create MatchTicket PDA on L1 |
102
+ | `delegateTicket(player, tenant, validator?)` | Delegate ticket to TEE |
103
+ | `joinQueue(queue, tenant, playerData, callbackProgram?)` | Join queue in TEE; callback fires via Tenant PDA on match |
104
+ | `cancelTicket(tenant)` | Cancel search (sets ticket to Cancelled) |
105
+ | `closeTicket(tenant)` | Reclaim rent after match or cancel |
106
+ | `pollForMatch(ticketPda, maxAttempts?, pollInterval?)` | Poll L1 until ticket is Matched |
107
+ | `getTicket(ticketPda)` | Fetch ticket account |
108
+ | `withRpc(teeUrl)` | Return a new client pointing at a different RPC |
109
+
110
+ ### `getAuthToken(rpcUrl, signer)`
111
+
112
+ Authenticate with the MagicBlock TEE. Returns `{ token, expiresAt }`. The token is passed as `?token=<jwt>` in the TEE RPC URL.
113
+
114
+ ### `waitUntilPermissionActive(teeUrlWithToken, pda)`
85
115
 
86
- **Methods**
87
- - `joinQueue(queue, tenant, playerData)` - Join matchmaking queue
116
+ Poll the TEE until a delegated PDA is active (authorized users list is non-empty).
88
117
 
89
- ## How It Works
118
+ ## How the Callback Works
119
+
120
+ When a match is found during `join_queue`, the duel program fires a CPI callback **signed by the Tenant PDA** via `invoke_signed`. This is cryptographically unforgeable — game programs can verify the signer is the Tenant PDA without any additional access control:
121
+
122
+ ```rust
123
+ pub fn on_match_found(
124
+ ctx: Context<OnMatchFound>,
125
+ player1: Pubkey,
126
+ player2: Pubkey,
127
+ match_id: u64,
128
+ ) -> Result<()> {
129
+ // ctx.accounts.signer.key() == Tenant PDA — verified by the runtime
130
+ // Only the duel program can produce this signer
131
+ Ok(())
132
+ }
133
+
134
+ #[derive(Accounts)]
135
+ pub struct OnMatchFound<'info> {
136
+ pub signer: Signer<'info>, // will be the Tenant PDA
137
+ }
138
+ ```
90
139
 
91
- 1. **Queue Joining**: Players call `joinQueue` directly via SDK
92
- 2. **Matching**: TEE automatically processes matches based on ELO windows
93
- 3. **Privacy**: Queue state is only visible within the TEE, not on L1
140
+ Pass the callback program as `remaining_accounts` in `joinQueue` (or via `callbackProgram` param in the SDK) for the callback to fire.
94
141
 
95
- ## Links
142
+ ## ELO Data Types
96
143
 
97
- - [GitHub Repository](https://github.com/val-samonte/private-matchmaking)
98
- - [Full Documentation](https://github.com/val-samonte/private-matchmaking#readme)
144
+ | Type | Bytes | Range |
145
+ |---|---|---|
146
+ | `"u8"` | 1 | 0–255 |
147
+ | `"u16"` | 2 | 0–65,535 |
148
+ | `"u32"` | 4 | 0–4,294,967,295 |
149
+ | `"u64"` | 8 | 0–18,446,744,073,709,551,615 |
99
150
 
100
151
  ## License
101
152
 
package/dist/admin.d.ts CHANGED
@@ -1,49 +1,34 @@
1
- import * as anchor from "@coral-xyz/anchor";
2
- import * as web3 from "@solana/web3.js";
3
- import type { Duel } from "./types.js";
4
- export type EloDataType = 'u8' | 'u16' | 'u32' | 'u64';
1
+ import { type Address, type TransactionSigner, type Rpc, type SolanaRpcApi } from "@solana/kit";
2
+ export type EloDataType = "u8" | "u16" | "u32" | "u64";
5
3
  export interface InitializeTenantOptions {
6
- authority?: web3.PublicKey;
7
- eloWindow?: number;
4
+ authority?: Address;
5
+ eloWindow?: bigint;
8
6
  eloOffset?: number;
9
7
  eloDataType?: EloDataType;
10
- callbackProgramId?: web3.PublicKey | null;
8
+ callbackProgramId?: Address | null;
11
9
  callbackDiscriminator?: number[] | null;
12
10
  }
13
11
  export declare class MatchmakingAdmin {
14
- program: anchor.Program<Duel>;
15
- provider: anchor.AnchorProvider;
16
- constructor(provider: anchor.AnchorProvider, programId: web3.PublicKey | string);
17
- getQueuePda(authority: web3.PublicKey): web3.PublicKey;
18
- getTenantPda(authority: web3.PublicKey): web3.PublicKey;
19
- getTicketPda(player: web3.PublicKey, tenant: web3.PublicKey): web3.PublicKey;
12
+ rpc: Rpc<SolanaRpcApi>;
13
+ signer: TransactionSigner;
14
+ programId: Address;
15
+ constructor(rpc: Rpc<SolanaRpcApi>, signer: TransactionSigner, programId?: Address);
16
+ getQueuePda(authority: Address): Promise<Address>;
17
+ getTenantPda(authority: Address): Promise<Address>;
18
+ getTicketPda(player: Address, tenant: Address): Promise<Address>;
19
+ getQueue(queuePda: Address): Promise<import("@solana/accounts").Account<import("./generated/duel/index.js").Queue, string>>;
20
+ initializeTenant(tenantProgramId: Address, options?: InitializeTenantOptions): Promise<string>;
21
+ initializeQueue(_authority: Address, tenant: Address): Promise<string>;
22
+ delegateQueue(authority: Address, validator?: Address): Promise<string>;
23
+ flushMatches(queue: Address, tenant: Address, ticketPdas: Address[]): Promise<string>;
24
+ commitTickets(tenant: Address, ticketPdas: Address[]): Promise<string>;
20
25
  /**
21
- * Fetch Queue account data
26
+ * High-level: full match resolution flow (runs on TEE).
27
+ * Reads the queue's pending matches, flushes them (updating opponent tickets),
28
+ * waits for TEE settlement, then commits all tickets back to L1.
29
+ * Use individual methods (flushMatches, commitTickets) as escape hatches if needed.
22
30
  */
23
- getQueue(queuePda: web3.PublicKey): Promise<any>;
24
- /**
25
- * Fetch Tenant account data
26
- */
27
- getTenant(tenantPda: web3.PublicKey): Promise<any>;
28
- /**
29
- * Initialize a Tenant (with optional callback config)
30
- */
31
- initializeTenant(tenantProgramId: web3.PublicKey, options?: InitializeTenantOptions, confirmOptions?: web3.ConfirmOptions, signers?: web3.Keypair[]): Promise<web3.TransactionSignature>;
32
- /**
33
- * Initialize a Queue
34
- */
35
- initializeQueue(authority: web3.PublicKey, tenant: web3.PublicKey, confirmOptions?: web3.ConfirmOptions, signers?: web3.Keypair[]): Promise<web3.TransactionSignature>;
36
- /**
37
- * Delegate Queue to TEE
38
- */
39
- delegateQueue(authority: web3.PublicKey, validator?: web3.PublicKey, confirmOptions?: web3.ConfirmOptions, signers?: web3.Keypair[]): Promise<web3.TransactionSignature>;
40
- /**
41
- * Flush pending matches - crank instruction to update opponent tickets
42
- * Can be called by any TEE-authenticated wallet (permissionless)
43
- */
44
- flushMatches(queue: web3.PublicKey, tenant: web3.PublicKey, ticketPdas: web3.PublicKey[], callbackProgram?: web3.PublicKey, confirmOptions?: web3.ConfirmOptions, signers?: web3.Keypair[]): Promise<web3.TransactionSignature>;
45
- /**
46
- * Commit matched tickets back to L1 (runs in TEE)
47
- */
48
- commitTickets(tenant: web3.PublicKey, ticketPdas: web3.PublicKey[], confirmOptions?: web3.ConfirmOptions, signers?: web3.Keypair[]): Promise<web3.TransactionSignature>;
31
+ resolveMatches(queue: Address, tenant: Address, ticketPdas: Address[], settlementDelayMs?: number): Promise<void>;
32
+ /** Create a new MatchmakingAdmin pointing at a TEE RPC endpoint. */
33
+ withRpc(teeUrl: string): MatchmakingAdmin;
49
34
  }
package/dist/admin.js CHANGED
@@ -1,137 +1,116 @@
1
- import * as anchor from "@coral-xyz/anchor";
2
- import { BN } from "bn.js";
3
- import * as web3 from "@solana/web3.js";
4
- import IDL from "./duel.json" with { type: "json" };
1
+ import { createSolanaRpc, } from "@solana/kit";
2
+ import { getInitializeTenantInstructionAsync, getInitializeQueueInstructionAsync, getDelegateQueueInstructionAsync, getFlushMatchesInstruction, getCommitTicketsInstruction, fetchQueue, accountType, } from "./generated/duel/index.js";
3
+ import { sendInstruction } from "./transaction.js";
5
4
  import * as utils from "./utils.js";
6
- const TICKET_SEED = "ticket";
7
5
  function getEloSize(dataType) {
8
6
  switch (dataType) {
9
- case 'u8': return 1;
10
- case 'u16': return 2;
11
- case 'u32': return 4;
12
- case 'u64': return 8;
7
+ case "u8": return 1;
8
+ case "u16": return 2;
9
+ case "u32": return 4;
10
+ case "u64": return 8;
13
11
  }
14
12
  }
13
+ const DUEL_PROGRAM_ID = "EdZzUwKd1X2ZWjxLPpz1cpEzMF7RUZC43Pq64v1VcK5X";
15
14
  export class MatchmakingAdmin {
16
- constructor(provider, programId) {
17
- const address = typeof programId === "string" ? programId : programId.toBase58();
18
- const idl = { ...IDL, address };
19
- this.program = new anchor.Program(idl, provider);
20
- this.provider = provider;
15
+ constructor(rpc, signer, programId = DUEL_PROGRAM_ID) {
16
+ this.rpc = rpc;
17
+ this.signer = signer;
18
+ this.programId = programId;
21
19
  }
22
- // Derive PDAs Helpers
23
- getQueuePda(authority) {
24
- return utils.deriveQueuePda(this.program.programId, authority);
20
+ async getQueuePda(authority) {
21
+ return utils.deriveQueuePda(this.programId, authority);
25
22
  }
26
- getTenantPda(authority) {
27
- return utils.deriveTenantPda(this.program.programId, authority);
23
+ async getTenantPda(authority) {
24
+ return utils.deriveTenantPda(this.programId, authority);
28
25
  }
29
- getTicketPda(player, tenant) {
30
- return utils.deriveTicketPda(this.program.programId, player, tenant);
26
+ async getTicketPda(player, tenant) {
27
+ return utils.deriveTicketPda(this.programId, player, tenant);
31
28
  }
32
- /**
33
- * Fetch Queue account data
34
- */
35
29
  async getQueue(queuePda) {
36
- return await this.program.account.queue.fetch(queuePda);
37
- }
38
- /**
39
- * Fetch Tenant account data
40
- */
41
- async getTenant(tenantPda) {
42
- return await this.program.account.tenant.fetch(tenantPda);
30
+ return fetchQueue(this.rpc, queuePda);
43
31
  }
44
- /**
45
- * Initialize a Tenant (with optional callback config)
46
- */
47
- async initializeTenant(tenantProgramId, options, confirmOptions, signers = []) {
48
- const { authority = tenantProgramId, eloWindow = 100, eloOffset = 40, eloDataType = 'u16', callbackProgramId = null, callbackDiscriminator = null, } = options || {};
32
+ async initializeTenant(tenantProgramId, options) {
33
+ const { eloWindow = 100n, eloOffset = 40, eloDataType = "u16", callbackProgramId = null, callbackDiscriminator = null, } = options || {};
49
34
  const eloSize = getEloSize(eloDataType);
50
- const tenantPda = this.getTenantPda(authority);
51
- return await this.program.methods
52
- .initializeTenant(tenantProgramId, eloOffset, eloSize, new BN(eloWindow), callbackProgramId || null, callbackDiscriminator ? Array.from(Buffer.from(callbackDiscriminator)) : null)
53
- .accountsPartial({
54
- tenant: tenantPda,
55
- authority: authority,
56
- })
57
- .signers(signers)
58
- .rpc(confirmOptions);
35
+ const ix = await getInitializeTenantInstructionAsync({
36
+ authority: this.signer,
37
+ tenantProgramId,
38
+ eloOffset,
39
+ eloSize,
40
+ eloWindow,
41
+ callbackProgramId: callbackProgramId ? { __option: "Some", value: callbackProgramId } : { __option: "None" },
42
+ callbackDiscriminator: callbackDiscriminator
43
+ ? { __option: "Some", value: new Uint8Array(callbackDiscriminator) }
44
+ : { __option: "None" },
45
+ }, { programAddress: this.programId });
46
+ return sendInstruction(this.rpc, ix, this.signer);
59
47
  }
60
- /**
61
- * Initialize a Queue
62
- */
63
- async initializeQueue(authority, tenant, confirmOptions, signers = []) {
64
- const queuePda = this.getQueuePda(authority);
65
- return await this.program.methods
66
- .initializeQueue()
67
- .accountsPartial({
68
- queue: queuePda,
69
- tenant: tenant,
70
- authority: authority,
71
- })
72
- .signers(signers)
73
- .rpc(confirmOptions);
48
+ async initializeQueue(_authority, tenant) {
49
+ const ix = await getInitializeQueueInstructionAsync({
50
+ authority: this.signer,
51
+ tenant,
52
+ }, { programAddress: this.programId });
53
+ return sendInstruction(this.rpc, ix, this.signer);
74
54
  }
75
- /**
76
- * Delegate Queue to TEE
77
- */
78
- async delegateQueue(authority, validator = new web3.PublicKey("FnE6VJT5QNZdedZPnCoLsARgBwoE6DeJNjBs2H1gySXA"), confirmOptions, signers = []) {
79
- const queuePda = this.getQueuePda(authority);
80
- return await this.program.methods
81
- .delegateQueue({ queue: { authority } })
82
- .accounts({
55
+ async delegateQueue(authority, validator) {
56
+ const queuePda = await this.getQueuePda(authority);
57
+ const ix = await getDelegateQueueInstructionAsync({
83
58
  pda: queuePda,
84
- payer: authority,
85
- validator: validator,
86
- })
87
- .signers(signers)
88
- .rpc(confirmOptions);
59
+ payer: this.signer,
60
+ validator,
61
+ accountType: accountType("Queue", { authority }),
62
+ }, { programAddress: this.programId });
63
+ return sendInstruction(this.rpc, ix, this.signer);
64
+ }
65
+ async flushMatches(queue, tenant, ticketPdas) {
66
+ const ix = getFlushMatchesInstruction({
67
+ queue,
68
+ tenant,
69
+ signer: this.signer,
70
+ }, { programAddress: this.programId });
71
+ const ixWithRemaining = {
72
+ ...ix,
73
+ accounts: [
74
+ ...ix.accounts,
75
+ ...ticketPdas.map((address) => ({ address, role: 1 })),
76
+ ],
77
+ };
78
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
79
+ return sendInstruction(this.rpc, ixWithRemaining, this.signer);
80
+ }
81
+ async commitTickets(tenant, ticketPdas) {
82
+ const ix = getCommitTicketsInstruction({
83
+ tenant,
84
+ payer: this.signer,
85
+ }, { programAddress: this.programId });
86
+ const ixWithRemaining = {
87
+ ...ix,
88
+ accounts: [
89
+ ...ix.accounts,
90
+ ...ticketPdas.map((address) => ({ address, role: 1 })),
91
+ ],
92
+ };
93
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
94
+ return sendInstruction(this.rpc, ixWithRemaining, this.signer);
89
95
  }
90
96
  /**
91
- * Flush pending matches - crank instruction to update opponent tickets
92
- * Can be called by any TEE-authenticated wallet (permissionless)
97
+ * High-level: full match resolution flow (runs on TEE).
98
+ * Reads the queue's pending matches, flushes them (updating opponent tickets),
99
+ * waits for TEE settlement, then commits all tickets back to L1.
100
+ * Use individual methods (flushMatches, commitTickets) as escape hatches if needed.
93
101
  */
94
- async flushMatches(queue, tenant, ticketPdas, callbackProgram, confirmOptions, signers = []) {
95
- const remainingAccounts = ticketPdas.map(pda => ({
96
- pubkey: pda,
97
- isSigner: false,
98
- isWritable: true,
99
- }));
100
- if (callbackProgram) {
101
- remainingAccounts.push({
102
- pubkey: callbackProgram,
103
- isSigner: false,
104
- isWritable: false,
105
- });
102
+ async resolveMatches(queue, tenant, ticketPdas, settlementDelayMs = 3000) {
103
+ const queueData = await this.getQueue(queue);
104
+ const pendingTicketPdas = await Promise.all(queueData.data.pendingMatches.map((pm) => utils.deriveTicketPda(this.programId, pm.player, tenant)));
105
+ if (pendingTicketPdas.length > 0) {
106
+ await this.flushMatches(queue, tenant, pendingTicketPdas);
107
+ await new Promise((r) => setTimeout(r, settlementDelayMs));
106
108
  }
107
- return await this.program.methods
108
- .flushMatches()
109
- .accountsPartial({
110
- queue: queue,
111
- tenant: tenant,
112
- signer: this.provider.publicKey,
113
- })
114
- .remainingAccounts(remainingAccounts)
115
- .signers(signers)
116
- .rpc(confirmOptions);
109
+ await this.commitTickets(tenant, ticketPdas);
117
110
  }
118
- /**
119
- * Commit matched tickets back to L1 (runs in TEE)
120
- */
121
- async commitTickets(tenant, ticketPdas, confirmOptions, signers = []) {
122
- return await this.program.methods
123
- .commitTickets()
124
- .accountsPartial({
125
- tenant: tenant,
126
- payer: this.provider.publicKey,
127
- })
128
- .remainingAccounts(ticketPdas.map(pda => ({
129
- pubkey: pda,
130
- isSigner: false,
131
- isWritable: true,
132
- })))
133
- .signers(signers)
134
- .rpc(confirmOptions);
111
+ /** Create a new MatchmakingAdmin pointing at a TEE RPC endpoint. */
112
+ withRpc(teeUrl) {
113
+ return new MatchmakingAdmin(createSolanaRpc(teeUrl), this.signer, this.programId);
135
114
  }
136
115
  }
137
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"admin.js","sourceRoot":"","sources":["../src/admin.ts"],"names":[],"mappings":"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","sourcesContent":["import * as anchor from \"@coral-xyz/anchor\";\nimport { BN } from \"bn.js\";\nimport * as web3 from \"@solana/web3.js\";\nimport type { Duel } from \"./types.js\";\nimport IDL from \"./duel.json\" with { type: \"json\" };\nimport * as utils from \"./utils.js\";\n\nconst TICKET_SEED = \"ticket\";\n\nexport type EloDataType = 'u8' | 'u16' | 'u32' | 'u64';\n\nexport interface InitializeTenantOptions {\n  authority?: web3.PublicKey;\n  eloWindow?: number;\n  eloOffset?: number;\n  eloDataType?: EloDataType;\n  callbackProgramId?: web3.PublicKey | null;\n  callbackDiscriminator?: number[] | null;\n}\n\nfunction getEloSize(dataType: EloDataType): number {\n  switch (dataType) {\n    case 'u8': return 1;\n    case 'u16': return 2;\n    case 'u32': return 4;\n    case 'u64': return 8;\n  }\n}\n\nexport class MatchmakingAdmin {\n  public program: anchor.Program<Duel>;\n  public provider: anchor.AnchorProvider;\n\n  constructor(provider: anchor.AnchorProvider, programId: web3.PublicKey | string) {\n    const address = typeof programId === \"string\" ? programId : programId.toBase58();\n    const idl = { ...IDL, address };\n    this.program = new anchor.Program(idl as any, provider);\n    this.provider = provider;\n  }\n\n  // Derive PDAs Helpers\n  getQueuePda(authority: web3.PublicKey): web3.PublicKey {\n    return utils.deriveQueuePda(this.program.programId, authority);\n  }\n\n  getTenantPda(authority: web3.PublicKey): web3.PublicKey {\n    return utils.deriveTenantPda(this.program.programId, authority);\n  }\n\n  getTicketPda(player: web3.PublicKey, tenant: web3.PublicKey): web3.PublicKey {\n    return utils.deriveTicketPda(this.program.programId, player, tenant);\n  }\n\n  /**\n   * Fetch Queue account data\n   */\n  async getQueue(queuePda: web3.PublicKey): Promise<any> {\n    return await this.program.account.queue.fetch(queuePda);\n  }\n\n  /**\n   * Fetch Tenant account data\n   */\n  async getTenant(tenantPda: web3.PublicKey): Promise<any> {\n    return await this.program.account.tenant.fetch(tenantPda);\n  }\n\n  /**\n   * Initialize a Tenant (with optional callback config)\n   */\n  async initializeTenant(\n    tenantProgramId: web3.PublicKey,\n    options?: InitializeTenantOptions,\n    confirmOptions?: web3.ConfirmOptions,\n    signers: web3.Keypair[] = []\n  ): Promise<web3.TransactionSignature> {\n    const {\n      authority = tenantProgramId,\n      eloWindow = 100,\n      eloOffset = 40,\n      eloDataType = 'u16',\n      callbackProgramId = null,\n      callbackDiscriminator = null,\n    } = options || {};\n\n    const eloSize = getEloSize(eloDataType);\n    const tenantPda = this.getTenantPda(authority);\n\n    return await this.program.methods\n      .initializeTenant(\n        tenantProgramId,\n        eloOffset,\n        eloSize,\n        new BN(eloWindow),\n        callbackProgramId || null,\n        callbackDiscriminator ? Array.from(Buffer.from(callbackDiscriminator)) : null\n      )\n      .accountsPartial({\n        tenant: tenantPda,\n        authority: authority,\n      })\n      .signers(signers)\n      .rpc(confirmOptions);\n  }\n\n  /**\n   * Initialize a Queue\n   */\n  async initializeQueue(\n    authority: web3.PublicKey,\n    tenant: web3.PublicKey,\n    confirmOptions?: web3.ConfirmOptions,\n    signers: web3.Keypair[] = []\n  ): Promise<web3.TransactionSignature> {\n    const queuePda = this.getQueuePda(authority);\n    return await this.program.methods\n      .initializeQueue()\n      .accountsPartial({\n        queue: queuePda,\n        tenant: tenant,\n        authority: authority,\n      })\n      .signers(signers)\n      .rpc(confirmOptions);\n  }\n\n  /**\n   * Delegate Queue to TEE\n   */\n  async delegateQueue(\n    authority: web3.PublicKey,\n    validator: web3.PublicKey = new web3.PublicKey(\"FnE6VJT5QNZdedZPnCoLsARgBwoE6DeJNjBs2H1gySXA\"),\n    confirmOptions?: web3.ConfirmOptions,\n    signers: web3.Keypair[] = []\n  ): Promise<web3.TransactionSignature> {\n      const queuePda = this.getQueuePda(authority);\n      return await this.program.methods\n        .delegateQueue({ queue: { authority } } as any)\n        .accounts({\n            pda: queuePda,\n            payer: authority,\n            validator: validator,\n        } as unknown as any)\n        .signers(signers)\n        .rpc(confirmOptions);\n  }\n\n  /**\n   * Flush pending matches - crank instruction to update opponent tickets\n   * Can be called by any TEE-authenticated wallet (permissionless)\n   */\n  async flushMatches(\n    queue: web3.PublicKey,\n    tenant: web3.PublicKey,\n    ticketPdas: web3.PublicKey[],\n    callbackProgram?: web3.PublicKey,\n    confirmOptions?: web3.ConfirmOptions,\n    signers: web3.Keypair[] = []\n  ): Promise<web3.TransactionSignature> {\n    const remainingAccounts = ticketPdas.map(pda => ({\n      pubkey: pda,\n      isSigner: false,\n      isWritable: true,\n    }));\n\n    if (callbackProgram) {\n      remainingAccounts.push({\n        pubkey: callbackProgram,\n        isSigner: false,\n        isWritable: false,\n      });\n    }\n\n    return await this.program.methods\n      .flushMatches()\n      .accountsPartial({\n        queue: queue,\n        tenant: tenant,\n        signer: this.provider.publicKey,\n      })\n      .remainingAccounts(remainingAccounts)\n      .signers(signers)\n      .rpc(confirmOptions);\n  }\n\n  /**\n   * Commit matched tickets back to L1 (runs in TEE)\n   */\n  async commitTickets(\n    tenant: web3.PublicKey,\n    ticketPdas: web3.PublicKey[],\n    confirmOptions?: web3.ConfirmOptions,\n    signers: web3.Keypair[] = []\n  ): Promise<web3.TransactionSignature> {\n    return await this.program.methods\n      .commitTickets()\n      .accountsPartial({\n        tenant: tenant,\n        payer: this.provider.publicKey,\n      })\n      .remainingAccounts(\n        ticketPdas.map(pda => ({\n          pubkey: pda,\n          isSigner: false,\n          isWritable: true,\n        }))\n      )\n      .signers(signers)\n      .rpc(confirmOptions);\n  }\n}\n"]}
116
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"admin.js","sourceRoot":"","sources":["../src/admin.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  createSolanaRpc,\n  type Address,\n  type TransactionSigner,\n  type Rpc,\n  type SolanaRpcApi,\n} from \"@solana/kit\";\nimport {\n  getInitializeTenantInstructionAsync,\n  getInitializeQueueInstructionAsync,\n  getDelegateQueueInstructionAsync,\n  getFlushMatchesInstruction,\n  getCommitTicketsInstruction,\n  fetchQueue,\n  accountType,\n} from \"./generated/duel/index.js\";\nimport { sendInstruction } from \"./transaction.js\";\nimport * as utils from \"./utils.js\";\n\nexport type EloDataType = \"u8\" | \"u16\" | \"u32\" | \"u64\";\n\nexport interface InitializeTenantOptions {\n  authority?: Address;\n  eloWindow?: bigint;\n  eloOffset?: number;\n  eloDataType?: EloDataType;\n  callbackProgramId?: Address | null;\n  callbackDiscriminator?: number[] | null;\n}\n\nfunction getEloSize(dataType: EloDataType): number {\n  switch (dataType) {\n    case \"u8\": return 1;\n    case \"u16\": return 2;\n    case \"u32\": return 4;\n    case \"u64\": return 8;\n  }\n}\n\nconst DUEL_PROGRAM_ID = \"EdZzUwKd1X2ZWjxLPpz1cpEzMF7RUZC43Pq64v1VcK5X\" as Address;\n\nexport class MatchmakingAdmin {\n  public rpc: Rpc<SolanaRpcApi>;\n  public signer: TransactionSigner;\n  public programId: Address;\n\n  constructor(\n    rpc: Rpc<SolanaRpcApi>,\n    signer: TransactionSigner,\n    programId: Address = DUEL_PROGRAM_ID,\n  ) {\n    this.rpc = rpc;\n    this.signer = signer;\n    this.programId = programId;\n  }\n\n  async getQueuePda(authority: Address): Promise<Address> {\n    return utils.deriveQueuePda(this.programId, authority);\n  }\n\n  async getTenantPda(authority: Address): Promise<Address> {\n    return utils.deriveTenantPda(this.programId, authority);\n  }\n\n  async getTicketPda(player: Address, tenant: Address): Promise<Address> {\n    return utils.deriveTicketPda(this.programId, player, tenant);\n  }\n\n  async getQueue(queuePda: Address) {\n    return fetchQueue(this.rpc, queuePda);\n  }\n\n  async initializeTenant(\n    tenantProgramId: Address,\n    options?: InitializeTenantOptions,\n  ): Promise<string> {\n    const {\n      eloWindow = 100n,\n      eloOffset = 40,\n      eloDataType = \"u16\",\n      callbackProgramId = null,\n      callbackDiscriminator = null,\n    } = options || {};\n\n    const eloSize = getEloSize(eloDataType);\n    const ix = await getInitializeTenantInstructionAsync({\n      authority: this.signer,\n      tenantProgramId,\n      eloOffset,\n      eloSize,\n      eloWindow,\n      callbackProgramId: callbackProgramId ? { __option: \"Some\", value: callbackProgramId } : { __option: \"None\" },\n      callbackDiscriminator: callbackDiscriminator\n        ? { __option: \"Some\", value: new Uint8Array(callbackDiscriminator) }\n        : { __option: \"None\" },\n    }, { programAddress: this.programId });\n\n    return sendInstruction(this.rpc, ix, this.signer);\n  }\n\n  async initializeQueue(\n    _authority: Address,\n    tenant: Address,\n  ): Promise<string> {\n    const ix = await getInitializeQueueInstructionAsync({\n      authority: this.signer,\n      tenant,\n    }, { programAddress: this.programId });\n    return sendInstruction(this.rpc, ix, this.signer);\n  }\n\n  async delegateQueue(\n    authority: Address,\n    validator?: Address,\n  ): Promise<string> {\n    const queuePda = await this.getQueuePda(authority);\n    const ix = await getDelegateQueueInstructionAsync({\n      pda: queuePda,\n      payer: this.signer,\n      validator,\n      accountType: accountType(\"Queue\", { authority }),\n    }, { programAddress: this.programId });\n    return sendInstruction(this.rpc, ix, this.signer);\n  }\n\n  async flushMatches(\n    queue: Address,\n    tenant: Address,\n    ticketPdas: Address[],\n  ): Promise<string> {\n    const ix = getFlushMatchesInstruction({\n      queue,\n      tenant,\n      signer: this.signer,\n    }, { programAddress: this.programId });\n\n    const ixWithRemaining = {\n      ...ix,\n      accounts: [\n        ...ix.accounts,\n        ...ticketPdas.map((address) => ({ address, role: 1 as const })),\n      ],\n    };\n\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    return sendInstruction(this.rpc, ixWithRemaining as any, this.signer);\n  }\n\n  async commitTickets(\n    tenant: Address,\n    ticketPdas: Address[],\n  ): Promise<string> {\n    const ix = getCommitTicketsInstruction({\n      tenant,\n      payer: this.signer,\n    }, { programAddress: this.programId });\n\n    const ixWithRemaining = {\n      ...ix,\n      accounts: [\n        ...ix.accounts,\n        ...ticketPdas.map((address) => ({ address, role: 1 as const })),\n      ],\n    };\n\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    return sendInstruction(this.rpc, ixWithRemaining as any, this.signer);\n  }\n\n  /**\n   * High-level: full match resolution flow (runs on TEE).\n   * Reads the queue's pending matches, flushes them (updating opponent tickets),\n   * waits for TEE settlement, then commits all tickets back to L1.\n   * Use individual methods (flushMatches, commitTickets) as escape hatches if needed.\n   */\n  async resolveMatches(\n    queue: Address,\n    tenant: Address,\n    ticketPdas: Address[],\n    settlementDelayMs = 3000,\n  ): Promise<void> {\n    const queueData = await this.getQueue(queue);\n    const pendingTicketPdas = await Promise.all(\n      queueData.data.pendingMatches.map((pm) =>\n        utils.deriveTicketPda(this.programId, pm.player, tenant),\n      ),\n    );\n\n    if (pendingTicketPdas.length > 0) {\n      await this.flushMatches(queue, tenant, pendingTicketPdas);\n      await new Promise((r) => setTimeout(r, settlementDelayMs));\n    }\n\n    await this.commitTickets(tenant, ticketPdas);\n  }\n\n  /** Create a new MatchmakingAdmin pointing at a TEE RPC endpoint. */\n  withRpc(teeUrl: string): MatchmakingAdmin {\n    return new MatchmakingAdmin(createSolanaRpc(teeUrl), this.signer, this.programId);\n  }\n}\n"]}
@@ -1,4 +1,3 @@
1
- // import { x25519 } from "@noble/curves/ed25519";
2
1
  export class EncryptionProvider {
3
2
  constructor() {
4
3
  this.keyPair = null;
@@ -94,4 +93,4 @@ export class EncryptionProvider {
94
93
  return new Uint8Array(raw);
95
94
  }
96
95
  }
97
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"encryption.js","sourceRoot":"","sources":["../src/encryption.ts"],"names":[],"mappings":"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","sourcesContent":["import { PublicKey } from \"@solana/web3.js\";\n// import { x25519 } from \"@noble/curves/ed25519\"; \n\nexport class EncryptionProvider {\n    private keyPair: CryptoKeyPair | null = null;\n    \n    // Check for crypto availability\n    private get crypto(): Crypto {\n        if (typeof globalThis.crypto !== 'undefined') {\n            return globalThis.crypto;\n        }\n        // Fallback for Node < 19 if global crypto not set (though SDK likely targets modern envs)\n        try {\n            return require('crypto').webcrypto;\n        } catch (e) {\n            throw new Error(\"WebCrypto API not available.\");\n        }\n    }\n\n    /**\n     * Generate a fresh ephemeral keypair for the session (X25519/P-256).\n     */\n    async generateSessionKey(): Promise<CryptoKey> {\n        this.keyPair = await this.crypto.subtle.generateKey(\n            {\n                name: \"ECDH\",\n                namedCurve: \"P-256\", \n            },\n            true,\n            [\"deriveKey\", \"deriveBits\"]\n        ) as CryptoKeyPair;\n        \n        return this.keyPair.publicKey;\n    }\n\n    /**\n     * Derive shared secret and encrypt payload.\n     */\n    async encryptPayload(\n        data: Uint8Array, \n        teePublicKeyBytes: Uint8Array\n    ): Promise<{ encrypted: Uint8Array, clientPublicKey: Uint8Array }> {\n        if (!this.keyPair) {\n            await this.generateSessionKey();\n        }\n\n        // Import TEE Public Key\n        // Fix: Explicitly cast to BufferSource/any because TS gets confused with SharedArrayBuffer in some envs\n        const teeKey = await this.crypto.subtle.importKey(\n            \"raw\",\n            teePublicKeyBytes as unknown as BufferSource,\n            { name: \"ECDH\", namedCurve: \"P-256\" }, \n            false,\n            []\n        );\n\n        // Derive Shared Secret (AES-GCM Key)\n        const sharedKey = await this.crypto.subtle.deriveKey(\n            {\n                name: \"ECDH\",\n                public: teeKey,\n            },\n            this.keyPair!.privateKey,\n            {\n                name: \"AES-GCM\",\n                length: 256,\n            },\n            false,\n            [\"encrypt\", \"decrypt\"]\n        );\n\n        // Encrypt Data\n        const iv = this.crypto.getRandomValues(new Uint8Array(12));\n        const encryptedBuffer = await this.crypto.subtle.encrypt(\n            {\n                name: \"AES-GCM\",\n                iv: iv,\n            },\n            sharedKey,\n            data as unknown as BufferSource\n        );\n\n        // Concatenate IV + CipherText\n        const encrypted = new Uint8Array(iv.length + encryptedBuffer.byteLength);\n        encrypted.set(iv);\n        encrypted.set(new Uint8Array(encryptedBuffer), iv.length);\n\n        // Export Client Public Key\n        const clientPubRaw = await this.crypto.subtle.exportKey(\"raw\", this.keyPair!.publicKey);\n\n        return {\n            encrypted,\n            clientPublicKey: new Uint8Array(clientPubRaw)\n        };\n    }\n    \n    /**\n     * Decrypt a response from the TEE.\n     */\n    async decryptResponse(\n        encryptedData: Uint8Array, \n        teePublicKeyBytes: Uint8Array\n    ): Promise<Uint8Array> {\n          if (!this.keyPair) throw new Error(\"No session key\");\n          \n           // Import TEE Public Key\n        const teeKey = await this.crypto.subtle.importKey(\n            \"raw\",\n            teePublicKeyBytes as unknown as BufferSource,\n            { name: \"ECDH\", namedCurve: \"P-256\" }, \n            false,\n            []\n        );\n\n        // Derive Shared Secret\n        const sharedKey = await this.crypto.subtle.deriveKey(\n            {\n                name: \"ECDH\",\n                public: teeKey,\n            },\n            this.keyPair!.privateKey,\n            {\n                name: \"AES-GCM\",\n                length: 256,\n            },\n            false,\n            [\"encrypt\", \"decrypt\"]\n        );\n        \n        const iv = encryptedData.slice(0, 12);\n        const ciphertext = encryptedData.slice(12);\n        \n        const decrypted = await this.crypto.subtle.decrypt(\n            {\n                name: \"AES-GCM\",\n                iv: iv,\n            },\n            sharedKey,\n            ciphertext as unknown as BufferSource\n        );\n        \n        return new Uint8Array(decrypted);\n    }\n\n    /**\n     * Helper to generate a valid random P-256 Public Key (65 bytes) for testing.\n     */\n    async createMockValidatorKey(): Promise<Uint8Array> {\n        const pair = await this.crypto.subtle.generateKey(\n            { name: \"ECDH\", namedCurve: \"P-256\" },\n            true, \n            [\"deriveKey\"]\n        ) as CryptoKeyPair;\n        const raw = await this.crypto.subtle.exportKey(\"raw\", pair.publicKey);\n        return new Uint8Array(raw);\n    }\n}\n"]}
96
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"encryption.js","sourceRoot":"","sources":["../src/encryption.ts"],"names":[],"mappings":"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","sourcesContent":["export class EncryptionProvider {\n    private keyPair: CryptoKeyPair | null = null;\n    \n    // Check for crypto availability\n    private get crypto(): Crypto {\n        if (typeof globalThis.crypto !== 'undefined') {\n            return globalThis.crypto;\n        }\n        // Fallback for Node < 19 if global crypto not set (though SDK likely targets modern envs)\n        try {\n            return require('crypto').webcrypto;\n        } catch (e) {\n            throw new Error(\"WebCrypto API not available.\");\n        }\n    }\n\n    /**\n     * Generate a fresh ephemeral keypair for the session (X25519/P-256).\n     */\n    async generateSessionKey(): Promise<CryptoKey> {\n        this.keyPair = await this.crypto.subtle.generateKey(\n            {\n                name: \"ECDH\",\n                namedCurve: \"P-256\", \n            },\n            true,\n            [\"deriveKey\", \"deriveBits\"]\n        ) as CryptoKeyPair;\n        \n        return this.keyPair.publicKey;\n    }\n\n    /**\n     * Derive shared secret and encrypt payload.\n     */\n    async encryptPayload(\n        data: Uint8Array, \n        teePublicKeyBytes: Uint8Array\n    ): Promise<{ encrypted: Uint8Array, clientPublicKey: Uint8Array }> {\n        if (!this.keyPair) {\n            await this.generateSessionKey();\n        }\n\n        // Import TEE Public Key\n        // Fix: Explicitly cast to BufferSource/any because TS gets confused with SharedArrayBuffer in some envs\n        const teeKey = await this.crypto.subtle.importKey(\n            \"raw\",\n            teePublicKeyBytes as unknown as BufferSource,\n            { name: \"ECDH\", namedCurve: \"P-256\" }, \n            false,\n            []\n        );\n\n        // Derive Shared Secret (AES-GCM Key)\n        const sharedKey = await this.crypto.subtle.deriveKey(\n            {\n                name: \"ECDH\",\n                public: teeKey,\n            },\n            this.keyPair!.privateKey,\n            {\n                name: \"AES-GCM\",\n                length: 256,\n            },\n            false,\n            [\"encrypt\", \"decrypt\"]\n        );\n\n        // Encrypt Data\n        const iv = this.crypto.getRandomValues(new Uint8Array(12));\n        const encryptedBuffer = await this.crypto.subtle.encrypt(\n            {\n                name: \"AES-GCM\",\n                iv: iv,\n            },\n            sharedKey,\n            data as unknown as BufferSource\n        );\n\n        // Concatenate IV + CipherText\n        const encrypted = new Uint8Array(iv.length + encryptedBuffer.byteLength);\n        encrypted.set(iv);\n        encrypted.set(new Uint8Array(encryptedBuffer), iv.length);\n\n        // Export Client Public Key\n        const clientPubRaw = await this.crypto.subtle.exportKey(\"raw\", this.keyPair!.publicKey);\n\n        return {\n            encrypted,\n            clientPublicKey: new Uint8Array(clientPubRaw)\n        };\n    }\n    \n    /**\n     * Decrypt a response from the TEE.\n     */\n    async decryptResponse(\n        encryptedData: Uint8Array, \n        teePublicKeyBytes: Uint8Array\n    ): Promise<Uint8Array> {\n          if (!this.keyPair) throw new Error(\"No session key\");\n          \n           // Import TEE Public Key\n        const teeKey = await this.crypto.subtle.importKey(\n            \"raw\",\n            teePublicKeyBytes as unknown as BufferSource,\n            { name: \"ECDH\", namedCurve: \"P-256\" }, \n            false,\n            []\n        );\n\n        // Derive Shared Secret\n        const sharedKey = await this.crypto.subtle.deriveKey(\n            {\n                name: \"ECDH\",\n                public: teeKey,\n            },\n            this.keyPair!.privateKey,\n            {\n                name: \"AES-GCM\",\n                length: 256,\n            },\n            false,\n            [\"encrypt\", \"decrypt\"]\n        );\n        \n        const iv = encryptedData.slice(0, 12);\n        const ciphertext = encryptedData.slice(12);\n        \n        const decrypted = await this.crypto.subtle.decrypt(\n            {\n                name: \"AES-GCM\",\n                iv: iv,\n            },\n            sharedKey,\n            ciphertext as unknown as BufferSource\n        );\n        \n        return new Uint8Array(decrypted);\n    }\n\n    /**\n     * Helper to generate a valid random P-256 Public Key (65 bytes) for testing.\n     */\n    async createMockValidatorKey(): Promise<Uint8Array> {\n        const pair = await this.crypto.subtle.generateKey(\n            { name: \"ECDH\", namedCurve: \"P-256\" },\n            true, \n            [\"deriveKey\"]\n        ) as CryptoKeyPair;\n        const raw = await this.crypto.subtle.exportKey(\"raw\", pair.publicKey);\n        return new Uint8Array(raw);\n    }\n}\n"]}