solforge 0.1.6 → 0.2.0
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/.agi/agi.sqlite +0 -0
- package/.claude/settings.local.json +9 -0
- package/.github/workflows/release-binaries.yml +133 -0
- package/.tmp/.787ebcdbf7b8fde8-00000000.hm +0 -0
- package/.tmp/.bffe6efebdf8aedc-00000000.hm +0 -0
- package/AGENTS.md +271 -0
- package/CLAUDE.md +106 -0
- package/PROJECT_STRUCTURE.md +124 -0
- package/README.md +367 -393
- package/SOLANA_KIT_GUIDE.md +251 -0
- package/SOLFORGE.md +119 -0
- package/biome.json +34 -0
- package/bun.lock +743 -0
- package/docs/bun-single-file-executable.md +585 -0
- package/docs/cli-plan.md +154 -0
- package/docs/data-indexing-plan.md +214 -0
- package/docs/gui-roadmap.md +202 -0
- package/drizzle/0000_friendly_millenium_guard.sql +53 -0
- package/drizzle/0001_stale_sentinels.sql +2 -0
- package/drizzle/meta/0000_snapshot.json +329 -0
- package/drizzle/meta/0001_snapshot.json +345 -0
- package/drizzle/meta/_journal.json +20 -0
- package/drizzle.config.ts +12 -0
- package/index.ts +21 -0
- package/mint.sh +47 -0
- package/package.json +45 -69
- package/postcss.config.js +6 -0
- package/rpc-server.ts.backup +519 -0
- package/server/index.ts +5 -0
- package/server/lib/base58.ts +33 -0
- package/server/lib/faucet.ts +110 -0
- package/server/lib/spl-token.ts +57 -0
- package/server/methods/TEMPLATE.md +117 -0
- package/server/methods/account/get-account-info.ts +90 -0
- package/server/methods/account/get-balance.ts +27 -0
- package/server/methods/account/get-multiple-accounts.ts +83 -0
- package/server/methods/account/get-parsed-account-info.ts +21 -0
- package/server/methods/account/index.ts +12 -0
- package/server/methods/account/parsers/index.ts +52 -0
- package/server/methods/account/parsers/loader-upgradeable.ts +66 -0
- package/server/methods/account/parsers/spl-token.ts +237 -0
- package/server/methods/account/parsers/system.ts +4 -0
- package/server/methods/account/request-airdrop.ts +219 -0
- package/server/methods/admin/adopt-mint-authority.ts +94 -0
- package/server/methods/admin/clone-program-accounts.ts +55 -0
- package/server/methods/admin/clone-program.ts +152 -0
- package/server/methods/admin/clone-token-accounts.ts +117 -0
- package/server/methods/admin/clone-token-mint.ts +82 -0
- package/server/methods/admin/create-mint.ts +114 -0
- package/server/methods/admin/create-token-account.ts +137 -0
- package/server/methods/admin/helpers.ts +70 -0
- package/server/methods/admin/index.ts +10 -0
- package/server/methods/admin/list-mints.ts +21 -0
- package/server/methods/admin/load-program.ts +52 -0
- package/server/methods/admin/mint-to.ts +278 -0
- package/server/methods/block/get-block-height.ts +5 -0
- package/server/methods/block/get-block.ts +35 -0
- package/server/methods/block/get-blocks-with-limit.ts +23 -0
- package/server/methods/block/get-latest-blockhash.ts +12 -0
- package/server/methods/block/get-slot.ts +5 -0
- package/server/methods/block/index.ts +6 -0
- package/server/methods/block/is-blockhash-valid.ts +23 -0
- package/server/methods/epoch/get-cluster-nodes.ts +17 -0
- package/server/methods/epoch/get-epoch-info.ts +16 -0
- package/server/methods/epoch/get-epoch-schedule.ts +15 -0
- package/server/methods/epoch/get-highest-snapshot-slot.ts +9 -0
- package/server/methods/epoch/get-leader-schedule.ts +8 -0
- package/server/methods/epoch/get-max-retransmit-slot.ts +9 -0
- package/server/methods/epoch/get-max-shred-insert-slot.ts +9 -0
- package/server/methods/epoch/get-slot-leader.ts +6 -0
- package/server/methods/epoch/get-slot-leaders.ts +9 -0
- package/server/methods/epoch/get-stake-activation.ts +9 -0
- package/server/methods/epoch/get-stake-minimum-delegation.ts +9 -0
- package/server/methods/epoch/get-vote-accounts.ts +19 -0
- package/server/methods/epoch/index.ts +13 -0
- package/server/methods/epoch/minimum-ledger-slot.ts +5 -0
- package/server/methods/fee/get-fee-calculator-for-blockhash.ts +12 -0
- package/server/methods/fee/get-fee-for-message.ts +8 -0
- package/server/methods/fee/get-fee-rate-governor.ts +16 -0
- package/server/methods/fee/get-fees.ts +14 -0
- package/server/methods/fee/get-recent-prioritization-fees.ts +22 -0
- package/server/methods/fee/index.ts +5 -0
- package/server/methods/get-address-lookup-table.ts +31 -0
- package/server/methods/index.ts +265 -0
- package/server/methods/performance/get-recent-performance-samples.ts +25 -0
- package/server/methods/performance/get-transaction-count.ts +5 -0
- package/server/methods/performance/index.ts +2 -0
- package/server/methods/program/get-block-commitment.ts +9 -0
- package/server/methods/program/get-block-production.ts +14 -0
- package/server/methods/program/get-block-time.ts +21 -0
- package/server/methods/program/get-blocks.ts +11 -0
- package/server/methods/program/get-first-available-block.ts +9 -0
- package/server/methods/program/get-genesis-hash.ts +6 -0
- package/server/methods/program/get-identity.ts +6 -0
- package/server/methods/program/get-inflation-governor.ts +15 -0
- package/server/methods/program/get-inflation-rate.ts +10 -0
- package/server/methods/program/get-inflation-reward.ts +12 -0
- package/server/methods/program/get-largest-accounts.ts +8 -0
- package/server/methods/program/get-parsed-program-accounts.ts +12 -0
- package/server/methods/program/get-parsed-token-accounts-by-delegate.ts +12 -0
- package/server/methods/program/get-parsed-token-accounts-by-owner.ts +12 -0
- package/server/methods/program/get-program-accounts.ts +221 -0
- package/server/methods/program/get-supply.ts +13 -0
- package/server/methods/program/get-token-account-balance.ts +64 -0
- package/server/methods/program/get-token-accounts-by-delegate.ts +81 -0
- package/server/methods/program/get-token-accounts-by-owner.ts +390 -0
- package/server/methods/program/get-token-largest-accounts.ts +80 -0
- package/server/methods/program/get-token-supply.ts +38 -0
- package/server/methods/program/index.ts +21 -0
- package/server/methods/solforge/index.ts +155 -0
- package/server/methods/system/get-health.ts +5 -0
- package/server/methods/system/get-minimum-balance-for-rent-exemption.ts +13 -0
- package/server/methods/system/get-version.ts +9 -0
- package/server/methods/system/index.ts +3 -0
- package/server/methods/transaction/get-confirmed-transaction.ts +11 -0
- package/server/methods/transaction/get-parsed-transaction.ts +21 -0
- package/server/methods/transaction/get-signature-statuses.ts +72 -0
- package/server/methods/transaction/get-signatures-for-address.ts +45 -0
- package/server/methods/transaction/get-transaction.ts +428 -0
- package/server/methods/transaction/index.ts +7 -0
- package/server/methods/transaction/send-transaction.ts +232 -0
- package/server/methods/transaction/simulate-transaction.ts +56 -0
- package/server/rpc-server.ts +474 -0
- package/server/types.ts +74 -0
- package/server/ws-server.ts +171 -0
- package/sf.config.json +38 -0
- package/src/cli/bootstrap.ts +67 -0
- package/src/cli/commands/airdrop.ts +37 -0
- package/src/cli/commands/config.ts +39 -0
- package/src/cli/commands/mint.ts +187 -0
- package/src/cli/commands/program-clone.ts +124 -0
- package/src/cli/commands/program-load.ts +64 -0
- package/src/cli/commands/rpc-start.ts +46 -0
- package/src/cli/commands/token-adopt-authority.ts +37 -0
- package/src/cli/commands/token-clone.ts +113 -0
- package/src/cli/commands/token-create.ts +81 -0
- package/src/cli/main.ts +130 -0
- package/src/cli/run-solforge.ts +98 -0
- package/src/cli/setup-utils.ts +54 -0
- package/src/cli/setup-wizard.ts +256 -0
- package/src/cli/utils/args.ts +15 -0
- package/src/config/index.ts +130 -0
- package/src/db/index.ts +83 -0
- package/src/db/schema/accounts.ts +23 -0
- package/src/db/schema/address-signatures.ts +31 -0
- package/src/db/schema/index.ts +5 -0
- package/src/db/schema/meta-kv.ts +9 -0
- package/src/db/schema/transactions.ts +29 -0
- package/src/db/schema/tx-accounts.ts +33 -0
- package/src/db/tx-store.ts +229 -0
- package/src/gui/public/app.css +1 -0
- package/src/gui/public/index.html +19 -0
- package/src/gui/server.ts +297 -0
- package/src/gui/src/api.ts +127 -0
- package/src/gui/src/app.tsx +390 -0
- package/src/gui/src/components/airdrop-mint-form.tsx +216 -0
- package/src/gui/src/components/clone-program-modal.tsx +183 -0
- package/src/gui/src/components/clone-token-modal.tsx +211 -0
- package/src/gui/src/components/modal.tsx +127 -0
- package/src/gui/src/components/programs-panel.tsx +112 -0
- package/src/gui/src/components/status-panel.tsx +122 -0
- package/src/gui/src/components/tokens-panel.tsx +116 -0
- package/src/gui/src/hooks/use-interval.ts +17 -0
- package/src/gui/src/index.css +529 -0
- package/src/gui/src/main.tsx +17 -0
- package/src/migrations-bundled.ts +17 -0
- package/src/rpc/start.ts +44 -0
- package/tailwind.config.js +27 -0
- package/test-client.ts +120 -0
- package/tmp/inspect-html.ts +4 -0
- package/tmp/response-test.ts +5 -0
- package/tmp/test-html.ts +5 -0
- package/tmp/test-server.ts +13 -0
- package/tsconfig.json +24 -23
- package/LICENSE +0 -21
- package/scripts/postinstall.cjs +0 -103
- package/src/api-server-entry.ts +0 -109
- package/src/commands/add-program.ts +0 -337
- package/src/commands/init.ts +0 -122
- package/src/commands/list.ts +0 -136
- package/src/commands/mint.ts +0 -336
- package/src/commands/start.ts +0 -878
- package/src/commands/status.ts +0 -99
- package/src/commands/stop.ts +0 -406
- package/src/config/manager.ts +0 -157
- package/src/index.ts +0 -188
- package/src/services/api-server.ts +0 -532
- package/src/services/port-manager.ts +0 -177
- package/src/services/process-registry.ts +0 -154
- package/src/services/program-cloner.ts +0 -317
- package/src/services/token-cloner.ts +0 -809
- package/src/services/validator.ts +0 -295
- package/src/types/config.ts +0 -110
- package/src/utils/shell.ts +0 -110
|
@@ -0,0 +1,229 @@
|
|
|
1
|
+
import { and, desc, eq, gt, inArray, lt } from "drizzle-orm";
|
|
2
|
+
import { db } from "./index";
|
|
3
|
+
import { accounts } from "./schema/accounts";
|
|
4
|
+
import { addressSignatures } from "./schema/address-signatures";
|
|
5
|
+
import { transactions } from "./schema/transactions";
|
|
6
|
+
import { txAccounts } from "./schema/tx-accounts";
|
|
7
|
+
|
|
8
|
+
export type InsertTxBundle = {
|
|
9
|
+
signature: string;
|
|
10
|
+
slot: number;
|
|
11
|
+
blockTime?: number;
|
|
12
|
+
version: 0 | "legacy";
|
|
13
|
+
fee: number;
|
|
14
|
+
err: unknown | null;
|
|
15
|
+
rawBase64: string;
|
|
16
|
+
preBalances: number[];
|
|
17
|
+
postBalances: number[];
|
|
18
|
+
logs: string[];
|
|
19
|
+
accounts: Array<{
|
|
20
|
+
address: string;
|
|
21
|
+
index: number;
|
|
22
|
+
signer: boolean;
|
|
23
|
+
writable: boolean;
|
|
24
|
+
programIdIndex?: number;
|
|
25
|
+
}>;
|
|
26
|
+
preTokenBalances?: any[];
|
|
27
|
+
postTokenBalances?: any[];
|
|
28
|
+
};
|
|
29
|
+
|
|
30
|
+
export type AccountSnapshot = {
|
|
31
|
+
address: string;
|
|
32
|
+
lamports: number;
|
|
33
|
+
ownerProgram: string;
|
|
34
|
+
executable: boolean;
|
|
35
|
+
rentEpoch: number;
|
|
36
|
+
dataLen: number;
|
|
37
|
+
dataBase64?: string | null;
|
|
38
|
+
lastSlot: number;
|
|
39
|
+
};
|
|
40
|
+
|
|
41
|
+
export class TxStore {
|
|
42
|
+
async insertTransactionBundle(bundle: InsertTxBundle): Promise<void> {
|
|
43
|
+
const errJson = bundle.err ? JSON.stringify(bundle.err) : null;
|
|
44
|
+
await db.transaction(async (tx) => {
|
|
45
|
+
await tx
|
|
46
|
+
.insert(transactions)
|
|
47
|
+
.values({
|
|
48
|
+
signature: bundle.signature,
|
|
49
|
+
slot: bundle.slot,
|
|
50
|
+
blockTime: bundle.blockTime ?? null,
|
|
51
|
+
version: String(bundle.version),
|
|
52
|
+
errJson,
|
|
53
|
+
fee: bundle.fee,
|
|
54
|
+
rawBase64: bundle.rawBase64,
|
|
55
|
+
preBalancesJson: JSON.stringify(bundle.preBalances ?? []),
|
|
56
|
+
postBalancesJson: JSON.stringify(bundle.postBalances ?? []),
|
|
57
|
+
logsJson: JSON.stringify(bundle.logs ?? []),
|
|
58
|
+
preTokenBalancesJson: JSON.stringify(bundle.preTokenBalances ?? []),
|
|
59
|
+
postTokenBalancesJson: JSON.stringify(bundle.postTokenBalances ?? []),
|
|
60
|
+
})
|
|
61
|
+
.onConflictDoNothing();
|
|
62
|
+
|
|
63
|
+
if (Array.isArray(bundle.accounts) && bundle.accounts.length > 0) {
|
|
64
|
+
await tx
|
|
65
|
+
.insert(txAccounts)
|
|
66
|
+
.values(
|
|
67
|
+
bundle.accounts.map((a) => ({
|
|
68
|
+
signature: bundle.signature,
|
|
69
|
+
accountIndex: a.index,
|
|
70
|
+
address: a.address,
|
|
71
|
+
signer: a.signer ? 1 : 0,
|
|
72
|
+
writable: a.writable ? 1 : 0,
|
|
73
|
+
programIdIndex: a.programIdIndex ?? null,
|
|
74
|
+
})),
|
|
75
|
+
)
|
|
76
|
+
.onConflictDoNothing();
|
|
77
|
+
|
|
78
|
+
await tx
|
|
79
|
+
.insert(addressSignatures)
|
|
80
|
+
.values(
|
|
81
|
+
bundle.accounts.map((a) => ({
|
|
82
|
+
address: a.address,
|
|
83
|
+
signature: bundle.signature,
|
|
84
|
+
slot: bundle.slot,
|
|
85
|
+
err: errJson ? 1 : 0,
|
|
86
|
+
blockTime: bundle.blockTime ?? null,
|
|
87
|
+
})),
|
|
88
|
+
)
|
|
89
|
+
.onConflictDoNothing();
|
|
90
|
+
}
|
|
91
|
+
});
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
async upsertAccounts(snapshots: AccountSnapshot[]): Promise<void> {
|
|
95
|
+
if (!Array.isArray(snapshots) || snapshots.length === 0) return;
|
|
96
|
+
// SQLite upsert via onConflictDoUpdate
|
|
97
|
+
for (const s of snapshots) {
|
|
98
|
+
await db
|
|
99
|
+
.insert(accounts)
|
|
100
|
+
.values({
|
|
101
|
+
address: s.address,
|
|
102
|
+
lamports: s.lamports,
|
|
103
|
+
ownerProgram: s.ownerProgram,
|
|
104
|
+
executable: s.executable ? 1 : 0,
|
|
105
|
+
rentEpoch: s.rentEpoch,
|
|
106
|
+
dataLen: s.dataLen,
|
|
107
|
+
dataBase64: s.dataBase64 ?? null,
|
|
108
|
+
lastSlot: s.lastSlot,
|
|
109
|
+
})
|
|
110
|
+
.onConflictDoUpdate({
|
|
111
|
+
target: accounts.address,
|
|
112
|
+
set: {
|
|
113
|
+
lamports: s.lamports,
|
|
114
|
+
ownerProgram: s.ownerProgram,
|
|
115
|
+
executable: s.executable ? 1 : 0,
|
|
116
|
+
rentEpoch: s.rentEpoch,
|
|
117
|
+
dataLen: s.dataLen,
|
|
118
|
+
dataBase64: s.dataBase64 ?? null,
|
|
119
|
+
lastSlot: s.lastSlot,
|
|
120
|
+
},
|
|
121
|
+
});
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
async getTransaction(signature: string) {
|
|
126
|
+
const rows = await db
|
|
127
|
+
.select()
|
|
128
|
+
.from(transactions)
|
|
129
|
+
.where(eq(transactions.signature, signature))
|
|
130
|
+
.limit(1);
|
|
131
|
+
return rows[0] || null;
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
async getStatuses(signatures: string[]) {
|
|
135
|
+
if (!Array.isArray(signatures) || signatures.length === 0)
|
|
136
|
+
return new Map<string, { slot: number; err: any | null }>();
|
|
137
|
+
const results = await db
|
|
138
|
+
.select({
|
|
139
|
+
signature: transactions.signature,
|
|
140
|
+
slot: transactions.slot,
|
|
141
|
+
errJson: transactions.errJson,
|
|
142
|
+
})
|
|
143
|
+
.from(transactions)
|
|
144
|
+
.where(inArraySafe(transactions.signature, signatures));
|
|
145
|
+
const map = new Map<string, { slot: number; err: any | null }>();
|
|
146
|
+
for (const r of results)
|
|
147
|
+
map.set(r.signature, {
|
|
148
|
+
slot: Number(r.slot),
|
|
149
|
+
err: r.errJson ? safeParse(r.errJson) : null,
|
|
150
|
+
});
|
|
151
|
+
return map;
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
async getSignaturesForAddress(
|
|
155
|
+
address: string,
|
|
156
|
+
opts: { before?: string; until?: string; limit?: number } = {},
|
|
157
|
+
) {
|
|
158
|
+
let beforeSlot: number | undefined;
|
|
159
|
+
let untilSlot: number | undefined;
|
|
160
|
+
if (opts.before) {
|
|
161
|
+
const row = await this.getTransaction(opts.before);
|
|
162
|
+
beforeSlot = row?.slot ? Number(row.slot) : undefined;
|
|
163
|
+
}
|
|
164
|
+
if (opts.until) {
|
|
165
|
+
const row = await this.getTransaction(opts.until);
|
|
166
|
+
untilSlot = row?.slot ? Number(row.slot) : undefined;
|
|
167
|
+
}
|
|
168
|
+
const limit = Math.min(Math.max(opts.limit ?? 1000, 1), 1000);
|
|
169
|
+
|
|
170
|
+
const whereClauses = [eq(addressSignatures.address, address)] as any[];
|
|
171
|
+
if (typeof beforeSlot === "number")
|
|
172
|
+
whereClauses.push(lt(addressSignatures.slot, beforeSlot));
|
|
173
|
+
if (typeof untilSlot === "number")
|
|
174
|
+
whereClauses.push(gt(addressSignatures.slot, untilSlot));
|
|
175
|
+
|
|
176
|
+
const rows = await db
|
|
177
|
+
.select({
|
|
178
|
+
signature: addressSignatures.signature,
|
|
179
|
+
slot: addressSignatures.slot,
|
|
180
|
+
blockTime: addressSignatures.blockTime,
|
|
181
|
+
err: addressSignatures.err,
|
|
182
|
+
})
|
|
183
|
+
.from(addressSignatures)
|
|
184
|
+
.where(and(...whereClauses))
|
|
185
|
+
.orderBy(desc(addressSignatures.slot))
|
|
186
|
+
.limit(limit);
|
|
187
|
+
|
|
188
|
+
return rows.map((r) => ({
|
|
189
|
+
signature: r.signature,
|
|
190
|
+
slot: Number(r.slot),
|
|
191
|
+
err: r.err ? {} : null,
|
|
192
|
+
memo: null as null,
|
|
193
|
+
blockTime: r.blockTime ? Number(r.blockTime) : null,
|
|
194
|
+
confirmationStatus: r.err ? "processed" : "confirmed",
|
|
195
|
+
}));
|
|
196
|
+
}
|
|
197
|
+
|
|
198
|
+
async getAccountsByOwner(ownerProgram: string, limit = 1000) {
|
|
199
|
+
const rows = await db
|
|
200
|
+
.select()
|
|
201
|
+
.from(accounts)
|
|
202
|
+
.where(eq(accounts.ownerProgram, ownerProgram))
|
|
203
|
+
.orderBy(desc(accounts.lastSlot))
|
|
204
|
+
.limit(Math.min(Math.max(limit, 1), 1000));
|
|
205
|
+
return rows;
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
async getBlockTimeForSlot(slot: number): Promise<number | null> {
|
|
209
|
+
const rows = await db
|
|
210
|
+
.select({ bt: transactions.blockTime })
|
|
211
|
+
.from(transactions)
|
|
212
|
+
.where(eq(transactions.slot, slot))
|
|
213
|
+
.limit(1);
|
|
214
|
+
const r = rows[0];
|
|
215
|
+
return r?.bt != null ? Number(r.bt) : null;
|
|
216
|
+
}
|
|
217
|
+
}
|
|
218
|
+
|
|
219
|
+
function safeParse<T = any>(s: string): T | null {
|
|
220
|
+
try {
|
|
221
|
+
return JSON.parse(s) as T;
|
|
222
|
+
} catch {
|
|
223
|
+
return null;
|
|
224
|
+
}
|
|
225
|
+
}
|
|
226
|
+
|
|
227
|
+
function inArraySafe<T>(col: any, arr: T[]) {
|
|
228
|
+
return arr.length > 0 ? inArray(col, arr as any) : eq(col, "__never__");
|
|
229
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
*,:after,:before{--tw-border-spacing-x:0;--tw-border-spacing-y:0;--tw-translate-x:0;--tw-translate-y:0;--tw-rotate:0;--tw-skew-x:0;--tw-skew-y:0;--tw-scale-x:1;--tw-scale-y:1;--tw-pan-x: ;--tw-pan-y: ;--tw-pinch-zoom: ;--tw-scroll-snap-strictness:proximity;--tw-gradient-from-position: ;--tw-gradient-via-position: ;--tw-gradient-to-position: ;--tw-ordinal: ;--tw-slashed-zero: ;--tw-numeric-figure: ;--tw-numeric-spacing: ;--tw-numeric-fraction: ;--tw-ring-inset: ;--tw-ring-offset-width:0px;--tw-ring-offset-color:#fff;--tw-ring-color:rgba(59,130,246,.5);--tw-ring-offset-shadow:0 0 #0000;--tw-ring-shadow:0 0 #0000;--tw-shadow:0 0 #0000;--tw-shadow-colored:0 0 #0000;--tw-blur: ;--tw-brightness: ;--tw-contrast: ;--tw-grayscale: ;--tw-hue-rotate: ;--tw-invert: ;--tw-saturate: ;--tw-sepia: ;--tw-drop-shadow: ;--tw-backdrop-blur: ;--tw-backdrop-brightness: ;--tw-backdrop-contrast: ;--tw-backdrop-grayscale: ;--tw-backdrop-hue-rotate: ;--tw-backdrop-invert: ;--tw-backdrop-opacity: ;--tw-backdrop-saturate: ;--tw-backdrop-sepia: ;--tw-contain-size: ;--tw-contain-layout: ;--tw-contain-paint: ;--tw-contain-style: }::backdrop{--tw-border-spacing-x:0;--tw-border-spacing-y:0;--tw-translate-x:0;--tw-translate-y:0;--tw-rotate:0;--tw-skew-x:0;--tw-skew-y:0;--tw-scale-x:1;--tw-scale-y:1;--tw-pan-x: ;--tw-pan-y: ;--tw-pinch-zoom: ;--tw-scroll-snap-strictness:proximity;--tw-gradient-from-position: ;--tw-gradient-via-position: ;--tw-gradient-to-position: ;--tw-ordinal: ;--tw-slashed-zero: ;--tw-numeric-figure: ;--tw-numeric-spacing: ;--tw-numeric-fraction: ;--tw-ring-inset: ;--tw-ring-offset-width:0px;--tw-ring-offset-color:#fff;--tw-ring-color:rgba(59,130,246,.5);--tw-ring-offset-shadow:0 0 #0000;--tw-ring-shadow:0 0 #0000;--tw-shadow:0 0 #0000;--tw-shadow-colored:0 0 #0000;--tw-blur: ;--tw-brightness: ;--tw-contrast: ;--tw-grayscale: ;--tw-hue-rotate: ;--tw-invert: ;--tw-saturate: ;--tw-sepia: ;--tw-drop-shadow: ;--tw-backdrop-blur: ;--tw-backdrop-brightness: ;--tw-backdrop-contrast: ;--tw-backdrop-grayscale: ;--tw-backdrop-hue-rotate: ;--tw-backdrop-invert: ;--tw-backdrop-opacity: ;--tw-backdrop-saturate: ;--tw-backdrop-sepia: ;--tw-contain-size: ;--tw-contain-layout: ;--tw-contain-paint: ;--tw-contain-style: }/*! tailwindcss v3.4.17 | MIT License | https://tailwindcss.com*/*,:after,:before{box-sizing:border-box;border:0 solid #e5e7eb}:after,:before{--tw-content:""}:host,html{line-height:1.5;-webkit-text-size-adjust:100%;-moz-tab-size:4;-o-tab-size:4;tab-size:4;font-family:ui-sans-serif,system-ui,sans-serif,Apple Color Emoji,Segoe UI Emoji,Segoe UI Symbol,Noto Color Emoji;font-feature-settings:normal;font-variation-settings:normal;-webkit-tap-highlight-color:transparent}body{margin:0;line-height:inherit}hr{height:0;color:inherit;border-top-width:1px}abbr:where([title]){-webkit-text-decoration:underline dotted;text-decoration:underline dotted}h1,h2,h3,h4,h5,h6{font-size:inherit;font-weight:inherit}a{color:inherit;text-decoration:inherit}b,strong{font-weight:bolder}code,kbd,pre,samp{font-family:ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,Liberation Mono,Courier New,monospace;font-feature-settings:normal;font-variation-settings:normal;font-size:1em}small{font-size:80%}sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:baseline}sub{bottom:-.25em}sup{top:-.5em}table{text-indent:0;border-color:inherit;border-collapse:collapse}button,input,optgroup,select,textarea{font-family:inherit;font-feature-settings:inherit;font-variation-settings:inherit;font-size:100%;font-weight:inherit;line-height:inherit;letter-spacing:inherit;color:inherit;margin:0;padding:0}button,select{text-transform:none}button,input:where([type=button]),input:where([type=reset]),input:where([type=submit]){-webkit-appearance:button;background-color:transparent;background-image:none}:-moz-focusring{outline:auto}:-moz-ui-invalid{box-shadow:none}progress{vertical-align:baseline}::-webkit-inner-spin-button,::-webkit-outer-spin-button{height:auto}[type=search]{-webkit-appearance:textfield;outline-offset:-2px}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-file-upload-button{-webkit-appearance:button;font:inherit}summary{display:list-item}blockquote,dd,dl,figure,h1,h2,h3,h4,h5,h6,hr,p,pre{margin:0}fieldset{margin:0}fieldset,legend{padding:0}menu,ol,ul{list-style:none;margin:0;padding:0}dialog{padding:0}textarea{resize:vertical}input::-moz-placeholder,textarea::-moz-placeholder{opacity:1;color:#9ca3af}input::placeholder,textarea::placeholder{opacity:1;color:#9ca3af}[role=button],button{cursor:pointer}:disabled{cursor:default}audio,canvas,embed,iframe,img,object,svg,video{display:block;vertical-align:middle}img,video{max-width:100%;height:auto}[hidden]:where(:not([hidden=until-found])){display:none}.glass-panel{border-radius:1rem;border-width:1px;background:linear-gradient(135deg,hsla(0,0%,100%,.03),hsla(0,0%,100%,.01));border-color:var(--color-border-default);backdrop-filter:blur(20px);box-shadow:var(--shadow-xl),inset 0 1px 0 0 hsla(0,0%,100%,.05);transition:var(--transition-base)}.glass-panel:hover{border-color:var(--color-border-strong);box-shadow:var(--shadow-2xl),var(--shadow-glow),inset 0 1px 0 0 hsla(0,0%,100%,.08)}.card{border-radius:.75rem;padding:1.5rem;background:var(--color-bg-surface);border:1px solid var(--color-border-subtle);transition:var(--transition-smooth);animation:fadeIn .5s ease-out}.card:hover{background:var(--color-bg-elevated);border-color:var(--color-border-default);transform:translateY(-2px);box-shadow:var(--shadow-lg)}.btn-primary{display:inline-flex;align-items:center;justify-content:center;gap:.5rem;border-radius:.75rem;padding:.625rem 1.25rem;font-weight:500;transition:var(--transition-base);position:relative;overflow:hidden;text-transform:none;letter-spacing:.01em;background:linear-gradient(135deg,var(--color-accent-primary),var(--color-accent-secondary));color:#fff;box-shadow:0 4px 15px rgba(139,92,246,.3)}.btn-primary:hover{box-shadow:0 6px 25px rgba(139,92,246,.5);transform:translateY(-2px)}.btn-primary:active{transform:translateY(0)}.btn-secondary{display:inline-flex;align-items:center;justify-content:center;gap:.5rem;border-radius:.75rem;padding:.625rem 1.25rem;font-weight:500;transition:var(--transition-base);position:relative;overflow:hidden;text-transform:none;letter-spacing:.01em;background:var(--color-bg-elevated);color:var(--color-text-primary);border:1px solid var(--color-border-default)}.btn-secondary:hover{background:var(--color-bg-hover);border-color:var(--color-accent-primary);box-shadow:inset 0 0 0 1px var(--color-accent-primary)}.btn-icon{display:inline-flex;align-items:center;justify-content:center;width:40px;height:40px;border-radius:12px;background:var(--color-bg-elevated);border:1px solid var(--color-border-subtle);color:var(--color-text-secondary);transition:var(--transition-base)}.btn-icon:hover{background:var(--color-bg-hover);color:var(--color-accent-primary);border-color:var(--color-accent-primary);transform:scale(1.05)}.\!input,.input{width:100%;border-radius:.75rem;padding:.75rem 1rem;font-size:.875rem;line-height:1.25rem}.\!input{background:var(--color-bg-surface)!important;border:1px solid var(--color-border-subtle)!important;color:var(--color-text-primary)!important;transition:var(--transition-base)!important;font-family:Inter,sans-serif!important}.input{background:var(--color-bg-surface);border:1px solid var(--color-border-subtle);color:var(--color-text-primary);transition:var(--transition-base);font-family:Inter,sans-serif}.\!input:hover{background:var(--color-bg-elevated)!important;border-color:var(--color-border-default)!important}.input:hover{background:var(--color-bg-elevated);border-color:var(--color-border-default)}.\!input:focus{outline:none!important;border-color:var(--color-accent-primary)!important;box-shadow:0 0 0 3px var(--color-accent-glow)!important;background:var(--color-bg-elevated)!important}.input:focus{outline:none;border-color:var(--color-accent-primary);box-shadow:0 0 0 3px var(--color-accent-glow);background:var(--color-bg-elevated)}.\!input::-moz-placeholder{color:var(--color-text-muted)!important}.\!input::placeholder{color:var(--color-text-muted)!important}.input::-moz-placeholder{color:var(--color-text-muted)}.input::placeholder{color:var(--color-text-muted)}.select{width:100%;border-radius:.75rem;padding:.75rem 1rem;font-size:.875rem;line-height:1.25rem;background:var(--color-bg-surface);border:1px solid var(--color-border-subtle);color:var(--color-text-primary);transition:var(--transition-base);font-family:Inter,sans-serif}.select:hover{border-color:var(--color-border-default)}.select:focus,.select:hover{background:var(--color-bg-elevated)}.select:focus{outline:none;border-color:var(--color-accent-primary);box-shadow:0 0 0 3px var(--color-accent-glow)}.select::-moz-placeholder{color:var(--color-text-muted)}.select::placeholder{color:var(--color-text-muted)}.select{cursor:pointer;padding-right:40px;background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' fill='none' stroke='%239ca3af' viewBox='0 0 24 24'%3E%3Cpath stroke-linecap='round' stroke-linejoin='round' stroke-width='2' d='m19 9-7 7-7-7'/%3E%3C/svg%3E");background-repeat:no-repeat;background-position:right 12px center;background-size:20px}.checkbox{height:1.25rem;width:1.25rem;border-radius:.375rem;background:var(--color-bg-surface);border:1px solid var(--color-border-default);cursor:pointer;transition:var(--transition-base)}.checkbox:checked{background:linear-gradient(135deg,var(--color-accent-primary),var(--color-accent-secondary));border-color:var(--color-accent-primary)}.checkbox:focus{outline:none;box-shadow:0 0 0 3px var(--color-accent-glow)}.table-modern{width:100%;border-collapse:separate;border-spacing:0}.table-modern thead{background:var(--color-bg-surface)}.table-modern th{text-align:left;font-size:.75rem;line-height:1rem;font-weight:600;text-transform:uppercase;letter-spacing:.05em;color:var(--color-text-tertiary)}.table-modern td,.table-modern th{padding:.75rem 1rem;border-bottom:1px solid var(--color-border-subtle)}.table-modern td{font-size:.875rem;line-height:1.25rem;color:var(--color-text-primary)}.table-modern tbody tr{transition:var(--transition-base)}.badge,.table-modern tbody tr:hover{background:var(--color-bg-elevated)}.badge{display:inline-flex;align-items:center;gap:.375rem;border-radius:9999px;padding:.25rem .75rem;font-size:.75rem;line-height:1rem;font-weight:500;color:var(--color-text-secondary);border:1px solid var(--color-border-subtle)}.badge-success{background:rgba(16,185,129,.1);color:#10b981;border-color:rgba(16,185,129,.2)}.badge-warning{background:rgba(245,158,11,.1);color:#f59e0b;border-color:rgba(245,158,11,.2)}.badge-info{background:rgba(59,130,246,.1);color:#3b82f6;border-color:rgba(59,130,246,.2)}.spinner{border:3px solid var(--color-border-subtle);border-top-color:var(--color-accent-primary);border-radius:50%;width:20px;height:20px;animation:spin 1s linear infinite}.tooltip.show{opacity:1}.status-dot{display:inline-block;height:.5rem;width:.5rem;border-radius:9999px;animation:pulse 2s infinite}.status-dot.online{background:var(--color-success);box-shadow:0 0 8px var(--color-success)}.status-dot.offline{background:var(--color-text-muted)}.status-dot.error{background:var(--color-error);box-shadow:0 0 8px var(--color-error)}.fixed{position:fixed}.absolute{position:absolute}.relative{position:relative}.inset-0{inset:0}.left-0{left:0}.left-3{left:.75rem}.left-4{left:1rem}.top-0{top:0}.top-1\/2{top:50%}.top-4{top:1rem}.z-30{z-index:30}.z-40{z-index:40}.z-50{z-index:50}.mx-auto{margin-left:auto;margin-right:auto}.mb-1{margin-bottom:.25rem}.mb-2{margin-bottom:.5rem}.mb-3{margin-bottom:.75rem}.mb-4{margin-bottom:1rem}.mb-6{margin-bottom:1.5rem}.ml-8{margin-left:2rem}.mr-1{margin-right:.25rem}.mt-0\.5{margin-top:.125rem}.mt-1{margin-top:.25rem}.mt-2{margin-top:.5rem}.mt-6{margin-top:1.5rem}.block{display:block}.flex{display:flex}.table{display:table}.grid{display:grid}.hidden{display:none}.h-10{height:2.5rem}.h-12{height:3rem}.h-16{height:4rem}.h-full{height:100%}.max-h-\[60vh\]{max-height:60vh}.min-h-screen{min-height:100vh}.w-10{width:2.5rem}.w-12{width:3rem}.w-16{width:4rem}.w-5{width:1.25rem}.w-72{width:18rem}.w-full{width:100%}.max-w-7xl{max-width:80rem}.max-w-lg{max-width:32rem}.flex-1{flex:1 1 0%}.-translate-x-full{--tw-translate-x:-100%}.-translate-x-full,.-translate-y-1\/2{transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.-translate-y-1\/2{--tw-translate-y:-50%}.translate-x-0{--tw-translate-x:0px}.transform,.translate-x-0{transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}@keyframes spin{to{transform:rotate(1turn)}}.animate-spin{animation:spin 1s linear infinite}.cursor-not-allowed{cursor:not-allowed}.cursor-pointer{cursor:pointer}.scroll-mt-24{scroll-margin-top:6rem}.appearance-none{-webkit-appearance:none;-moz-appearance:none;appearance:none}.flex-col{flex-direction:column}.flex-wrap{flex-wrap:wrap}.items-start{align-items:flex-start}.items-center{align-items:center}.items-stretch{align-items:stretch}.justify-center{justify-content:center}.justify-between{justify-content:space-between}.gap-2{gap:.5rem}.gap-3{gap:.75rem}.gap-4{gap:1rem}.space-y-2>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(.5rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(.5rem*var(--tw-space-y-reverse))}.space-y-3>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(.75rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(.75rem*var(--tw-space-y-reverse))}.space-y-4>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(1rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(1rem*var(--tw-space-y-reverse))}.space-y-5>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(1.25rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(1.25rem*var(--tw-space-y-reverse))}.space-y-6>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(1.5rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(1.5rem*var(--tw-space-y-reverse))}.space-y-8>:not([hidden])~:not([hidden]){--tw-space-y-reverse:0;margin-top:calc(2rem*(1 - var(--tw-space-y-reverse)));margin-bottom:calc(2rem*var(--tw-space-y-reverse))}.overflow-hidden{overflow:hidden}.overflow-x-auto{overflow-x:auto}.overflow-y-auto{overflow-y:auto}.truncate{overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.break-all{word-break:break-all}.rounded-2xl{border-radius:1rem}.rounded-full{border-radius:9999px}.rounded-lg{border-radius:.5rem}.rounded-none{border-radius:0}.rounded-xl{border-radius:.75rem}.border{border-width:1px}.border-b{border-bottom-width:1px}.border-r{border-right-width:1px}.border-t{border-top-width:1px}.border-green-500\/30{border-color:rgba(34,197,94,.3)}.border-purple-500\/30{border-color:rgba(168,85,247,.3)}.border-red-500\/30{border-color:rgba(239,68,68,.3)}.border-white\/10{border-color:hsla(0,0%,100%,.1)}.border-white\/20{border-color:hsla(0,0%,100%,.2)}.border-white\/5{border-color:hsla(0,0%,100%,.05)}.bg-black\/50{background-color:rgba(0,0,0,.5)}.bg-black\/80{background-color:rgba(0,0,0,.8)}.bg-gray-900\/95{background-color:rgba(17,24,39,.95)}.bg-green-500\/10{background-color:rgba(34,197,94,.1)}.bg-red-500\/10{background-color:rgba(239,68,68,.1)}.bg-white\/5{background-color:hsla(0,0%,100%,.05)}.bg-gradient-to-br{background-image:linear-gradient(to bottom right,var(--tw-gradient-stops))}.bg-gradient-to-r{background-image:linear-gradient(to right,var(--tw-gradient-stops))}.from-amber-500\/20{--tw-gradient-from:rgba(245,158,11,.2) var(--tw-gradient-from-position);--tw-gradient-to:rgba(245,158,11,0) var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to)}.from-amber-600\/20{--tw-gradient-from:rgba(217,119,6,.2) var(--tw-gradient-from-position);--tw-gradient-to:rgba(217,119,6,0) var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to)}.from-blue-500\/20{--tw-gradient-from:rgba(59,130,246,.2) var(--tw-gradient-from-position);--tw-gradient-to:rgba(59,130,246,0) var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to)}.from-blue-600\/20{--tw-gradient-from:rgba(37,99,235,.2) var(--tw-gradient-from-position);--tw-gradient-to:rgba(37,99,235,0) var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to)}.from-gray-600\/20{--tw-gradient-from:rgba(75,85,99,.2) var(--tw-gradient-from-position);--tw-gradient-to:rgba(75,85,99,0) var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to)}.from-green-500\/20{--tw-gradient-from:rgba(34,197,94,.2) var(--tw-gradient-from-position);--tw-gradient-to:rgba(34,197,94,0) var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to)}.from-purple-400{--tw-gradient-from:#c084fc var(--tw-gradient-from-position);--tw-gradient-to:rgba(192,132,252,0) var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to)}.from-purple-500\/20{--tw-gradient-from:rgba(168,85,247,.2) var(--tw-gradient-from-position);--tw-gradient-to:rgba(168,85,247,0) var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to)}.from-purple-600{--tw-gradient-from:#9333ea var(--tw-gradient-from-position);--tw-gradient-to:rgba(147,51,234,0) var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to)}.from-purple-600\/20{--tw-gradient-from:rgba(147,51,234,.2) var(--tw-gradient-from-position);--tw-gradient-to:rgba(147,51,234,0) var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to)}.from-red-500\/10{--tw-gradient-from:rgba(239,68,68,.1) var(--tw-gradient-from-position);--tw-gradient-to:rgba(239,68,68,0) var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to)}.from-violet-500\/20{--tw-gradient-from:rgba(139,92,246,.2) var(--tw-gradient-from-position);--tw-gradient-to:rgba(139,92,246,0) var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to)}.from-white\/5{--tw-gradient-from:hsla(0,0%,100%,.05) var(--tw-gradient-from-position);--tw-gradient-to:hsla(0,0%,100%,0) var(--tw-gradient-to-position);--tw-gradient-stops:var(--tw-gradient-from),var(--tw-gradient-to)}.to-cyan-500\/20{--tw-gradient-to:rgba(6,182,212,.2) var(--tw-gradient-to-position)}.to-cyan-600\/20{--tw-gradient-to:rgba(8,145,178,.2) var(--tw-gradient-to-position)}.to-emerald-500\/20{--tw-gradient-to:rgba(16,185,129,.2) var(--tw-gradient-to-position)}.to-gray-700\/20{--tw-gradient-to:rgba(55,65,81,.2) var(--tw-gradient-to-position)}.to-orange-500\/20{--tw-gradient-to:rgba(249,115,22,.2) var(--tw-gradient-to-position)}.to-orange-600\/20{--tw-gradient-to:rgba(234,88,12,.2) var(--tw-gradient-to-position)}.to-pink-500\/10{--tw-gradient-to:rgba(236,72,153,.1) var(--tw-gradient-to-position)}.to-purple-500\/20{--tw-gradient-to:rgba(168,85,247,.2) var(--tw-gradient-to-position)}.to-transparent{--tw-gradient-to:transparent var(--tw-gradient-to-position)}.to-violet-400{--tw-gradient-to:#a78bfa var(--tw-gradient-to-position)}.to-violet-500\/20{--tw-gradient-to:rgba(139,92,246,.2) var(--tw-gradient-to-position)}.to-violet-600{--tw-gradient-to:#7c3aed var(--tw-gradient-to-position)}.to-violet-600\/20{--tw-gradient-to:rgba(124,58,237,.2) var(--tw-gradient-to-position)}.bg-clip-text{-webkit-background-clip:text;background-clip:text}.p-0{padding:0}.p-3{padding:.75rem}.p-4{padding:1rem}.p-6{padding:1.5rem}.px-4{padding-left:1rem;padding-right:1rem}.py-12{padding-top:3rem;padding-bottom:3rem}.py-3{padding-top:.75rem;padding-bottom:.75rem}.pl-10{padding-left:2.5rem}.pt-2{padding-top:.5rem}.pt-3{padding-top:.75rem}.text-left{text-align:left}.text-center{text-align:center}.font-mono{font-family:ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,Liberation Mono,Courier New,monospace}.text-2xl{font-size:1.5rem;line-height:2rem}.text-3xl{font-size:1.875rem;line-height:2.25rem}.text-sm{font-size:.875rem;line-height:1.25rem}.text-xl{font-size:1.25rem;line-height:1.75rem}.text-xs{font-size:.75rem;line-height:1rem}.font-bold{font-weight:700}.font-medium{font-weight:500}.font-semibold{font-weight:600}.uppercase{text-transform:uppercase}.tracking-wider{letter-spacing:.05em}.text-amber-300{--tw-text-opacity:1;color:rgb(252 211 77/var(--tw-text-opacity,1))}.text-amber-400{--tw-text-opacity:1;color:rgb(251 191 36/var(--tw-text-opacity,1))}.text-amber-500{--tw-text-opacity:1;color:rgb(245 158 11/var(--tw-text-opacity,1))}.text-blue-300{--tw-text-opacity:1;color:rgb(147 197 253/var(--tw-text-opacity,1))}.text-blue-400{--tw-text-opacity:1;color:rgb(96 165 250/var(--tw-text-opacity,1))}.text-blue-500{--tw-text-opacity:1;color:rgb(59 130 246/var(--tw-text-opacity,1))}.text-gray-300{--tw-text-opacity:1;color:rgb(209 213 219/var(--tw-text-opacity,1))}.text-gray-400{--tw-text-opacity:1;color:rgb(156 163 175/var(--tw-text-opacity,1))}.text-gray-500{--tw-text-opacity:1;color:rgb(107 114 128/var(--tw-text-opacity,1))}.text-green-300{--tw-text-opacity:1;color:rgb(134 239 172/var(--tw-text-opacity,1))}.text-green-400{--tw-text-opacity:1;color:rgb(74 222 128/var(--tw-text-opacity,1))}.text-purple-300{--tw-text-opacity:1;color:rgb(216 180 254/var(--tw-text-opacity,1))}.text-purple-400{--tw-text-opacity:1;color:rgb(192 132 252/var(--tw-text-opacity,1))}.text-red-300{--tw-text-opacity:1;color:rgb(252 165 165/var(--tw-text-opacity,1))}.text-red-400{--tw-text-opacity:1;color:rgb(248 113 113/var(--tw-text-opacity,1))}.text-transparent{color:transparent}.text-violet-300{--tw-text-opacity:1;color:rgb(196 181 253/var(--tw-text-opacity,1))}.text-violet-400{--tw-text-opacity:1;color:rgb(167 139 250/var(--tw-text-opacity,1))}.text-white{--tw-text-opacity:1;color:rgb(255 255 255/var(--tw-text-opacity,1))}.opacity-50{opacity:.5}.shadow-2xl{--tw-shadow:0 25px 50px -12px rgba(0,0,0,.25);--tw-shadow-colored:0 25px 50px -12px var(--tw-shadow-color)}.shadow-2xl,.shadow-lg{box-shadow:var(--tw-ring-offset-shadow,0 0 #0000),var(--tw-ring-shadow,0 0 #0000),var(--tw-shadow)}.shadow-lg{--tw-shadow:0 10px 15px -3px rgba(0,0,0,.1),0 4px 6px -4px rgba(0,0,0,.1);--tw-shadow-colored:0 10px 15px -3px var(--tw-shadow-color),0 4px 6px -4px var(--tw-shadow-color)}.backdrop-blur-sm{--tw-backdrop-blur:blur(4px)}.backdrop-blur-sm,.backdrop-blur-xl{backdrop-filter:var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia)}.backdrop-blur-xl{--tw-backdrop-blur:blur(24px)}.transition-all{transition-property:all;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-colors{transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-transform{transition-property:transform;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.duration-200{transition-duration:.2s}.duration-300{transition-duration:.3s}.ease-out{transition-timing-function:cubic-bezier(0,0,.2,1)}:root{color-scheme:dark;--color-bg-base:#0a0a0f;--color-bg-surface:#12121a;--color-bg-elevated:#1a1a24;--color-bg-hover:#22222e;--color-bg-active:#2a2a38;--color-accent-primary:#8b5cf6;--color-accent-secondary:#a78bfa;--color-accent-tertiary:#c4b5fd;--color-accent-glow:rgba(139,92,246,.4);--color-success:#10b981;--color-warning:#f59e0b;--color-error:#ef4444;--color-info:#3b82f6;--color-text-primary:#f3f4f6;--color-text-secondary:#9ca3af;--color-text-tertiary:#6b7280;--color-text-muted:#4b5563;--color-border-subtle:hsla(0,0%,100%,.06);--color-border-default:hsla(0,0%,100%,.1);--color-border-strong:hsla(0,0%,100%,.15);--shadow-sm:0 1px 2px 0 rgba(0,0,0,.05);--shadow-md:0 4px 6px -1px rgba(0,0,0,.1),0 2px 4px -2px rgba(0,0,0,.1);--shadow-lg:0 10px 15px -3px rgba(0,0,0,.1),0 4px 6px -4px rgba(0,0,0,.1);--shadow-xl:0 20px 25px -5px rgba(0,0,0,.1),0 8px 10px -6px rgba(0,0,0,.1);--shadow-2xl:0 25px 50px -12px rgba(0,0,0,.25);--shadow-glow:0 0 50px rgba(139,92,246,.3);--transition-base:all 0.2s cubic-bezier(0.4,0,0.2,1);--transition-smooth:all 0.3s cubic-bezier(0.4,0,0.2,1);--transition-spring:all 0.5s cubic-bezier(0.68,-0.55,0.265,1.55)}*{box-sizing:border-box;margin:0;padding:0}html{font-family:Inter,system-ui,-apple-system,sans-serif;font-feature-settings:"cv11","ss01";font-variation-settings:"opsz" 32;scroll-behavior:smooth}body{-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;background:var(--color-bg-base);color:var(--color-text-primary);min-height:100vh;position:relative;overflow-x:hidden}body:before{content:"";position:fixed;top:0;left:0;right:0;bottom:0;background:radial-gradient(ellipse at top left,rgba(139,92,246,.15) 0,transparent 50%),radial-gradient(ellipse at bottom right,rgba(167,139,250,.1) 0,transparent 50%);pointer-events:none;z-index:0}#root{position:relative;z-index:1}::-webkit-scrollbar{width:12px;height:12px}::-webkit-scrollbar-track{background:var(--color-bg-surface);border-radius:6px}::-webkit-scrollbar-thumb{background:linear-gradient(180deg,var(--color-accent-primary),var(--color-accent-secondary));border-radius:6px;border:2px solid var(--color-bg-surface)}::-webkit-scrollbar-thumb:hover{background:linear-gradient(180deg,var(--color-accent-secondary),var(--color-accent-tertiary))}@keyframes fadeIn{0%{opacity:0;transform:translateY(10px)}to{opacity:1;transform:translateY(0)}}@keyframes slideIn{0%{opacity:0;transform:translateX(-20px)}to{opacity:1;transform:translateX(0)}}@keyframes pulse{0%,to{opacity:1}50%{opacity:.5}}@keyframes shimmer{0%{background-position:-1000px 0}to{background-position:1000px 0}}@keyframes glow{0%,to{box-shadow:0 0 20px rgba(139,92,246,.5),inset 0 0 20px rgba(139,92,246,.1)}50%{box-shadow:0 0 40px rgba(139,92,246,.8),inset 0 0 30px rgba(139,92,246,.2)}}.text-mono{font-family:JetBrains Mono,monospace}:focus-visible{outline:2px solid var(--color-accent-primary);outline-offset:2px}::-moz-selection{background:var(--color-accent-primary);color:#fff}::selection{background:var(--color-accent-primary);color:#fff}.hover\:scale-\[1\.02\]:hover{--tw-scale-x:1.02;--tw-scale-y:1.02;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.hover\:border-red-500\/30:hover{border-color:rgba(239,68,68,.3)}.hover\:bg-red-500\/20:hover{background-color:rgba(239,68,68,.2)}.hover\:bg-white\/5:hover{background-color:hsla(0,0%,100%,.05)}.hover\:text-red-300:hover{--tw-text-opacity:1;color:rgb(252 165 165/var(--tw-text-opacity,1))}.hover\:text-red-400:hover{--tw-text-opacity:1;color:rgb(248 113 113/var(--tw-text-opacity,1))}.group:hover .group-hover\:scale-110{--tw-scale-x:1.1;--tw-scale-y:1.1;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.group:hover .group-hover\:text-purple-300{--tw-text-opacity:1;color:rgb(216 180 254/var(--tw-text-opacity,1))}@media (min-width:640px){.sm\:flex-initial{flex:0 1 auto}.sm\:grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.sm\:flex-row{flex-direction:row}.sm\:items-center{align-items:center}}@media (min-width:768px){.md\:block{display:block}.md\:hidden{display:none}.md\:flex-row{flex-direction:row}.md\:items-center{align-items:center}.md\:justify-between{justify-content:space-between}.md\:p-8{padding:2rem}}@media (min-width:1024px){.lg\:ml-72{margin-left:18rem}.lg\:hidden{display:none}.lg\:translate-x-0{--tw-translate-x:0px;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.lg\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.lg\:grid-cols-4{grid-template-columns:repeat(4,minmax(0,1fr))}.lg\:flex-row{flex-direction:row}.lg\:items-center{align-items:center}}
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
<!doctype html>
|
|
2
|
+
<html lang="en">
|
|
3
|
+
<head>
|
|
4
|
+
<meta charset="utf-8" />
|
|
5
|
+
<meta name="viewport" content="width=device-width,initial-scale=1" />
|
|
6
|
+
<meta name="color-scheme" content="dark" />
|
|
7
|
+
<meta name="theme-color" content="#0a0a0f" />
|
|
8
|
+
<title>SolForge - Solana Development Suite</title>
|
|
9
|
+
<link rel="preconnect" href="https://fonts.googleapis.com">
|
|
10
|
+
<link rel="preconnect" href="https://fonts.gstatic.com" crossorigin>
|
|
11
|
+
<link href="https://fonts.googleapis.com/css2?family=Inter:wght@300;400;500;600;700;800;900&family=JetBrains+Mono:wght@400;500;600;700&display=swap" rel="stylesheet">
|
|
12
|
+
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.5.1/css/all.min.css">
|
|
13
|
+
</head>
|
|
14
|
+
<body>
|
|
15
|
+
<link rel="stylesheet" href="./app.css" />
|
|
16
|
+
<div id="root"></div>
|
|
17
|
+
<script defer src="./build/main.js"></script>
|
|
18
|
+
</body>
|
|
19
|
+
</html>
|
|
@@ -0,0 +1,297 @@
|
|
|
1
|
+
import { serve } from "bun";
|
|
2
|
+
import type { LiteSVMRpcServer } from "../../server/rpc-server";
|
|
3
|
+
import type { JsonRpcResponse } from "../../server/types";
|
|
4
|
+
import { readConfig, writeConfig } from "../config";
|
|
5
|
+
import indexHtml from "./public/index.html";
|
|
6
|
+
import { file } from "bun";
|
|
7
|
+
// Embed built GUI assets as files so the compiled binary can stream them
|
|
8
|
+
import appCssFile from "./public/app.css" with { type: "file" };
|
|
9
|
+
import mainJsFile from "./public/build/main.js" with { type: "file" };
|
|
10
|
+
import bundledCssFile from "./public/build/main.css" with { type: "file" };
|
|
11
|
+
|
|
12
|
+
type GuiStartOptions = {
|
|
13
|
+
port?: number;
|
|
14
|
+
host?: string;
|
|
15
|
+
rpcPort?: number;
|
|
16
|
+
rpcServer?: LiteSVMRpcServer;
|
|
17
|
+
};
|
|
18
|
+
|
|
19
|
+
const LAMPORTS_PER_SOL = 1_000_000_000n;
|
|
20
|
+
const CORS = {
|
|
21
|
+
"Access-Control-Allow-Origin": "*",
|
|
22
|
+
"Access-Control-Allow-Headers": "content-type",
|
|
23
|
+
"Access-Control-Allow-Methods": "GET,POST,OPTIONS",
|
|
24
|
+
};
|
|
25
|
+
|
|
26
|
+
class HttpError extends Error {
|
|
27
|
+
constructor(
|
|
28
|
+
public status: number,
|
|
29
|
+
message: string,
|
|
30
|
+
public details?: unknown,
|
|
31
|
+
) {
|
|
32
|
+
super(message);
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
const json = (data: unknown, status = 200) =>
|
|
37
|
+
new Response(JSON.stringify(data), {
|
|
38
|
+
status,
|
|
39
|
+
headers: { ...CORS, "Content-Type": "application/json" },
|
|
40
|
+
});
|
|
41
|
+
|
|
42
|
+
const text = (value: string, status = 200) =>
|
|
43
|
+
new Response(value, {
|
|
44
|
+
status,
|
|
45
|
+
headers: { ...CORS, "Content-Type": "text/plain" },
|
|
46
|
+
});
|
|
47
|
+
|
|
48
|
+
const okOptions = () => new Response(null, { status: 204, headers: CORS });
|
|
49
|
+
const css = (fpath: string) =>
|
|
50
|
+
new Response(file(fpath), {
|
|
51
|
+
headers: { ...CORS, "Content-Type": "text/css" },
|
|
52
|
+
});
|
|
53
|
+
const js = (fpath: string) =>
|
|
54
|
+
new Response(file(fpath), {
|
|
55
|
+
headers: { ...CORS, "Content-Type": "application/javascript" },
|
|
56
|
+
});
|
|
57
|
+
|
|
58
|
+
const handleError = (error: unknown) => {
|
|
59
|
+
if (error instanceof HttpError)
|
|
60
|
+
return json({ error: error.message, details: error.details }, error.status);
|
|
61
|
+
console.error("[gui] API error", error);
|
|
62
|
+
return json({ error: "Internal server error" }, 500);
|
|
63
|
+
};
|
|
64
|
+
|
|
65
|
+
const wrap =
|
|
66
|
+
(handler: (req: Request) => Promise<unknown> | unknown) =>
|
|
67
|
+
async (req: Request) => {
|
|
68
|
+
try {
|
|
69
|
+
const result = await handler(req);
|
|
70
|
+
return result instanceof Response ? result : json(result);
|
|
71
|
+
} catch (error) {
|
|
72
|
+
return handleError(error);
|
|
73
|
+
}
|
|
74
|
+
};
|
|
75
|
+
|
|
76
|
+
const readJson = async <T extends Record<string, unknown>>(
|
|
77
|
+
req: Request,
|
|
78
|
+
): Promise<T> => {
|
|
79
|
+
try {
|
|
80
|
+
return (await req.json()) as T;
|
|
81
|
+
} catch (error) {
|
|
82
|
+
throw new HttpError(400, "Invalid JSON body", { cause: error });
|
|
83
|
+
}
|
|
84
|
+
};
|
|
85
|
+
|
|
86
|
+
const parseLamports = (payload: {
|
|
87
|
+
lamports?: string | number;
|
|
88
|
+
sol?: string | number;
|
|
89
|
+
}) => {
|
|
90
|
+
if (payload.lamports != null) {
|
|
91
|
+
const lamports = BigInt(payload.lamports);
|
|
92
|
+
if (lamports <= 0n) throw new HttpError(400, "lamports must be positive");
|
|
93
|
+
return lamports;
|
|
94
|
+
}
|
|
95
|
+
if (payload.sol != null) {
|
|
96
|
+
const sol = Number(payload.sol);
|
|
97
|
+
if (!Number.isFinite(sol) || sol <= 0)
|
|
98
|
+
throw new HttpError(400, "sol must be positive");
|
|
99
|
+
return BigInt(Math.floor(sol * Number(LAMPORTS_PER_SOL)));
|
|
100
|
+
}
|
|
101
|
+
throw new HttpError(400, "lamports or sol is required");
|
|
102
|
+
};
|
|
103
|
+
|
|
104
|
+
export function startGuiServer(opts: GuiStartOptions = {}) {
|
|
105
|
+
const host = opts.host ?? "127.0.0.1";
|
|
106
|
+
const port = Number(opts.port ?? 42069);
|
|
107
|
+
const rpcPort = Number(opts.rpcPort ?? 8899);
|
|
108
|
+
const rpcServer = opts.rpcServer;
|
|
109
|
+
const rpcUrl = `http://${host}:${rpcPort}`;
|
|
110
|
+
|
|
111
|
+
const callRpc = async (method: string, params: any[] = []) => {
|
|
112
|
+
if (!rpcServer) throw new HttpError(503, "RPC server not available");
|
|
113
|
+
const response: JsonRpcResponse = await rpcServer.handleRequest({
|
|
114
|
+
jsonrpc: "2.0",
|
|
115
|
+
id: `gui-${Date.now()}`,
|
|
116
|
+
method,
|
|
117
|
+
params,
|
|
118
|
+
});
|
|
119
|
+
if (response.error)
|
|
120
|
+
throw new HttpError(
|
|
121
|
+
400,
|
|
122
|
+
response.error.message ?? "RPC error",
|
|
123
|
+
response.error,
|
|
124
|
+
);
|
|
125
|
+
return response.result;
|
|
126
|
+
};
|
|
127
|
+
|
|
128
|
+
async function recordTokenClone(mint: string) {
|
|
129
|
+
try {
|
|
130
|
+
const cfg = await readConfig();
|
|
131
|
+
const set = new Set(cfg.clone.tokens || []);
|
|
132
|
+
if (!set.has(mint)) {
|
|
133
|
+
set.add(mint);
|
|
134
|
+
cfg.clone.tokens = Array.from(set);
|
|
135
|
+
await writeConfig(cfg);
|
|
136
|
+
}
|
|
137
|
+
} catch (error) {
|
|
138
|
+
console.warn("[gui] failed to update config for token clone:", error);
|
|
139
|
+
}
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
async function recordProgramClone(programId: string) {
|
|
143
|
+
try {
|
|
144
|
+
const cfg = await readConfig();
|
|
145
|
+
const set = new Set(cfg.clone.programs || []);
|
|
146
|
+
if (!set.has(programId)) {
|
|
147
|
+
set.add(programId);
|
|
148
|
+
cfg.clone.programs = Array.from(set);
|
|
149
|
+
await writeConfig(cfg);
|
|
150
|
+
}
|
|
151
|
+
} catch (error) {
|
|
152
|
+
console.warn("[gui] failed to update config for program clone:", error);
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
const routes = {
|
|
157
|
+
"/": indexHtml,
|
|
158
|
+
"/app.css": { GET: () => css(appCssFile) },
|
|
159
|
+
"/build/main.css": { GET: () => css(bundledCssFile) },
|
|
160
|
+
"/build/main.js": { GET: () => js(mainJsFile) },
|
|
161
|
+
"/health": { GET: () => text("ok") },
|
|
162
|
+
"/api/config": { GET: () => json({ rpcUrl }), OPTIONS: okOptions },
|
|
163
|
+
"/api/status": {
|
|
164
|
+
GET: wrap(() => callRpc("solforgeGetStatus")),
|
|
165
|
+
OPTIONS: okOptions,
|
|
166
|
+
},
|
|
167
|
+
"/api/programs": {
|
|
168
|
+
GET: wrap(() => callRpc("solforgeListPrograms")),
|
|
169
|
+
OPTIONS: okOptions,
|
|
170
|
+
},
|
|
171
|
+
"/api/tokens": {
|
|
172
|
+
GET: wrap(() => callRpc("solforgeListTokensDetailed")),
|
|
173
|
+
OPTIONS: okOptions,
|
|
174
|
+
},
|
|
175
|
+
"/api/airdrop": {
|
|
176
|
+
POST: wrap(async (req) => {
|
|
177
|
+
const body = await readJson<{
|
|
178
|
+
address?: string;
|
|
179
|
+
lamports?: string | number;
|
|
180
|
+
sol?: string | number;
|
|
181
|
+
}>(req);
|
|
182
|
+
const address =
|
|
183
|
+
typeof body.address === "string" ? body.address.trim() : "";
|
|
184
|
+
if (!address) throw new HttpError(400, "address is required");
|
|
185
|
+
const lamports = parseLamports(body);
|
|
186
|
+
const signature = await callRpc("requestAirdrop", [
|
|
187
|
+
address,
|
|
188
|
+
Number(lamports),
|
|
189
|
+
]);
|
|
190
|
+
return { ok: true, signature };
|
|
191
|
+
}),
|
|
192
|
+
OPTIONS: okOptions,
|
|
193
|
+
},
|
|
194
|
+
"/api/mint": {
|
|
195
|
+
POST: wrap(async (req) => {
|
|
196
|
+
const body = await readJson<{
|
|
197
|
+
mint?: string;
|
|
198
|
+
owner?: string;
|
|
199
|
+
amountRaw?: string;
|
|
200
|
+
}>(req);
|
|
201
|
+
const mint = typeof body.mint === "string" ? body.mint.trim() : "";
|
|
202
|
+
const owner = typeof body.owner === "string" ? body.owner.trim() : "";
|
|
203
|
+
const amountRaw =
|
|
204
|
+
typeof body.amountRaw === "string" ? body.amountRaw.trim() : "";
|
|
205
|
+
if (!mint || !owner || !amountRaw)
|
|
206
|
+
throw new HttpError(400, "mint, owner and amountRaw are required");
|
|
207
|
+
return callRpc("solforgeMintTo", [mint, owner, amountRaw]);
|
|
208
|
+
}),
|
|
209
|
+
OPTIONS: okOptions,
|
|
210
|
+
},
|
|
211
|
+
"/api/clone/program": {
|
|
212
|
+
POST: wrap(async (req) => {
|
|
213
|
+
const body = await readJson<{
|
|
214
|
+
programId?: string;
|
|
215
|
+
endpoint?: string;
|
|
216
|
+
withAccounts?: boolean;
|
|
217
|
+
accountsLimit?: number;
|
|
218
|
+
}>(req);
|
|
219
|
+
const programId =
|
|
220
|
+
typeof body.programId === "string" ? body.programId.trim() : "";
|
|
221
|
+
if (!programId) throw new HttpError(400, "programId is required");
|
|
222
|
+
const endpoint =
|
|
223
|
+
typeof body.endpoint === "string" && body.endpoint.trim()
|
|
224
|
+
? body.endpoint.trim()
|
|
225
|
+
: undefined;
|
|
226
|
+
const withAccounts = body.withAccounts === true;
|
|
227
|
+
const accountsLimit =
|
|
228
|
+
typeof body.accountsLimit === "number"
|
|
229
|
+
? body.accountsLimit
|
|
230
|
+
: undefined;
|
|
231
|
+
const res = await callRpc("solforgeAdminCloneProgram", [
|
|
232
|
+
programId,
|
|
233
|
+
{ endpoint, withAccounts, accountsLimit },
|
|
234
|
+
]);
|
|
235
|
+
await recordProgramClone(programId);
|
|
236
|
+
return res;
|
|
237
|
+
}),
|
|
238
|
+
OPTIONS: okOptions,
|
|
239
|
+
},
|
|
240
|
+
"/api/clone/token": {
|
|
241
|
+
POST: wrap(async (req) => {
|
|
242
|
+
const body = await readJson<{
|
|
243
|
+
mint?: string;
|
|
244
|
+
endpoint?: string;
|
|
245
|
+
cloneAccounts?: boolean;
|
|
246
|
+
holders?: number;
|
|
247
|
+
allAccounts?: boolean;
|
|
248
|
+
}>(req);
|
|
249
|
+
const mint = typeof body.mint === "string" ? body.mint.trim() : "";
|
|
250
|
+
if (!mint) throw new HttpError(400, "mint is required");
|
|
251
|
+
const endpoint =
|
|
252
|
+
typeof body.endpoint === "string" && body.endpoint.trim()
|
|
253
|
+
? body.endpoint.trim()
|
|
254
|
+
: undefined;
|
|
255
|
+
const cloneAccounts = body.cloneAccounts === true;
|
|
256
|
+
const holders =
|
|
257
|
+
typeof body.holders === "number" ? body.holders : undefined;
|
|
258
|
+
const allAccounts = body.allAccounts === true;
|
|
259
|
+
const mintResult = await callRpc("solforgeAdminCloneTokenMint", [
|
|
260
|
+
mint,
|
|
261
|
+
{ endpoint },
|
|
262
|
+
]);
|
|
263
|
+
// Ensure local faucet is the mint authority so GUI minting works out of the box
|
|
264
|
+
try {
|
|
265
|
+
await callRpc("solforgeAdoptMintAuthority", [mint]);
|
|
266
|
+
} catch (e) {
|
|
267
|
+
console.warn("[gui] adopt mint authority failed (continuing)", e);
|
|
268
|
+
}
|
|
269
|
+
let accountsResult: unknown = null;
|
|
270
|
+
if (cloneAccounts) {
|
|
271
|
+
accountsResult = await callRpc("solforgeAdminCloneTokenAccounts", [
|
|
272
|
+
mint,
|
|
273
|
+
{ endpoint, holders, allAccounts },
|
|
274
|
+
]);
|
|
275
|
+
}
|
|
276
|
+
await recordTokenClone(mint);
|
|
277
|
+
return {
|
|
278
|
+
mint: mintResult,
|
|
279
|
+
adoptedAuthority: true,
|
|
280
|
+
accounts: accountsResult,
|
|
281
|
+
};
|
|
282
|
+
}),
|
|
283
|
+
OPTIONS: okOptions,
|
|
284
|
+
},
|
|
285
|
+
} as const;
|
|
286
|
+
|
|
287
|
+
const server = serve({
|
|
288
|
+
port,
|
|
289
|
+
hostname: host,
|
|
290
|
+
routes,
|
|
291
|
+
development: false,
|
|
292
|
+
});
|
|
293
|
+
console.log(`🖥️ Solforge GUI available at http://${host}:${server.port}`);
|
|
294
|
+
return { server, port: server.port };
|
|
295
|
+
}
|
|
296
|
+
|
|
297
|
+
export type { GuiStartOptions };
|