Skip to content

Commit 2724816

Browse files
committed
feat: add a staking endpoint to return a stake summary per account
1 parent ba09a52 commit 2724816

File tree

7 files changed

+255
-5
lines changed

7 files changed

+255
-5
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,53 @@
1+
import type { PositionState } from "@pythnetwork/staking-sdk";
2+
import {
3+
PythStakingClient,
4+
summarizeAccountPositions,
5+
getCurrentEpoch,
6+
} from "@pythnetwork/staking-sdk";
7+
import { WalletAdapterNetwork } from "@solana/wallet-adapter-base";
8+
import { clusterApiUrl, Connection } from "@solana/web3.js";
9+
10+
import {
11+
AMOUNT_STAKED_PER_ACCOUNT_SECRET,
12+
MAINNET_API_RPC,
13+
} from "../../../../config/server";
14+
15+
export const maxDuration = 800;
16+
17+
export const GET = async (req: Request) => {
18+
if (
19+
AMOUNT_STAKED_PER_ACCOUNT_SECRET === undefined ||
20+
req.headers.get("authorization") ===
21+
`Bearer ${AMOUNT_STAKED_PER_ACCOUNT_SECRET}`
22+
) {
23+
const [accounts, epoch] = await Promise.all([
24+
client.getAllStakeAccountPositionsAllOwners(),
25+
getCurrentEpoch(client.connection),
26+
]);
27+
return Response.json(
28+
accounts.map((account) => {
29+
const summary = summarizeAccountPositions(account, epoch);
30+
return [
31+
account.data.owner,
32+
{
33+
voting: stringifySummaryValues(summary.voting),
34+
integrityPool: stringifySummaryValues(summary.integrityPool),
35+
},
36+
];
37+
}),
38+
);
39+
} else {
40+
return new Response("Unauthorized", { status: 400 });
41+
}
42+
};
43+
44+
const stringifySummaryValues = (values: Record<PositionState, bigint>) =>
45+
Object.fromEntries(
46+
Object.entries(values).map(([state, value]) => [state, value.toString()]),
47+
);
48+
49+
const client = new PythStakingClient({
50+
connection: new Connection(
51+
MAINNET_API_RPC ?? clusterApiUrl(WalletAdapterNetwork.Mainnet),
52+
),
53+
});

apps/staking/src/config/server.ts

+4
Original file line numberDiff line numberDiff line change
@@ -80,6 +80,10 @@ export const SIMULATION_PAYER_ADDRESS = getOr(
8080
"SIMULATION_PAYER_ADDRESS",
8181
"E5KR7yfb9UyVB6ZhmhQki1rM1eBcxHvyGKFZakAC5uc",
8282
);
83+
export const AMOUNT_STAKED_PER_ACCOUNT_SECRET = demandInProduction(
84+
"AMOUNT_STAKED_PER_ACCOUNT_SECRET",
85+
);
86+
8387
class MissingEnvironmentError extends Error {
8488
constructor(name: string) {
8589
super(`Missing environment variable: ${name}!`);

apps/staking/turbo.json

+2-1
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,8 @@
1212
"MAINNET_API_RPC",
1313
"BLOCKED_REGIONS",
1414
"AMPLITUDE_API_KEY",
15-
"GOOGLE_ANALYTICS_ID"
15+
"GOOGLE_ANALYTICS_ID",
16+
"AMOUNT_STAKED_PER_ACCOUNT_SECRET"
1617
]
1718
},
1819
"start:dev": {

governance/pyth_staking_sdk/package.json

+8-1
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,9 @@
88
"files": [
99
"dist/**/*"
1010
],
11+
"engines": {
12+
"node": "22"
13+
},
1114
"publishConfig": {
1215
"access": "public"
1316
},
@@ -28,6 +31,7 @@
2831
"@solana/wallet-adapter-react": "catalog:",
2932
"@types/jest": "catalog:",
3033
"@types/node": "catalog:",
34+
"@types/stream-json": "^1.7.8",
3135
"eslint": "catalog:",
3236
"jest": "catalog:",
3337
"prettier": "catalog:",
@@ -39,6 +43,9 @@
3943
"@pythnetwork/solana-utils": "workspace:*",
4044
"@solana/spl-governance": "^0.3.28",
4145
"@solana/spl-token": "^0.3.7",
42-
"@solana/web3.js": "catalog:"
46+
"@solana/web3.js": "catalog:",
47+
"stream-chain": "^3.4.0",
48+
"stream-json": "^1.9.1",
49+
"zod": "catalog:"
4350
}
4451
}

governance/pyth_staking_sdk/src/pyth-staking-client.ts

+116
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,12 @@ import {
2222
Transaction,
2323
TransactionInstruction,
2424
} from "@solana/web3.js";
25+
import { chain } from "stream-chain";
26+
import { parser } from "stream-json";
27+
import { ignore } from "stream-json/filters/Ignore";
28+
import { pick } from "stream-json/filters/Pick";
29+
import { streamArray } from "stream-json/streamers/StreamArray";
30+
import { z } from "zod";
2531

2632
import {
2733
GOVERNANCE_ADDRESS,
@@ -1031,4 +1037,114 @@ export class PythStakingClient {
10311037

10321038
return getAccount(this.connection, rewardCustodyAccountAddress);
10331039
}
1040+
1041+
/**
1042+
* Return all stake account positions for all owners. Note that this method
1043+
* is unique in a few ways:
1044+
*
1045+
* 1. It's very, very expensive. Don't call it if you don't _really_ need it,
1046+
* and expect it to take a few minutes to respond.
1047+
* 2. Because the full positionData is so large, json parsing it with a
1048+
* typical json parser would involve buffering to a string that's too large
1049+
* for node. So instead we use `stream-json` to parse it as a stream.
1050+
*/
1051+
public async getAllStakeAccountPositionsAllOwners(): Promise<
1052+
StakeAccountPositions[]
1053+
> {
1054+
const res = await fetch(this.connection.rpcEndpoint, {
1055+
method: "POST",
1056+
headers: {
1057+
"Content-Type": "application/json",
1058+
},
1059+
body: JSON.stringify({
1060+
jsonrpc: "2.0",
1061+
id: 1,
1062+
method: "getProgramAccounts",
1063+
params: [
1064+
this.stakingProgram.programId.toBase58(),
1065+
{
1066+
encoding: "base64",
1067+
filters: [
1068+
{
1069+
memcmp: this.stakingProgram.coder.accounts.memcmp(
1070+
"positionData",
1071+
) as {
1072+
offset: number;
1073+
bytes: string;
1074+
},
1075+
},
1076+
],
1077+
},
1078+
],
1079+
}),
1080+
});
1081+
1082+
if (res.ok) {
1083+
const { body } = res;
1084+
if (body) {
1085+
const accounts = await new Promise<unknown[]>((resolve) => {
1086+
const pipeline = chain([
1087+
body,
1088+
parser(),
1089+
pick({ filter: "result" }),
1090+
ignore({ filter: /(owner|executable|lamports|rentEpoch)/i }),
1091+
streamArray(),
1092+
]);
1093+
1094+
const accounts: unknown[] = [];
1095+
pipeline.on("data", (data: unknown) => accounts.push(data));
1096+
pipeline.on("end", () => {
1097+
resolve(accounts);
1098+
});
1099+
});
1100+
1101+
return accountSchema
1102+
.parse(accounts)
1103+
.map(({ value }) =>
1104+
deserializeStakeAccountPositions(
1105+
value.pubkey,
1106+
value.account.data,
1107+
this.stakingProgram.idl,
1108+
),
1109+
);
1110+
} else {
1111+
throw new NoBodyError();
1112+
}
1113+
} else {
1114+
throw new NotOKError(res);
1115+
}
1116+
}
1117+
}
1118+
1119+
const accountSchema = z.array(
1120+
z.object({
1121+
value: z.object({
1122+
account: z.object({
1123+
data: z
1124+
.array(z.string())
1125+
.min(1)
1126+
.transform((data) =>
1127+
// Safe because `min(1)` guarantees that `data` is nonempty
1128+
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
1129+
Buffer.from(data[0]!, "base64"),
1130+
),
1131+
}),
1132+
pubkey: z.string().transform((value) => new PublicKey(value)),
1133+
}),
1134+
}),
1135+
);
1136+
1137+
class NotOKError extends Error {
1138+
constructor(result: Response) {
1139+
super(`Received a ${result.status.toString()} response for ${result.url}`);
1140+
this.cause = result;
1141+
this.name = "NotOKError";
1142+
}
1143+
}
1144+
1145+
class NoBodyError extends Error {
1146+
constructor() {
1147+
super("Response did not contain a body!");
1148+
this.name = "NoBodyError";
1149+
}
10341150
}

governance/pyth_staking_sdk/src/utils/position.ts

+30
Original file line numberDiff line numberDiff line change
@@ -111,3 +111,33 @@ export const getVotingTokenAmount = (
111111
);
112112
return totalVotingTokenAmount;
113113
};
114+
115+
export const summarizeAccountPositions = (
116+
positions: StakeAccountPositions,
117+
epoch: bigint,
118+
) => {
119+
const summary = {
120+
voting: {
121+
[PositionState.LOCKED]: 0n,
122+
[PositionState.LOCKING]: 0n,
123+
[PositionState.PREUNLOCKING]: 0n,
124+
[PositionState.UNLOCKED]: 0n,
125+
[PositionState.UNLOCKING]: 0n,
126+
},
127+
integrityPool: {
128+
[PositionState.LOCKED]: 0n,
129+
[PositionState.LOCKING]: 0n,
130+
[PositionState.PREUNLOCKING]: 0n,
131+
[PositionState.UNLOCKED]: 0n,
132+
[PositionState.UNLOCKING]: 0n,
133+
},
134+
};
135+
for (const position of positions.data.positions) {
136+
const category = position.targetWithParameters.voting
137+
? "voting"
138+
: "integrityPool";
139+
const state = getPositionState(position, epoch);
140+
summary[category][state] += position.amount;
141+
}
142+
return summary;
143+
};

pnpm-lock.yaml

+42-3
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

0 commit comments

Comments
 (0)