@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.
- package/README.md +110 -59
- package/dist/admin.d.ts +25 -40
- package/dist/admin.js +93 -114
- package/dist/encryption.js +1 -2
- package/dist/generated/duel/accounts/index.d.ts +10 -0
- package/dist/generated/duel/accounts/index.js +11 -0
- package/dist/generated/duel/accounts/matchTicket.d.ts +38 -0
- package/dist/generated/duel/accounts/matchTicket.js +45 -0
- package/dist/generated/duel/accounts/queue.d.ts +42 -0
- package/dist/generated/duel/accounts/queue.js +45 -0
- package/dist/generated/duel/accounts/tenant.d.ts +41 -0
- package/dist/generated/duel/accounts/tenant.js +44 -0
- package/dist/generated/duel/errors/duel.d.ts +31 -0
- package/dist/generated/duel/errors/duel.js +35 -0
- package/dist/generated/duel/errors/index.d.ts +8 -0
- package/dist/generated/duel/errors/index.js +9 -0
- package/dist/generated/duel/index.d.ts +12 -0
- package/dist/generated/duel/index.js +13 -0
- package/dist/generated/duel/instructions/cancelTicket.d.ts +45 -0
- package/dist/generated/duel/instructions/cancelTicket.js +56 -0
- package/dist/generated/duel/instructions/closeTicket.d.ts +45 -0
- package/dist/generated/duel/instructions/closeTicket.js +56 -0
- package/dist/generated/duel/instructions/commitTickets.d.ts +39 -0
- package/dist/generated/duel/instructions/commitTickets.js +50 -0
- package/dist/generated/duel/instructions/createTicket.d.ts +48 -0
- package/dist/generated/duel/instructions/createTicket.js +63 -0
- package/dist/generated/duel/instructions/delegateQueue.d.ts +69 -0
- package/dist/generated/duel/instructions/delegateQueue.js +90 -0
- package/dist/generated/duel/instructions/delegateTicket.d.ts +69 -0
- package/dist/generated/duel/instructions/delegateTicket.js +90 -0
- package/dist/generated/duel/instructions/flushMatches.d.ts +37 -0
- package/dist/generated/duel/instructions/flushMatches.js +43 -0
- package/dist/generated/duel/instructions/index.d.ts +19 -0
- package/dist/generated/duel/instructions/index.js +20 -0
- package/dist/generated/duel/instructions/initializeQueue.d.ts +48 -0
- package/dist/generated/duel/instructions/initializeQueue.js +63 -0
- package/dist/generated/duel/instructions/initializeTenant.d.ts +70 -0
- package/dist/generated/duel/instructions/initializeTenant.js +67 -0
- package/dist/generated/duel/instructions/joinQueue.d.ts +51 -0
- package/dist/generated/duel/instructions/joinQueue.js +56 -0
- package/dist/generated/duel/instructions/processUndelegation.d.ts +43 -0
- package/dist/generated/duel/instructions/processUndelegation.js +49 -0
- package/dist/generated/duel/instructions/setupTicketPermission.d.ts +54 -0
- package/dist/generated/duel/instructions/setupTicketPermission.js +63 -0
- package/dist/generated/duel/programs/duel.d.ts +92 -0
- package/dist/generated/duel/programs/duel.js +146 -0
- package/dist/generated/duel/programs/index.d.ts +8 -0
- package/dist/generated/duel/programs/index.js +9 -0
- package/dist/generated/duel/types/accountType.d.ts +25 -0
- package/dist/generated/duel/types/accountType.js +25 -0
- package/dist/generated/duel/types/index.d.ts +13 -0
- package/dist/generated/duel/types/index.js +14 -0
- package/dist/generated/duel/types/matchEntry.d.ts +23 -0
- package/dist/generated/duel/types/matchEntry.js +18 -0
- package/dist/generated/duel/types/matchFound.d.ts +23 -0
- package/dist/generated/duel/types/matchFound.js +18 -0
- package/dist/generated/duel/types/pendingMatch.d.ts +21 -0
- package/dist/generated/duel/types/pendingMatch.js +18 -0
- package/dist/generated/duel/types/queueEntry.d.ts +19 -0
- package/dist/generated/duel/types/queueEntry.js +18 -0
- package/dist/generated/duel/types/ticketStatus.d.ts +40 -0
- package/dist/generated/duel/types/ticketStatus.js +25 -0
- package/dist/generated/rps-game/accounts/gameSession.d.ts +40 -0
- package/dist/generated/rps-game/accounts/gameSession.js +45 -0
- package/dist/generated/rps-game/accounts/index.d.ts +9 -0
- package/dist/generated/rps-game/accounts/index.js +10 -0
- package/dist/generated/rps-game/accounts/playerProfile.d.ts +36 -0
- package/dist/generated/rps-game/accounts/playerProfile.js +47 -0
- package/dist/generated/rps-game/errors/index.d.ts +8 -0
- package/dist/generated/rps-game/errors/index.js +9 -0
- package/dist/generated/rps-game/errors/rpsGame.d.ts +25 -0
- package/dist/generated/rps-game/errors/rpsGame.js +29 -0
- package/dist/generated/rps-game/index.d.ts +12 -0
- package/dist/generated/rps-game/index.js +13 -0
- package/dist/generated/rps-game/instructions/closePlayer.d.ts +45 -0
- package/dist/generated/rps-game/instructions/closePlayer.js +56 -0
- package/dist/generated/rps-game/instructions/delegatePda.d.ts +69 -0
- package/dist/generated/rps-game/instructions/delegatePda.js +90 -0
- package/dist/generated/rps-game/instructions/index.d.ts +16 -0
- package/dist/generated/rps-game/instructions/index.js +17 -0
- package/dist/generated/rps-game/instructions/initializePlayer.d.ts +48 -0
- package/dist/generated/rps-game/instructions/initializePlayer.js +63 -0
- package/dist/generated/rps-game/instructions/makeChoice.d.ts +44 -0
- package/dist/generated/rps-game/instructions/makeChoice.js +46 -0
- package/dist/generated/rps-game/instructions/onMatchFound.d.ts +43 -0
- package/dist/generated/rps-game/instructions/onMatchFound.js +45 -0
- package/dist/generated/rps-game/instructions/persistResults.d.ts +43 -0
- package/dist/generated/rps-game/instructions/persistResults.js +50 -0
- package/dist/generated/rps-game/instructions/processUndelegation.d.ts +43 -0
- package/dist/generated/rps-game/instructions/processUndelegation.js +49 -0
- package/dist/generated/rps-game/instructions/startGame.d.ts +54 -0
- package/dist/generated/rps-game/instructions/startGame.js +67 -0
- package/dist/generated/rps-game/instructions/startGameWithTicket.d.ts +57 -0
- package/dist/generated/rps-game/instructions/startGameWithTicket.js +67 -0
- package/dist/generated/rps-game/programs/index.d.ts +8 -0
- package/dist/generated/rps-game/programs/index.js +9 -0
- package/dist/generated/rps-game/programs/rpsGame.d.ts +78 -0
- package/dist/generated/rps-game/programs/rpsGame.js +118 -0
- package/dist/generated/rps-game/types/accountType.d.ts +34 -0
- package/dist/generated/rps-game/types/accountType.js +25 -0
- package/dist/generated/rps-game/types/choice.d.ts +17 -0
- package/dist/generated/rps-game/types/choice.js +25 -0
- package/dist/generated/rps-game/types/gameResult.d.ts +26 -0
- package/dist/generated/rps-game/types/gameResult.js +25 -0
- package/dist/generated/rps-game/types/index.d.ts +10 -0
- package/dist/generated/rps-game/types/index.js +11 -0
- package/dist/index.d.ts +2 -1
- package/dist/index.js +3 -2
- package/dist/player.d.ts +25 -46
- package/dist/player.js +81 -140
- package/dist/tee.d.ts +14 -0
- package/dist/tee.js +62 -0
- package/dist/transaction.d.ts +11 -0
- package/dist/transaction.js +50 -0
- package/dist/utils.d.ts +4 -4
- package/dist/utils.js +23 -8
- package/package.json +3 -6
- package/src/admin.ts +151 -161
- package/src/duel.json +66 -7
- package/src/encryption.ts +0 -3
- package/src/generated/duel/accounts/index.ts +11 -0
- package/src/generated/duel/accounts/matchTicket.ts +77 -0
- package/src/generated/duel/accounts/queue.ts +77 -0
- package/src/generated/duel/accounts/tenant.ts +76 -0
- package/src/generated/duel/errors/duel.ts +46 -0
- package/src/generated/duel/errors/index.ts +9 -0
- package/src/generated/duel/index.ts +13 -0
- package/src/generated/duel/instructions/cancelTicket.ts +100 -0
- package/src/generated/duel/instructions/closeTicket.ts +100 -0
- package/src/generated/duel/instructions/commitTickets.ts +84 -0
- package/src/generated/duel/instructions/createTicket.ts +109 -0
- package/src/generated/duel/instructions/delegateQueue.ts +157 -0
- package/src/generated/duel/instructions/delegateTicket.ts +157 -0
- package/src/generated/duel/instructions/flushMatches.ts +76 -0
- package/src/generated/duel/instructions/index.ts +20 -0
- package/src/generated/duel/instructions/initializeQueue.ts +109 -0
- package/src/generated/duel/instructions/initializeTenant.ts +126 -0
- package/src/generated/duel/instructions/joinQueue.ts +106 -0
- package/src/generated/duel/instructions/processUndelegation.ts +86 -0
- package/src/generated/duel/instructions/setupTicketPermission.ts +115 -0
- package/src/generated/duel/programs/duel.ts +108 -0
- package/src/generated/duel/programs/index.ts +9 -0
- package/src/generated/duel/types/accountType.ts +36 -0
- package/src/generated/duel/types/index.ts +14 -0
- package/src/generated/duel/types/matchEntry.ts +25 -0
- package/src/generated/duel/types/matchFound.ts +25 -0
- package/src/generated/duel/types/pendingMatch.ts +25 -0
- package/src/generated/duel/types/queueEntry.ts +25 -0
- package/src/generated/duel/types/ticketStatus.ts +38 -0
- package/src/generated/rps-game/accounts/gameSession.ts +77 -0
- package/src/generated/rps-game/accounts/index.ts +10 -0
- package/src/generated/rps-game/accounts/playerProfile.ts +80 -0
- package/src/generated/rps-game/errors/index.ts +9 -0
- package/src/generated/rps-game/errors/rpsGame.ts +40 -0
- package/src/generated/rps-game/index.ts +13 -0
- package/src/generated/rps-game/instructions/closePlayer.ts +100 -0
- package/src/generated/rps-game/instructions/delegatePda.ts +157 -0
- package/src/generated/rps-game/instructions/index.ts +17 -0
- package/src/generated/rps-game/instructions/initializePlayer.ts +109 -0
- package/src/generated/rps-game/instructions/makeChoice.ts +84 -0
- package/src/generated/rps-game/instructions/onMatchFound.ts +79 -0
- package/src/generated/rps-game/instructions/persistResults.ts +88 -0
- package/src/generated/rps-game/instructions/processUndelegation.ts +86 -0
- package/src/generated/rps-game/instructions/startGame.ts +118 -0
- package/src/generated/rps-game/instructions/startGameWithTicket.ts +121 -0
- package/src/generated/rps-game/programs/index.ts +9 -0
- package/src/generated/rps-game/programs/rpsGame.ts +95 -0
- package/src/generated/rps-game/types/accountType.ts +36 -0
- package/src/generated/rps-game/types/choice.ts +25 -0
- package/src/generated/rps-game/types/gameResult.ts +37 -0
- package/src/generated/rps-game/types/index.ts +11 -0
- package/src/index.ts +2 -1
- package/src/player.ts +129 -192
- package/src/{idl/private_matchmaking.json → rps_game.json} +547 -583
- package/src/tee.ts +79 -0
- package/src/transaction.ts +90 -0
- package/src/utils.ts +35 -20
- package/tsconfig.json +2 -2
- package/dist/client.d.ts +0 -54
- package/dist/client.js +0 -265
- package/dist/duel.json +0 -1207
- package/dist/private_matchmaking.json +0 -534
- package/dist/types.d.ts +0 -635
- package/dist/types.js +0 -2
- package/src/idl/private_matchmaking.ts +0 -1033
- package/src/types.ts +0 -300
package/src/tee.ts
ADDED
|
@@ -0,0 +1,79 @@
|
|
|
1
|
+
import {
|
|
2
|
+
getBase58Decoder,
|
|
3
|
+
createSignableMessage,
|
|
4
|
+
type Address,
|
|
5
|
+
type MessagePartialSigner,
|
|
6
|
+
} from "@solana/kit";
|
|
7
|
+
|
|
8
|
+
export type { MessagePartialSigner as MessageSigner };
|
|
9
|
+
|
|
10
|
+
/**
|
|
11
|
+
* Authenticate with the MagicBlock TEE via challenge-sign flow.
|
|
12
|
+
*/
|
|
13
|
+
export async function getAuthToken(
|
|
14
|
+
rpcUrl: string,
|
|
15
|
+
signer: MessagePartialSigner,
|
|
16
|
+
): Promise<{ token: string; expiresAt: number }> {
|
|
17
|
+
const challengeRes = await fetch(
|
|
18
|
+
`${rpcUrl}/auth/challenge?pubkey=${signer.address}`
|
|
19
|
+
);
|
|
20
|
+
if (!challengeRes.ok) {
|
|
21
|
+
throw new Error(`TEE challenge failed: ${challengeRes.statusText}`);
|
|
22
|
+
}
|
|
23
|
+
const { challenge } = (await challengeRes.json()) as { challenge: string };
|
|
24
|
+
|
|
25
|
+
const challengeBytes = new TextEncoder().encode(challenge);
|
|
26
|
+
const [sigDict] = await signer.signMessages([createSignableMessage(challengeBytes)]);
|
|
27
|
+
const signature = sigDict[signer.address as Address];
|
|
28
|
+
const signatureString = getBase58Decoder().decode(signature);
|
|
29
|
+
|
|
30
|
+
const tokenRes = await fetch(`${rpcUrl}/auth/login`, {
|
|
31
|
+
method: "POST",
|
|
32
|
+
headers: { "Content-Type": "application/json" },
|
|
33
|
+
body: JSON.stringify({
|
|
34
|
+
pubkey: signer.address,
|
|
35
|
+
challenge,
|
|
36
|
+
signature: signatureString,
|
|
37
|
+
}),
|
|
38
|
+
});
|
|
39
|
+
const authJson = (await tokenRes.json()) as { token: string; expiresAt?: number; error?: string };
|
|
40
|
+
if (tokenRes.status !== 200) {
|
|
41
|
+
throw new Error(`Failed to authenticate: ${authJson.error}`);
|
|
42
|
+
}
|
|
43
|
+
const expiresAt = authJson.expiresAt ?? Date.now() + 1000 * 60 * 60 * 24 * 30;
|
|
44
|
+
return { token: authJson.token, expiresAt };
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
/**
|
|
48
|
+
* Poll the TEE /permission endpoint until the given PDA has authorized users,
|
|
49
|
+
* indicating delegation is active. Returns false on timeout (does not throw).
|
|
50
|
+
*/
|
|
51
|
+
export async function waitUntilPermissionActive(
|
|
52
|
+
teeUrlWithToken: string,
|
|
53
|
+
pda: Address,
|
|
54
|
+
timeoutMs = 30000,
|
|
55
|
+
): Promise<boolean> {
|
|
56
|
+
// Parse URL: "https://host/path?token=xxx" -> baseUrl="https://host/path", tokenParam="token=xxx"
|
|
57
|
+
const [baseUrl, tokenParam] = teeUrlWithToken.replace("/?", "?").split("?");
|
|
58
|
+
let permissionUrl: string;
|
|
59
|
+
if (tokenParam) {
|
|
60
|
+
permissionUrl = `${baseUrl}/permission?${tokenParam}&pubkey=${pda}`;
|
|
61
|
+
} else {
|
|
62
|
+
permissionUrl = `${baseUrl}/permission?pubkey=${pda}`;
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
const start = Date.now();
|
|
66
|
+
while (Date.now() - start < timeoutMs) {
|
|
67
|
+
try {
|
|
68
|
+
const res = await fetch(permissionUrl);
|
|
69
|
+
if (res.ok) {
|
|
70
|
+
const { authorizedUsers } = (await res.json()) as { authorizedUsers?: unknown[] };
|
|
71
|
+
if (authorizedUsers && authorizedUsers.length > 0) return true;
|
|
72
|
+
}
|
|
73
|
+
} catch {
|
|
74
|
+
// ignore transient errors, keep polling
|
|
75
|
+
}
|
|
76
|
+
await new Promise((r) => setTimeout(r, 400));
|
|
77
|
+
}
|
|
78
|
+
return false;
|
|
79
|
+
}
|
|
@@ -0,0 +1,90 @@
|
|
|
1
|
+
import {
|
|
2
|
+
createTransactionMessage,
|
|
3
|
+
setTransactionMessageFeePayerSigner,
|
|
4
|
+
setTransactionMessageLifetimeUsingBlockhash,
|
|
5
|
+
appendTransactionMessageInstruction,
|
|
6
|
+
appendTransactionMessageInstructions,
|
|
7
|
+
signTransactionMessageWithSigners,
|
|
8
|
+
getBase64EncodedWireTransaction,
|
|
9
|
+
pipe,
|
|
10
|
+
type Instruction,
|
|
11
|
+
type SolanaRpcApi,
|
|
12
|
+
type Rpc,
|
|
13
|
+
type TransactionSigner,
|
|
14
|
+
} from "@solana/kit";
|
|
15
|
+
|
|
16
|
+
type SolanaRpc = Rpc<SolanaRpcApi>;
|
|
17
|
+
|
|
18
|
+
/**
|
|
19
|
+
* Build, sign with a Kit keypair signer, and send a single instruction.
|
|
20
|
+
*/
|
|
21
|
+
export async function sendInstruction(
|
|
22
|
+
rpc: SolanaRpc,
|
|
23
|
+
instruction: Instruction,
|
|
24
|
+
signer: TransactionSigner,
|
|
25
|
+
): Promise<string> {
|
|
26
|
+
const { value: latestBlockhash } = await rpc.getLatestBlockhash().send();
|
|
27
|
+
|
|
28
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
29
|
+
const tx: any = pipe(
|
|
30
|
+
createTransactionMessage({ version: 0 as const }),
|
|
31
|
+
(m) => setTransactionMessageFeePayerSigner(signer, m),
|
|
32
|
+
(m) => setTransactionMessageLifetimeUsingBlockhash(latestBlockhash, m),
|
|
33
|
+
(m) => appendTransactionMessageInstruction(instruction, m),
|
|
34
|
+
);
|
|
35
|
+
|
|
36
|
+
const signedTx = await signTransactionMessageWithSigners(tx);
|
|
37
|
+
const encoded = getBase64EncodedWireTransaction(signedTx);
|
|
38
|
+
|
|
39
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
40
|
+
const sig = await (rpc.sendTransaction(encoded as any, {
|
|
41
|
+
encoding: "base64",
|
|
42
|
+
skipPreflight: true,
|
|
43
|
+
}).send() as Promise<string>);
|
|
44
|
+
|
|
45
|
+
// Poll for status to detect runtime errors
|
|
46
|
+
for (let i = 0; i < 8; i++) {
|
|
47
|
+
await new Promise(r => setTimeout(r, 1000));
|
|
48
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
49
|
+
const statuses = await (rpc as any).getSignatureStatuses([sig], { searchTransactionHistory: false }).send().catch(() => null);
|
|
50
|
+
const status = statuses?.value?.[0];
|
|
51
|
+
if (status) {
|
|
52
|
+
if (status.err) {
|
|
53
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
54
|
+
console.error(`[TX] ${sig.slice(0, 16)}... FAILED:`, JSON.stringify(status.err, (_, v) => typeof v === 'bigint' ? v.toString() : v));
|
|
55
|
+
} else if (status.confirmationStatus) {
|
|
56
|
+
console.log(`[TX] ${sig.slice(0, 16)}... ${status.confirmationStatus}`);
|
|
57
|
+
}
|
|
58
|
+
break;
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
return sig;
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
/**
|
|
65
|
+
* Build, sign, and send multiple instructions in a single transaction.
|
|
66
|
+
*/
|
|
67
|
+
export async function sendInstructions(
|
|
68
|
+
rpc: SolanaRpc,
|
|
69
|
+
instructions: Instruction[],
|
|
70
|
+
signer: TransactionSigner,
|
|
71
|
+
): Promise<string> {
|
|
72
|
+
const { value: latestBlockhash } = await rpc.getLatestBlockhash().send();
|
|
73
|
+
|
|
74
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
75
|
+
const tx: any = pipe(
|
|
76
|
+
createTransactionMessage({ version: 0 as const }),
|
|
77
|
+
(m) => setTransactionMessageFeePayerSigner(signer, m),
|
|
78
|
+
(m) => setTransactionMessageLifetimeUsingBlockhash(latestBlockhash, m),
|
|
79
|
+
(m) => appendTransactionMessageInstructions(instructions, m),
|
|
80
|
+
);
|
|
81
|
+
|
|
82
|
+
const signedTx = await signTransactionMessageWithSigners(tx);
|
|
83
|
+
const encoded = getBase64EncodedWireTransaction(signedTx);
|
|
84
|
+
|
|
85
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
86
|
+
return rpc.sendTransaction(encoded as any, {
|
|
87
|
+
encoding: "base64",
|
|
88
|
+
skipPreflight: true,
|
|
89
|
+
}).send() as Promise<string>;
|
|
90
|
+
}
|
package/src/utils.ts
CHANGED
|
@@ -1,30 +1,45 @@
|
|
|
1
|
-
import
|
|
2
|
-
|
|
1
|
+
import {
|
|
2
|
+
getProgramDerivedAddress,
|
|
3
|
+
getAddressEncoder,
|
|
4
|
+
getUtf8Encoder,
|
|
5
|
+
type Address,
|
|
6
|
+
} from "@solana/kit";
|
|
7
|
+
|
|
8
|
+
const addressEncoder = getAddressEncoder();
|
|
9
|
+
const utf8Encoder = getUtf8Encoder();
|
|
3
10
|
|
|
4
11
|
export const QUEUE_SEED = "queue";
|
|
5
12
|
export const TENANT_SEED = "tenant";
|
|
6
13
|
export const TICKET_SEED = "ticket";
|
|
7
14
|
|
|
8
|
-
export function deriveQueuePda(programId:
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
15
|
+
export async function deriveQueuePda(programId: Address, authority: Address): Promise<Address> {
|
|
16
|
+
const [pda] = await getProgramDerivedAddress({
|
|
17
|
+
programAddress: programId,
|
|
18
|
+
seeds: [utf8Encoder.encode(QUEUE_SEED), addressEncoder.encode(authority)],
|
|
19
|
+
});
|
|
20
|
+
return pda;
|
|
14
21
|
}
|
|
15
22
|
|
|
16
|
-
export function deriveTenantPda(programId:
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
23
|
+
export async function deriveTenantPda(programId: Address, authority: Address): Promise<Address> {
|
|
24
|
+
const [pda] = await getProgramDerivedAddress({
|
|
25
|
+
programAddress: programId,
|
|
26
|
+
seeds: [utf8Encoder.encode(TENANT_SEED), addressEncoder.encode(authority)],
|
|
27
|
+
});
|
|
28
|
+
return pda;
|
|
22
29
|
}
|
|
23
30
|
|
|
24
|
-
export function deriveTicketPda(
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
31
|
+
export async function deriveTicketPda(
|
|
32
|
+
programId: Address,
|
|
33
|
+
player: Address,
|
|
34
|
+
tenant: Address
|
|
35
|
+
): Promise<Address> {
|
|
36
|
+
const [pda] = await getProgramDerivedAddress({
|
|
37
|
+
programAddress: programId,
|
|
38
|
+
seeds: [
|
|
39
|
+
utf8Encoder.encode(TICKET_SEED),
|
|
40
|
+
addressEncoder.encode(player),
|
|
41
|
+
addressEncoder.encode(tenant),
|
|
42
|
+
],
|
|
43
|
+
});
|
|
44
|
+
return pda;
|
|
30
45
|
}
|
package/tsconfig.json
CHANGED
package/dist/client.d.ts
DELETED
|
@@ -1,54 +0,0 @@
|
|
|
1
|
-
import { Program, AnchorProvider } from "@coral-xyz/anchor";
|
|
2
|
-
import { PublicKey, Keypair, TransactionSignature } from "@solana/web3.js";
|
|
3
|
-
import { PrivateMatchmaking } from "./idl/private_matchmaking";
|
|
4
|
-
import { MatchmakingClientConfig, QueueHead, QueuePage, PlayerStatus, JoinQueueResult } from "./types";
|
|
5
|
-
import { EncryptionProvider } from "./encryption";
|
|
6
|
-
export declare class MatchmakingClient {
|
|
7
|
-
program: Program<PrivateMatchmaking>;
|
|
8
|
-
provider: AnchorProvider;
|
|
9
|
-
config: MatchmakingClientConfig;
|
|
10
|
-
encryption: EncryptionProvider;
|
|
11
|
-
constructor(provider: AnchorProvider, programId: PublicKey, config?: MatchmakingClientConfig);
|
|
12
|
-
/**
|
|
13
|
-
* Initialize a new matchmaking queue.
|
|
14
|
-
*/
|
|
15
|
-
initializeQueue(queueId: string, config: any, capacity: number, pageSize?: number): Promise<PublicKey>;
|
|
16
|
-
/**
|
|
17
|
-
* Initialize a specific page for the queue.
|
|
18
|
-
*/
|
|
19
|
-
initializePage(queue: PublicKey, index: number): Promise<PublicKey>;
|
|
20
|
-
/**
|
|
21
|
-
* Delegate the queue to the Privacy Layer (Ephemeral Rollup).
|
|
22
|
-
*/
|
|
23
|
-
delegateQueue(queueId: string, validatorOverride?: PublicKey): Promise<TransactionSignature>;
|
|
24
|
-
/**
|
|
25
|
-
* Join the queue.
|
|
26
|
-
*/
|
|
27
|
-
joinQueue(queue: PublicKey, playerGameAccount: PublicKey, tenantProgramId: PublicKey): Promise<JoinQueueResult>;
|
|
28
|
-
/**
|
|
29
|
-
* Unlock a player manually (refund rent).
|
|
30
|
-
*/
|
|
31
|
-
unlockPlayer(playerGameAccount: PublicKey, playerWallet: PublicKey): Promise<TransactionSignature>;
|
|
32
|
-
/**
|
|
33
|
-
* Process matches on a specific page.
|
|
34
|
-
* Usually called by the off-chain worker or manually for testing.
|
|
35
|
-
*/
|
|
36
|
-
processMatch(queue: PublicKey, pageIndex: number): Promise<TransactionSignature>;
|
|
37
|
-
/**
|
|
38
|
-
* Resize the queue capacity.
|
|
39
|
-
*/
|
|
40
|
-
resizeQueue(queue: PublicKey, newCapacity: number): Promise<TransactionSignature>;
|
|
41
|
-
getQueue(queuePda: PublicKey): Promise<QueueHead>;
|
|
42
|
-
getPage(pagePda: PublicKey): Promise<QueuePage>;
|
|
43
|
-
getPlayerStatus(statusPda: PublicKey): Promise<PlayerStatus>;
|
|
44
|
-
getPlayerStatusForGameAccount(playerGameAccount: PublicKey): Promise<PlayerStatus>;
|
|
45
|
-
createMockPlayer(playerAccount: Keypair, elo: number): Promise<TransactionSignature>;
|
|
46
|
-
/**
|
|
47
|
-
* Close a queue page to reclaim rent.
|
|
48
|
-
*/
|
|
49
|
-
closePage(queue: PublicKey, index: number): Promise<TransactionSignature>;
|
|
50
|
-
/**
|
|
51
|
-
* Close a queue head to reclaim rent.
|
|
52
|
-
*/
|
|
53
|
-
closeQueue(queueId: string): Promise<TransactionSignature>;
|
|
54
|
-
}
|
package/dist/client.js
DELETED
|
@@ -1,265 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
-
if (k2 === undefined) k2 = k;
|
|
4
|
-
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
-
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
-
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
-
}
|
|
8
|
-
Object.defineProperty(o, k2, desc);
|
|
9
|
-
}) : (function(o, m, k, k2) {
|
|
10
|
-
if (k2 === undefined) k2 = k;
|
|
11
|
-
o[k2] = m[k];
|
|
12
|
-
}));
|
|
13
|
-
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
-
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
-
}) : function(o, v) {
|
|
16
|
-
o["default"] = v;
|
|
17
|
-
});
|
|
18
|
-
var __importStar = (this && this.__importStar) || function (mod) {
|
|
19
|
-
if (mod && mod.__esModule) return mod;
|
|
20
|
-
var result = {};
|
|
21
|
-
if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
|
|
22
|
-
__setModuleDefault(result, mod);
|
|
23
|
-
return result;
|
|
24
|
-
};
|
|
25
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
26
|
-
exports.MatchmakingClient = void 0;
|
|
27
|
-
const anchor = __importStar(require("@coral-xyz/anchor"));
|
|
28
|
-
const anchor_1 = require("@coral-xyz/anchor");
|
|
29
|
-
const web3_js_1 = require("@solana/web3.js");
|
|
30
|
-
const utils_1 = require("./utils");
|
|
31
|
-
const encryption_1 = require("./encryption");
|
|
32
|
-
// Default Validator for Devnet (MagicBlock)
|
|
33
|
-
const DEFAULT_VALIDATOR = "FnE6VJT5QNZdedZPnCoLsARgBwoE6DeJNjBs2H1gySXA";
|
|
34
|
-
class MatchmakingClient {
|
|
35
|
-
constructor(provider, programId, config = {}) {
|
|
36
|
-
this.provider = provider;
|
|
37
|
-
this.config = config;
|
|
38
|
-
this.encryption = new encryption_1.EncryptionProvider();
|
|
39
|
-
// Load the IDL (we assume it's bundled or we can require it if running in node)
|
|
40
|
-
// For SDK purity, we should import the JSON.
|
|
41
|
-
const idl = require("./idl/private_matchmaking.json");
|
|
42
|
-
idl.address = programId.toBase58();
|
|
43
|
-
this.program = new anchor_1.Program(idl, provider);
|
|
44
|
-
}
|
|
45
|
-
/**
|
|
46
|
-
* Initialize a new matchmaking queue.
|
|
47
|
-
*/
|
|
48
|
-
async initializeQueue(queueId, config, capacity, pageSize = 50) {
|
|
49
|
-
const queuePda = (0, utils_1.deriveQueuePda)(this.program.programId, this.provider.wallet.publicKey, queueId);
|
|
50
|
-
// Ensure tenant ID is set
|
|
51
|
-
const tenantProgramId = config.tenantProgramId || this.provider.wallet.publicKey;
|
|
52
|
-
const tx = await this.program.methods
|
|
53
|
-
.initializeQueue(queueId, config, capacity, pageSize)
|
|
54
|
-
.accounts({
|
|
55
|
-
// queue: queuePda,
|
|
56
|
-
// authority: this.provider.wallet.publicKey,
|
|
57
|
-
tenantProgramId: tenantProgramId,
|
|
58
|
-
// systemProgram: SystemProgram.programId,
|
|
59
|
-
})
|
|
60
|
-
.rpc(this.config.confirmOptions);
|
|
61
|
-
console.log(`Initialized Queue: ${queuePda.toBase58()} in tx: ${tx}`);
|
|
62
|
-
// Initialize Pages (Ring Buffer)
|
|
63
|
-
for (let i = 0; i < capacity; i++) {
|
|
64
|
-
await this.initializePage(queuePda, i);
|
|
65
|
-
}
|
|
66
|
-
return queuePda;
|
|
67
|
-
}
|
|
68
|
-
/**
|
|
69
|
-
* Initialize a specific page for the queue.
|
|
70
|
-
*/
|
|
71
|
-
async initializePage(queue, index) {
|
|
72
|
-
const pagePda = (0, utils_1.derivePagePda)(this.program.programId, queue, index);
|
|
73
|
-
const tx = await this.program.methods
|
|
74
|
-
.initializePage(new anchor.BN(index))
|
|
75
|
-
.accounts({
|
|
76
|
-
queue: queue,
|
|
77
|
-
// page: pagePda,
|
|
78
|
-
// authority: this.provider.wallet.publicKey,
|
|
79
|
-
// systemProgram: SystemProgram.programId,
|
|
80
|
-
})
|
|
81
|
-
.rpc(this.config.confirmOptions);
|
|
82
|
-
console.log(`Initialized Page ${index}: ${pagePda.toBase58()}`);
|
|
83
|
-
return pagePda;
|
|
84
|
-
}
|
|
85
|
-
/**
|
|
86
|
-
* Delegate the queue to the Privacy Layer (Ephemeral Rollup).
|
|
87
|
-
*/
|
|
88
|
-
async delegateQueue(queueId, validatorOverride) {
|
|
89
|
-
const validator = validatorOverride || new web3_js_1.PublicKey(DEFAULT_VALIDATOR);
|
|
90
|
-
const queuePda = (0, utils_1.deriveQueuePda)(this.program.programId, this.provider.wallet.publicKey, queueId);
|
|
91
|
-
const tx = await this.program.methods
|
|
92
|
-
.delegateQueue(queueId)
|
|
93
|
-
.preInstructions([
|
|
94
|
-
web3_js_1.ComputeBudgetProgram.requestHeapFrame({ bytes: 256 * 1024 })
|
|
95
|
-
])
|
|
96
|
-
.accounts({
|
|
97
|
-
// pda: queuePda,
|
|
98
|
-
// authority: this.provider.wallet.publicKey,
|
|
99
|
-
// payer: this.provider.wallet.publicKey,
|
|
100
|
-
validator: validator,
|
|
101
|
-
})
|
|
102
|
-
.rpc(this.config.confirmOptions);
|
|
103
|
-
console.log(`Delegated Queue ${queueId}: ${tx}`);
|
|
104
|
-
return tx;
|
|
105
|
-
}
|
|
106
|
-
/**
|
|
107
|
-
* Join the queue.
|
|
108
|
-
*/
|
|
109
|
-
async joinQueue(queue, playerGameAccount, tenantProgramId) {
|
|
110
|
-
// Fetch queue to determine current WRITE index
|
|
111
|
-
const queueAccount = await this.getQueue(queue);
|
|
112
|
-
const capacity = queueAccount.capacity;
|
|
113
|
-
const writeIndex = queueAccount.writePageIndex.toNumber();
|
|
114
|
-
const currentIndex = writeIndex % capacity;
|
|
115
|
-
const pagePda = (0, utils_1.derivePagePda)(this.program.programId, queue, currentIndex);
|
|
116
|
-
const statusPda = (0, utils_1.derivePlayerStatusPda)(this.program.programId, playerGameAccount);
|
|
117
|
-
// Encryption Handshake Integration
|
|
118
|
-
let instructionData;
|
|
119
|
-
if (this.config.encrypted) {
|
|
120
|
-
console.log("🔒 Encrypted Queue Join Init...");
|
|
121
|
-
// TODO: Fetch Validator Key from on-chain Registry or Delegate Account
|
|
122
|
-
// For now, using a mock dummy key for demonstration of the flow
|
|
123
|
-
// In PROD: const validatorKey = await this.getValidatorKey(queue);
|
|
124
|
-
const mockValidatorKey = await this.encryption.createMockValidatorKey(); // Valid P-256 Public Key
|
|
125
|
-
// Perform Encryption
|
|
126
|
-
// Real payload would vary (e.g. Rock/Paper/Scissor choice)
|
|
127
|
-
const payload = Buffer.from("PLAYER_CHOICE_ROCK");
|
|
128
|
-
const { encrypted, clientPublicKey } = await this.encryption.encryptPayload(payload, mockValidatorKey);
|
|
129
|
-
console.log("🔒 Payload Encrypted. Client PubKey:", Buffer.from(clientPublicKey).toString('hex'));
|
|
130
|
-
// In a real implementation, 'encrypted' and 'clientPublicKey' would be passed
|
|
131
|
-
// as arguments to the 'joinQueue' instruction modification.
|
|
132
|
-
// Since the IDL isn't updated for arguments yet, we just log it.
|
|
133
|
-
}
|
|
134
|
-
const tx = await this.program.methods
|
|
135
|
-
.joinQueue()
|
|
136
|
-
.accounts({
|
|
137
|
-
queue: queue,
|
|
138
|
-
// page: pagePda,
|
|
139
|
-
page: pagePda,
|
|
140
|
-
// playerStatus: statusPda,
|
|
141
|
-
// playerAuthority: this.provider.wallet.publicKey,
|
|
142
|
-
playerGameAccount: playerGameAccount,
|
|
143
|
-
tenantProgram: tenantProgramId, // Added for Matchable Interface CPI
|
|
144
|
-
// systemProgram: SystemProgram.programId,
|
|
145
|
-
})
|
|
146
|
-
.rpc(this.config.confirmOptions);
|
|
147
|
-
console.log(`Joined Queue at Page ${currentIndex}: ${tx} (Lock: ${statusPda.toBase58()})`);
|
|
148
|
-
// Future Logic: If SDK receives a return log/event indicating "Instant Match", parse it.
|
|
149
|
-
// For now, default to "Queued".
|
|
150
|
-
return {
|
|
151
|
-
status: "Queued",
|
|
152
|
-
tx,
|
|
153
|
-
statusPda
|
|
154
|
-
};
|
|
155
|
-
}
|
|
156
|
-
/**
|
|
157
|
-
* Unlock a player manually (refund rent).
|
|
158
|
-
*/
|
|
159
|
-
async unlockPlayer(playerGameAccount, playerWallet) {
|
|
160
|
-
const statusPda = (0, utils_1.derivePlayerStatusPda)(this.program.programId, playerGameAccount);
|
|
161
|
-
const statusAccount = await this.getPlayerStatus(statusPda);
|
|
162
|
-
const queueAddress = statusAccount.queue;
|
|
163
|
-
const tx = await this.program.methods
|
|
164
|
-
.unlockPlayer()
|
|
165
|
-
.accounts({
|
|
166
|
-
queue: queueAddress,
|
|
167
|
-
// authority: this.provider.wallet.publicKey,
|
|
168
|
-
// playerStatus: statusPda,
|
|
169
|
-
player: playerWallet,
|
|
170
|
-
playerGameAccount: playerGameAccount
|
|
171
|
-
})
|
|
172
|
-
.rpc(this.config.confirmOptions);
|
|
173
|
-
console.log(`Unlocked Player: ${tx}`);
|
|
174
|
-
return tx;
|
|
175
|
-
}
|
|
176
|
-
/**
|
|
177
|
-
* Process matches on a specific page.
|
|
178
|
-
* Usually called by the off-chain worker or manually for testing.
|
|
179
|
-
*/
|
|
180
|
-
async processMatch(queue, pageIndex) {
|
|
181
|
-
const pagePda = (0, utils_1.derivePagePda)(this.program.programId, queue, pageIndex);
|
|
182
|
-
const tx = await this.program.methods
|
|
183
|
-
.processMatch(new anchor.BN(pageIndex))
|
|
184
|
-
.accounts({
|
|
185
|
-
queueAccount: queue,
|
|
186
|
-
// page: pagePda,
|
|
187
|
-
})
|
|
188
|
-
.rpc(this.config.confirmOptions);
|
|
189
|
-
return tx;
|
|
190
|
-
}
|
|
191
|
-
/**
|
|
192
|
-
* Resize the queue capacity.
|
|
193
|
-
*/
|
|
194
|
-
async resizeQueue(queue, newCapacity) {
|
|
195
|
-
const tx = await this.program.methods
|
|
196
|
-
.resizeQueue(newCapacity)
|
|
197
|
-
.accounts({
|
|
198
|
-
queue: queue,
|
|
199
|
-
// authority: this.provider.wallet.publicKey,
|
|
200
|
-
})
|
|
201
|
-
.rpc(this.config.confirmOptions);
|
|
202
|
-
console.log(`Resized Queue to ${newCapacity}: ${tx}`);
|
|
203
|
-
return tx;
|
|
204
|
-
}
|
|
205
|
-
// --- State Fetchers ---
|
|
206
|
-
async getQueue(queuePda) {
|
|
207
|
-
return await this.program.account.queueHead.fetch(queuePda);
|
|
208
|
-
}
|
|
209
|
-
async getPage(pagePda) {
|
|
210
|
-
return await this.program.account.queuePage.fetch(pagePda);
|
|
211
|
-
}
|
|
212
|
-
async getPlayerStatus(statusPda) {
|
|
213
|
-
return await this.program.account.playerStatus.fetch(statusPda);
|
|
214
|
-
}
|
|
215
|
-
async getPlayerStatusForGameAccount(playerGameAccount) {
|
|
216
|
-
const statusPda = (0, utils_1.derivePlayerStatusPda)(this.program.programId, playerGameAccount);
|
|
217
|
-
return await this.getPlayerStatus(statusPda);
|
|
218
|
-
}
|
|
219
|
-
// --- Dev Helpers ---
|
|
220
|
-
async createMockPlayer(playerAccount, elo) {
|
|
221
|
-
const tx = await this.program.methods
|
|
222
|
-
.createMockPlayer(new anchor.BN(elo))
|
|
223
|
-
.accounts({
|
|
224
|
-
playerAccount: playerAccount.publicKey,
|
|
225
|
-
authority: this.provider.wallet.publicKey,
|
|
226
|
-
// systemProgram: SystemProgram.programId,
|
|
227
|
-
})
|
|
228
|
-
.signers([playerAccount])
|
|
229
|
-
.rpc(this.config.confirmOptions);
|
|
230
|
-
return tx;
|
|
231
|
-
}
|
|
232
|
-
/**
|
|
233
|
-
* Close a queue page to reclaim rent.
|
|
234
|
-
*/
|
|
235
|
-
async closePage(queue, index) {
|
|
236
|
-
const pagePda = (0, utils_1.derivePagePda)(this.program.programId, queue, index);
|
|
237
|
-
const tx = await this.program.methods
|
|
238
|
-
.closePage(new anchor.BN(index))
|
|
239
|
-
.accounts({
|
|
240
|
-
queue: queue,
|
|
241
|
-
// page: pagePda, // Auto-resolved
|
|
242
|
-
authority: this.provider.wallet.publicKey,
|
|
243
|
-
})
|
|
244
|
-
.rpc(this.config.confirmOptions);
|
|
245
|
-
console.log(`Closed Page ${index}: ${pagePda.toBase58()}`);
|
|
246
|
-
return tx;
|
|
247
|
-
}
|
|
248
|
-
/**
|
|
249
|
-
* Close a queue head to reclaim rent.
|
|
250
|
-
*/
|
|
251
|
-
async closeQueue(queueId) {
|
|
252
|
-
const queuePda = (0, utils_1.deriveQueuePda)(this.program.programId, this.provider.wallet.publicKey, queueId);
|
|
253
|
-
const tx = await this.program.methods
|
|
254
|
-
.closeQueue(queueId)
|
|
255
|
-
.accounts({
|
|
256
|
-
// queue: queuePda, // Auto-resolved
|
|
257
|
-
authority: this.provider.wallet.publicKey,
|
|
258
|
-
})
|
|
259
|
-
.rpc(this.config.confirmOptions);
|
|
260
|
-
console.log(`Closed Queue: ${queuePda.toBase58()}`);
|
|
261
|
-
return tx;
|
|
262
|
-
}
|
|
263
|
-
}
|
|
264
|
-
exports.MatchmakingClient = MatchmakingClient;
|
|
265
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"client.js","sourceRoot":"","sources":["../src/client.ts"],"names":[],"mappings":"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","sourcesContent":["import * as anchor from \"@coral-xyz/anchor\";\nimport { Program, Idl, AnchorProvider } from \"@coral-xyz/anchor\";\nimport { \n    PublicKey, \n    SystemProgram, \n    Keypair, \n    TransactionSignature,\n    ComputeBudgetProgram\n} from \"@solana/web3.js\";\nimport { PrivateMatchmaking } from \"./idl/private_matchmaking\";\nimport { MatchmakingClientConfig, QueueHead, QueuePage, PlayerStatus, JoinQueueResult } from \"./types\";\nimport { derivePagePda, derivePlayerStatusPda, deriveQueuePda } from \"./utils\";\nimport { EncryptionProvider } from \"./encryption\";\n\n// Default Validator for Devnet (MagicBlock)\nconst DEFAULT_VALIDATOR = \"FnE6VJT5QNZdedZPnCoLsARgBwoE6DeJNjBs2H1gySXA\";\n\nexport class MatchmakingClient {\n    program: Program<PrivateMatchmaking>;\n    provider: AnchorProvider;\n    config: MatchmakingClientConfig;\n    encryption: EncryptionProvider;\n\n    constructor(\n        provider: AnchorProvider, \n        programId: PublicKey,\n        config: MatchmakingClientConfig = {}\n    ) {\n        this.provider = provider;\n        this.config = config;\n        this.encryption = new EncryptionProvider();\n        \n        // Load the IDL (we assume it's bundled or we can require it if running in node)\n        // For SDK purity, we should import the JSON.\n        const idl = require(\"./idl/private_matchmaking.json\");\n        idl.address = programId.toBase58();\n        this.program = new Program(idl, provider);\n    }\n\n    /**\n     * Initialize a new matchmaking queue.\n     */\n    async initializeQueue(\n        queueId: string, \n        config: any, \n        capacity: number,\n        pageSize: number = 50\n    ): Promise<PublicKey> {\n        const queuePda = deriveQueuePda(this.program.programId, this.provider.wallet.publicKey, queueId);\n\n        // Ensure tenant ID is set\n        const tenantProgramId = config.tenantProgramId || this.provider.wallet.publicKey;\n\n        const tx = await this.program.methods\n            .initializeQueue(queueId, config, capacity, pageSize)\n            .accounts({\n                // queue: queuePda,\n                // authority: this.provider.wallet.publicKey,\n                tenantProgramId: tenantProgramId,\n                // systemProgram: SystemProgram.programId,\n            })\n            .rpc(this.config.confirmOptions);\n        \n        console.log(`Initialized Queue: ${queuePda.toBase58()} in tx: ${tx}`);\n        \n        // Initialize Pages (Ring Buffer)\n        for (let i = 0; i < capacity; i++) {\n            await this.initializePage(queuePda, i);\n        }\n        \n        return queuePda;\n    }\n\n    /**\n     * Initialize a specific page for the queue.\n     */\n    async initializePage(queue: PublicKey, index: number): Promise<PublicKey> {\n        const pagePda = derivePagePda(this.program.programId, queue, index);\n        \n        const tx = await this.program.methods\n            .initializePage(new anchor.BN(index))\n            .accounts({\n                queue: queue,\n                // page: pagePda,\n                // authority: this.provider.wallet.publicKey,\n                // systemProgram: SystemProgram.programId,\n            })\n            .rpc(this.config.confirmOptions);\n            \n        console.log(`Initialized Page ${index}: ${pagePda.toBase58()}`);\n        return pagePda;\n    }\n\n    /**\n     * Delegate the queue to the Privacy Layer (Ephemeral Rollup).\n     */\n    async delegateQueue(queueId: string, validatorOverride?: PublicKey): Promise<TransactionSignature> {\n        const validator = validatorOverride || new PublicKey(DEFAULT_VALIDATOR);\n        const queuePda = deriveQueuePda(this.program.programId, this.provider.wallet.publicKey, queueId);\n\n        const tx = await this.program.methods\n            .delegateQueue(queueId)\n            .preInstructions([\n                ComputeBudgetProgram.requestHeapFrame({ bytes: 256 * 1024 })\n            ])\n            .accounts({\n                // pda: queuePda,\n                // authority: this.provider.wallet.publicKey,\n                // payer: this.provider.wallet.publicKey,\n                validator: validator,\n            })\n            .rpc(this.config.confirmOptions);\n            \n        console.log(`Delegated Queue ${queueId}: ${tx}`);\n        return tx;\n    }\n\n    /**\n     * Join the queue.\n     */\n    async joinQueue(\n        queue: PublicKey, \n        playerGameAccount: PublicKey, \n        tenantProgramId: PublicKey\n    ): Promise<JoinQueueResult> {\n        // Fetch queue to determine current WRITE index\n        const queueAccount = await this.getQueue(queue);\n        const capacity = queueAccount.capacity;\n        const writeIndex = queueAccount.writePageIndex.toNumber();\n        const currentIndex = writeIndex % capacity;\n\n        const pagePda = derivePagePda(this.program.programId, queue, currentIndex);\n        const statusPda = derivePlayerStatusPda(this.program.programId, playerGameAccount);\n\n        // Encryption Handshake Integration\n        let instructionData: Buffer | undefined;\n        if (this.config.encrypted) {\n            console.log(\"🔒 Encrypted Queue Join Init...\");\n            // TODO: Fetch Validator Key from on-chain Registry or Delegate Account\n            // For now, using a mock dummy key for demonstration of the flow\n            // In PROD: const validatorKey = await this.getValidatorKey(queue);\n            const mockValidatorKey = await this.encryption.createMockValidatorKey(); // Valid P-256 Public Key\n            \n            // Perform Encryption\n            // Real payload would vary (e.g. Rock/Paper/Scissor choice)\n            const payload = Buffer.from(\"PLAYER_CHOICE_ROCK\");\n            const { encrypted, clientPublicKey } = await this.encryption.encryptPayload(payload, mockValidatorKey);\n            console.log(\"🔒 Payload Encrypted. Client PubKey:\", Buffer.from(clientPublicKey).toString('hex'));\n            \n            // In a real implementation, 'encrypted' and 'clientPublicKey' would be passed \n            // as arguments to the 'joinQueue' instruction modification.\n            // Since the IDL isn't updated for arguments yet, we just log it.\n        }\n\n        const tx = await this.program.methods\n            .joinQueue()\n            .accounts({\n                queue: queue,\n                // page: pagePda,\n                page: pagePda, \n                // playerStatus: statusPda,\n                // playerAuthority: this.provider.wallet.publicKey,\n                playerGameAccount: playerGameAccount,\n                tenantProgram: tenantProgramId, // Added for Matchable Interface CPI\n                // systemProgram: SystemProgram.programId,\n            })\n            .rpc(this.config.confirmOptions);\n\n        console.log(`Joined Queue at Page ${currentIndex}: ${tx} (Lock: ${statusPda.toBase58()})`);\n        \n        // Future Logic: If SDK receives a return log/event indicating \"Instant Match\", parse it.\n        // For now, default to \"Queued\".\n        return { \n            status: \"Queued\", \n            tx, \n            statusPda \n        };\n    }\n\n    /**\n     * Unlock a player manually (refund rent).\n     */\n    async unlockPlayer(\n        playerGameAccount: PublicKey, \n        playerWallet: PublicKey\n    ): Promise<TransactionSignature> {\n        const statusPda = derivePlayerStatusPda(this.program.programId, playerGameAccount);\n        const statusAccount = await this.getPlayerStatus(statusPda);\n        const queueAddress = statusAccount.queue;\n\n        const tx = await this.program.methods\n            .unlockPlayer()\n            .accounts({\n                queue: queueAddress,\n                // authority: this.provider.wallet.publicKey,\n                // playerStatus: statusPda,\n                player: playerWallet,\n                playerGameAccount: playerGameAccount\n            })\n            .rpc(this.config.confirmOptions);\n        \n        console.log(`Unlocked Player: ${tx}`);\n        return tx;\n    }\n\n    /**\n     * Process matches on a specific page.\n     * Usually called by the off-chain worker or manually for testing.\n     */\n    async processMatch(\n        queue: PublicKey,\n        pageIndex: number\n    ): Promise<TransactionSignature> {\n        const pagePda = derivePagePda(this.program.programId, queue, pageIndex);\n\n        const tx = await this.program.methods\n            .processMatch(new anchor.BN(pageIndex))\n            .accounts({\n                queueAccount: queue,\n                // page: pagePda,\n            })\n            .rpc(this.config.confirmOptions);\n\n        return tx;\n    }\n    \n    /**\n     * Resize the queue capacity.\n     */\n    async resizeQueue(queue: PublicKey, newCapacity: number): Promise<TransactionSignature> {\n        const tx = await this.program.methods\n            .resizeQueue(newCapacity)\n            .accounts({\n                queue: queue,\n                // authority: this.provider.wallet.publicKey,\n            })\n            .rpc(this.config.confirmOptions);\n        console.log(`Resized Queue to ${newCapacity}: ${tx}`);\n        return tx;\n    }\n\n    // --- State Fetchers ---\n\n    async getQueue(queuePda: PublicKey): Promise<QueueHead> {\n        return await this.program.account.queueHead.fetch(queuePda);\n    }\n\n    async getPage(pagePda: PublicKey): Promise<QueuePage> {\n        return await this.program.account.queuePage.fetch(pagePda);\n    }\n\n    async getPlayerStatus(statusPda: PublicKey): Promise<PlayerStatus> {\n        return await this.program.account.playerStatus.fetch(statusPda);\n    }\n    \n    async getPlayerStatusForGameAccount(playerGameAccount: PublicKey): Promise<PlayerStatus> {\n         const statusPda = derivePlayerStatusPda(this.program.programId, playerGameAccount);\n         return await this.getPlayerStatus(statusPda);\n    }\n\n    // --- Dev Helpers ---\n\n    async createMockPlayer(\n        playerAccount: Keypair, \n        elo: number\n    ): Promise<TransactionSignature> {\n        const tx = await this.program.methods\n            .createMockPlayer(new anchor.BN(elo))\n            .accounts({\n                playerAccount: playerAccount.publicKey,\n                authority: this.provider.wallet.publicKey,\n                // systemProgram: SystemProgram.programId,\n            })\n            .signers([playerAccount])\n            .rpc(this.config.confirmOptions);\n        return tx;\n    }\n\n    /**\n     * Close a queue page to reclaim rent.\n     */\n    async closePage(queue: PublicKey, index: number): Promise<TransactionSignature> {\n        const pagePda = derivePagePda(this.program.programId, queue, index);\n        const tx = await this.program.methods\n            .closePage(new anchor.BN(index))\n            .accounts({\n                queue: queue,\n                // page: pagePda, // Auto-resolved\n                authority: this.provider.wallet.publicKey,\n            })\n            .rpc(this.config.confirmOptions);\n        console.log(`Closed Page ${index}: ${pagePda.toBase58()}`);\n        return tx;\n    }\n\n    /**\n     * Close a queue head to reclaim rent.\n     */\n    async closeQueue(queueId: string): Promise<TransactionSignature> {\n        const queuePda = deriveQueuePda(this.program.programId, this.provider.wallet.publicKey, queueId);\n        const tx = await this.program.methods\n            .closeQueue(queueId)\n            .accounts({\n                // queue: queuePda, // Auto-resolved\n                authority: this.provider.wallet.publicKey,\n            })\n            .rpc(this.config.confirmOptions);\n        console.log(`Closed Queue: ${queuePda.toBase58()}`);\n        return tx;\n    }\n}\n"]}
|