solforge 0.2.12 → 0.2.14
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/package.json +1 -5
- package/start.cjs +19 -23
- package/docs/API.md +0 -379
- package/docs/CONFIGURATION.md +0 -407
- package/docs/bun-single-file-executable.md +0 -585
- package/docs/cli-plan.md +0 -154
- package/docs/data-indexing-plan.md +0 -214
- package/docs/gui-roadmap.md +0 -202
- package/scripts/decode-b58.ts +0 -10
- package/scripts/install.sh +0 -112
- package/server/index.ts +0 -5
- package/server/lib/base58.ts +0 -33
- package/server/lib/faucet.ts +0 -110
- package/server/lib/instruction-parser.ts +0 -328
- package/server/lib/parsers/spl-associated-token-account.ts +0 -50
- package/server/lib/parsers/spl-token.ts +0 -340
- package/server/lib/spl-token.ts +0 -57
- package/server/methods/TEMPLATE.md +0 -117
- package/server/methods/account/get-account-info.ts +0 -86
- package/server/methods/account/get-balance.ts +0 -23
- package/server/methods/account/get-multiple-accounts.ts +0 -84
- package/server/methods/account/get-parsed-account-info.ts +0 -17
- package/server/methods/account/index.ts +0 -12
- package/server/methods/account/parsers/index.ts +0 -52
- package/server/methods/account/parsers/loader-upgradeable.ts +0 -79
- package/server/methods/account/parsers/spl-token.ts +0 -256
- package/server/methods/account/parsers/system.ts +0 -4
- package/server/methods/account/request-airdrop.ts +0 -271
- package/server/methods/admin/adopt-mint-authority.ts +0 -94
- package/server/methods/admin/clone-program-accounts.ts +0 -55
- package/server/methods/admin/clone-program.ts +0 -152
- package/server/methods/admin/clone-token-accounts.ts +0 -117
- package/server/methods/admin/clone-token-mint.ts +0 -82
- package/server/methods/admin/create-mint.ts +0 -114
- package/server/methods/admin/create-token-account.ts +0 -137
- package/server/methods/admin/helpers.ts +0 -70
- package/server/methods/admin/index.ts +0 -10
- package/server/methods/admin/list-mints.ts +0 -21
- package/server/methods/admin/load-program.ts +0 -52
- package/server/methods/admin/mint-to.ts +0 -266
- package/server/methods/block/get-block-height.ts +0 -5
- package/server/methods/block/get-block.ts +0 -31
- package/server/methods/block/get-blocks-with-limit.ts +0 -19
- package/server/methods/block/get-latest-blockhash.ts +0 -12
- package/server/methods/block/get-slot.ts +0 -5
- package/server/methods/block/index.ts +0 -6
- package/server/methods/block/is-blockhash-valid.ts +0 -19
- package/server/methods/epoch/get-cluster-nodes.ts +0 -17
- package/server/methods/epoch/get-epoch-info.ts +0 -16
- package/server/methods/epoch/get-epoch-schedule.ts +0 -15
- package/server/methods/epoch/get-highest-snapshot-slot.ts +0 -9
- package/server/methods/epoch/get-leader-schedule.ts +0 -8
- package/server/methods/epoch/get-max-retransmit-slot.ts +0 -9
- package/server/methods/epoch/get-max-shred-insert-slot.ts +0 -9
- package/server/methods/epoch/get-slot-leader.ts +0 -6
- package/server/methods/epoch/get-slot-leaders.ts +0 -9
- package/server/methods/epoch/get-stake-activation.ts +0 -9
- package/server/methods/epoch/get-stake-minimum-delegation.ts +0 -9
- package/server/methods/epoch/get-vote-accounts.ts +0 -19
- package/server/methods/epoch/index.ts +0 -13
- package/server/methods/epoch/minimum-ledger-slot.ts +0 -5
- package/server/methods/fee/get-fee-calculator-for-blockhash.ts +0 -12
- package/server/methods/fee/get-fee-for-message.ts +0 -8
- package/server/methods/fee/get-fee-rate-governor.ts +0 -16
- package/server/methods/fee/get-fees.ts +0 -14
- package/server/methods/fee/get-recent-prioritization-fees.ts +0 -22
- package/server/methods/fee/index.ts +0 -5
- package/server/methods/get-address-lookup-table.ts +0 -27
- package/server/methods/index.ts +0 -265
- package/server/methods/performance/get-recent-performance-samples.ts +0 -25
- package/server/methods/performance/get-transaction-count.ts +0 -5
- package/server/methods/performance/index.ts +0 -2
- package/server/methods/program/get-block-commitment.ts +0 -9
- package/server/methods/program/get-block-production.ts +0 -14
- package/server/methods/program/get-block-time.ts +0 -21
- package/server/methods/program/get-blocks.ts +0 -11
- package/server/methods/program/get-first-available-block.ts +0 -9
- package/server/methods/program/get-genesis-hash.ts +0 -6
- package/server/methods/program/get-identity.ts +0 -6
- package/server/methods/program/get-inflation-governor.ts +0 -15
- package/server/methods/program/get-inflation-rate.ts +0 -10
- package/server/methods/program/get-inflation-reward.ts +0 -12
- package/server/methods/program/get-largest-accounts.ts +0 -8
- package/server/methods/program/get-parsed-program-accounts.ts +0 -12
- package/server/methods/program/get-parsed-token-accounts-by-delegate.ts +0 -12
- package/server/methods/program/get-parsed-token-accounts-by-owner.ts +0 -12
- package/server/methods/program/get-program-accounts.ts +0 -221
- package/server/methods/program/get-supply.ts +0 -13
- package/server/methods/program/get-token-account-balance.ts +0 -60
- package/server/methods/program/get-token-accounts-by-delegate.ts +0 -82
- package/server/methods/program/get-token-accounts-by-owner.ts +0 -416
- package/server/methods/program/get-token-largest-accounts.ts +0 -81
- package/server/methods/program/get-token-supply.ts +0 -39
- package/server/methods/program/index.ts +0 -21
- package/server/methods/solforge/index.ts +0 -158
- package/server/methods/system/get-health.ts +0 -5
- package/server/methods/system/get-minimum-balance-for-rent-exemption.ts +0 -13
- package/server/methods/system/get-version.ts +0 -9
- package/server/methods/system/index.ts +0 -3
- package/server/methods/transaction/get-confirmed-transaction.ts +0 -11
- package/server/methods/transaction/get-parsed-transaction.ts +0 -17
- package/server/methods/transaction/get-signature-statuses.ts +0 -79
- package/server/methods/transaction/get-signatures-for-address.ts +0 -41
- package/server/methods/transaction/get-transaction.ts +0 -639
- package/server/methods/transaction/index.ts +0 -7
- package/server/methods/transaction/inner-instructions.test.ts +0 -104
- package/server/methods/transaction/send-transaction.ts +0 -469
- package/server/methods/transaction/simulate-transaction.ts +0 -57
- package/server/rpc-server.ts +0 -521
- package/server/types.ts +0 -109
- package/server/ws-server.ts +0 -178
- package/src/api-server-entry.ts +0 -109
- package/src/cli/bootstrap.ts +0 -67
- package/src/cli/commands/airdrop.ts +0 -37
- package/src/cli/commands/config.ts +0 -39
- package/src/cli/commands/mint.ts +0 -187
- package/src/cli/commands/program-clone.ts +0 -122
- package/src/cli/commands/program-load.ts +0 -64
- package/src/cli/commands/rpc-start.ts +0 -49
- package/src/cli/commands/token-adopt-authority.ts +0 -37
- package/src/cli/commands/token-clone.ts +0 -112
- package/src/cli/commands/token-create.ts +0 -81
- package/src/cli/main.ts +0 -158
- package/src/cli/run-solforge.ts +0 -112
- package/src/cli/setup-utils.ts +0 -54
- package/src/cli/setup-wizard.ts +0 -258
- package/src/cli/utils/args.ts +0 -15
- package/src/commands/add-program.ts +0 -333
- package/src/commands/init.ts +0 -122
- package/src/commands/list.ts +0 -136
- package/src/commands/mint.ts +0 -287
- package/src/commands/start.ts +0 -881
- package/src/commands/status.ts +0 -99
- package/src/commands/stop.ts +0 -405
- package/src/config/index.ts +0 -146
- package/src/config/manager.ts +0 -157
- package/src/db/index.ts +0 -83
- package/src/db/schema/accounts.ts +0 -23
- package/src/db/schema/address-signatures.ts +0 -31
- package/src/db/schema/index.ts +0 -6
- package/src/db/schema/meta-kv.ts +0 -9
- package/src/db/schema/transactions.ts +0 -36
- package/src/db/schema/tx-account-states.ts +0 -23
- package/src/db/schema/tx-accounts.ts +0 -33
- package/src/db/tx-store.ts +0 -264
- package/src/gui/public/app.css +0 -1556
- package/src/gui/public/build/main.css +0 -1569
- package/src/gui/public/build/main.js +0 -303
- package/src/gui/public/build/main.js.txt +0 -231
- package/src/gui/public/index.html +0 -19
- package/src/gui/server.ts +0 -296
- package/src/gui/src/api.ts +0 -127
- package/src/gui/src/app.tsx +0 -441
- package/src/gui/src/components/airdrop-mint-form.tsx +0 -246
- package/src/gui/src/components/clone-program-modal.tsx +0 -202
- package/src/gui/src/components/clone-token-modal.tsx +0 -230
- package/src/gui/src/components/modal.tsx +0 -134
- package/src/gui/src/components/programs-panel.tsx +0 -124
- package/src/gui/src/components/status-panel.tsx +0 -136
- package/src/gui/src/components/tokens-panel.tsx +0 -122
- package/src/gui/src/hooks/use-interval.ts +0 -17
- package/src/gui/src/index.css +0 -557
- package/src/gui/src/main.tsx +0 -17
- package/src/index.ts +0 -216
- package/src/migrations-bundled.ts +0 -23
- package/src/rpc/start.ts +0 -44
- package/src/services/api-server.ts +0 -504
- package/src/services/port-manager.ts +0 -174
- package/src/services/process-registry.ts +0 -153
- package/src/services/program-cloner.ts +0 -317
- package/src/services/token-cloner.ts +0 -811
- package/src/services/validator.ts +0 -293
- package/src/types/config.ts +0 -110
- package/src/utils/shell.ts +0 -110
- package/src/utils/token-loader.ts +0 -115
package/server/rpc-server.ts
DELETED
|
@@ -1,521 +0,0 @@
|
|
|
1
|
-
import {
|
|
2
|
-
type Keypair,
|
|
3
|
-
PublicKey,
|
|
4
|
-
type VersionedTransaction,
|
|
5
|
-
} from "@solana/web3.js";
|
|
6
|
-
import { LiteSVM } from "litesvm";
|
|
7
|
-
import { sqlite } from "../src/db/index";
|
|
8
|
-
import { TxStore } from "../src/db/tx-store";
|
|
9
|
-
import { decodeBase58, encodeBase58 } from "./lib/base58";
|
|
10
|
-
import { fundFaucetIfNeeded, loadOrCreateFaucet } from "./lib/faucet";
|
|
11
|
-
import { rpcMethods } from "./methods";
|
|
12
|
-
import type {
|
|
13
|
-
JsonRpcRequest,
|
|
14
|
-
JsonRpcResponse,
|
|
15
|
-
RpcMethodContext,
|
|
16
|
-
} from "./types";
|
|
17
|
-
|
|
18
|
-
export class LiteSVMRpcServer {
|
|
19
|
-
private svm: LiteSVM;
|
|
20
|
-
private slot: bigint = 1n;
|
|
21
|
-
private blockHeight: bigint = 1n;
|
|
22
|
-
private txCount: bigint = 0n;
|
|
23
|
-
private signatureListeners: Set<(sig: string) => void> = new Set();
|
|
24
|
-
private knownMints: Set<string> = new Set();
|
|
25
|
-
private knownPrograms: Set<string> = new Set();
|
|
26
|
-
private faucet: Keypair;
|
|
27
|
-
private txRecords: Map<
|
|
28
|
-
string,
|
|
29
|
-
{
|
|
30
|
-
tx: VersionedTransaction;
|
|
31
|
-
logs: string[];
|
|
32
|
-
err: unknown;
|
|
33
|
-
fee: number;
|
|
34
|
-
slot: number;
|
|
35
|
-
blockTime?: number;
|
|
36
|
-
preBalances?: number[];
|
|
37
|
-
postBalances?: number[];
|
|
38
|
-
preTokenBalances?: unknown[];
|
|
39
|
-
postTokenBalances?: unknown[];
|
|
40
|
-
}
|
|
41
|
-
> = new Map();
|
|
42
|
-
private store: TxStore;
|
|
43
|
-
|
|
44
|
-
constructor() {
|
|
45
|
-
this.svm = new LiteSVM()
|
|
46
|
-
.withSysvars()
|
|
47
|
-
.withBuiltins()
|
|
48
|
-
.withDefaultPrograms()
|
|
49
|
-
// Mint 1,000,000 SOL (1e15 lamports) for local dev
|
|
50
|
-
.withLamports(1_000_000_000_000_000n)
|
|
51
|
-
.withBlockhashCheck(true)
|
|
52
|
-
// keep some tx history so getTransaction/getSignatureStatuses can work
|
|
53
|
-
.withTransactionHistory(1000n)
|
|
54
|
-
.withSigverify(false);
|
|
55
|
-
this.store = new TxStore();
|
|
56
|
-
// Seed slot/blockHeight/txCount from DB if available for continuity
|
|
57
|
-
try {
|
|
58
|
-
const maxRow = sqlite
|
|
59
|
-
.prepare("SELECT MAX(slot) as m FROM transactions")
|
|
60
|
-
.get() as { m?: number } | undefined;
|
|
61
|
-
const cntRow = sqlite
|
|
62
|
-
.prepare("SELECT COUNT(1) as c FROM transactions")
|
|
63
|
-
.get() as { c?: number } | undefined;
|
|
64
|
-
const maxSlot = (maxRow?.m ?? 0) as number;
|
|
65
|
-
const txc = (cntRow?.c ?? 0) as number;
|
|
66
|
-
if (maxSlot > 0) {
|
|
67
|
-
this.slot = BigInt(maxSlot + 1);
|
|
68
|
-
this.blockHeight = BigInt(maxSlot + 1);
|
|
69
|
-
}
|
|
70
|
-
if (txc > 0) {
|
|
71
|
-
this.txCount = BigInt(txc);
|
|
72
|
-
}
|
|
73
|
-
} catch {}
|
|
74
|
-
|
|
75
|
-
// Load or create faucet; fund once at startup
|
|
76
|
-
this.faucet = loadOrCreateFaucet();
|
|
77
|
-
try {
|
|
78
|
-
const bal = fundFaucetIfNeeded(this.svm, this.faucet);
|
|
79
|
-
console.log(
|
|
80
|
-
`💧 Faucet loaded: ${this.faucet.publicKey.toBase58()} with ${(Number(bal) / 1_000_000_000).toFixed(0)} SOL`,
|
|
81
|
-
);
|
|
82
|
-
} catch (e) {
|
|
83
|
-
console.warn("⚠️ Faucet funding failed:", e);
|
|
84
|
-
try {
|
|
85
|
-
const bal =
|
|
86
|
-
this.svm.getBalance(this.faucet.publicKey as PublicKey) || 0n;
|
|
87
|
-
console.log(
|
|
88
|
-
`💧 Faucet balance: ${(Number(bal) / 1_000_000_000).toFixed(9)} SOL`,
|
|
89
|
-
);
|
|
90
|
-
} catch {}
|
|
91
|
-
}
|
|
92
|
-
}
|
|
93
|
-
|
|
94
|
-
// base58 helpers moved to server/lib/base58
|
|
95
|
-
|
|
96
|
-
private createSuccessResponse(
|
|
97
|
-
id: string | number,
|
|
98
|
-
result: unknown,
|
|
99
|
-
): JsonRpcResponse {
|
|
100
|
-
return {
|
|
101
|
-
jsonrpc: "2.0",
|
|
102
|
-
id,
|
|
103
|
-
result,
|
|
104
|
-
};
|
|
105
|
-
}
|
|
106
|
-
|
|
107
|
-
private createErrorResponse(
|
|
108
|
-
id: string | number,
|
|
109
|
-
code: number,
|
|
110
|
-
message: string,
|
|
111
|
-
data?: unknown,
|
|
112
|
-
): JsonRpcResponse {
|
|
113
|
-
return {
|
|
114
|
-
jsonrpc: "2.0",
|
|
115
|
-
id,
|
|
116
|
-
error: { code, message, data },
|
|
117
|
-
};
|
|
118
|
-
}
|
|
119
|
-
|
|
120
|
-
private getContext(): RpcMethodContext {
|
|
121
|
-
return {
|
|
122
|
-
svm: this.svm,
|
|
123
|
-
slot: this.slot,
|
|
124
|
-
blockHeight: this.blockHeight,
|
|
125
|
-
store: this.store,
|
|
126
|
-
encodeBase58,
|
|
127
|
-
decodeBase58,
|
|
128
|
-
createSuccessResponse: this.createSuccessResponse.bind(this),
|
|
129
|
-
createErrorResponse: this.createErrorResponse.bind(this),
|
|
130
|
-
notifySignature: (signature: string) => {
|
|
131
|
-
for (const cb of this.signatureListeners) cb(signature);
|
|
132
|
-
},
|
|
133
|
-
getFaucet: () => this.faucet,
|
|
134
|
-
getTxCount: () => this.txCount,
|
|
135
|
-
registerMint: (mint: PublicKey | string) => {
|
|
136
|
-
try {
|
|
137
|
-
const pk =
|
|
138
|
-
typeof mint === "string" ? mint : new PublicKey(mint).toBase58();
|
|
139
|
-
this.knownMints.add(pk);
|
|
140
|
-
} catch {}
|
|
141
|
-
},
|
|
142
|
-
listMints: () => Array.from(this.knownMints),
|
|
143
|
-
registerProgram: (program: PublicKey | string) => {
|
|
144
|
-
try {
|
|
145
|
-
const pk =
|
|
146
|
-
typeof program === "string"
|
|
147
|
-
? program
|
|
148
|
-
: new PublicKey(program).toBase58();
|
|
149
|
-
this.knownPrograms.add(pk);
|
|
150
|
-
} catch {}
|
|
151
|
-
},
|
|
152
|
-
listPrograms: () => Array.from(this.knownPrograms),
|
|
153
|
-
recordTransaction: (signature, tx, meta) => {
|
|
154
|
-
this.txRecords.set(signature, {
|
|
155
|
-
tx,
|
|
156
|
-
logs: meta?.logs || [],
|
|
157
|
-
err: meta?.err ?? null,
|
|
158
|
-
fee: meta?.fee ?? 5000,
|
|
159
|
-
slot: Number(this.slot),
|
|
160
|
-
blockTime: meta?.blockTime,
|
|
161
|
-
preBalances: meta?.preBalances,
|
|
162
|
-
postBalances: meta?.postBalances,
|
|
163
|
-
preTokenBalances: (
|
|
164
|
-
meta as { preTokenBalances?: unknown[] } | undefined
|
|
165
|
-
)?.preTokenBalances,
|
|
166
|
-
postTokenBalances: (
|
|
167
|
-
meta as { postTokenBalances?: unknown[] } | undefined
|
|
168
|
-
)?.postTokenBalances,
|
|
169
|
-
innerInstructions: meta?.innerInstructions || [],
|
|
170
|
-
computeUnits:
|
|
171
|
-
meta?.computeUnits == null ? null : Number(meta.computeUnits),
|
|
172
|
-
returnData: meta?.returnData ?? null,
|
|
173
|
-
});
|
|
174
|
-
try {
|
|
175
|
-
if (process.env.DEBUG_TX_CAPTURE === "1") {
|
|
176
|
-
console.debug(
|
|
177
|
-
`[tx-capture] recordTransaction: sig=${signature} slot=${this.slot} logs=${meta?.logs?.length || 0} inner=${Array.isArray(meta?.innerInstructions) ? meta?.innerInstructions?.length : 0} cu=${meta?.computeUnits ?? null} returnData=${meta?.returnData ? "yes" : "no"}`,
|
|
178
|
-
);
|
|
179
|
-
}
|
|
180
|
-
} catch {}
|
|
181
|
-
|
|
182
|
-
// Persist to SQLite for durability and history queries
|
|
183
|
-
try {
|
|
184
|
-
const msg = tx.message as unknown as {
|
|
185
|
-
staticAccountKeys?: unknown[];
|
|
186
|
-
accountKeys?: unknown[];
|
|
187
|
-
header?: unknown;
|
|
188
|
-
isAccountSigner?: (i: number) => boolean;
|
|
189
|
-
isAccountWritable?: (i: number) => boolean;
|
|
190
|
-
version?: number;
|
|
191
|
-
};
|
|
192
|
-
const rawKeys: unknown[] = Array.isArray(msg.staticAccountKeys)
|
|
193
|
-
? msg.staticAccountKeys
|
|
194
|
-
: Array.isArray(msg.accountKeys)
|
|
195
|
-
? msg.accountKeys
|
|
196
|
-
: [];
|
|
197
|
-
const keys: string[] = rawKeys.map((k) => {
|
|
198
|
-
try {
|
|
199
|
-
return typeof k === "string" ? k : (k as PublicKey).toBase58();
|
|
200
|
-
} catch {
|
|
201
|
-
return String(k);
|
|
202
|
-
}
|
|
203
|
-
});
|
|
204
|
-
const header = msg.header || {
|
|
205
|
-
numRequiredSignatures: (tx.signatures || []).length,
|
|
206
|
-
numReadonlySignedAccounts: 0,
|
|
207
|
-
numReadonlyUnsignedAccounts: 0,
|
|
208
|
-
};
|
|
209
|
-
const numReq = Number(header.numRequiredSignatures || 0);
|
|
210
|
-
const numRoSigned = Number(header.numReadonlySignedAccounts || 0);
|
|
211
|
-
const numRoUnsigned = Number(header.numReadonlyUnsignedAccounts || 0);
|
|
212
|
-
const total = keys.length;
|
|
213
|
-
const unsignedCount = Math.max(0, total - numReq);
|
|
214
|
-
const writableSignedCutoff = Math.max(0, numReq - numRoSigned);
|
|
215
|
-
const writableUnsignedCount = Math.max(
|
|
216
|
-
0,
|
|
217
|
-
unsignedCount - numRoUnsigned,
|
|
218
|
-
);
|
|
219
|
-
|
|
220
|
-
const accounts = keys.map((addr, i) => {
|
|
221
|
-
const signer =
|
|
222
|
-
typeof msg.isAccountSigner === "function"
|
|
223
|
-
? !!msg.isAccountSigner(i)
|
|
224
|
-
: i < numReq;
|
|
225
|
-
let writable = true;
|
|
226
|
-
if (typeof msg.isAccountWritable === "function")
|
|
227
|
-
writable = !!msg.isAccountWritable(i);
|
|
228
|
-
else {
|
|
229
|
-
if (i < numReq) writable = i < writableSignedCutoff;
|
|
230
|
-
else writable = i - numReq < writableUnsignedCount;
|
|
231
|
-
}
|
|
232
|
-
return { address: addr, index: i, signer, writable };
|
|
233
|
-
});
|
|
234
|
-
const version: 0 | "legacy" =
|
|
235
|
-
typeof msg.version === "number"
|
|
236
|
-
? msg.version === 0
|
|
237
|
-
? 0
|
|
238
|
-
: "legacy"
|
|
239
|
-
: 0;
|
|
240
|
-
const rawBase64 = Buffer.from(tx.serialize()).toString("base64");
|
|
241
|
-
this.store
|
|
242
|
-
.insertTransactionBundle({
|
|
243
|
-
signature,
|
|
244
|
-
slot: Number(this.slot),
|
|
245
|
-
blockTime: meta?.blockTime,
|
|
246
|
-
version,
|
|
247
|
-
fee: Number(meta?.fee ?? 5000),
|
|
248
|
-
err: meta?.err ?? null,
|
|
249
|
-
rawBase64,
|
|
250
|
-
preBalances: Array.isArray(meta?.preBalances)
|
|
251
|
-
? (meta?.preBalances as number[])
|
|
252
|
-
: [],
|
|
253
|
-
postBalances: Array.isArray(meta?.postBalances)
|
|
254
|
-
? (meta?.postBalances as number[])
|
|
255
|
-
: [],
|
|
256
|
-
logs: Array.isArray(meta?.logs) ? (meta?.logs as string[]) : [],
|
|
257
|
-
preTokenBalances: (() => {
|
|
258
|
-
const arr = (
|
|
259
|
-
meta as { preTokenBalances?: unknown[] } | undefined
|
|
260
|
-
)?.preTokenBalances;
|
|
261
|
-
return Array.isArray(arr) ? arr : [];
|
|
262
|
-
})(),
|
|
263
|
-
postTokenBalances: (() => {
|
|
264
|
-
const arr = (
|
|
265
|
-
meta as { postTokenBalances?: unknown[] } | undefined
|
|
266
|
-
)?.postTokenBalances;
|
|
267
|
-
return Array.isArray(arr) ? arr : [];
|
|
268
|
-
})(),
|
|
269
|
-
innerInstructions: Array.isArray(meta?.innerInstructions)
|
|
270
|
-
? meta?.innerInstructions
|
|
271
|
-
: [],
|
|
272
|
-
computeUnits:
|
|
273
|
-
meta?.computeUnits == null ? null : Number(meta.computeUnits),
|
|
274
|
-
returnData: meta?.returnData ?? null,
|
|
275
|
-
accounts,
|
|
276
|
-
accountStates: Array.isArray(meta?.accountStates)
|
|
277
|
-
? meta?.accountStates
|
|
278
|
-
: [],
|
|
279
|
-
})
|
|
280
|
-
.catch(() => {});
|
|
281
|
-
|
|
282
|
-
// Upsert account snapshots for static keys
|
|
283
|
-
const snapshots = keys
|
|
284
|
-
.map((addr) => {
|
|
285
|
-
try {
|
|
286
|
-
const acc = this.svm.getAccount(new PublicKey(addr));
|
|
287
|
-
if (!acc) return null;
|
|
288
|
-
const ownerStr = new PublicKey(acc.owner).toBase58();
|
|
289
|
-
let dataBase64: string | undefined;
|
|
290
|
-
// Store raw data for SPL Token accounts to reflect balance changes
|
|
291
|
-
try {
|
|
292
|
-
if (
|
|
293
|
-
ownerStr ===
|
|
294
|
-
"TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA" ||
|
|
295
|
-
ownerStr === "TokenzQdBNbLqP5VEhdkAS6EPFLC1PHnBqCXEpPxuEb"
|
|
296
|
-
) {
|
|
297
|
-
if (acc.data && acc.data.length > 0) {
|
|
298
|
-
dataBase64 = Buffer.from(acc.data).toString("base64");
|
|
299
|
-
}
|
|
300
|
-
}
|
|
301
|
-
} catch {}
|
|
302
|
-
return {
|
|
303
|
-
address: addr,
|
|
304
|
-
lamports: Number(acc.lamports || 0n),
|
|
305
|
-
ownerProgram: ownerStr,
|
|
306
|
-
executable: !!acc.executable,
|
|
307
|
-
rentEpoch: Number(acc.rentEpoch || 0),
|
|
308
|
-
dataLen: acc.data?.length ?? 0,
|
|
309
|
-
dataBase64,
|
|
310
|
-
lastSlot: Number(this.slot),
|
|
311
|
-
};
|
|
312
|
-
} catch {
|
|
313
|
-
return null;
|
|
314
|
-
}
|
|
315
|
-
})
|
|
316
|
-
.filter(Boolean) as import("../src/db/tx-store").AccountSnapshot[];
|
|
317
|
-
if (snapshots.length > 0)
|
|
318
|
-
this.store.upsertAccounts(snapshots).catch(() => {});
|
|
319
|
-
} catch {}
|
|
320
|
-
},
|
|
321
|
-
getRecordedTransaction: (signature) => this.txRecords.get(signature),
|
|
322
|
-
};
|
|
323
|
-
}
|
|
324
|
-
|
|
325
|
-
onSignatureRecorded(cb: (sig: string) => void) {
|
|
326
|
-
this.signatureListeners.add(cb);
|
|
327
|
-
return () => this.signatureListeners.delete(cb);
|
|
328
|
-
}
|
|
329
|
-
|
|
330
|
-
getFaucetAddress(): string {
|
|
331
|
-
try {
|
|
332
|
-
return this.faucet.publicKey.toBase58();
|
|
333
|
-
} catch {
|
|
334
|
-
return "";
|
|
335
|
-
}
|
|
336
|
-
}
|
|
337
|
-
|
|
338
|
-
getFaucetBalance(): bigint {
|
|
339
|
-
try {
|
|
340
|
-
return this.svm.getBalance(this.faucet.publicKey as PublicKey) ?? 0n;
|
|
341
|
-
} catch {
|
|
342
|
-
return 0n;
|
|
343
|
-
}
|
|
344
|
-
}
|
|
345
|
-
|
|
346
|
-
listKnownMints(): string[] {
|
|
347
|
-
return Array.from(this.knownMints);
|
|
348
|
-
}
|
|
349
|
-
|
|
350
|
-
listKnownPrograms(): string[] {
|
|
351
|
-
return Array.from(this.knownPrograms);
|
|
352
|
-
}
|
|
353
|
-
|
|
354
|
-
getSignatureStatus(
|
|
355
|
-
signature: string,
|
|
356
|
-
): { slot: number; err: unknown | null } | null {
|
|
357
|
-
// Prefer local record for reliability
|
|
358
|
-
const rec = this.txRecords.get(signature);
|
|
359
|
-
if (rec) {
|
|
360
|
-
return { slot: rec.slot, err: rec.err ?? null };
|
|
361
|
-
}
|
|
362
|
-
try {
|
|
363
|
-
const sigBytes = decodeBase58(signature);
|
|
364
|
-
const tx = this.svm.getTransaction(sigBytes);
|
|
365
|
-
if (!tx) return null;
|
|
366
|
-
let errVal: unknown = null;
|
|
367
|
-
try {
|
|
368
|
-
const raw = (tx as { err?: unknown }).err;
|
|
369
|
-
errVal = typeof raw === "function" ? (raw as () => unknown)() : raw;
|
|
370
|
-
} catch {
|
|
371
|
-
errVal = null;
|
|
372
|
-
}
|
|
373
|
-
return { slot: Number(this.slot), err: errVal };
|
|
374
|
-
} catch {
|
|
375
|
-
return null;
|
|
376
|
-
}
|
|
377
|
-
}
|
|
378
|
-
|
|
379
|
-
async handleRequest(request: JsonRpcRequest): Promise<JsonRpcResponse> {
|
|
380
|
-
const { method, params, id } = request;
|
|
381
|
-
|
|
382
|
-
try {
|
|
383
|
-
const methodHandler = rpcMethods[method];
|
|
384
|
-
|
|
385
|
-
if (!methodHandler) {
|
|
386
|
-
return this.createErrorResponse(
|
|
387
|
-
id,
|
|
388
|
-
-32601,
|
|
389
|
-
`Method not found: ${method}`,
|
|
390
|
-
);
|
|
391
|
-
}
|
|
392
|
-
|
|
393
|
-
const context = this.getContext();
|
|
394
|
-
const result = await methodHandler(id, params, context);
|
|
395
|
-
|
|
396
|
-
// Update slot and blockHeight for methods that modify state
|
|
397
|
-
if (["sendTransaction", "requestAirdrop"].includes(method)) {
|
|
398
|
-
this.slot += 1n;
|
|
399
|
-
this.blockHeight += 1n;
|
|
400
|
-
this.txCount += 1n;
|
|
401
|
-
}
|
|
402
|
-
|
|
403
|
-
return result;
|
|
404
|
-
} catch (error: unknown) {
|
|
405
|
-
const message = error instanceof Error ? error.message : String(error);
|
|
406
|
-
return this.createErrorResponse(id, -32603, "Internal error", message);
|
|
407
|
-
}
|
|
408
|
-
}
|
|
409
|
-
}
|
|
410
|
-
|
|
411
|
-
export function createLiteSVMRpcServer(port: number = 8899, host?: string) {
|
|
412
|
-
const server = new LiteSVMRpcServer();
|
|
413
|
-
|
|
414
|
-
const bunServer = Bun.serve({
|
|
415
|
-
port,
|
|
416
|
-
hostname: host || process.env.RPC_HOST || "127.0.0.1",
|
|
417
|
-
async fetch(req) {
|
|
418
|
-
const DEBUG = process.env.DEBUG_RPC_LOG === "1";
|
|
419
|
-
const acrh = req.headers.get("Access-Control-Request-Headers");
|
|
420
|
-
const allowHeaders =
|
|
421
|
-
acrh && acrh.length > 0
|
|
422
|
-
? acrh
|
|
423
|
-
: "Content-Type, Accept, Origin, solana-client";
|
|
424
|
-
const corsHeaders = {
|
|
425
|
-
"Access-Control-Allow-Origin": "*",
|
|
426
|
-
"Access-Control-Allow-Methods": "GET, POST, OPTIONS, HEAD",
|
|
427
|
-
"Access-Control-Allow-Headers": allowHeaders,
|
|
428
|
-
"Access-Control-Max-Age": "600",
|
|
429
|
-
// Help Chrome when accessing local network/localhost from secure context
|
|
430
|
-
"Access-Control-Allow-Private-Network": "true",
|
|
431
|
-
} as const;
|
|
432
|
-
|
|
433
|
-
if (req.method === "GET") {
|
|
434
|
-
const url = new URL(req.url);
|
|
435
|
-
if (url.pathname === "/" || url.pathname === "") {
|
|
436
|
-
return new Response("ok", {
|
|
437
|
-
headers: { "Content-Type": "text/plain", ...corsHeaders },
|
|
438
|
-
});
|
|
439
|
-
}
|
|
440
|
-
if (url.pathname === "/health") {
|
|
441
|
-
return new Response("ok", {
|
|
442
|
-
headers: { "Content-Type": "text/plain", ...corsHeaders },
|
|
443
|
-
});
|
|
444
|
-
}
|
|
445
|
-
return new Response("Not found", { status: 404, headers: corsHeaders });
|
|
446
|
-
}
|
|
447
|
-
if (req.method === "POST") {
|
|
448
|
-
try {
|
|
449
|
-
const body = await req.json();
|
|
450
|
-
|
|
451
|
-
if (Array.isArray(body)) {
|
|
452
|
-
if (DEBUG) {
|
|
453
|
-
try {
|
|
454
|
-
console.log(
|
|
455
|
-
"RPC batch:",
|
|
456
|
-
body.map((b: { method?: string }) => b.method),
|
|
457
|
-
);
|
|
458
|
-
} catch {}
|
|
459
|
-
}
|
|
460
|
-
const responses = await Promise.all(
|
|
461
|
-
body.map((request) => server.handleRequest(request)),
|
|
462
|
-
);
|
|
463
|
-
return new Response(JSON.stringify(responses), {
|
|
464
|
-
headers: { "Content-Type": "application/json", ...corsHeaders },
|
|
465
|
-
});
|
|
466
|
-
} else {
|
|
467
|
-
const reqObj = body as JsonRpcRequest;
|
|
468
|
-
if (DEBUG) {
|
|
469
|
-
try {
|
|
470
|
-
console.log(
|
|
471
|
-
"RPC:",
|
|
472
|
-
reqObj.method,
|
|
473
|
-
JSON.stringify(reqObj.params),
|
|
474
|
-
);
|
|
475
|
-
} catch {}
|
|
476
|
-
}
|
|
477
|
-
const response = await server.handleRequest(reqObj);
|
|
478
|
-
return new Response(JSON.stringify(response), {
|
|
479
|
-
headers: { "Content-Type": "application/json", ...corsHeaders },
|
|
480
|
-
});
|
|
481
|
-
}
|
|
482
|
-
} catch (_error) {
|
|
483
|
-
return new Response(
|
|
484
|
-
JSON.stringify({
|
|
485
|
-
jsonrpc: "2.0",
|
|
486
|
-
id: null,
|
|
487
|
-
error: {
|
|
488
|
-
code: -32700,
|
|
489
|
-
message: "Parse error",
|
|
490
|
-
},
|
|
491
|
-
}),
|
|
492
|
-
{ headers: { "Content-Type": "application/json", ...corsHeaders } },
|
|
493
|
-
);
|
|
494
|
-
}
|
|
495
|
-
}
|
|
496
|
-
|
|
497
|
-
if (req.method === "OPTIONS") {
|
|
498
|
-
return new Response(null, { headers: corsHeaders });
|
|
499
|
-
}
|
|
500
|
-
if (req.method === "HEAD") {
|
|
501
|
-
return new Response(null, { headers: corsHeaders });
|
|
502
|
-
}
|
|
503
|
-
|
|
504
|
-
return new Response("Method not allowed", {
|
|
505
|
-
status: 405,
|
|
506
|
-
headers: corsHeaders,
|
|
507
|
-
});
|
|
508
|
-
},
|
|
509
|
-
error(error) {
|
|
510
|
-
console.error("Server error:", error);
|
|
511
|
-
return new Response("Internal Server Error", { status: 500 });
|
|
512
|
-
},
|
|
513
|
-
});
|
|
514
|
-
|
|
515
|
-
const hostname = (host || process.env.RPC_HOST || "127.0.0.1").toString();
|
|
516
|
-
console.log(`🚀 LiteSVM RPC Server running on http://${hostname}:${port}`);
|
|
517
|
-
console.log(` Compatible with Solana RPC API`);
|
|
518
|
-
console.log(` Use with: solana config set -u http://${hostname}:${port}`);
|
|
519
|
-
|
|
520
|
-
return { httpServer: bunServer, rpcServer: server };
|
|
521
|
-
}
|
package/server/types.ts
DELETED
|
@@ -1,109 +0,0 @@
|
|
|
1
|
-
import type { Keypair, PublicKey, VersionedTransaction } from "@solana/web3.js";
|
|
2
|
-
import type { LiteSVM } from "litesvm";
|
|
3
|
-
import type { TxStore } from "../src/db/tx-store";
|
|
4
|
-
|
|
5
|
-
export interface JsonRpcRequest {
|
|
6
|
-
jsonrpc: "2.0";
|
|
7
|
-
id: string | number;
|
|
8
|
-
method: string;
|
|
9
|
-
params?: unknown;
|
|
10
|
-
}
|
|
11
|
-
|
|
12
|
-
export interface JsonRpcResponse {
|
|
13
|
-
jsonrpc: "2.0";
|
|
14
|
-
id: string | number;
|
|
15
|
-
result?: unknown;
|
|
16
|
-
error?: {
|
|
17
|
-
code: number;
|
|
18
|
-
message: string;
|
|
19
|
-
data?: unknown;
|
|
20
|
-
};
|
|
21
|
-
}
|
|
22
|
-
|
|
23
|
-
export interface RpcMethodContext {
|
|
24
|
-
svm: LiteSVM;
|
|
25
|
-
slot: bigint;
|
|
26
|
-
blockHeight: bigint;
|
|
27
|
-
store?: TxStore;
|
|
28
|
-
encodeBase58: (bytes: Uint8Array) => string;
|
|
29
|
-
decodeBase58: (str: string) => Uint8Array;
|
|
30
|
-
createSuccessResponse: (
|
|
31
|
-
id: string | number,
|
|
32
|
-
result: unknown,
|
|
33
|
-
) => JsonRpcResponse;
|
|
34
|
-
createErrorResponse: (
|
|
35
|
-
id: string | number,
|
|
36
|
-
code: number,
|
|
37
|
-
message: string,
|
|
38
|
-
data?: unknown,
|
|
39
|
-
) => JsonRpcResponse;
|
|
40
|
-
notifySignature: (signature: string) => void;
|
|
41
|
-
getFaucet: () => Keypair;
|
|
42
|
-
getTxCount: () => bigint;
|
|
43
|
-
registerMint?: (mint: PublicKey | string) => void;
|
|
44
|
-
listMints?: () => string[];
|
|
45
|
-
registerProgram?: (program: PublicKey | string) => void;
|
|
46
|
-
listPrograms?: () => string[];
|
|
47
|
-
recordTransaction: (
|
|
48
|
-
signature: string,
|
|
49
|
-
tx: VersionedTransaction,
|
|
50
|
-
meta?: {
|
|
51
|
-
logs?: string[];
|
|
52
|
-
err?: unknown;
|
|
53
|
-
fee?: number;
|
|
54
|
-
blockTime?: number;
|
|
55
|
-
preBalances?: number[];
|
|
56
|
-
postBalances?: number[];
|
|
57
|
-
preTokenBalances?: unknown[];
|
|
58
|
-
postTokenBalances?: unknown[];
|
|
59
|
-
innerInstructions?: unknown[];
|
|
60
|
-
computeUnits?: number | bigint | null;
|
|
61
|
-
returnData?: { programId: string; dataBase64: string } | null;
|
|
62
|
-
// Optional rich per-account snapshots captured around execution
|
|
63
|
-
accountStates?: Array<{
|
|
64
|
-
address: string;
|
|
65
|
-
pre?: {
|
|
66
|
-
lamports?: number;
|
|
67
|
-
ownerProgram?: string;
|
|
68
|
-
executable?: boolean;
|
|
69
|
-
rentEpoch?: number;
|
|
70
|
-
dataLen?: number;
|
|
71
|
-
dataBase64?: string | null;
|
|
72
|
-
lastSlot?: number;
|
|
73
|
-
} | null;
|
|
74
|
-
post?: {
|
|
75
|
-
lamports?: number;
|
|
76
|
-
ownerProgram?: string;
|
|
77
|
-
executable?: boolean;
|
|
78
|
-
rentEpoch?: number;
|
|
79
|
-
dataLen?: number;
|
|
80
|
-
dataBase64?: string | null;
|
|
81
|
-
lastSlot?: number;
|
|
82
|
-
} | null;
|
|
83
|
-
}>;
|
|
84
|
-
},
|
|
85
|
-
) => void;
|
|
86
|
-
getRecordedTransaction: (signature: string) =>
|
|
87
|
-
| {
|
|
88
|
-
tx: VersionedTransaction;
|
|
89
|
-
logs: string[];
|
|
90
|
-
err: unknown;
|
|
91
|
-
fee: number;
|
|
92
|
-
slot: number;
|
|
93
|
-
blockTime?: number;
|
|
94
|
-
preBalances?: number[];
|
|
95
|
-
postBalances?: number[];
|
|
96
|
-
preTokenBalances?: unknown[];
|
|
97
|
-
postTokenBalances?: unknown[];
|
|
98
|
-
innerInstructions?: unknown[];
|
|
99
|
-
computeUnits?: number | null;
|
|
100
|
-
returnData?: { programId: string; dataBase64: string } | null;
|
|
101
|
-
}
|
|
102
|
-
| undefined;
|
|
103
|
-
}
|
|
104
|
-
|
|
105
|
-
export type RpcMethodHandler = (
|
|
106
|
-
id: string | number,
|
|
107
|
-
params: unknown[] | undefined,
|
|
108
|
-
context: RpcMethodContext,
|
|
109
|
-
) => JsonRpcResponse | Promise<JsonRpcResponse>;
|