solforge 0.2.11 → 0.2.13

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 (175) hide show
  1. package/package.json +4 -8
  2. package/docs/API.md +0 -379
  3. package/docs/CONFIGURATION.md +0 -407
  4. package/docs/bun-single-file-executable.md +0 -585
  5. package/docs/cli-plan.md +0 -154
  6. package/docs/data-indexing-plan.md +0 -214
  7. package/docs/gui-roadmap.md +0 -202
  8. package/scripts/decode-b58.ts +0 -10
  9. package/scripts/install.sh +0 -112
  10. package/server/index.ts +0 -5
  11. package/server/lib/base58.ts +0 -33
  12. package/server/lib/faucet.ts +0 -110
  13. package/server/lib/instruction-parser.ts +0 -328
  14. package/server/lib/parsers/spl-associated-token-account.ts +0 -50
  15. package/server/lib/parsers/spl-token.ts +0 -340
  16. package/server/lib/spl-token.ts +0 -57
  17. package/server/methods/TEMPLATE.md +0 -117
  18. package/server/methods/account/get-account-info.ts +0 -86
  19. package/server/methods/account/get-balance.ts +0 -23
  20. package/server/methods/account/get-multiple-accounts.ts +0 -84
  21. package/server/methods/account/get-parsed-account-info.ts +0 -17
  22. package/server/methods/account/index.ts +0 -12
  23. package/server/methods/account/parsers/index.ts +0 -52
  24. package/server/methods/account/parsers/loader-upgradeable.ts +0 -79
  25. package/server/methods/account/parsers/spl-token.ts +0 -256
  26. package/server/methods/account/parsers/system.ts +0 -4
  27. package/server/methods/account/request-airdrop.ts +0 -271
  28. package/server/methods/admin/adopt-mint-authority.ts +0 -94
  29. package/server/methods/admin/clone-program-accounts.ts +0 -55
  30. package/server/methods/admin/clone-program.ts +0 -152
  31. package/server/methods/admin/clone-token-accounts.ts +0 -117
  32. package/server/methods/admin/clone-token-mint.ts +0 -82
  33. package/server/methods/admin/create-mint.ts +0 -114
  34. package/server/methods/admin/create-token-account.ts +0 -137
  35. package/server/methods/admin/helpers.ts +0 -70
  36. package/server/methods/admin/index.ts +0 -10
  37. package/server/methods/admin/list-mints.ts +0 -21
  38. package/server/methods/admin/load-program.ts +0 -52
  39. package/server/methods/admin/mint-to.ts +0 -266
  40. package/server/methods/block/get-block-height.ts +0 -5
  41. package/server/methods/block/get-block.ts +0 -31
  42. package/server/methods/block/get-blocks-with-limit.ts +0 -19
  43. package/server/methods/block/get-latest-blockhash.ts +0 -12
  44. package/server/methods/block/get-slot.ts +0 -5
  45. package/server/methods/block/index.ts +0 -6
  46. package/server/methods/block/is-blockhash-valid.ts +0 -19
  47. package/server/methods/epoch/get-cluster-nodes.ts +0 -17
  48. package/server/methods/epoch/get-epoch-info.ts +0 -16
  49. package/server/methods/epoch/get-epoch-schedule.ts +0 -15
  50. package/server/methods/epoch/get-highest-snapshot-slot.ts +0 -9
  51. package/server/methods/epoch/get-leader-schedule.ts +0 -8
  52. package/server/methods/epoch/get-max-retransmit-slot.ts +0 -9
  53. package/server/methods/epoch/get-max-shred-insert-slot.ts +0 -9
  54. package/server/methods/epoch/get-slot-leader.ts +0 -6
  55. package/server/methods/epoch/get-slot-leaders.ts +0 -9
  56. package/server/methods/epoch/get-stake-activation.ts +0 -9
  57. package/server/methods/epoch/get-stake-minimum-delegation.ts +0 -9
  58. package/server/methods/epoch/get-vote-accounts.ts +0 -19
  59. package/server/methods/epoch/index.ts +0 -13
  60. package/server/methods/epoch/minimum-ledger-slot.ts +0 -5
  61. package/server/methods/fee/get-fee-calculator-for-blockhash.ts +0 -12
  62. package/server/methods/fee/get-fee-for-message.ts +0 -8
  63. package/server/methods/fee/get-fee-rate-governor.ts +0 -16
  64. package/server/methods/fee/get-fees.ts +0 -14
  65. package/server/methods/fee/get-recent-prioritization-fees.ts +0 -22
  66. package/server/methods/fee/index.ts +0 -5
  67. package/server/methods/get-address-lookup-table.ts +0 -27
  68. package/server/methods/index.ts +0 -265
  69. package/server/methods/performance/get-recent-performance-samples.ts +0 -25
  70. package/server/methods/performance/get-transaction-count.ts +0 -5
  71. package/server/methods/performance/index.ts +0 -2
  72. package/server/methods/program/get-block-commitment.ts +0 -9
  73. package/server/methods/program/get-block-production.ts +0 -14
  74. package/server/methods/program/get-block-time.ts +0 -21
  75. package/server/methods/program/get-blocks.ts +0 -11
  76. package/server/methods/program/get-first-available-block.ts +0 -9
  77. package/server/methods/program/get-genesis-hash.ts +0 -6
  78. package/server/methods/program/get-identity.ts +0 -6
  79. package/server/methods/program/get-inflation-governor.ts +0 -15
  80. package/server/methods/program/get-inflation-rate.ts +0 -10
  81. package/server/methods/program/get-inflation-reward.ts +0 -12
  82. package/server/methods/program/get-largest-accounts.ts +0 -8
  83. package/server/methods/program/get-parsed-program-accounts.ts +0 -12
  84. package/server/methods/program/get-parsed-token-accounts-by-delegate.ts +0 -12
  85. package/server/methods/program/get-parsed-token-accounts-by-owner.ts +0 -12
  86. package/server/methods/program/get-program-accounts.ts +0 -221
  87. package/server/methods/program/get-supply.ts +0 -13
  88. package/server/methods/program/get-token-account-balance.ts +0 -60
  89. package/server/methods/program/get-token-accounts-by-delegate.ts +0 -82
  90. package/server/methods/program/get-token-accounts-by-owner.ts +0 -416
  91. package/server/methods/program/get-token-largest-accounts.ts +0 -81
  92. package/server/methods/program/get-token-supply.ts +0 -39
  93. package/server/methods/program/index.ts +0 -21
  94. package/server/methods/solforge/index.ts +0 -158
  95. package/server/methods/system/get-health.ts +0 -5
  96. package/server/methods/system/get-minimum-balance-for-rent-exemption.ts +0 -13
  97. package/server/methods/system/get-version.ts +0 -9
  98. package/server/methods/system/index.ts +0 -3
  99. package/server/methods/transaction/get-confirmed-transaction.ts +0 -11
  100. package/server/methods/transaction/get-parsed-transaction.ts +0 -17
  101. package/server/methods/transaction/get-signature-statuses.ts +0 -79
  102. package/server/methods/transaction/get-signatures-for-address.ts +0 -41
  103. package/server/methods/transaction/get-transaction.ts +0 -639
  104. package/server/methods/transaction/index.ts +0 -7
  105. package/server/methods/transaction/inner-instructions.test.ts +0 -104
  106. package/server/methods/transaction/send-transaction.ts +0 -469
  107. package/server/methods/transaction/simulate-transaction.ts +0 -57
  108. package/server/rpc-server.ts +0 -521
  109. package/server/types.ts +0 -109
  110. package/server/ws-server.ts +0 -178
  111. package/src/api-server-entry.ts +0 -109
  112. package/src/cli/bootstrap.ts +0 -67
  113. package/src/cli/commands/airdrop.ts +0 -37
  114. package/src/cli/commands/config.ts +0 -39
  115. package/src/cli/commands/mint.ts +0 -187
  116. package/src/cli/commands/program-clone.ts +0 -122
  117. package/src/cli/commands/program-load.ts +0 -64
  118. package/src/cli/commands/rpc-start.ts +0 -49
  119. package/src/cli/commands/token-adopt-authority.ts +0 -37
  120. package/src/cli/commands/token-clone.ts +0 -112
  121. package/src/cli/commands/token-create.ts +0 -81
  122. package/src/cli/main.ts +0 -152
  123. package/src/cli/run-solforge.ts +0 -112
  124. package/src/cli/setup-utils.ts +0 -54
  125. package/src/cli/setup-wizard.ts +0 -258
  126. package/src/cli/utils/args.ts +0 -15
  127. package/src/commands/add-program.ts +0 -333
  128. package/src/commands/init.ts +0 -122
  129. package/src/commands/list.ts +0 -136
  130. package/src/commands/mint.ts +0 -287
  131. package/src/commands/start.ts +0 -881
  132. package/src/commands/status.ts +0 -99
  133. package/src/commands/stop.ts +0 -405
  134. package/src/config/index.ts +0 -146
  135. package/src/config/manager.ts +0 -157
  136. package/src/db/index.ts +0 -83
  137. package/src/db/schema/accounts.ts +0 -23
  138. package/src/db/schema/address-signatures.ts +0 -31
  139. package/src/db/schema/index.ts +0 -6
  140. package/src/db/schema/meta-kv.ts +0 -9
  141. package/src/db/schema/transactions.ts +0 -36
  142. package/src/db/schema/tx-account-states.ts +0 -23
  143. package/src/db/schema/tx-accounts.ts +0 -33
  144. package/src/db/tx-store.ts +0 -264
  145. package/src/gui/public/app.css +0 -1556
  146. package/src/gui/public/build/main.css +0 -1569
  147. package/src/gui/public/build/main.js +0 -303
  148. package/src/gui/public/build/main.js.txt +0 -231
  149. package/src/gui/public/index.html +0 -19
  150. package/src/gui/server.ts +0 -296
  151. package/src/gui/src/api.ts +0 -127
  152. package/src/gui/src/app.tsx +0 -441
  153. package/src/gui/src/components/airdrop-mint-form.tsx +0 -246
  154. package/src/gui/src/components/clone-program-modal.tsx +0 -202
  155. package/src/gui/src/components/clone-token-modal.tsx +0 -230
  156. package/src/gui/src/components/modal.tsx +0 -134
  157. package/src/gui/src/components/programs-panel.tsx +0 -124
  158. package/src/gui/src/components/status-panel.tsx +0 -136
  159. package/src/gui/src/components/tokens-panel.tsx +0 -122
  160. package/src/gui/src/hooks/use-interval.ts +0 -17
  161. package/src/gui/src/index.css +0 -557
  162. package/src/gui/src/main.tsx +0 -17
  163. package/src/index.ts +0 -216
  164. package/src/migrations-bundled.ts +0 -23
  165. package/src/rpc/start.ts +0 -44
  166. package/src/services/api-server.ts +0 -504
  167. package/src/services/port-manager.ts +0 -174
  168. package/src/services/process-registry.ts +0 -153
  169. package/src/services/program-cloner.ts +0 -317
  170. package/src/services/token-cloner.ts +0 -811
  171. package/src/services/validator.ts +0 -293
  172. package/src/types/config.ts +0 -110
  173. package/src/utils/shell.ts +0 -110
  174. package/src/utils/token-loader.ts +0 -115
  175. /package/{start.js → start.cjs} +0 -0
@@ -1,333 +0,0 @@
1
- import chalk from "chalk";
2
- import { existsSync } from "node:fs";
3
- import inquirer from "inquirer";
4
- import { configManager } from "../config/manager.js";
5
- import { processRegistry } from "../services/process-registry.js";
6
- import { ProgramCloner } from "../services/program-cloner.js";
7
- import type { ProgramConfig } from "../types/config.js";
8
-
9
- export async function addProgramCommand(
10
- options: { name?: string; programId?: string; interactive?: boolean } = {},
11
- ): Promise<void> {
12
- console.log(chalk.blue("šŸ“¦ Adding program to configuration...\n"));
13
-
14
- // Check if config exists
15
- if (!existsSync("./sf.config.json")) {
16
- console.error(chalk.red("āŒ No sf.config.json found in current directory"));
17
- console.log(chalk.yellow("šŸ’” Run `solforge init` to create one"));
18
- return;
19
- }
20
-
21
- // Load current config
22
- try {
23
- await configManager.load("./sf.config.json");
24
- } catch (error) {
25
- console.error(chalk.red("āŒ Failed to load sf.config.json"));
26
- console.error(
27
- chalk.red(error instanceof Error ? error.message : String(error)),
28
- );
29
- return;
30
- }
31
-
32
- const config = configManager.getConfig();
33
-
34
- // Get program details
35
- let programId: string;
36
- let programName: string | undefined;
37
- let upgradeable: boolean = false;
38
-
39
- if (options.interactive !== false && !options.programId) {
40
- // Interactive mode - show common programs + custom option
41
- const { programChoice } = await inquirer.prompt([
42
- {
43
- type: "list",
44
- name: "programChoice",
45
- message: "Select a program to add:",
46
- choices: [
47
- {
48
- name: "Token Metadata Program",
49
- value: {
50
- id: "metaqbxxUerdq28cj1RbAWkYQm3ybzjb6a8bt518x1s",
51
- name: "Token Metadata",
52
- upgradeable: true,
53
- },
54
- },
55
- {
56
- name: "Associated Token Account Program",
57
- value: {
58
- id: "ATokenGPvbdGVxr1b2hvZbsiqW5xWH25efTNsLJA8knL",
59
- name: "Associated Token Account",
60
- upgradeable: false,
61
- },
62
- },
63
- {
64
- name: "System Program",
65
- value: {
66
- id: "11111111111111111111111111111112",
67
- name: "System Program",
68
- upgradeable: false,
69
- },
70
- },
71
- {
72
- name: "Token Program",
73
- value: {
74
- id: "TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA",
75
- name: "Token Program",
76
- upgradeable: false,
77
- },
78
- },
79
- {
80
- name: "Token Program 2022",
81
- value: {
82
- id: "TokenzQdBNbLqP5VEhdkAS6EPFLC1PHnBqCXEpPxuEb",
83
- name: "Token Program 2022",
84
- upgradeable: true,
85
- },
86
- },
87
- {
88
- name: "Rent Sysvar",
89
- value: {
90
- id: "SysvarRent111111111111111111111111111111111",
91
- name: "Rent Sysvar",
92
- upgradeable: false,
93
- },
94
- },
95
- {
96
- name: "Clock Sysvar",
97
- value: {
98
- id: "SysvarC1ock11111111111111111111111111111111",
99
- name: "Clock Sysvar",
100
- upgradeable: false,
101
- },
102
- },
103
- {
104
- name: "Jupiter V6 Program",
105
- value: {
106
- id: "JUP6LkbZbjS1jKKwapdHNy74zcZ3tLUZoi5QNyVTaV4",
107
- name: "Jupiter V6",
108
- upgradeable: true,
109
- },
110
- },
111
- {
112
- name: "Raydium AMM Program",
113
- value: {
114
- id: "675kPX9MHTjS2zt1qfr1NYHuzeLXfQM9H24wFSUt1Mp8",
115
- name: "Raydium AMM",
116
- upgradeable: false,
117
- },
118
- },
119
- {
120
- name: "Serum DEX Program",
121
- value: {
122
- id: "9xQeWvG816bUx9EPjHmaT23yvVM2ZWbrrpZb9PusVFin",
123
- name: "Serum DEX",
124
- upgradeable: false,
125
- },
126
- },
127
- {
128
- name: "Custom (enter program ID manually)",
129
- value: "custom",
130
- },
131
- ],
132
- },
133
- ]);
134
-
135
- if (programChoice === "custom") {
136
- const answers = await inquirer.prompt([
137
- {
138
- type: "input",
139
- name: "programId",
140
- message: "Enter program ID:",
141
- validate: (input) =>
142
- input.trim().length > 0 || "Program ID is required",
143
- },
144
- {
145
- type: "input",
146
- name: "name",
147
- message: "Enter program name (optional):",
148
- default: "",
149
- },
150
- {
151
- type: "confirm",
152
- name: "upgradeable",
153
- message: "Is this an upgradeable program?",
154
- default: false,
155
- },
156
- ]);
157
-
158
- programId = answers.programId;
159
- programName = answers.name || undefined;
160
- upgradeable = answers.upgradeable;
161
- } else {
162
- programId = programChoice.id;
163
- programName = programChoice.name;
164
- upgradeable = programChoice.upgradeable;
165
- }
166
- } else {
167
- // Non-interactive mode
168
- if (!options.programId) {
169
- console.error(chalk.red("āŒ Program ID is required"));
170
- console.log(
171
- chalk.gray(
172
- "šŸ’” Use --program-id flag, or run without flags for interactive mode",
173
- ),
174
- );
175
- return;
176
- }
177
-
178
- programId = options.programId;
179
- programName = options.name;
180
- // Default to false for CLI, could add --upgradeable flag later
181
- upgradeable = false;
182
- }
183
-
184
- try {
185
- // Check if program already exists in config
186
- const existingProgram = config.programs.find(
187
- (p) => p.mainnetProgramId === programId,
188
- );
189
- if (existingProgram) {
190
- console.log(
191
- chalk.yellow(`āš ļø Program ${programId} already exists in configuration`),
192
- );
193
- console.log(chalk.gray(` Name: ${existingProgram.name || "unnamed"}`));
194
- return;
195
- }
196
-
197
- // Verify program exists on mainnet first
198
- console.log(chalk.gray("šŸ” Verifying program exists on mainnet..."));
199
- const programCloner = new ProgramCloner();
200
- const programInfo = await programCloner.getProgramInfo(
201
- programId,
202
- "mainnet-beta",
203
- );
204
-
205
- if (!programInfo.exists) {
206
- console.error(chalk.red(`āŒ Program ${programId} not found on mainnet`));
207
- return;
208
- }
209
-
210
- if (!programInfo.executable) {
211
- console.error(chalk.red(`āŒ ${programId} is not an executable program`));
212
- return;
213
- }
214
-
215
- console.log(
216
- chalk.gray(` āœ“ Found executable program (${programInfo.size} bytes)`),
217
- );
218
-
219
- // Create new program config
220
- const newProgram: ProgramConfig = {
221
- name: programName,
222
- mainnetProgramId: programId,
223
- upgradeable: upgradeable,
224
- cluster: "mainnet-beta",
225
- dependencies: [],
226
- };
227
-
228
- // Add to config
229
- config.programs.push(newProgram);
230
-
231
- // Save updated config
232
- await configManager.save("./sf.config.json");
233
-
234
- console.log(
235
- chalk.green(
236
- `\nāœ… Successfully added ${programName || programId} to configuration!`,
237
- ),
238
- );
239
- console.log(chalk.cyan(`šŸ“ Updated sf.config.json`));
240
- console.log(chalk.gray(` Program ID: ${programId}`));
241
- console.log(chalk.gray(` Upgradeable: ${upgradeable ? "Yes" : "No"}`));
242
-
243
- // Check if there are running validators
244
- await processRegistry.cleanup();
245
- const validators = processRegistry.getRunning();
246
-
247
- if (validators.length > 0) {
248
- console.log(
249
- chalk.yellow(`\nāš ļø Found ${validators.length} running validator(s)`),
250
- );
251
- console.log(
252
- chalk.gray("šŸ’” Programs are added to genesis config and require RESET"),
253
- );
254
- console.log(chalk.red(" āš ļø RESET will WIPE all ledger data!"));
255
- console.log(
256
- chalk.gray(
257
- ` 1. Stop validators using "${config.name}": solforge stop <validator-id>`,
258
- ),
259
- );
260
- console.log(chalk.gray(" 2. Start with reset: solforge start"));
261
-
262
- const { shouldRestart } = await inquirer.prompt([
263
- {
264
- type: "confirm",
265
- name: "shouldRestart",
266
- message: "āš ļø Reset validators now? (This will WIPE all ledger data)",
267
- default: false,
268
- },
269
- ]);
270
-
271
- if (shouldRestart) {
272
- console.log(chalk.yellow("\nšŸ”„ Resetting validators..."));
273
- console.log(chalk.red("āš ļø This will delete all ledger data!"));
274
-
275
- // Import the commands we need
276
- const { stopCommand } = await import("./stop.js");
277
- const { startCommand } = await import("./start.js");
278
-
279
- // Get current config for modifications
280
- const currentConfig = configManager.getConfig();
281
-
282
- // Stop validators that use this config
283
- const configName = currentConfig.name;
284
- const matchingValidators = validators.filter((v) =>
285
- v.name.startsWith(configName),
286
- );
287
-
288
- if (matchingValidators.length > 0) {
289
- console.log(
290
- chalk.gray(
291
- `šŸ›‘ Stopping ${matchingValidators.length} validator(s) using config "${configName}"...`,
292
- ),
293
- );
294
-
295
- for (const validator of matchingValidators) {
296
- await stopCommand(validator.id, {});
297
- }
298
- }
299
-
300
- // Wait a moment for clean shutdown
301
- await new Promise((resolve) => setTimeout(resolve, 2000));
302
-
303
- // Temporarily enable reset for this start
304
- const originalReset = currentConfig.localnet.reset;
305
- currentConfig.localnet.reset = true;
306
-
307
- console.log(
308
- chalk.cyan(
309
- "šŸš€ Starting validator with reset to apply program changes...",
310
- ),
311
- );
312
-
313
- // Start new validator (will reset due to config)
314
- await startCommand(false);
315
-
316
- // Restore original reset setting
317
- currentConfig.localnet.reset = originalReset;
318
- await configManager.save("./sf.config.json");
319
- }
320
- } else {
321
- console.log(
322
- chalk.gray(
323
- "\nšŸ’” Start a validator to use the new program: solforge start",
324
- ),
325
- );
326
- }
327
- } catch (error) {
328
- console.error(chalk.red("āŒ Failed to add program:"));
329
- console.error(
330
- chalk.red(` ${error instanceof Error ? error.message : String(error)}`),
331
- );
332
- }
333
- }
@@ -1,122 +0,0 @@
1
- import chalk from "chalk";
2
- import { existsSync, writeFileSync } from "node:fs";
3
- import inquirer from "inquirer";
4
- import { resolve } from "node:path";
5
- import type { Config } from "../types/config.js";
6
-
7
- const defaultConfig: Config = {
8
- name: "my-localnet",
9
- description: "Local Solana development environment",
10
- tokens: [
11
- {
12
- symbol: "USDC",
13
- mainnetMint: "EPjFWdd5AufqSSqeM2qN1xzybapC8G4wEGGkZwyTDt1v",
14
- recipients: [],
15
- mintAmount: 1000000,
16
- cloneMetadata: true,
17
- },
18
- ],
19
- programs: [],
20
- localnet: {
21
- airdropAmount: 100,
22
- faucetAccounts: [],
23
- port: 8899,
24
- faucetPort: 9900,
25
- reset: false,
26
- logLevel: "info",
27
- bindAddress: "127.0.0.1",
28
- quiet: false,
29
- rpc: "https://api.mainnet-beta.solana.com",
30
- limitLedgerSize: 100000,
31
- },
32
- };
33
-
34
- export async function initCommand(): Promise<void> {
35
- const configPath = resolve(process.cwd(), "sf.config.json");
36
-
37
- // Check if config already exists
38
- if (existsSync(configPath)) {
39
- const { overwrite } = await inquirer.prompt([
40
- {
41
- type: "confirm",
42
- name: "overwrite",
43
- message: "sf.config.json already exists. Overwrite?",
44
- default: false,
45
- },
46
- ]);
47
-
48
- if (!overwrite) {
49
- console.log(chalk.yellow("āš ļø Initialization cancelled"));
50
- return;
51
- }
52
- }
53
-
54
- // Gather basic configuration
55
- const answers = await inquirer.prompt([
56
- {
57
- type: "input",
58
- name: "name",
59
- message: "Project name:",
60
- default: "my-localnet",
61
- },
62
- {
63
- type: "input",
64
- name: "description",
65
- message: "Description:",
66
- default: "Local Solana development environment",
67
- },
68
- {
69
- type: "number",
70
- name: "port",
71
- message: "RPC port:",
72
- default: 8899,
73
- },
74
- {
75
- type: "confirm",
76
- name: "includeUsdc",
77
- message: "Include USDC token?",
78
- default: true,
79
- },
80
- ]);
81
-
82
- // Build config
83
- const config: Config = {
84
- ...defaultConfig,
85
- name: answers.name,
86
- description: answers.description,
87
- localnet: {
88
- ...defaultConfig.localnet,
89
- port: answers.port,
90
- },
91
- };
92
-
93
- if (!answers.includeUsdc) {
94
- config.tokens = [];
95
- }
96
-
97
- try {
98
- // Write config file
99
- writeFileSync(configPath, JSON.stringify(config, null, 2));
100
-
101
- console.log(chalk.green("āœ… sf.config.json created successfully!"));
102
- console.log(chalk.gray(`šŸ“„ Config saved to: ${configPath}`));
103
- console.log();
104
- console.log(chalk.blue("Next steps:"));
105
- console.log(
106
- chalk.gray("1. Edit sf.config.json to add your tokens and programs"),
107
- );
108
- console.log(chalk.gray("2. Run `solforge start` to launch your localnet"));
109
- console.log();
110
- console.log(
111
- chalk.yellow(
112
- "šŸ’” Tip: Check configs/example.sf.config.json for more examples",
113
- ),
114
- );
115
- } catch (error) {
116
- console.error(chalk.red("āŒ Failed to create sf.config.json"));
117
- console.error(
118
- chalk.red(error instanceof Error ? error.message : String(error)),
119
- );
120
- process.exit(1);
121
- }
122
- }
@@ -1,136 +0,0 @@
1
- import chalk from "chalk";
2
- import type { RunningValidator } from "../services/process-registry.js";
3
- import { processRegistry } from "../services/process-registry.js";
4
-
5
- export async function listCommand(): Promise<void> {
6
- console.log(chalk.blue("šŸ“‹ Listing running validators...\n"));
7
-
8
- // Clean up dead processes first
9
- await processRegistry.cleanup();
10
-
11
- const validators = processRegistry.getRunning();
12
-
13
- if (validators.length === 0) {
14
- console.log(chalk.yellow("āš ļø No running validators found"));
15
- console.log(chalk.gray("šŸ’” Use `solforge start` to start a validator"));
16
- return;
17
- }
18
-
19
- // Verify each validator is actually still running
20
- const activeValidators: RunningValidator[] = [];
21
- for (const validator of validators) {
22
- const isRunning = await processRegistry.isProcessRunning(validator.pid);
23
- if (isRunning) {
24
- activeValidators.push({ ...validator, status: "running" });
25
- } else {
26
- processRegistry.updateStatus(validator.id, "stopped");
27
- }
28
- }
29
-
30
- if (activeValidators.length === 0) {
31
- console.log(
32
- chalk.yellow(
33
- "āš ļø No active validators found (all processes have stopped)",
34
- ),
35
- );
36
- console.log(chalk.gray("šŸ’” Use `solforge start` to start a validator"));
37
- return;
38
- }
39
-
40
- console.log(
41
- chalk.green(
42
- `āœ… Found ${activeValidators.length} running validator${
43
- activeValidators.length > 1 ? "s" : ""
44
- }\n`,
45
- ),
46
- );
47
-
48
- // Display validators in a table format
49
- displayValidatorsTable(activeValidators);
50
-
51
- console.log(
52
- chalk.gray("\nšŸ’” Use `solforge stop <id>` to stop a specific validator"),
53
- );
54
- console.log(
55
- chalk.gray("šŸ’” Use `solforge stop --all` to stop all validators"),
56
- );
57
- }
58
-
59
- function displayValidatorsTable(validators: RunningValidator[]): void {
60
- // Calculate column widths
61
- const maxIdWidth = Math.max(2, ...validators.map((v) => v.id.length));
62
- const maxNameWidth = Math.max(4, ...validators.map((v) => v.name.length));
63
- const maxPidWidth = Math.max(
64
- 3,
65
- ...validators.map((v) => v.pid.toString().length),
66
- );
67
- const maxPortWidth = 9; // "8899/9900" format
68
- const maxUptimeWidth = 7;
69
-
70
- // Header
71
- const header =
72
- chalk.bold("ID".padEnd(maxIdWidth)) +
73
- " | " +
74
- chalk.bold("Name".padEnd(maxNameWidth)) +
75
- " | " +
76
- chalk.bold("PID".padEnd(maxPidWidth)) +
77
- " | " +
78
- chalk.bold("RPC/Faucet".padEnd(maxPortWidth)) +
79
- " | " +
80
- chalk.bold("Uptime".padEnd(maxUptimeWidth)) +
81
- " | " +
82
- chalk.bold("Status");
83
-
84
- console.log(header);
85
- console.log("-".repeat(header.length - 20)); // Subtract ANSI codes length
86
-
87
- // Rows
88
- validators.forEach((validator) => {
89
- const uptime = formatUptime(validator.startTime);
90
- const ports = `${validator.rpcPort}/${validator.faucetPort}`;
91
- const status =
92
- validator.status === "running" ? chalk.green("ā—") : chalk.red("ā—");
93
-
94
- const row =
95
- validator.id.padEnd(maxIdWidth) +
96
- " | " +
97
- validator.name.padEnd(maxNameWidth) +
98
- " | " +
99
- validator.pid.toString().padEnd(maxPidWidth) +
100
- " | " +
101
- ports.padEnd(maxPortWidth) +
102
- " | " +
103
- uptime.padEnd(maxUptimeWidth) +
104
- " | " +
105
- status;
106
-
107
- console.log(row);
108
- });
109
-
110
- console.log(); // Empty line
111
-
112
- // URLs section
113
- console.log(chalk.cyan("🌐 Connection URLs:"));
114
- validators.forEach((validator) => {
115
- console.log(chalk.gray(` ${validator.name} (${validator.id}):`));
116
- console.log(chalk.gray(` RPC: ${validator.rpcUrl}`));
117
- console.log(chalk.gray(` Faucet: ${validator.faucetUrl}`));
118
- });
119
- }
120
-
121
- function formatUptime(startTime: Date): string {
122
- const now = new Date();
123
- const uptimeMs = now.getTime() - startTime.getTime();
124
- const uptimeSeconds = Math.floor(uptimeMs / 1000);
125
-
126
- if (uptimeSeconds < 60) {
127
- return `${uptimeSeconds}s`;
128
- } else if (uptimeSeconds < 3600) {
129
- const minutes = Math.floor(uptimeSeconds / 60);
130
- return `${minutes}m`;
131
- } else {
132
- const hours = Math.floor(uptimeSeconds / 3600);
133
- const minutes = Math.floor((uptimeSeconds % 3600) / 60);
134
- return `${hours}h${minutes}m`;
135
- }
136
- }