midnight-wallet-cli 0.2.5 → 0.4.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/LICENSE +201 -0
- package/README.md +95 -20
- package/dist/mcp-server.js +1130 -237
- package/dist/wallet.js +1528 -377
- package/docs/SKILL.md +136 -0
- package/package.json +34 -8
package/dist/mcp-server.js
CHANGED
|
@@ -1,224 +1,312 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
|
-
var
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
Valid
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
Valid
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
2
|
+
import{createRequire as w5}from"node:module";var y5=Object.create;var{getPrototypeOf:b5,defineProperty:h2,getOwnPropertyNames:S5}=Object;var k5=Object.prototype.hasOwnProperty;var R4=($,Z,Q)=>{Q=$!=null?y5(b5($)):{};let X=Z||!$||!$.__esModule?h2(Q,"default",{value:$,enumerable:!0}):Q;for(let z of S5($))if(!k5.call(X,z))h2(X,z,{get:()=>$[z],enumerable:!0});return X};var k=($,Z)=>{for(var Q in Z)h2($,Q,{get:Z[Q],enumerable:!0,configurable:!0,set:(X)=>Z[Q]=()=>X})};var N=($,Z)=>()=>($&&(Z=$($=0)),Z);var r0=w5(import.meta.url);function A($,Z){let Q=$.flags[Z];if(Q===void 0||Q===!0)return;return Q}function C($,Z){return Z in $.flags}function S0($){return C($,"verbose")}function s0($){return C($,v1)&&!C($,a4)}function A4($){if(C($,"no-cache"))throw new Error(`--no-cache is not supported on write commands (transfer, airdrop, dust register, serve).
|
|
3
|
+
`+`Writes always use the cache — the SDK's fresh-sync is too slow on hosted networks.
|
|
4
|
+
`+"To reset: midnight cache clear --wallet <name> --network <name>")}function D4($,Z,Q){let X=A($,Z);if(X===void 0)throw new Error(`Missing required flag: --${Z} <${Q}>`);return X}var v1="_minimal",a4="_full";var O3={};k(O3,{writeJsonResult:()=>x,writeJsonError:()=>f5,setCaptureTarget:()=>y1});function y1($){r4=$}function x($){let Z=JSON.stringify($)+`
|
|
5
|
+
`;if(r4)r4(Z);else process.stdout.write(Z)}function f5($,Z,Q){let X=JSON.stringify({error:!0,code:Z,message:$.message,exitCode:Q})+`
|
|
6
|
+
`;if(r4)r4(X);else process.stdout.write(X)}var r4=null;var l1={};k(l1,{isValidWalletName:()=>$4,WALLETS_DIR_NAME:()=>e4,TX_TTL_MINUTES:()=>N4,TOKEN_MULTIPLIER:()=>w1,TOKEN_DECIMALS:()=>k1,SYNC_TIMEOUT_MS:()=>p2,SYNC_ATTEMPT_TIMEOUT_MS:()=>f1,SYNC_ATTEMPT_REMOTE_TIMEOUT_MS:()=>u5,STALE_UTXO_ERROR_CODE:()=>i2,RETRY_BASE_DELAY_MS:()=>p5,PROOF_TIMEOUT_MS:()=>s4,PRE_SEND_SYNC_TIMEOUT_MS:()=>c2,NATIVE_TOKEN_TYPE:()=>S1,MIN_DUST_FOR_TRANSFER:()=>E4,MIDNIGHT_DIR:()=>K0,MAX_RETRY_ATTEMPTS:()=>l2,LOCALNET_DIR_NAME:()=>n2,GENESIS_SEED:()=>t0,FILE_MODE:()=>D0,DUST_TIMEOUT_MS:()=>h1,DUST_RETRY_DELAY_MS:()=>p1,DUST_RETRY_ATTEMPTS:()=>x4,DUST_REGISTRATION_TIMEOUT_MS:()=>d2,DUST_REGISTRATION_RETRY_DELAY_MS:()=>g1,DUST_FEE_BLOCKS_MARGIN:()=>u2,DUST_COST_OVERHEAD:()=>g2,DIR_MODE:()=>A0,DEFAULT_WALLET_NAME:()=>C4,DEFAULT_WALLET_FILENAME:()=>t4,DEFAULT_SERVE_PORT:()=>c1,DEFAULT_CONFIG_FILENAME:()=>o2,DASHBOARD_BASE_URL:()=>$1,CACHE_VERSION:()=>u1,CACHE_DIR_NAME:()=>e0,BALANCE_OVERSPEND_ERROR_CODE:()=>c5,BALANCE_CHECK_TIMEOUT_MS:()=>m1,ABANDONED_TX_TIMEOUT_MS:()=>a2});function $4($){if(!$||$!==$.trim())return!1;if(/[\/\\]/.test($))return!1;if($.endsWith(".json"))return!1;if($==="."||$==="..")return!1;if(/[\x00-\x1f]/.test($))return!1;return!0}var t0="0000000000000000000000000000000000000000000000000000000000000001",S1="0000000000000000000000000000000000000000000000000000000000000000",k1=6,w1=1e6,g2=300000000000000n,u2=5,E4=800000000000000n,p2=300000,f1=30000,u5=120000,c2=1e4,h1=120000,s4=300000,m1=60000,N4=10,l2=3,p5=1000,d2=600000,g1=15000,i2=115,c5=138,K0=".midnight",t4="wallet.json",o2="config.json",A0=448,D0=384,n2="localnet",u1=1,e0="cache",e4="wallets",C4="default",x4=10,p1=3000,a2=120000,c1=9932,$1="https://midnight-comp-tracker.vercel.app";var M3={};k(M3,{unsetConfigValue:()=>Z$,setConfigValue:()=>$$,saveCliConfig:()=>v4,loadCliConfig:()=>E0,getValidConfigKeys:()=>i1,getConfigValue:()=>e2});import*as d0 from"fs";import*as r2 from"path";import{homedir as l5}from"os";function s2($){return d5[$]??$}function i5($){return/^(https?|wss?):\/\/\S+/.test($)}function _3($){return $??r2.join(l5(),K0)}function T3($){return r2.join(_3($),o2)}function o5($){let Z=_3($);if(!d0.existsSync(Z))d0.mkdirSync(Z,{recursive:!0,mode:A0})}function E0($){let Z=T3($);if(!d0.existsSync(Z))return{...Z1};let Q;try{Q=d0.readFileSync(Z,"utf-8")}catch{return{...Z1}}let X;try{X=JSON.parse(Q)}catch{return{...Z1}}let z={network:X.network&&F0(X.network)?X.network:Z1.network};if(X["proof-server"]&&typeof X["proof-server"]==="string")z["proof-server"]=X["proof-server"];if(X.node&&typeof X.node==="string")z.node=X.node;if(X["indexer-ws"]&&typeof X["indexer-ws"]==="string")z["indexer-ws"]=X["indexer-ws"];if(X.wallet&&typeof X.wallet==="string")z.wallet=X.wallet;return z}function v4($,Z){o5(Z);let Q=T3(Z);d0.writeFileSync(Q,JSON.stringify($,null,2)+`
|
|
7
|
+
`,{mode:D0})}function e2($,Z){let Q=s2($),X=E0(Z);if(Q==="network")return X.network;if(Q==="wallet")return X.wallet??"(not set)";if(t2.has(Q)){let z=X[Q];return typeof z==="string"?z:"(not set)"}throw new Error(`Unknown config key: "${$}"
|
|
8
|
+
Valid keys: ${d1.join(", ")}`)}function $$($,Z,Q){let X=s2($),z=E0(Q);if(X==="network"){if(!F0(Z))throw new Error(`Invalid network: "${Z}"
|
|
9
|
+
Valid networks: preprod, preview, undeployed`);z.network=Z}else if(X==="wallet"){if(!$4(Z))throw new Error(`Invalid wallet name: "${Z}"
|
|
10
|
+
Wallet name must be a simple name (no path separators, .json suffix, or special characters).`);z.wallet=Z}else if(t2.has(X)){if(!i5(Z))throw new Error(`Invalid URL for "${$}": "${Z}"
|
|
11
|
+
Must start with http://, https://, ws://, or wss://`);z[X]=Z}else throw new Error(`Unknown config key: "${$}"
|
|
12
|
+
Valid keys: ${d1.join(", ")}`);v4(z,Q)}function Z$($,Z){let Q=s2($),X=E0(Z);if(Q==="network")X.network=Z1.network;else if(Q==="wallet")delete X.wallet;else if(t2.has(Q))delete X[Q];else throw new Error(`Unknown config key: "${$}"
|
|
13
|
+
Valid keys: ${d1.join(", ")}`);v4(X,Z)}function i1(){return d1}var Z1,d1,d5,t2;var y4=N(()=>{y0();Z1={network:"undeployed"},d1=["network","proof-server","node","indexer-ws","wallet"],d5={"network-id":"network"};t2=new Set(["proof-server","node","indexer-ws"])});var D3={};k(D3,{resolveNetworkConfig:()=>Q$,isValidNetworkName:()=>F0,getValidNetworkNames:()=>b4,getNetworkConfig:()=>o1,detectTestcontainerPorts:()=>A3,detectNetworkFromAddress:()=>r5,applyEndpointOverrides:()=>k0});import{execSync as n5}from"child_process";function F0($){return R3.includes($)}function o1($){return{...a5[$]}}function b4(){return R3}function r5($){if($.startsWith("mn_addr_preprod1"))return"preprod";if($.startsWith("mn_addr_preview1"))return"preview";if($.startsWith("mn_addr_undeployed1"))return"undeployed";return null}function A3(){try{let $=n5('docker ps --format "{{.Image}}|{{.Ports}}"',{encoding:"utf-8",timeout:5000}),Z={};for(let Q of $.trim().split(`
|
|
14
|
+
`)){if(!Q)continue;let[X,z]=Q.split("|"),Y=(G)=>{let J=new RegExp(`0\\.0\\.0\\.0:(\\d+)->${G}/tcp`),K=z?.match(J);return K?parseInt(K[1],10):void 0};if(X.includes("indexer-standalone")||X.includes("indexer")){let G=Y(8088);if(G)Z.indexerPort=G}if(X.includes("midnight-node")){let G=Y(9944);if(G)Z.nodePort=G}if(X.includes("proof-server")){let G=Y(6300);if(G)Z.proofServerPort=G}}return Z}catch{return{}}}function Q$($){let Z=o1($);if($==="undeployed"){let Q=A3();if(Q.indexerPort)Z.indexer=`http://localhost:${Q.indexerPort}/api/v3/graphql`,Z.indexerWS=`ws://localhost:${Q.indexerPort}/api/v3/graphql/ws`;if(Q.nodePort)Z.node=`ws://localhost:${Q.nodePort}`;if(Q.proofServerPort)Z.proofServer=`http://localhost:${Q.proofServerPort}`}return Z}function k0($,Z,Q){let X=E0(Q);if($.proofServer=Z.proofServer??X["proof-server"]??$.proofServer,$.node=Z.node??X.node??$.node,$.indexerWS=Z.indexerWS??X["indexer-ws"]??$.indexerWS,Z.indexerWS??X["indexer-ws"]){let z=$.indexerWS;$.indexer=z.replace(/^wss:/,"https:").replace(/^ws:/,"http:").replace(/\/ws$/,"")}return $}var a5,R3;var y0=N(()=>{y4();a5={preprod:{indexer:"https://indexer.preprod.midnight.network/api/v3/graphql",indexerWS:"wss://indexer.preprod.midnight.network/api/v3/graphql/ws",node:"wss://rpc.preprod.midnight.network",proofServer:"http://localhost:6300",networkId:"PreProd"},preview:{indexer:"https://indexer.preview.midnight.network/api/v3/graphql",indexerWS:"wss://indexer.preview.midnight.network/api/v3/graphql/ws",node:"wss://rpc.preview.midnight.network",proofServer:"http://localhost:6300",networkId:"Preview"},undeployed:{indexer:"http://localhost:8088/api/v3/graphql",indexerWS:"ws://localhost:8088/api/v3/graphql/ws",node:"ws://localhost:9944",proofServer:"http://localhost:6300",networkId:"Undeployed"}},R3=["preprod","preview","undeployed"]});import{HDWallet as s5,Roles as X$}from"@midnight-ntwrk/wallet-sdk-hd";function z$($,Z){let Q=s5.fromSeed($);if(Q.type!=="seedOk")throw new Error("Invalid seed for HD wallet");let X=Q.hdWallet.selectAccount(0).selectRole(Z).deriveKeyAt(0);if(X.type==="keyOutOfBounds")throw new Error("Key derivation out of bounds");return X.key}function n1($){return z$($,X$.Zswap)}function E3($){return z$($,X$.NightExternal)}function J4($){return z$($,X$.Dust)}var Q1=()=>{};var x3={};k(x3,{deriveUnshieldedAddress:()=>N0,deriveShieldedAddress:()=>a1,deriveAllShieldedAddresses:()=>S4,deriveAllAddresses:()=>Z4});import{HDWallet as t5,Roles as e5}from"@midnight-ntwrk/wallet-sdk-hd";import{createKeystore as $6,PublicKey as Z6}from"@midnight-ntwrk/wallet-sdk-unshielded-wallet";import{MidnightBech32m as Q6,ShieldedAddress as X6,ShieldedCoinPublicKey as z6,ShieldedEncryptionPublicKey as Y6}from"@midnight-ntwrk/wallet-sdk-address-format";import{NetworkId as Y$}from"@midnight-ntwrk/wallet-sdk-abstractions";import*as N3 from"@midnight-ntwrk/ledger-v8";function N0($,Z,Q=0){let X=C3[Z],z=t5.fromSeed($);if(z.type!=="seedOk")throw new Error("Invalid seed for HD wallet");let Y=z.hdWallet.selectAccount(0).selectRole(e5.NightExternal).deriveKeyAt(Q);if(Y.type==="keyOutOfBounds")throw new Error(`Key index ${Q} out of bounds`);let G=$6(Y.key,X);return Z6.fromKeyStore(G).address}function Z4($,Z=0){let Q={};for(let X of b4())Q[X]=N0($,X,Z);return Q}function S4($){let Z=a1($),Q={};for(let X of b4()){let z=C3[X];Q[X]=Q6.encode(z,Z).asString()}return Q}function a1($){let Z=n1($),Q=N3.ZswapSecretKeys.fromSeed(Z);return new X6(new z6(Buffer.from(Q.coinPublicKey,"hex")),new Y6(Buffer.from(Q.encryptionPublicKey,"hex")))}var C3;var m0=N(()=>{y0();Q1();C3={preprod:Y$.NetworkId.PreProd,preview:Y$.NetworkId.Preview,undeployed:Y$.NetworkId.Undeployed}});var k4={};k(k4,{resolveNetworkName:()=>P0,resolveNetwork:()=>q0});function P0($){let Z=A($.args,"network");if(Z!==void 0){if(!F0(Z))throw new Error(`Invalid network: "${Z}"
|
|
15
|
+
Valid networks: ${b4().join(", ")}`);return Z}let Q=E0($.configDir);if(Q.network&&F0(Q.network))return Q.network;return"undeployed"}function q0($){let Z=P0($),Q=Q$(Z);return{name:Z,config:Q}}var V0=N(()=>{y0();y4()});var b3={};k(b3,{setActiveWallet:()=>r1,saveWalletConfig:()=>X1,saveShieldedAddress:()=>K4,resolveWalletPath:()=>Q0,removeWallet:()=>K$,migrateOldWallet:()=>V6,loadWalletConfig:()=>z0,listWallets:()=>q$,getAddress:()=>J6,getActiveWalletName:()=>V4});import*as c from"fs";import*as B0 from"path";import{homedir as G6}from"os";function K4($,Z,Q){let X=B0.resolve($);if(!c.existsSync(X))return;try{let z=JSON.parse(c.readFileSync(X,"utf-8")),Y=z.shieldedAddresses&&typeof z.shieldedAddresses==="object"?z.shieldedAddresses:{};if(Y[Z]=Q,z.shieldedAddresses=Y,z.shieldedAddress&&z.shieldedAddress===Q)delete z.shieldedAddress;c.writeFileSync(X,JSON.stringify(z,null,2)+`
|
|
16
|
+
`,{mode:D0})}catch{}}function J6($,Z){return $.addresses[Z]}function G$(){return B0.join(G6(),K0)}function q4(){return B0.join(G$(),e4)}function J$(){return B0.join(G$(),t4)}function y3(){let $=G$();if(!c.existsSync($))c.mkdirSync($,{recursive:!0,mode:A0})}function q6(){y3();let $=q4();if(!c.existsSync($))c.mkdirSync($,{recursive:!0,mode:A0})}function K6($){return $.includes("/")||$.includes("\\")||$.endsWith(".json")}function Q0($){if($!==void 0){if(K6($))return B0.resolve($);return B0.join(q4(),`${$}.json`)}let Q=E0().wallet??C4;return B0.join(q4(),`${Q}.json`)}function V4(){return E0().wallet??C4}function r1($){if(!$4($))throw new Error(`Invalid wallet name: "${$}"
|
|
17
|
+
Wallet name must be a simple name (no path separators, .json suffix, or special characters).`);let Z=B0.join(q4(),`${$}.json`);if(!c.existsSync(Z))throw new Error(`Wallet "${$}" not found.
|
|
18
|
+
Run "midnight wallet list" to see available wallets.`);let Q=E0();Q.wallet=$,v4(Q)}function q$(){let $=q4();if(!c.existsSync($))return[];let Z=V4();return c.readdirSync($).filter((X)=>X.endsWith(".json")).sort().map((X)=>{let z=X.replace(/\.json$/,""),Y=B0.join($,X);try{let G=JSON.parse(c.readFileSync(Y,"utf-8")),J;if(G.addresses)J=G.addresses;else if(G.address&&G.seed)try{let q=Buffer.from(G.seed,"hex");J=Z4(q)}catch{J={undeployed:G.address,preprod:"(unknown)",preview:"(unknown)"}}else J={undeployed:"(unknown)",preprod:"(unknown)",preview:"(unknown)"};let K;if(G.shieldedAddresses&&typeof G.shieldedAddresses==="object")K=G.shieldedAddresses;else if(G.seed)try{K=S4(Buffer.from(G.seed,"hex"))}catch{}return{name:z,addresses:J,shieldedAddresses:K,isActive:z===Z,createdAt:typeof G.createdAt==="string"?G.createdAt:void 0,file:Y}}catch{return{name:z,addresses:{undeployed:"(invalid)",preprod:"(invalid)",preview:"(invalid)"},isActive:z===Z,file:Y}}})}function K$($){if(!$4($))throw new Error(`Invalid wallet name: "${$}"
|
|
19
|
+
Wallet name must be a simple name (no path separators, .json suffix, or special characters).`);let Z=q4(),Q=B0.join(Z,`${$}.json`);if(!c.existsSync(Q))throw new Error(`Wallet "${$}" not found.
|
|
20
|
+
Run "midnight wallet list" to see available wallets.`);let X=V4();if($===X)throw new Error(`Cannot remove the active wallet "${$}".
|
|
21
|
+
Switch to another wallet first: midnight wallet use <other-wallet>`);if(c.readdirSync(Z).filter((Y)=>Y.endsWith(".json")).length<=1)throw new Error(`Cannot remove "${$}" — it is the only wallet.
|
|
22
|
+
`+"Create another wallet first: midnight wallet generate <name>");c.unlinkSync(Q)}function V6(){let $=J$();if(!c.existsSync($))return;let Z=q4();if(c.existsSync(Z)){if(c.readdirSync(Z).filter((Y)=>Y.endsWith(".json")).length>0)return}q6();let Q=B0.join(Z,`${C4}.json`);c.copyFileSync($,Q),c.chmodSync(Q,D0),c.unlinkSync($);let X=E0();X.wallet=C4,v4(X)}function z0($){let Z=$?B0.resolve($):J$();if(!c.existsSync(Z))throw new Error(`Wallet file not found: ${Z}
|
|
23
|
+
Generate a wallet first: midnight wallet generate <name> --network <name>`);let Q;try{Q=c.readFileSync(Z,"utf-8")}catch(K){throw new Error(`Failed to read wallet file: ${Z}
|
|
24
|
+
${K.message}`)}let X;try{X=JSON.parse(Q)}catch{throw new Error(`Invalid JSON in wallet file: ${Z}`)}if(!X.seed||!X.createdAt){let K=["seed","createdAt"];if(!X.addresses)K.push("address");let q=K.filter((V)=>!X[V]);if(q.length>0)throw new Error(`Wallet file is missing required fields (${q.join(", ")}): ${Z}`)}if(!/^[0-9a-fA-F]+$/.test(X.seed))throw new Error(`Invalid seed format in wallet file (expected hex string): ${Z}`);if(!X.addresses){if(!X.address)throw new Error(`Wallet file is missing required fields (address): ${Z}`);let K=Buffer.from(X.seed,"hex"),q=Z4(K),V={seed:X.seed,addresses:q,createdAt:X.createdAt};if(X.mnemonic)V.mnemonic=X.mnemonic;let B=v3(X,K);if(B)V.shieldedAddresses=B;let H={...X,addresses:q};if(B)H.shieldedAddresses=B,delete H.shieldedAddress;return c.writeFileSync(Z,JSON.stringify(H,null,2)+`
|
|
25
|
+
`,{mode:D0}),V}if(typeof X.addresses!=="object")throw new Error(`Wallet file has invalid addresses field: ${Z}`);let z={seed:X.seed,addresses:X.addresses,createdAt:X.createdAt};if(X.mnemonic)z.mnemonic=X.mnemonic;let Y=Buffer.from(X.seed,"hex"),G=v3(X,Y);if(G)z.shieldedAddresses=G;if(G&&(!X.shieldedAddresses||typeof X.shieldedAddresses!=="object"||Object.keys(X.shieldedAddresses).length<Object.keys(G).length))try{let K={...X,shieldedAddresses:G};delete K.shieldedAddress,c.writeFileSync(Z,JSON.stringify(K,null,2)+`
|
|
26
|
+
`,{mode:D0})}catch{}return z}function v3($,Z){let Q;try{Q=S4(Z)}catch{}let X=$.shieldedAddresses,z=X&&typeof X==="object"?X:null,Y=z?Object.fromEntries(Object.entries(z).filter(([,G])=>typeof G==="string"&&G.length>0)):{};if(!Q)return Object.keys(Y).length>0?Y:void 0;return{...Q,...Y}}function X1($,Z){let Q=Z?B0.resolve(Z):J$();if(!Z)y3();else{let X=B0.dirname(Q);if(!c.existsSync(X))c.mkdirSync(X,{recursive:!0,mode:A0})}return c.writeFileSync(Q,JSON.stringify($,null,2)+`
|
|
27
|
+
`,{mode:D0}),Q}var b0=N(()=>{y4();m0()});function w4(){return!("NO_COLOR"in process.env)}function z1($,Z){if(!w4())return $;return`\x1B[38;5;${Z}m${$}\x1B[0m`}function y($){if(!w4())return $;return`\x1B[1m${$}\x1B[0m`}function U($){if(!w4())return $;return`\x1B[2m${$}\x1B[0m`}function l($){return z1($,38)}function X0($){return z1($,196)}function m($){return z1($,40)}function e($){return z1($,226)}function Y1($){return z1($,245)}function g($,Z=k3){let Q=` ${$} `,X=Z-Q.length;if(X<=0)return y(Q);let z=Math.floor(X/2),Y=X-z;return y("═".repeat(z)+Q+"═".repeat(Y))}function d($=k3){return U("─".repeat($))}function M($,Z,Q=16){let X=($+":").padEnd(Q);return` ${Y1(X)}${Z}`}function B4($){let Z=$<0n,Q=Z?-$:$,X=BigInt(10**k1),z=Q/X,G=(Q%X).toString().padStart(k1,"0");return`${Z?"-":""}${z}.${G}`}function s1($){return`${B4($)} NIGHT`}function J1($){let Z=$<0n,Q=Z?-$:$,X=10n**BigInt(S3),z=Q/X,J=(Q%X).toString().padStart(S3,"0").replace(/0+$/,"").padEnd(6,"0");return`${Z?"-":""}${z}.${J}`}function t1($){return`${J1($)} DUST`}function s($,Z=!1){let Q=Z&&$.length>20?$.slice(0,10)+"…"+$.slice(-8):$;return l(Q)}function B6($,Z){if(G1($).length<=Z)return[$];let X=$.split(/(\s+)/),z=[],Y="",G=0;for(let J of X){let K=G1(J).length;if(G+K>Z&&G>0)z.push(Y),Y=J.trimStart(),G=G1(Y).length;else Y+=J,G+=K}if(Y.length>0)z.push(Y);return z}function w3($,Z="light",Q=70){let X=Z==="heavy"?{tl:"╔",tr:"╗",bl:"╚",br:"╝",h:"═",v:"║"}:{tl:"┌",tr:"┐",bl:"└",br:"┘",h:"─",v:"│"},z=Q-4,Y=[];for(let B of $){let H=B.split(`
|
|
28
|
+
`);for(let j of H)Y.push(...B6(j,z))}let G=Math.max(...Y.map((B)=>G1(B).length)),J=Math.max(G+2,20),K=X.tl+X.h.repeat(J)+X.tr,q=X.bl+X.h.repeat(J)+X.br,V=Y.map((B)=>{let H=G1(B).length,j=J-H-2;return`${X.v} ${B}${" ".repeat(Math.max(0,j))} ${X.v}`});return[K,...V,q].join(`
|
|
29
|
+
`)}function Q4($,Z){let X=[`${m("✓")} ${$}`];if(Z)X.push(M("Transaction",l(Z)));return X.join(`
|
|
30
|
+
`)}var k3=60,S3=15,G1=($)=>$.replace(/\x1b\[[0-9;]*m/g,"");var H0=()=>{};var u3={};k(u3,{default:()=>g3});import*as m3 from"fs";import*as e1 from"path";import{homedir as H6}from"os";import{generateMnemonic as W6,mnemonicToSeedSync as f3,validateMnemonic as U6}from"@scure/bip39";import{wordlist as h3}from"@scure/bip39/wordlists/english.js";async function g3($){let Z=P0({args:$}),Q=A($,"output"),X=A($,"seed"),z=A($,"mnemonic");if(X!==void 0&&z!==void 0)throw new Error("Cannot specify both --seed and --mnemonic. Use one or the other.");let Y=Q?e1.resolve(Q):e1.join(H6(),K0,t4);if(m3.existsSync(Y)&&!C($,"force"))throw new Error(`Wallet file already exists: ${Y}
|
|
31
|
+
Use --force to overwrite, or --output <file> to save to a different path.`);let G,J;if(X!==void 0){let H=X.replace(/^0x/,"");if(H.length!==64||!/^[0-9a-fA-F]+$/.test(H))throw new Error("Seed must be a 64-character hex string (32 bytes)");G=Buffer.from(H,"hex")}else if(z!==void 0){if(!U6(z,h3))throw new Error("Invalid BIP-39 mnemonic. Expected 12 or 24 words from the English wordlist.");J=z,G=Buffer.from(f3(J))}else J=W6(h3,256),G=Buffer.from(f3(J));let K=Z4(G),q=K[Z],V={seed:G.toString("hex"),addresses:K,createdAt:new Date().toISOString()};if(J)V.mnemonic=J;let B=X1(V,Q);if(C($,"json")){let H={addresses:K,activeAddress:q,activeNetwork:Z,seed:G.toString("hex"),file:B,createdAt:V.createdAt};if(J)H.mnemonic=J;x(H);return}if(process.stdout.write(q+`
|
|
25
32
|
`),process.stderr.write(`
|
|
26
|
-
`+
|
|
33
|
+
`+g("Wallet Generated")+`
|
|
27
34
|
|
|
28
|
-
`),process.stderr.write(
|
|
29
|
-
`),process.stderr.write(
|
|
30
|
-
`),process.stderr.write(
|
|
35
|
+
`),process.stderr.write(M("Network",Z)+`
|
|
36
|
+
`),process.stderr.write(M("Address",s(q))+`
|
|
37
|
+
`),process.stderr.write(M("File",B)+`
|
|
31
38
|
`),process.stderr.write(`
|
|
32
|
-
`),
|
|
33
|
-
`),process.stderr.write(` ${
|
|
39
|
+
`),J)process.stderr.write(e(y(" MNEMONIC (save securely!):"))+`
|
|
40
|
+
`),process.stderr.write(` ${J}
|
|
34
41
|
|
|
35
|
-
`);process.stderr.write(
|
|
36
|
-
`),process.stderr.write(` ${
|
|
42
|
+
`);process.stderr.write(e(y(" SEED (hex):"))+`
|
|
43
|
+
`),process.stderr.write(` ${G.toString("hex")}
|
|
37
44
|
|
|
38
|
-
`),process.stderr.write(
|
|
39
|
-
`),process.stderr.write(
|
|
45
|
+
`),process.stderr.write(d()+`
|
|
46
|
+
`),process.stderr.write(U(" Next: midnight info | midnight balance")+`
|
|
40
47
|
|
|
41
|
-
`),process.stderr.write(
|
|
42
|
-
`)}var
|
|
48
|
+
`),process.stderr.write(m("✓")+` Wallet saved
|
|
49
|
+
`)}var p3=N(()=>{m0();V0();b0();H0()});var l3={};k(l3,{default:()=>c3});async function c3($){let Z=Q0(A($,"wallet")),Q=z0(Z),X=P0({args:$}),z=Q.addresses[X];if(C($,"json")){let G={addresses:Q.addresses,activeNetwork:X,activeAddress:z,createdAt:Q.createdAt,file:Z};if(Q.shieldedAddresses)G.shieldedAddresses=Q.shieldedAddresses;x(G);return}process.stdout.write(z+`
|
|
43
50
|
`),process.stderr.write(`
|
|
44
|
-
`+
|
|
51
|
+
`+g("Wallet Info")+`
|
|
45
52
|
|
|
46
|
-
`);
|
|
53
|
+
`);let Y=Object.keys(Q.addresses);for(let G=0;G<Y.length;G++){let J=Y[G],K=J===X,q=Q.addresses[J],V=Q.shieldedAddresses?.[J],B=K?y(l(J))+U(" (active)"):J;if(process.stderr.write(` ${B}
|
|
54
|
+
`),process.stderr.write(` ${U("unshielded")} ${s(q)}
|
|
55
|
+
`),V)process.stderr.write(` ${U("shielded ")} ${s(V)}
|
|
56
|
+
`);else process.stderr.write(` ${U("shielded ")} ${U("(unavailable)")}
|
|
57
|
+
`);if(G<Y.length-1)process.stderr.write(`
|
|
47
58
|
`)}process.stderr.write(`
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
`),process.stderr.write(
|
|
51
|
-
`),process.stderr.write(
|
|
52
|
-
`),process.stderr.write(`
|
|
53
|
-
`+C()+`
|
|
59
|
+
`+d()+`
|
|
60
|
+
|
|
61
|
+
`),process.stderr.write(` ${U("created")} ${Q.createdAt}
|
|
62
|
+
`),process.stderr.write(` ${U("file ")} ${Z}
|
|
54
63
|
|
|
55
|
-
`)}var
|
|
64
|
+
`)}var d3=N(()=>{b0();V0();H0();H0()});function w0(){i3=!0}function w($,Z){if(!i3)return;let Q=new Date().toISOString().slice(11,23);process.stderr.write(U(` [${Q}] ${$}: ${Z}`)+`
|
|
65
|
+
`)}var i3=!1;var H4=()=>{};import{NetworkId as V$}from"@midnight-ntwrk/wallet-sdk-abstractions";function f4($){let Z=L6[$];if(Z===void 0)throw new Error(`Unknown networkId: ${$}`);return Z}var L6;var $2=N(()=>{L6={PreProd:V$.NetworkId.PreProd,Preview:V$.NetworkId.Preview,Undeployed:V$.NetworkId.Undeployed}});import j6 from"ws";function h4($,Z,Q){return new Promise((X,z)=>{let Y=new j6(Z,["graphql-transport-ws"]),G=new Map,J=0,K=0,q=0,V=!1,B=!1,H,j=()=>{let F=new Map,O=0,_=0,R=0;for(let T of G.values())if(!T.spent){O++;let v=F.get(T.tokenType)??0n;if(F.set(T.tokenType,v+T.value),T.tokenType===S1)if(T.registeredForDustGeneration)_++;else R++}return{balances:F,utxoCount:O,txCount:J,highestTxId:K,registeredUtxos:_,unregisteredUtxos:R}},L=()=>{clearTimeout(H)},W=()=>{if(!B&&V&&(K===0||q>=K))B=!0,L(),Y.send(JSON.stringify({id:"1",type:"complete"})),Y.close(),X(j())};Y.on("open",()=>{Y.send(JSON.stringify({type:"connection_init"}))}),Y.on("message",(F)=>{let O=JSON.parse(F.toString());switch(O.type){case"connection_ack":Y.send(JSON.stringify({id:"1",type:"subscribe",payload:{query:O6,variables:{address:$}}}));break;case"next":{if(O.payload?.errors){let R=O.payload.errors[0]?.message||"Unknown GraphQL error";if(!B)B=!0,L(),Y.close(),z(new Error(`GraphQL error: ${R}`));return}let _=O.payload?.data?.unshieldedTransactions;if(!_)return;if(_.__typename==="UnshieldedTransaction"){J++;let R=_;q=Math.max(q,R.transaction.id);for(let T of R.createdUtxos){let v=`${T.intentHash}:${T.outputIndex}`;G.set(v,{value:BigInt(T.value),tokenType:T.tokenType,spent:!1,registeredForDustGeneration:T.registeredForDustGeneration===!0})}for(let T of R.spentUtxos){let v=`${T.intentHash}:${T.outputIndex}`,b=G.get(v);if(b)b.spent=!0}if(Q)Q(q,K);W()}else if(_.__typename==="UnshieldedTransactionsProgress")K=_.highestTransactionId,V=!0,W();break}case"error":if(!B)B=!0,L(),Y.close(),z(new Error(`GraphQL subscription error: ${JSON.stringify(O.payload)}`));break;case"complete":break}}),Y.on("error",(F)=>{if(!B){B=!0,L();let O=F.message?.trim()?F.message:`unable to reach ${Z} (connection refused, DNS, or TLS error). Is the indexer running?`;z(new Error(`WebSocket connection failed: ${O}`))}}),Y.on("close",()=>{if(!B)B=!0,L(),z(new Error(`Indexer closed the connection before balance sync completed. Indexer: ${Z}`))}),H=setTimeout(()=>{if(!B)B=!0,Y.close(),z(new Error(`Balance check timed out after ${m1/1000}s. Indexer: ${Z}`))},m1)})}function q1($){return $===S1}var O6=`
|
|
56
66
|
subscription UnshieldedTransactions($address: UnshieldedAddress!) {
|
|
57
67
|
unshieldedTransactions(address: $address) {
|
|
58
68
|
__typename
|
|
59
69
|
... on UnshieldedTransaction {
|
|
60
70
|
transaction { id hash }
|
|
61
|
-
createdUtxos { value owner tokenType intentHash outputIndex }
|
|
62
|
-
spentUtxos { value owner tokenType intentHash outputIndex }
|
|
71
|
+
createdUtxos { value owner tokenType intentHash outputIndex registeredForDustGeneration }
|
|
72
|
+
spentUtxos { value owner tokenType intentHash outputIndex registeredForDustGeneration }
|
|
63
73
|
}
|
|
64
74
|
... on UnshieldedTransactionsProgress {
|
|
65
75
|
highestTransactionId
|
|
66
76
|
}
|
|
67
77
|
}
|
|
68
78
|
}
|
|
69
|
-
`;var
|
|
79
|
+
`;var K1=()=>{};import F6 from"ws";import*as m4 from"@midnight-ntwrk/ledger-v8";function M6(){let $=new m4.DustParameters(P6,I6,_6);return new m4.DustLocalState($)}function Z2($,Z,Q={}){let{onProgress:X,onCheckpoint:z,timeoutMs:Y=600000,idleMs:G=5000,initialSilenceMs:J=3000,signal:K,initialState:q,startFromId:V=0}=Q,B=500;return new Promise((H,j)=>{let L=new F6(Z,["graphql-transport-ws"]),W=q??M6(),F=[],O=0,_=V-1,R=-1,T=!1,v=!1,b,u,p,L0=()=>{if(clearTimeout(b),u)clearTimeout(u);if(p)clearTimeout(p);try{L.close()}catch{}K?.removeEventListener("abort",E)},f=()=>{if(F.length===0)return;if(W=W.replayEvents($,F),O+=F.length,F.length=0,z&&_>=0)try{z(W,_)}catch{}},o=()=>{if(u)clearTimeout(u);u=setTimeout(()=>{if(T&&!v)Y0()},G)},Y0=(I=!1)=>{if(v)return;v=!0,L0();try{f();let P=new Date;W=W.processTtls(P),H({balance:W.walletBalance(P),availableCoins:W.utxos.length,eventCount:O,ownedUtxoCount:W.utxos.length,syncTime:W.syncTime,state:W,lastAppliedEventId:_,partial:I})}catch(P){j(new Error(`Failed to build dust state: ${P.message}`))}},$0=(I)=>{if(v)return;v=!0,L0(),j(I)},E=()=>$0(new Error("Operation cancelled"));K?.addEventListener("abort",E,{once:!0}),L.on("open",()=>{L.send(JSON.stringify({type:"connection_init"}))}),L.on("message",(I)=>{let P;try{P=JSON.parse(I.toString())}catch{return}if(P.type==="connection_ack"){if(L.send(JSON.stringify({id:"1",type:"subscribe",payload:{query:T6,variables:{id:V}}})),q)p=setTimeout(()=>{if(!T&&!v)Y0()},J);return}if(P.type==="error"){$0(new Error(`GraphQL subscription error: ${JSON.stringify(P.payload)}`));return}if(P.type!=="next")return;if(P.payload?.errors){$0(new Error(`GraphQL error: ${P.payload.errors[0]?.message||"unknown"}`));return}let D=P.payload?.data?.dustLedgerEvents;if(!D)return;if(T=!0,p)clearTimeout(p),p=void 0;try{let h=Buffer.from(D.raw,"hex");F.push(m4.Event.deserialize(h))}catch(h){$0(new Error(`Failed to deserialize dust event ${D.id}: ${h.message}`));return}if(_=D.id,D.maxId>R)R=D.maxId;if(F.length>=500)try{f()}catch(h){$0(new Error(`Failed applying dust events: ${h.message}`));return}if(X?.(O+F.length,R),o(),_>=R)Y0()}),L.on("error",(I)=>$0(new Error(`WebSocket error: ${I.message}`))),L.on("close",()=>{if(v)return;if(!T)Y0();else $0(new Error("Indexer closed connection before dust sync completed"))}),b=setTimeout(()=>{Y0(!0)},Y)})}var P6=5000000000n,I6=8267n,_6,T6=`
|
|
80
|
+
subscription DustLedgerEvents($id: Int) {
|
|
81
|
+
dustLedgerEvents(id: $id) {
|
|
82
|
+
type: __typename
|
|
83
|
+
id
|
|
84
|
+
raw
|
|
85
|
+
maxId
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
`;var B$=N(()=>{_6=3n*60n*60n});import R6 from"ws";async function g4($,Z,Q=[]){let X=$.timeoutMs??A6;return new Promise((z,Y)=>{let G=new R6($.url),J=!1,K=(V)=>{if(!J)J=!0,V()},q=setTimeout(()=>{K(()=>{try{G.close()}catch{}Y(new Error(`RPC ${Z} timed out after ${X}ms`))})},X);G.on("open",()=>{try{G.send(JSON.stringify({jsonrpc:"2.0",id:1,method:Z,params:Q}))}catch(V){clearTimeout(q),K(()=>Y(V instanceof Error?V:new Error(String(V))))}}),G.on("message",(V)=>{clearTimeout(q),K(()=>{try{let B=JSON.parse(V.toString());try{G.close()}catch{}if(B.error)Y(new Error(`RPC ${Z}: ${B.error.message}`));else z(B.result)}catch(B){Y(B instanceof Error?B:new Error(String(B)))}})}),G.on("error",(V)=>{clearTimeout(q),K(()=>Y(V instanceof Error?V:new Error(String(V))))})})}var A6=5000;var Q2=()=>{};var W$={};k(W$,{getChainGenesisHash:()=>E6,clearChainIdMemo:()=>N6});async function E6($){let Z=H$.get($);if(Z&&Date.now()-Z.fetchedAt<D6)return Z.hash;try{let Q=await g4({url:$,timeoutMs:5000},"chain_getBlockHash",[0]);if(typeof Q!=="string"||!Q.startsWith("0x"))return null;return H$.set($,{hash:Q,fetchedAt:Date.now()}),Q}catch{return null}}function N6(){H$.clear()}var H$,D6=60000;var U$=N(()=>{Q2();H$=new Map});var r3={};k(r3,{validateDustCacheChainId:()=>O$,saveDustCache:()=>u4,primeDustCache:()=>S6,loadDustCache:()=>L4,getDustCachePath:()=>z2,dustPublicKeyHexFromSeed:()=>U4,dustPublicKeyHex:()=>H1,clearDustDirectCache:()=>j4});import{existsSync as V1,mkdirSync as C6,writeFileSync as x6,readFileSync as o3,unlinkSync as X2,readdirSync as L$,renameSync as v6}from"node:fs";import{join as W4,dirname as y6}from"node:path";import{homedir as n3}from"node:os";import{randomBytes as b6}from"node:crypto";import*as B1 from"@midnight-ntwrk/ledger-v8";function H1($){return $.toString(16).padStart(64,"0")}function U4($){let Z=J4($),Q=B1.DustSecretKey.fromSeed(Z);return H1(Q.publicKey)}function j$($,Z){let Q=Z??W4(n3(),K0,e0);return W4(Q,$)}function z2($,Z,Q){let X=Z.slice(0,20);return W4(j$($,Q),`dust-${X}.json`)}function L4($,Z,Q){let X=z2($,Z,Q);if(!V1(X))return null;try{let z=JSON.parse(o3(X,"utf-8"));if(z.version!==a3)return null;if(z.network!==$)return null;if(z.dustPublicKeyHex!==Z)return null;if(typeof z.lastAppliedEventId!=="number")return null;if(typeof z.dustState!=="string")return null;let Y=Buffer.from(z.dustState,"hex");return{state:B1.DustLocalState.deserialize(Y),lastAppliedEventId:z.lastAppliedEventId}}catch{return null}}function u4($,Z,Q,X,z,Y){let G={version:a3,network:$,dustPublicKeyHex:Z,lastAppliedEventId:X,timestamp:new Date().toISOString(),dustState:Buffer.from(Q.serialize()).toString("hex"),...Y?{chainId:Y}:{}},J=z2($,Z,z),K=y6(J);if(!V1(K))C6(K,{recursive:!0,mode:A0});let q=J+`.tmp.${b6(4).toString("hex")}`;try{x6(q,JSON.stringify(G),{mode:D0}),v6(q,J)}catch(V){try{X2(q)}catch{}throw V}}async function O$($,Z,Q){let{getChainGenesisHash:X}=await Promise.resolve().then(() => (U$(),W$)),z=await X(Z);if(!z)return[];let Y=j$($,Q);if(!V1(Y))return[];let G=[],J;try{J=L$(Y)}catch{return[]}for(let K of J){if(!K.startsWith("dust-")||!K.endsWith(".json"))continue;let q=W4(Y,K);try{let V=JSON.parse(o3(q,"utf-8"));if(!V.chainId)continue;if(V.chainId!==z)X2(q),G.push(q)}catch{}}return G}function j4($,Z,Q){let X=Q??W4(n3(),K0,e0);if($&&Z){let Y=z2($,Z,Q);try{X2(Y)}catch{}return}let z=(Y)=>{if(!V1(Y))return;try{for(let G of L$(Y))if(G.startsWith("dust-")&&G.endsWith(".json"))try{X2(W4(Y,G))}catch{}}catch{}};if($){z(j$($,Q));return}if(!V1(X))return;try{for(let Y of L$(X))z(W4(X,Y))}catch{}}async function S6($,Z,Q,X={}){let z=J4($),Y=B1.DustSecretKey.fromSeed(z),G=H1(Y.publicKey),J=L4(Z,G),K=J?J.lastAppliedEventId+1:0,q=await Z2(Y,Q,{initialState:J?.state,startFromId:K,onProgress:X.onProgress,signal:X.signal});if(q.lastAppliedEventId>=0)u4(Z,G,q.state,q.lastAppliedEventId);else if(!J)u4(Z,G,q.state,-1);return{eventCount:q.eventCount,lastAppliedEventId:q.lastAppliedEventId>=0?q.lastAppliedEventId:J?.lastAppliedEventId??-1,fromCache:J!==null}}var a3=1;var W1=N(()=>{Q1();B$()});import{ShieldedWallet as s3}from"@midnight-ntwrk/wallet-sdk-shielded";import{UnshieldedWallet as t3,createKeystore as k6,PublicKey as w6}from"@midnight-ntwrk/wallet-sdk-unshielded-wallet";import{DustWallet as e3}from"@midnight-ntwrk/wallet-sdk-dust-wallet";import{WalletFacade as $8}from"@midnight-ntwrk/wallet-sdk-facade";import*as p4 from"@midnight-ntwrk/ledger-v8";import{NetworkId as F$,InMemoryTransactionHistoryStorage as f6,TransactionHistoryStorage as h6}from"@midnight-ntwrk/wallet-sdk-abstractions";import*as I0 from"rxjs";async function X4($,Z,Q){let X=m6[Z.networkId];if(X===void 0)throw new Error(`Unknown networkId: ${Z.networkId}`);w("facade",`Building facade for network ${Z.networkId}`),w("facade",`Node: ${Z.node}`),w("facade",`Indexer: ${Z.indexerWS}`),w("facade",`Proof server: ${Z.proofServer}`);let z=n1($),Y=E3($),G=J4($),J=p4.ZswapSecretKeys.fromSeed(z),K=p4.DustSecretKey.fromSeed(G),q=k6(Y,X),V={networkId:X,indexerClientConnection:{indexerHttpUrl:Z.indexer,indexerWsUrl:Z.indexerWS},costParameters:{additionalFeeOverhead:g2,feeBlocksMargin:u2},txHistoryStorage:new f6(h6.TransactionHistoryCommonSchema),provingServerUrl:new URL(Z.proofServer),relayURL:new URL(Z.node)},B=()=>$8.init({configuration:V,shielded:(W)=>s3(W).startWithSecretKeys(J),unshielded:(W)=>t3(W).startWithPublicKey(w6.fromKeyStore(q)),dust:(W)=>e3(W).startWithSecretKey(K,p4.LedgerParameters.initialParameters().dust)}),H=Q?g6(Q,Z,K.publicKey):null,j=!1,L;if(H){w("facade","Restoring from cache...");try{L=await $8.init({configuration:V,shielded:(W)=>s3(W).restore(H.shielded),unshielded:(W)=>t3(W).restore(H.unshielded),dust:(W)=>e3(W).restore(H.dust)}),j=!0,w("facade","Cache restore successful")}catch(W){w("facade",`Cache restore failed: ${W.message}`),process.stderr.write(` Cache restore failed, building from scratch: ${W.message}
|
|
89
|
+
`),L=await B()}}else w("facade","No cache, building fresh"),L=await B();return{facade:L,keystore:q,zswapSecretKeys:J,dustSecretKey:K,restoredFromCache:j}}function g6($,Z,Q){try{let X=Z.networkId.toLowerCase(),z=H1(Q),Y=L4(X,z);if(!Y)return $;let G=JSON.parse($.dust),J=G.offset!==void 0?Number(G.offset):-1;if(J>=Y.lastAppliedEventId)return w("facade",`Facade dust offset=${J} >= dust-direct offset=${Y.lastAppliedEventId}; skipping bridge`),$;return G.state=Buffer.from(Y.state.serialize()).toString("hex"),G.offset=Y.lastAppliedEventId.toString(),w("facade",`Bridged dust-direct cache (facade offset ${J} → dust-direct offset ${Y.lastAppliedEventId})`),{...$,dust:JSON.stringify(G)}}catch(X){return w("facade",`Dust-direct bridge skipped: ${X.message}`),$}}function P$($,Z="full"){let Q=$.unshielded?.progress?.isStrictlyComplete()??!1,X=Z!=="no-dust",z=!X;if(X){if(z=$.dust?.state?.progress?.isStrictlyComplete()??!1,!z)try{let G=$.dust?.state?.progress;if(G&&G.highestRelevantWalletIndex>0&&G.appliedIndex>=G.highestRelevantWalletIndex)z=!0;if(G&&G.highestRelevantWalletIndex===0&&G.appliedIndex===0&&Q)z=!0}catch{}}if(Z==="lite")return Q&&z;let Y=$.shielded?.state?.progress?.isStrictlyComplete()??!1;if(!Y&&Q)try{let G=$.shielded?.state?.progress;if(G&&G.highestRelevantWalletIndex===0&&G.appliedIndex===0)Y=!0}catch{}return Y&&Q&&z}function u6($){if($.dust?.state?.progress?.isStrictlyComplete())return!1;try{let Z=$.dust?.state?.progress;if(Z&&Z.highestRelevantWalletIndex>0&&Z.appliedIndex>=Z.highestRelevantWalletIndex)return!1;if(Z&&Z.highestRelevantWalletIndex===0&&Z.appliedIndex===0&&$.unshielded?.progress?.isStrictlyComplete())return!1}catch{}return!0}async function z4($,Z={}){let{onProgress:Q,onSyncDetail:X,timeoutMs:z,syncMode:Y="full",requireStrictSync:G=!1}=Z,{facade:J,zswapSecretKeys:K,dustSecretKey:q}=$;w("sync","Starting facade (connecting to node and indexer)..."),await J.start(K,q),w("sync","Facade started, subscribing to state...");let V=z??p2;w("sync",`Sync timeout: ${V/1000}s, mode: ${Y}`);let B=!G&&$.restoredFromCache,H=Date.now();return new Promise((j,L)=>{let W=!1,F=0,O="",_=null,R=setTimeout(()=>{if(!W){if(w("sync",`Sync timed out after ${V/1000}s (${F} emissions)`),_)try{let T=_.unshielded?.progress;w("sync",` unshielded: applied=${T?.appliedId} highest=${T?.highestTransactionId} complete=${T?.isStrictlyComplete()}`);let v=_.dust?.state?.progress;if(w("sync",` dust: applied=${v?.appliedIndex} highest=${v?.highestRelevantWalletIndex} complete=${v?.isStrictlyComplete?.()} connected=${v?.isConnected}`),Y==="full"){let b=_.shielded?.state?.progress;w("sync",` shielded: complete=${b?.isStrictlyComplete()}`)}}catch{}L(new Error("Wallet sync timed out"))}},V);$.keepAlive=J.state().subscribe({next:(T)=>{if(W)return;if(F++,_=T,$.restoredFromCache){let b=c6(T);if(b){W=!0,clearTimeout(R),w("sync",`Stale cache detected: ${b}`),L(new Y2(b));return}}if(Q){let b=T.unshielded.progress;if(b){let u=Number(b.appliedId),p=Number(b.highestTransactionId);Q(Math.min(u,p),p)}}let v=[];try{if((Y==="full"||Y==="no-dust")&&!T.shielded?.state?.progress?.isStrictlyComplete())v.push("shielded");if(Y!=="no-dust"&&u6(T))v.push("dust");if(!T.unshielded?.progress?.isStrictlyComplete())v.push("unshielded")}catch{}if(v.length>0){X?.(v.join(", "));let b=v.join(",");if(F===1||b!==O||F%100===0)w("sync",`Waiting on: ${v.join(", ")} (emission #${F})`),O=b}if(P$(T,Y)){W=!0,clearTimeout(R),w("sync",`Sync complete after ${F} emissions`),j(T);return}if(B){let b=Date.now()-H;if(b>=p6&&P$(T,"no-dust"))W=!0,clearTimeout(R),w("sync",`Sync resolved via cached-restore grace (${b}ms, ${F} emissions)`),j(T)}},error:(T)=>{if(!W)w("sync",`Sync error: ${T.message}`),clearTimeout(R),L(T)}})})}function c6($){try{let Z=$.unshielded?.progress;if(!Z)return;let Q=Number(Z.appliedId??0),X=Number(Z.highestTransactionId??0);if(X>0&&Q>X)return`unshielded cache applied=${Q} but chain highest=${X}.`}catch{}return}async function Z8($){let Z=(Q)=>{let X=Q.unshielded?.progress?.isStrictlyComplete()??!1,z=Q.dust?.state?.progress?.isStrictlyComplete()??!1;return X&&z};try{return await I0.firstValueFrom($.facade.state().pipe(I0.filter(Z),I0.timeout(15000)))}catch{return await I0.firstValueFrom($.facade.state())}}async function G2($,Z=60000){let Q=(X)=>{try{let z=X.dust;return z?.availableCoins?.length>0||z?.balance(new Date)>0n}catch{return!1}};try{return await I0.firstValueFrom($.facade.state().pipe(I0.filter(Q),I0.timeout(Z)))}catch{return await I0.firstValueFrom($.facade.state())}}async function I$($,Z="full"){return I0.firstValueFrom($.facade.state().pipe(I0.filter((Q)=>P$(Q,Z)),I0.timeout(c2)))}async function g0($){$.keepAlive?.unsubscribe(),await Promise.race([$.facade.stop(),new Promise((Z)=>setTimeout(Z,5000))])}function j0($){let Z=(X)=>{let z=X?._tag;if(typeof z==="string"&&z.startsWith("Wallet.")){let Y=X?.message??"transient error";$?.(z,Y);return}Q("Unhandled rejection:",X),process.exit(1)},Q=console.error;return console.error=(...X)=>{let z=X[0];if(typeof z==="object"&&z?._tag?.startsWith("Wallet.")){$?.(z._tag,z?.message??"transient error");return}if(typeof z==="string"&&z.startsWith("Wallet.")){$?.("Wallet.Sync","transient error");return}Q(...X)},process.on("unhandledRejection",Z),()=>{process.removeListener("unhandledRejection",Z),console.error=Q}}var m6,p6=1e4,Y2;var i0=N(()=>{Q1();W1();H4();m6={PreProd:F$.NetworkId.PreProd,Preview:F$.NetworkId.Preview,Undeployed:F$.NetworkId.Undeployed};Y2=class Y2 extends Error{code="STALE_CACHE";constructor($){super(`Cached wallet state is stale (from a previous chain). ${$}
|
|
90
|
+
Run: midnight cache clear --wallet <name> --network <name>
|
|
91
|
+
Or: midnight cache clear (wipe all caches)`);this.name="StaleCacheError"}}});function l6($){let Z=$;while(Z){let Q=String(Z?.message??"").toLowerCase();if(Q.includes("submission error"))return!0;if(Q.includes("transaction")&&Q.includes("invalid"))return!0;if(Q.includes("138"))return!0;let X=Z?._tag;if(X==="TransactionInvalidError"||X==="SubmissionError")return!0;Z=Z.cause}return!1}function J2($){let Z=$?.message?.toLowerCase()??"";return Z.includes("not enough dust")||Z.includes("dust generated")||Z.includes("insufficient funds")||Z.includes("no dust tokens")||l6($)}function U1($){let Z=$?.message?.toLowerCase()??"";if($?._tag==="Wallet.InsufficientFunds")return!0;return Z==="insufficient funds"||Z.startsWith("insufficient funds")}var X8={};k(X8,{validateWalletCacheChainId:()=>T$,saveWalletCache:()=>p0,loadWalletCache:()=>o0,getCachePath:()=>K2,clearWalletCache:()=>c0});import{existsSync as j1,mkdirSync as d6,writeFileSync as i6,readFileSync as Q8,unlinkSync as L1,readdirSync as q2,renameSync as o6}from"node:fs";import{join as u0,dirname as n6}from"node:path";import{homedir as _$}from"node:os";import{randomBytes as a6}from"node:crypto";function K2($,Z,Q){let X=Q??u0(_$(),K0,e0),z=$.slice(0,20);return u0(X,Z,`${z}.json`)}function o0($,Z,Q){let X=K2($,Z,Q);if(!j1(X))return null;try{let z=Q8(X,"utf-8"),Y=JSON.parse(z);if(Y.version!==u1)return null;if(Y.network!==Z)return null;if(Y.address!==$)return null;if(!Y.wallets||typeof Y.wallets.shielded!=="string"||typeof Y.wallets.unshielded!=="string"||typeof Y.wallets.dust!=="string")return null;return Y.wallets}catch{return null}}async function p0($,Z,Q,X,z){let[Y,G,J]=await Promise.all([Q.shielded.serializeState(),Q.unshielded.serializeState(),Q.dust.serializeState()]),K={version:u1,network:Z,address:$,timestamp:new Date().toISOString(),wallets:{shielded:Y,unshielded:G,dust:J},...z?{chainId:z}:{}},q=K2($,Z,X),V=n6(q);if(!j1(V))d6(V,{recursive:!0,mode:A0});let B=q+`.tmp.${a6(4).toString("hex")}`;try{i6(B,JSON.stringify(K),{mode:D0}),o6(B,q)}catch(H){try{L1(B)}catch{}throw H}}async function T$($,Z,Q){let{getChainGenesisHash:X}=await Promise.resolve().then(() => (U$(),W$)),z=await X(Z);if(!z)return[];let Y=Q??u0(_$(),K0,e0),G=u0(Y,$);if(!j1(G))return[];let J=[],K;try{K=q2(G)}catch{return[]}for(let q of K){if(!q.endsWith(".json"))continue;let V=u0(G,q);try{let B=JSON.parse(Q8(V,"utf-8"));if(!B.chainId)continue;if(B.chainId!==z)L1(V),J.push(V)}catch{}}return J}function c0($,Z,Q){let X=Q??u0(_$(),K0,e0);if($&&Z){let z=K2($,Z,Q);try{L1(z)}catch{}return}if(Z){let z=u0(X,Z);if(!j1(z))return;try{for(let Y of q2(z))if(Y.endsWith(".json"))L1(u0(z,Y))}catch{}return}if(!j1(X))return;try{for(let z of q2(X)){let Y=u0(X,z);try{for(let G of q2(Y))if(G.endsWith(".json"))L1(u0(Y,G))}catch{}}}catch{}}var O4=()=>{};var G8={};k(G8,{setDefaultRepository:()=>zQ,defaultRepository:()=>_0,WalletDataRepository:()=>M$});import*as Y8 from"@midnight-ntwrk/ledger-v8";class M${now;fetchTip;fetchUnshielded;fetchDust;cacheDir;dustMemo=new Map;unshieldedMemo=new Map;tipMemo=new Map;constructor($={}){this.now=$.now??Date.now,this.fetchTip=$.fetchTip??ZQ,this.fetchUnshielded=$.fetchUnshielded??QQ,this.fetchDust=$.fetchDust??XQ,this.cacheDir=$.cacheDir}async dust($,Z,Q={}){let X=V2(Z),z=U4($),Y=`${X}:${z}`;if(!Q.forceFresh){let H=await this.tryMemo(Y,Z,this.dustMemo,Q.signal);if(H)return{...H,fromCache:!0,eventsApplied:0}}await O$(X,z,Z.node);let G=Q.forceFresh?null:L4(X,z,this.cacheDir),J=G!==null,K=0,q=null;for(let H=0;H<s6;H++){if(Q.signal?.aborted)throw new Error("Operation cancelled");let j=G?G.lastAppliedEventId+1:0;if(Q.onStatus?.(H>0?`Resuming dust from event ${j} (continuation ${H+1})…`:G?`Resuming dust from event ${j}…`:"Reading dust events…"),q=await this.fetchDust($,Z,{startFromId:j,initialState:G?.state,signal:Q.signal,onProgress:(L,W)=>{let F=Math.max(1,W+1-j);Q.onStatus?.(`Reading dust events… ${L}/${F}`)},onCheckpoint:(L,W)=>{try{u4(X,z,L,W,this.cacheDir)}catch{}}}),q.lastAppliedEventId>=0||!G){let L=q.lastAppliedEventId>=0?q.lastAppliedEventId:G?.lastAppliedEventId??-1;try{u4(X,z,q.state,L,this.cacheDir)}catch{}}if(K+=q.eventCount,!q.partial)break;G=L4(X,z,this.cacheDir)}if(!q)throw new Error("Dust sync exhausted retries without producing a result");let V={state:q.state,balance:q.balance,availableCoins:q.availableCoins,ownedUtxoCount:q.ownedUtxoCount,syncTime:q.syncTime,fromCache:J,eventsApplied:K,fetchedAt:this.now()},B=await this.getTip(Z,Q.signal);return this.dustMemo.set(Y,{value:V,fetchedAt:V.fetchedAt,tipAtFetch:B}),V}async unshielded($,Z,Q={}){let X=V2(Z),z=typeof $==="string"?$:N0($,X),Y=`${X}:${z}`;if(!Q.forceFresh){let q=await this.tryMemo(Y,Z,this.unshieldedMemo,Q.signal);if(q)return{...q,fromCache:!0}}let J={...await this.fetchUnshielded(z,Z,Q.onProgress),fromCache:!1,fetchedAt:this.now()},K=await this.getTip(Z,Q.signal);return this.unshieldedMemo.set(Y,{value:J,fetchedAt:J.fetchedAt,tipAtFetch:K}),J}async withFacade($,Z,Q,X={}){let z=X.syncMode??"full",Y=X.requireStrictSync??!0,G=Y&&!X.readOnly,J=V2(Z),K=N0($,J),q=Z.networkId!=="Undeployed",V=X.syncTimeoutMs??(q?e6:t6);if(await T$(K,J,Z.node),G&&!X.noPrime&&!X.forceFresh)try{await this.dust($,Z,{signal:X.signal,onStatus:X.onStatus})}catch(W){X.onSyncWarning?.("PrePrime",W.message)}let B,H,j=!1,L=async()=>{if(j||!B)return;j=!0;try{await g0(B)}catch{}};try{for(let O=1;O<=O1;O++){if(X.signal?.aborted)throw new Error("Operation cancelled");let _=X.forceFresh?null:o0(K,J,this.cacheDir);B=await X4($,Z,_);try{H=await z4(B,{onProgress:X.onSyncProgress,onSyncDetail:X.onSyncDetail,timeoutMs:V,syncMode:z,requireStrictSync:Y});break}catch(R){if(X.signal?.aborted)throw new Error("Operation cancelled");if(R instanceof Y2&&O<O1&&G){X.onStatus?.(`Cache is stale, clearing and rebuilding (attempt ${O+1}/${O1})...`),await g0(B).catch(()=>{}),B=void 0,c0(K,J,this.cacheDir),j4(J,U4($),this.cacheDir);try{await this.dust($,Z,{signal:X.signal,onStatus:X.onStatus})}catch{}continue}if(O<O1&&String(R?.message??"").includes("timed out")){try{await p0(K,J,B.facade,this.cacheDir)}catch{}X.onStatus?.(`Sync timed out, retrying (attempt ${O+1}/${O1})...`),await g0(B).catch(()=>{}),B=void 0;continue}throw R}}if(!B||!H)throw new Error("Sync failed: no facade state after retries");let W,F=0;while(!0){if(F++,X.signal?.aborted)throw new Error("Operation cancelled");try{W=await Q({bundle:B,state:H});break}catch(O){if(!(G&&F<z8&&U1(O)))throw O;if(X.onStatus?.(`Refreshing wallet state (attempt ${F+1}/${z8})...`),await g0(B).catch(()=>{}),B=void 0,await new Promise((T)=>setTimeout(T,$Q)),X.signal?.aborted)throw new Error("Operation cancelled");let R=o0(K,J,this.cacheDir);B=await X4($,Z,R),H=await z4(B,{onProgress:X.onSyncProgress,onSyncDetail:X.onSyncDetail,timeoutMs:V,syncMode:z,requireStrictSync:Y})}}if(!X.skipAutoSave)try{await p0(K,J,B.facade,this.cacheDir)}catch{}if(!X.readOnly)this.invalidate({network:Z,seed:$});return W}finally{await L()}}invalidate($){let Z=V2($.network),Q=$.kinds??["dust","unshielded","facade"];if(Q.includes("dust"))if($.seed){let X=U4($.seed);this.dustMemo.delete(`${Z}:${X}`)}else this.deleteByPrefix(this.dustMemo,`${Z}:`);if(Q.includes("unshielded")){let X=$.address??($.seed?N0($.seed,Z):null);if(X)this.unshieldedMemo.delete(`${Z}:${X}`);else this.deleteByPrefix(this.unshieldedMemo,`${Z}:`)}if(Q.includes("facade")&&$.seed){let X=N0($.seed,Z);try{c0(X,Z,this.cacheDir)}catch{}}}resetTipMemo(){this.tipMemo.clear()}async tryMemo($,Z,Q,X){let z=Q.get($);if(!z)return null;let Y;try{Y=await this.getTip(Z,X)}catch{return z.value}if(Y===z.tipAtFetch)return z.value;return Q.delete($),null}async getTip($,Z){let Q=this.tipMemo.get($.networkId);if(Q&&this.now()-Q.fetchedAt<r6)return Q.tip;let X=await this.fetchTip($,Z);return this.tipMemo.set($.networkId,{tip:X,fetchedAt:this.now()}),X}deleteByPrefix($,Z){for(let Q of $.keys())if(Q.startsWith(Z))$.delete(Q)}}function ZQ($){return g4({url:$.node,timeoutMs:3000},"chain_getBlockHash",[])}function QQ($,Z,Q){return h4($,Z.indexerWS,Q)}function XQ($,Z,Q){let X=J4($),z=Y8.DustSecretKey.fromSeed(X);return Z2(z,Z.indexerWS,Q)}function V2($){let Z=$.networkId.toLowerCase();if(F0(Z))return Z;throw new Error(`Unsupported network: ${$.networkId}`)}function _0(){if(!B2)B2=new M$;return B2}function zQ($){B2=$}var r6=5000,s6=6,t6=30000,e6=120000,O1=3,z8=5,$Q=5000,B2=null;var F4=N(()=>{y0();i0();W1();O4();K1();B$();Q2();Q1();m0()});function J8($={}){let Z=Math.min(Math.max($.windowMs??15000,1000),60000),Q=[],X=null,z=(G)=>{let J=G-Z;while(Q.length>0&&Q[0].t<J)Q.shift()},Y=()=>{let G=Q[Q.length-1],J=Q[0],K=G.highest>0?Math.min(100,Math.round(G.applied/G.highest*100)):null,q=null,V=null;if(Q.length>=2){let B=G.applied-J.applied,H=G.t-J.t;if(H>=1000&&B>0){q=B/(H/1000);let j=Math.max(0,G.highest-G.applied);V=j>0?j/q:0}}return{applied:G.applied,highest:G.highest,percent:K,ratePerSec:q,etaSec:V}};return{sample(G){return Q.push(G),z(G.t),X=Y(),X},snapshot(){return X},reset(){Q.length=0,X=null}}}function q8($,Z="Syncing"){let Q=[Z];if($.percent!==null)Q.push(`${$.percent}%`);else if($.highest===0)Q.push(`${$.applied} events`);if($.etaSec!==null&&$.etaSec>0)Q.push(`ETA ${YQ($.etaSec)}`);if($.ratePerSec!==null&&$.ratePerSec>0)Q.push(`${Math.round($.ratePerSec)} evt/s`);return Q.join(" · ")}function YQ($){if(!Number.isFinite($)||$<0)return"?";if($<60)return`${Math.ceil($)}s`;let Z=Math.floor($/60),Q=Math.round($-Z*60);if(Z<60)return Q>0?`${Z}m ${Q}s`:`${Z}m`;let X=Math.floor(Z/60),z=Z-X*60;return z>0?`${X}h ${z}m`:`${X}h`}import*as D$ from"@midnight-ntwrk/ledger-v8";import{MidnightBech32m as GQ,UnshieldedAddress as JQ}from"@midnight-ntwrk/wallet-sdk-address-format";import{NetworkId as R$}from"@midnight-ntwrk/wallet-sdk-abstractions";function F1($){if($<=0)throw new Error("Amount must be greater than 0");if(!Number.isFinite($))throw new Error("Amount must be a finite number");let Z=$.toFixed(6),[Q,X]=Z.split("."),z=Q+(X??"").padEnd(6,"0"),Y=BigInt(z);if(Y<=0n)throw new Error("Amount too small — minimum is 0.000001 NIGHT");return Y}function P1($){let Z=Number($);if(Number.isNaN(Z)||!Number.isFinite(Z))throw new Error(`Invalid amount: "${$}" — must be a positive number`);if(Z<=0)throw new Error(`Invalid amount: "${$}" — must be greater than 0`);return Z}function KQ($,Z){let Q=qQ[Z.networkId];if(Q===void 0)throw new Error(`Unknown networkId: ${Z.networkId}`);try{return GQ.parse($).decode(JQ,Q)}catch(X){throw new Error(`Invalid recipient address: ${X.message}
|
|
92
|
+
Expected a bech32m address for network "${Z.networkId}"`)}}function H2($){let Z=$<0n?-$:$,Q=Z/1000000000000000n,X=Z%1000000000000000n;return`${$<0n?"-":""}${Q}.${X.toString().padStart(15,"0").slice(0,6)}`}function A$($){let Z=Math.round($/1000);if(Z<60)return`${Z}s`;let Q=Math.floor(Z/60),X=Z%60;return`${Q}m ${X}s`}function T0(){let{warn:$,error:Z}=console,Q=(X)=>X.some((z)=>String(z).includes("RPC-CORE"));return console.warn=(...X)=>{if(!Q(X))$(...X)},console.error=(...X)=>{if(!Q(X))Z(...X)},()=>{console.warn=$,console.error=Z}}async function VQ($,Z,Q){let X=new Date(Date.now()+N4*60*1000);await Promise.race([$.facade.dust.waitForSyncedState(),new Promise((q,V)=>setTimeout(()=>V(new Error("Insufficient funds: dust wallet sync timed out")),f1))]);let z=await $.facade.dust.createDustGenerationTransaction(new Date,X,Z,$.keystore.getPublicKey(),Q),Y=z.intents?.get(1);if(!Y)throw new Error("Dust generation intent not found on transaction");let G=$.keystore.signData(Y.signatureData(1)),J=await $.facade.dust.addDustGenerationSignature(z,G),K=await $.facade.finalizeTransaction(J);return await $.facade.submitTransaction(K)}async function BQ($,Z,Q,X){let z=Date.now(),Y=z+d2,G,J=!1,K=setInterval(()=>{if(J&&X){let q=A$(Date.now()-z);X(`Waiting for dust generation capacity (${q} elapsed, ~5 min on fresh wallets)...`)}},1000);try{while(Date.now()<Y)try{return await VQ($,Z,Q)}catch(q){if(G=q,J2(q)&&Date.now()+g1<Y){J=!0;let V=A$(Date.now()-z);X?.(`Waiting for dust generation capacity (${V} elapsed, ~5 min on fresh wallets)...`),await new Promise((B)=>setTimeout(B,g1));continue}throw q}throw G??new Error("Dust registration timed out")}finally{clearInterval(K)}}async function l0($,Z,Q){let X=Q??await $.facade.waitForSyncedState();if(X.dust.availableCoins.length>0||X.dust.balance(new Date)>0n)return Z?.("Dust available"),{alreadyAvailable:!0};let z=X.unshielded.availableCoins.filter((J)=>J.meta?.registeredForDustGeneration!==!0),Y;if(z.length>0){Z?.(`Registering ${z.length} UTXO(s) for dust generation...`);let J=z.map((K)=>({...K.utxo,ctime:new Date(K.meta.ctime)}));Y=await BQ($,J,X.dust.address,Z)}else Z?.("UTXOs already registered, waiting for dust generation...");Z?.("Waiting for dust tokens...");let G=Date.now();while(Date.now()-G<h1){try{if((await Promise.race([$.facade.waitForSyncedState(),new Promise((K,q)=>setTimeout(()=>q(new Error("Poll sync timed out")),f1))])).dust.balance(new Date)>0n)return Z?.("Dust available"),{alreadyAvailable:!1,txHash:Y}}catch{}await new Promise((J)=>setTimeout(J,5000))}throw new Error("Timed out waiting for dust tokens. Try running: midnight dust register")}async function HQ($,Z,Q,X,z,Y,G){let J=Date.now(),K=J+h1,q,V=0;while(!0)try{if(q)await I$($,"lite");let B=new Date(Date.now()+N4*60*1000);w("transfer","Building transfer transaction...");let H=await $.facade.transferTransaction([{type:"unshielded",outputs:[{amount:Q,receiverAddress:Z,type:D$.unshieldedToken().raw}]}],{shieldedSecretKeys:$.zswapSecretKeys,dustSecretKey:$.dustSecretKey},{ttl:B,payFees:!0});w("transfer","Signing recipe...");let j=await $.facade.signRecipe(H,(_)=>$.keystore.signData(_));w("transfer","Generating ZK proof..."),X?.();let L=await Promise.race([$.facade.finalizeRecipe(j),new Promise((_,R)=>{setTimeout(()=>R(new Error("ZK proof generation timed out")),s4)})]);w("transfer","Submitting transaction to node..."),z?.();let W=Date.now(),F=G?setInterval(()=>G(Date.now()-W),1000):void 0,O;try{O=await $.facade.submitTransaction(L)}finally{if(F)clearInterval(F)}return w("transfer",`Transaction submitted: ${O}`),O}catch(B){if(q=B,(B?.code===i2||B?.message?.includes("115")||B?.message?.toLowerCase().includes("stale"))&&++V<l2)continue;if(U1(B))throw B;if(J2(B)&&Date.now()<K){let j=A$(Date.now()-J);Y?.(`Dust insufficient, re-ensuring (${j} elapsed)...`);try{let L=await I$($,"lite"),W=L.dust.balance(new Date);if(W>0n&&W<E4)throw new Error(`Insufficient dust for transaction fees.
|
|
93
|
+
Available: ${H2(W)} DUST, need ≥${H2(E4)} DUST.
|
|
94
|
+
Dust regenerates over time from registered NIGHT UTXOs.
|
|
95
|
+
Check status: midnight dust status`);await l0($,Y,L)}catch(L){if(String(L?.message).startsWith("Insufficient dust"))throw L;await new Promise((W)=>setTimeout(W,5000))}continue}throw B}}async function c4($){let{seedBuffer:Z,networkConfig:Q,recipientAddress:X,amountNight:z,signal:Y,onSync:G,onSyncDetail:J,onDust:K,onProving:q,onSubmitting:V,onSubmittingTick:B,onSyncWarning:H}=$,j=F1(z),L=KQ(X,Q),W=j0(H),F=T0();try{return await _0().withFacade(Z,Q,async({bundle:O,state:_})=>{if(Y?.aborted)throw new Error("Operation cancelled");w("transfer","Sync complete, checking balance...");let R=_.unshielded.balances[D$.unshieldedToken().raw]??0n,T=_.unshielded.availableCoins?.length??0;if(w("transfer",`Balance: ${Number(R)/w1} NIGHT, coins: ${T}`),R<j){let u=Number(R)/w1;throw new Error(`Insufficient balance: ${u.toFixed(6)} NIGHT available, ${z} NIGHT requested`)}let v=_.dust.balance(new Date);if(v>0n&&v<E4)throw new Error(`Insufficient dust for transaction fees.
|
|
96
|
+
Available: ${H2(v)} DUST, need ≥${H2(E4)} DUST.
|
|
97
|
+
Dust regenerates over time from registered NIGHT UTXOs.
|
|
98
|
+
Check status: midnight dust status`);if(Y?.aborted)throw new Error("Operation cancelled");if(w("transfer","Ensuring dust availability..."),await l0(O,K,_),w("transfer","Dust available"),Y?.aborted)throw new Error("Operation cancelled");return w("transfer","Building and submitting transaction..."),{txHash:await HQ(O,L,j,q,V,K,B),amountMicroNight:j}},{syncMode:"lite",requireStrictSync:!0,signal:Y,onStatus:K,onSyncProgress:G,onSyncDetail:J,onSyncWarning:H})}finally{F(),W()}}var qQ;var P4=N(()=>{i0();F4();H4();qQ={PreProd:R$.NetworkId.PreProd,Preview:R$.NetworkId.Preview,Undeployed:R$.NetworkId.Undeployed}});function V8($){return $.replace(/\x1b\[[0-9;]*m/g,"")}function i($){let Z=!1;if(!w4()){let J=$;process.stderr.write(`⠋ ${J}`);let K={update(q){J=q,process.stderr.write(`\r⠋ ${J}\x1B[K`)},stop(q){if(Z)return;Z=!0,l4=null,process.stderr.write(`\r✓ ${q??J}\x1B[K
|
|
99
|
+
`)},fail(q){if(Z)return;Z=!0,l4=null,process.stderr.write(`\r✗ ${q??J}\x1B[K
|
|
100
|
+
`)},log(q){process.stderr.write(`\r\x1B[K${q}
|
|
101
|
+
`),process.stderr.write(`⠋ ${J}`)}};return l4=K,K}let Q=0,X=$,z=()=>{let J=l(K8[Q]),q=(process.stderr.columns||80)-4,V=X;if(V8(V).length>q)V=V8(V).slice(0,q);process.stderr.write(`\r${J} ${V}\x1B[K`),Q=(Q+1)%K8.length};z();let Y=setInterval(z,WQ),G={update(J){X=J},stop(J){if(Z)return;Z=!0,clearInterval(Y),l4=null;let K=J??X;process.stderr.write(`\r\x1B[32m✓\x1B[0m ${K}\x1B[K
|
|
102
|
+
`)},fail(J){if(Z)return;Z=!0,clearInterval(Y),l4=null;let K=J??X;process.stderr.write(`\r\x1B[31m✗\x1B[0m ${K}\x1B[K
|
|
70
103
|
`)},log(J){process.stderr.write(`\r\x1B[K${J}
|
|
71
|
-
`),
|
|
72
|
-
|
|
73
|
-
`)
|
|
74
|
-
`);else for(let[Y,G]of q.balances)if(b0(Y))process.stdout.write(`NIGHT=${G}
|
|
75
|
-
`);else process.stdout.write(`${Y}=${G}
|
|
104
|
+
`),z()}};return l4=G,G}async function W2($,Z,Q){let X=Date.now(),z=(J)=>{let K=Math.floor(J/1000),q=Math.floor(K/60).toString().padStart(2,"0"),V=(K%60).toString().padStart(2,"0");return`${q}:${V}`},Y=()=>$.update(`${Z} ${z(Date.now()-X)}`);Y();let G=setInterval(Y,1000);try{return await Q}finally{clearInterval(G)}}var K8,WQ=80,l4=null;var n0=N(()=>{K8=["⠋","⠙","⠹","⠸","⠼","⠴","⠦","⠧","⠇","⠏"]});var W8={};k(W8,{default:()=>H8});import*as B8 from"@midnight-ntwrk/ledger-v8";import{MidnightBech32m as UQ}from"@midnight-ntwrk/wallet-sdk-address-format";async function H8($){if($.subcommand)return jQ($);return OQ($)}function LQ($){let Z=$.match(/^mn_addr_([a-z]+)1/);if(!Z)return null;let Q=Z[1];return F0(Q)?Q:null}async function jQ($){if(S0($))w0();let Z=$.subcommand;w("balance",`address=${Z}`);let Q=LQ(Z),X=A($,"network");if(X&&Q&&X!==Q)throw new Error(`Address belongs to ${Q} but --network is ${X}.
|
|
105
|
+
Drop --network (we'll infer from the address) or pass an address for ${X}.`);if(!X&&Q)$={...$,flags:{...$.flags,network:Q}};let{name:z,config:Y}=q0({args:$});k0(Y,{proofServer:A($,"proof-server"),node:A($,"node"),indexerWS:A($,"indexer-ws")});let G=i(`Checking balance on ${z}...`);try{let J=await _0().unshielded(Z,Y,{forceFresh:C($,"no-cache"),onProgress:(K,q)=>{if(q>0){let V=Math.round(K/q*100);G.update(`Syncing transactions... ${V}%`)}}});if(G.stop(`Synced ${J.txCount} transactions`),C($,"json")){let K={};for(let[V,B]of J.balances){let H=q1(V)?"NIGHT":V;K[H]=q1(V)?B4(B):B.toString()}let q=K.NIGHT!==void 0?{NIGHT:K.NIGHT}:{};if(s0($)){x({network:z,balances:K,...q,utxoCount:J.utxoCount});return}x({address:Z,network:z,balances:K,...q,utxoCount:J.utxoCount,txCount:J.txCount});return}if(J.balances.size===0)process.stdout.write(`0
|
|
106
|
+
`);else for(let[K,q]of J.balances)process.stdout.write(`${q1(K)?"NIGHT":K}=${q}
|
|
76
107
|
`);if(process.stderr.write(`
|
|
77
|
-
`+
|
|
108
|
+
`+g("Balance")+`
|
|
78
109
|
|
|
79
|
-
`),process.stderr.write(
|
|
80
|
-
`),process.stderr.write(
|
|
81
|
-
`),process.stderr.write(
|
|
82
|
-
`),process.stderr.write(
|
|
110
|
+
`),process.stderr.write(M("Address",s(Z))+`
|
|
111
|
+
`),process.stderr.write(M("Network",z)+`
|
|
112
|
+
`),process.stderr.write(M("UTXOs",J.utxoCount.toString())+`
|
|
113
|
+
`),process.stderr.write(M("Transactions",J.txCount.toString())+`
|
|
83
114
|
`),process.stderr.write(`
|
|
84
|
-
`),
|
|
85
|
-
`);else for(let[
|
|
86
|
-
`);else{let K
|
|
115
|
+
`),J.balances.size===0)process.stderr.write(` ${U("No balance found")}
|
|
116
|
+
`);else for(let[K,q]of J.balances)if(q1(K))process.stderr.write(M("NIGHT",y(s1(q)))+`
|
|
117
|
+
`);else{let V=K.slice(0,8)+"…"+K.slice(-8);process.stderr.write(M(`Token ${V}`,y(q.toString()))+`
|
|
87
118
|
`)}process.stderr.write(`
|
|
88
|
-
`+
|
|
119
|
+
`+d()+`
|
|
120
|
+
|
|
121
|
+
`)}catch(J){throw G.fail("Failed"),J}}async function OQ($){if(S0($))w0();let{name:Z,config:Q}=q0({args:$});w("balance",`network=${Z} indexerWS=${Q.indexerWS}`);let X=Q0(A($,"wallet"));w("balance",`wallet path=${X}`);let z=z0(X),Y=Buffer.from(z.seed,"hex"),G=z.addresses[Z],J=z.shieldedAddresses?.[Z]??"";k0(Q,{proofServer:A($,"proof-server"),node:A($,"node"),indexerWS:A($,"indexer-ws")});let K=j0(),q=T0(),V=C($,"no-cache"),B=C($,"json"),H=B8.unshieldedToken().raw,j=null;try{if(!B){if(process.stderr.write(`
|
|
122
|
+
`+g("Balance")+`
|
|
89
123
|
|
|
90
|
-
`)
|
|
124
|
+
`),process.stderr.write(M("Address",s(G))+`
|
|
125
|
+
`),J)process.stderr.write(M("Shielded",s(J))+`
|
|
126
|
+
`);process.stderr.write(M("Network",Z)+`
|
|
127
|
+
`),process.stderr.write(`
|
|
128
|
+
`+y("Unshielded")+`
|
|
129
|
+
`)}if(!B)j=i("Checking unshielded...");w("balance",`phase=unshielded forceFresh=${V}`);let L=await _0().unshielded(G,Q,{forceFresh:V}),W=L.balances.get(H)??0n,F=L.utxoCount;if(w("balance",`unshielded fromCache=${L.fromCache} utxos=${F} txs=${L.txCount}`),j)if(j.stop("Unshielded ready"),j=null,W>0n)process.stderr.write(M(" NIGHT",y(s1(W)))+`
|
|
130
|
+
`),process.stderr.write(M(" UTXOs",F.toString())+`
|
|
131
|
+
`);else process.stderr.write(` ${U("No unshielded balance")}
|
|
132
|
+
`);if(!B)process.stdout.write(`NIGHT=${W}
|
|
91
133
|
`),process.stderr.write(`
|
|
92
|
-
|
|
93
|
-
`),process.stderr.write(
|
|
94
|
-
`),process.stderr.write(
|
|
95
|
-
`)
|
|
96
|
-
`),process.
|
|
134
|
+
`+y("Shielded")+`
|
|
135
|
+
`);if(!B)j=i("Syncing shielded...");let O=J8(),_=f4(Q.networkId),{liveShieldedAddrStr:R,shieldedBalance:T,shieldedCoins:v,pendingCoins:b}=await _0().withFacade(Y,Q,async({state:u})=>{let p=UQ.encode(_,u.shielded.address).asString();return K4(X,Z,p),{liveShieldedAddrStr:p,shieldedBalance:u.shielded.balances[H]??0n,shieldedCoins:u.shielded.availableCoins.length,pendingCoins:u.shielded.pendingCoins.length}},{syncMode:"no-dust",requireStrictSync:!1,readOnly:!0,forceFresh:V,onSyncProgress:(u,p)=>{if(!j)return;let L0=O.sample({applied:u,highest:p,t:Date.now()});j.update(q8(L0,"Syncing shielded"))},onSyncDetail:(u)=>j?.update(`Syncing shielded (waiting on: ${u})`)});if(j)if(j.stop("Shielded ready"),j=null,T>0n)process.stderr.write(M(" NIGHT",y(s1(T)))+`
|
|
136
|
+
`),process.stderr.write(M(" Coins",`${v} available, ${b} pending`)+`
|
|
137
|
+
`);else process.stderr.write(` ${U("No shielded balance")}
|
|
138
|
+
`);if(B){if(s0($)){x({network:Z,unshielded:{NIGHT:B4(W),utxoCount:F},shielded:{NIGHT:B4(T),availableCoins:v,pendingCoins:b}});return}x({address:G,shieldedAddress:R,network:Z,unshielded:{NIGHT:B4(W),utxoCount:F},shielded:{NIGHT:B4(T),availableCoins:v,pendingCoins:b}});return}process.stdout.write(`SHIELDED_NIGHT=${T}
|
|
139
|
+
`),process.stderr.write(`
|
|
140
|
+
`+d()+`
|
|
97
141
|
|
|
98
|
-
`)}var
|
|
142
|
+
`)}catch(L){throw j?.fail("Failed"),L}finally{q(),K()}}var U8=N(()=>{H4();y0();b0();V0();y0();$2();K1();F4();i0();P4();H0();n0()});var j8={};k(j8,{default:()=>L8});async function L8($){let Z=D4($,"seed","hex").replace(/^0x/,"");if(Z.length!==64||!/^[0-9a-fA-F]+$/.test(Z))throw new Error("Seed must be a 64-character hex string (32 bytes)");let Q=A($,"index"),X=Q!==void 0?parseInt(Q,10):0;if(isNaN(X)||X<0||!Number.isInteger(Number(Q??"0")))throw new Error("Key index must be a non-negative integer");let z=Buffer.from(Z,"hex"),Y=P0({args:$}),G=N0(z,Y,X),J=`m/44'/2400'/0'/NightExternal/${X}`;if(C($,"json")){x({address:G,network:Y,index:X,path:J});return}process.stdout.write(G+`
|
|
99
143
|
`),process.stderr.write(`
|
|
100
|
-
`),process.stderr.write(
|
|
101
|
-
`),process.stderr.write(
|
|
102
|
-
`),process.stderr.write(
|
|
103
|
-
`),process.stderr.write(
|
|
144
|
+
`),process.stderr.write(M("Network",Y)+`
|
|
145
|
+
`),process.stderr.write(M("Index",X.toString())+`
|
|
146
|
+
`),process.stderr.write(M("Address",s(G))+`
|
|
147
|
+
`),process.stderr.write(M("Path",U(J))+`
|
|
148
|
+
`),process.stderr.write(d()+`
|
|
104
149
|
|
|
105
|
-
`)}var
|
|
150
|
+
`)}var O8=N(()=>{m0();V0();H0()});var P8={};k(P8,{default:()=>F8});async function F8($){let Z=P0({args:$}),Q=Buffer.from(t0,"hex"),X=N0(Q,Z);if(C($,"json")){x({address:X,network:Z});return}process.stdout.write(X+`
|
|
151
|
+
`),process.stderr.write(`
|
|
152
|
+
`),process.stderr.write(M("Network",Z)+`
|
|
153
|
+
`),process.stderr.write(M("Address",s(X))+`
|
|
154
|
+
`),process.stderr.write(M("Seed",U("0x01 (genesis)"))+`
|
|
155
|
+
`),process.stderr.write(d()+`
|
|
156
|
+
|
|
157
|
+
`)}var I8=N(()=>{m0();V0();H0()});var M8={};k(M8,{default:()=>T8});import*as _8 from"@midnight-ntwrk/ledger-v8";function I1($,Z,Q){let X={readTime:0n,computeTime:0n,blockUsage:0n,bytesWritten:0n,bytesChurned:0n};X[Z]=Q;let Y=$.normalizeFullness(X)[Z];return Math.round(Number(Q)/Y)}function FQ($){return{readTime:I1($,"readTime",1000000000n),computeTime:I1($,"computeTime",1000000000n),blockUsage:I1($,"blockUsage",10000n),bytesWritten:I1($,"bytesWritten",10000n),bytesChurned:I1($,"bytesChurned",1000000n)}}async function T8($){let Z=_8.LedgerParameters.initialParameters(),Q=FQ(Z);if(C($,"json")){x(Q);return}for(let[X,z]of Object.entries(Q))process.stdout.write(`${X}=${z}
|
|
106
158
|
`);process.stderr.write(`
|
|
107
|
-
`+
|
|
159
|
+
`+g("Block Limits")+`
|
|
108
160
|
|
|
109
|
-
`),process.stderr.write(
|
|
161
|
+
`),process.stderr.write(U(" Derived from LedgerParameters.initialParameters()")+`
|
|
110
162
|
|
|
111
|
-
`);for(let[X,z]of Object.entries(Q)){let
|
|
163
|
+
`);for(let[X,z]of Object.entries(Q)){let Y=PQ[X]??"";process.stderr.write(M(X,`${y(z.toLocaleString())} ${U(Y)}`)+`
|
|
112
164
|
`)}process.stderr.write(`
|
|
113
|
-
`+
|
|
114
|
-
`),process.stderr.write(
|
|
115
|
-
`),process.stderr.write(
|
|
116
|
-
|
|
117
|
-
`)}var
|
|
118
|
-
`)}var K3=!1;var I0=()=>{};import{HDWallet as v$,Roles as u1}from"@midnight-ntwrk/wallet-sdk-hd";function m1($,Z){let Q=v$.fromSeed($);if(Q.type!=="seedOk")throw new Error("Invalid seed for HD wallet");let X=Q.hdWallet.selectAccount(0).selectRole(Z).deriveKeyAt(0);if(X.type==="keyOutOfBounds")throw new Error("Key derivation out of bounds");return X.key}function J3($){return m1($,u1.Zswap)}function U3($){return m1($,u1.NightExternal)}function j3($){return m1($,u1.Dust)}var B3=()=>{};import{ShieldedWallet as V3}from"@midnight-ntwrk/wallet-sdk-shielded";import{UnshieldedWallet as H3,createKeystore as k$,PublicKey as w$,InMemoryTransactionHistoryStorage as h$}from"@midnight-ntwrk/wallet-sdk-unshielded-wallet";import{DustWallet as W3}from"@midnight-ntwrk/wallet-sdk-dust-wallet";import{WalletFacade as L3}from"@midnight-ntwrk/wallet-sdk-facade";import*as M0 from"@midnight-ntwrk/ledger-v8";import{NetworkId as c1}from"@midnight-ntwrk/wallet-sdk-abstractions";import*as s from"rxjs";async function k0($,Z,Q){let X=f$[Z.networkId];if(X===void 0)throw new Error(`Unknown networkId: ${Z.networkId}`);T("facade",`Building facade for network ${Z.networkId}`),T("facade",`Node: ${Z.node}`),T("facade",`Indexer: ${Z.indexerWS}`),T("facade",`Proof server: ${Z.proofServer}`);let z=J3($),q=U3($),Y=j3($),G=M0.ZswapSecretKeys.fromSeed(z),K=M0.DustSecretKey.fromSeed(Y),J=k$(q,X),B={networkId:X,indexerClientConnection:{indexerHttpUrl:Z.indexer,indexerWsUrl:Z.indexerWS},costParameters:{additionalFeeOverhead:W2,feeBlocksMargin:L2},txHistoryStorage:new h$,provingServerUrl:new URL(Z.proofServer),relayURL:new URL(Z.node)},j=()=>L3.init({configuration:B,shielded:(V)=>V3(V).startWithSecretKeys(G),unshielded:(V)=>H3(V).startWithPublicKey(w$.fromKeyStore(J)),dust:(V)=>W3(V).startWithSecretKey(K,M0.LedgerParameters.initialParameters().dust)}),U=!1,W;if(Q){T("facade","Restoring from cache...");try{W=await L3.init({configuration:B,shielded:(V)=>V3(V).restore(Q.shielded),unshielded:(V)=>H3(V).restore(Q.unshielded),dust:(V)=>W3(V).restore(Q.dust)}),U=!0,T("facade","Cache restore successful")}catch(V){T("facade",`Cache restore failed: ${V.message}`),process.stderr.write(` Cache restore failed, building from scratch: ${V.message}
|
|
119
|
-
`),W=await j()}}else T("facade","No cache, building fresh"),W=await j();return{facade:W,keystore:J,zswapSecretKeys:G,dustSecretKey:K,restoredFromCache:U}}function P3($,Z="full"){let Q=$.unshielded?.progress?.isStrictlyComplete()??!1,X=$.dust?.state?.progress?.isStrictlyComplete()??!1;if(!X)try{let q=$.dust?.state?.progress;if(q&&q.highestRelevantWalletIndex>0&&q.appliedIndex>=q.highestRelevantWalletIndex)X=!0}catch{}if(Z==="lite")return Q&&X;return($.shielded?.state?.progress?.isStrictlyComplete()??!1)&&Q&&X}function p$($){if($.dust?.state?.progress?.isStrictlyComplete())return!1;try{let Z=$.dust?.state?.progress;if(Z&&Z.highestRelevantWalletIndex>0&&Z.appliedIndex>=Z.highestRelevantWalletIndex)return!1}catch{}return!0}async function w0($,Z={}){let{onProgress:Q,onSyncDetail:X,timeoutMs:z,syncMode:q="full"}=Z,{facade:Y,zswapSecretKeys:G,dustSecretKey:K}=$;T("sync","Starting facade (connecting to node and indexer)..."),await Y.start(G,K),T("sync","Facade started, subscribing to state...");let J=z??P2;return T("sync",`Sync timeout: ${J/1000}s, mode: ${q}`),new Promise((B,j)=>{let U=!1,W=0,V="",P=null,x=setTimeout(()=>{if(!U){if(T("sync",`Sync timed out after ${J/1000}s (${W} emissions)`),P)try{let F=P.unshielded?.progress;T("sync",` unshielded: applied=${F?.appliedId} highest=${F?.highestTransactionId} complete=${F?.isStrictlyComplete()}`);let O=P.dust?.state?.progress;if(T("sync",` dust: applied=${O?.appliedIndex} highest=${O?.highestRelevantWalletIndex} complete=${O?.isStrictlyComplete?.()} connected=${O?.isConnected}`),q==="full"){let A=P.shielded?.state?.progress;T("sync",` shielded: complete=${A?.isStrictlyComplete()}`)}}catch{}j(new Error("Wallet sync timed out"))}},J);$.keepAlive=Y.state().subscribe({next:(F)=>{if(U)return;if(W++,P=F,Q){let A=F.unshielded.progress;if(A){let k=Number(A.appliedId),h=Number(A.highestTransactionId);Q(Math.min(k,h),h)}}let O=[];try{if(q==="full"&&!F.shielded?.state?.progress?.isStrictlyComplete())O.push("shielded");if(p$(F))O.push("dust");if(!F.unshielded?.progress?.isStrictlyComplete())O.push("unshielded")}catch{}if(O.length>0){X?.(O.join(", "));let A=O.join(",");if(W===1||A!==V||W%100===0)T("sync",`Waiting on: ${O.join(", ")} (emission #${W})`),V=A}if(P3(F,q))U=!0,clearTimeout(x),T("sync",`Sync complete after ${W} emissions`),B(F)},error:(F)=>{if(!U)T("sync",`Sync error: ${F.message}`),clearTimeout(x),j(F)}})})}async function g1($){let Z=(Q)=>{let X=Q.unshielded?.progress?.isStrictlyComplete()??!1,z=Q.dust?.state?.progress?.isStrictlyComplete()??!1;return X&&z};try{return await s.firstValueFrom($.facade.state().pipe(s.filter(Z),s.timeout(15000)))}catch{return await s.firstValueFrom($.facade.state())}}async function d1($,Z="full"){return s.firstValueFrom($.facade.state().pipe(s.filter((Q)=>P3(Q,Z)),s.timeout(O2)))}async function h0($){$.keepAlive?.unsubscribe(),await Promise.race([$.facade.stop(),new Promise((Z)=>setTimeout(Z,5000))])}function Q1($){let Z=(X)=>{let z=X?._tag;if(typeof z==="string"&&z.startsWith("Wallet.")){let q=X?.message??"transient error";$?.(z,q);return}Q("Unhandled rejection:",X),process.exit(1)},Q=console.error;return console.error=(...X)=>{let z=X[0];if(typeof z==="object"&&z?._tag?.startsWith("Wallet.")){$?.(z._tag,z?.message??"transient error");return}if(typeof z==="string"&&z.startsWith("Wallet.")){$?.("Wallet.Sync","transient error");return}Q(...X)},process.on("unhandledRejection",Z),()=>{process.removeListener("unhandledRejection",Z),console.error=Q}}var f$;var l1=E(()=>{B3();I0();f$={PreProd:c1.NetworkId.PreProd,Preview:c1.NetworkId.Preview,Undeployed:c1.NetworkId.Undeployed}});import{existsSync as z1,mkdirSync as u$,writeFileSync as m$,readFileSync as c$,unlinkSync as X1,readdirSync as i1,renameSync as g$}from"node:fs";import{join as H0,dirname as d$}from"node:path";import{homedir as F3}from"node:os";import{randomBytes as l$}from"node:crypto";function o1($,Z,Q){let X=Q??H0(F3(),g,C1),z=$.slice(0,20);return H0(X,Z,`${z}.json`)}function f0($,Z,Q){let X=o1($,Z,Q);if(!z1(X))return null;try{let z=c$(X,"utf-8"),q=JSON.parse(z);if(q.version!==D1)return null;if(q.network!==Z)return null;if(q.address!==$)return null;if(!q.wallets||typeof q.wallets.shielded!=="string"||typeof q.wallets.unshielded!=="string"||typeof q.wallets.dust!=="string")return null;return q.wallets}catch{return null}}async function _0($,Z,Q,X){let[z,q,Y]=await Promise.all([Q.shielded.serializeState(),Q.unshielded.serializeState(),Q.dust.serializeState()]),G={version:D1,network:Z,address:$,timestamp:new Date().toISOString(),wallets:{shielded:z,unshielded:q,dust:Y}},K=o1($,Z,X),J=d$(K);if(!z1(J))u$(J,{recursive:!0,mode:e});let B=K+`.tmp.${l$(4).toString("hex")}`;try{m$(B,JSON.stringify(G),{mode:G0}),g$(B,K)}catch(j){try{X1(B)}catch{}throw j}}function q1($,Z,Q){let X=Q??H0(F3(),g,C1);if($&&Z){let z=o1($,Z,Q);try{X1(z)}catch{}return}if(Z){let z=H0(X,Z);if(!z1(z))return;try{for(let q of i1(z))if(q.endsWith(".json"))X1(H0(z,q))}catch{}return}if(!z1(X))return;try{for(let z of i1(X)){let q=H0(X,z);try{for(let Y of i1(q))if(Y.endsWith(".json"))X1(H0(q,Y))}catch{}}}catch{}}var Y1=()=>{};import*as r1 from"@midnight-ntwrk/ledger-v8";import{MidnightBech32m as i$,UnshieldedAddress as o$}from"@midnight-ntwrk/wallet-sdk-address-format";import{NetworkId as n1}from"@midnight-ntwrk/wallet-sdk-abstractions";function t$($){if($<=0)throw new Error("Amount must be greater than 0");if(!Number.isFinite($))throw new Error("Amount must be a finite number");let Z=$.toFixed(6),[Q,X]=Z.split("."),z=Q+(X??"").padEnd(6,"0"),q=BigInt(z);if(q<=0n)throw new Error("Amount too small — minimum is 0.000001 NIGHT");return q}function K1($){let Z=Number($);if(Number.isNaN(Z)||!Number.isFinite(Z))throw new Error(`Invalid amount: "${$}" — must be a positive number`);if(Z<=0)throw new Error(`Invalid amount: "${$}" — must be greater than 0`);return Z}function r$($,Z){let Q=n$[Z.networkId];if(Q===void 0)throw new Error(`Unknown networkId: ${Z.networkId}`);try{return i$.parse($).decode(o$,Q)}catch(X){throw new Error(`Invalid recipient address: ${X.message}
|
|
120
|
-
Expected a bech32m address for network "${Z.networkId}"`)}}function a$($){let Z=$;while(Z){let Q=String(Z?.message??"").toLowerCase();if(Q.includes("submission error"))return!0;if(Q.includes("transaction")&&Q.includes("invalid"))return!0;if(Q.includes("138"))return!0;let X=Z?._tag;if(X==="TransactionInvalidError"||X==="SubmissionError")return!0;Z=Z.cause}return!1}function O3($){let Z=$?.message?.toLowerCase()??"";return Z.includes("not enough dust")||Z.includes("dust generated")||Z.includes("insufficient funds")||Z.includes("no dust tokens")||a$($)}function G1($){let Z=$<0n?-$:$,Q=Z/1000000000000000n,X=Z%1000000000000000n;return`${$<0n?"-":""}${Q}.${X.toString().padStart(15,"0").slice(0,6)}`}function t1($){let Z=Math.round($/1000);if(Z<60)return`${Z}s`;let Q=Math.floor(Z/60),X=Z%60;return`${Q}m ${X}s`}function a1(){let{warn:$,error:Z}=console,Q=(X)=>X.some((z)=>String(z).includes("RPC-CORE"));return console.warn=(...X)=>{if(!Q(X))$(...X)},console.error=(...X)=>{if(!Q(X))Z(...X)},()=>{console.warn=$,console.error=Z}}async function s$($,Z,Q){let X=new Date(Date.now()+A1*60*1000);await Promise.race([$.facade.dust.waitForSyncedState(),new Promise((J,B)=>setTimeout(()=>B(new Error("Insufficient funds: dust wallet sync timed out")),g0))]);let z=await $.facade.dust.createDustGenerationTransaction(new Date,X,Z,$.keystore.getPublicKey(),Q),q=z.intents?.get(1);if(!q)throw new Error("Dust generation intent not found on transaction");let Y=$.keystore.signData(q.signatureData(1)),G=await $.facade.dust.addDustGenerationSignature(z,Y),K=await $.facade.finalizeTransaction(G);return await $.facade.submitTransaction(K)}async function e$($,Z,Q,X){let z=Date.now(),q=z+_2,Y,G=!1,K=setInterval(()=>{if(G&&X){let J=t1(Date.now()-z);X(`Waiting for dust generation capacity (${J} elapsed, ~5 min on fresh wallets)...`)}},1000);try{while(Date.now()<q)try{return await s$($,Z,Q)}catch(J){if(Y=J,O3(J)&&Date.now()+E1<q){G=!0;let B=t1(Date.now()-z);X?.(`Waiting for dust generation capacity (${B} elapsed, ~5 min on fresh wallets)...`),await new Promise((j)=>setTimeout(j,E1));continue}throw J}throw Y??new Error("Dust registration timed out")}finally{clearInterval(K)}}async function J1($,Z,Q){let X=Q??await $.facade.waitForSyncedState();if(X.dust.availableCoins.length>0||X.dust.balance(new Date)>0n)return Z?.("Dust available"),{alreadyAvailable:!0};let z=X.unshielded.availableCoins.filter((G)=>G.meta?.registeredForDustGeneration!==!0),q;if(z.length>0){Z?.(`Registering ${z.length} UTXO(s) for dust generation...`);let G=z.map((K)=>({...K.utxo,ctime:new Date(K.meta.ctime)}));q=await e$($,G,X.dust.address,Z)}else Z?.("UTXOs already registered, waiting for dust generation...");Z?.("Waiting for dust tokens...");let Y=Date.now();while(Date.now()-Y<R1){try{if((await Promise.race([$.facade.waitForSyncedState(),new Promise((K,J)=>setTimeout(()=>J(new Error("Poll sync timed out")),g0))])).dust.balance(new Date)>0n)return Z?.("Dust available"),{alreadyAvailable:!1,txHash:q}}catch{}await new Promise((G)=>setTimeout(G,5000))}throw new Error("Timed out waiting for dust tokens. Try running: midnight dust register")}async function $8($,Z,Q,X,z,q){let Y=Date.now(),G=Y+R1,K,J=0;while(!0)try{if(K)await d1($,"lite");let B=new Date(Date.now()+A1*60*1000);T("transfer","Building transfer transaction...");let j=await $.facade.transferTransaction([{type:"unshielded",outputs:[{amount:Q,receiverAddress:Z,type:r1.unshieldedToken().raw}]}],{shieldedSecretKeys:$.zswapSecretKeys,dustSecretKey:$.dustSecretKey},{ttl:B,payFees:!0});T("transfer","Signing recipe...");let U=await $.facade.signRecipe(j,(P)=>$.keystore.signData(P));T("transfer","Generating ZK proof..."),X?.();let W=await Promise.race([$.facade.finalizeRecipe(U),new Promise((P,x)=>{setTimeout(()=>x(new Error("ZK proof generation timed out")),I2)})]);T("transfer","Submitting transaction to node..."),z?.();let V=await $.facade.submitTransaction(W);return T("transfer",`Transaction submitted: ${V}`),V}catch(B){if(K=B,(B?.code===T2||B?.message?.includes("115")||B?.message?.toLowerCase().includes("stale"))&&++J<M2)continue;if(O3(B)&&Date.now()<G){let U=t1(Date.now()-Y);q?.(`Dust insufficient, re-ensuring (${U} elapsed)...`);try{let W=await d1($,"lite"),V=W.dust.balance(new Date);if(V>0n&&V<x0)throw new Error(`Insufficient dust for transaction fees.
|
|
121
|
-
Available: ${G1(V)} DUST, need ≥${G1(x0)} DUST.
|
|
122
|
-
Dust regenerates over time from registered NIGHT UTXOs.
|
|
123
|
-
Check status: midnight dust status`);await J1($,q,W)}catch(W){if(String(W?.message).startsWith("Insufficient dust"))throw W;await new Promise((V)=>setTimeout(V,5000))}continue}throw B}}async function U1($){let{seedBuffer:Z,networkConfig:Q,recipientAddress:X,amountNight:z,signal:q,onSync:Y,onSyncDetail:G,onDust:K,onProving:J,onSubmitting:B,onSyncWarning:j,noCache:U,walletAddress:W,networkName:V}=$,P=t$(z),x=r$(X,Q),F=Q1(j),O=a1(),A=!U&&W&&V,k=A?f0(W,V):null;T("transfer","Building facade...");let h=await k0(Z,Q,k),R0=!1,Y2=async()=>{if(!R0){R0=!0;try{await h0(h)}catch{}}},G2=()=>{Y2()};q?.addEventListener("abort",G2,{once:!0});try{let K2=Q.networkId!=="Undeployed",J2=K2?F2:g0,m8=3,u0;T("transfer",`Sync timeout: ${J2/1000}s (${K2?"remote":"local"} network)`);for(let W0=1;W0<=3;W0++)try{u0=await w0(h,{onProgress:Y,onSyncDetail:G,timeoutMs:J2,syncMode:"lite"});break}catch(U2){if(q?.aborted)throw new Error("Operation cancelled");if(W0<3&&String(U2?.message).includes("timed out")){if(A)try{T("transfer","Saving partial sync progress to cache..."),await _0(W,V,h.facade)}catch{}K?.(`Sync timed out, retrying (attempt ${W0+1}/3)...`),await h0(h).catch(()=>{});let J$=A?f0(W,V):null;h=await k0(Z,Q,J$);continue}throw U2}if(q?.aborted)throw new Error("Operation cancelled");T("transfer","Sync complete, checking balance...");let F1=u0.unshielded.balances[r1.unshieldedToken().raw]??0n;if(T("transfer",`Balance: ${Number(F1)/T1} NIGHT`),F1<P){let W0=Number(F1)/T1;throw new Error(`Insufficient balance: ${W0.toFixed(6)} NIGHT available, ${z} NIGHT requested`)}if(q?.aborted)throw new Error("Operation cancelled");T("transfer","Ensuring dust availability..."),await J1(h,K,u0),T("transfer","Dust available");let O1=u0.dust.balance(new Date);if(O1>0n&&O1<x0)throw new Error(`Insufficient dust for transaction fees.
|
|
124
|
-
Available: ${G1(O1)} DUST, need ≥${G1(x0)} DUST.
|
|
125
|
-
Dust regenerates over time from registered NIGHT UTXOs.
|
|
126
|
-
Check status: midnight dust status`);if(q?.aborted)throw new Error("Operation cancelled");T("transfer","Building and submitting transaction...");let K$=await $8(h,x,P,J,B,K);if(A)try{await _0(W,V,h.facade)}catch{}return{txHash:K$,amountMicroNight:P}}finally{q?.removeEventListener("abort",G2),O(),F(),await Y2()}}var n$;var j1=E(()=>{l1();Y1();I0();n$={PreProd:n1.NetworkId.PreProd,Preview:n1.NetworkId.Preview,Undeployed:n1.NetworkId.Undeployed}});var M3={};f(M3,{default:()=>I3});async function I3($,Z){let Q=$.subcommand;if(!Q)throw new Error(`Missing amount.
|
|
165
|
+
`+d()+`
|
|
166
|
+
`),process.stderr.write(U(" bytesWritten is typically the tightest constraint")+`
|
|
167
|
+
`),process.stderr.write(U(" for large contract deployments.")+`
|
|
168
|
+
|
|
169
|
+
`)}var PQ;var R8=N(()=>{H0();PQ={readTime:"picoseconds",computeTime:"picoseconds",blockUsage:"bytes",bytesWritten:"bytes",bytesChurned:"bytes"}});var E8={};k(E8,{default:()=>D8});import*as A8 from"@midnight-ntwrk/ledger-v8";import{MidnightBech32m as IQ}from"@midnight-ntwrk/wallet-sdk-address-format";async function D8($,Z){A4($);let Q=$.subcommand;if(!Q)throw new Error(`Missing amount.
|
|
127
170
|
Usage: midnight airdrop <amount>
|
|
128
|
-
Example: midnight airdrop 1000`);let X=
|
|
129
|
-
Current network: "${
|
|
130
|
-
On preprod/preview, use a faucet or transfer from another wallet.`);
|
|
131
|
-
`+
|
|
132
|
-
|
|
133
|
-
`),process.stderr.write(
|
|
134
|
-
`),process.stderr.write(
|
|
135
|
-
`),process.stderr.write(
|
|
136
|
-
`),process.stderr.write(
|
|
171
|
+
Example: midnight airdrop 1000`);let X=P1(Q),z=z0(Q0(A($,"wallet"))),{name:Y,config:G}=q0({args:$});if(k0(G,{proofServer:A($,"proof-server"),node:A($,"node"),indexerWS:A($,"indexer-ws")}),Y!=="undeployed")throw new Error(`Airdrop is only available on the "undeployed" network (local devnet).
|
|
172
|
+
Current network: "${Y}"
|
|
173
|
+
On preprod/preview, use a faucet or transfer from another wallet.`);if(C($,"shielded"))return TQ($,z,X,Y,G,Z);return _Q($,z,X,Y,G,Z)}async function _Q($,Z,Q,X,z,Y){if(S0($))w0();let G=Z.addresses[X],J=Buffer.from(t0,"hex");process.stderr.write(`
|
|
174
|
+
`+g("Airdrop")+`
|
|
175
|
+
|
|
176
|
+
`),process.stderr.write(M("Network",X)+`
|
|
177
|
+
`),process.stderr.write(M("From",U("genesis (seed 0x01)"))+`
|
|
178
|
+
`),process.stderr.write(M("To",s(G,!0))+`
|
|
179
|
+
`),process.stderr.write(M("Amount",y(Q+" NIGHT"))+`
|
|
137
180
|
`),process.stderr.write(`
|
|
138
|
-
`);let
|
|
181
|
+
`);let K=i("Starting genesis wallet...");try{let q=await c4({seedBuffer:J,networkConfig:z,recipientAddress:G,amountNight:Q,signal:Y,onSync(V,B){if(B>0){let H=Math.round(V/B*100);K.update(`Syncing genesis wallet... ${H}%`)}},onDust(V){K.update(`Dust: ${V}`)},onProving(){K.update("Generating ZK proof (this may take a few minutes)...")},onSubmitting(){K.update("Submitting and waiting for finalization (typically 12 to 30s)...")},onSubmittingTick(V){let B=Math.floor(V/1000),H=Math.floor(B/60).toString().padStart(2,"0"),j=(B%60).toString().padStart(2,"0");K.update(`Submitting and waiting for finalization... ${H}:${j} elapsed (typically 12 to 30s)`)},onSyncWarning(V,B){K.update(`Syncing genesis wallet... (${B}, retrying)`)}});if(K.stop("Transaction submitted"),C($,"json")){x({txHash:q.txHash,amount:Q,recipient:G,network:X});return}process.stdout.write(q.txHash+`
|
|
139
182
|
`),process.stderr.write(`
|
|
140
|
-
`+
|
|
183
|
+
`+Q4(`Airdropped ${Q} NIGHT to your wallet`,q.txHash)+`
|
|
141
184
|
`),process.stderr.write(`
|
|
142
|
-
`+
|
|
143
|
-
`),process.stderr.write(
|
|
144
|
-
`),process.stderr.write(
|
|
145
|
-
`),process.stderr.write(
|
|
146
|
-
`),process.stderr.write(
|
|
185
|
+
`+d()+`
|
|
186
|
+
`),process.stderr.write(U(" Verify: midnight balance")+`
|
|
187
|
+
`),process.stderr.write(U(" Register dust: midnight dust register")+`
|
|
188
|
+
`),process.stderr.write(U(" Note: Dust generation takes a few minutes on a fresh wallet.")+`
|
|
189
|
+
`),process.stderr.write(U(" It will happen automatically on your first transfer.")+`
|
|
147
190
|
|
|
148
|
-
`)}catch(
|
|
191
|
+
`)}catch(q){if(K.fail("Failed"),q instanceof Error&&q.message.toLowerCase().includes("dust"))throw new Error(`${q.message}
|
|
149
192
|
|
|
150
193
|
On a fresh localnet, the minimum airdrop is ~1 NIGHT.
|
|
151
|
-
Try: midnight airdrop 1`);throw
|
|
194
|
+
Try: midnight airdrop 1`);throw q}}async function TQ($,Z,Q,X,z,Y){let G=F1(Q);if(S0($))w0();let J=Buffer.from(Z.seed,"hex"),K=Buffer.from(t0,"hex"),q=f4(z.networkId),V=A8.unshieldedToken().raw,B=a1(J),H=IQ.encode(q,B).asString();K4(Q0(A($,"wallet")),X,H),process.stderr.write(`
|
|
195
|
+
`+g("Shielded Airdrop")+`
|
|
196
|
+
|
|
197
|
+
`),process.stderr.write(M("Network",X)+`
|
|
198
|
+
`),process.stderr.write(M("From",U("genesis shielded (seed 0x01)"))+`
|
|
199
|
+
`),process.stderr.write(M("To",s(H,!0))+`
|
|
200
|
+
`),process.stderr.write(M("Amount",y(Q+" NIGHT (shielded)"))+`
|
|
201
|
+
`),process.stderr.write(`
|
|
202
|
+
`);let j=j0(),L=T0(),W=i("Syncing genesis wallet...");try{let F=await _0().withFacade(K,z,async({bundle:O,state:_})=>{let R=_.shielded.balances[V]??0n;if(R<G){let p=Number(R)/1e6;throw new Error(`Genesis wallet has insufficient shielded balance: ${p.toFixed(6)} NIGHT available, ${Q} NIGHT requested`)}if(W.update("Checking dust..."),await l0(O,(p)=>W.update(p)),Y?.aborted)throw new Error("Operation cancelled");W.update("Building shielded transaction...");let T=await O.facade.transferTransaction([{type:"shielded",outputs:[{type:V,amount:G,receiverAddress:B}]}],{shieldedSecretKeys:O.zswapSecretKeys,dustSecretKey:O.dustSecretKey},{ttl:new Date(Date.now()+3600000)});W.update("Signing...");let v=await O.facade.signRecipe(T,(p)=>O.keystore.signData(p));W.update("Generating ZK proof (this may take a few minutes)...");let b=await O.facade.finalizeRecipe(v),u=await W2(W,"Submitting and waiting for finalization (typically 12 to 30s)...",O.facade.submitTransaction(b));return String(u)},{syncMode:"full",requireStrictSync:!0,signal:Y,onStatus:(O)=>W.update(O),onSyncProgress:(O,_)=>{if(_>0){let R=Math.min(Math.round(O/_*100),100);W.update(R>=100?"Syncing genesis wallet...":`Syncing genesis wallet... ${R}%`)}},onSyncDetail:(O)=>W.update(`Syncing genesis wallet... (waiting on: ${O})`)});if(W.stop("Transaction submitted"),C($,"json")){x({txHash:F,amount:Q,shieldedAddress:H,network:X,type:"shielded"});return}process.stdout.write(F+`
|
|
203
|
+
`),process.stderr.write(`
|
|
204
|
+
`+Q4(`Airdropped ${Q} shielded NIGHT to your wallet`,F)+`
|
|
205
|
+
`),process.stderr.write(`
|
|
206
|
+
`+d()+`
|
|
207
|
+
`),process.stderr.write(U(" Verify: midnight balance --shielded")+`
|
|
208
|
+
|
|
209
|
+
`)}catch(F){throw W.fail("Failed"),F}finally{L(),j()}}var N8=N(()=>{H4();b0();V0();y0();$2();m0();P4();i0();F4();H0();n0()});var y8={};k(y8,{default:()=>v8});import*as x8 from"@midnight-ntwrk/ledger-v8";import{MidnightBech32m as C8,ShieldedAddress as MQ}from"@midnight-ntwrk/wallet-sdk-address-format";async function v8($,Z){A4($);let Q=$.subcommand,X=$.positionals[0];if(!Q)throw new Error(`Missing recipient address.
|
|
152
210
|
Usage: midnight transfer <to> <amount>
|
|
153
|
-
Example: midnight transfer mn_addr_undeployed1... 100
|
|
211
|
+
Example: midnight transfer mn_addr_undeployed1... 100
|
|
212
|
+
Example: midnight transfer alice 100`);if(!X)throw new Error(`Missing amount.
|
|
154
213
|
Usage: midnight transfer <to> <amount>
|
|
155
|
-
Example: midnight transfer mn_addr_undeployed1... 100`);let z=
|
|
156
|
-
|
|
214
|
+
Example: midnight transfer mn_addr_undeployed1... 100`);let z=C($,"shielded"),Y=RQ(Q,$,z);if(z)return DQ($,Y,X,Z);return AQ($,Y,X,Z)}function RQ($,Z,Q){if($.startsWith("mn_addr_")||$.startsWith("mn_shield-addr_"))return $;let X=Q0($),z=z0(X),{name:Y}=q0({args:Z});if(Q){let G=z.shieldedAddresses?.[Y];if(!G)throw new Error(`Wallet "${$}" has no shielded address for network "${Y}".
|
|
215
|
+
Regenerate the wallet or run "midnight balance --shielded" first.`);return G}return z.addresses[Y]}async function AQ($,Z,Q,X){let z=P1(Q),Y=Q0(A($,"wallet")),G=z0(Y),J=Buffer.from(G.seed,"hex"),{name:K,config:q}=q0({args:$}),V=G.addresses[K];if(k0(q,{proofServer:A($,"proof-server"),node:A($,"node"),indexerWS:A($,"indexer-ws")}),process.stderr.write(`
|
|
216
|
+
`+g("Transfer")+`
|
|
157
217
|
|
|
158
|
-
`),process.stderr.write(
|
|
159
|
-
`),process.stderr.write(
|
|
160
|
-
`),process.stderr.write(
|
|
161
|
-
`),process.stderr.write(
|
|
218
|
+
`),process.stderr.write(M("Network",K)+`
|
|
219
|
+
`),process.stderr.write(M("From",s(V,!0))+`
|
|
220
|
+
`),process.stderr.write(M("To",s(Z,!0))+`
|
|
221
|
+
`),process.stderr.write(M("Amount",y(z+" NIGHT"))+`
|
|
162
222
|
`),process.stderr.write(`
|
|
163
|
-
`)
|
|
223
|
+
`),S0($))w0();let B=i("Starting wallet...");try{let H=await c4({seedBuffer:J,networkConfig:q,recipientAddress:Z,amountNight:z,signal:X,onSync(j,L){if(L>0){let W=Math.min(Math.round(j/L*100),100);B.update(W>=100?"Syncing wallet...":`Syncing wallet... ${W}%`)}},onSyncDetail(j){B.update(`Syncing wallet... (waiting on: ${j})`)},onDust(j){B.update(`Dust: ${j}`)},onProving(){B.update("Generating ZK proof (this may take a few minutes)...")},onSubmitting(){B.update("Submitting and waiting for finalization (typically 12 to 30s)...")},onSubmittingTick(j){let L=Math.floor(j/1000),W=Math.floor(L/60).toString().padStart(2,"0"),F=(L%60).toString().padStart(2,"0");B.update(`Submitting and waiting for finalization... ${W}:${F} elapsed (typically 12 to 30s)`)},onSyncWarning(j,L){B.update(`Syncing wallet... (${L}, retrying)`)}});if(B.stop("Transaction submitted"),C($,"json")){x({txHash:H.txHash,amount:z,recipient:Z,network:K});return}process.stdout.write(H.txHash+`
|
|
164
224
|
`),process.stderr.write(`
|
|
165
|
-
`+
|
|
225
|
+
`+Q4(`Transferred ${z} NIGHT`,H.txHash)+`
|
|
166
226
|
`),process.stderr.write(`
|
|
167
|
-
`+
|
|
168
|
-
`),process.stderr.write(
|
|
227
|
+
`+d()+`
|
|
228
|
+
`),process.stderr.write(U(" Verify: midnight balance")+`
|
|
169
229
|
|
|
170
|
-
`)}catch(
|
|
230
|
+
`)}catch(H){throw B.fail("Failed"),H}}async function DQ($,Z,Q,X){let z=P1(Q),Y=F1(z),G=Q0(A($,"wallet")),J=z0(G),K=Buffer.from(J.seed,"hex"),{name:q,config:V}=q0({args:$}),B=J.addresses[q],H=f4(V.networkId),j=x8.unshieldedToken().raw;k0(V,{proofServer:A($,"proof-server"),node:A($,"node"),indexerWS:A($,"indexer-ws")});let L;try{L=C8.parse(Z).decode(MQ,H)}catch(_){throw new Error(`Invalid shielded address: ${_.message}
|
|
231
|
+
Expected a shielded address (mn_shield-addr_...) for network "${V.networkId}"`)}if(process.stderr.write(`
|
|
232
|
+
`+g("Shielded Transfer")+`
|
|
233
|
+
|
|
234
|
+
`),process.stderr.write(M("Network",q)+`
|
|
235
|
+
`),process.stderr.write(M("To",s(Z,!0))+`
|
|
236
|
+
`),process.stderr.write(M("Amount",y(z+" NIGHT (shielded)"))+`
|
|
237
|
+
`),process.stderr.write(`
|
|
238
|
+
`),S0($))w0();let W=j0(),F=T0(),O=i("Syncing wallet...");try{let _=await _0().withFacade(K,V,async({bundle:R,state:T})=>{let v=C8.encode(H,T.shielded.address).asString();K4(G,q,v);let b=T.shielded.balances[j]??0n;if(b<Y){let o=Number(b)/1e6;throw new Error(`Insufficient shielded balance: ${o.toFixed(6)} NIGHT available, ${z} NIGHT requested.
|
|
239
|
+
Fund shielded balance: midnight airdrop ${z} --shielded`)}if(O.update("Checking dust..."),await l0(R,(o)=>O.update(o)),X?.aborted)throw new Error("Operation cancelled");O.update("Building shielded transaction...");let u=await R.facade.transferTransaction([{type:"shielded",outputs:[{type:j,amount:Y,receiverAddress:L}]}],{shieldedSecretKeys:R.zswapSecretKeys,dustSecretKey:R.dustSecretKey},{ttl:new Date(Date.now()+3600000)});O.update("Signing...");let p=await R.facade.signRecipe(u,(o)=>R.keystore.signData(o));O.update("Generating ZK proof (this may take a few minutes)...");let L0=await R.facade.finalizeRecipe(p),f=await W2(O,"Submitting and waiting for finalization (typically 12 to 30s)...",R.facade.submitTransaction(L0));return String(f)},{syncMode:"full",requireStrictSync:!0,signal:X,onStatus:(R)=>O.update(R),onSyncProgress:(R,T)=>{if(T>0){let v=Math.min(Math.round(R/T*100),100);O.update(v>=100?"Syncing wallet...":`Syncing wallet... ${v}%`)}},onSyncDetail:(R)=>O.update(`Syncing wallet... (waiting on: ${R})`)});if(O.stop("Transaction submitted"),C($,"json")){x({txHash:_,amount:z,recipient:Z,network:q,type:"shielded"});return}process.stdout.write(_+`
|
|
240
|
+
`),process.stderr.write(`
|
|
241
|
+
`+Q4(`Transferred ${z} shielded NIGHT`,_)+`
|
|
242
|
+
`),process.stderr.write(`
|
|
243
|
+
`+d()+`
|
|
244
|
+
`),process.stderr.write(U(" Verify: midnight balance --shielded")+`
|
|
245
|
+
|
|
246
|
+
`)}catch(_){throw O.fail("Failed"),_}finally{F(),W()}}var b8=N(()=>{H4();b0();V0();y0();$2();P4();i0();F4();H0();n0()});var Z0;var Y4=N(()=>{Z0=class Z0 extends Error{constructor($){super($);this.name="UsageError",Object.setPrototypeOf(this,Z0.prototype)}}});var k8={};k(k8,{default:()=>S8});async function S8($,Z){let Q=$.subcommand;if(!Q||Q!=="register"&&Q!=="status")throw new Z0(`Missing or invalid subcommand.
|
|
171
247
|
Usage:
|
|
172
248
|
midnight dust register Register NIGHT UTXOs for dust generation
|
|
173
|
-
midnight dust status Check dust registration status`);let X=
|
|
174
|
-
`+
|
|
175
|
-
|
|
176
|
-
`),process.stderr.write(L("Network",Z)+`
|
|
249
|
+
midnight dust status Check dust registration status`);let X=Q0(A($,"wallet")),z=z0(X),Y=Buffer.from(z.seed,"hex"),{name:G,config:J}=q0({args:$}),K=z.addresses[G];if(k0(J,{proofServer:A($,"proof-server"),node:A($,"node"),indexerWS:A($,"indexer-ws")}),S0($))w0();let q=C($,"json"),V=s0($);if(Q==="status"){let L=await EQ(K,G,J.indexerWS,Z);if(!L.isRegistered){NQ(G,L,q,V);return}let W=!C($,"no-cache");await xQ(Y,J,L,q,V,W,Z);return}A4($);let B={},H=j0((L,W)=>B.current?.(L,W)),j=T0();try{await CQ(Y,J,q,Z,B)}finally{j(),H()}}async function EQ($,Z,Q,X){process.stderr.write(`
|
|
250
|
+
`+g("Dust Status")+`
|
|
177
251
|
|
|
178
|
-
`)
|
|
179
|
-
`),J.alreadyAvailable)process.stderr.write(`
|
|
180
|
-
`+B0(`Dust tokens already available: ${e0(j)}`)+`
|
|
252
|
+
`),process.stderr.write(M("Network",Z)+`
|
|
181
253
|
|
|
254
|
+
`);let z=i(`Checking registration on ${Z}...`);try{let Y=await h4($,Q,(K,q)=>{if(q>0){let V=Math.min(Math.round(K/q*100),100);z.update(V>=100?"Checking registration...":`Checking registration... ${V}%`)}});if(X?.aborted)throw new Error("Operation cancelled");let{registeredUtxos:G,unregisteredUtxos:J}=Y;return z.stop("Registration checked"),{isRegistered:G>0,registeredUtxos:G,unregisteredUtxos:J}}catch(Y){throw z.fail("Failed"),Y}}function NQ($,Z,Q,X){let{registeredUtxos:z,unregisteredUtxos:Y}=Z,G=z+Y;if(Q){if(X){x({network:$,registered:!1,registeredUtxos:z,unregisteredUtxos:Y});return}x({subcommand:"status",registered:!1,registeredUtxos:z,unregisteredUtxos:Y,network:$});return}if(process.stdout.write(`registered=no
|
|
255
|
+
`),process.stdout.write(`registered_utxos=${z}
|
|
256
|
+
`),process.stdout.write(`unregistered_utxos=${Y}
|
|
257
|
+
`),process.stderr.write(M("Registered",y("no"))+`
|
|
258
|
+
`),process.stderr.write(M("UTXOs",`${z} registered, ${Y} unregistered`)+`
|
|
259
|
+
`),G===0)process.stderr.write(`
|
|
260
|
+
`+U("No NIGHT UTXOs at this address. Fund the wallet first.")+`
|
|
182
261
|
`);else process.stderr.write(`
|
|
183
|
-
`+
|
|
262
|
+
`+U("Not generating dust. Run: midnight dust register")+`
|
|
263
|
+
`);process.stderr.write(`
|
|
264
|
+
`+d()+`
|
|
184
265
|
|
|
185
|
-
`)}
|
|
186
|
-
`+
|
|
266
|
+
`)}async function CQ($,Z,Q,X,z){let Y=Z.networkId.toLowerCase();process.stderr.write(`
|
|
267
|
+
`+g("Dust Register")+`
|
|
187
268
|
|
|
188
|
-
`),process.stderr.write(
|
|
269
|
+
`),process.stderr.write(M("Network",Y)+`
|
|
189
270
|
|
|
190
|
-
`);let G=
|
|
191
|
-
`),process.stdout.write(`registered=${W}
|
|
192
|
-
`),process.stdout.write(`unregistered=${U.length}
|
|
193
|
-
`),process.stderr.write(L("NIGHT Balance",R(a0(V)))+`
|
|
194
|
-
`),process.stderr.write(L("Dust Balance",R(e0(J)))+`
|
|
195
|
-
`),process.stderr.write(L("Dust Available",B?"yes":"no")+`
|
|
196
|
-
`),process.stderr.write(L("Registered",W.toString()+" UTXO(s)")+`
|
|
197
|
-
`),process.stderr.write(L("Unregistered",U.length.toString()+" UTXO(s)")+`
|
|
271
|
+
`);let G=i("Syncing wallet...");if(z)z.current=(J,K)=>G.update(`Syncing wallet... (${K}, retrying)`);try{let{result:J,dustBal:K}=await _0().withFacade($,Z,async({bundle:q,state:V})=>{if(X?.aborted)throw new Error("Operation cancelled");G.update("Checking dust status...");let B=await l0(q,(j)=>G.update(j),V);if(X?.aborted)throw new Error("Operation cancelled");let H=await Z8(q);return{result:B,dustBal:H.dust.balance(new Date)}},{syncMode:"lite",requireStrictSync:!0,signal:X,onStatus:(q)=>G.update(q),onSyncProgress:(q,V)=>{if(V>0){let B=Math.min(Math.round(q/V*100),100);G.update(B>=100?"Syncing wallet...":`Syncing wallet... ${B}%`)}},onSyncDetail:(q)=>G.update(`Syncing wallet... (waiting on: ${q})`)});if(G.stop(J.alreadyAvailable?"Dust already available":"Dust registration complete"),Q){let q={subcommand:"register",dustBalance:J1(K)};if(J.txHash)q.txHash=J.txHash;x(q);return}process.stdout.write(K.toString()+`
|
|
198
272
|
`),process.stderr.write(`
|
|
199
|
-
`+
|
|
200
|
-
|
|
201
|
-
`)}catch(
|
|
202
|
-
`)
|
|
203
|
-
`)
|
|
204
|
-
`)
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
`)
|
|
208
|
-
`)
|
|
209
|
-
|
|
210
|
-
`)
|
|
273
|
+
`+Q4(J.alreadyAvailable?`Dust tokens already available: ${t1(K)}`:`Dust tokens available: ${t1(K)}`)+`
|
|
274
|
+
|
|
275
|
+
`)}catch(J){throw G.fail("Failed"),J}}async function xQ($,Z,Q,X,z,Y,G){let J=Z.networkId.toLowerCase(),K=i(`Reading dust events from ${J}...`);try{let q=await _0().dust($,Z,{forceFresh:!Y,signal:G,onStatus:(H)=>K.update(H)});K.stop(q.fromCache&&q.eventsApplied===0?"Cache up to date":"Dust events applied");let{registeredUtxos:V,unregisteredUtxos:B}=Q;if(X){if(z){x({network:J,registered:!0,registeredUtxos:V,unregisteredUtxos:B,dustBalance:J1(q.balance),dustAvailable:q.availableCoins>0});return}x({subcommand:"status",registered:!0,registeredUtxos:V,unregisteredUtxos:B,dustBalance:J1(q.balance),dustAvailable:q.availableCoins>0,eventsApplied:q.eventsApplied,ownedUtxos:q.ownedUtxoCount,cached:q.fromCache,network:J});return}if(process.stdout.write(`registered=yes
|
|
276
|
+
`),process.stdout.write(`dust=${q.balance}
|
|
277
|
+
`),process.stdout.write(`registered_utxos=${V}
|
|
278
|
+
`),process.stdout.write(`unregistered_utxos=${B}
|
|
279
|
+
`),process.stderr.write(M("Registered",y("yes"))+`
|
|
280
|
+
`),process.stderr.write(M("UTXOs",`${V} registered, ${B} unregistered`)+`
|
|
281
|
+
`),process.stderr.write(M("Dust Balance",y(t1(q.balance)))+`
|
|
282
|
+
`),process.stderr.write(M("Dust Available",q.availableCoins>0?"yes":"no")+`
|
|
283
|
+
`),process.stderr.write(M("Events applied",q.eventsApplied.toString()+(q.fromCache?" (delta)":""))+`
|
|
284
|
+
`),B>0)process.stderr.write(`
|
|
285
|
+
`+U(`${B} UTXO(s) not yet registered. Run: midnight dust register`)+`
|
|
286
|
+
`);process.stderr.write(`
|
|
287
|
+
`+d()+`
|
|
288
|
+
|
|
289
|
+
`)}catch(q){throw K.fail("Failed"),q}}var w8=N(()=>{Y4();H4();b0();V0();y0();i0();P4();K1();F4();H0();n0()});var h8={};k(h8,{default:()=>f8});async function f8($){if($.subcommand!=="clear")throw new Z0("Usage: midnight cache clear [--network <name>] [--wallet <name|file>]");let Q=A($,"wallet"),X=A($,"network"),z=C($,"json");if(Q){let Y=Q0(Q),G=z0(Y),{name:J}=q0({args:$}),K=G.addresses[J];c0(K,J);let q=U4(Buffer.from(G.seed,"hex"));if(j4(J,q),z){x({action:"clear",scope:"wallet",wallet:Q,network:J});return}process.stderr.write(m("✓")+` Cache cleared for wallet "${Q}" on ${J}
|
|
290
|
+
`)}else if(X){if(c0(void 0,X),j4(X),z){x({action:"clear",scope:"network",network:X});return}process.stderr.write(m("✓")+` Cache cleared for network "${X}"
|
|
291
|
+
`)}else{if(c0(),j4(),z){x({action:"clear",scope:"all"});return}process.stderr.write(m("✓")+` Cache cleared
|
|
292
|
+
`)}}var m8=N(()=>{Y4();O4();W1();b0();V0()});var u8={};k(u8,{default:()=>g8});async function g8($){let Z=$.subcommand;if(!Z||!["get","set","unset"].includes(Z))throw new Z0(`Usage: midnight config <get|set|unset> <key> [value]
|
|
293
|
+
Valid keys: ${i1().join(", ")}`);let Q=$.positionals[0];if(!Q)throw new Z0(`Missing config key.
|
|
294
|
+
Valid keys: ${i1().join(", ")}`);if(Z==="get"){let X=e2(Q);if(C($,"json")){x({action:"get",key:Q,value:X});return}process.stdout.write(X+`
|
|
295
|
+
`)}else if(Z==="unset"){Z$(Q);let X=Q==="network"?"(default)":"(removed)";if(C($,"json")){x({action:"unset",key:Q,value:X});return}process.stderr.write(m("✓")+` ${Q} ${X}
|
|
296
|
+
`)}else{let X=$.positionals[1];if(X===void 0)throw new Z0(`Missing value for config set.
|
|
297
|
+
Usage: midnight config set ${Q} <value>`);if($$(Q,X),C($,"json")){x({action:"set",key:Q,value:X});return}process.stderr.write(m("✓")+` ${Q} = ${X}
|
|
298
|
+
`)}}var p8=N(()=>{Y4();y4()});import{execSync as _1}from"child_process";import{existsSync as c8,mkdirSync as vQ,readFileSync as yQ,writeFileSync as l8}from"fs";import{homedir as bQ}from"os";import{join as N$}from"path";function j2(){try{return _1("docker compose version",{...L2,timeout:1e4}).trim()}catch{try{throw _1("docker --version",{...L2,timeout:5000}),new Error(`Docker Compose v2 is required.
|
|
211
299
|
Install it from https://docs.docker.com/compose/install/`)}catch($){if($ instanceof Error&&$.message.includes("Docker Compose v2"))throw $;throw new Error(`Docker is required but was not found.
|
|
212
|
-
Install Docker from https://docs.docker.com/get-docker/`)}}}function
|
|
213
|
-
`),Q=[];for(let X of Z){if(!X.trim())continue;try{let z=JSON.parse(X),
|
|
300
|
+
Install Docker from https://docs.docker.com/get-docker/`)}}}function O2(){if(c8(E$)&&c8(U2)&&yQ(E$,"utf-8").trim()===d8)return!1;return vQ(C$,{recursive:!0,mode:A0}),l8(U2,fQ,"utf-8"),l8(E$,d8,"utf-8"),!0}function a0($){return _1(`docker compose -f "${U2}" ${$}`,L2)}function I4(){try{let $=a0("ps --format json");if(!$.trim())return[];let Z=$.trim().split(`
|
|
301
|
+
`),Q=[];for(let X of Z){if(!X.trim())continue;try{let z=JSON.parse(X),Y=z.Service??"unknown";Q.push({name:Y,state:z.State??"unknown",health:z.Health??"",port:hQ[Y]??"",image:z.Image??""})}catch{}}return Q}catch{return[]}}function F2($=120000,Z=3000){let Q=Date.now()+$;while(Date.now()<Q){let X=I4();if(X.length===3&&X.every((Y)=>Y.state==="running")){if(X.every((G)=>G.health==="healthy"||G.health===""))return!0}_1(`sleep ${Z/1000}`,{timeout:Z+1000})}return!1}function x$(){return U2}function i8(){let $=[];for(let Z of mQ)try{_1(`docker rm -f "${Z}"`,{...L2,timeout:1e4}),$.push(Z)}catch{}return $}var d8="3.0.1",SQ="midnightntwrk/midnight-node:0.22.3",kQ="midnightntwrk/indexer-standalone:4.0.1",wQ="midnightntwrk/proof-server:8.0.3",C$,U2,E$,fQ,L2,hQ,mQ;var v$=N(()=>{C$=N$(bQ(),K0,n2),U2=N$(C$,"compose.yml"),E$=N$(C$,".version"),fQ=`services:
|
|
214
302
|
proof-server:
|
|
215
|
-
image: '
|
|
303
|
+
image: '${wQ}'
|
|
216
304
|
container_name: "proof-server"
|
|
217
305
|
ports:
|
|
218
306
|
- "6300:6300"
|
|
219
307
|
|
|
220
308
|
indexer:
|
|
221
|
-
image: '
|
|
309
|
+
image: '${kQ}'
|
|
222
310
|
container_name: "indexer"
|
|
223
311
|
ports:
|
|
224
312
|
- '8088:8088'
|
|
@@ -240,7 +328,7 @@ Install Docker from https://docs.docker.com/get-docker/`)}}}function u3(){if(w3(
|
|
|
240
328
|
condition: service_started
|
|
241
329
|
|
|
242
330
|
node:
|
|
243
|
-
image: '
|
|
331
|
+
image: '${SQ}'
|
|
244
332
|
container_name: "node"
|
|
245
333
|
ports:
|
|
246
334
|
- "9944:9944"
|
|
@@ -252,34 +340,38 @@ Install Docker from https://docs.docker.com/get-docker/`)}}}function u3(){if(w3(
|
|
|
252
340
|
start_period: 5s
|
|
253
341
|
environment:
|
|
254
342
|
CFG_PRESET: "dev"
|
|
255
|
-
`,
|
|
256
|
-
`)}async function
|
|
257
|
-
`);let Q=
|
|
258
|
-
`+
|
|
259
|
-
`);else
|
|
343
|
+
`,L2={encoding:"utf-8",timeout:30000};hQ={node:"9944",indexer:"8088","proof-server":"6300"};mQ=["node","indexer","proof-server"]});var o8={};k(o8,{waitForFirstBlock:()=>gQ});async function gQ($,Z={}){let Q=Z.timeoutMs??30000,X=Z.pollIntervalMs??1000,z=Date.now()+Q;while(Date.now()<z){try{let Y=await g4({url:$,timeoutMs:3000},"chain_getHeader"),G=typeof Y?.number==="string"?Y.number:"0x0",J=Number.parseInt(G,16);if(Number.isFinite(J)&&J>=1)return}catch{}await new Promise((Y)=>setTimeout(Y,X))}throw new Error(`Node did not produce a block within ${Q/1000}s`)}var n8=N(()=>{Q2()});var a8={};k(a8,{waitForAddressFunded:()=>uQ});async function uQ($,Z,Q={}){let X=Q.timeoutMs??30000,z=Q.pollIntervalMs??1000,Y=Date.now()+X;while(Date.now()<Y){try{if((await h4(Z,$)).utxoCount>0)return}catch{}await new Promise((G)=>setTimeout(G,z))}throw new Error(`Indexer did not report funds for ${Z.slice(0,20)}… within ${X/1000}s`)}var r8=N(()=>{K1()});var s8={};k(s8,{waitForProofServerReady:()=>pQ});async function pQ($,Z={}){let Q=Z.timeoutMs??30000,X=Z.pollIntervalMs??1000,z=Z.requestTimeoutMs??2000,Y=Date.now()+Q,G=$.endsWith("/")?$:$+"/";while(Date.now()<Y){try{let J=new AbortController,K=setTimeout(()=>J.abort(),z);try{if((await fetch(G,{method:"GET",signal:J.signal})).ok)return}finally{clearTimeout(K)}}catch{}await new Promise((J)=>setTimeout(J,X))}throw new Error(`Proof server at ${G} did not respond within ${Q/1000}s`)}var Q7={};k(Q7,{default:()=>Z7});import{spawn as t8}from"child_process";import{existsSync as cQ,rmSync as lQ}from"node:fs";import{resolve as dQ}from"node:path";function iQ($){return e8.includes($)}function $7($){let Z=[];for(let Q of $){let X=Q.state==="running"?m:X0,z=Q.health?` (${Q.health})`:"",Y=Q.port?`:${Q.port}`:"",G=Q.image?` ${U(Q.image)}`:"";if(Z.push(` ${Q.name.padEnd(16)}${X(Q.state)}${U(z)}${U(Y)}`),G)Z.push(G)}return Z.join(`
|
|
344
|
+
`)}async function oQ($){if(O2())process.stderr.write(U(` Wrote compose.yml to ${x$()}`)+`
|
|
345
|
+
`);let Q=i("Starting local network...");try{if(a0("up -d"),Q.update("Waiting for services to be healthy..."),!F2(120000))Q.stop(e("Services started but not all healthy yet")),process.stderr.write(`
|
|
346
|
+
`+U(" Tip: run ")+y("midnight localnet logs")+U(" to check for errors")+`
|
|
347
|
+
`);else{Q.update("Waiting for chain to produce first block...");try{let{waitForFirstBlock:Y}=await Promise.resolve().then(() => (n8(),o8)),{waitForAddressFunded:G}=await Promise.resolve().then(() => (r8(),a8)),{waitForProofServerReady:J}=await Promise.resolve().then(() => s8),{getNetworkConfig:K}=await Promise.resolve().then(() => (y0(),D3)),{deriveUnshieldedAddress:q}=await Promise.resolve().then(() => (m0(),x3)),{GENESIS_SEED:V}=await Promise.resolve().then(() => l1),B=K("undeployed");await Y(B.node,{timeoutMs:30000}),Q.update("Waiting for indexer to ingest genesis state...");let H=q(Buffer.from(V,"hex"),"undeployed");await G(B.indexerWS,H,{timeoutMs:30000}),Q.update("Waiting for proof server..."),await J(B.proofServer,{timeoutMs:30000}),Q.stop("Local network is running")}catch(Y){Q.stop(e("Services running but chain/indexer/proof-server not fully ready yet")),process.stderr.write(`
|
|
348
|
+
`+U(" Tip: give it a few more seconds then retry. Run ")+y("midnight localnet logs")+U(" if this persists.")+`
|
|
349
|
+
`)}}}catch(z){if(Q.fail("Failed to start local network"),z instanceof Error){if(z.message.includes("is already in use by container"))throw new Error(`Container name conflict — containers with the same names already exist
|
|
260
350
|
`+`(likely from a previous midnight-local-network setup).
|
|
261
351
|
|
|
262
352
|
Run "midnight localnet clean" to remove them, then try again.`);if(z.message.includes("address already in use"))throw new Error(`Port conflict detected — another process is using a required port.
|
|
263
|
-
`+"Check ports 9944, 8088, and 6300, then try again.")}throw z}let X=
|
|
264
|
-
|
|
353
|
+
`+"Check ports 9944, 8088, and 6300, then try again.")}throw z}let X=I4();if($){x({subcommand:"up",services:X.map((z)=>({name:z.name,state:z.state,port:z.port,health:z.health,image:z.image}))});return}if(X.length>0)process.stderr.write(`
|
|
354
|
+
`+$7(X)+`
|
|
265
355
|
`);for(let z of X)process.stdout.write(`${z.name}=${z.state}:${z.port}
|
|
266
356
|
`);process.stderr.write(`
|
|
267
|
-
`+
|
|
268
|
-
`)}async function
|
|
269
|
-
|
|
357
|
+
`+U(" Next: ")+y("midnight wallet generate dev --network undeployed")+`
|
|
358
|
+
`)}async function nQ($){let Z=i("Stopping local network...");try{if(a0("stop"),Z.stop("Local network stopped (containers preserved)"),$){x({subcommand:"stop",status:"stopped"});return}}catch(Q){throw Z.fail("Failed to stop local network"),Q}}async function aQ($){let Z=i("Tearing down local network...");try{a0("down --volumes");let{clearWalletCache:Q}=await Promise.resolve().then(() => (O4(),X8)),{clearDustDirectCache:X}=await Promise.resolve().then(() => (W1(),r3));Q(void 0,"undeployed"),X("undeployed");let z=rQ();if(Z.stop("Local network removed (containers, networks, volumes, caches)"),$){x({subcommand:"down",status:"removed",cachesCleared:!0,dappLevelDbWiped:z});return}if(z)process.stderr.write(U(" Also cleared dapp leveldb at ./midnight-level-db (private state from old chain)")+`
|
|
359
|
+
`)}catch(Q){throw Z.fail("Failed to tear down local network"),Q}}function rQ(){let $=dQ(process.cwd(),"midnight-level-db");if(!cQ($))return!1;try{return lQ($,{recursive:!0,force:!0}),!0}catch{return!1}}async function sQ($){let Z=I4();if($){x({subcommand:"status",services:Z.map((Q)=>({name:Q.name,state:Q.state,port:Q.port,health:Q.health,image:Q.image}))});return}if(Z.length===0){process.stderr.write(`
|
|
360
|
+
`+g("Localnet Status")+`
|
|
270
361
|
|
|
271
|
-
`),process.stderr.write(
|
|
272
|
-
`),process.stderr.write(
|
|
362
|
+
`),process.stderr.write(U(" No services running.")+`
|
|
363
|
+
`),process.stderr.write(U(" Run ")+y("midnight localnet up")+U(" to start.")+`
|
|
273
364
|
|
|
274
365
|
`);return}process.stderr.write(`
|
|
275
|
-
`+
|
|
366
|
+
`+g("Localnet Status")+`
|
|
276
367
|
|
|
277
|
-
`),process.stderr.write(
|
|
368
|
+
`),process.stderr.write($7(Z)+`
|
|
278
369
|
`),process.stderr.write(`
|
|
279
|
-
`+
|
|
370
|
+
`+d()+`
|
|
280
371
|
|
|
281
372
|
`);for(let Q of Z)process.stdout.write(`${Q.name}=${Q.state}:${Q.port}
|
|
282
|
-
`)}async function
|
|
373
|
+
`)}async function tQ($){let Z=i("Removing conflicting containers...");try{try{a0("down")}catch{}let Q=i8();if(Q.length>0)Z.stop(`Removed ${Q.length} container${Q.length>1?"s":""}: ${Q.join(", ")}`);else Z.stop("No conflicting containers found");if($){x({subcommand:"clean",status:"cleaned",removed:Q});return}}catch(Q){throw Z.fail("Failed to clean up"),Q}}async function eQ($){let Z=x$(),Q=A($,"tail"),X=C($,"json");if(Q!==void 0||X){let Y=Q??"200",G=t8("docker",["compose","-f",Z,"logs","--tail",Y,"--no-color"],{stdio:["ignore","pipe","pipe"]}),J="",K="";return G.stdout.on("data",(q)=>{J+=q.toString()}),G.stderr.on("data",(q)=>{K+=q.toString()}),new Promise((q,V)=>{G.on("close",(B)=>{if(B!==0&&B!==null){V(new Error(`docker compose logs exited with code ${B}: ${K.trim()}`));return}if(X)x({tail:Number(Y),lines:J.split(`
|
|
374
|
+
`).filter(Boolean)});else process.stdout.write(J);q()}),G.on("error",V)})}let z=t8("docker",["compose","-f",Z,"logs","-f"],{stdio:"inherit"});return new Promise((Y,G)=>{z.on("close",(J)=>{if(J===0||J===130||J===null)Y();else G(new Error(`docker compose logs exited with code ${J}`))}),z.on("error",G)})}async function Z7($){let Z=$.subcommand;if(!Z||!iQ(Z))throw new Z0(`Usage: midnight localnet <${e8.join("|")}>
|
|
283
375
|
|
|
284
376
|
Subcommands:
|
|
285
377
|
up Start the local network
|
|
@@ -289,84 +381,885 @@ Subcommands:
|
|
|
289
381
|
logs Stream service logs
|
|
290
382
|
clean Remove conflicting containers
|
|
291
383
|
|
|
292
|
-
Example: midnight localnet up`);
|
|
293
|
-
`+
|
|
384
|
+
Example: midnight localnet up`);j2();let Q=C($,"json");switch(process.stderr.write(`
|
|
385
|
+
`+g("Localnet")+`
|
|
294
386
|
|
|
295
|
-
`),Z){case"up":return
|
|
296
|
-
Available: generate, list, use, info, remove
|
|
297
|
-
Run "midnight help wallet" for usage.`)}}async function
|
|
298
|
-
Usage: midnight wallet generate <name> [--network <name>]`);if(
|
|
299
|
-
Wallet name must be a simple name (no path separators, .json suffix, or special characters).`);let Q=
|
|
300
|
-
Use --force to overwrite.`);let
|
|
387
|
+
`),Z){case"up":return oQ(Q);case"stop":return nQ(Q);case"down":return aQ(Q);case"status":return sQ(Q);case"logs":return eQ($);case"clean":return tQ(Q)}}var e8;var X7=N(()=>{Y4();v$();H0();n0();e8=["up","stop","down","status","logs","clean"]});var J7={};k(J7,{default:()=>G7});import*as Y7 from"fs";import*as y$ from"path";import{homedir as $9}from"os";import{generateMnemonic as Z9,mnemonicToSeedSync as z7,mnemonicToEntropy as Q9,validateMnemonic as X9}from"@scure/bip39";import{wordlist as b$}from"@scure/bip39/wordlists/english.js";async function G7($){let Z=$.subcommand;switch(Z){case"generate":return z9($);case"list":case"ls":return Y9($);case"use":return G9($);case"info":return J9($);case"remove":case"rm":return q9($);case"seed":return K9($);default:throw new Z0(`Unknown wallet subcommand: "${Z??"(none)"}"
|
|
388
|
+
Available: generate, list, use, info, remove, seed
|
|
389
|
+
Run "midnight help wallet" for usage.`)}}async function z9($){let Z=$.positionals[0];if(!Z)throw new Error(`Missing wallet name.
|
|
390
|
+
Usage: midnight wallet generate <name> [--network <name>]`);if(!$4(Z))throw new Error(`Invalid wallet name: "${Z}"
|
|
391
|
+
Wallet name must be a simple name (no path separators, .json suffix, or special characters).`);let Q=P0({args:$}),X=A($,"seed"),z=A($,"mnemonic");if(X!==void 0&&z!==void 0)throw new Error("Cannot specify both --seed and --mnemonic. Use one or the other.");let Y=y$.join($9(),K0,e4),G=y$.join(Y,`${Z}.json`);if(Y7.existsSync(G)&&!C($,"force"))throw new Error(`Wallet "${Z}" already exists: ${G}
|
|
392
|
+
Use --force to overwrite.`);let J,K;if(X!==void 0){let W=X.replace(/^0x/,"");if(W.length!==64||!/^[0-9a-fA-F]+$/.test(W))throw new Error("Seed must be a 64-character hex string (32 bytes)");J=Buffer.from(W,"hex")}else if(z!==void 0){if(!X9(z,b$))throw new Error("Invalid BIP-39 mnemonic. Expected 12 or 24 words from the English wordlist.");K=z,J=Buffer.from(z7(K))}else K=Z9(b$,256),J=Buffer.from(z7(K));let q=Z4(J),V=S4(J),B=q[Q],H={seed:J.toString("hex"),addresses:q,shieldedAddresses:V,createdAt:new Date().toISOString()};if(K)H.mnemonic=K;let j=X1(H,G);if(r1(Z),C($,"json")){let W={name:Z,addresses:q,shieldedAddresses:V,activeAddress:B,activeNetwork:Q,seed:J.toString("hex"),file:j,createdAt:H.createdAt,active:!0};if(K)W.mnemonic=K;x(W);return}process.stdout.write(B+`
|
|
301
393
|
`),process.stderr.write(`
|
|
302
|
-
`+
|
|
394
|
+
`+g("Wallet Generated")+`
|
|
303
395
|
|
|
304
|
-
`),process.stderr.write(
|
|
305
|
-
`),process.stderr.write(
|
|
306
|
-
`),process.stderr.write(
|
|
307
|
-
`)
|
|
308
|
-
`)
|
|
396
|
+
`),process.stderr.write(M("Name",Z)+`
|
|
397
|
+
`),process.stderr.write(M("Network",Q)+`
|
|
398
|
+
`),process.stderr.write(M("Address",s(B))+`
|
|
399
|
+
`);let L=V[Q];if(L)process.stderr.write(M("Shielded",s(L))+`
|
|
400
|
+
`);if(process.stderr.write(M("File",j)+`
|
|
401
|
+
`),process.stderr.write(M("Active","yes")+`
|
|
309
402
|
`),process.stderr.write(`
|
|
310
|
-
`),K)process.stderr.write(
|
|
403
|
+
`),K)process.stderr.write(e(y(" MNEMONIC (save securely!):"))+`
|
|
311
404
|
`),process.stderr.write(` ${K}
|
|
312
405
|
|
|
313
|
-
`);process.stderr.write(
|
|
314
|
-
`),process.stderr.write(` ${
|
|
406
|
+
`);process.stderr.write(e(y(" SEED (hex):"))+`
|
|
407
|
+
`),process.stderr.write(` ${J.toString("hex")}
|
|
315
408
|
|
|
316
|
-
`),process.stderr.write(
|
|
317
|
-
`),process.stderr.write(
|
|
409
|
+
`),process.stderr.write(d()+`
|
|
410
|
+
`),process.stderr.write(U(" Next: midnight wallet list | midnight balance")+`
|
|
318
411
|
|
|
319
|
-
`),process.stderr.write(
|
|
320
|
-
`)}async function
|
|
412
|
+
`),process.stderr.write(m("✓")+` Wallet saved
|
|
413
|
+
`)}async function Y9($){let Z=q$(),Q=P0({args:$});if(C($,"json")){if(s0($)){x({activeNetwork:Q,wallets:Z.map((J)=>({name:J.name,active:J.isActive,network:Q,address:J.addresses[Q]??null,shieldedAddress:J.shieldedAddresses?.[Q]??null}))});return}x({activeNetwork:Q,wallets:Z.map((J)=>({name:J.name,active:J.isActive,addresses:J.addresses,shieldedAddresses:J.shieldedAddresses,createdAt:J.createdAt,file:J.file}))});return}if(Z.length===0){process.stderr.write(`
|
|
321
414
|
No wallets found.
|
|
322
|
-
`),process.stderr.write(
|
|
415
|
+
`),process.stderr.write(U(" Create one: midnight wallet generate <name> --network <name>")+`
|
|
323
416
|
|
|
324
|
-
`);return}
|
|
325
|
-
`+
|
|
417
|
+
`);return}process.stderr.write(`
|
|
418
|
+
`+g("Wallets")+`
|
|
419
|
+
|
|
420
|
+
`);let X=Math.max(...Z.map((J)=>J.name.length)),z=19,Y=19,G=" ".repeat(X-4);process.stderr.write(` ${U("name")}${G} ${U("unshielded".padEnd(z))} ${U("shielded")}
|
|
421
|
+
`);for(let J of Z){let K=J.isActive?m("●"):" ",q=J.isActive?y(l(J.name)):l(J.name),V=" ".repeat(X-J.name.length),B=J.addresses[Q]??"(unknown)",H=J.shieldedAddresses?.[Q],j=H?s(H,!0):U("—".padEnd(Y));process.stderr.write(` ${K} ${q}${V} ${s(B,!0)} ${j}
|
|
422
|
+
`)}process.stderr.write(`
|
|
423
|
+
`+d()+`
|
|
424
|
+
`),process.stderr.write(U(` ● = active wallet · network: ${Q} · details: midnight wallet info <name>`)+`
|
|
425
|
+
|
|
426
|
+
`)}async function G9($){let Z=$.positionals[0];if(!Z)throw new Error(`Missing wallet name.
|
|
427
|
+
Usage: midnight wallet use <name>`);if(r1(Z),C($,"json")){x({wallet:Z,active:!0});return}process.stderr.write(m("✓")+` Active wallet set to "${Z}"
|
|
428
|
+
`)}async function J9($){let Z=$.positionals[0]??V4(),Q=Q0(Z),X=z0(Q),z=Z===V4(),Y=P0({args:$}),G=X.addresses[Y];if(C($,"json")){if(s0($)){x({name:Z,active:z,network:Y,address:G,shieldedAddress:X.shieldedAddresses?.[Y]??null});return}let K={name:Z,addresses:X.addresses,activeAddress:G,activeNetwork:Y,createdAt:X.createdAt,file:Q,active:z};if(X.shieldedAddresses)K.shieldedAddresses=X.shieldedAddresses;x(K);return}process.stdout.write(G+`
|
|
429
|
+
`),process.stderr.write(`
|
|
430
|
+
`+g(`Wallet: ${Z}`)+`
|
|
326
431
|
|
|
327
|
-
`);
|
|
432
|
+
`);let J=Object.keys(X.addresses);for(let K=0;K<J.length;K++){let q=J[K],V=q===Y,B=X.addresses[q],H=X.shieldedAddresses?.[q],j=V?y(l(q))+U(" (active)"):q;if(process.stderr.write(` ${j}
|
|
433
|
+
`),process.stderr.write(` ${U("unshielded")} ${s(B)}
|
|
434
|
+
`),H)process.stderr.write(` ${U("shielded ")} ${s(H)}
|
|
435
|
+
`);else process.stderr.write(` ${U("shielded ")} ${U("(unavailable)")}
|
|
436
|
+
`);if(K<J.length-1)process.stderr.write(`
|
|
328
437
|
`)}process.stderr.write(`
|
|
329
|
-
`+
|
|
330
|
-
|
|
438
|
+
`+d()+`
|
|
439
|
+
|
|
440
|
+
`),process.stderr.write(` ${U("created")} ${X.createdAt}
|
|
441
|
+
`),process.stderr.write(` ${U("file ")} ${Q}
|
|
442
|
+
`),process.stderr.write(` ${U("active ")} ${z?m("yes"):"no"}
|
|
443
|
+
|
|
444
|
+
`)}async function q9($){let Z=$.positionals[0];if(!Z)throw new Error(`Missing wallet name.
|
|
445
|
+
Usage: midnight wallet remove <name>`);if(K$(Z),C($,"json")){x({wallet:Z,removed:!0});return}process.stderr.write(m("✓")+` Wallet "${Z}" removed
|
|
446
|
+
`)}async function K9($){let Z=$.positionals[0]??V4(),Q=Q0(Z),X=await import("fs"),z=JSON.parse(X.readFileSync(Q,"utf-8"));if(!z.seed)throw new Error(`Wallet "${Z}" has no seed in ${Q}`);let Y=C($,"entropy"),G;if(Y){if(!z.mnemonic)throw new Error(`Wallet "${Z}" has no mnemonic on file — cannot derive BIP-39 entropy.
|
|
447
|
+
`+"The 32-byte entropy can only be computed from the original 24-word mnemonic.");G=Buffer.from(Q9(z.mnemonic,b$)).toString("hex")}if(C($,"json")){let K={name:Z,seed:z.seed};if(z.mnemonic)K.mnemonic=z.mnemonic;if(G)K.entropy=G;x(K);return}if(process.stderr.write(`
|
|
448
|
+
`),process.stderr.write(e(y(" ⚠ WARNING: This will display your wallet seed and mnemonic."))+`
|
|
449
|
+
`),process.stderr.write(e(" Anyone with this seed can access your funds.")+`
|
|
450
|
+
`),process.stderr.write(e(" Never share it. Never paste it into a website.")+`
|
|
451
|
+
`),process.stderr.write(`
|
|
452
|
+
`),!await V9(" Show seed? (y/N) ")){process.stderr.write(U(" Cancelled.")+`
|
|
453
|
+
|
|
454
|
+
`);return}if(process.stdout.write(z.seed+`
|
|
455
|
+
`),process.stderr.write(`
|
|
456
|
+
`),process.stderr.write(M("Name",Z)+`
|
|
457
|
+
`),process.stderr.write(M("Seed (64-byte PBKDF2)",z.seed)+`
|
|
458
|
+
`),G)process.stderr.write(M("Entropy (32-byte BIP-39)",G)+`
|
|
459
|
+
`);if(z.mnemonic)process.stderr.write(M("Mnemonic",z.mnemonic)+`
|
|
460
|
+
`);if(process.stderr.write(M("File",Q)+`
|
|
461
|
+
`),G)process.stderr.write(`
|
|
462
|
+
`),process.stderr.write(U(" Note: seed and entropy derive DIFFERENT Midnight wallets from the")+`
|
|
463
|
+
`),process.stderr.write(U(" same mnemonic. Use the value your target project accepts.")+`
|
|
464
|
+
`);process.stderr.write(`
|
|
465
|
+
`)}function V9($){return new Promise((Z)=>{let X=r0("readline").createInterface({input:process.stdin,output:process.stderr});X.question($,(z)=>{X.close(),Z(z.trim().toLowerCase()==="y")})})}var q7=N(()=>{Y4();m0();V0();b0();H0()});var W7={};k(W7,{default:()=>H7});function k$($){let Z=Date.now()-new Date($).getTime();if(Z<60000)return"just now";if(Z<3600000)return`${Math.floor(Z/60000)}m ago`;if(Z<86400000)return`${Math.floor(Z/3600000)}h ago`;return`${Math.floor(Z/86400000)}d ago`}function S$($){let Z=!1,Q=!1;for(let X of Object.values($)){if(X.status==="down")Z=!0;if(X.status==="degraded")Q=!0}if(Z)return"down";if(Q)return"degraded";return"up"}async function P2($){let Z=`${$1}${$}`,Q=await fetch(Z,{signal:AbortSignal.timeout(1e4)});if(!Q.ok)throw new Error(`Dashboard returned HTTP ${Q.status} for ${$}`);return Q.json()}async function U9($){let Z=A($,"network");if(Z)return Z;try{let{loadCliConfig:Q}=await Promise.resolve().then(() => (y4(),M3)),X=Q();if(X.network&&F0(X.network))return X.network}catch{}return}async function _2($,Z={}){let Q=new AbortController,X=setTimeout(()=>Q.abort(),W9),z=Date.now();try{return{response:await fetch($,{...Z,signal:Q.signal}),latencyMs:Date.now()-z}}finally{clearTimeout(X)}}async function L9($){let Z=new Date().toISOString();try{let{response:Q,latencyMs:X}=await _2($,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({query:"{ block { height } }"})});if(!Q.ok){if(Q.status===403)return{status:"degraded",latencyMs:X,lastChecked:Z,notes:"HTTP 403 — likely WAF blocking"};return{status:"down",latencyMs:X,lastChecked:Z,notes:`HTTP ${Q.status}`}}let z=await Q.json();if(!z?.data)return{status:"down",latencyMs:X,lastChecked:Z,notes:"No data field"};let Y=z.data?.block?.height;return{status:X>I2?"degraded":"up",latencyMs:X,lastChecked:Z,...Y!==void 0?{blockHeight:Y}:{},...X>I2?{notes:`Slow (${X}ms)`}:{}}}catch(Q){return{status:"down",latencyMs:0,lastChecked:Z,notes:Q.message}}}async function j9($){let Z=new Date().toISOString(),Q=$.replace(/^wss:/,"https:").replace(/^ws:/,"http:");try{let{response:X,latencyMs:z}=await _2(Q,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({jsonrpc:"2.0",id:1,method:"system_health",params:[]})});if(!X.ok){if(X.status===403)return{status:"degraded",latencyMs:z,lastChecked:Z,notes:"HTTP 403 — likely WAF blocking"};return{status:"down",latencyMs:z,lastChecked:Z,notes:`HTTP ${X.status}`}}let G=(await X.json())?.result;if(!G)return{status:"down",latencyMs:z,lastChecked:Z,notes:"No result in response"};let J=G.peers??0,K=G.isSyncing??!1,q="up",V;if(K)q="degraded",V="Node is syncing";else if(J===0)q="degraded",V="No peers";return{status:q,latencyMs:z,lastChecked:Z,peers:J,isSyncing:K,...V?{notes:V}:{}}}catch(X){return{status:"down",latencyMs:0,lastChecked:Z,notes:X.message}}}async function O9($){let Z=new Date().toISOString();if(!$)return{status:"unknown",latencyMs:0,lastChecked:Z,notes:"No faucet URL"};try{let{response:Q,latencyMs:X}=await _2($);if(!Q.ok)return{status:"down",latencyMs:X,lastChecked:Z,notes:`HTTP ${Q.status}`};return{status:X>V7?"degraded":"up",latencyMs:X,lastChecked:Z,...X>V7?{notes:`Slow (${X}ms)`}:{}}}catch(Q){return{status:"down",latencyMs:0,lastChecked:Z,notes:Q.message}}}async function F9($){let Z=new Date().toISOString();if(!$)return{status:"unknown",latencyMs:0,lastChecked:Z,notes:"URL not configured"};try{let{response:Q,latencyMs:X}=await _2($);if(!Q.ok)return{status:"down",latencyMs:X,lastChecked:Z,notes:`HTTP ${Q.status}`};let z=await Q.text();if(!z||z.length===0)return{status:"down",latencyMs:X,lastChecked:Z,notes:"Empty response"};return{status:X>I2?"degraded":"up",latencyMs:X,lastChecked:Z,...X>I2?{notes:`Slow (${X}ms)`}:{}}}catch(Q){return{status:"down",latencyMs:0,lastChecked:Z,notes:Q.message}}}async function B7($){if(!F0($))return{};let Z=o1($),Q=B9[$];if(!Q)return{};let[X,z,Y,G]=await Promise.all([L9(Z.indexer),j9(Z.node),O9(Q.faucet),F9(Q.explorer)]);return{indexer:X,rpc:z,faucet:Y,explorer:G}}function P9($,Z,Q){process.stderr.write(`
|
|
466
|
+
`+y($)+`
|
|
467
|
+
`),process.stderr.write(U(" "+"─".repeat(62))+`
|
|
468
|
+
`);let X=new Set([...Object.keys(Q),...Z?Object.keys(Z):[]]);for(let z of X){let Y=Q[z],G=Z?.[z],J=(H9[z]??z).padEnd(18);if(Y){let K=K7[Y.status]??U("—"),q=Y.latencyMs>0?U(` ${Y.latencyMs}ms`):"",V=Y.notes?U(` — ${Y.notes}`):"",B="";if(G&&G.status!==Y.status&&G.status!=="unknown")B=U(` (canary: ${G.status}${G.lastChecked?" "+k$(G.lastChecked):""})`);process.stderr.write(` ${Y1(J)}${K}${q}${V}${B}
|
|
469
|
+
`)}else if(G){let K=K7[G.status]??U("—"),q=G.latencyMs>0?U(` ${G.latencyMs}ms`):"",V=G.lastChecked?U(` (${k$(G.lastChecked)})`):"",B=G.notes?U(` — ${G.notes}`):"";process.stderr.write(` ${Y1(J)}${K}${q}${V}${B}
|
|
470
|
+
`)}}}function I9($,Z){let Q=Z?$.filter((X)=>X.affects.toLowerCase().includes(Z.toLowerCase())):$;if(Q.length===0)return;process.stderr.write(`
|
|
471
|
+
`+g("Known Issues")+`
|
|
472
|
+
|
|
473
|
+
`);for(let X of Q)process.stderr.write(` ${X0(X.id.padEnd(28))}${X.summary}
|
|
474
|
+
`),process.stderr.write(` ${" ".repeat(28)}${U(X.status)}
|
|
475
|
+
`)}async function H7($){let Z=C($,"json"),Q=C($,"all"),X=C($,"watch"),z=await U9($)??"preprod",Y=i("Checking network status..."),G=null,J=null,K,q;if(Q)q=["preprod","preview"];else q=[z];let[V,...B]=await Promise.all([Promise.all([P2("/api/status"),P2("/api/issues")]).catch((W)=>{return K=W.message,null}),...q.map((W)=>B7(W))]);if(V)[G,J]=V;Y.stop("Done");let H={};for(let W=0;W<q.length;W++)H[q[W]]=B[W];if(Z){let W={lastUpdated:G?.lastUpdated??null,dashboard:$1,canaryAvailable:G!==null,networks:{}};for(let O of q){let _=H[O]??{},R=G?.networks[O],T={...R,..._};W.networks[O]={overall:S$(T),live:_,canary:R??null}}if(J){let O=Q?void 0:q[0];W.issues=O?J.issues.filter((_)=>_.affects.toLowerCase().includes(O.toLowerCase())):J.issues}if(K)W.canaryError=K;x(W);let F=q.reduce((O,_)=>{let R={...G?.networks[_]??{},...H[_]??{}},T=S$(R);if(T==="down")return"down";if(T==="degraded"&&O!=="down")return"degraded";return O},"up");if(F==="down")process.exitCode=2;else if(F==="degraded")process.exitCode=1;return}let j=()=>{if(process.stderr.write(`
|
|
476
|
+
`+g("Midnight Network Status")+`
|
|
477
|
+
`),G?.lastUpdated)process.stderr.write(U(` Canary: ${k$(G.lastUpdated)}`)+" ");if(process.stderr.write(U("Live: just now")+`
|
|
478
|
+
`),K)process.stderr.write(U(` (Dashboard unreachable: ${K})`)+`
|
|
479
|
+
`);if(q.length===0){process.stderr.write(`
|
|
480
|
+
`+U("No network data available.")+`
|
|
481
|
+
|
|
482
|
+
`);return}for(let W of q)P9(W,G?.networks[W],H[W]??{});if(J){let W=Q?void 0:q[0];I9(J.issues,W)}process.stderr.write(`
|
|
483
|
+
`+d()+`
|
|
484
|
+
`),process.stderr.write(U(" Dashboard: ")+l($1)+`
|
|
485
|
+
|
|
486
|
+
`)};if(j(),X){let W=setInterval(async()=>{try{let[F,...O]=await Promise.all([Promise.all([P2("/api/status"),P2("/api/issues")]).catch(()=>null),...q.map((_)=>B7(_))]);if(F)[G,J]=F;for(let _=0;_<q.length;_++)H[q[_]]=O[_];process.stderr.write("\x1B[2J\x1B[H"),j()}catch{process.stderr.write(U(" Refresh failed — retrying in 30s")+`
|
|
487
|
+
`)}},30000);await new Promise(()=>{process.on("SIGINT",()=>{clearInterval(W),process.exit(0)})})}let L=q.reduce((W,F)=>{let O={...G?.networks[F]??{},...H[F]??{}},_=S$(O);if(_==="down")return"down";if(_==="degraded"&&W!=="down")return"degraded";return W},"up");if(L==="down")process.exitCode=2;else if(L==="degraded")process.exitCode=1}var B9,H9,K7,W9=15000,I2=5000,V7=1e4;var U7=N(()=>{y0();H0();n0();B9={preprod:{faucet:"https://faucet.preprod.midnight.network/",explorer:"https://preprod.midnightexplorer.com/"},preview:{faucet:"https://faucet.preview.midnight.network/",explorer:null},undeployed:{faucet:"",explorer:""}},H9={indexer:"Indexer",rpc:"RPC Node",faucet:"Faucet",explorer:"Explorer",chain:"Chain",dust:"Dust Generation",wallet:"Wallet Ops",dapp:"DApp Flow"},K7={up:m("UP"),down:X0("DOWN"),degraded:y("\x1B[33mDEGRADED\x1B[0m"),unknown:U("—")}});var O7={};k(O7,{toJsonOutput:()=>g$,formatWitnessSignature:()=>m$,formatCompactType:()=>M0,formatCircuitSignature:()=>f$,formatCircuitFlags:()=>h$,findContractInfo:()=>G4});import{existsSync as w$,readFileSync as _9,readdirSync as T9}from"node:fs";import{join as d4,basename as M9}from"node:path";function G4($,Z){let Q=d4($,"compiler",T2);if(w$(Q))return j7(Q,$,[]);for(let X of L7){let z=X?d4($,X,"managed"):d4($,"managed");if(!w$(z))continue;let Y;try{Y=T9(z).sort()}catch{continue}let G=Y.filter((V)=>w$(d4(z,V,"compiler",T2)));if(G.length===0)continue;let J;if(Z){if(!G.includes(Z))throw new Error(`Contract "${Z}" not found in ${z}
|
|
488
|
+
Available: ${G.join(", ")}`);J=Z}else J=G[0];let K=G.filter((V)=>V!==J),q=d4(z,J,"compiler",T2);return j7(q,d4(z,J),K)}throw new Error(`No compiled contract found in ${$}
|
|
489
|
+
Expected: managed/<name>/compiler/${T2}
|
|
490
|
+
Searched: ${L7.map((X)=>X?`${X}/managed/`:"managed/").join(", ")}
|
|
491
|
+
Run "compact compile" first, or use --managed <path> to specify the managed directory.`)}function j7($,Z,Q){let X;try{X=JSON.parse(_9($,"utf-8"))}catch(Y){throw new Error(`Failed to parse ${$}: ${Y.message}`)}return{info:R9(X,$,Z,Q),infoPath:$}}function R9($,Z,Q,X){if(typeof $!=="object"||$===null||Array.isArray($))throw new Error(`${Z}: must be a JSON object`);let z=$,Y=typeof z["compiler-version"]==="string"?z["compiler-version"]:"unknown",G=typeof z["language-version"]==="string"?z["language-version"]:"unknown",J=typeof z["runtime-version"]==="string"?z["runtime-version"]:"unknown",K=Array.isArray(z.circuits)?z.circuits:[],q=Array.isArray(z.witnesses)?z.witnesses:[];return{name:M9(Q),managedDir:Q,compilerVersion:Y,languageVersion:G,runtimeVersion:J,circuits:K,witnesses:q,siblings:X}}function M0($){switch($["type-name"]){case"Uint":return"bigint";case"Bytes":return"Uint8Array";case"Opaque":return $.tsType??"unknown";case"Tuple":if(!$.types||$.types.length===0)return"void";return`[${$.types.map(M0).join(", ")}]`;case"Boolean":return"boolean";case"String":return"string";case"Vector":return $.types?.[0]?`${M0($.types[0])}[]`:"unknown[]";case"Map":if($.types&&$.types.length>=2)return`Map<${M0($.types[0])}, ${M0($.types[1])}>`;return"Map<unknown, unknown>";case"Set":return $.types?.[0]?`Set<${M0($.types[0])}>`:"Set<unknown>";case"Option":return $.types?.[0]?`${M0($.types[0])} | null`:"unknown | null";default:return $["type-name"]??"unknown"}}function f$($){let Z=$.arguments.map((Q)=>`${Q.name}: ${M0(Q.type)}`).join(", ");return`${$.name}(${Z})`}function h$($){if($.pure)return"pure";let Z=["impure"];if($.proof)Z.push("proof");return Z.join(", ")}function m$($){let Z=$.arguments.map((z)=>`${z.name}: ${M0(z.type)}`).join(", "),Q=M0($["result type"]),X=Q!=="void"?` → ${Q}`:"";return`${$.name}(${Z})${X}`}function g$($){return{name:$.name,compilerVersion:$.compilerVersion,languageVersion:$.languageVersion,runtimeVersion:$.runtimeVersion,managedDir:$.managedDir,siblings:$.siblings,circuits:$.circuits.map((Z)=>({name:Z.name,pure:Z.pure,proof:Z.proof,arguments:Z.arguments.map((Q)=>({name:Q.name,type:M0(Q.type)})),returnType:M0(Z["result-type"])})),witnesses:$.witnesses.map((Z)=>({name:Z.name,arguments:Z.arguments.map((Q)=>({name:Q.name,type:M0(Q.type)})),returnType:M0(Z["result type"])}))}}var T2="contract-info.json",L7;var M2=N(()=>{L7=["","contract/src","contract","contracts/src","contracts","src"]});import*as F7 from"node:readline";function E9($){return A9.has($)}function N9($){return D9.has($)}function C9($){let Z=[],Q=$.dappName?`Request from "${$.dappName}"`:"DApp Request";if(Z.push(y(Q)),Z.push(""),Z.push(` ${U("Action:")} ${l($.method)}`),Z.push(` ${U("Network:")} ${$.network}`),$.details.length>0){Z.push("");for(let X of $.details)Z.push(` ${U(X.label+":")} ${X.value}`)}return Z.push(""),Z.push(` ${m("[A]pprove")} ${X0("[R]eject")}`),w3(Z,"heavy")}async function P7($,Z={}){if(Z.approveAll)return process.stderr.write(U(` Auto-approved: ${$.method}`)+`
|
|
492
|
+
`),"approve";if(Z.autoApproveReads&&E9($.method))return process.stderr.write(U(` Auto-approved (read-only): ${$.method}`)+`
|
|
493
|
+
`),"approve";if(Z.autoApproveReads&&N9($.method))return process.stderr.write(U(` Auto-approved (prep): ${$.method}`)+`
|
|
494
|
+
`),"approve";if(!process.stdin.isTTY)return process.stderr.write(X0(" Cannot prompt for approval: stdin is not a TTY")+`
|
|
495
|
+
`),process.stderr.write(U(" Use --approve-all for non-interactive environments")+`
|
|
496
|
+
`),"reject";if(T1)return process.stderr.write(X0(" Rejected: another approval prompt is active")+`
|
|
497
|
+
`),"reject";process.stderr.write(`
|
|
498
|
+
`+C9($)+`
|
|
499
|
+
|
|
500
|
+
`),T1=!0;let Q;try{Q=F7.createInterface({input:process.stdin,output:process.stderr})}catch{return T1=!1,process.stderr.write(X0(" Cannot create readline interface")+`
|
|
501
|
+
`),"reject"}return new Promise((X)=>{let z=(Y)=>{T1=!1,Q.close(),X(Y)};Q.on("close",()=>{if(T1)z("reject")}),Q.question(e(" Approve? [A/r] "),(Y)=>{let G=Y.trim().toLowerCase();if(G==="r"||G==="reject")z("reject");else z("approve")})})}var A9,D9,T1=!1;var I7=N(()=>{H0();A9=new Set(["getShieldedBalances","getUnshieldedBalances","getDustBalance","getShieldedAddresses","getUnshieldedAddress","getDustAddress","getTxHistory","getConfiguration","getConnectionStatus"]),D9=new Set(["balanceUnsealedTransaction","balanceSealedTransaction"])});import{WebSocketServer as x9}from"ws";function y9($){if(u$($))return v9[$.code]??-32603;return-32603}function u$($){return typeof $==="object"&&$!==null&&"type"in $&&$.type==="DAppConnectorAPIError"}function b9($){if(!($ instanceof Error)&&!(typeof $==="object"&&$!==null&&("_tag"in $)))return;let Z=[],Q=$.cause,X=new Set;while(Q&&!X.has(Q)){X.add(Q);let z=Q;if(Q instanceof Error||typeof Q==="object"&&z._tag){let Y=z._tag,G=z.message??"";Z.push(Y?`[${Y}] ${G}`:G),Q=z.cause}else if(typeof Q==="string"){Z.push(Q);break}else{try{Z.push(JSON.stringify(Q))}catch{Z.push(String(Q))}break}}if(Z.length===0)return;return{causes:Z}}function R2($,Z){if(typeof Z==="bigint")return Z.toString();return Z}function S9($,Z){return Z}function k9($){let Z;try{Z=JSON.parse($,S9)}catch{throw Object.assign(new Error("Parse error"),{rpcCode:-32700})}if(typeof Z!=="object"||Z===null||Z.jsonrpc!=="2.0"||typeof Z.method!=="string")throw Object.assign(new Error("Invalid Request"),{rpcCode:-32600});let Q=Z;if(Q.id===void 0||Q.id===null)throw Object.assign(new Error("Missing request id"),{rpcCode:-32600});return Q}function _7($){let{port:Z,handlers:Q,onConnect:X,onDisconnect:z,onRequest:Y,onResponse:G}=$,J=new Map,K=new x9({port:Z,host:"127.0.0.1"});K.on("error",(V)=>{process.stderr.write(`WebSocket server error: ${V.message}
|
|
502
|
+
`)}),K.on("connection",(V)=>{let B=`conn_${++w9}`,H={ws:V,id:B,connectedAt:new Date,authenticated:!1,requestCount:0,notify(W,F){if(V.readyState!==V.OPEN)return;let O={jsonrpc:"2.0",method:W,...F&&{params:F}};V.send(JSON.stringify(O,R2))}};J.set(B,H),X?.(H);let j=!1,L=()=>{if(j)return;j=!0,J.delete(B),z?.(H)};V.on("message",async(W)=>{let F=null,O,_,R=Date.now();try{O=k9(W.toString("utf-8")),F=O.id,H.requestCount++,Y?.(H,O);let T=Q[O.method];if(!T){let p={jsonrpc:"2.0",id:O.id,error:{code:-32601,message:`Method not found: ${O.method}`}};V.send(JSON.stringify(p,R2));return}_={notify:H.notify.bind(H),connectionId:B,requestId:O.id,metadata:{}};let v=await T(O.params??{},_),b=Date.now()-R;if(O.method==="connect"&&v&&typeof v==="object")H.authenticated=!0,H.networkId=v.networkId;let u={jsonrpc:"2.0",id:O.id,result:v};V.send(JSON.stringify(u,R2)),G?.(H,O,b,v,void 0,_.metadata)}catch(T){let v=Date.now()-R,b=T?.rpcCode??y9(T),u=T instanceof Error?T.message:"Internal error",p=u$(T)?{type:T.type,code:T.code}:b9(T),L0={jsonrpc:"2.0",id:F,error:{code:b,message:u,data:p}};if(V.send(JSON.stringify(L0,R2)),O){let f={message:u,code:u$(T)?T.code:void 0};G?.(H,O,v,void 0,f,_?.metadata)}}}),V.on("close",L),V.on("error",()=>{L();try{V.close()}catch{}})});async function q(){for(let V of J.values())V.ws.close(1001,"Server shutting down");return J.clear(),new Promise((V,B)=>{K.close((H)=>{if(H)B(H);else V()})})}return{wss:K,connections:J,close:q}}function C0($,Z){let Q=new Error(Z);return Q.type="DAppConnectorAPIError",Q.code=$,Q.reason=Z,Q}var v9,w9=0;var p$=N(()=>{v9={Rejected:-32000,PermissionRejected:-32001,Disconnected:-32002,InvalidRequest:-32602,InternalError:-32603}});function T7($){let Z=[],Q,X;function z(){if(Q!==void 0&&X!==void 0){let J=Date.now()-X;Z.push({phase:Q,durationMs:J}),$?.onComplete?.(Q,J),Q=void 0,X=void 0}}function Y(J){z(),Q=J,X=Date.now(),$?.onStart?.(J)}function G(){return[...Z]}return{start:Y,complete:z,getTimings:G}}import{Transaction as M7}from"@midnight-ntwrk/ledger-v8";function m9($){return Buffer.from($).toString("hex")}function A2($){if(!/^[0-9a-fA-F]*$/.test($))throw new Error("Invalid hex string: contains non-hex characters");if($.length%2!==0)throw new Error("Invalid hex string: odd length");return new Uint8Array(Buffer.from($,"hex"))}function D7($){return m9($.serialize())}function E7($){return M7.deserialize(R7,A7,h9,A2($))}function c$($){return M7.deserialize(R7,A7,f9,A2($))}var R7="signature",A7="proof",f9="binding",h9="pre-binding";var N7=()=>{};import{Transaction as g9}from"@midnight-ntwrk/ledger-v8";function D2($,Z){let Q=[];Q.push({label:"Tx size",value:l9($.length/2)});let X=u9($,Z);if(X){let z=p9(X);if(z.action)Q.push({label:"Type",value:z.action});if(z.circuits.length>0)Q.push({label:"Circuits",value:z.circuits.join(", ")});if(z.ttl)Q.push({label:"TTL",value:z.ttl})}return Q}function u9($,Z){let Q=c9($),X=Z==="sealed"?[["signature","proof","binding"],["signature","proof","pre-binding"]]:[["signature","proof","pre-binding"],["signature","pre-proof","pre-binding"]];for(let[z,Y,G]of X)try{return g9.deserialize(z,Y,G,Q).toString(!0)}catch{}return null}function p9($){let Z={networkId:null,action:null,circuits:[],ttl:null},Q=$.match(/network_id:\s*"([^"]+)"/);if(Q)Z.networkId=Q[1];if($.includes("Deploy ContractState"))Z.action="Deploy contract";else if($.match(/Call\s/))Z.action="Call contract";let X=/(\w+):\s*<verifier key>/g,z;while((z=X.exec($))!==null)Z.circuits.push(z[1]);if(Z.circuits.length===0){let G=$.match(/Call\s+\S+\s+(\w+)/);if(G)Z.circuits.push(G[1])}let Y=$.match(/ttl:\s*Timestamp\((\d+)\)/);if(Y){let G=parseInt(Y[1],10),J=new Date(G*1000);Z.ttl=J.toISOString().replace("T"," ").slice(0,19)}return Z}function c9($){return new Uint8Array(Buffer.from($,"hex"))}function l9($){if($<1024)return`${$} B`;let Z=$/1024;if(Z<1024)return`${Z.toFixed(1)} KB`;return`${(Z/1024).toFixed(1)} MB`}var C7=()=>{};import{DustLocalState as d9}from"@midnight-ntwrk/ledger-v8";import{CoreWallet as f0}from"@midnight-ntwrk/wallet-sdk-dust-wallet/v1";var _4,i9,o9,ZB;var x7=N(()=>{_4=new Map,i9=f0.applyEventsWithChanges;f0.applyEventsWithChanges=function $(Z,Q,X,z){let Y=i9.call(f0,Z,Q,X,z),G=Array.isArray(Y)?Y[0]:Y;if(_4.size>0&&G?.pendingDust){let J=new Set(G.pendingDust.map((K)=>K.nullifier));for(let K of _4.keys())if(!J.has(K))_4.delete(K)}return Y};o9=f0.spendCoins;f0.spendCoins=function $(Z,Q,X,z){let Y=null;try{Y=Z.state.serialize()}catch{}let G=o9.call(f0,Z,Q,X,z),J=G[1]??G;if(Y&&J?.pendingDust){for(let K of J.pendingDust)if(!_4.has(K.nullifier))_4.set(K.nullifier,Y)}return G};ZB=f0.revertTransaction;f0.revertTransaction=function $(Z,Q){let X=f0.pendingDustToMap(Z.pendingDust),z=[],Y=Q.intents;if(Y)for(let J of Y.values()){let K=J.dustActions?.spends;if(!K)continue;for(let q of K)if(X.has(q.oldNullifier))z.push(q.oldNullifier)}let G=Z.state;for(let J of z){let K=_4.get(J);if(K){try{G=d9.deserialize(K)}catch{}_4.delete(J)}}return{...Z,state:G,pendingDust:Z.pendingDust.filter((J)=>!z.includes(J.nullifier))}};f0.applyFailed=f0.revertTransaction});import*as T4 from"rxjs";import{MidnightBech32m as l$,UnshieldedAddress as n9,ShieldedAddress as a9}from"@midnight-ntwrk/wallet-sdk-address-format";import{NetworkId as d$}from"@midnight-ntwrk/wallet-sdk-abstractions";function s9($){let Z=[],Q=$,X=new Set;while(Q&&!X.has(Q)){X.add(Q);let Y=Q;if(Q instanceof Error||typeof Q==="object"&&Y._tag){let G=Y._tag,J=Y.message??"";if(G&&J)Z.push(`[${G}] ${J}`);else if(J)Z.push(J);else if(G)Z.push(`[${G}]`);if(Y.data&&typeof Y.data==="object")try{Z.push(`data=${JSON.stringify(Y.data)}`)}catch{}Q=Y.cause}else if(typeof Q==="string"){Z.push(Q);break}else{try{let G=JSON.stringify(Q);if(G&&G!=="{}")Z.push(G);else Z.push(String(Q))}catch{Z.push(String(Q))}break}}let z=[];for(let Y of Z)if(z.length===0||z[z.length-1]!==Y)z.push(Y);return z.join(" → ")||"Unknown error"}function v7($){let{bundle:Z,networkConfig:Q,approvalOptions:X,callbacks:z}=$,{facade:Y,keystore:G,zswapSecretKeys:J,dustSecretKey:K}=Z,q=r9[Q.networkId];if(q===void 0)throw new Error(`Unknown networkId: ${Q.networkId}`);let V,B=Y.state().pipe(T4.filter((E)=>E.isSynced)).subscribe((E)=>{V=E});function H(){if(!V)throw C0("Disconnected","Wallet not synced yet");return V}let j={shieldedSecretKeys:J,dustSecretKey:K},L=new Map;function W(E,I,P){let D=L.get(E);if(!D)D=new Map,L.set(E,D);let h=setTimeout(async()=>{if(D.delete(I),D.size===0)L.delete(E);try{await Y.revert(P)}catch{}process.stderr.write(U(` abandoned tx reverted (${E})`)+`
|
|
503
|
+
`)},a2);D.set(I,{recipe:P,timer:h})}function F(E,I){let P=L.get(E);if(!P)return;let D=P.get(I);if(D)clearTimeout(D.timer),P.delete(I);if(P.size===0)L.delete(E)}async function O(E){let I=L.get(E);if(!I||I.size===0)return;L.delete(E);for(let[,P]of I){clearTimeout(P.timer);try{await Y.revert(P.recipe)}catch{}}process.stderr.write(U(` reverted ${I.size} pending tx(s) on disconnect`)+`
|
|
504
|
+
`)}function _(){return new Date(Date.now()+N4*60*1000)}async function R(E,I){I?.start("signing");let P=await Y.signRecipe(E,(S)=>G.signData(S));I?.start("proving");let D,h=await Promise.race([Y.finalizeRecipe(P),new Promise((S,n)=>{D=setTimeout(()=>n(new Error("ZK proof generation timed out")),s4)})]);return clearTimeout(D),I?.complete(),{hex:D7(h),finalized:h}}async function T(E,I=[],P){P?.notify("approval:pending",{method:E});let D=await P7({method:E,network:Q.networkId,details:I,dappName:P?.connectionId},X),h=D==="reject"?"rejected":"approved";if(P?.notify("approval:resolved",{method:E,result:h}),D==="reject")throw C0("Rejected","User rejected the request")}function v(E){return l$.encode(q,E).asString()}function b(E){let I={};for(let P of E){let D=P.kind;if(D!=="shielded"&&D!=="unshielded")throw C0("InvalidRequest",`Invalid output kind: "${D}" — must be "shielded" or "unshielded"`);if(!I[D])I[D]=[];let h=BigInt(P.value),S;if(D==="unshielded")S=l$.parse(P.recipient).decode(n9,q);else S=l$.parse(P.recipient).decode(a9,q);I[D].push({type:P.type,receiverAddress:S,amount:h})}return Object.entries(I).map(([P,D])=>({type:P,outputs:D}))}function u(E,I){return T7({onStart:(P)=>{let D=I?.connectionId??"unknown";z?.onPhaseStart?.(D,E,P),I?.notify("progress",{method:E,phase:P,status:"started"})},onComplete:(P,D)=>{let h=I?.connectionId??"unknown";z?.onPhaseComplete?.(h,E,P,D),I?.notify("progress",{method:E,phase:P,status:"completed",durationMs:D})}})}function p(){if(!V)return!1;try{let E=V.dust;return E?.availableCoins?.length>0||E?.balance(new Date)>0n}catch{return!1}}async function L0(E){return new Promise((I)=>{if(p()){I(!0);return}let P=Y.state().pipe(T4.filter(()=>p()),T4.take(1),T4.timeout(E)).subscribe({next:()=>{P.unsubscribe(),I(!0)},error:()=>{P.unsubscribe(),I(!1)}})})}async function f(E){for(let I=1;I<=x4;I++)try{return await E()}catch(P){let D=String(P?.message??P??"");if(!(/no dust tokens/i.test(D)||/dust.*unavailable/i.test(D))||I===x4)throw P;if(process.stderr.write(U(` dust unavailable, waiting for recovery (${I}/${x4})... [${D.slice(0,60)}]`)+`
|
|
505
|
+
`),!await L0(p1)&&I<x4)await new Promise((n)=>setTimeout(n,p1))}throw new Error("unreachable")}let o={connect:async(E)=>{let I=String(E.networkId??"");if(I.toLowerCase()!==Q.networkId.toLowerCase())throw C0("InvalidRequest",`Network mismatch: wallet is on ${Q.networkId}, requested ${I}`);return{networkId:Q.networkId}},getUnshieldedBalances:async()=>{return H().unshielded.balances},getShieldedBalances:async()=>{return H().shielded.balances},getDustBalance:async()=>{let I=H().dust.balance(new Date);return{cap:I,balance:I}},getUnshieldedAddress:async()=>{let E=H();return{unshieldedAddress:v(E.unshielded.address)}},getShieldedAddresses:async()=>{let I=H().shielded.address;return{shieldedAddress:v(I),shieldedCoinPublicKey:I.coinPublicKeyString(),shieldedEncryptionPublicKey:I.encryptionPublicKeyString()}},getDustAddress:async()=>{let E=H();return{dustAddress:v(E.dust.address)}},getTxHistory:async(E)=>{let I=H(),P=Number(E.pageNumber??0),D=Number(E.pageSize??20);try{let h=I.unshielded.transactionHistory,S=[],n=0,U0=P*D;for await(let t of h.getAll()){if(n>=U0+D)break;if(n>=U0)S.push({txHash:t.hash,txStatus:t.status==="SUCCESS"?{status:"finalized"}:{status:"pending"}});n++}return S}catch{return[]}},getConfiguration:async()=>{return{indexerUri:Q.indexer,indexerWsUri:Q.indexerWS,proverServerUri:Q.proofServer,substrateNodeUri:Q.node,networkId:Q.networkId}},getConnectionStatus:async()=>{return{status:"connected",networkId:q}},makeTransfer:async(E,I)=>{let P=E.desiredOutputs;if(!Array.isArray(P)||P.length===0)throw C0("InvalidRequest","desiredOutputs must be a non-empty array");let D=u("makeTransfer",I),h=P.map((x0,v0)=>({label:`Output ${v0+1}`,value:`${x0.value} → ${String(x0.recipient).slice(0,20)}... (${x0.kind})`}));D.start("approve"),await T("makeTransfer",h,I),D.start("building");let S=b(P),n=E.options?.payFees??!0,U0=await f(()=>Y.transferTransaction(S,j,{ttl:_(),payFees:n})),{hex:t,finalized:G0}=await R(U0,D);return W(I.connectionId,t,U0),I.metadata.phases=D.getTimings(),{tx:t}},submitTransaction:async(E,I)=>{let P=String(E.tx??"");if(!P)throw C0("InvalidRequest","tx is required");let D=u("submitTransaction",I);D.start("approve");try{await T("submitTransaction",D2(P,"sealed"),I)}catch(S){let U0=L.get(I.connectionId)?.get(P);if(U0)try{await Y.revert(U0.recipe)}catch{}throw F(I.connectionId,P),S}let h=c$(P);D.start("submitting");try{let S=await Y.submitTransaction(h);return D.complete(),F(I.connectionId,P),I.metadata.phases=D.getTimings(),{txHash:S}}catch(S){D.complete();let U0=L.get(I.connectionId)?.get(P);if(U0)try{await Y.revert(U0.recipe)}catch{}F(I.connectionId,P);let t=s9(S),G0=new Error(`Transaction submission failed: ${t}`);throw G0.cause=S,G0}},balanceUnsealedTransaction:async(E,I)=>{let P=String(E.tx??"");if(!P)throw C0("InvalidRequest","tx is required");let D=u("balanceUnsealedTransaction",I);D.start("approve"),await T("balanceUnsealedTransaction",D2(P,"unsealed"),I),D.start("building");let h=E7(P),S=await f(()=>Y.balanceUnboundTransaction(h,j,{ttl:_()})),{hex:n,finalized:U0}=await R(S,D);return W(I.connectionId,n,S),I.metadata.phases=D.getTimings(),{tx:n}},balanceSealedTransaction:async(E,I)=>{let P=String(E.tx??"");if(!P)throw C0("InvalidRequest","tx is required");let D=u("balanceSealedTransaction",I);D.start("approve"),await T("balanceSealedTransaction",D2(P,"sealed"),I),D.start("building");let h=c$(P),S=await f(()=>Y.balanceFinalizedTransaction(h,j,{ttl:_()})),{hex:n,finalized:U0}=await R(S,D);return W(I.connectionId,n,S),I.metadata.phases=D.getTimings(),{tx:n}},makeIntent:async(E,I)=>{let{desiredInputs:P,desiredOutputs:D,options:h}=E;if(!h)throw C0("InvalidRequest","options is required for makeIntent");let S=u("makeIntent",I);S.start("approve"),await T("makeIntent",[],I),S.start("building");let n={};if(Array.isArray(P))for(let v0 of P){let R0=v0.kind;if(!n[R0])n[R0]={};n[R0][v0.type]=BigInt(v0.value)}let U0=b(D??[]),t=await f(()=>Y.initSwap(n,U0,j,{ttl:_(),payFees:h.payFees??!0})),{hex:G0,finalized:x0}=await R(t,S);return W(I.connectionId,G0,t),I.metadata.phases=S.getTimings(),{tx:G0}},signData:async(E,I)=>{let P=String(E.data??""),D=E.options;if(!P||!D?.encoding)throw C0("InvalidRequest","data and options.encoding are required");if(D.keyType&&D.keyType!=="unshielded")throw C0("InvalidRequest",`Unsupported keyType: "${D.keyType}" — only "unshielded" is supported`);await T("signData",[{label:"Encoding",value:D.encoding},{label:"Data",value:P.length>64?P.slice(0,64)+"...":P}],I);let h;switch(D.encoding){case"hex":h=A2(P);break;case"base64":h=new Uint8Array(Buffer.from(P,"base64"));break;case"text":h=new Uint8Array(Buffer.from(P,"utf-8"));break;default:throw C0("InvalidRequest",`Unknown encoding: ${D.encoding}`)}let S=G.signData(h),n=G.getPublicKey();return{data:P,signature:String(S),verifyingKey:String(n)}},getProvingProvider:async()=>{return{provingProvider:"ready",proverServerUri:Q.proofServer}},hintUsage:async(E)=>{let I=E.methodNames??[];process.stderr.write(U(` DApp hints usage: ${I.join(", ")}`)+`
|
|
506
|
+
`)}};function Y0(){B.unsubscribe();for(let[,E]of L)for(let[,I]of E)clearTimeout(I.timer);L.clear()}function $0(){for(let[,E]of L)if(E.size>0)return!0;return!1}return{handlers:o,revertPendingTxs:O,hasPendingTxs:$0,dispose:Y0}}var r9;var y7=N(()=>{I7();p$();N7();C7();x7();r9={PreProd:d$.NetworkId.PreProd,Preview:d$.NetworkId.Preview,Undeployed:d$.NetworkId.Undeployed}});var w7={};k(w7,{startServeOrReuse:()=>i$,startServe:()=>k7,probeServeNetwork:()=>S7,isPortInUse:()=>b7});import{Socket as t9}from"node:net";function b7($){return new Promise((Z)=>{let Q=new t9;Q.setTimeout(1000),Q.once("connect",()=>{Q.destroy(),Z(!0)}),Q.once("timeout",()=>{Q.destroy(),Z(!1)}),Q.once("error",()=>Z(!1)),Q.connect($,"127.0.0.1")})}async function S7($){let Z=(await import("ws")).default;return new Promise((Q)=>{let X=new Z(`ws://127.0.0.1:${$}`),z=setTimeout(()=>{try{X.close()}catch{}Q(null)},2000);X.on("open",()=>{X.send(JSON.stringify({jsonrpc:"2.0",id:1,method:"getConnectionStatus",params:{}}))}),X.on("message",(Y)=>{clearTimeout(z);try{let J=JSON.parse(Y.toString())?.result?.networkId;X.close(),Q(typeof J==="string"?J:null)}catch{X.close(),Q(null)}}),X.on("error",()=>{clearTimeout(z),Q(null)})})}async function i$($){let Z=$.port??c1,Q=($.network??"undeployed").toLowerCase(),X=$.onMessage??(()=>{});if(await b7(Z)){let z=await S7(Z);if(!z)throw new Error(`Port ${Z} is in use by a process that does not look like mn serve. Free the port (e.g. \`lsof -nP -iTCP:${Z} -sTCP:LISTEN\` then \`kill <pid>\`) and retry.`);let Y=z.toLowerCase();if(Y!==Q)throw new Error(`mn serve is already running on port ${Z} for network ${Y}, but this run needs ${Q}. Stop the wrong-network serve first (\`pkill -f 'mn serve'\`) and retry.`);return X(`Reusing existing mn serve on port ${Z}`),{port:Z,async stop(){}}}return k7($)}async function k7($){let Z=$.port??c1,Q=$.onMessage??(()=>{}),X=z0(Q0($.wallet)),z=Buffer.from(X.seed,"hex"),{name:Y,config:G}=q0({args:{command:"serve",subcommand:void 0,positionals:[],flags:$.network?{network:$.network}:{}}}),J=X.addresses[Y],K=j0((L,W)=>{Q(`SDK: ${W}`)}),q=T0();Q("Building wallet facade...");let V=o0(J,Y),B=await X4(z,G,V);Q("Syncing wallet..."),await z4(B,{onProgress:(L,W)=>{if(W>0){let F=Math.min(Math.round(L/W*100),100);Q(`Syncing wallet... ${F}%`)}}}),Q("Wallet synced"),Q("Waiting for dust..."),await G2(B),Q("Dust ready");try{await p0(J,Y,B.facade)}catch{}let H=v7({bundle:B,networkConfig:G,approvalOptions:{approveAll:!0,autoApproveReads:!0},callbacks:{onPhaseStart:(L,W,F)=>{Q(`${F}...`)},onPhaseComplete:(L,W,F,O)=>{Q(`${F} ${Math.round(O)}ms`)}}}),j=_7({port:Z,handlers:H.handlers,onConnect:(L)=>{Q(`connected ${L.id}`)},onDisconnect:(L)=>{Q(`disconnected ${L.id}`),H.revertPendingTxs(L.id).catch(()=>{})},onRequest:(L,W)=>{Q(`${L.id} → ${W.method}`)},onResponse:(L,W,F,O,_)=>{if(_)Q(`${L.id} ← ${W.method} FAILED: ${_.message}`);else Q(`${L.id} ← ${W.method} (${Math.round(F)}ms)`)}});return Q(`Server ready on ws://localhost:${Z}`),{port:Z,async stop(){j.close(),H.dispose(),q(),K();try{await g0(B)}catch{}}}}var o$=N(()=>{b0();V0();i0();O4();P4();y7();p$()});var u7={};k(u7,{findWitnessSource:()=>g7,findWitnessFile:()=>e9,buildMissingWitnessError:()=>$X,WITNESS_SOURCE_CANDIDATES:()=>m7,WITNESS_FILE_CANDIDATES:()=>M1});import{existsSync as f7}from"node:fs";import{join as h7}from"node:path";function e9($){for(let Z of M1){let Q=h7($,Z);if(f7(Q))return Q}return null}function g7($){for(let Z of m7){let Q=h7($,Z);if(f7(Q))return Q}return null}function $X($){let{projectRoot:Z,witnessNames:Q}=$,X=g7(Z),z=[`Contract declares ${Q.length} witness(es): ${Q.join(", ")}`,"but no compiled witnesses module was found.","",`Searched (relative to ${Z}):`,...M1.map((Y)=>` - ${Y}`),""];if(X)z.push(`Found ${X} — looks like it isn't compiled yet.`,'Build it (e.g. "npm run build") so a .js exists, then retry.');else z.push('Add a witnesses module exporting a "witnesses" object that maps each',"declared name to its implementation. Example: src/witnesses.ts → tsc → dist/witnesses.js.");return z.join(`
|
|
507
|
+
`)}var M1,m7;var n$=N(()=>{M1=["dist/witnesses.js","src/witnesses.js","contract/dist/witnesses.js","contract/src/witnesses.js"],m7=["src/witnesses.ts","contract/src/witnesses.ts"]});function E2($){if(typeof $==="number")return Number.isInteger($)?BigInt($):$;if(typeof $==="string"){if(/^-?\d+n$/.test($))return BigInt($.slice(0,-1));return $}if(Array.isArray($)){if($.length>0&&$.every((Z)=>typeof Z==="number"&&Number.isInteger(Z)&&Z>=0&&Z<=255))return Uint8Array.from($);return $.map(E2)}if($&&typeof $==="object"&&Object.getPrototypeOf($)===Object.prototype){let Z={};for(let Q in $)if(Object.prototype.hasOwnProperty.call($,Q))Z[Q]=E2($[Q]);return Z}return $}var C2={};k(C2,{runState:()=>A1,runDeploy:()=>r$,runCall:()=>s$});import{spawn as ZX}from"node:child_process";import{writeFileSync as QX,unlinkSync as p7,existsSync as N2,symlinkSync as c7}from"node:fs";import{join as R1,dirname as l7}from"node:path";import{fileURLToPath as XX}from"node:url";function zX(){let $=l7(XX(import.meta.url));for(let Z=0;Z<8;Z++){let Q=R1($,"node_modules");if(N2(Q))return Q;let X=l7($);if(X===$)break;$=X}return null}async function A1($){let Z=JX($),Q=await t$($.dappDir,Z,$.onMessage);try{return JSON.parse(Q)}catch{throw new Error(`State script returned unexpected output:
|
|
508
|
+
${Q}`)}}async function r$($){let Z=YX($),Q=await t$($.dappDir,Z,$.onMessage);try{return JSON.parse(Q)}catch{throw new Error(`Deploy script returned unexpected output:
|
|
509
|
+
${Q}`)}}async function s$($){let Z=GX($),Q=await t$($.dappDir,Z,$.onMessage);try{return JSON.parse(Q)}catch{throw new Error(`Call script returned unexpected output:
|
|
510
|
+
${Q}`)}}function d7($){return`
|
|
511
|
+
import WebSocket from 'ws';
|
|
512
|
+
|
|
513
|
+
const RPC_URL = 'ws://127.0.0.1:${$}';
|
|
514
|
+
let rpcId = 0;
|
|
515
|
+
let rpcWs;
|
|
516
|
+
|
|
517
|
+
async function rpcConnect() {
|
|
518
|
+
return new Promise((resolve, reject) => {
|
|
519
|
+
rpcWs = new WebSocket(RPC_URL);
|
|
520
|
+
rpcWs.on('open', () => resolve(rpcWs));
|
|
521
|
+
rpcWs.on('error', (err) => reject(new Error('Cannot connect to mn serve at ' + RPC_URL + ': ' + err.message)));
|
|
522
|
+
setTimeout(() => reject(new Error('Timeout connecting to mn serve at ' + RPC_URL)), 5000);
|
|
523
|
+
});
|
|
524
|
+
}
|
|
525
|
+
|
|
526
|
+
function rpcCall(method, params = {}) {
|
|
527
|
+
return new Promise((resolve, reject) => {
|
|
528
|
+
const id = ++rpcId;
|
|
529
|
+
const timeout = setTimeout(() => reject(new Error('RPC timeout for ' + method)), 300000);
|
|
530
|
+
|
|
531
|
+
const handler = (data) => {
|
|
532
|
+
try {
|
|
533
|
+
const msg = JSON.parse(data.toString());
|
|
534
|
+
if (msg.id === id) {
|
|
535
|
+
clearTimeout(timeout);
|
|
536
|
+
rpcWs.removeListener('message', handler);
|
|
537
|
+
if (msg.error) reject(new Error(msg.error.message || JSON.stringify(msg.error)));
|
|
538
|
+
else resolve(msg.result);
|
|
539
|
+
}
|
|
540
|
+
} catch {}
|
|
541
|
+
};
|
|
542
|
+
|
|
543
|
+
rpcWs.on('message', handler);
|
|
544
|
+
rpcWs.send(JSON.stringify({ jsonrpc: '2.0', id, method, params }));
|
|
545
|
+
});
|
|
546
|
+
}
|
|
547
|
+
|
|
548
|
+
function rpcClose() {
|
|
549
|
+
if (rpcWs) rpcWs.close();
|
|
550
|
+
}
|
|
551
|
+
`}function i7(){return`
|
|
552
|
+
// Wallet provider backed by mn serve RPC
|
|
553
|
+
const walletProvider = {
|
|
554
|
+
getCoinPublicKey: () => {
|
|
555
|
+
// Will be populated after getShieldedAddresses
|
|
556
|
+
return walletState?.shieldedCoinPublicKey ?? '';
|
|
557
|
+
},
|
|
558
|
+
getEncryptionPublicKey: () => {
|
|
559
|
+
return walletState?.shieldedEncryptionPublicKey ?? '';
|
|
560
|
+
},
|
|
561
|
+
async balanceTx(tx, ttl) {
|
|
562
|
+
const { toHex } = await import('@midnight-ntwrk/midnight-js-utils');
|
|
563
|
+
const txHex = toHex(tx.serialize());
|
|
564
|
+
const result = await rpcCall('balanceUnsealedTransaction', { tx: txHex });
|
|
565
|
+
|
|
566
|
+
const { fromHex } = await import('@midnight-ntwrk/midnight-js-utils');
|
|
567
|
+
const { Transaction } = await import('@midnight-ntwrk/midnight-js-types');
|
|
568
|
+
const bytes = fromHex(result.tx);
|
|
569
|
+
return Transaction.deserialize('signature', 'proof', 'binding', bytes);
|
|
570
|
+
},
|
|
571
|
+
async submitTx(tx) {
|
|
572
|
+
const { toHex } = await import('@midnight-ntwrk/midnight-js-utils');
|
|
573
|
+
const txHex = toHex(tx.serialize());
|
|
574
|
+
await rpcCall('submitTransaction', { tx: txHex });
|
|
575
|
+
return tx.identifiers()[0];
|
|
576
|
+
},
|
|
577
|
+
};
|
|
578
|
+
|
|
579
|
+
// Get wallet state for coin/encryption public keys
|
|
580
|
+
const walletState = await rpcCall('getShieldedAddresses', {});
|
|
581
|
+
`}function o7($,Z){return`
|
|
582
|
+
import { pathToFileURL } from 'node:url';
|
|
583
|
+
import { resolve } from 'node:path';
|
|
584
|
+
import { CompiledContract } from '@midnight-ntwrk/compact-js';
|
|
585
|
+
|
|
586
|
+
const MANAGED_DIR = ${JSON.stringify(Z)};
|
|
587
|
+
|
|
588
|
+
// Load compiled contract class
|
|
589
|
+
const contractMod = await import(pathToFileURL(resolve(MANAGED_DIR, 'contract', 'index.js')).href);
|
|
590
|
+
|
|
591
|
+
// Try to load witnesses and private state factory.
|
|
592
|
+
// Both relative-to-cwd locations and contract/<...> locations are checked
|
|
593
|
+
// so mn dev (which chdirs into the contract sub-package) and mn contract
|
|
594
|
+
// deploy run from a workspace root both work.
|
|
595
|
+
const WITNESS_CANDIDATES = ${JSON.stringify(M1)};
|
|
596
|
+
let witnesses;
|
|
597
|
+
let createPrivateState;
|
|
598
|
+
for (const p of WITNESS_CANDIDATES) {
|
|
599
|
+
try {
|
|
600
|
+
const wMod = await import(pathToFileURL(resolve(p)).href);
|
|
601
|
+
if (wMod.witnesses) { witnesses = wMod.witnesses; }
|
|
602
|
+
// Look for createInitialPrivateState or create*PrivateState
|
|
603
|
+
for (const key of Object.keys(wMod)) {
|
|
604
|
+
if (typeof wMod[key] === 'function' && key.startsWith('create') && key.toLowerCase().includes('privatestate')) {
|
|
605
|
+
createPrivateState = wMod[key];
|
|
606
|
+
}
|
|
607
|
+
}
|
|
608
|
+
if (witnesses) break;
|
|
609
|
+
} catch {}
|
|
610
|
+
}
|
|
611
|
+
|
|
612
|
+
// Note: callers (commands/contract.ts, commands/dev.ts) preflight when the
|
|
613
|
+
// contract's compiler-info declares witnesses, so a missing module here means
|
|
614
|
+
// the contract truly has no witnesses (vacant is correct). The runtime
|
|
615
|
+
// warning still names every path we tried — useful when the SDK later
|
|
616
|
+
// complains about a specific witness name despite preflight passing.
|
|
617
|
+
if (!witnesses) {
|
|
618
|
+
process.stderr.write('Warning: No witnesses module found — using vacant witnesses. Searched:\\n');
|
|
619
|
+
for (const p of WITNESS_CANDIDATES) process.stderr.write(' - ' + p + '\\n');
|
|
620
|
+
}
|
|
621
|
+
|
|
622
|
+
// Generate initial private state.
|
|
623
|
+
// Uses a deterministic seed derived from the wallet so post/takeDown use the same key.
|
|
624
|
+
// The private state provider (leveldb) persists between calls, but initialPrivateState
|
|
625
|
+
// is the fallback when no stored state exists yet.
|
|
626
|
+
function makeInitialPrivateState() {
|
|
627
|
+
if (createPrivateState) {
|
|
628
|
+
// Derive a deterministic key from the wallet's coin public key.
|
|
629
|
+
// This ensures post and takeDown always use the same secret key
|
|
630
|
+
// for the same wallet, even across separate CLI invocations.
|
|
631
|
+
const cpk = walletState?.shieldedCoinPublicKey ?? '';
|
|
632
|
+
let secretKey;
|
|
633
|
+
if (cpk && cpk.length >= 64) {
|
|
634
|
+
// Use first 32 bytes of coin public key as seed
|
|
635
|
+
secretKey = new Uint8Array(32);
|
|
636
|
+
for (let i = 0; i < 32; i++) {
|
|
637
|
+
secretKey[i] = parseInt(cpk.substr(i * 2, 2), 16);
|
|
638
|
+
}
|
|
639
|
+
} else {
|
|
640
|
+
// Fallback: random (will break takeDown if state isn't persisted)
|
|
641
|
+
secretKey = new Uint8Array(32);
|
|
642
|
+
globalThis.crypto.getRandomValues(secretKey);
|
|
643
|
+
}
|
|
644
|
+
try { return createPrivateState(secretKey); } catch {}
|
|
645
|
+
try { return createPrivateState(); } catch {}
|
|
646
|
+
}
|
|
647
|
+
return {};
|
|
648
|
+
}
|
|
649
|
+
|
|
650
|
+
const compiled = witnesses
|
|
651
|
+
? CompiledContract.make(${JSON.stringify($)}, contractMod.Contract).pipe(
|
|
652
|
+
(c) => CompiledContract.withWitnesses(c, witnesses),
|
|
653
|
+
(c) => CompiledContract.withCompiledFileAssets(c, MANAGED_DIR),
|
|
654
|
+
)
|
|
655
|
+
: CompiledContract.make(${JSON.stringify($)}, contractMod.Contract).pipe(
|
|
656
|
+
CompiledContract.withVacantWitnesses,
|
|
657
|
+
(c) => CompiledContract.withCompiledFileAssets(c, MANAGED_DIR),
|
|
658
|
+
);
|
|
659
|
+
`}function n7($,Z,Q){return`
|
|
660
|
+
import { NodeZkConfigProvider } from '@midnight-ntwrk/midnight-js-node-zk-config-provider';
|
|
661
|
+
import { httpClientProofProvider } from '@midnight-ntwrk/midnight-js-http-client-proof-provider';
|
|
662
|
+
import { indexerPublicDataProvider } from '@midnight-ntwrk/midnight-js-indexer-public-data-provider';
|
|
663
|
+
import { levelPrivateStateProvider } from '@midnight-ntwrk/midnight-js-level-private-state-provider';
|
|
664
|
+
|
|
665
|
+
const zkConfigProvider = new NodeZkConfigProvider(MANAGED_DIR);
|
|
666
|
+
|
|
667
|
+
const providers = {
|
|
668
|
+
privateStateProvider: levelPrivateStateProvider({
|
|
669
|
+
privateStateStoreName: ${JSON.stringify(Q)},
|
|
670
|
+
privateStoragePasswordProvider: () => Promise.resolve('mn-contract-default-pwd-16ch'),
|
|
671
|
+
accountId: 'mn-contract-runner',
|
|
672
|
+
}),
|
|
673
|
+
publicDataProvider: indexerPublicDataProvider(
|
|
674
|
+
${JSON.stringify(Z.indexer)},
|
|
675
|
+
${JSON.stringify(Z.indexerWS??Z.indexer.replace("http","ws"))},
|
|
676
|
+
),
|
|
677
|
+
zkConfigProvider,
|
|
678
|
+
proofProvider: httpClientProofProvider(${JSON.stringify(Z.proofServer)}, zkConfigProvider),
|
|
679
|
+
walletProvider,
|
|
680
|
+
midnightProvider: walletProvider,
|
|
681
|
+
};
|
|
682
|
+
`}function YX($){let Z=$.privateStateKey??`${$.contractName}PrivateState`,Q=JSON.stringify($.args??[]);return`
|
|
683
|
+
import { setNetworkId } from '@midnight-ntwrk/midnight-js-network-id';
|
|
684
|
+
setNetworkId(${JSON.stringify($.networkConfig.networkId.toLowerCase())});
|
|
685
|
+
|
|
686
|
+
${d7($.servePort)}
|
|
687
|
+
${o7($.contractName,$.managedDir)}
|
|
688
|
+
|
|
689
|
+
process.stderr.write('Connecting to mn serve...\\n');
|
|
690
|
+
await rpcConnect();
|
|
691
|
+
process.stderr.write('Connected\\n');
|
|
692
|
+
|
|
693
|
+
${i7()}
|
|
694
|
+
${n7($.managedDir,$.networkConfig,Z)}
|
|
695
|
+
|
|
696
|
+
import { deployContract } from '@midnight-ntwrk/midnight-js-contracts';
|
|
697
|
+
|
|
698
|
+
${a7}
|
|
699
|
+
const constructorArgs = (${Q}).map(coerceArg);
|
|
700
|
+
process.stderr.write('Deploying contract' + (constructorArgs.length ? ' with ' + constructorArgs.length + ' constructor arg(s)...' : '...') + '\\n');
|
|
701
|
+
const deployed = await deployContract(providers, {
|
|
702
|
+
compiledContract: compiled,
|
|
703
|
+
privateStateId: ${JSON.stringify(Z)},
|
|
704
|
+
initialPrivateState: makeInitialPrivateState(),
|
|
705
|
+
args: constructorArgs,
|
|
706
|
+
});
|
|
707
|
+
|
|
708
|
+
const address = deployed.deployTxData?.public?.contractAddress ?? 'unknown';
|
|
709
|
+
process.stderr.write('Deploy complete: ' + address + '\\n');
|
|
710
|
+
|
|
711
|
+
console.log(JSON.stringify({ contractAddress: address }));
|
|
712
|
+
rpcClose();
|
|
713
|
+
process.exit(0);
|
|
714
|
+
`}function GX($){let Z=$.privateStateKey??`${$.contractName}PrivateState`;return`
|
|
715
|
+
import { setNetworkId } from '@midnight-ntwrk/midnight-js-network-id';
|
|
716
|
+
setNetworkId(${JSON.stringify($.networkConfig.networkId.toLowerCase())});
|
|
717
|
+
|
|
718
|
+
${d7($.servePort)}
|
|
719
|
+
${o7($.contractName,$.managedDir)}
|
|
720
|
+
|
|
721
|
+
process.stderr.write('Connecting to mn serve...\\n');
|
|
722
|
+
await rpcConnect();
|
|
723
|
+
process.stderr.write('Connected\\n');
|
|
724
|
+
|
|
725
|
+
${i7()}
|
|
726
|
+
${n7($.managedDir,$.networkConfig,Z)}
|
|
727
|
+
|
|
728
|
+
import { findDeployedContract } from '@midnight-ntwrk/midnight-js-contracts';
|
|
729
|
+
|
|
730
|
+
process.stderr.write('Finding deployed contract...\\n');
|
|
731
|
+
const deployed = await findDeployedContract(providers, {
|
|
732
|
+
compiledContract: compiled,
|
|
733
|
+
contractAddress: ${JSON.stringify($.contractAddress)},
|
|
734
|
+
privateStateId: ${JSON.stringify(Z)},
|
|
735
|
+
initialPrivateState: makeInitialPrivateState(),
|
|
736
|
+
});
|
|
737
|
+
|
|
738
|
+
process.stderr.write('Calling ${$.circuit}...\\n');
|
|
739
|
+
${a7}
|
|
740
|
+
const args = (${JSON.stringify($.args??[])}).map(coerceArg);
|
|
741
|
+
await deployed.callTx.${$.circuit}(...args);
|
|
742
|
+
|
|
743
|
+
console.log(JSON.stringify({ status: 'success', circuit: ${JSON.stringify($.circuit)} }));
|
|
744
|
+
rpcClose();
|
|
745
|
+
process.exit(0);
|
|
746
|
+
`}function JX($){return`
|
|
747
|
+
import { setNetworkId } from '@midnight-ntwrk/midnight-js-network-id';
|
|
748
|
+
setNetworkId(${JSON.stringify($.networkConfig.networkId.toLowerCase())});
|
|
749
|
+
|
|
750
|
+
import { pathToFileURL } from 'node:url';
|
|
751
|
+
import { resolve } from 'node:path';
|
|
752
|
+
import { indexerPublicDataProvider } from '@midnight-ntwrk/midnight-js-indexer-public-data-provider';
|
|
753
|
+
|
|
754
|
+
const MANAGED_DIR = ${JSON.stringify($.managedDir)};
|
|
331
755
|
|
|
332
|
-
|
|
333
|
-
|
|
334
|
-
|
|
756
|
+
// Load the contract module which exports the ledger() function
|
|
757
|
+
const contractMod = await import(pathToFileURL(resolve(MANAGED_DIR, 'contract', 'index.js')).href);
|
|
758
|
+
|
|
759
|
+
if (typeof contractMod.ledger !== 'function') {
|
|
760
|
+
console.error('Contract module does not export a ledger() function');
|
|
761
|
+
process.exit(1);
|
|
762
|
+
}
|
|
763
|
+
|
|
764
|
+
process.stderr.write('Querying contract state...\\n');
|
|
765
|
+
const provider = indexerPublicDataProvider(
|
|
766
|
+
${JSON.stringify($.networkConfig.indexer)},
|
|
767
|
+
${JSON.stringify($.networkConfig.indexerWS??$.networkConfig.indexer.replace("http","ws"))},
|
|
768
|
+
);
|
|
769
|
+
|
|
770
|
+
const contractState = await provider.queryContractState(${JSON.stringify($.contractAddress)});
|
|
771
|
+
if (!contractState) {
|
|
772
|
+
console.error('No contract found at address ${$.contractAddress}');
|
|
773
|
+
process.exit(1);
|
|
774
|
+
}
|
|
775
|
+
|
|
776
|
+
process.stderr.write('Parsing ledger state...\\n');
|
|
777
|
+
// ledger() expects contractState.data (ChargedState), not the full ContractState
|
|
778
|
+
const state = contractMod.ledger(contractState.data ?? contractState);
|
|
779
|
+
|
|
780
|
+
// Extract scalar fields and map fields
|
|
781
|
+
const fields = {};
|
|
782
|
+
const maps = {};
|
|
783
|
+
|
|
784
|
+
for (const key of Object.keys(state)) {
|
|
785
|
+
const val = state[key];
|
|
786
|
+
|
|
787
|
+
// Check if it's a map-like (has size() and Symbol.iterator)
|
|
788
|
+
if (val && typeof val === 'object' && typeof val.size === 'function') {
|
|
789
|
+
try {
|
|
790
|
+
maps[key] = { size: val.size().toString() };
|
|
791
|
+
} catch {
|
|
792
|
+
maps[key] = { size: '?' };
|
|
793
|
+
}
|
|
794
|
+
} else if (typeof val === 'bigint') {
|
|
795
|
+
fields[key] = val.toString();
|
|
796
|
+
} else if (typeof val === 'string') {
|
|
797
|
+
fields[key] = val;
|
|
798
|
+
} else if (typeof val === 'boolean') {
|
|
799
|
+
fields[key] = String(val);
|
|
800
|
+
} else if (val instanceof Uint8Array) {
|
|
801
|
+
fields[key] = Array.from(val).map(b => b.toString(16).padStart(2, '0')).join('');
|
|
802
|
+
} else if (val !== undefined && val !== null) {
|
|
803
|
+
try { fields[key] = JSON.stringify(val); } catch { fields[key] = '?'; }
|
|
804
|
+
}
|
|
805
|
+
}
|
|
806
|
+
|
|
807
|
+
console.log(JSON.stringify({ fields, maps }));
|
|
808
|
+
process.exit(0);
|
|
809
|
+
`}function KX($){if(!a$)return[];let Z=[];if(!N2($)){try{c7(a$,$),Z.push($)}catch{}return Z}for(let Q of qX){let X=R1(a$,Q),z=R1($,Q);if(!N2(X)||N2(z))continue;try{c7(X,z),Z.push(z)}catch{}}return Z}async function t$($,Z,Q){let X=R1($,"node_modules"),z=KX(X),Y=R1($,`.mn-contract-${Date.now()}.mjs`);QX(Y,Z);let G=()=>{try{p7(Y)}catch{}for(let J of z.slice().reverse())try{p7(J)}catch{}};return new Promise((J,K)=>{let q=ZX("node",[Y],{cwd:$,stdio:["ignore","pipe","pipe"],env:{...process.env,NODE_NO_WARNINGS:"1"}}),V="",B="";q.stdout?.on("data",(H)=>{V+=H.toString()}),q.stderr?.on("data",(H)=>{let j=H.toString().trim();if(j)B+=j+`
|
|
810
|
+
`,Q?.(j)}),q.on("close",(H)=>{if(G(),H!==0){K(new Error(B.trim()||`Script exited with code ${H}`));return}J(V.trim())}),q.on("error",(H)=>{G(),K(H)})})}var a$,a7,qX;var D1=N(()=>{n$();a$=zX();a7=`
|
|
811
|
+
const coerceArg = (${E2.toString()});
|
|
812
|
+
`;qX=["@midnight-ntwrk","ws"]});var r7={};k(r7,{buildStateProvider:()=>BX});import{indexerPublicDataProvider as VX}from"@midnight-ntwrk/midnight-js-indexer-public-data-provider";function BX($){return VX($.indexer,$.indexerWS??$.indexer.replace("http","ws"))}var s7=()=>{};var QZ={};k(QZ,{default:()=>e7});import{resolve as x2}from"node:path";function E1(){return{update(){},stop(){},fail(){},log(){}}}function HX($){return t7.includes($)}async function e7($,Z){let Q=$.subcommand;if(!Q||!HX(Q))throw new Z0(`Usage: midnight contract <${t7.join("|")}>
|
|
813
|
+
|
|
814
|
+
inspect Show circuits, witnesses, and types for a compiled contract
|
|
815
|
+
deploy Deploy a contract to the network
|
|
816
|
+
call Call a circuit on a deployed contract
|
|
817
|
+
state Read the ledger state of a deployed contract
|
|
818
|
+
|
|
819
|
+
Run "midnight help contract" for more info.`);switch(Q){case"inspect":return WX($);case"deploy":return UX($);case"call":return jX($);case"state":return OX($)}}function v2($){let Z=A($,"managed"),Q=A($,"path");return x2(Z??Q??process.cwd())}async function WX($){let Z=C($,"json"),Q=v2($),X=A($,"name"),{info:z}=G4(Q,X);if(Z){x(g$(z));return}if(process.stderr.write(`
|
|
820
|
+
`+g(`Contract: ${z.name}`)+`
|
|
821
|
+
|
|
822
|
+
`),process.stderr.write(M("Compiler",z.compilerVersion)+`
|
|
823
|
+
`),process.stderr.write(M("Language",z.languageVersion)+`
|
|
824
|
+
`),process.stderr.write(M("Runtime",z.runtimeVersion)+`
|
|
335
825
|
`),process.stderr.write(`
|
|
336
|
-
`+
|
|
337
|
-
|
|
338
|
-
`),process.stderr.write(
|
|
339
|
-
`)
|
|
340
|
-
|
|
341
|
-
`),process.stderr.write(
|
|
342
|
-
`)
|
|
343
|
-
`)
|
|
826
|
+
`+y(" Circuits")+`
|
|
827
|
+
`),z.circuits.length===0)process.stderr.write(U(" (none)")+`
|
|
828
|
+
`);else for(let Y of z.circuits){let G=f$(Y),J=h$(Y),K=Y.pure?l(J):e(J);process.stderr.write(` ${G} ${U("—")} ${K}
|
|
829
|
+
`)}if(process.stderr.write(`
|
|
830
|
+
`+y(" Witnesses")+`
|
|
831
|
+
`),z.witnesses.length===0)process.stderr.write(U(" (none)")+`
|
|
832
|
+
`);else for(let Y of z.witnesses)process.stderr.write(` ${m$(Y)}
|
|
833
|
+
`);if(z.siblings.length>0){process.stderr.write(`
|
|
834
|
+
`+y(" Other contracts in this project")+`
|
|
835
|
+
`);for(let Y of z.siblings)process.stderr.write(` ${l(Y)}
|
|
836
|
+
`);process.stderr.write(`
|
|
837
|
+
`+U(" Inspect a sibling: mn contract inspect --name <name>")+`
|
|
838
|
+
`)}process.stderr.write(`
|
|
839
|
+
`)}async function $Z($,Z,Q){let{loadWalletConfig:X,resolveWalletPath:z}=await Promise.resolve().then(() => (b0(),b3)),{resolveNetwork:Y}=await Promise.resolve().then(() => (V0(),k4)),{defaultRepository:G}=await Promise.resolve().then(() => (F4(),G8)),{NATIVE_TOKEN_TYPE:J}=await Promise.resolve().then(() => l1),{config:K}=Y({args:{command:"contract",subcommand:void 0,positionals:[],flags:{network:$}}}),q=X(z(Q)),V=Buffer.from(q.seed,"hex"),B=q.addresses[$],H=Z?E1():i("Checking wallet..."),j=G();try{let W=(await j.unshielded(B,K)).balances.get(J)??0n;if(W===0n)throw H.fail("No NIGHT balance"),new Error(`Wallet has 0 NIGHT on ${$}.
|
|
840
|
+
|
|
841
|
+
Fund your wallet before deploying:
|
|
842
|
+
Address: ${B}
|
|
843
|
+
`+($==="undeployed"?` Run: midnight airdrop 1000
|
|
844
|
+
`:` Use the Midnight faucet or transfer from another wallet.
|
|
845
|
+
`));let F=await j.dust(V,K,{onStatus:(O)=>H.update(O)});if(F.balance===0n&&F.availableCoins===0)throw H.fail("No dust available"),new Error(`Wallet has no dust on ${$}. Dust is required to pay transaction fees.
|
|
846
|
+
|
|
847
|
+
Register for dust generation:
|
|
848
|
+
Run: midnight dust register --network ${$}
|
|
849
|
+
Then wait: midnight dust status --network ${$}
|
|
850
|
+
`);H.stop(`Wallet OK (${W} NIGHT, dust available)`)}catch(L){throw H.fail("Wallet check failed"),L}}async function ZZ($,Z,Q){let{startServeOrReuse:X}=await Promise.resolve().then(() => (o$(),w7)),z=Z?E1():i("Starting mn serve..."),Y=await X({network:$,wallet:Q,onMessage:(G)=>z.update(G)});return z.stop("mn serve ready (port "+Y.port+")"),{port:Y.port,stop:()=>Y.stop()}}async function UX($){let Z=C($,"json"),Q=x2(A($,"path")??process.cwd()),X=v2($),z=A($,"name"),{runDeploy:Y}=await Promise.resolve().then(() => (D1(),C2)),{resolveNetwork:G}=await Promise.resolve().then(() => (V0(),k4));if(!Z)process.stderr.write(`
|
|
851
|
+
`+g("Contract Deploy")+`
|
|
852
|
+
|
|
853
|
+
`);let{info:J}=G4(X,z),K=J.witnesses.map((W)=>W.name);if(K.length>0){let{findWitnessFile:W,buildMissingWitnessError:F}=await Promise.resolve().then(() => (n$(),u7));if(!W(Q))throw new Error(F({projectRoot:Q,witnessNames:K}))}let q=A($,"network")??"undeployed",{config:V}=G({args:{command:"contract",subcommand:"deploy",positionals:[],flags:{network:q}}}),B=A($,"args"),H=[];if(B)try{let W=JSON.parse(B);H=Array.isArray(W)?W:Object.values(W)}catch(W){throw new Z0(`Invalid --args JSON: ${W.message}`)}if(!Z){if(process.stderr.write(M("Contract",J.name)+`
|
|
854
|
+
`),process.stderr.write(M("Network",q)+`
|
|
855
|
+
`),H.length>0)process.stderr.write(M("Constructor args",String(H.length))+`
|
|
856
|
+
`);process.stderr.write(`
|
|
857
|
+
`)}await $Z(q,Z,A($,"wallet"));let j=await ZZ(q,Z,A($,"wallet")),L=Z?E1():i("Deploying contract...");try{let W=await Y({dappDir:Q,networkConfig:V,managedDir:J.managedDir,contractName:J.name,servePort:j.port,args:H,onMessage:(F)=>L.update(F)});if(L.stop(m("✓")+" Contract deployed"),Z)x({subcommand:"deploy",contractName:J.name,address:W.contractAddress,network:q});else process.stderr.write(`
|
|
858
|
+
`),process.stderr.write(M("Address",W.contractAddress)+`
|
|
344
859
|
`),process.stderr.write(`
|
|
345
|
-
`+
|
|
346
|
-
|
|
347
|
-
`)}
|
|
348
|
-
|
|
349
|
-
`)
|
|
350
|
-
|
|
351
|
-
`)
|
|
352
|
-
`)
|
|
353
|
-
|
|
354
|
-
|
|
355
|
-
|
|
356
|
-
|
|
357
|
-
`)
|
|
358
|
-
|
|
359
|
-
`)
|
|
360
|
-
`+
|
|
361
|
-
|
|
362
|
-
`),
|
|
363
|
-
|
|
364
|
-
|
|
365
|
-
|
|
366
|
-
`)
|
|
367
|
-
|
|
368
|
-
`),process.stderr.write(
|
|
369
|
-
|
|
370
|
-
`)
|
|
371
|
-
`)}},30000);await new Promise(()=>{process.on("SIGINT",()=>{clearInterval(P),process.exit(0)})})}let V=J.reduce((P,x)=>{let F={...Y?.networks[x]??{},...U[x]??{}},O=X2(F);if(O==="down")return"down";if(O==="degraded"&&P!=="down")return"degraded";return P},"up");if(V==="down")process.exitCode=2;else if(V==="degraded")process.exitCode=1}var x8,A8,Z$,E8=15000,L1=5000,Q$=1e4;var Y$=E(()=>{Q0();c();V0();x8={preprod:{faucet:"https://faucet.preprod.midnight.network/",explorer:"https://preprod.midnightexplorer.com/"},preview:{faucet:"https://faucet.preview.midnight.network/",explorer:null},undeployed:{faucet:"",explorer:""}},A8={indexer:"Indexer",rpc:"RPC Node",faucet:"Faucet",explorer:"Explorer",chain:"Chain",dust:"Dust Generation",wallet:"Wallet Ops",dapp:"DApp Flow"},Z$={up:b("UP"),down:a("DOWN"),degraded:R("\x1B[33mDEGRADED\x1B[0m"),unknown:H("—")}});import{Server as k8}from"@modelcontextprotocol/sdk/server/index.js";import{StdioServerTransport as w8}from"@modelcontextprotocol/sdk/server/stdio.js";import{ListToolsRequestSchema as h8,CallToolRequestSchema as f8}from"@modelcontextprotocol/sdk/types.js";async function N($,Z,Q){let X=[];M1((q)=>X.push(q));let z=process.stderr.write;process.stderr.write=()=>!0;try{Z.flags.json=!0,await $(Z,Q);let q=X.join("").trim();if(!q)return{};return JSON.parse(q)}finally{M1(null),process.stderr.write=z}}var n={INVALID_ARGS:"INVALID_ARGS",WALLET_NOT_FOUND:"WALLET_NOT_FOUND",NETWORK_ERROR:"NETWORK_ERROR",INSUFFICIENT_BALANCE:"INSUFFICIENT_BALANCE",TX_REJECTED:"TX_REJECTED",STALE_UTXO:"STALE_UTXO",PROOF_TIMEOUT:"PROOF_TIMEOUT",DUST_REQUIRED:"DUST_REQUIRED",CANCELLED:"CANCELLED",UNKNOWN:"UNKNOWN"};function j2($){let Z=($.message??"").toLowerCase();if(Z.includes("operation cancelled")||Z.includes("operation aborted")||Z==="cancelled"||Z==="aborted")return{exitCode:7,errorCode:n.CANCELLED};if(Z.includes("wallet file not found")||Z.includes("wallet")&&Z.includes("not found"))return{exitCode:3,errorCode:n.WALLET_NOT_FOUND};if(Z.includes("missing required flag")||Z.includes("missing amount")||Z.includes("missing recipient")||Z.includes("missing config key")||Z.includes("missing or invalid subcommand")||Z.includes("unknown command")||Z.includes("cannot specify both")||Z.includes("invalid bip-39")||Z.includes("seed must be")||Z.includes("key index must be")||Z.includes("usage:"))return{exitCode:2,errorCode:n.INVALID_ARGS};if(Z.includes("proof")&&Z.includes("timeout"))return{exitCode:6,errorCode:n.PROOF_TIMEOUT};if(Z.includes("stale utxo")||Z.includes("error code 115")||Z.includes("errorcode: 115"))return{exitCode:6,errorCode:n.STALE_UTXO};if(Z.includes("no dust")||Z.includes("dust")&&(Z.includes("required")||Z.includes("available")||Z.includes("insufficient")))return{exitCode:5,errorCode:n.DUST_REQUIRED};if(Z.includes("insufficient")||Z.includes("not enough"))return{exitCode:5,errorCode:n.INSUFFICIENT_BALANCE};if(Z.includes("rejected")||Z.includes("transaction failed"))return{exitCode:6,errorCode:n.TX_REJECTED};if(Z.includes("econnrefused")||Z.includes("enotfound")||Z.includes("etimedout")||Z.includes("websocket")||Z.includes("connection refused")||Z.includes("network")&&Z.includes("error"))return{exitCode:4,errorCode:n.NETWORK_ERROR};return{exitCode:1,errorCode:n.UNKNOWN}}var m0={name:"midnight-wallet-cli",version:"0.2.5",type:"module",description:"Git-style CLI wallet for the Midnight blockchain",license:"Apache-2.0",workspaces:["packages/*"],bin:{midnight:"dist/wallet.js",mn:"dist/wallet.js","midnight-wallet-cli":"dist/wallet.js","midnight-wallet-mcp":"dist/mcp-server.js"},files:["dist"],scripts:{wallet:"tsx src/wallet.ts",build:'bun build src/wallet.ts --outfile dist/wallet.js --target node --format esm --packages external --minify --banner "#!/usr/bin/env node" && bun build src/mcp-server.ts --outfile dist/mcp-server.js --target node --format esm --packages external --minify --banner "#!/usr/bin/env node"',mcp:"tsx src/mcp-server.ts",prepublishOnly:"npm run build && npm run test",test:"vitest run","test:watch":"vitest",typecheck:"tsc --noEmit"},dependencies:{"@midnight-ntwrk/dapp-connector-api":"^4.0.1","@midnight-ntwrk/ledger-v8":"^8.0.3","@midnight-ntwrk/midnight-js-network-id":"^4.0.2","@midnight-ntwrk/midnight-js-types":"^4.0.2","@midnight-ntwrk/wallet-sdk-abstractions":"^2.0.0","@midnight-ntwrk/wallet-sdk-address-format":"^3.1.0","@midnight-ntwrk/wallet-sdk-capabilities":"^3.2.0","@midnight-ntwrk/wallet-sdk-dust-wallet":"^3.0.0","@midnight-ntwrk/wallet-sdk-facade":"^3.0.0","@midnight-ntwrk/wallet-sdk-hd":"^3.0.1","@midnight-ntwrk/wallet-sdk-shielded":"^2.1.0","@midnight-ntwrk/wallet-sdk-unshielded-wallet":"^2.1.0","@modelcontextprotocol/sdk":"^1.27.1","@scure/bip39":"^2.0.1",rxjs:"^7.8.1",ws:"^8.19.0"},devDependencies:{"@types/node":"^22.19.13","@types/ws":"^8.18.1",tsx:"^4.21.0",typescript:"^5.9.3",vitest:"^3.2.4"}};var t8=m0.name,B2=m0.version,r8=m0.description;function i($,Z,Q){let X={json:!0},z=[];for(let[q,Y]of Object.entries(Z)){if(Y===void 0||Y===null)continue;if(typeof Y==="boolean"){if(Y)X[q]=!0}else X[q]=String(Y)}return{command:$,subcommand:Q,positionals:z,flags:X}}var p8={generate:()=>Promise.resolve().then(() => (m2(),u2)),info:()=>Promise.resolve().then(() => (d2(),g2)),balance:()=>Promise.resolve().then(() => (a2(),r2)),address:()=>Promise.resolve().then(() => ($3(),e2)),"genesis-address":()=>Promise.resolve().then(() => (X3(),Q3)),"inspect-cost":()=>Promise.resolve().then(() => (G3(),Y3)),airdrop:()=>Promise.resolve().then(() => (_3(),M3)),transfer:()=>Promise.resolve().then(() => (x3(),R3)),dust:()=>Promise.resolve().then(() => (C3(),D3)),cache:()=>Promise.resolve().then(() => (S3(),y3)),config:()=>Promise.resolve().then(() => (k3(),v3)),localnet:()=>Promise.resolve().then(() => (n3(),o3)),wallet:()=>Promise.resolve().then(() => ($$(),e3)),status:()=>Promise.resolve().then(() => (Y$(),q$))};async function y($){let Z=p8[$];if(!Z)throw new Error(`Unknown command handler: ${$}`);return(await Z()).default}var G$=[{name:"midnight_generate",description:"Generate a new wallet (deprecated — use midnight_wallet_generate instead)",inputSchema:{type:"object",properties:{network:{type:"string",description:"Network: preprod, preview, undeployed",enum:["preprod","preview","undeployed"]},seed:{type:"string",description:"Restore from existing seed (64-char hex)"},mnemonic:{type:"string",description:"Restore from BIP-39 mnemonic (24 words)"},output:{type:"string",description:"Custom output path (deprecated — use midnight_wallet_generate instead)"},force:{type:"string",description:'Set to "true" to overwrite existing wallet file'}}},async handler($){let Z=i("generate",$);if($.force==="true"||$.force===!0)Z.flags.force=!0;let Q=await y("generate");return N(Q,Z)}},{name:"midnight_wallet_generate",description:"Create a new named wallet and set it as active",inputSchema:{type:"object",properties:{name:{type:"string",description:'Wallet name (e.g. "alice", "dev")'},network:{type:"string",description:"Network: preprod, preview, undeployed",enum:["preprod","preview","undeployed"]},seed:{type:"string",description:"Restore from existing seed (64-char hex)"},mnemonic:{type:"string",description:"Restore from BIP-39 mnemonic (24 words)"},force:{type:"string",description:'Set to "true" to overwrite existing wallet'}},required:["name"]},async handler($){let Z=$.name,Q=i("wallet",$,"generate");if(Q.positionals=[Z],delete Q.flags.name,$.force==="true"||$.force===!0)Q.flags.force=!0;let X=await y("wallet");return N(X,Q)}},{name:"midnight_wallet_list",description:"List all wallets with name, address, network, and active marker",inputSchema:{type:"object",properties:{}},async handler(){let $={command:"wallet",subcommand:"list",positionals:[],flags:{json:!0}},Z=await y("wallet");return N(Z,$)}},{name:"midnight_wallet_use",description:"Set the active wallet by name",inputSchema:{type:"object",properties:{name:{type:"string",description:"Wallet name to activate"}},required:["name"]},async handler($){let Q={command:"wallet",subcommand:"use",positionals:[$.name],flags:{json:!0}},X=await y("wallet");return N(X,Q)}},{name:"midnight_wallet_info",description:"Show details for a named wallet or the active wallet",inputSchema:{type:"object",properties:{name:{type:"string",description:"Wallet name (default: active wallet)"}}},async handler($){let Z=$.name,Q={command:"wallet",subcommand:"info",positionals:Z?[Z]:[],flags:{json:!0}},X=await y("wallet");return N(X,Q)}},{name:"midnight_wallet_remove",description:"Remove a named wallet (refuses active or last wallet)",inputSchema:{type:"object",properties:{name:{type:"string",description:"Wallet name to remove"}},required:["name"]},async handler($){let Q={command:"wallet",subcommand:"remove",positionals:[$.name],flags:{json:!0}},X=await y("wallet");return N(X,Q)}},{name:"midnight_info",description:"Display wallet address, network, creation date (no secrets shown)",inputSchema:{type:"object",properties:{wallet:{type:"string",description:"Wallet name or path"}}},async handler($){let Z=i("info",$),Q=await y("info");return N(Q,Z)}},{name:"midnight_balance",description:"Check unshielded balance via indexer subscription",inputSchema:{type:"object",properties:{address:{type:"string",description:"Address to check (or reads from wallet file)"},wallet:{type:"string",description:"Wallet name or path"},network:{type:"string",description:"Override network detection",enum:["preprod","preview","undeployed"]},"indexer-ws":{type:"string",description:"Custom indexer WebSocket URL"}}},async handler($){let Z=$.address,Q=i("balance",$,Z);delete Q.flags.address;let X=await y("balance");return N(X,Q)}},{name:"midnight_address",description:"Derive and display an unshielded address from a seed",inputSchema:{type:"object",properties:{seed:{type:"string",description:"Seed to derive from (required, 64-char hex)"},network:{type:"string",description:"Network for address prefix",enum:["preprod","preview","undeployed"]},index:{type:"string",description:"Key derivation index (default: 0)"}},required:["seed"]},async handler($){let Z=i("address",$),Q=await y("address");return N(Q,Z)}},{name:"midnight_genesis_address",description:"Display the genesis wallet address (seed 0x01) for a network",inputSchema:{type:"object",properties:{network:{type:"string",description:"Network for address prefix",enum:["preprod","preview","undeployed"]}}},async handler($){let Z=i("genesis-address",$),Q=await y("genesis-address");return N(Q,Z)}},{name:"midnight_inspect_cost",description:"Display current block limits derived from LedgerParameters",inputSchema:{type:"object",properties:{}},async handler(){let $=i("inspect-cost",{}),Z=await y("inspect-cost");return N(Z,$)}},{name:"midnight_airdrop",description:"Fund your wallet from the genesis wallet (undeployed network only)",inputSchema:{type:"object",properties:{amount:{type:"string",description:"Amount in NIGHT to airdrop"},wallet:{type:"string",description:"Wallet name or path"},"no-cache":{type:"string",description:'Set to "true" to bypass wallet state cache'}},required:["amount"]},async handler($){let Z=$.amount,Q=i("airdrop",$,Z);delete Q.flags.amount;let X=await y("airdrop");return N(X,Q)}},{name:"midnight_transfer",description:"Send NIGHT tokens to another address",inputSchema:{type:"object",properties:{to:{type:"string",description:"Recipient bech32m address"},amount:{type:"string",description:"Amount in NIGHT to send"},wallet:{type:"string",description:"Wallet name or path"},"proof-server":{type:"string",description:"Override proof server URL"},node:{type:"string",description:"Override substrate node RPC URL"},"indexer-ws":{type:"string",description:"Override indexer WebSocket URL"},"no-cache":{type:"string",description:'Set to "true" to bypass wallet state cache'}},required:["to","amount"]},async handler($){let{to:Z,amount:Q}=$,X=i("transfer",$,Z);X.positionals=[Q],delete X.flags.to,delete X.flags.amount;let z=await y("transfer");return N(z,X)}},{name:"midnight_dust_register",description:"Register NIGHT UTXOs for dust (fee token) generation",inputSchema:{type:"object",properties:{wallet:{type:"string",description:"Wallet name or path"},"proof-server":{type:"string",description:"Override proof server URL"},node:{type:"string",description:"Override substrate node RPC URL"},"indexer-ws":{type:"string",description:"Override indexer WebSocket URL"},"no-cache":{type:"string",description:'Set to "true" to bypass wallet state cache'}}},async handler($){let Z=i("dust",$,"register"),Q=await y("dust");return N(Q,Z)}},{name:"midnight_dust_status",description:"Check dust registration status and balance",inputSchema:{type:"object",properties:{wallet:{type:"string",description:"Wallet name or path"},"proof-server":{type:"string",description:"Override proof server URL"},node:{type:"string",description:"Override substrate node RPC URL"},"indexer-ws":{type:"string",description:"Override indexer WebSocket URL"},"no-cache":{type:"string",description:'Set to "true" to bypass wallet state cache'}}},async handler($){let Z=i("dust",$,"status"),Q=await y("dust");return N(Q,Z)}},{name:"midnight_config_get",description:"Read a persistent config value",inputSchema:{type:"object",properties:{key:{type:"string",description:'Config key to read (e.g. "network")'}},required:["key"]},async handler($){let Q={command:"config",subcommand:"get",positionals:[$.key],flags:{json:!0}},X=await y("config");return N(X,Q)}},{name:"midnight_config_set",description:"Write a persistent config value",inputSchema:{type:"object",properties:{key:{type:"string",description:'Config key to set (e.g. "network")'},value:{type:"string",description:"Config value to set"}},required:["key","value"]},async handler($){let{key:Z,value:Q}=$,X={command:"config",subcommand:"set",positionals:[Z,Q],flags:{json:!0}},z=await y("config");return N(z,X)}},{name:"midnight_cache_clear",description:"Clear cached wallet sync state",inputSchema:{type:"object",properties:{network:{type:"string",description:"Only clear cache for this network",enum:["preprod","preview","undeployed"]},wallet:{type:"string",description:"Only clear cache for this wallet (name or path)"}}},async handler($){let Z=i("cache",$,"clear"),Q=await y("cache");return N(Q,Z)}},{name:"midnight_config_unset",description:"Reset a persistent config value to its default",inputSchema:{type:"object",properties:{key:{type:"string",description:"Config key to reset"}},required:["key"]},async handler($){let Q={command:"config",subcommand:"unset",positionals:[$.key],flags:{json:!0}},X=await y("config");return N(X,Q)}},{name:"midnight_localnet_up",description:"Start a local Midnight network via Docker Compose",inputSchema:{type:"object",properties:{}},async handler(){let $={command:"localnet",subcommand:"up",positionals:[],flags:{json:!0}},Z=await y("localnet");return N(Z,$)}},{name:"midnight_localnet_stop",description:"Stop local network containers (preserves state for fast restart)",inputSchema:{type:"object",properties:{}},async handler(){let $={command:"localnet",subcommand:"stop",positionals:[],flags:{json:!0}},Z=await y("localnet");return N(Z,$)}},{name:"midnight_localnet_down",description:"Remove local network containers, networks, and volumes (full teardown)",inputSchema:{type:"object",properties:{}},async handler(){let $={command:"localnet",subcommand:"down",positionals:[],flags:{json:!0}},Z=await y("localnet");return N(Z,$)}},{name:"midnight_localnet_status",description:"Show local network service status and ports",inputSchema:{type:"object",properties:{}},async handler(){let $={command:"localnet",subcommand:"status",positionals:[],flags:{json:!0}},Z=await y("localnet");return N(Z,$)}},{name:"midnight_localnet_clean",description:"Remove conflicting containers from other setups",inputSchema:{type:"object",properties:{}},async handler(){let $={command:"localnet",subcommand:"clean",positionals:[],flags:{json:!0}},Z=await y("localnet");return N(Z,$)}}],q2=new k8({name:"midnight-wallet-cli",version:B2},{capabilities:{tools:{}}});q2.setRequestHandler(h8,async()=>{return{tools:G$.map(($)=>({name:$.name,description:$.description,inputSchema:$.inputSchema}))}});q2.setRequestHandler(f8,async($)=>{let{name:Z,arguments:Q}=$.params,X=G$.find((z)=>z.name===Z);if(!X)return{content:[{type:"text",text:`Unknown tool: ${Z}`}],isError:!0};try{let z=await X.handler(Q??{});return{content:[{type:"text",text:JSON.stringify(z,null,2)}]}}catch(z){let q=z instanceof Error?z:new Error(String(z)),{errorCode:Y}=j2(q);return{content:[{type:"text",text:JSON.stringify({error:!0,code:Y,message:q.message})}],isError:!0}}});async function u8(){let $=new w8;await q2.connect($)}u8().catch(($)=>{process.stderr.write(`MCP server error: ${$.message}
|
|
860
|
+
`+m(" ✓ Deploy successful")+`
|
|
861
|
+
`),LX(W.contractAddress,J.circuits,q),process.stdout.write(W.contractAddress+`
|
|
862
|
+
`)}catch(W){throw L.fail("Deploy failed"),W}finally{await j.stop()}}function LX($,Z,Q){let X=Q==="undeployed"?"":` --network ${Q}`;process.stderr.write(`
|
|
863
|
+
`+U(" Next:")+`
|
|
864
|
+
`),process.stderr.write(U(" mn contract state --address ")+l($)+U(X)+`
|
|
865
|
+
`);let z=Z.find((Y)=>Y.arguments.length===0);if(z)process.stderr.write(U(" mn contract call --address ")+l($)+U(" --circuit ")+l(z.name)+U(X)+`
|
|
866
|
+
`);process.stderr.write(`
|
|
867
|
+
`)}async function jX($){let Z=C($,"json"),Q=x2(A($,"path")??process.cwd()),X=v2($),z=A($,"name"),Y=D4($,"address","contract address"),G=D4($,"circuit","circuit name"),J=A($,"args"),{runCall:K}=await Promise.resolve().then(() => (D1(),C2)),{resolveNetwork:q}=await Promise.resolve().then(() => (V0(),k4));if(!Z)process.stderr.write(`
|
|
868
|
+
`+g("Contract Call")+`
|
|
869
|
+
|
|
870
|
+
`);let{info:V}=G4(X,z),B=[];if(J)try{let F=JSON.parse(J);B=Array.isArray(F)?F:Object.values(F)}catch(F){throw new Z0(`Invalid --args JSON: ${F.message}`)}let H=A($,"network")??"undeployed",{config:j}=q({args:{command:"contract",subcommand:"call",positionals:[],flags:{network:H}}});if(!Z)process.stderr.write(M("Contract",V.name)+`
|
|
871
|
+
`),process.stderr.write(M("Circuit",G)+`
|
|
872
|
+
`),process.stderr.write(M("Address",Y.slice(0,20)+"...")+`
|
|
873
|
+
|
|
874
|
+
`);await $Z(H,Z,A($,"wallet"));let L=await ZZ(H,Z,A($,"wallet")),W=Z?E1():i(`Calling ${G}...`);try{let F=await K({dappDir:Q,networkConfig:j,managedDir:V.managedDir,contractName:V.name,contractAddress:Y,circuit:G,args:B,servePort:L.port,onMessage:(O)=>W.update(O)});if(W.stop(m("✓")+` ${G} called`),Z)x({subcommand:"call",contractName:V.name,circuit:G,address:Y,network:H,status:F.status});else process.stderr.write(`
|
|
875
|
+
`+m(" ✓ Circuit call successful")+`
|
|
876
|
+
|
|
877
|
+
`)}catch(F){throw W.fail("Call failed"),F}finally{await L.stop()}}async function OX($){let Z=C($,"json"),Q=x2(A($,"path")??process.cwd()),X=v2($),z=A($,"name"),Y=D4($,"address","contract address"),{resolveNetwork:G}=await Promise.resolve().then(() => (V0(),k4)),{runState:J}=await Promise.resolve().then(() => (D1(),C2)),K=A($,"network")??"undeployed",{config:q}=G({args:{command:"contract",subcommand:"state",positionals:[],flags:{network:K}}});if(!Z)process.stderr.write(`
|
|
878
|
+
`+g("Contract State")+`
|
|
879
|
+
|
|
880
|
+
`),process.stderr.write(M("Address",Y.slice(0,20)+"...")+`
|
|
881
|
+
`),process.stderr.write(M("Network",K)+`
|
|
882
|
+
|
|
883
|
+
`);let V=Z?E1():i("Querying contract state...");try{let{info:B}=G4(X,z),H=await J({dappDir:Q,networkConfig:q,managedDir:B.managedDir,contractName:B.name,contractAddress:Y,onMessage:(j)=>V.update(j)});if(V.stop(m("✓")+" State retrieved"),Z)x({subcommand:"state",address:Y,network:K,...H});else{let j=Object.keys(H.fields).length>0,L=Object.keys(H.maps).length>0;if(j||L){process.stderr.write(y(" Ledger State")+`
|
|
884
|
+
`);for(let[W,F]of Object.entries(H.fields))process.stderr.write(` ${W}: ${l(F)}
|
|
885
|
+
`);for(let[W,F]of Object.entries(H.maps))process.stderr.write(` ${W}: ${U(`Map (${F.size} entries)`)}
|
|
886
|
+
`)}else process.stderr.write(U(" (no ledger fields found)")+`
|
|
887
|
+
`);process.stderr.write(`
|
|
888
|
+
`)}}catch(B){V.stop(e("⚠")+" Parsed state unavailable, showing raw");let{buildStateProvider:H}=await Promise.resolve().then(() => (s7(),r7)),L=await H(q).queryContractState(Y);if(!L)throw new Error(`No contract found at address ${Y} on ${K}`);let W=L.data??L;if(Z)x({subcommand:"state",address:Y,network:K,raw:W});else process.stderr.write(y(" Raw State")+`
|
|
889
|
+
`),process.stderr.write(U(" (run from dApp root for parsed ledger fields)")+`
|
|
890
|
+
`),process.stderr.write(` ${JSON.stringify(W).slice(0,200)}
|
|
891
|
+
|
|
892
|
+
`)}}var t7;var XZ=N(()=>{Y4();H0();n0();M2();t7=["inspect","deploy","call","state"]});import{existsSync as i4,readFileSync as N1,readdirSync as FX}from"node:fs";import{join as M4,resolve as PX}from"node:path";function y2($){let Z=PX($??process.cwd()),Q=M4(Z,e$);if(!i4(Q))throw new Error(`No ${e$} found in ${Z}
|
|
893
|
+
Run this command from the root of a dApp project, or create a ${e$} file.`);let X;try{X=JSON.parse(N1(Q,"utf-8"))}catch(Y){throw new Error(`Failed to parse ${Q}: ${Y.message}`)}return{config:TX(X,Q),configPath:Q,dappDir:Z}}function $3($){let Z=M4($,IX);if(!i4(Z))return[];let Q=FX(Z,{withFileTypes:!0}),X=[];for(let z of Q){if(!z.isDirectory())continue;let Y=M4(Z,z.name),G=M4(Y,"suite.json");if(!i4(G))continue;let J;try{J=JSON.parse(N1(G,"utf-8"))}catch(K){throw new Error(`Failed to parse ${G}: ${K.message}`)}X.push({suite:MX(J,G),suiteDir:Y})}return X}function qZ($){let Z=M4($,"assertions.json");if(!i4(Z))return null;let Q;try{Q=JSON.parse(N1(Z,"utf-8"))}catch(X){throw new Error(`Failed to parse ${Z}: ${X.message}`)}return RX(Q,Z)}function KZ($){let Z=M4($,"actions.json");if(!i4(Z))return null;let Q;try{Q=JSON.parse(N1(Z,"utf-8"))}catch(X){throw new Error(`Failed to parse ${Z}: ${X.message}`)}return AX(Q,Z)}function VZ($){let Z=M4($,"prompt.md");if(!i4(Z))return null;return N1(Z,"utf-8")}function TX($,Z){if(typeof $!=="object"||$===null||Array.isArray($))throw new Error(`${Z}: must be a JSON object`);let Q=$;if(typeof Q.name!=="string"||Q.name.length===0)throw new Error(`${Z}: "name" is required and must be a non-empty string`);if(Q.network!==void 0&&!zZ.includes(Q.network))throw new Error(`${Z}: "network" must be one of: ${zZ.join(", ")}`);if(Q.port!==void 0&&(typeof Q.port!=="number"||Q.port<1||Q.port>65535))throw new Error(`${Z}: "port" must be a number between 1 and 65535`);if(Q.buildCmd!==void 0&&typeof Q.buildCmd!=="string")throw new Error(`${Z}: "buildCmd" must be a string`);if(Q.buildDir!==void 0&&typeof Q.buildDir!=="string")throw new Error(`${Z}: "buildDir" must be a string`);if(Q.url!==void 0&&typeof Q.url!=="string")throw new Error(`${Z}: "url" must be a string`);if(!Array.isArray(Q.prep))throw new Error(`${Z}: "prep" is required and must be an array of prep step IDs`);for(let X of Q.prep)if(typeof X!=="string"||!_X.test(X))throw new Error(`${Z}: invalid prep step "${X}". Valid steps: cache-clear, localnet-up, balance:<amount>, dust-register, dust-wait, mn-serve, build-and-serve`);return{name:Q.name,network:Q.network??"undeployed",port:Q.port,buildCmd:Q.buildCmd,buildDir:Q.buildDir,url:Q.url,contractEntry:Q.contractEntry,prep:Q.prep}}function MX($,Z){if(typeof $!=="object"||$===null||Array.isArray($))throw new Error(`${Z}: must be a JSON object`);let Q=$;if(typeof Q.name!=="string"||Q.name.length===0)throw new Error(`${Z}: "name" is required and must be a non-empty string`);if(typeof Q.description!=="string")throw new Error(`${Z}: "description" is required and must be a string`);if(!YZ.includes(Q.strategy))throw new Error(`${Z}: "strategy" must be one of: ${YZ.join(", ")}`);if(Q.timeout!==void 0&&(typeof Q.timeout!=="number"||Q.timeout<=0))throw new Error(`${Z}: "timeout" must be a positive number (seconds)`);if(Q.browserMode!==void 0&&!GZ.includes(Q.browserMode))throw new Error(`${Z}: "browserMode" must be one of: ${GZ.join(", ")}`);return{name:Q.name,description:Q.description,strategy:Q.strategy,browserMode:Q.browserMode,model:Q.model,effort:Q.effort,timeout:Q.timeout,depends:Array.isArray(Q.depends)?Q.depends:void 0}}function RX($,Z){if(typeof $!=="object"||$===null||Array.isArray($))throw new Error(`${Z}: must be a JSON object`);let Q=$;if(!Array.isArray(Q.post))throw new Error(`${Z}: "post" is required and must be an array of assertion checks`);for(let X of Q.post){if(typeof X!=="object"||X===null)throw new Error(`${Z}: each assertion check must be an object`);let z=X;if(typeof z.id!=="string")throw new Error(`${Z}: each assertion check must have a string "id"`);if(typeof z.type!=="string")throw new Error(`${Z}: assertion "${z.id}" must have a string "type"`);if(typeof z.expect!=="string"||z.expect!=="pass"&&z.expect!=="fail")throw new Error(`${Z}: assertion "${z.id}" must have "expect" set to "pass" or "fail"`)}return{pre:Array.isArray(Q.pre)?Q.pre:void 0,post:Q.post}}function AX($,Z){if(typeof $!=="object"||$===null||Array.isArray($))throw new Error(`${Z}: must be a JSON object`);let Q=$;if(!Array.isArray(Q.actions))throw new Error(`${Z}: "actions" is required and must be an array`);for(let X of Q.actions){if(typeof X!=="object"||X===null)throw new Error(`${Z}: each action must be an object`);let z=X;if(typeof z.id!=="string")throw new Error(`${Z}: each action must have a string "id"`);if(typeof z.type!=="string"||!JZ.includes(z.type))throw new Error(`${Z}: action "${z.id}" has invalid type "${z.type}". Valid: ${JZ.join(", ")}`);if(z.type==="contract-call"&&typeof z.circuit!=="string")throw new Error(`${Z}: action "${z.id}" (contract-call) requires a "circuit" field`)}return{actions:Q.actions}}var e$="dapp.test.json",IX="tests/suites",zZ,YZ,GZ,_X,JZ;var BZ=N(()=>{zZ=["undeployed","preprod","preview"],YZ=["browser","cli","hybrid"],GZ=["dom","script","vision","auto"],_X=/^(cache-clear|localnet-up|balance:\d+|dust|dust-register|dust-wait|mn-serve|build-and-serve)$/;JZ=["contract-deploy","contract-call","contract-state","wallet-cmd"]});function HZ(){let $=[];return{cleanups:$,addCleanup(Z){$.push(Z)}}}import{spawn as DX}from"node:child_process";import{createWriteStream as EX}from"node:fs";import{join as NX}from"node:path";import CX from"node:http";import xX from"node:net";async function UZ($){let{dappDir:Z,buildCmd:Q,buildDir:X=".",port:z,url:Y,logFile:G,timeoutMs:J=240000,onMessage:K=()=>{}}=$;if(await vX(z)){if(await WZ(Y))return K(`Reusing existing dev server on port ${z} (URL responds 200)`),{port:z,child:{kill:()=>!0},stop(){}};K(`Port ${z} in use by an unresponsive process — killing it`),await yX(z),await new Promise((W)=>setTimeout(W,2000))}let q=NX(Z,X),V=EX(G,{flags:"a"}),B=DX("bash",["-lc",Q],{cwd:q,stdio:["ignore","pipe","pipe"],detached:!1});B.stdout?.pipe(V),B.stderr?.pipe(V);let H=!1,j=null;B.on("exit",(W)=>{H=!0,j=W}),K(`Building (${Q})...`);let L=Date.now()+J;while(Date.now()<L){if(H)throw new Error(`Build process exited with code ${j} before the UI was ready.
|
|
894
|
+
Check the build log: ${G}`);if(await WZ(Y))return K("UI ready"),{port:z,child:B,stop(){if(!H)B.kill("SIGTERM"),setTimeout(()=>{if(!H)B.kill("SIGKILL")},5000).unref();V.end()}};await new Promise((W)=>setTimeout(W,2000))}throw B.kill("SIGTERM"),V.end(),new Error(`UI did not start within ${Math.round(J/1000)}s at ${Y}
|
|
895
|
+
Check the build log: ${G}`)}function vX($){return new Promise((Z)=>{let Q=new xX.Socket;Q.setTimeout(1000),Q.on("connect",()=>{Q.destroy(),Z(!0)}),Q.on("timeout",()=>{Q.destroy(),Z(!1)}),Q.on("error",()=>{Z(!1)}),Q.connect($,"127.0.0.1")})}async function yX($){let{execSync:Z}=await import("node:child_process");try{let Q=Z(`lsof -ti:${$}`,{encoding:"utf-8"}).trim();if(Q)Z(`kill ${Q}`,{stdio:"pipe"})}catch{}}function WZ($){return new Promise((Z)=>{let Q=CX.get($,{timeout:3000},(X)=>{Z(X.statusCode===200),X.resume()});Q.on("error",()=>Z(!1)),Q.on("timeout",()=>{Q.destroy(),Z(!1)})})}var LZ=()=>{};import*as PZ from"@midnight-ntwrk/ledger-v8";import{execSync as bX}from"node:child_process";import{join as SX}from"node:path";async function IZ($,Z,Q,X){let z=[];for(let Y of $.prep){let G=Date.now();X.onStepStart(Y);try{await kX(Y,$,Z,Q,X);let J=Date.now()-G;z.push({step:Y,status:"pass",duration:J}),X.onStepComplete(Y,"pass",J)}catch(J){let K=Date.now()-G,q=J.message;throw z.push({step:Y,status:"fail",duration:K,error:q}),X.onStepComplete(Y,"fail",K,q),new Error(`Prep step "${Y}" failed: ${q}`)}}return z}async function kX($,Z,Q,X,z){if($==="cache-clear")return wX(Z);if($==="localnet-up")return mX(z);if($.startsWith("balance:")){let Y=parseInt($.split(":")[1],10);return gX(Y,Z,z)}if($==="dust"||$==="dust-register"||$==="dust-wait")return uX(Z,z);if($==="mn-serve")return pX(Z,X,z);if($==="build-and-serve")return cX(Z,Q,X,z);throw new Error(`Unknown prep step: ${$}`)}async function wX($){let Z=$.network??"undeployed";c0(void 0,Z)}function hX(){try{return bX("curl -s -o /dev/null http://localhost:8088/api/v3/graphql --max-time 3",{timeout:5000,stdio:["pipe","pipe","pipe"]}),!0}catch{return!1}}function OZ($,Z){let Q=Date.now();while(Date.now()-Q<$){if(hX())return!0;Z?.("Waiting for indexer...");let{execSync:X}=r0("node:child_process");try{X("sleep 2",{timeout:3000})}catch{}}return!1}async function mX($){j2(),O2();let Z=["node","indexer","proof-server"],Q=I4();if(Q.length>=3&&Z.every((G)=>Q.some((J)=>J.name===G&&J.state==="running"))&&Q.every((G)=>!G.health||G.health==="healthy")){if(OZ(1e4,(G)=>$.onMessage(G))){$.onMessage("Localnet OK (3 services healthy, indexer responding)");return}$.onMessage("Localnet containers running but indexer not responding. Restarting...")}let z=Q.filter((G)=>G.state==="running").map((G)=>G.name),Y=Z.filter((G)=>!z.includes(G));if(Y.length>0)$.onMessage(`Localnet missing: ${Y.join(", ")}. Restarting clean...`);else $.onMessage("Localnet unhealthy. Restarting clean...");for(let G=1;G<=b2;G++){$.onMessage(`Attempt ${G}/${b2}: tearing down...`),a0("down"),$.onMessage(`Attempt ${G}/${b2}: starting...`),a0("up -d");let J=F2(fX);if(J&&OZ(jZ,(H)=>$.onMessage(H))){$.onMessage("Localnet running (indexer verified)");return}if(J){$.onMessage(`Attempt ${G}: Docker healthy but indexer not responding after ${jZ/1000}s`);continue}let K=I4(),q=Z.filter((H)=>!K.some((j)=>j.name===H&&j.state==="running")),V=K.filter((H)=>H.state==="running"&&H.health&&H.health!=="healthy"),B=[];if(q.length>0)B.push(`not running: ${q.join(", ")}`);if(V.length>0)B.push(`unhealthy: ${V.map((H)=>H.name).join(", ")}`);$.onMessage(`Attempt ${G} failed: ${B.join("; ")||"timeout"}`)}throw new Error(`Localnet failed to start after ${b2} attempts`)}async function gX($,Z,Q){let X=Z.network??"undeployed",{config:z}=q0({args:{command:"test",subcommand:void 0,positionals:[],flags:{network:X}}}),Y=z0(Q0()),G=Buffer.from(Y.seed,"hex"),J=Y.addresses[X];Q.onMessage("Checking balance...");let K=j0((L,W)=>{Q.onMessage(`SDK: ${W}`)}),q=T0(),V=o0(J,X),B=await X4(G,z,V);try{let L=await z4(B,{syncMode:"lite",onProgress:(O,_)=>{if(_>0){let R=Math.min(Math.round(O/_*100),100);Q.onMessage(`Syncing wallet... ${R}%`)}},onSyncDetail:(O)=>{Q.onMessage(`Syncing wallet... (${O})`)}}),W=PZ.unshieldedToken().raw,F=L.unshielded.balances[W]??0n;if(F>0n){Q.onMessage(`Balance OK: ${F}`);try{await p0(J,X,B.facade)}catch{}return}}finally{q(),K();try{await g0(B)}catch{}}if(X!=="undeployed")throw new Error(`Wallet has 0 NIGHT on ${X}. Fund your wallet before running tests:
|
|
896
|
+
mn airdrop ${$} (if faucet available)
|
|
897
|
+
Or transfer NIGHT from another wallet.`);Q.onMessage(`Balance is 0. Airdropping ${$} NIGHT from genesis...`);let H=Buffer.from(t0,"hex"),j=N0(H,X);await c4({seedBuffer:H,networkConfig:z,recipientAddress:J,amountNight:$,onSync(L,W){},onDust(L){Q.onMessage(`Dust: ${L}`)},onProving(){Q.onMessage("Generating ZK proof...")},onSubmitting(){Q.onMessage("Submitting airdrop transaction...")},onSyncWarning(L,W){Q.onMessage(`Syncing genesis... (${W})`)}}),Q.onMessage(`Airdropped ${$} NIGHT`)}async function uX($,Z){let Q=$.network??"undeployed",{config:X}=q0({args:{command:"test",subcommand:void 0,positionals:[],flags:{network:Q}}}),z=z0(Q0()),Y=Buffer.from(z.seed,"hex"),G=z.addresses[Q],J=j0(),K=T0(),q=o0(G,Q),V=await X4(Y,X,q);try{if(Z.onMessage("Syncing wallet..."),await z4(V,{syncMode:"lite"}),Z.onMessage("Ensuring dust..."),(await l0(V,(L)=>Z.onMessage(`Dust: ${L}`))).alreadyAvailable){Z.onMessage("Dust already available");try{await p0(G,Q,V.facade)}catch{}return}Z.onMessage("Dust registered. Waiting for coins to become available...");let H=await G2(V,FZ);if(!(()=>{try{let L=H.dust;return L?.availableCoins?.length>0||L?.balance(new Date)>0n}catch{return!1}})()){if(Q!=="undeployed")throw new Error(`Dust not available on ${Q}. Register dust and wait for it to generate:
|
|
898
|
+
mn dust register
|
|
899
|
+
mn dust status (check until dustAvailable: true)`);throw new Error(`Dust not available after ${FZ/1000}s. The chain may be slow — try again.`)}Z.onMessage("Dust is available");try{await p0(G,Q,V.facade)}catch{}}finally{K(),J();try{await g0(V)}catch{}}}async function pX($,Z,Q){Q.onMessage("Starting mn serve...");let X=await i$({port:void 0,network:$.network,onMessage:(z)=>Q.onMessage(`[serve] ${z}`)});Z.serveHandle=X,Z.addCleanup(async()=>X.stop()),Q.onMessage(`mn serve ready on port ${X.port}`)}async function cX($,Z,Q,X){if(!$.buildCmd){X.onMessage("No buildCmd in config, skipping build");return}let z=$.port??4173,Y=$.url??`http://localhost:${z}/`,G=SX(Z,"tests","results",`build_${Date.now()}.log`),J=await UZ({dappDir:Z,buildCmd:$.buildCmd,buildDir:$.buildDir,port:z,url:Y,logFile:G,onMessage:(K)=>X.onMessage(`[build] ${K}`)});Q.buildHandle=J,Q.addCleanup(async()=>J.stop())}var fX=30000,b2=2,jZ=30000,FZ=90000;var _Z=N(()=>{O4();b0();V0();v$();i0();O4();P4();m0();o$();LZ()});import{execSync as TZ,spawn as lX}from"node:child_process";import{createWriteStream as dX,existsSync as iX,mkdirSync as oX,readFileSync as nX}from"node:fs";import{dirname as aX,join as rX}from"node:path";import{homedir as sX}from"node:os";function tX(){let $=rX(sX(),".claude.json");if(!iX($))return!1;try{let Z=JSON.parse(nX($,"utf-8"));return Boolean(Z.mcpServers?.["chrome-devtools"])}catch{return!1}}function $z($){let Z;try{Z=TZ(`ps -ax -o pid,command | grep "${eX}" | grep -v grep | awk '{print $1}'`,{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim()}catch{return}if(!Z)return;let Q=Z.split(`
|
|
900
|
+
`).filter(Boolean);$(`Found ${Q.length} orphan chrome-devtools-mcp Chrome process(es), killing`);for(let X of Q)try{TZ(`kill ${X}`,{stdio:"pipe"})}catch{}}function Z3($){let Z=$.browserMode??"vision";if(Z==="auto")return"vision";return Z}async function MZ($){let{suite:Z,prompt:Q,dappDir:X,logFile:z,onMessage:Y=()=>{}}=$,G=(Z.timeout??600)*1000,J=Z3(Z);if(J==="dom"||J==="script"){if(!tX())throw new Error(`Browser mode "${J}" requires the chrome-devtools MCP server, which is not registered in ~/.claude.json.
|
|
901
|
+
|
|
902
|
+
Install it:
|
|
903
|
+
claude mcp add chrome-devtools --scope user npx chrome-devtools-mcp@latest
|
|
904
|
+
|
|
905
|
+
Then re-run the suite. Alternatively, switch the suite to "browserMode": "vision" `+"(no MCP needed, but slower — uses screenshots).");$z(Y)}oX(aX(z),{recursive:!0});let K=dX(z),q=[];if(J==="vision")q.push("--chrome");if(q.push("--dangerously-skip-permissions"),Z.model)q.push("--model",Z.model);if(Z.effort)q.push("--effort",Z.effort);let V=Q;if(J==="dom")V=Zz+Q;else if(J==="script")V=Qz+Q;return q.push("-p",V),Y(`Launching Claude (mode: ${J}, model: ${Z.model??"default"}, timeout: ${Z.timeout??600}s)`),new Promise((B)=>{let H=lX("claude",q,{cwd:X,stdio:["inherit","pipe","inherit"]}),j=!1;H.stdout?.on("data",(W)=>{process.stdout.write(W),K.write(W)});let L=setTimeout(()=>{j=!0,Y(`Timeout: test exceeded ${Z.timeout??600}s`),H.kill("SIGTERM"),setTimeout(()=>{if(!H.killed)H.kill("SIGKILL")},1e4).unref()},G);H.on("close",(W)=>{clearTimeout(L),K.end(),B({exitCode:W??1,logFile:z,timedOut:j})}),H.on("error",(W)=>{clearTimeout(L),K.end(),Y(`Claude process error: ${W.message}`),B({exitCode:1,logFile:z,timedOut:!1})})})}var eX="chrome-devtools-mcp/chrome-profile",Zz="## Browser Automation Mode: DOM (Accessibility Tree)\n\nYou have chrome-devtools MCP tools available. You MUST follow these rules:\n\n1. Use `navigate_page` to open URLs in Chrome.\n2. Use `take_snapshot` to read the page — it returns a structured accessibility tree with element UIDs. This is your primary tool for understanding page state.\n3. Use `click`, `fill`, and `type_text` with element UIDs from the snapshot to interact with the page.\n4. Use `press_key` for keyboard input (Enter, Escape, Tab, etc.).\n5. Only use `take_screenshot` for final visual verification or when the accessibility tree is insufficient.\n6. Do NOT rely on screenshots for navigation — the accessibility tree is faster and more reliable.\n7. After each interaction, take a new `take_snapshot` to verify the result.\n\n---\n\n",Qz="## Browser Automation Mode: Script (Canvas App)\n\nYou have chrome-devtools MCP tools available. This is a canvas-based app — the accessibility tree will be empty. You MUST follow these rules:\n\n1. Use `navigate_page` to open the URL in Chrome.\n2. Use `evaluate_script` to dispatch keyboard input via `window.dispatchEvent(new KeyboardEvent(...))`.\n3. Use `evaluate_script` to read DOM state (e.g., toast notifications via `document.querySelector`).\n4. Use `list_console_messages` to monitor app output (contract addresses, errors, status changes).\n5. Use `take_screenshot` at key checkpoints to verify visual state — but minimize usage (a few, not every step).\n6. Do NOT use `take_snapshot` — the accessibility tree is empty for canvas content.\n7. Between actions, use `take_screenshot` or `list_console_messages` to verify transitions — not rapid polling.\n\n---\n\n";var RZ=()=>{};import{existsSync as Xz,readFileSync as zz}from"node:fs";import Yz from"node:net";async function AZ($,Z){let Q=[];for(let X of $)try{let z=await Gz(X,Z),Y=X.expect==="pass"?z:!z;Q.push({id:X.id,status:Y?"pass":"fail",message:Y?void 0:`Expected ${X.expect}, got ${z?"pass":"fail"}`})}catch(z){Q.push({id:X.id,status:"fail",message:z.message})}return Q}async function Gz($,Z){switch($.type){case"balance-changed":return Jz($,Z);case"process-exit-code":return qz($,Z);case"port-listening":return Kz($);case"agent-report-no-failure":return Bz($,Z);default:throw new Error(`Unsupported assertion type: "${$.type}" (available in v2)`)}}function Jz($,Z){if(Z.preBalance===void 0||Z.postBalance===void 0)throw new Error("Balance data not available — balance:N prep step may not have run");let Q=BigInt(Z.preBalance),X=BigInt(Z.postBalance),z=$.params.direction;if(z==="decreased")return X<Q;if(z==="increased")return X>Q;if(z==="unchanged")return X===Q;return X!==Q}function qz($,Z){if(Z.processExitCode===void 0)throw new Error("Process exit code not available");let Q=$.params.code;return Z.processExitCode===Q}function Kz($){let Z=$.params.port;return new Promise((Q)=>{let X=new Yz.Socket;X.setTimeout(2000),X.on("connect",()=>{X.destroy(),Q(!0)}),X.on("timeout",()=>{X.destroy(),Q(!1)}),X.on("error",()=>{Q(!1)}),X.connect(Z,"127.0.0.1")})}function Bz($,Z){let Q=$.params??{},X=Q.logPath??Z.agentLogPath;if(!X||!Xz(X))return!0;let z;try{z=zz(X,"utf-8")}catch{return!0}let Y=Wz(z),G=Q.markers??Vz;for(let J of G)if(Hz(Y,J))return!1;return!0}function Hz($,Z){if(/\s/.test(Z)||/[^ -~]/.test(Z))return $.toLowerCase().includes(Z.toLowerCase());let Q=Z.replace(/[.*+?^${}()|[\]\\]/g,"\\$&");return new RegExp(`\\b${Q}\\b`,"i").test($)}function Wz($){return $.replace(/```[\s\S]*?```/g,"").replace(/`[^`]*`/g,"")}var Vz;var DZ=N(()=>{Vz=["FAILED","❌","did not pass","test failed","overall: fail","result: fail"]});import{existsSync as Uz,mkdirSync as Lz,writeFileSync as jz,readFileSync as Oz,readdirSync as Fz}from"node:fs";import{join as S2}from"node:path";function NZ($,Z){let Q=S2(Z,EZ);Lz(Q,{recursive:!0});let X=`${$.suite}_${$.timestamp.replace(/[:.]/g,"-")}.json`,z=S2(Q,X);return jz(z,JSON.stringify($,null,2)+`
|
|
906
|
+
`),z}function CZ($,Z){let Q=Q3($,Z);return Q.length>0?Q[0]:null}function Q3($,Z){let Q=S2($,EZ);if(!Uz(Q))return[];let X=Fz(Q).filter((Y)=>Y.endsWith(".json")).filter((Y)=>!Z||Y.startsWith(Z+"_")),z=[];for(let Y of X)try{let G=Oz(S2(Q,Y),"utf-8");z.push(JSON.parse(G))}catch{}return z.sort((Y,G)=>G.timestamp.localeCompare(Y.timestamp)),z}var EZ="tests/results";var xZ=()=>{};import{execSync as Pz}from"node:child_process";async function X3($,Z){if($.buildHandle)try{$.buildHandle.stop(),Z?.("Stopped build process")}catch{Z?.("Failed to stop build process (may have already exited)")}if($.serveHandle)try{await $.serveHandle.stop(),Z?.("Stopped mn serve")}catch{Z?.("Failed to stop mn serve")}for(let Q of $.cleanups.reverse())try{await Q()}catch{}if($.buildHandle?.port)Iz($.buildHandle.port,Z)}function Iz($,Z){try{Pz(`osascript -e 'tell application "Google Chrome"
|
|
907
|
+
repeat with theWindow in every window
|
|
908
|
+
repeat with theTab in every tab of theWindow
|
|
909
|
+
if URL of theTab contains "localhost:${$}" then
|
|
910
|
+
close theTab
|
|
911
|
+
end if
|
|
912
|
+
end repeat
|
|
913
|
+
end repeat
|
|
914
|
+
end tell'`,{timeout:5000,stdio:"pipe"}),Z?.(`Closed Chrome tabs for localhost:${$}`)}catch{}}var vZ=()=>{};var fZ={};k(fZ,{placeholderArgsFor:()=>k2,placeholderArg:()=>z3,buildScaffold:()=>Dz,buildPromptMarkdown:()=>wZ});function z3($){switch($["type-name"]){case"Uint":return 0;case"Bytes":{let Z=$.length??32;return Array.from({length:Z},()=>0)}case"Boolean":return!1;case"String":return"test";case"Opaque":return $.tsType==="string"?"test":null;case"Vector":return[];case"Map":return{};case"Set":return[];case"Option":return null;case"Tuple":return($.types??[]).map(z3);default:return null}}function k2($){let Z={};for(let Q of $.arguments)Z[Q.name]=z3(Q.type);return Z}function Dz($,Z){if((Z.strategy??"cli")==="browser"){if(!Z.browser)throw new Error("Browser strategy requires browser options (port, buildCmd, buildDir?, url?)");return Nz(Z,Z.browser)}return Ez($,Z)}function Ez($,Z){let Q=Z.suiteName??_z,X=Z.network??yZ,z=Z.servePort??bZ;return{dappConfig:{name:Z.contractName,network:X,prep:SZ},suite:{name:Q,description:`Auto-generated CLI test suite for ${Z.contractName}. Edit args and add ledger-field assertions to taste.`,strategy:"cli",timeout:Z.timeoutSeconds??Mz},actions:{actions:Cz($)},assertions:{post:[kZ(z)]},prompt:null,suiteName:Q}}function Nz($,Z){let Q=$.suiteName??Tz,X=$.network??yZ,z=$.servePort??bZ,Y=Z.url??`http://localhost:${Z.port}/`,G={name:$.contractName,network:X,port:Z.port,buildCmd:Z.buildCmd,url:Y,prep:Az};if(Z.buildDir)G.buildDir=Z.buildDir;return{dappConfig:G,suite:{name:Q,description:`Auto-generated browser test suite for ${$.contractName}. Edit prompt.md to describe the dApp-specific user flow.`,strategy:"browser",timeout:$.timeoutSeconds??Rz,...Z.browserMode?{browserMode:Z.browserMode}:{}},actions:null,assertions:{post:[{id:"claude-exit-ok",type:"process-exit-code",params:{code:0},expect:"pass"},kZ(z),{id:"agent-no-failure",type:"agent-report-no-failure",params:{},expect:"pass"}]},prompt:wZ($.contractName,Y),suiteName:Q}}function kZ($){return{id:"serve-port-listening",type:"port-listening",params:{port:$},expect:"pass"}}function wZ($,Z){return`You are running an automated E2E test of ${$}.
|
|
915
|
+
|
|
916
|
+
The UI is at ${Z} — open it in Chrome.
|
|
917
|
+
mn serve is running with --approve-all so contract calls will auto-approve.
|
|
918
|
+
|
|
919
|
+
> TODO: replace the steps below with the real user flow for your dApp.
|
|
920
|
+
> Keep them numbered — Claude follows numbered lists more reliably than prose.
|
|
921
|
+
> Use specific text Claude can find on screen ("WALLET OK", "Submit", etc.)
|
|
922
|
+
> rather than vague phrases ("the wallet button").
|
|
923
|
+
|
|
924
|
+
Follow these steps:
|
|
925
|
+
|
|
926
|
+
1. Open ${Z} in Chrome.
|
|
927
|
+
2. Wait for the wallet connection indicator to show success
|
|
928
|
+
(TODO: name the exact text/element your dApp shows on connect).
|
|
929
|
+
3. TODO: dApp-specific actions. Examples:
|
|
930
|
+
- Click the button labelled "Deploy".
|
|
931
|
+
- Fill the input named "amount" with 100.
|
|
932
|
+
- Press Enter to submit.
|
|
933
|
+
4. Wait for the on-chain confirmation.
|
|
934
|
+
Contract calls take 30–60 seconds because of ZK proof generation.
|
|
935
|
+
5. Verify the result on screen
|
|
936
|
+
(TODO: name what your dApp shows after success — a tx hash, a state change, etc.).
|
|
937
|
+
6. Report:
|
|
938
|
+
- Did each step succeed?
|
|
939
|
+
- What was the final on-screen state?
|
|
940
|
+
- Any errors?
|
|
941
|
+
|
|
942
|
+
If any step fails, report exactly what you see on screen and stop.
|
|
943
|
+
Take screenshots at each major step to document the test.
|
|
944
|
+
`}function Cz($){let Z=[{id:"deploy",type:"contract-deploy"},{id:"check-initial",type:"contract-state"}],Q=$.filter((X)=>!X.pure);for(let X of Q){let z={id:`call-${X.name}`,type:"contract-call",circuit:X.name};if(X.arguments.length>0)z.args=k2(X);Z.push(z)}return Z.push({id:"check-final",type:"contract-state"}),Z}var yZ="undeployed",_z="cli-default",Tz="ui-default",Mz=300,Rz=600,bZ=9932,SZ,Az;var Y3=N(()=>{SZ=["cache-clear","localnet-up","balance:1000","dust","mn-serve"],Az=[...SZ,"build-and-serve"]});var gZ={};k(gZ,{writeScaffold:()=>yz});import{existsSync as mZ,mkdirSync as xz,readFileSync as vz,writeFileSync as hZ}from"node:fs";import{join as o4}from"node:path";function yz($,Z){let{dappDir:Q,force:X=!1}=Z,z=o4(Q,"tests","suites",$.suiteName),Y=o4(Q,"dapp.test.json"),G=[{path:o4(z,"suite.json"),kind:"json",body:$.suite},{path:o4(z,"assertions.json"),kind:"json",body:$.assertions}];if($.actions)G.push({path:o4(z,"actions.json"),kind:"json",body:$.actions});if($.prompt)G.push({path:o4(z,"prompt.md"),kind:"text",body:$.prompt});if(!X){let V=G.filter((B)=>mZ(B.path)).map((B)=>B.path);if(V.length>0)throw new Error(`Suite "${$.suiteName}" already exists at ${z}.
|
|
945
|
+
Pick a different suite name (--suite <name>) or pass --force to overwrite.
|
|
946
|
+
Existing files:
|
|
947
|
+
`+V.map((B)=>` - ${B}`).join(`
|
|
948
|
+
`))}let J=bz(Y,$,X);xz(z,{recursive:!0});let K=[],q=[];if(J.action==="write")hZ(Y,JSON.stringify($.dappConfig,null,2)+`
|
|
949
|
+
`),K.push(Y);else q.push(Y);for(let V of G){let B=V.kind==="json"?JSON.stringify(V.body,null,2)+`
|
|
950
|
+
`:V.body;hZ(V.path,B),K.push(V.path)}return{written:K,preserved:q}}function bz($,Z,Q){if(Q||!mZ($))return{action:"write"};let X;try{X=JSON.parse(vz($,"utf-8"))}catch(z){let Y=z instanceof Error?z.message:String(z);throw new Error(`${$} exists but is not valid JSON: ${Y}
|
|
951
|
+
Fix or remove it (or pass --force to overwrite).`)}if(X.name!==Z.dappConfig.name)throw new Error(`${$} exists with name "${String(X.name)}" but this scaffold targets "${Z.dappConfig.name}". Use --force to overwrite, or run from the right project directory.`);return{action:"preserve"}}var uZ=()=>{};var dZ={};k(dZ,{discoverScreensInDir:()=>G3,discoverScreens:()=>uz});import{existsSync as cZ,readdirSync as Sz,readFileSync as kz,statSync as wz}from"node:fs";import{basename as fz,extname as hz,join as lZ,relative as mz}from"node:path";function uz($){let Z=new Map;for(let Q of gz){let X=lZ($,Q);if(!cZ(X))continue;J3(X,$,Z)}return[...Z.values()].sort((Q,X)=>Q.name.localeCompare(X.name))}function G3($,Z=$){if(!cZ($))return[];let Q=new Map;return J3($,Z,Q),[...Q.values()].sort((X,z)=>X.name.localeCompare(z.name))}function J3($,Z,Q){let X;try{X=Sz($)}catch{return}for(let z of X){let Y=lZ($,z),G;try{G=wz(Y)}catch{continue}if(G.isDirectory()){if(pz.has(z.toLowerCase()))continue;J3(Y,Z,Q);continue}if(hz(z)!==pZ)continue;if(z.endsWith(".test.tsx")||z.endsWith(".spec.tsx"))continue;let J=fz(z,pZ);if(!cz(J,Y))continue;let K=dz(J);if(Q.has(K))continue;Q.set(K,{name:K,component:J,path:Y,relativePath:mz(Z,Y)})}}function cz($,Z){if(!/^[A-Z][A-Za-z0-9]*$/.test($))return!1;let Q;try{Q=kz(Z,"utf-8")}catch{return!1}return lz.test(Q)}function dz($){return $.replace(/([a-z0-9])([A-Z])/g,"$1-$2").replace(/([A-Z]+)([A-Z][a-z])/g,"$1-$2").toLowerCase()}var gz,pZ=".tsx",pz,lz;var q3=N(()=>{gz=["src/components","src/pages","src/screens","src/views"];pz=new Set(["layout","layouts","shared","common","ui","icons","styles"]);lz=/\bexport\s+(default\b|const\s+[A-Z]|function\s+[A-Z])/});var aZ={};k(aZ,{promptSuiteName:()=>$Y,promptStrategy:()=>az,promptScreen:()=>nZ,promptGoal:()=>ez,promptCircuit:()=>sz,promptBrowserOptions:()=>rz,promptBrowserMode:()=>oZ,isInteractive:()=>nz});import{existsSync as iz,statSync as oz}from"node:fs";import*as iZ from"node:readline";function nz(){return process.stdin.isTTY===!0}function h0($,Z){let Q=iZ.createInterface({input:process.stdin,output:process.stderr}),X=Z!==void 0?U(` [${Z}]`):"";return new Promise((z)=>{Q.question(` ${$}${X}: `,(Y)=>{Q.close();let G=Y.trim();z(G===""&&Z!==void 0?Z:G)})})}async function az(){process.stderr.write(`
|
|
952
|
+
`+l("Test scaffold setup")+`
|
|
953
|
+
`);for(;;){let $=(await h0("Strategy? cli or ui","cli")).toLowerCase();if($==="cli")return"cli";if($==="ui"||$==="browser")return"browser";process.stderr.write(U(` Unknown strategy "${$}". Pick "cli" or "ui".
|
|
954
|
+
`))}}async function rz($={}){process.stderr.write(`
|
|
955
|
+
`+U("Browser test scaffold needs your dApp UI details:")+`
|
|
956
|
+
`);let Z=$.port!==void 0?String($.port):await h0("Dev server port","4173"),Q=parseInt(Z,10);if(!Number.isFinite(Q)||Q<=0||Q>65535)throw new Error(`Invalid port "${Z}" — must be a number between 1 and 65535.`);let X=$.buildCmd??await h0("Build command","npm run dev");if(!X)throw new Error("Build command is required for browser strategy.");let z=$.buildDir!==void 0?$.buildDir:await h0("Build dir (blank for project root)",""),Y=z===""?void 0:z,G=$.url??await h0("URL",`http://localhost:${Q}/`),J=$.browserMode??await oZ();return{port:Q,buildCmd:X,buildDir:Y,url:G,browserMode:J}}async function oZ(){process.stderr.write(`
|
|
957
|
+
`+l("Browser mode — how should Claude perceive the page?")+`
|
|
958
|
+
`),process.stderr.write(` ${U("1.")} dom ${U("— accessibility tree (text). Fast. HTML / React / Vue UIs.")}
|
|
959
|
+
`),process.stderr.write(` ${U("2.")} vision ${U("— screenshots. Slow. Canvas games (no DOM).")}
|
|
960
|
+
`),process.stderr.write(` ${U("3.")} script ${U("— direct JS. Fastest. Needs dApp-side hooks (advanced).")}
|
|
961
|
+
`),process.stderr.write(U(` dom and script need chrome-devtools-mcp installed.
|
|
962
|
+
`));for(;;){let $=(await h0("Mode","dom")).toLowerCase().trim();if($==="1"||$==="dom")return"dom";if($==="2"||$==="vision")return"vision";if($==="3"||$==="script")return"script";process.stderr.write(e(` Pick 1 (dom), 2 (vision), or 3 (script).
|
|
963
|
+
`))}}async function sz($){let Z=$.filter((Q)=>!Q.pure);if(Z.length===0){process.stderr.write(U(` No impure circuits in this contract — nothing to test as a write path.
|
|
964
|
+
`));return}process.stderr.write(`
|
|
965
|
+
`+l("Pick the circuit this suite will exercise:")+`
|
|
966
|
+
`);for(let Q=0;Q<Z.length;Q++)process.stderr.write(` ${U(`${Q+1}.`)} ${Z[Q].name}
|
|
967
|
+
`);process.stderr.write(U(` (or type "skip" to scaffold all circuits without AI)
|
|
968
|
+
`));for(;;){let Q=(await h0("Circuit","1")).trim();if(Q.toLowerCase()==="skip")return;let X=parseInt(Q,10);if(Number.isFinite(X)&&X>=1&&X<=Z.length)return Z[X-1];let z=Z.find((Y)=>Y.name===Q);if(z)return z;process.stderr.write(U(` Pick a number 1–${Z.length}, the circuit name, or "skip".
|
|
969
|
+
`))}}async function nZ($){if($.length===0)return tz();process.stderr.write(`
|
|
970
|
+
`+l("Pick the screen this suite will test:")+`
|
|
971
|
+
`);for(let Z=0;Z<$.length;Z++)process.stderr.write(` ${U(`${Z+1}.`)} ${$[Z].name} ${U(`(${$[Z].relativePath})`)}
|
|
972
|
+
`);process.stderr.write(U(` (or type "skip" to scaffold a generic prompt.md without AI)
|
|
973
|
+
`));for(;;){let Z=(await h0("Screen","1")).trim();if(Z.toLowerCase()==="skip")return;let Q=parseInt(Z,10);if(Number.isFinite(Q)&&Q>=1&&Q<=$.length)return $[Q-1];let X=$.find((z)=>z.name===Z||z.component===Z);if(X)return X;process.stderr.write(U(` Pick a number 1–${$.length}, the screen name, or "skip".
|
|
974
|
+
`))}}async function tz(){process.stderr.write(U(`
|
|
975
|
+
Could not auto-discover screens (no src/components, src/pages, or src/screens under the build dir).
|
|
976
|
+
`));for(;;){let $=(await h0('Path to a .tsx file or a UI directory (or "skip")')).trim();if($.toLowerCase()==="skip"||$==="")return;if(!iz($)){process.stderr.write(U(` No such path: ${$}
|
|
977
|
+
`));continue}let Z;try{Z=oz($)}catch{process.stderr.write(U(` Cannot stat: ${$}
|
|
978
|
+
`));continue}if(Z.isDirectory()){let Q=G3($);if(Q.length===0){process.stderr.write(U(` No PascalCase .tsx components in ${$}
|
|
979
|
+
`));continue}return nZ(Q)}if(!$.endsWith(".tsx")){process.stderr.write(U(` Not a .tsx file: ${$}
|
|
980
|
+
`));continue}return QY($)}}async function ez(){let $=(await h0("What does success look like? (one line, optional)")).trim();return $===""?void 0:$}async function $Y($){for(;;){let Z=(await h0("Suite name",$)).trim();if(ZY(Z))return Z;process.stderr.write(e(` Suite names must be kebab-case (a–z, 0–9, dashes, underscores). Try again.
|
|
981
|
+
`))}}function ZY($){return/^[a-z0-9][a-z0-9_-]{0,63}$/.test($)}function QY($){let Q=($.split("/").pop()??$).replace(/\.tsx$/,"");return{name:Q.replace(/([a-z0-9])([A-Z])/g,"$1-$2").replace(/([A-Z]+)([A-Z][a-z])/g,"$1-$2").toLowerCase(),component:Q,path:$,relativePath:$}}var rZ=N(()=>{q3()});function eZ($){let{contractName:Z,contractSummary:Q,contractSource:X,targetCircuit:z,goal:Y,witnessDependentCircuits:G}=$,J=G&&G.size>0?`
|
|
982
|
+
## Circuits NOT testable from a CLI suite
|
|
983
|
+
|
|
984
|
+
These circuits call witnesses that read from private state. In a
|
|
985
|
+
CLI test the private state is empty (no UI to populate it), so
|
|
986
|
+
they crash with "Cannot read properties of undefined". Do NOT
|
|
987
|
+
include them as actions, even as setup steps:
|
|
988
|
+
|
|
989
|
+
`+[...G.entries()].map(([V,B])=>` - ${V} (uses ${B.join(", ")})`).join(`
|
|
990
|
+
`)+`
|
|
991
|
+
${z&&G.has(z.name)?`
|
|
992
|
+
The user explicitly asked for "${z.name}" — try anyway,
|
|
993
|
+
`+`but expect a runtime witness-undefined error and call that out
|
|
994
|
+
in the description so the user knows the suite needs the UI to
|
|
995
|
+
populate state first.
|
|
996
|
+
`:""}`:"",K=z?`## Target
|
|
997
|
+
|
|
998
|
+
Focus this suite on ONE circuit:
|
|
999
|
+
|
|
1000
|
+
Circuit: ${z.name} (${z.pure?"pure":"impure"})
|
|
1001
|
+
Args:
|
|
1002
|
+
${z.arguments.length===0?" (takes no arguments)":z.arguments.map((V)=>` - ${V.name}: ${V.type["type-name"]}`).join(`
|
|
1003
|
+
`)}
|
|
1004
|
+
${$.startingArgs&&Object.keys($.startingArgs).length>0?`
|
|
1005
|
+
Starting args (REQUIRED — keep these exact field names; replace
|
|
1006
|
+
placeholder values with values that satisfy the goal):
|
|
1007
|
+
|
|
1008
|
+
\`\`\`json
|
|
1009
|
+
${JSON.stringify($.startingArgs,null,2)}
|
|
1010
|
+
\`\`\`
|
|
1011
|
+
|
|
1012
|
+
If the contract source declares Struct or Alias arg types, find the
|
|
1013
|
+
inner field names by reading the source below — do NOT invent them.
|
|
1014
|
+
For Struct args, the JSON object MUST have the inner fields (e.g.
|
|
1015
|
+
\`providerPk: { x: ..., y: ... }\`); leaving inner fields out crashes
|
|
1016
|
+
the runtime with "Cannot read properties of undefined".
|
|
1017
|
+
`:""}
|
|
1018
|
+
For circuits that need prior state (e.g. registerProvider before
|
|
1019
|
+
requestLoan), include the setup actions explicitly. Each setup action
|
|
1020
|
+
must follow the same arg-shape rules.`:`## Target
|
|
1021
|
+
|
|
1022
|
+
No specific circuit was selected — pick the impure circuit (or short
|
|
1023
|
+
chain of circuits) from the contract that best matches the goal below.
|
|
1024
|
+
Prefer the most representative "main path" circuit. If the goal
|
|
1025
|
+
implies setup (e.g. registerProvider before requestLoan), include the
|
|
1026
|
+
setup actions explicitly.`,q=Y?`Success criterion (from the user):
|
|
1027
|
+
> ${Y}
|
|
1028
|
+
`:`No specific success criterion was given. Pick a reasonable one and reflect it in the suite description.
|
|
1029
|
+
`;return`You are scaffolding a CLI test suite for the Midnight contract \`${Z}\`.
|
|
1030
|
+
Strategy is "cli" — no browser. The test will run via \`mn test run\`, which
|
|
1031
|
+
deploys the contract, executes the actions in order, and asserts on the
|
|
1032
|
+
final ledger state.
|
|
1033
|
+
|
|
1034
|
+
${K}
|
|
1035
|
+
${J}
|
|
1036
|
+
${q}
|
|
1037
|
+
|
|
1038
|
+
## Contract context
|
|
1039
|
+
|
|
1040
|
+
${Q}
|
|
1041
|
+
|
|
1042
|
+
${X?`
|
|
1043
|
+
--- contract source ---
|
|
1044
|
+
${X}
|
|
1045
|
+
--- end source ---
|
|
1046
|
+
`:""}
|
|
1047
|
+
|
|
1048
|
+
## What to produce
|
|
1049
|
+
|
|
1050
|
+
A JSON object with EXACTLY this shape (top-level keys: actions, assertions, description):
|
|
1051
|
+
|
|
1052
|
+
\`\`\`json
|
|
1053
|
+
{
|
|
1054
|
+
"description": "one short sentence",
|
|
1055
|
+
"actions": [
|
|
1056
|
+
{ "id": "deploy", "type": "contract-deploy" },
|
|
1057
|
+
{ "id": "<short-slug>", "type": "contract-call", "circuit": "<name>", "args": { ... } },
|
|
1058
|
+
{ "id": "<short-slug>", "type": "contract-state", "assert": { "<fieldOrMap>": { "<op>": <value> } } }
|
|
1059
|
+
],
|
|
1060
|
+
"assertions": {
|
|
1061
|
+
"post": [
|
|
1062
|
+
{ "id": "serve-port-listening", "type": "port-listening", "params": { "port": 9932 }, "expect": "pass" }
|
|
1063
|
+
]
|
|
1064
|
+
}
|
|
1065
|
+
}
|
|
1066
|
+
\`\`\`
|
|
1067
|
+
|
|
1068
|
+
Notes:
|
|
1069
|
+
- \`actions\` is a top-level ARRAY of action objects (not nested under another \`actions\` key).
|
|
1070
|
+
- Always start with \`{ "id": "deploy", "type": "contract-deploy" }\`.
|
|
1071
|
+
- End with a contract-state read so the final assertion runs on the result.
|
|
1072
|
+
- For circuits that need prior state (e.g. registerProvider before requestLoan),
|
|
1073
|
+
include the setup actions explicitly between deploy and the target call.
|
|
1074
|
+
|
|
1075
|
+
Action shape:
|
|
1076
|
+
{ "id": "<short-slug>", "type": "contract-deploy" }
|
|
1077
|
+
{ "id": "<short-slug>", "type": "contract-state", "assert": { "<fieldOrMap>": { "<op>": <value> } } }
|
|
1078
|
+
{ "id": "<short-slug>", "type": "contract-call", "circuit": "<name>", "args": { ... } }
|
|
1079
|
+
|
|
1080
|
+
Args coercion the runner applies (so JSON arg values can be):
|
|
1081
|
+
- number → BigInt for Uint<N>
|
|
1082
|
+
- "123n" string → BigInt for values beyond Number.MAX_SAFE_INTEGER
|
|
1083
|
+
- [0..255] int array → Uint8Array for Bytes<N>
|
|
1084
|
+
- object → recurses into Struct fields
|
|
1085
|
+
|
|
1086
|
+
${tZ}
|
|
1087
|
+
|
|
1088
|
+
Now produce the JSON.`}function $5($){let{contractName:Z,contractSummary:Q,screenComponent:X,screenSource:z,relatedSources:Y,url:G,goal:J}=$,K=(Y??[]).map((B)=>`
|
|
1089
|
+
--- ${B.path} ---
|
|
1090
|
+
${B.source}
|
|
1091
|
+
--- end ${B.path} ---`).join(`
|
|
1092
|
+
`),q=X&&z?`## Target screen
|
|
1093
|
+
|
|
1094
|
+
Component: ${X}
|
|
1095
|
+
Served at: ${G}
|
|
1096
|
+
|
|
1097
|
+
## Screen source
|
|
1098
|
+
|
|
1099
|
+
\`\`\`tsx
|
|
1100
|
+
${z}
|
|
1101
|
+
\`\`\`
|
|
1102
|
+
${K?`
|
|
1103
|
+
## Imported components from the same UI
|
|
1104
|
+
${K}
|
|
1105
|
+
`:""}`:`## Target
|
|
1106
|
+
|
|
1107
|
+
No specific screen was selected — generate a generic Midnight dApp
|
|
1108
|
+
flow grounded in the contract circuits + the user's goal below.
|
|
1109
|
+
The dApp is served at ${G}.
|
|
1110
|
+
|
|
1111
|
+
Conventional patterns Midnight dApps follow that you can rely on:
|
|
1112
|
+
- A "Connect Wallet" button in the header.
|
|
1113
|
+
- After connect, a contract-address paste field labelled near
|
|
1114
|
+
"Contract address" with a "Connect →" or "Link" action.
|
|
1115
|
+
- Per-circuit forms with a labelled "Submit" / "Send" / "Run" action
|
|
1116
|
+
and progress feedback during ZK proof generation (30–90 s).
|
|
1117
|
+
- A history / state panel that updates after a successful tx.
|
|
1118
|
+
- Toast or inline error messages on failure.
|
|
1119
|
+
|
|
1120
|
+
Use generic-but-specific selectors (\`getByRole('button', { name: /connect wallet/i })\`)
|
|
1121
|
+
in the prompt steps, since you don't have the actual JSX.`,V=J?`Success criterion (from the user):
|
|
1122
|
+
> ${J}
|
|
1123
|
+
`:X?`No specific success criterion was given. Infer one from the screen source: what is the obvious "happy path" outcome a user would see on success?
|
|
1124
|
+
`:`No specific success criterion was given. Infer one from the contract circuits: what is the obvious "happy path" outcome the dApp exists to support?
|
|
1125
|
+
`;return`You are scaffolding a browser test for the Midnight dApp \`${Z}\`.
|
|
1126
|
+
The test runs via \`mn test run\` — Claude (you, in the test session)
|
|
1127
|
+
drives Chrome and follows a prompt.md file.
|
|
1128
|
+
|
|
1129
|
+
${q}
|
|
1130
|
+
|
|
1131
|
+
${V}
|
|
1132
|
+
|
|
1133
|
+
## Contract context (so you know what circuits the dApp calls)
|
|
1134
|
+
|
|
1135
|
+
${Q}
|
|
1136
|
+
|
|
1137
|
+
## What to produce
|
|
1138
|
+
|
|
1139
|
+
A JSON object with three keys:
|
|
1140
|
+
|
|
1141
|
+
- prompt: the markdown body of prompt.md the test runner will hand to
|
|
1142
|
+
Claude in the actual test session. Numbered steps, terse.
|
|
1143
|
+
${z?`CRITICAL: reference the EXACT button labels and field labels that appear in the source above ("Save PIN", "Request loan →", etc.). Don't invent labels.`:"Use the conventional Midnight dApp patterns above. The test session's Claude will adapt to the dApp's actual labels."}
|
|
1144
|
+
Steps should:
|
|
1145
|
+
1. open ${G} in Chrome
|
|
1146
|
+
2. wait for any wallet/connection state the dApp needs
|
|
1147
|
+
3. perform the on-screen actions for the goal
|
|
1148
|
+
4. verify the success criterion on screen
|
|
1149
|
+
5. report pass/fail per step + final on-screen text
|
|
1150
|
+
Keep it under 25 lines.
|
|
1151
|
+
- assertions: { "post": [ ... ] } — at minimum
|
|
1152
|
+
{ "id": "claude-exit-ok", "type": "process-exit-code",
|
|
1153
|
+
"params": { "code": 0 }, "expect": "pass" },
|
|
1154
|
+
{ "id": "serve-port-listening", "type": "port-listening",
|
|
1155
|
+
"params": { "port": 9932 }, "expect": "pass" },
|
|
1156
|
+
and
|
|
1157
|
+
{ "id": "agent-no-failure", "type": "agent-report-no-failure",
|
|
1158
|
+
"params": {}, "expect": "pass" }
|
|
1159
|
+
(this last one parses your final report — without it, you
|
|
1160
|
+
writing "FAILED" still counts as a pass).
|
|
1161
|
+
Add others if the success state can be checked from the
|
|
1162
|
+
chain (e.g. ledger-field on a contract-state read).
|
|
1163
|
+
- description: one short sentence summarising what this suite verifies.
|
|
1164
|
+
|
|
1165
|
+
${tZ}
|
|
1166
|
+
|
|
1167
|
+
Now produce the JSON.`}function K3($){let{name:Z,circuits:Q,witnesses:X}=$,z=[`Contract: ${Z}`];if(Q.length>0){z.push("","Circuits:");for(let Y of Q){let G=Y.arguments.length===0?"()":"("+Y.arguments.map((J)=>`${J.name}: ${J.type["type-name"]}`).join(", ")+")";z.push(` ${Y.pure?"pure ":"impure"} ${Y.name}${G}`)}}if(X.length>0){z.push("","Witnesses:");for(let Y of X){let G=Y.arguments.length===0?"()":"("+Y.arguments.map((J)=>`${J.name}: ${J.type["type-name"]}`).join(", ")+")";z.push(` ${Y.name}${G}`)}}return z.join(`
|
|
1168
|
+
`)}var sZ="```json",tZ;var Z5=N(()=>{tZ=`Output rules:
|
|
1169
|
+
- Reply with EXACTLY one ${"```json"} ... \`\`\` fenced code block.
|
|
1170
|
+
- Inside the fence: valid JSON, nothing else (no comments, no trailing text).
|
|
1171
|
+
- Fields you don't know about: omit them. Don't invent fields.
|
|
1172
|
+
- Use only the circuit / ledger field names that appear in the inputs below.
|
|
1173
|
+
- Be concise — fewer, well-named actions beat many vague ones.`});var Q5={};k(Q5,{analyzeWitnessDependencies:()=>C1});function C1($,Z){if(Z.length===0)return{byCircuit:new Map};let Q=YY($),X=GY(Q);if(X.size===0)return{byCircuit:new Map};let z=new Set(Z),Y=new Map,G=new Map;for(let[q,V]of X){Y.set(q,new Set),G.set(q,new Set);for(let B of qY(V.body,V.argNames))if(z.has(B))Y.get(q).add(B);else if(X.has(B))G.get(q).add(B)}let J=!0;while(J){J=!1;for(let[q,V]of G){let B=Y.get(q);for(let H of V)for(let j of Y.get(H)??[])if(!B.has(j))B.add(j),J=!0}}let K=new Map;for(let[q,V]of X){if(!V.isCircuit)continue;let B=Y.get(q);if(B&&B.size>0)K.set(q,[...B].sort())}return{byCircuit:K}}function YY($){return $.replace(zY,"").replace(XY,"")}function GY($){let Z=new Map,Q=/\b(circuit|function)\s+([A-Za-z_][A-Za-z0-9_]*)\s*\(([^)]*)\)/gms,X;while((X=Q.exec($))!==null){let z=X[1],Y=X[2],G=X[3],J=X.index+X[0].length,K=$.indexOf("{",J);if(K<0)continue;let q=1,V=K+1;for(;V<$.length&&q>0;V++){let j=$[V];if(j==="{")q++;else if(j==="}")q--}if(q!==0)continue;let B=$.slice(K+1,V-1),H=JY(G);Z.set(Y,{body:B,argNames:H,isCircuit:z==="circuit"})}return Z}function JY($){let Z=new Set;for(let Q of $.split(",")){let X=Q.trim();if(!X)continue;let z=X.indexOf(":"),Y=z>-1?X.slice(0,z).trim():X;if(/^[A-Za-z_][A-Za-z0-9_]*$/.test(Y))Z.add(Y)}return Z}function qY($,Z){let Q=new Set,X=/\b([A-Za-z_][A-Za-z0-9_]*)\s*\(/g,z;while((z=X.exec($))!==null){let Y=z[1];if(KY.has(Y)||Z.has(Y))continue;Q.add(Y)}return Q}var XY,zY,KY;var w2=N(()=>{XY=/\/\/[^\n]*/g,zY=/\/\*[\s\S]*?\*\//g;KY=new Set(["if","for","while","switch","return","new","assert","disclose","transientHash","persistentHash","ownPublicKey","publicKey","Map","Set","Vector","Tuple","Option","Bytes","Field","Uint","Boolean"])});var H5={};k(H5,{isClaudeAvailable:()=>jY,generateUiScaffoldWithAI:()=>TY,generateCliScaffoldWithAI:()=>_Y,findContractSourcePath:()=>H3,claudeSubprocessRunner:()=>B3});import{execFileSync as VY,spawn as BY}from"node:child_process";import{existsSync as V3,readdirSync as HY,readFileSync as Y5}from"node:fs";import{dirname as n4,join as f2}from"node:path";function jY(){try{return VY("claude",["--version"],{stdio:"ignore",timeout:5000}),!0}catch{return!1}}function K5($){let Q=new RegExp("```json\\s*\\n([\\s\\S]*?)\\n```").exec($);if(!Q)throw new Error(`Claude response did not contain a ${sZ} ... \`\`\` fenced block. First 200 chars: ${$.slice(0,200)}`);try{return JSON.parse(Q[1])}catch(X){let z=X instanceof Error?X.message:String(X);throw new Error(`JSON inside the fence failed to parse: ${z}`)}}function OY($,Z){if(!$||typeof $!=="object")throw new Error("AI response must be an object");if(typeof $.description!=="string"||!$.description)throw new Error("AI response missing string `description`");if(!$.assertions||!Array.isArray($.assertions.post))throw new Error("AI response missing `assertions.post` array");let Q=PY($);if(!Q)throw new Error("AI response missing actions — expected either `actions: [...]` (top-level array) "+"or `actions: { actions: [...] }` (wrapped). Got: "+JSON.stringify($.actions??null).slice(0,200));let X=new Map(Z.circuits.map((z)=>[z.name,z]));for(let z of Q){if(z.type!=="contract-call"||!z.circuit)continue;let Y=X.get(z.circuit);if(!Y)throw new Error(`AI response references unknown circuit "${z.circuit}". Valid circuits: ${[...X.keys()].join(", ")||"(none)"}`);FY(z,Y)}return{actions:Q}}function FY($,Z){if(Z.arguments.length===0)return;let Q=$.args??{};for(let X of Z.arguments){let z=Q[X.name];if(z===void 0)throw new Error(`Action "${$.id}" (circuit ${Z.name}) is missing required arg "${X.name}" of type ${X.type["type-name"]}. Declared args: ${Z.arguments.map((G)=>`${G.name}: ${G.type["type-name"]}`).join(", ")}`);let Y=X.type["type-name"];if((Y==="Struct"||Y==="Alias")&&(z===null||typeof z!=="object"||Array.isArray(z)))throw new Error(`Action "${$.id}" (circuit ${Z.name}) arg "${X.name}" must be an object for ${Y} type; got ${z===null?"null":typeof z}. Read the contract source to find the inner field names.`)}}function PY($){if(Array.isArray($.actions))return $.actions;let Z=$.actions;if(Z&&Array.isArray(Z.actions))return Z.actions;return null}function IY($){if(!$||typeof $!=="object")throw new Error("AI response must be an object");if(typeof $.description!=="string"||!$.description)throw new Error("AI response missing string `description`");if(typeof $.prompt!=="string"||!$.prompt.trim())throw new Error("AI response missing non-empty `prompt`");if(!$.assertions||!Array.isArray($.assertions.post))throw new Error("AI response missing `assertions.post` array")}async function _Y($,Z=B3){let Q=$.contractSourcePath&&V3($.contractSourcePath)?Y5($.contractSourcePath,"utf-8"):void 0,X=$.contract.witnesses.map((O)=>O.name),z=Q?C1(Q,X).byCircuit:new Map,Y=$.targetCircuit?k2($.targetCircuit):void 0,G=eZ({contractName:$.contract.name,contractSummary:K3({name:$.contract.name,circuits:$.contract.circuits,witnesses:$.contract.witnesses}),contractSource:Q,targetCircuit:$.targetCircuit,startingArgs:Y,witnessDependentCircuits:z,goal:$.goal}),J=await Z(G),K=K5(J),q=OY(K,$.contract),V=$.suiteName??($.targetCircuit?`cli-${$.targetCircuit.name.toLowerCase().replace(/_/g,"-")}`:`cli-${B5($.goal)??"ai"}`),B=$.network??G5,H=$.servePort??J5,j={name:$.contract.name,network:B,prep:q5},L={name:V,description:K.description,strategy:"cli",timeout:WY},W=MY(K.assertions,H),F=[...z.entries()].filter(([O])=>O!==$.targetCircuit?.name).map(([O,_])=>({name:O,reason:`uses witness ${_.join(", ")} (private state — UI-only)`}));return{dappConfig:j,suite:L,actions:q,assertions:W,prompt:null,suiteName:V,excludedCircuits:F.length>0?F:void 0}}async function TY($,Z=B3){let Q=$.screen?Y5($.screen.path,"utf-8"):void 0,X=$5({contractName:$.contract.name,contractSummary:K3({name:$.contract.name,circuits:$.contract.circuits,witnesses:$.contract.witnesses}),screenComponent:$.screen?.component,screenSource:Q,relatedSources:$.relatedSources,url:$.url,goal:$.goal}),z=await Z(X),Y=K5(z);IY(Y);let G=$.suiteName??($.screen?`ui-${$.screen.name}`:`ui-${B5($.goal)??"ai"}`),J=$.network??G5,K=$.servePort??J5,q=$.url,V={name:$.contract.name,network:J,port:$.port,buildCmd:$.buildCmd,url:q,prep:LY};if($.buildDir)V.buildDir=$.buildDir;let B={name:G,description:Y.description,strategy:"browser",timeout:UY,...$.browserMode?{browserMode:$.browserMode}:{}},H=RY(Y.assertions,K);return{dappConfig:V,suite:B,actions:null,assertions:H,prompt:Y.prompt,suiteName:G}}function V5($){let Z=$.params;return typeof Z?.port==="number"?Z.port:void 0}function MY($,Z){if($.post.some((X)=>X.type==="port-listening"&&V5(X)===Z))return $;return{...$,post:[...$.post,{id:"serve-port-listening",type:"port-listening",params:{port:Z},expect:"pass"}]}}function RY($,Z){let Q=$.post;if(!Q.some((G)=>G.type==="process-exit-code"))Q=[...Q,{id:"claude-exit-ok",type:"process-exit-code",params:{code:0},expect:"pass"}];if(!Q.some((G)=>G.type==="port-listening"&&V5(G)===Z))Q=[...Q,{id:"serve-port-listening",type:"port-listening",params:{port:Z},expect:"pass"}];if(!Q.some((G)=>G.type==="agent-report-no-failure"))Q=[...Q,{id:"agent-no-failure",type:"agent-report-no-failure",params:{},expect:"pass"}];return{...$,post:Q}}function B5($){if(!$)return;let Z=$.toLowerCase().replace(/[^a-z0-9]+/g,"-").replace(/^-+|-+$/g,"").slice(0,32);return Z.length>=2?Z:void 0}function H3($){let Z=$.split("/").pop(),Q=[n4($),f2(n4($),"src"),n4(n4($)),f2(n4(n4($)),"src")];for(let X of Q){if(!V3(X))continue;if(Z){let G=f2(X,`${Z}.compact`);if(V3(G))return G}let z;try{z=HY(X)}catch{continue}let Y=z.find((G)=>G.endsWith(".compact"));if(Y)return f2(X,Y)}return}var G5="undeployed",J5=9932,WY=300,UY=600,q5,LY,X5=300000,z5=4194304,B3=($)=>{return new Promise((Z,Q)=>{let X=BY("claude",["--print"],{stdio:["pipe","pipe","pipe"]}),z="",Y="",G=0,J=!1,K=setTimeout(()=>{X.kill("SIGTERM"),Q(new Error(`claude --print timed out after ${X5/1000}s`))},X5);X.stdout?.on("data",(q)=>{if(G+=q.length,G>z5){J=!0,X.kill("SIGTERM");return}z+=q.toString()}),X.stderr?.on("data",(q)=>{Y+=q.toString()}),X.on("error",(q)=>{if(clearTimeout(K),q.code==="ENOENT"){Q(new Error("claude CLI not found on PATH. Install Claude Code (npm install -g @anthropic-ai/claude-code) or run mn test create without --goal / --screen for the deterministic scaffolder."));return}Q(q)}),X.on("exit",(q,V)=>{if(clearTimeout(K),J){Q(new Error(`claude --print response exceeded ${z5/1024/1024}MB; aborted.`));return}if(V==="SIGTERM")return;if(q!==0){let B=Y.trim().slice(0,300)||`(no stderr; exit ${q})`;Q(new Error(`claude --print exited with code ${q}: ${B}`));return}Z(z)}),X.stdin?.write($),X.stdin?.end()})};var W3=N(()=>{Z5();Y3();w2();q5=["cache-clear","localnet-up","balance:1000","dust","mn-serve"],LY=[...q5,"build-and-serve"]});var O5={};k(O5,{runActions:()=>EY,diffState:()=>SY});import{existsSync as W5,readFileSync as U5,writeFileSync as AY,mkdirSync as DY}from"node:fs";import{join as L5}from"node:path";async function EY($){let{actions:Z,config:Q,dappDir:X,suiteName:z,networkConfig:Y,servePort:G,redeploy:J,onActionStart:K,onActionComplete:q,onMessage:V}=$,B=[],H=Q.network??"undeployed",j;if(!J)j=kY(X,z,H);let{info:L}=G4(X),W=bY(L),F;for(let O of Z){K?.(O);let _=Date.now();if(F&&(O.type==="contract-call"||O.type==="contract-state")){let R={id:O.id,type:O.type,status:"skip",duration:0,message:`cascaded skip — depends on ${F}`};B.push(R),q?.(O,R);continue}try{let R=await NY(O,{dappDir:X,networkConfig:Y,managedDir:L.managedDir,contractName:L.name,servePort:G,contractAddress:j,witnessDeps:W,onMessage:V});if(R.contractAddress){if(j=R.contractAddress,wY(X,z,H,j),O.type==="contract-deploy")F=void 0}let T=Date.now()-_,v={...R,duration:T};if(B.push(v),q?.(O,v),R.status==="skip"&&O.type==="contract-call")F=`skipped action "${O.id}"`}catch(R){let T=Date.now()-_,v={id:O.id,type:O.type,status:"fail",duration:T,message:R.message};B.push(v),q?.(O,v);break}}return B}async function NY($,Z){switch($.type){case"contract-deploy":return CY($,Z);case"contract-call":return xY($,Z);case"contract-state":return vY($,Z);case"wallet-cmd":return yY($,Z);default:throw new Error(`Unknown action type: ${$.type}`)}}async function CY($,Z){if(Z.contractAddress)return{id:$.id,type:$.type,status:"pass",duration:0,contractAddress:Z.contractAddress,message:`Reusing cached contract ${Z.contractAddress.slice(0,16)}...`};let Q=await r$({dappDir:Z.dappDir,networkConfig:Z.networkConfig,managedDir:Z.managedDir,contractName:Z.contractName,servePort:Z.servePort,onMessage:Z.onMessage});return{id:$.id,type:$.type,status:"pass",duration:0,contractAddress:Q.contractAddress,message:`Deployed at ${Q.contractAddress.slice(0,16)}...`}}async function xY($,Z){if(!Z.contractAddress)throw new Error(`Action "${$.id}": no contract address. Add a contract-deploy action first.`);if(!$.circuit)throw new Error(`Action "${$.id}": missing circuit name.`);let Q=Z.witnessDeps.get($.circuit);if(Q&&Q.length>0)return{id:$.id,type:$.type,status:"skip",duration:0,message:"not CLI-testable — circuit reads private state via witness "+`${Q.join(", ")}. Cover with mn test create --strategy ui.`};let X;try{X=await A1({dappDir:Z.dappDir,networkConfig:Z.networkConfig,managedDir:Z.managedDir,contractName:Z.contractName,contractAddress:Z.contractAddress,onMessage:()=>{}})}catch{}let z=$.args?Object.values($.args):[];await s$({dappDir:Z.dappDir,networkConfig:Z.networkConfig,managedDir:Z.managedDir,contractName:Z.contractName,contractAddress:Z.contractAddress,circuit:$.circuit,args:z,servePort:Z.servePort,onMessage:Z.onMessage});let Y;try{Y=await A1({dappDir:Z.dappDir,networkConfig:Z.networkConfig,managedDir:Z.managedDir,contractName:Z.contractName,contractAddress:Z.contractAddress,onMessage:()=>{}})}catch{}return{id:$.id,type:$.type,status:"pass",duration:0,message:`${$.circuit} called`,stateBefore:X,stateAfter:Y}}async function vY($,Z){if(!Z.contractAddress)throw new Error(`Action "${$.id}": no contract address. Add a contract-deploy action first.`);let Q=await A1({dappDir:Z.dappDir,networkConfig:Z.networkConfig,managedDir:Z.managedDir,contractName:Z.contractName,contractAddress:Z.contractAddress,onMessage:Z.onMessage});if($.assert){let X=[];for(let[z,Y]of Object.entries($.assert)){let G=Q.fields[z];if(G===void 0){let J=Q.maps[z];if(J)U3(z,BigInt(J.size),Y,X);else X.push(`Field "${z}" not found in ledger state`);continue}try{let J=BigInt(G);U3(z,J,Y,X)}catch{U3(z,G,Y,X)}}if(X.length>0)throw new Error(`State assertion failed:
|
|
1174
|
+
${X.join(`
|
|
1175
|
+
`)}`)}return{id:$.id,type:$.type,status:"pass",duration:0,message:`State checked (${Object.keys(Q.fields).length} fields, ${Object.keys(Q.maps).length} maps)`,stateAfter:Q}}function U3($,Z,Q,X){for(let[z,Y]of Object.entries(Q)){let G=typeof Y==="number"?BigInt(Y):Y;if(typeof Z==="bigint"&&(typeof G==="bigint"||typeof G==="number")){let J=BigInt(G);switch(z){case">":if(!(Z>J))X.push(`${$}: expected > ${J}, got ${Z}`);break;case">=":if(!(Z>=J))X.push(`${$}: expected >= ${J}, got ${Z}`);break;case"<":if(!(Z<J))X.push(`${$}: expected < ${J}, got ${Z}`);break;case"<=":if(!(Z<=J))X.push(`${$}: expected <= ${J}, got ${Z}`);break;case"==":if(Z!==J)X.push(`${$}: expected == ${J}, got ${Z}`);break;case"!=":if(Z===J)X.push(`${$}: expected != ${J}, got ${Z}`);break;default:X.push(`${$}: unknown operator "${z}"`)}}else{let J=String(G);switch(z){case"==":if(Z!==J)X.push(`${$}: expected == "${J}", got "${Z}"`);break;case"!=":if(Z===J)X.push(`${$}: expected != "${J}", got "${Z}"`);break;default:X.push(`${$}: operator "${z}" not supported for string values`)}}}}async function yY($,Z){return{id:$.id,type:$.type,status:"pass",duration:0,message:`wallet-cmd: ${$.cmd??"no command"} (not yet implemented)`}}function bY($){if($.witnesses.length===0)return new Map;let Z=H3($.managedDir);if(!Z||!W5(Z))return new Map;let Q=U5(Z,"utf-8"),X=$.witnesses.map((z)=>z.name);return C1(Q,X).byCircuit}function SY($,Z){if(!$||!Z)return[];let Q=[],X=new Set([...Object.keys($.fields),...Object.keys(Z.fields)]);for(let Y of X){let G=$.fields[Y]??"(absent)",J=Z.fields[Y]??"(absent)";if(G!==J)Q.push({field:Y,before:G,after:J})}let z=new Set([...Object.keys($.maps),...Object.keys(Z.maps)]);for(let Y of z){let G=$.maps[Y]?.size??"0",J=Z.maps[Y]?.size??"0";if(G!==J)Q.push({field:`${Y} (entries)`,before:G,after:J})}return Q}function j5($,Z){return L5($,"tests","results",`.contract-cache-${Z}.json`)}function kY($,Z,Q){let X=j5($,Z);if(!W5(X))return;try{let z=JSON.parse(U5(X,"utf-8"));if(z.network!==Q)return;return z.address}catch{return}}function wY($,Z,Q,X){let z=j5($,Z);DY(L5($,"tests","results"),{recursive:!0});let Y={address:X,network:Q,timestamp:new Date().toISOString()};AY(z,JSON.stringify(Y,null,2)+`
|
|
1176
|
+
`)}var F5=N(()=>{D1();M2();w2();W3()});var M5={};k(M5,{default:()=>_5});import{join as P5}from"node:path";import{mkdirSync as fY}from"node:fs";function hY($){return I5.includes($)}async function _5($,Z){let Q=$.subcommand;if(!Q||!hY(Q))throw new Z0(`Usage: midnight test <${I5.join("|")}>
|
|
1177
|
+
|
|
1178
|
+
create Generate dapp.test.json + a CLI test suite from the contract
|
|
1179
|
+
run Run test suites for the current dApp
|
|
1180
|
+
list List available test suites
|
|
1181
|
+
results Show latest test results
|
|
1182
|
+
|
|
1183
|
+
Run from the root of a dApp project containing dapp.test.json.`);let X=C($,"json");switch(Q){case"create":return cY($,X);case"run":return oY($,X,Z);case"list":return nY(X);case"results":return aY($,X)}}function mY($){let Z=($??"").toLowerCase();if(Z==="")return;if(Z==="cli"||Z==="browser")return Z;if(Z==="ui")return"browser";throw new Z0(`Unknown strategy "${$}". Use "cli" or "ui".`)}function gY($){return $==="preprod"||$==="preview"||$==="undeployed"?$:void 0}function uY($){let Z={},Q=A($,"port");if(Q!==void 0){let J=parseInt(Q,10);if(!Number.isFinite(J)||J<=0||J>65535)throw new Z0(`Invalid --port "${Q}" — must be 1–65535.`);Z.port=J}let X=A($,"build-cmd");if(X!==void 0)Z.buildCmd=X;let z=A($,"build-dir");if(z!==void 0)Z.buildDir=z;let Y=A($,"url");if(Y!==void 0)Z.url=Y;let G=A($,"browser-mode");if(G!==void 0){if(G!=="dom"&&G!=="vision"&&G!=="script"&&G!=="auto")throw new Z0(`Invalid --browser-mode "${G}" — must be dom, vision, or script.`);Z.browserMode=G}return Z}function pY($){if($.port===void 0||$.buildCmd===void 0)throw new Z0("Browser strategy needs --port and --build-cmd (and optionally --build-dir, --url) when running non-interactively.");return{port:$.port,buildCmd:$.buildCmd,buildDir:$.buildDir,url:$.url}}async function cY($,Z){let{resolve:Q}=await import("node:path"),{findContractInfo:X}=await Promise.resolve().then(() => (M2(),O7)),{buildScaffold:z}=await Promise.resolve().then(() => (Y3(),fZ)),{writeScaffold:Y}=await Promise.resolve().then(() => (uZ(),gZ)),G=await Promise.resolve().then(() => (rZ(),aZ)),{writeJsonResult:J}=await Promise.resolve().then(() => O3),{isInteractive:K,promptStrategy:q,promptBrowserOptions:V,promptCircuit:B,promptScreen:H,promptGoal:j,promptSuiteName:L}=G,W=await Promise.resolve().then(() => (W3(),H5)),{discoverScreens:F}=await Promise.resolve().then(() => (q3(),dZ)),O=Q(A($,"path")??process.cwd()),_=A($,"name"),R=A($,"suite"),T=gY(A($,"network")),v=C($,"force"),b=!Z&&K(),u=A($,"goal"),p=A($,"screen"),L0=C($,"no-ai"),o=mY(A($,"strategy"))??(b?await q():"cli"),Y0;if(o==="browser"){let S=uY($);Y0=b?await V(S):pY(S)}let{info:$0}=X(O,_),E=!L0&&W.isClaudeAvailable(),I=!L0&&(u!==void 0||p!==void 0||b&&E),P=I?await lY({strategy:o,info:$0,dappDir:O,browser:Y0,network:T,suiteName:R,goalFlag:u,screenFlag:p,interactive:b,jsonMode:Z,ai:W,promptCircuit:B,promptScreen:H,promptGoal:j,promptSuiteName:L,discoverScreens:F})??z($0.circuits,{contractName:$0.name,suiteName:R,strategy:o,browser:Y0,network:T}):z($0.circuits,{contractName:$0.name,suiteName:R,strategy:o,browser:Y0,network:T});if(b&&!R){let S=await L(P.suiteName);if(S!==P.suiteName)P.suiteName=S,P.suite.name=S}let D=Y(P,{dappDir:O,force:v});if(Z){J({subcommand:"create",contractName:$0.name,suiteName:P.suiteName,strategy:o,aiAssisted:I&&P.suiteName!=="cli-default"&&P.suiteName!=="ui-default",written:D.written,excludedCircuits:P.excludedCircuits});return}process.stderr.write(`
|
|
1184
|
+
`+g(`Test scaffold: ${$0.name}`)+`
|
|
1185
|
+
|
|
1186
|
+
`);for(let S of D.written)process.stderr.write(` ${m("✓")} ${S}
|
|
1187
|
+
`);let h=o==="browser"?{file:"prompt.md",hint:"verify the UI steps match your dApp — exact button labels, expected screens, success criteria. Claude follows these literally."}:{file:"actions.json",hint:'review args — placeholder values like 0 may violate contract assertions (e.g. "amount > 0").'};if(P.excludedCircuits&&P.excludedCircuits.length>0){process.stderr.write(`
|
|
1188
|
+
`+U(" Skipped (not CLI-testable):")+`
|
|
1189
|
+
`);for(let S of P.excludedCircuits)process.stderr.write(` ${e("!")} ${S.name} ${U("— "+S.reason)}
|
|
1190
|
+
`);process.stderr.write(U(` These need the dApp UI to populate private state first.
|
|
1191
|
+
`)),process.stderr.write(U(` Cover them with a browser-strategy suite (mn test create --strategy ui).
|
|
1192
|
+
`))}process.stderr.write(`
|
|
1193
|
+
`+U(" Next:")+`
|
|
1194
|
+
`),process.stderr.write(U(" Edit ")+l(`tests/suites/${P.suiteName}/${h.file}`)+`
|
|
1195
|
+
`),process.stderr.write(U(" ")+U(h.hint)+`
|
|
1196
|
+
`),process.stderr.write(U(" Run ")+l(`mn test run --suite ${P.suiteName}`)+`
|
|
1197
|
+
`),process.stderr.write(U(" List ")+l("mn test list")+U(" (see every suite in this project)")+`
|
|
1198
|
+
|
|
1199
|
+
`)}async function lY($){try{if(!$.ai.isClaudeAvailable())return null;let Z=$.goalFlag??($.interactive?await $.promptGoal():void 0);if($.strategy==="cli")return await dY($,Z);return await iY($,Z)}catch(Z){return process.stderr.write(`
|
|
1200
|
+
${U("AI scaffold failed, falling back to deterministic:")} ${Z.message}
|
|
1201
|
+
`),null}}async function T5($,Z,Q){if(!Q)return Z();let X=i($);try{let z=await Z();return X.stop(),z}catch(z){throw X.fail("AI scaffold failed"),z}}async function dY($,Z){let Q=$.interactive?await $.promptCircuit($.info.circuits):$.info.circuits.find((z)=>!z.pure);if(!Q&&!Z)return null;let X=$.ai.findContractSourcePath($.info.managedDir);if(Q&&X&&$.interactive){let{readFileSync:z}=await import("node:fs"),{analyzeWitnessDependencies:Y}=await Promise.resolve().then(() => (w2(),Q5)),G=z(X,"utf-8"),J=$.info.witnesses.map((V)=>V.name),q=Y(G,J).byCircuit.get(Q.name);if(q&&q.length>0)process.stderr.write(`
|
|
1202
|
+
`+e("!")+` ${Q.name} calls ${q.join(", ")} — a witness that reads private state.
|
|
1203
|
+
`),process.stderr.write(U(` The CLI suite will deploy + run, but expect a "Cannot read properties of undefined" failure
|
|
1204
|
+
`)+U(` unless the dApp UI populates that state first. Consider --strategy ui instead.
|
|
1205
|
+
`))}return T5("Asking Claude to scaffold the test suite (30–60s)...",()=>$.ai.generateCliScaffoldWithAI({contract:$.info,contractSourcePath:X,targetCircuit:Q,goal:Z,network:$.network,suiteName:$.suiteName}),$.interactive&&!$.jsonMode)}async function iY($,Z){if(!$.browser)return null;let{join:Q}=await import("node:path"),X=$.browser.buildDir?Q($.dappDir,$.browser.buildDir):$.dappDir,z=$.discoverScreens(X),Y=$.screenFlag?z.find((G)=>G.name===$.screenFlag||G.component===$.screenFlag):$.interactive?await $.promptScreen(z):z[0];if(!Y&&!Z)return null;return T5("Asking Claude to scaffold the test suite (30–60s)...",()=>$.ai.generateUiScaffoldWithAI({contract:$.info,screen:Y,url:$.browser.url??`http://localhost:${$.browser.port}/`,port:$.browser.port,buildCmd:$.browser.buildCmd,buildDir:$.browser.buildDir,browserMode:$.browser.browserMode,goal:Z,network:$.network,suiteName:$.suiteName}),$.interactive&&!$.jsonMode)}async function oY($,Z,Q){let{config:X,dappDir:z}=y2(),Y=$3(z),G=A($,"suite"),J;if(G){if(J=Y.find((j)=>j.suite.name===G),!J){let j=Y.map((L)=>L.suite.name).join(", ")||"none found";throw new Z0(`Suite "${G}" not found. Available: ${j}`)}}else if(Y.length>0)J=Y[0];fY(P5(z,"tests","results"),{recursive:!0});let K=Date.now(),q=new Date().toISOString();if(!Z){if(process.stderr.write(`
|
|
1206
|
+
`+g(`E2E Test: ${X.name}`)+`
|
|
1207
|
+
|
|
1208
|
+
`),process.stderr.write(M("Network",X.network??"undeployed")+`
|
|
1209
|
+
`),X.port)process.stderr.write(M("Port",String(X.port))+`
|
|
1210
|
+
`);if(J){if(process.stderr.write(M("Suite",J.suite.name)+`
|
|
1211
|
+
`),process.stderr.write(M("Strategy",J.suite.strategy)+`
|
|
1212
|
+
`),J.suite.strategy==="browser"){let j=Z3(J.suite);process.stderr.write(M("Browser Mode",j)+`
|
|
1213
|
+
`)}}process.stderr.write(`
|
|
1214
|
+
`)}let V=HZ(),B=[],H=(j)=>{if(!Z)process.stderr.write(U(` [teardown] ${j}`)+`
|
|
1215
|
+
`)};Q?.addEventListener("abort",()=>X3(V,H),{once:!0});try{if(!Z)process.stderr.write(y(` Prep
|
|
1216
|
+
`));let j=i("Starting prep..."),L=await IZ(X,z,V,{onStepStart(f){j.update(`[prep] ${f}...`)},onStepComplete(f,o,Y0,$0){if(o==="pass")j.stop(`${m("✓")} ${f} (${rY(Y0)})`);else j.stop(`${X0("✗")} ${f}: ${$0??"failed"}`);j=i("")},onMessage(f){j.update(f)}});if(j.stop(""),B.push(...L),!Z)process.stderr.write(`
|
|
1217
|
+
`);let W=0,F="";if(J){let f=J.suite;if(f.strategy==="browser"){let o=VZ(J.suiteDir);if(!o)throw new Error(`No prompt.md found in ${J.suiteDir} — required for browser strategy`);if(F=P5(z,"tests","results",`${f.name}_${q.replace(/[:.]/g,"-")}_claude.log`),!Z)process.stderr.write(y(` Browser Test
|
|
1218
|
+
`)),process.stderr.write(U(` Claude will take over the terminal. Output is also logged.
|
|
1219
|
+
|
|
1220
|
+
`));let Y0=await MZ({suite:f,prompt:o,dappDir:z,logFile:F,onMessage:($0)=>{if(!Z)process.stderr.write(U(` ${$0}`)+`
|
|
1221
|
+
`)}});if(W=Y0.exitCode,!Z)if(process.stderr.write(`
|
|
1222
|
+
`),Y0.timedOut)process.stderr.write(` ${X0("✗")} Browser test timed out
|
|
1223
|
+
`);else process.stderr.write(` ${W===0?m("✓"):X0("✗")} Claude exited with code ${W}
|
|
1224
|
+
`)}else if(f.strategy==="cli"){let o=KZ(J.suiteDir);if(!o)throw new Error(`No actions.json found in ${J.suiteDir} — required for CLI strategy`);if(!Z)process.stderr.write(y(` CLI Test
|
|
1225
|
+
`)),process.stderr.write(U(` Executing ${o.actions.length} action(s)
|
|
1226
|
+
|
|
1227
|
+
`));let{runActions:Y0,diffState:$0}=await Promise.resolve().then(() => (F5(),O5)),{resolveNetwork:E}=await Promise.resolve().then(() => (V0(),k4)),I=X.network??"undeployed",{config:P}=E({args:{command:"test",subcommand:"run",positionals:[],flags:{network:I}}}),{DEFAULT_SERVE_PORT:D}=await Promise.resolve().then(() => l1),h=V.serveHandle?.port??D,S=C($,"redeploy"),n=await Y0({actions:o.actions,config:X,dappDir:z,suiteName:f.name,networkConfig:P,servePort:h,redeploy:S,onActionStart:(t)=>{if(!Z){let G0=i(`[${t.id}] ${t.type}${t.circuit?` ${t.circuit}`:""}...`);t._spinner=G0}},onActionComplete:(t,G0)=>{if(!Z){let x0=t._spinner;if(x0)if(G0.status==="pass")x0.stop(`${m("✓")} ${t.id}: ${G0.message??"pass"}`);else if(G0.status==="skip")x0.stop(`${e("⊘")} ${t.id}: ${G0.message??"skipped"}`);else x0.stop(`${X0("✗")} ${t.id}: ${G0.message??"fail"}`);if(G0.stateBefore&&G0.stateAfter){let v0=$0(G0.stateBefore,G0.stateAfter);if(v0.length>0)for(let R0 of v0)process.stderr.write(U(` ${R0.field}: ${R0.before} → `)+l(R0.after)+`
|
|
1228
|
+
`)}}},onMessage:(t)=>{}});if(W=n.some((t)=>t.status==="fail")?1:0,!Z){process.stderr.write(`
|
|
1229
|
+
`);let t=n.filter((R0)=>R0.status==="pass").length,G0=n.filter((R0)=>R0.status==="skip").length,x0=n.filter((R0)=>R0.status==="fail").length,v0=m(String(t)+" passed");if(G0>0)v0+=`, ${e(String(G0)+" skipped")}`;if(x0>0)v0+=`, ${X0(String(x0)+" failed")}`;process.stderr.write(` Actions: ${v0}
|
|
1230
|
+
`)}V._actionResults=n}}else if(!Z)process.stderr.write(U(` No test suites found — only prep was run
|
|
1231
|
+
`));let O={processExitCode:W,agentLogPath:F},_=[];if(J){let f=qZ(J.suiteDir);if(f&&f.post.length>0){if(!Z)process.stderr.write(`
|
|
1232
|
+
`+y(` Assertions
|
|
1233
|
+
`));if(_=await AZ(f.post,O),!Z)for(let o of _){let Y0=o.status==="pass"?m("✓"):X0("✗"),$0=o.message?U(` — ${o.message}`):"";process.stderr.write(` ${Y0} ${o.id}${$0}
|
|
1234
|
+
`)}}}let R=Math.round((Date.now()-K)/1000),T=B.every((f)=>f.status==="pass"),v=_.every((f)=>f.status==="pass"),b=T&&v&&W===0?"pass":"fail",u=V._actionResults,p={id:`${X.name}_${q}`,dapp:X.name,suite:J?.suite.name??"prep-only",timestamp:q,duration:R,network:X.network??"undeployed",strategy:J?.suite.strategy??"none",model:J?.suite.model,status:b,prep:B,actions:u?.map((f)=>({id:f.id,type:f.type,status:f.status,duration:f.duration,message:f.message,contractAddress:f.contractAddress})),assertions:_,testOutput:F?{exitCode:W,logFile:F}:void 0},L0=NZ(p,z);if(Z)x(p);else process.stderr.write(`
|
|
1235
|
+
`+d()+`
|
|
1236
|
+
`),process.stderr.write(` ${b==="pass"?m(y("PASS")):X0(y("FAIL"))} — ${X.name} (${R}s)
|
|
1237
|
+
`),process.stderr.write(U(` Results: ${L0}`)+`
|
|
1238
|
+
|
|
1239
|
+
`);if(b==="fail")process.exitCode=1}finally{await X3(V,H)}}function nY($){let{config:Z,dappDir:Q}=y2(),X=$3(Q);if($){x({dapp:Z.name,suites:X.map((z)=>({name:z.suite.name,description:z.suite.description,strategy:z.suite.strategy,timeout:z.suite.timeout}))});return}if(process.stderr.write(`
|
|
1240
|
+
`+g(`Tests: ${Z.name}`)+`
|
|
1241
|
+
|
|
1242
|
+
`),X.length===0)process.stderr.write(U(` No test suites found.
|
|
1243
|
+
`)),process.stderr.write(U(` Create tests/suites/<name>/suite.json to add one.
|
|
1244
|
+
`));else for(let{suite:z}of X)process.stderr.write(` ${y(l(z.name))}
|
|
1245
|
+
`),process.stderr.write(` ${U(z.description)}
|
|
1246
|
+
`),process.stderr.write(U(` strategy: ${z.strategy}`)+(z.timeout?U(`, timeout: ${z.timeout}s`):"")+`
|
|
1247
|
+
|
|
1248
|
+
`);process.stderr.write(`
|
|
1249
|
+
`)}function aY($,Z){let{config:Q,dappDir:X}=y2(),z=C($,"all"),Y=A($,"suite"),G=z?Q3(X,Y):(()=>{let J=CZ(X,Y);return J?[J]:[]})();if(Z){x(z?{dapp:Q.name,results:G}:G[0]??{});return}if(process.stderr.write(`
|
|
1250
|
+
`+g(`Results: ${Q.name}`)+`
|
|
1251
|
+
|
|
1252
|
+
`),G.length===0)process.stderr.write(U(` No test results found.
|
|
1253
|
+
`)),process.stderr.write(U(` Run "midnight test run" to generate results.
|
|
1254
|
+
`));else for(let J of G){let K=J.status==="pass"?m("✓"):X0("✗"),q=J.status==="pass"?m(y("PASS")):X0(y("FAIL"));process.stderr.write(` ${K} ${y(J.suite)} — ${q} (${J.duration}s)
|
|
1255
|
+
`),process.stderr.write(U(` ${J.timestamp} | ${J.network} | ${J.strategy}`)+`
|
|
1256
|
+
`);let V=J.prep.filter((H)=>H.status==="fail");if(V.length>0)process.stderr.write(X0(` prep: ${V.length} failed`)+`
|
|
1257
|
+
`);let B=J.assertions.filter((H)=>H.status==="fail");if(B.length>0)process.stderr.write(X0(` assertions: ${B.length} failed`)+`
|
|
1258
|
+
`);process.stderr.write(`
|
|
1259
|
+
`)}process.stderr.write(`
|
|
1260
|
+
`)}function rY($){if($<1000)return`${$}ms`;return`${($/1000).toFixed(1)}s`}var I5;var R5=N(()=>{Y4();H0();n0();BZ();_Z();RZ();DZ();xZ();vZ();I5=["run","list","results","create"]});import{readFileSync as A5}from"node:fs";import{fileURLToPath as N5}from"node:url";import{Server as sY}from"@modelcontextprotocol/sdk/server/index.js";import{StdioServerTransport as tY}from"@modelcontextprotocol/sdk/server/stdio.js";import{ListToolsRequestSchema as eY,CallToolRequestSchema as $G,ListResourcesRequestSchema as ZG,ReadResourceRequestSchema as QG}from"@modelcontextprotocol/sdk/types.js";async function a($,Z,Q){let X=[];y1((z)=>X.push(z));try{if(Z.flags.json=!0,!(v1 in Z.flags))Z.flags[v1]=!0;await $(Z,Q);let z=X.join("").trim();if(!z)return{};return JSON.parse(z)}finally{y1(null)}}var O0={INVALID_ARGS:"INVALID_ARGS",WALLET_NOT_FOUND:"WALLET_NOT_FOUND",NETWORK_ERROR:"NETWORK_ERROR",INSUFFICIENT_BALANCE:"INSUFFICIENT_BALANCE",TX_REJECTED:"TX_REJECTED",STALE_UTXO:"STALE_UTXO",STALE_CACHE:"STALE_CACHE",PROOF_TIMEOUT:"PROOF_TIMEOUT",PROOF_FAILURE:"PROOF_FAILURE",INVALID_DUST_PROOF:"INVALID_DUST_PROOF",DUST_REQUIRED:"DUST_REQUIRED",SYNC_TIMEOUT:"SYNC_TIMEOUT",CANCELLED:"CANCELLED",UNKNOWN:"UNKNOWN"};function F3($){let Z=($.message??"").toLowerCase();if(Z.includes("operation cancelled")||Z.includes("operation aborted")||Z==="cancelled"||Z==="aborted")return{exitCode:7,errorCode:O0.CANCELLED};if(Z.includes("wallet file not found")||Z.includes("wallet")&&Z.includes("not found"))return{exitCode:3,errorCode:O0.WALLET_NOT_FOUND};if(Z.includes("missing required flag")||Z.includes("missing amount")||Z.includes("missing recipient")||Z.includes("missing config key")||Z.includes("missing or invalid subcommand")||Z.includes("unknown command")||Z.includes("cannot specify both")||Z.includes("invalid bip-39")||Z.includes("seed must be")||Z.includes("key index must be")||Z.includes("usage:"))return{exitCode:2,errorCode:O0.INVALID_ARGS};if(Z.includes("proof")&&Z.includes("timeout"))return{exitCode:6,errorCode:O0.PROOF_TIMEOUT};if(Z.includes("stale utxo")||Z.includes("error code 115")||Z.includes("errorcode: 115"))return{exitCode:6,errorCode:O0.STALE_UTXO};if(Z.includes("error 170")||Z.includes("errorcode: 170")||Z.includes("invaliddustspendproof"))return{exitCode:6,errorCode:O0.INVALID_DUST_PROOF};if(Z.includes("stale cache")||Z.includes("chain reset")||Z.includes("chainid mismatch")||Z.includes("applied > highest"))return{exitCode:6,errorCode:O0.STALE_CACHE};if(Z.includes("failed to prove"))return{exitCode:6,errorCode:O0.PROOF_FAILURE};if(Z.includes("wallet sync timed out")||Z.includes("sync timed out")||Z.includes("timed out waiting for dust")||Z.includes("did not respond within")||Z.includes("did not produce a block")||Z.includes("did not report funds"))return{exitCode:4,errorCode:O0.SYNC_TIMEOUT};if(Z.includes("no dust")||Z.includes("dust")&&(Z.includes("required")||Z.includes("available")||Z.includes("insufficient")))return{exitCode:5,errorCode:O0.DUST_REQUIRED};if(Z.includes("insufficient")||Z.includes("not enough"))return{exitCode:5,errorCode:O0.INSUFFICIENT_BALANCE};if(Z.includes("rejected")||Z.includes("transaction failed"))return{exitCode:6,errorCode:O0.TX_REJECTED};if(Z.includes("econnrefused")||Z.includes("enotfound")||Z.includes("etimedout")||Z.includes("websocket")||Z.includes("connection refused")||Z.includes("request to ")&&Z.includes(" failed, reason:")||Z.includes("network")&&Z.includes("error"))return{exitCode:4,errorCode:O0.NETWORK_ERROR};return{exitCode:1,errorCode:O0.UNKNOWN}}function P3($){let Z=$.split(`
|
|
1261
|
+
|
|
1262
|
+
`)[0].trim(),Q=Z.split(`
|
|
1263
|
+
`),X=[];for(let z of Q){if(/(?:^|\s)(midnight|mn)\s+[a-z]/i.test(z))break;if(/^(Try|Run|See|Open):\s/i.test(z.trim()))break;X.push(z)}return X.join(`
|
|
1264
|
+
`).trim()||Z}var b1={name:"midnight-wallet-cli",version:"0.4.0",type:"module",description:"Git-style CLI wallet for the Midnight blockchain",license:"Apache-2.0",homepage:"https://github.com/nel349/midnight-wallet-cli#readme",repository:{type:"git",url:"git+https://github.com/nel349/midnight-wallet-cli.git"},bugs:{url:"https://github.com/nel349/midnight-wallet-cli/issues"},keywords:["midnight","blockchain","wallet","cli","zk","zero-knowledge","compact","mcp","ai-agent"],workspaces:["packages/*"],bin:{midnight:"dist/wallet.js",mn:"dist/wallet.js","midnight-wallet-cli":"dist/wallet.js","midnight-wallet-mcp":"dist/mcp-server.js"},files:["dist","docs/SKILL.md"],scripts:{wallet:"tsx src/wallet.ts",build:'bun build src/wallet.ts --outfile dist/wallet.js --target node --format esm --packages external --minify --banner "#!/usr/bin/env node" && bun build src/mcp-server.ts --outfile dist/mcp-server.js --target node --format esm --packages external --minify --banner "#!/usr/bin/env node"',mcp:"tsx src/mcp-server.ts",prepublishOnly:"npm run build && npm run test",test:"vitest run","test:watch":"vitest",typecheck:"tsc --noEmit"},dependencies:{"@midnight-ntwrk/dapp-connector-api":"^4.0.1","@midnight-ntwrk/ledger-v8":"^8.0.3","@midnight-ntwrk/midnight-js-contracts":"^4.0.4","@midnight-ntwrk/midnight-js-http-client-proof-provider":"^4.0.4","@midnight-ntwrk/midnight-js-indexer-public-data-provider":"^4.0.4","@midnight-ntwrk/midnight-js-level-private-state-provider":"^4.0.4","@midnight-ntwrk/midnight-js-network-id":"^4.0.4","@midnight-ntwrk/midnight-js-node-zk-config-provider":"^4.0.4","@midnight-ntwrk/midnight-js-types":"^4.0.4","@midnight-ntwrk/midnight-js-utils":"^4.0.4","@midnight-ntwrk/wallet-sdk-abstractions":"^2.0.0","@midnight-ntwrk/wallet-sdk-address-format":"^3.1.0","@midnight-ntwrk/wallet-sdk-capabilities":"^3.2.0","@midnight-ntwrk/wallet-sdk-dust-wallet":"^4.0.0","@midnight-ntwrk/wallet-sdk-facade":"^4.0.0","@midnight-ntwrk/wallet-sdk-hd":"^3.0.1","@midnight-ntwrk/wallet-sdk-shielded":"^3.0.0","@midnight-ntwrk/wallet-sdk-unshielded-wallet":"^3.0.0","@modelcontextprotocol/sdk":"^1.27.1","@scure/bip39":"^2.0.1",rxjs:"^7.8.1",ws:"^8.19.0"},devDependencies:{"@types/node":"^22.19.13","@types/ws":"^8.18.1",tsx:"^4.21.0",typescript:"^5.9.3",vitest:"^3.2.4"}};var IG=b1.name,m2=b1.version,_G=b1.description;import{randomUUID as m5}from"node:crypto";var g5=300000;function I3($={}){let Z=$.ttlMs??g5,Q=$.now??(()=>Date.now()),X=new Map,z=(Y)=>{let G=0;for(let[J,K]of X)if(K.expiresAt<=Y)X.delete(J),G+=1;return G};return{create(Y){let G=Q();z(G);let J=m5(),K={token:J,tool:Y.tool,args:Y.args,description:Y.description,createdAt:G,expiresAt:G+Z};return X.set(J,K),K},redeem(Y){let G=X.get(Y);if(!G)return null;if(G.expiresAt<=Q())return X.delete(Y),null;return X.delete(Y),G},sweep(Y=Q()){return z(Y)},size(){return X.size}}}var C5="midnight-wallet://skill/core",x5="midnight-wallet://skill/full",XG="midnight-wallet://skill",zG=N5(new URL("../docs/SKILL-CORE.md",import.meta.url)),YG=N5(new URL("../docs/SKILL.md",import.meta.url));function W0($,Z,Q){let X={json:!0},z=[];for(let[Y,G]of Object.entries(Z)){if(G===void 0||G===null)continue;let J=Y==="full"?a4:Y;if(typeof G==="boolean"){if(G)X[J]=!0}else X[J]=String(G)}return{command:$,subcommand:Q,positionals:z,flags:X}}var GG={generate:()=>Promise.resolve().then(() => (p3(),u3)),info:()=>Promise.resolve().then(() => (d3(),l3)),balance:()=>Promise.resolve().then(() => (U8(),W8)),address:()=>Promise.resolve().then(() => (O8(),j8)),"genesis-address":()=>Promise.resolve().then(() => (I8(),P8)),"inspect-cost":()=>Promise.resolve().then(() => (R8(),M8)),airdrop:()=>Promise.resolve().then(() => (N8(),E8)),transfer:()=>Promise.resolve().then(() => (b8(),y8)),dust:()=>Promise.resolve().then(() => (w8(),k8)),cache:()=>Promise.resolve().then(() => (m8(),h8)),config:()=>Promise.resolve().then(() => (p8(),u8)),localnet:()=>Promise.resolve().then(() => (X7(),Q7)),wallet:()=>Promise.resolve().then(() => (q7(),J7)),status:()=>Promise.resolve().then(() => (U7(),W7)),contract:()=>Promise.resolve().then(() => (XZ(),QZ)),test:()=>Promise.resolve().then(() => (R5(),M5))};async function r($){let Z=GG[$];if(!Z)throw new Error(`Unknown command handler: ${$}`);return(await Z()).default}var v5=[{name:"midnight_wallet_generate",description:"Create wallet.",annotations:{destructiveHint:!0},inputSchema:{type:"object",properties:{name:{type:"string"},network:{type:"string",enum:["preprod","preview","undeployed"]},seed:{type:"string",description:"64-char hex"},mnemonic:{type:"string",description:"BIP-39 24 words"},force:{type:"string"}},required:["name"]},async handler($){let Z=$.name,Q=W0("wallet",$,"generate");if(Q.positionals=[Z],delete Q.flags.name,$.force==="true"||$.force===!0)Q.flags.force=!0;let X=await r("wallet");return a(X,Q)}},{name:"midnight_wallet_list",description:"List wallets. Default: per-wallet { name, active, network, address, shieldedAddress } scoped to the active network. Pass { full: true } for the 3-network addresses + shieldedAddresses maps.",annotations:{readOnlyHint:!0,idempotentHint:!0},inputSchema:{type:"object",properties:{full:{type:"boolean",description:"Return full per-network addresses + shieldedAddresses maps (the same shape `mn wallet list --json` emits). Default false (slim)."}}},async handler({full:$}={}){let Z={json:!0};if($)Z[a4]=!0;let Q={command:"wallet",subcommand:"list",positionals:[],flags:Z},X=await r("wallet");return a(X,Q)}},{name:"midnight_wallet_use",description:"Set active wallet.",annotations:{idempotentHint:!0},inputSchema:{type:"object",properties:{name:{type:"string"}},required:["name"]},async handler($){let Q={command:"wallet",subcommand:"use",positionals:[$.name],flags:{json:!0}},X=await r("wallet");return a(X,Q)}},{name:"midnight_wallet_info",description:"Show wallet details. Default: { name, active, network, address, shieldedAddress } scoped to the active network. Pass { full: true } for the per-network maps + createdAt + file path.",annotations:{readOnlyHint:!0,idempotentHint:!0},inputSchema:{type:"object",properties:{name:{type:"string"},full:{type:"boolean",description:"Return the full per-network addresses + shieldedAddresses maps and bookkeeping fields (matches `mn wallet info <name> --json`). Default false (slim)."}}},async handler($){let{name:Z,full:Q}=$,X={json:!0};if(Q)X[a4]=!0;let z={command:"wallet",subcommand:"info",positionals:Z?[Z]:[],flags:X},Y=await r("wallet");return a(Y,z)}},{name:"midnight_wallet_remove",description:"Remove wallet.",annotations:{destructiveHint:!0},inputSchema:{type:"object",properties:{name:{type:"string"}},required:["name"]},async handler($){let Q={command:"wallet",subcommand:"remove",positionals:[$.name],flags:{json:!0}},X=await r("wallet");return a(X,Q)}},{name:"midnight_info",description:"Wallet metadata.",annotations:{readOnlyHint:!0,idempotentHint:!0},inputSchema:{type:"object",properties:{wallet:{type:"string"}}},async handler($){let Z=W0("info",$),Q=await r("info");return a(Q,Z)}},{name:"midnight_balance",description:"NIGHT balance. Default: { network, unshielded, shielded } — no echoed addresses. Pass { full: true } for the human shape including address + shieldedAddress.",annotations:{readOnlyHint:!0,openWorldHint:!0},inputSchema:{type:"object",properties:{address:{type:"string"},wallet:{type:"string"},network:{type:"string",enum:["preprod","preview","undeployed"]},"indexer-ws":{type:"string"},full:{type:"boolean",description:"Include the queried address + shieldedAddress in the response (matches `mn balance --json`). Default false (slim)."}}},async handler($){let Z=$.address,Q=W0("balance",$,Z);delete Q.flags.address;let X=await r("balance");return a(X,Q)}},{name:"midnight_address",description:"Derive address from seed.",annotations:{readOnlyHint:!0,idempotentHint:!0},inputSchema:{type:"object",properties:{seed:{type:"string",description:"64-char hex"},network:{type:"string",enum:["preprod","preview","undeployed"]},index:{type:"string"}},required:["seed"]},async handler($){let Z=W0("address",$),Q=await r("address");return a(Q,Z)}},{name:"midnight_genesis_address",description:"Genesis address.",annotations:{readOnlyHint:!0,idempotentHint:!0},inputSchema:{type:"object",properties:{network:{type:"string",enum:["preprod","preview","undeployed"]}}},async handler($){let Z=W0("genesis-address",$),Q=await r("genesis-address");return a(Q,Z)}},{name:"midnight_inspect_cost",description:"Block limits.",annotations:{readOnlyHint:!0,idempotentHint:!0},inputSchema:{type:"object",properties:{}},async handler(){let $=W0("inspect-cost",{}),Z=await r("inspect-cost");return a(Z,$)}},{name:"midnight_airdrop",description:"Fund from genesis (undeployed).",annotations:{destructiveHint:!0,openWorldHint:!0},inputSchema:{type:"object",properties:{amount:{type:"string",description:"NIGHT"},wallet:{type:"string"}},required:["amount"]},async handler($){let Z=$.amount,Q=W0("airdrop",$,Z);delete Q.flags.amount;let X=await r("airdrop");return a(X,Q)}},{name:"midnight_transfer",description:"Send NIGHT (returns pending token; see skill).",annotations:{destructiveHint:!0,openWorldHint:!0},inputSchema:{type:"object",properties:{to:{type:"string",description:"bech32m"},amount:{type:"string",description:"NIGHT"},wallet:{type:"string"}},required:["to","amount"]},async handler($){let{to:Z,amount:Q}=$,X=W0("transfer",$,Z);X.positionals=[Q],delete X.flags.to,delete X.flags.amount;let z=await r("transfer");return a(z,X)}},{name:"midnight_dust_register",description:"Register for dust.",annotations:{destructiveHint:!0,openWorldHint:!0},inputSchema:{type:"object",properties:{wallet:{type:"string"}}},async handler($){let Z=W0("dust",$,"register"),Q=await r("dust");return a(Q,Z)}},{name:"midnight_dust_status",description:"Dust status. Default: { network, registered, registeredUtxos, unregisteredUtxos, dustBalance, dustAvailable }. Pass { full: true } for sync internals (eventsApplied, ownedUtxos, cached).",annotations:{readOnlyHint:!0,openWorldHint:!0},inputSchema:{type:"object",properties:{wallet:{type:"string"},"proof-server":{type:"string"},node:{type:"string"},"indexer-ws":{type:"string"},"no-cache":{type:"string"},full:{type:"boolean",description:"Include sync internals (eventsApplied, ownedUtxos, cached) in the response (matches `mn dust status --json`). Default false (slim)."}}},async handler($){let Z=W0("dust",$,"status"),Q=await r("dust");return a(Q,Z)}},{name:"midnight_config_get",description:"Read config.",annotations:{readOnlyHint:!0,idempotentHint:!0},inputSchema:{type:"object",properties:{key:{type:"string"}},required:["key"]},async handler($){let Q={command:"config",subcommand:"get",positionals:[$.key],flags:{json:!0}},X=await r("config");return a(X,Q)}},{name:"midnight_config_set",description:"Write config.",annotations:{idempotentHint:!0},inputSchema:{type:"object",properties:{key:{type:"string"},value:{type:"string"}},required:["key","value"]},async handler($){let{key:Z,value:Q}=$,X={command:"config",subcommand:"set",positionals:[Z,Q],flags:{json:!0}},z=await r("config");return a(z,X)}},{name:"midnight_cache_clear",description:"Clear sync cache.",annotations:{destructiveHint:!0,idempotentHint:!0},inputSchema:{type:"object",properties:{network:{type:"string",enum:["preprod","preview","undeployed"]},wallet:{type:"string"}}},async handler($){let Z=W0("cache",$,"clear"),Q=await r("cache");return a(Q,Z)}},{name:"midnight_config_unset",description:"Reset config.",annotations:{idempotentHint:!0},inputSchema:{type:"object",properties:{key:{type:"string"}},required:["key"]},async handler($){let Q={command:"config",subcommand:"unset",positionals:[$.key],flags:{json:!0}},X=await r("config");return a(X,Q)}},{name:"midnight_localnet_up",description:"Start localnet (Docker).",annotations:{openWorldHint:!0},inputSchema:{type:"object",properties:{}},async handler(){let $={command:"localnet",subcommand:"up",positionals:[],flags:{json:!0}},Z=await r("localnet");return a(Z,$)}},{name:"midnight_localnet_stop",description:"Stop localnet (preserves state).",annotations:{idempotentHint:!0,openWorldHint:!0},inputSchema:{type:"object",properties:{}},async handler(){let $={command:"localnet",subcommand:"stop",positionals:[],flags:{json:!0}},Z=await r("localnet");return a(Z,$)}},{name:"midnight_localnet_down",description:"Localnet teardown (volumes + undeployed cache).",annotations:{destructiveHint:!0,openWorldHint:!0},inputSchema:{type:"object",properties:{}},async handler(){let $={command:"localnet",subcommand:"down",positionals:[],flags:{json:!0}},Z=await r("localnet");return a(Z,$)}},{name:"midnight_localnet_status",description:"Localnet status.",annotations:{readOnlyHint:!0,openWorldHint:!0},inputSchema:{type:"object",properties:{}},async handler(){let $={command:"localnet",subcommand:"status",positionals:[],flags:{json:!0}},Z=await r("localnet");return a(Z,$)}},{name:"midnight_localnet_clean",description:"Remove stray containers.",annotations:{destructiveHint:!0,openWorldHint:!0},inputSchema:{type:"object",properties:{}},async handler(){let $={command:"localnet",subcommand:"clean",positionals:[],flags:{json:!0}},Z=await r("localnet");return a(Z,$)}},{name:"midnight_contract_inspect",description:"Inspect a compiled Compact contract: name, circuits (with arg/return types), witnesses, compiler/language/runtime versions. Reads managed/<name>/compiler/contract-info.json under the dapp dir. Returns a `siblings` array listing other contracts in the same project — call again with `name` to inspect a sibling.",annotations:{readOnlyHint:!0,idempotentHint:!0},inputSchema:{type:"object",properties:{path:{type:"string",description:"Path to dApp directory (defaults to cwd)"},managed:{type:"string",description:"Direct path to a managed/<name>/ directory (overrides path)"},name:{type:"string",description:"Specific contract name to inspect when the project has multiple (see siblings field)"}}},async handler($){let Z=W0("contract",$,"inspect"),Q=await r("contract");return a(Q,Z)}},{name:"midnight_contract_state",description:"Read the current ledger state of a deployed contract. Returns the ledger fields as a JSON object.",annotations:{readOnlyHint:!0,openWorldHint:!0},inputSchema:{type:"object",properties:{address:{type:"string",description:"Contract address (hex)"},wallet:{type:"string"},network:{type:"string",enum:["preprod","preview","undeployed"]},path:{type:"string",description:"dApp directory (defaults to cwd; needed to find the compiled artifact for state decoding)"},managed:{type:"string",description:"Direct path to a managed/<name>/ directory (overrides path-based contract scan)"},name:{type:"string",description:"Specific contract name when the project has multiple"}},required:["address"]},async handler($){let Z=W0("contract",$,"state"),Q=await r("contract");return a(Q,Z)}},{name:"midnight_contract_deploy",description:"Deploy a compiled Compact contract (returns pending token; agent must show the description and redeem via midnight_confirm_operation).",annotations:{destructiveHint:!0,openWorldHint:!0},inputSchema:{type:"object",properties:{wallet:{type:"string"},network:{type:"string",enum:["preprod","preview","undeployed"]},args:{type:"string",description:"JSON-encoded array or object of constructor arguments"},path:{type:"string",description:"dApp directory (defaults to cwd)"},managed:{type:"string",description:"Direct path to a managed/<name>/ directory (overrides path-based contract scan)"},name:{type:"string",description:"Specific contract name when the project has multiple"}}},async handler($){let Z=W0("contract",$,"deploy"),Q=await r("contract");return a(Q,Z)}},{name:"midnight_contract_call",description:"Call a circuit on a deployed contract (returns pending token; agent must show the description and redeem via midnight_confirm_operation).",annotations:{destructiveHint:!0,openWorldHint:!0},inputSchema:{type:"object",properties:{address:{type:"string",description:"Contract address (hex)"},circuit:{type:"string",description:"Circuit name to invoke"},wallet:{type:"string"},network:{type:"string",enum:["preprod","preview","undeployed"]},args:{type:"string",description:"JSON-encoded array or object of circuit arguments"},path:{type:"string",description:"dApp directory (defaults to cwd)"},managed:{type:"string",description:"Direct path to a managed/<name>/ directory (overrides path-based contract scan)"},name:{type:"string",description:"Specific contract name when the project has multiple"}},required:["address","circuit"]},async handler($){let Z=W0("contract",$,"call"),Q=await r("contract");return a(Q,Z)}},{name:"midnight_test_create",description:"Generate a test scaffold for the contract. Two paths: (1) AI-assisted — pass `goal` (and optionally `screen` for UI strategy) and the scaffold is generated by Claude reading the contract + screen source, producing a focused suite with realistic args / on-screen labels. (2) Deterministic — pass `no-ai: true` (or omit goal/screen in non-interactive mode) for boilerplate scaffolds that the user reviews. CLI strategy emits dapp.test.json + tests/suites/<name>/{suite,actions,assertions}.json. Browser strategy emits prompt.md instead of actions.json plus the browser fields in dapp.test.json — required: port, build-cmd. Use force:true to overwrite.",annotations:{destructiveHint:!0},inputSchema:{type:"object",properties:{path:{type:"string",description:"dApp directory (defaults to cwd)"},name:{type:"string",description:"Specific contract name when the project has multiple"},suite:{type:"string",description:"Suite directory name under tests/suites/ (auto-derived from circuit/screen when AI scaffolds)"},strategy:{type:"string",enum:["cli","browser"],description:"cli (default): drive contracts via actions.json. browser: drive a real UI via Claude + Chrome via prompt.md."},network:{type:"string",enum:["preprod","preview","undeployed"]},goal:{type:"string",description:'AI mode: one-line success criterion this suite should verify (e.g. "round goes from 0 to 1 after increment"). Triggers AI scaffolding.'},screen:{type:"string",description:'AI mode + browser strategy: name of the screen component to focus on (e.g. "loan-request-form" or "LoanRequestForm"). Auto-discovered from src/components, src/pages, src/screens.'},"no-ai":{type:"boolean",description:"Force deterministic scaffolder even when claude CLI is available."},port:{type:"string",description:'Browser strategy only — dev server port (e.g. "4173").'},"build-cmd":{type:"string",description:'Browser strategy only — shell command that builds + serves the UI (e.g. "npm run dev").'},"build-dir":{type:"string",description:"Browser strategy only — subdir the build runs in (monorepo case)."},url:{type:"string",description:"Browser strategy only — full URL Claude opens (default http://localhost:<port>/)."},"browser-mode":{type:"string",enum:["dom","vision","script"],description:"Browser strategy only — how Claude perceives the page. dom (HTML/React UIs, fast, needs chrome-devtools-mcp), vision (canvas games, slow), script (advanced, needs hooks). Default: dom for AI scaffolds."},force:{type:"boolean",description:"Overwrite existing files instead of aborting on collision"}}},async handler($){let Z=W0("test",$,"create"),Q=await r("test");return a(Q,Z)}},{name:"midnight_localnet_logs",description:"Snapshot of recent localnet logs (last N lines per service, no streaming). Returns { tail, lines: string[] }.",annotations:{readOnlyHint:!0,openWorldHint:!0},inputSchema:{type:"object",properties:{tail:{type:"string",description:"Number of lines to retrieve (default 200)"}}},async handler($){let Z=W0("localnet",$,"logs");if(!Z.flags.tail)Z.flags.tail="200";let Q=await r("localnet");return a(Q,Z)}},{name:"midnight_confirm_operation",description:"Redeem a pending token (confirm step).",annotations:{destructiveHint:!0,openWorldHint:!0},inputSchema:{type:"object",properties:{token:{type:"string"}},required:["token"]},async handler(){throw new Error("midnight_confirm_operation reached the inner handler — this is an MCP wiring bug")}}],JG=new Set(["midnight_transfer","midnight_contract_deploy","midnight_contract_call"]),D5=I3();function qG($,Z){let Q=Z.network??"active network",X=Z.wallet??"active wallet";switch($){case"midnight_transfer":return`Send ${Z.amount} NIGHT from ${X} to ${Z.to} on ${Q}`;case"midnight_contract_deploy":{let z=Z.args?` with args ${Z.args}`:"";return`Deploy contract from ${Z.path??"current directory"} as ${X} on ${Q}${z}`}case"midnight_contract_call":{let z=Z.args?` with args ${Z.args}`:"";return`Call circuit ${Z.circuit} on contract ${Z.address} as ${X} on ${Q}${z}`}default:return`Execute ${$} with ${JSON.stringify(Z)}`}}var x1=new sY({name:"midnight-wallet-cli",version:m2},{capabilities:{tools:{},resources:{}}});x1.setRequestHandler(ZG,async()=>({resources:[{uri:C5,name:"midnight-wallet skill (core)",description:"Read this first. Intent routing table + non-negotiable safety rules. ~830 tokens. Fetch /full on demand for canonical flows, error recovery, and concept primers.",mimeType:"text/markdown"},{uri:x5,name:"midnight-wallet skill (full)",description:"Canonical multi-step flows, error-recovery recipes, concept primers (NIGHT/DUST/shielded), network selection. Fetch on demand when you hit an error, start a multi-step flow, or need to explain a concept.",mimeType:"text/markdown"}]}));x1.setRequestHandler(QG,async($)=>{let Z=$.params.uri;if(Z===C5){let Q=A5(zG,"utf-8");return{contents:[{uri:Z,mimeType:"text/markdown",text:Q}]}}if(Z===x5||Z===XG){let Q=A5(YG,"utf-8");return{contents:[{uri:Z,mimeType:"text/markdown",text:Q}]}}throw new Error(`Unknown resource: ${Z}`)});x1.setRequestHandler(eY,async()=>{return{tools:v5.map(($)=>({name:$.name,description:$.description,inputSchema:$.inputSchema,...$.annotations?{annotations:$.annotations}:{}}))}});x1.setRequestHandler($G,async($)=>{let{name:Z,arguments:Q}=$.params,X=Q??{};if(Z==="midnight_confirm_operation"){let z=typeof X.token==="string"?X.token:"",Y=D5.redeem(z);if(!Y)return L3(new Error("Unknown or expired confirmation token. The first-step tool call may need to be re-issued."));return E5(Y.tool,Y.args)}if(JG.has(Z)){let z=D5.create({tool:Z,args:X,description:qG(Z,X)});return{content:[{type:"text",text:JSON.stringify(j3({pending:!0,token:z.token,description:z.description,tool:z.tool,expiresAt:new Date(z.expiresAt).toISOString(),nextStep:"Show the description to the user, get explicit consent, then call midnight_confirm_operation with this token."}),null,2)}]}}return E5(Z,X)});function j3($){return{...$,_serverVersion:m2}}async function E5($,Z){let Q=v5.find((X)=>X.name===$);if(!Q)return L3(new Error(`Unknown tool: ${$}`));try{let X=await Q.handler(Z);return{content:[{type:"text",text:JSON.stringify(j3(X),null,2)}]}}catch(X){return L3(X instanceof Error?X:new Error(String(X)))}}function L3($){let{errorCode:Z}=F3($);return{content:[{type:"text",text:JSON.stringify(j3({error:!0,code:Z,message:P3($.message)}))}],isError:!0}}async function KG(){let $=new tY;await x1.connect($)}KG().catch(($)=>{process.stderr.write(`MCP server error: ${$.message}
|
|
372
1265
|
`),process.exit(1)});
|