midnight-wallet-cli 0.2.4 → 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 +37 -10
package/dist/wallet.js
CHANGED
|
@@ -1,228 +1,318 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
|
-
var
|
|
3
|
-
|
|
4
|
-
`)}function
|
|
5
|
-
`).
|
|
6
|
-
`)}
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
`))
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
Valid
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
2
|
+
import{createRequire as O9}from"node:module";var H9=Object.create;var{getPrototypeOf:W9,defineProperty:H2,getOwnPropertyNames:U9}=Object;var L9=Object.prototype.hasOwnProperty;var b4=($,Z,Q)=>{Q=$!=null?H9(W9($)):{};let X=Z||!$||!$.__esModule?H2(Q,"default",{value:$,enumerable:!0}):Q;for(let z of U9($))if(!L9.call(X,z))H2(X,z,{get:()=>$[z],enumerable:!0});return X};var m=($,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 X4=O9(import.meta.url);function N7($){let Z=$??process.argv.slice(2),Q=[],X={},z=0;while(z<Z.length){let Y=Z[z];if(Y.startsWith("--")){let J=Y.slice(2),G=Z[z+1];if(G!==void 0&&!G.startsWith("-"))X[J]=G,z+=2;else X[J]=!0,z+=1}else if(Y.startsWith("-")&&Y.length===2){let J=Y.slice(1),G=Z[z+1];if(G!==void 0&&!G.startsWith("-"))X[J]=G,z+=2;else X[J]=!0,z+=1}else Q.push(Y),z+=1}return{command:Q[0],subcommand:Q[1],positionals:Q.slice(2),flags:X}}function D($,Z){let Q=$.flags[Z];if(Q===void 0||Q===!0)return;return Q}function x($,Z){return Z in $.flags}function D0($){return x($,"verbose")}function z4($){return x($,l1)&&!x($,q1)}function Y4($){if(x($,"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 S4($,Z,Q){let X=D($,Z);if(X===void 0)throw new Error(`Missing required flag: --${Z} <${Q}>`);return X}var l1="_minimal",q1="_full";var Z$={};m(Z$,{isValidWalletName:()=>q4,WALLETS_DIR_NAME:()=>F4,TX_TTL_MINUTES:()=>w4,TOKEN_MULTIPLIER:()=>o1,TOKEN_DECIMALS:()=>n1,SYNC_TIMEOUT_MS:()=>L2,SYNC_ATTEMPT_TIMEOUT_MS:()=>a1,SYNC_ATTEMPT_REMOTE_TIMEOUT_MS:()=>j9,STALE_UTXO_ERROR_CODE:()=>F2,RETRY_BASE_DELAY_MS:()=>P9,PROOF_TIMEOUT_MS:()=>K1,PRE_SEND_SYNC_TIMEOUT_MS:()=>O2,NATIVE_TOKEN_TYPE:()=>i1,MIN_DUST_FOR_TRANSFER:()=>k4,MIDNIGHT_DIR:()=>B0,MAX_RETRY_ATTEMPTS:()=>j2,LOCALNET_DIR_NAME:()=>_2,GENESIS_SEED:()=>J4,FILE_MODE:()=>E0,DUST_TIMEOUT_MS:()=>r1,DUST_RETRY_DELAY_MS:()=>$$,DUST_RETRY_ATTEMPTS:()=>h4,DUST_REGISTRATION_TIMEOUT_MS:()=>P2,DUST_REGISTRATION_RETRY_DELAY_MS:()=>t1,DUST_FEE_BLOCKS_MARGIN:()=>U2,DUST_COST_OVERHEAD:()=>W2,DIR_MODE:()=>N0,DEFAULT_WALLET_NAME:()=>f4,DEFAULT_WALLET_FILENAME:()=>V1,DEFAULT_SERVE_PORT:()=>m4,DEFAULT_CONFIG_FILENAME:()=>I2,DASHBOARD_BASE_URL:()=>B1,CACHE_VERSION:()=>e1,CACHE_DIR_NAME:()=>G4,BALANCE_OVERSPEND_ERROR_CODE:()=>F9,BALANCE_CHECK_TIMEOUT_MS:()=>s1,ABANDONED_TX_TIMEOUT_MS:()=>T2});function q4($){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 J4="0000000000000000000000000000000000000000000000000000000000000001",i1="0000000000000000000000000000000000000000000000000000000000000000",n1=6,o1=1e6,W2=300000000000000n,U2=5,k4=800000000000000n,L2=300000,a1=30000,j9=120000,O2=1e4,r1=120000,K1=300000,s1=60000,w4=10,j2=3,P9=1000,P2=600000,t1=15000,F2=115,F9=138,B0=".midnight",V1="wallet.json",I2="config.json",N0=448,E0=384,_2="localnet",e1=1,G4="cache",F4="wallets",f4="default",h4=10,$$=3000,T2=120000,m4=9932,B1="https://midnight-comp-tracker.vercel.app";function h0(){return!("NO_COLOR"in process.env)}function g4($,Z){if(!h0())return $;return`\x1B[38;5;${Z}m${$}\x1B[0m`}function M($){if(!h0())return $;return`\x1B[1m${$}\x1B[0m`}function V($){if(!h0())return $;return`\x1B[2m${$}\x1B[0m`}function f($){return g4($,38)}function n($){return g4($,196)}function p($){return g4($,40)}function S($){return g4($,226)}function m0($){return g4($,15)}function t0($){return g4($,245)}function g($,Z=x7){let Q=` ${$} `,X=Z-Q.length;if(X<=0)return M(Q);let z=Math.floor(X/2),Y=X-z;return M("═".repeat(z)+Q+"═".repeat(Y))}function o($=x7){return V("─".repeat($))}function A($,Z,Q=16){let X=($+":").padEnd(Q);return` ${t0(X)}${Z}`}function I4($){let Z=$<0n,Q=Z?-$:$,X=BigInt(10**n1),z=Q/X,J=(Q%X).toString().padStart(n1,"0");return`${Z?"-":""}${z}.${J}`}function Q$($){return`${I4($)} NIGHT`}function W1($){let Z=$<0n,Q=Z?-$:$,X=10n**BigInt(E7),z=Q/X,G=(Q%X).toString().padStart(E7,"0").replace(/0+$/,"").padEnd(6,"0");return`${Z?"-":""}${z}.${G}`}function X$($){return`${W1($)} DUST`}function $0($,Z=!1){let Q=Z&&$.length>20?$.slice(0,10)+"…"+$.slice(-8):$;return f(Q)}function I9($,Z){if(H1($).length<=Z)return[$];let X=$.split(/(\s+)/),z=[],Y="",J=0;for(let G of X){let q=H1(G).length;if(J+q>Z&&J>0)z.push(Y),Y=G.trimStart(),J=H1(Y).length;else Y+=G,J+=q}if(Y.length>0)z.push(Y);return z}function z$($,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 H of $){let U=H.split(`
|
|
5
|
+
`);for(let j of U)Y.push(...I9(j,z))}let J=Math.max(...Y.map((H)=>H1(H).length)),G=Math.max(J+2,20),q=X.tl+X.h.repeat(G)+X.tr,K=X.bl+X.h.repeat(G)+X.br,B=Y.map((H)=>{let U=H1(H).length,j=G-U-2;return`${X.v} ${H}${" ".repeat(Math.max(0,j))} ${X.v}`});return[q,...B,K].join(`
|
|
6
|
+
`)}function C7($,Z){let X=$.split(`
|
|
7
|
+
`).map((Y,J)=>J===0?n(M("Error: "))+n(Y):n(Y));if(Z)X.push(""),X.push(V("Suggestion: ")+Z);let z=z$(X,"heavy");if(h0())return z.replace(/[╔╗╚╝═║]/g,(Y)=>n(Y));return z}function v7($,Z){let z=$.replace(/^Usage:\s*/i,"").split(`
|
|
8
|
+
`).map((J,G)=>G===0?S(M("Usage: "))+S(J):S(J));if(Z)z.push(""),z.push(V("Hint: ")+Z);let Y=z$(z,"light");if(h0())return Y.replace(/[┌┐└┘─│]/g,(J)=>S(J));return Y}function K4($,Z){let X=[`${p("✓")} ${$}`];if(Z)X.push(A("Transaction",f(Z)));return X.join(`
|
|
9
|
+
`)}var x7=60,E7=15,H1=($)=>$.replace(/\x1b\[[0-9;]*m/g,"");var K0=()=>{};function U1($){let Z=($.message??"").toLowerCase();if(Z.includes("operation cancelled")||Z.includes("operation aborted")||Z==="cancelled"||Z==="aborted")return{exitCode:7,errorCode:I0.CANCELLED};if(Z.includes("wallet file not found")||Z.includes("wallet")&&Z.includes("not found"))return{exitCode:3,errorCode:I0.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:I0.INVALID_ARGS};if(Z.includes("proof")&&Z.includes("timeout"))return{exitCode:6,errorCode:I0.PROOF_TIMEOUT};if(Z.includes("stale utxo")||Z.includes("error code 115")||Z.includes("errorcode: 115"))return{exitCode:6,errorCode:I0.STALE_UTXO};if(Z.includes("error 170")||Z.includes("errorcode: 170")||Z.includes("invaliddustspendproof"))return{exitCode:6,errorCode:I0.INVALID_DUST_PROOF};if(Z.includes("stale cache")||Z.includes("chain reset")||Z.includes("chainid mismatch")||Z.includes("applied > highest"))return{exitCode:6,errorCode:I0.STALE_CACHE};if(Z.includes("failed to prove"))return{exitCode:6,errorCode:I0.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:I0.SYNC_TIMEOUT};if(Z.includes("no dust")||Z.includes("dust")&&(Z.includes("required")||Z.includes("available")||Z.includes("insufficient")))return{exitCode:5,errorCode:I0.DUST_REQUIRED};if(Z.includes("insufficient")||Z.includes("not enough"))return{exitCode:5,errorCode:I0.INSUFFICIENT_BALANCE};if(Z.includes("rejected")||Z.includes("transaction failed"))return{exitCode:6,errorCode:I0.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:I0.NETWORK_ERROR};return{exitCode:1,errorCode:I0.UNKNOWN}}function y7($){let Z=$.match(/^request to (\S+) failed, reason:\s*$/i);if(Z){let Q=Z[1];return`request to ${Q} failed: unable to reach ${Q} (connection refused, DNS, or TLS error). Is the indexer running?`}return $}var M2=2,I0;var R2=N(()=>{I0={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 b7($){return $ instanceof t}var t;var l0=N(()=>{t=class t extends Error{constructor($){super($);this.name="UsageError",Object.setPrototypeOf(this,t.prototype)}}});var A2={};m(A2,{writeJsonResult:()=>y,writeJsonError:()=>J$,setCaptureTarget:()=>Y$});function Y$($){L1=$}function y($){let Z=JSON.stringify($)+`
|
|
10
|
+
`;if(L1)L1(Z);else process.stdout.write(Z)}function J$($,Z,Q){let X=JSON.stringify({error:!0,code:Z,message:$.message,exitCode:Q})+`
|
|
11
|
+
`;if(L1)L1(X);else process.stdout.write(X)}var L1=null;var G$;var S7=N(()=>{G$={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 O1,g0,k7;var j1=N(()=>{S7();O1=G$.name,g0=G$.version,k7=G$.description});var h7={};m(h7,{resolveNetworkConfig:()=>D2,isValidNetworkName:()=>_0,getValidNetworkNames:()=>u4,getNetworkConfig:()=>q$,detectTestcontainerPorts:()=>f7,detectNetworkFromAddress:()=>R9,applyEndpointOverrides:()=>x0});import{execSync as T9}from"child_process";function _0($){return w7.includes($)}function q$($){return{...M9[$]}}function u4(){return w7}function R9($){if($.startsWith("mn_addr_preprod1"))return"preprod";if($.startsWith("mn_addr_preview1"))return"preview";if($.startsWith("mn_addr_undeployed1"))return"undeployed";return null}function f7(){try{let $=T9('docker ps --format "{{.Image}}|{{.Ports}}"',{encoding:"utf-8",timeout:5000}),Z={};for(let Q of $.trim().split(`
|
|
12
|
+
`)){if(!Q)continue;let[X,z]=Q.split("|"),Y=(J)=>{let G=new RegExp(`0\\.0\\.0\\.0:(\\d+)->${J}/tcp`),q=z?.match(G);return q?parseInt(q[1],10):void 0};if(X.includes("indexer-standalone")||X.includes("indexer")){let J=Y(8088);if(J)Z.indexerPort=J}if(X.includes("midnight-node")){let J=Y(9944);if(J)Z.nodePort=J}if(X.includes("proof-server")){let J=Y(6300);if(J)Z.proofServerPort=J}}return Z}catch{return{}}}function D2($){let Z=q$($);if($==="undeployed"){let Q=f7();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 x0($,Z,Q){let X=v0(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 M9,w7;var C0=N(()=>{p4();M9={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"}},w7=["preprod","preview","undeployed"]});var u7={};m(u7,{unsetConfigValue:()=>y2,setConfigValue:()=>v2,saveCliConfig:()=>c4,loadCliConfig:()=>v0,getValidConfigKeys:()=>V$,getConfigValue:()=>C2});import*as e0 from"fs";import*as N2 from"path";import{homedir as A9}from"os";function E2($){return D9[$]??$}function N9($){return/^(https?|wss?):\/\/\S+/.test($)}function m7($){return $??N2.join(A9(),B0)}function g7($){return N2.join(m7($),I2)}function E9($){let Z=m7($);if(!e0.existsSync(Z))e0.mkdirSync(Z,{recursive:!0,mode:N0})}function v0($){let Z=g7($);if(!e0.existsSync(Z))return{...P1};let Q;try{Q=e0.readFileSync(Z,"utf-8")}catch{return{...P1}}let X;try{X=JSON.parse(Q)}catch{return{...P1}}let z={network:X.network&&_0(X.network)?X.network:P1.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 c4($,Z){E9(Z);let Q=g7(Z);e0.writeFileSync(Q,JSON.stringify($,null,2)+`
|
|
13
|
+
`,{mode:E0})}function C2($,Z){let Q=E2($),X=v0(Z);if(Q==="network")return X.network;if(Q==="wallet")return X.wallet??"(not set)";if(x2.has(Q)){let z=X[Q];return typeof z==="string"?z:"(not set)"}throw new Error(`Unknown config key: "${$}"
|
|
14
|
+
Valid keys: ${K$.join(", ")}`)}function v2($,Z,Q){let X=E2($),z=v0(Q);if(X==="network"){if(!_0(Z))throw new Error(`Invalid network: "${Z}"
|
|
15
|
+
Valid networks: preprod, preview, undeployed`);z.network=Z}else if(X==="wallet"){if(!q4(Z))throw new Error(`Invalid wallet name: "${Z}"
|
|
16
|
+
Wallet name must be a simple name (no path separators, .json suffix, or special characters).`);z.wallet=Z}else if(x2.has(X)){if(!N9(Z))throw new Error(`Invalid URL for "${$}": "${Z}"
|
|
17
|
+
Must start with http://, https://, ws://, or wss://`);z[X]=Z}else throw new Error(`Unknown config key: "${$}"
|
|
18
|
+
Valid keys: ${K$.join(", ")}`);c4(z,Q)}function y2($,Z){let Q=E2($),X=v0(Z);if(Q==="network")X.network=P1.network;else if(Q==="wallet")delete X.wallet;else if(x2.has(Q))delete X[Q];else throw new Error(`Unknown config key: "${$}"
|
|
19
|
+
Valid keys: ${K$.join(", ")}`);c4(X,Z)}function V$(){return K$}var P1,K$,D9,x2;var p4=N(()=>{C0();P1={network:"undeployed"},K$=["network","proof-server","node","indexer-ws","wallet"],D9={"network-id":"network"};x2=new Set(["proof-server","node","indexer-ws"])});import{HDWallet as x9,Roles as b2}from"@midnight-ntwrk/wallet-sdk-hd";function S2($,Z){let Q=x9.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 B$($){return S2($,b2.Zswap)}function p7($){return S2($,b2.NightExternal)}function _4($){return S2($,b2.Dust)}var F1=()=>{};var l7={};m(l7,{deriveUnshieldedAddress:()=>y0,deriveShieldedAddress:()=>H$,deriveAllShieldedAddresses:()=>d4,deriveAllAddresses:()=>V4});import{HDWallet as C9,Roles as v9}from"@midnight-ntwrk/wallet-sdk-hd";import{createKeystore as y9,PublicKey as b9}from"@midnight-ntwrk/wallet-sdk-unshielded-wallet";import{MidnightBech32m as S9,ShieldedAddress as k9,ShieldedCoinPublicKey as w9,ShieldedEncryptionPublicKey as f9}from"@midnight-ntwrk/wallet-sdk-address-format";import{NetworkId as k2}from"@midnight-ntwrk/wallet-sdk-abstractions";import*as c7 from"@midnight-ntwrk/ledger-v8";function y0($,Z,Q=0){let X=d7[Z],z=C9.fromSeed($);if(z.type!=="seedOk")throw new Error("Invalid seed for HD wallet");let Y=z.hdWallet.selectAccount(0).selectRole(v9.NightExternal).deriveKeyAt(Q);if(Y.type==="keyOutOfBounds")throw new Error(`Key index ${Q} out of bounds`);let J=y9(Y.key,X);return b9.fromKeyStore(J).address}function V4($,Z=0){let Q={};for(let X of u4())Q[X]=y0($,X,Z);return Q}function d4($){let Z=H$($),Q={};for(let X of u4()){let z=d7[X];Q[X]=S9.encode(z,Z).asString()}return Q}function H$($){let Z=B$($),Q=c7.ZswapSecretKeys.fromSeed(Z);return new k9(new w9(Buffer.from(Q.coinPublicKey,"hex")),new f9(Buffer.from(Q.encryptionPublicKey,"hex")))}var d7;var i0=N(()=>{C0();F1();d7={preprod:k2.NetworkId.PreProd,preview:k2.NetworkId.Preview,undeployed:k2.NetworkId.Undeployed}});var o7={};m(o7,{setActiveWallet:()=>l4,saveWalletConfig:()=>I1,saveShieldedAddress:()=>M4,resolveWalletPath:()=>z0,removeWallet:()=>m2,migrateOldWallet:()=>g2,loadWalletConfig:()=>Y0,listWallets:()=>h2,getAddress:()=>m9,getActiveWalletName:()=>$4});import*as r from"fs";import*as W0 from"path";import{homedir as h9}from"os";function M4($,Z,Q){let X=W0.resolve($);if(!r.existsSync(X))return;try{let z=JSON.parse(r.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;r.writeFileSync(X,JSON.stringify(z,null,2)+`
|
|
20
|
+
`,{mode:E0})}catch{}}function m9($,Z){return $.addresses[Z]}function w2(){return W0.join(h9(),B0)}function T4(){return W0.join(w2(),F4)}function f2(){return W0.join(w2(),V1)}function n7(){let $=w2();if(!r.existsSync($))r.mkdirSync($,{recursive:!0,mode:N0})}function g9(){n7();let $=T4();if(!r.existsSync($))r.mkdirSync($,{recursive:!0,mode:N0})}function u9($){return $.includes("/")||$.includes("\\")||$.endsWith(".json")}function z0($){if($!==void 0){if(u9($))return W0.resolve($);return W0.join(T4(),`${$}.json`)}let Q=v0().wallet??f4;return W0.join(T4(),`${Q}.json`)}function $4(){return v0().wallet??f4}function l4($){if(!q4($))throw new Error(`Invalid wallet name: "${$}"
|
|
21
|
+
Wallet name must be a simple name (no path separators, .json suffix, or special characters).`);let Z=W0.join(T4(),`${$}.json`);if(!r.existsSync(Z))throw new Error(`Wallet "${$}" not found.
|
|
22
|
+
Run "midnight wallet list" to see available wallets.`);let Q=v0();Q.wallet=$,c4(Q)}function h2(){let $=T4();if(!r.existsSync($))return[];let Z=$4();return r.readdirSync($).filter((X)=>X.endsWith(".json")).sort().map((X)=>{let z=X.replace(/\.json$/,""),Y=W0.join($,X);try{let J=JSON.parse(r.readFileSync(Y,"utf-8")),G;if(J.addresses)G=J.addresses;else if(J.address&&J.seed)try{let K=Buffer.from(J.seed,"hex");G=V4(K)}catch{G={undeployed:J.address,preprod:"(unknown)",preview:"(unknown)"}}else G={undeployed:"(unknown)",preprod:"(unknown)",preview:"(unknown)"};let q;if(J.shieldedAddresses&&typeof J.shieldedAddresses==="object")q=J.shieldedAddresses;else if(J.seed)try{q=d4(Buffer.from(J.seed,"hex"))}catch{}return{name:z,addresses:G,shieldedAddresses:q,isActive:z===Z,createdAt:typeof J.createdAt==="string"?J.createdAt:void 0,file:Y}}catch{return{name:z,addresses:{undeployed:"(invalid)",preprod:"(invalid)",preview:"(invalid)"},isActive:z===Z,file:Y}}})}function m2($){if(!q4($))throw new Error(`Invalid wallet name: "${$}"
|
|
23
|
+
Wallet name must be a simple name (no path separators, .json suffix, or special characters).`);let Z=T4(),Q=W0.join(Z,`${$}.json`);if(!r.existsSync(Q))throw new Error(`Wallet "${$}" not found.
|
|
24
|
+
Run "midnight wallet list" to see available wallets.`);let X=$4();if($===X)throw new Error(`Cannot remove the active wallet "${$}".
|
|
25
|
+
Switch to another wallet first: midnight wallet use <other-wallet>`);if(r.readdirSync(Z).filter((Y)=>Y.endsWith(".json")).length<=1)throw new Error(`Cannot remove "${$}" — it is the only wallet.
|
|
26
|
+
`+"Create another wallet first: midnight wallet generate <name>");r.unlinkSync(Q)}function g2(){let $=f2();if(!r.existsSync($))return;let Z=T4();if(r.existsSync(Z)){if(r.readdirSync(Z).filter((Y)=>Y.endsWith(".json")).length>0)return}g9();let Q=W0.join(Z,`${f4}.json`);r.copyFileSync($,Q),r.chmodSync(Q,E0),r.unlinkSync($);let X=v0();X.wallet=f4,c4(X)}function Y0($){let Z=$?W0.resolve($):f2();if(!r.existsSync(Z))throw new Error(`Wallet file not found: ${Z}
|
|
27
|
+
Generate a wallet first: midnight wallet generate <name> --network <name>`);let Q;try{Q=r.readFileSync(Z,"utf-8")}catch(q){throw new Error(`Failed to read wallet file: ${Z}
|
|
28
|
+
${q.message}`)}let X;try{X=JSON.parse(Q)}catch{throw new Error(`Invalid JSON in wallet file: ${Z}`)}if(!X.seed||!X.createdAt){let q=["seed","createdAt"];if(!X.addresses)q.push("address");let K=q.filter((B)=>!X[B]);if(K.length>0)throw new Error(`Wallet file is missing required fields (${K.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 q=Buffer.from(X.seed,"hex"),K=V4(q),B={seed:X.seed,addresses:K,createdAt:X.createdAt};if(X.mnemonic)B.mnemonic=X.mnemonic;let H=i7(X,q);if(H)B.shieldedAddresses=H;let U={...X,addresses:K};if(H)U.shieldedAddresses=H,delete U.shieldedAddress;return r.writeFileSync(Z,JSON.stringify(U,null,2)+`
|
|
29
|
+
`,{mode:E0}),B}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"),J=i7(X,Y);if(J)z.shieldedAddresses=J;if(J&&(!X.shieldedAddresses||typeof X.shieldedAddresses!=="object"||Object.keys(X.shieldedAddresses).length<Object.keys(J).length))try{let q={...X,shieldedAddresses:J};delete q.shieldedAddress,r.writeFileSync(Z,JSON.stringify(q,null,2)+`
|
|
30
|
+
`,{mode:E0})}catch{}return z}function i7($,Z){let Q;try{Q=d4(Z)}catch{}let X=$.shieldedAddresses,z=X&&typeof X==="object"?X:null,Y=z?Object.fromEntries(Object.entries(z).filter(([,J])=>typeof J==="string"&&J.length>0)):{};if(!Q)return Object.keys(Y).length>0?Y:void 0;return{...Q,...Y}}function I1($,Z){let Q=Z?W0.resolve(Z):f2();if(!Z)n7();else{let X=W0.dirname(Q);if(!r.existsSync(X))r.mkdirSync(X,{recursive:!0,mode:N0})}return r.writeFileSync(Q,JSON.stringify($,null,2)+`
|
|
31
|
+
`,{mode:E0}),Q}var F0=N(()=>{p4();i0()});async function a($,Z,Q){let X=[];Y$((z)=>X.push(z));try{if(Z.flags.json=!0,!(l1 in Z.flags))Z.flags[l1]=!0;await $(Z,Q);let z=X.join("").trim();if(!z)return{};return JSON.parse(z)}finally{Y$(null)}}var W$=()=>{};function a7($){let Z=$.split(`
|
|
32
|
+
|
|
33
|
+
`)[0].trim(),Q=Z.split(`
|
|
34
|
+
`),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(`
|
|
35
|
+
`).trim()||Z}import{randomUUID as p9}from"node:crypto";function r7($={}){let Z=$.ttlMs??c9,Q=$.now??(()=>Date.now()),X=new Map,z=(Y)=>{let J=0;for(let[G,q]of X)if(q.expiresAt<=Y)X.delete(G),J+=1;return J};return{create(Y){let J=Q();z(J);let G=p9(),q={token:G,tool:Y.tool,args:Y.args,description:Y.description,createdAt:J,expiresAt:J+Z};return X.set(G,q),q},redeem(Y){let J=X.get(Y);if(!J)return null;if(J.expiresAt<=Q())return X.delete(Y),null;return X.delete(Y),J},sweep(Y=Q()){return z(Y)},size(){return X.size}}}var c9=300000;var s7=()=>{};var i4={};m(i4,{resolveNetworkName:()=>T0,resolveNetwork:()=>V0});function T0($){let Z=D($.args,"network");if(Z!==void 0){if(!_0(Z))throw new Error(`Invalid network: "${Z}"
|
|
36
|
+
Valid networks: ${u4().join(", ")}`);return Z}let Q=v0($.configDir);if(Q.network&&_0(Q.network))return Q.network;return"undeployed"}function V0($){let Z=T0($),Q=D2(Z);return{name:Z,config:Q}}var H0=N(()=>{C0();p4()});var u2={};m(u2,{default:()=>ZZ});import*as $Z from"fs";import*as U$ from"path";import{homedir as d9}from"os";import{generateMnemonic as l9,mnemonicToSeedSync as t7,validateMnemonic as i9}from"@scure/bip39";import{wordlist as e7}from"@scure/bip39/wordlists/english.js";async function ZZ($){let Z=T0({args:$}),Q=D($,"output"),X=D($,"seed"),z=D($,"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?U$.resolve(Q):U$.join(d9(),B0,V1);if($Z.existsSync(Y)&&!x($,"force"))throw new Error(`Wallet file already exists: ${Y}
|
|
37
|
+
Use --force to overwrite, or --output <file> to save to a different path.`);let J,G;if(X!==void 0){let U=X.replace(/^0x/,"");if(U.length!==64||!/^[0-9a-fA-F]+$/.test(U))throw new Error("Seed must be a 64-character hex string (32 bytes)");J=Buffer.from(U,"hex")}else if(z!==void 0){if(!i9(z,e7))throw new Error("Invalid BIP-39 mnemonic. Expected 12 or 24 words from the English wordlist.");G=z,J=Buffer.from(t7(G))}else G=l9(e7,256),J=Buffer.from(t7(G));let q=V4(J),K=q[Z],B={seed:J.toString("hex"),addresses:q,createdAt:new Date().toISOString()};if(G)B.mnemonic=G;let H=I1(B,Q);if(x($,"json")){let U={addresses:q,activeAddress:K,activeNetwork:Z,seed:J.toString("hex"),file:H,createdAt:B.createdAt};if(G)U.mnemonic=G;y(U);return}if(process.stdout.write(K+`
|
|
29
38
|
`),process.stderr.write(`
|
|
30
|
-
`+
|
|
39
|
+
`+g("Wallet Generated")+`
|
|
31
40
|
|
|
32
|
-
`),process.stderr.write(
|
|
33
|
-
`),process.stderr.write(
|
|
34
|
-
`),process.stderr.write(
|
|
41
|
+
`),process.stderr.write(A("Network",Z)+`
|
|
42
|
+
`),process.stderr.write(A("Address",$0(K))+`
|
|
43
|
+
`),process.stderr.write(A("File",H)+`
|
|
35
44
|
`),process.stderr.write(`
|
|
36
|
-
`),G)process.stderr.write(
|
|
45
|
+
`),G)process.stderr.write(S(M(" MNEMONIC (save securely!):"))+`
|
|
37
46
|
`),process.stderr.write(` ${G}
|
|
38
47
|
|
|
39
|
-
`);process.stderr.write(
|
|
48
|
+
`);process.stderr.write(S(M(" SEED (hex):"))+`
|
|
40
49
|
`),process.stderr.write(` ${J.toString("hex")}
|
|
41
50
|
|
|
42
|
-
`),process.stderr.write(
|
|
43
|
-
`),process.stderr.write(
|
|
51
|
+
`),process.stderr.write(o()+`
|
|
52
|
+
`),process.stderr.write(V(" Next: midnight info | midnight balance")+`
|
|
44
53
|
|
|
45
|
-
`),process.stderr.write(
|
|
46
|
-
`)}var
|
|
54
|
+
`),process.stderr.write(p("✓")+` Wallet saved
|
|
55
|
+
`)}var p2=N(()=>{i0();H0();F0();K0()});var c2={};m(c2,{default:()=>QZ});async function QZ($){let Z=z0(D($,"wallet")),Q=Y0(Z),X=T0({args:$}),z=Q.addresses[X];if(x($,"json")){let J={addresses:Q.addresses,activeNetwork:X,activeAddress:z,createdAt:Q.createdAt,file:Z};if(Q.shieldedAddresses)J.shieldedAddresses=Q.shieldedAddresses;y(J);return}process.stdout.write(z+`
|
|
47
56
|
`),process.stderr.write(`
|
|
48
|
-
`+
|
|
57
|
+
`+g("Wallet Info")+`
|
|
49
58
|
|
|
50
|
-
`);
|
|
59
|
+
`);let Y=Object.keys(Q.addresses);for(let J=0;J<Y.length;J++){let G=Y[J],q=G===X,K=Q.addresses[G],B=Q.shieldedAddresses?.[G],H=q?M(f(G))+V(" (active)"):G;if(process.stderr.write(` ${H}
|
|
60
|
+
`),process.stderr.write(` ${V("unshielded")} ${$0(K)}
|
|
61
|
+
`),B)process.stderr.write(` ${V("shielded ")} ${$0(B)}
|
|
62
|
+
`);else process.stderr.write(` ${V("shielded ")} ${V("(unavailable)")}
|
|
63
|
+
`);if(J<Y.length-1)process.stderr.write(`
|
|
51
64
|
`)}process.stderr.write(`
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
`),process.stderr.write(
|
|
55
|
-
`),process.stderr.write(
|
|
56
|
-
`),process.stderr.write(`
|
|
57
|
-
`+h()+`
|
|
65
|
+
`+o()+`
|
|
66
|
+
|
|
67
|
+
`),process.stderr.write(` ${V("created")} ${Q.createdAt}
|
|
68
|
+
`),process.stderr.write(` ${V("file ")} ${Z}
|
|
58
69
|
|
|
59
|
-
`)}var
|
|
70
|
+
`)}var d2=N(()=>{F0();H0();K0();K0()});function b0(){XZ=!0}function d($,Z){if(!XZ)return;let Q=new Date().toISOString().slice(11,23);process.stderr.write(V(` [${Q}] ${$}: ${Z}`)+`
|
|
71
|
+
`)}var XZ=!1;var B4=()=>{};import{NetworkId as l2}from"@midnight-ntwrk/wallet-sdk-abstractions";function n4($){let Z=n9[$];if(Z===void 0)throw new Error(`Unknown networkId: ${$}`);return Z}var n9;var L$=N(()=>{n9={PreProd:l2.NetworkId.PreProd,Preview:l2.NetworkId.Preview,Undeployed:l2.NetworkId.Undeployed}});import o9 from"ws";function o4($,Z,Q){return new Promise((X,z)=>{let Y=new o9(Z,["graphql-transport-ws"]),J=new Map,G=0,q=0,K=0,B=!1,H=!1,U,j=()=>{let P=new Map,O=0,F=0,I=0;for(let _ of J.values())if(!_.spent){O++;let v=P.get(_.tokenType)??0n;if(P.set(_.tokenType,v+_.value),_.tokenType===i1)if(_.registeredForDustGeneration)F++;else I++}return{balances:P,utxoCount:O,txCount:G,highestTxId:q,registeredUtxos:F,unregisteredUtxos:I}},L=()=>{clearTimeout(U)},W=()=>{if(!H&&B&&(q===0||K>=q))H=!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",(P)=>{let O=JSON.parse(P.toString());switch(O.type){case"connection_ack":Y.send(JSON.stringify({id:"1",type:"subscribe",payload:{query:a9,variables:{address:$}}}));break;case"next":{if(O.payload?.errors){let I=O.payload.errors[0]?.message||"Unknown GraphQL error";if(!H)H=!0,L(),Y.close(),z(new Error(`GraphQL error: ${I}`));return}let F=O.payload?.data?.unshieldedTransactions;if(!F)return;if(F.__typename==="UnshieldedTransaction"){G++;let I=F;K=Math.max(K,I.transaction.id);for(let _ of I.createdUtxos){let v=`${_.intentHash}:${_.outputIndex}`;J.set(v,{value:BigInt(_.value),tokenType:_.tokenType,spent:!1,registeredForDustGeneration:_.registeredForDustGeneration===!0})}for(let _ of I.spentUtxos){let v=`${_.intentHash}:${_.outputIndex}`,b=J.get(v);if(b)b.spent=!0}if(Q)Q(K,q);W()}else if(F.__typename==="UnshieldedTransactionsProgress")q=F.highestTransactionId,B=!0,W();break}case"error":if(!H)H=!0,L(),Y.close(),z(new Error(`GraphQL subscription error: ${JSON.stringify(O.payload)}`));break;case"complete":break}}),Y.on("error",(P)=>{if(!H){H=!0,L();let O=P.message?.trim()?P.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(!H)H=!0,L(),z(new Error(`Indexer closed the connection before balance sync completed. Indexer: ${Z}`))}),U=setTimeout(()=>{if(!H)H=!0,Y.close(),z(new Error(`Balance check timed out after ${s1/1000}s. Indexer: ${Z}`))},s1)})}function _1($){return $===i1}var a9=`
|
|
60
72
|
subscription UnshieldedTransactions($address: UnshieldedAddress!) {
|
|
61
73
|
unshieldedTransactions(address: $address) {
|
|
62
74
|
__typename
|
|
63
75
|
... on UnshieldedTransaction {
|
|
64
76
|
transaction { id hash }
|
|
65
|
-
createdUtxos { value owner tokenType intentHash outputIndex }
|
|
66
|
-
spentUtxos { value owner tokenType intentHash outputIndex }
|
|
77
|
+
createdUtxos { value owner tokenType intentHash outputIndex registeredForDustGeneration }
|
|
78
|
+
spentUtxos { value owner tokenType intentHash outputIndex registeredForDustGeneration }
|
|
67
79
|
}
|
|
68
80
|
... on UnshieldedTransactionsProgress {
|
|
69
81
|
highestTransactionId
|
|
70
82
|
}
|
|
71
83
|
}
|
|
72
84
|
}
|
|
73
|
-
`;var
|
|
85
|
+
`;var T1=()=>{};import r9 from"ws";import*as a4 from"@midnight-ntwrk/ledger-v8";function ZQ(){let $=new a4.DustParameters(s9,t9,e9);return new a4.DustLocalState($)}function O$($,Z,Q={}){let{onProgress:X,onCheckpoint:z,timeoutMs:Y=600000,idleMs:J=5000,initialSilenceMs:G=3000,signal:q,initialState:K,startFromId:B=0}=Q,H=500;return new Promise((U,j)=>{let L=new r9(Z,["graphql-transport-ws"]),W=K??ZQ(),P=[],O=0,F=B-1,I=-1,_=!1,v=!1,b,c,k,J0=()=>{if(clearTimeout(b),c)clearTimeout(c);if(k)clearTimeout(k);try{L.close()}catch{}q?.removeEventListener("abort",C)},w=()=>{if(P.length===0)return;if(W=W.replayEvents($,P),O+=P.length,P.length=0,z&&F>=0)try{z(W,F)}catch{}},i=()=>{if(c)clearTimeout(c);c=setTimeout(()=>{if(_&&!v)e()},J)},e=(R=!1)=>{if(v)return;v=!0,J0();try{w();let T=new Date;W=W.processTtls(T),U({balance:W.walletBalance(T),availableCoins:W.utxos.length,eventCount:O,ownedUtxoCount:W.utxos.length,syncTime:W.syncTime,state:W,lastAppliedEventId:F,partial:R})}catch(T){j(new Error(`Failed to build dust state: ${T.message}`))}},Q0=(R)=>{if(v)return;v=!0,J0(),j(R)},C=()=>Q0(new Error("Operation cancelled"));q?.addEventListener("abort",C,{once:!0}),L.on("open",()=>{L.send(JSON.stringify({type:"connection_init"}))}),L.on("message",(R)=>{let T;try{T=JSON.parse(R.toString())}catch{return}if(T.type==="connection_ack"){if(L.send(JSON.stringify({id:"1",type:"subscribe",payload:{query:$Q,variables:{id:B}}})),K)k=setTimeout(()=>{if(!_&&!v)e()},G);return}if(T.type==="error"){Q0(new Error(`GraphQL subscription error: ${JSON.stringify(T.payload)}`));return}if(T.type!=="next")return;if(T.payload?.errors){Q0(new Error(`GraphQL error: ${T.payload.errors[0]?.message||"unknown"}`));return}let E=T.payload?.data?.dustLedgerEvents;if(!E)return;if(_=!0,k)clearTimeout(k),k=void 0;try{let u=Buffer.from(E.raw,"hex");P.push(a4.Event.deserialize(u))}catch(u){Q0(new Error(`Failed to deserialize dust event ${E.id}: ${u.message}`));return}if(F=E.id,E.maxId>I)I=E.maxId;if(P.length>=500)try{w()}catch(u){Q0(new Error(`Failed applying dust events: ${u.message}`));return}if(X?.(O+P.length,I),i(),F>=I)e()}),L.on("error",(R)=>Q0(new Error(`WebSocket error: ${R.message}`))),L.on("close",()=>{if(v)return;if(!_)e();else Q0(new Error("Indexer closed connection before dust sync completed"))}),b=setTimeout(()=>{e(!0)},Y)})}var s9=5000000000n,t9=8267n,e9,$Q=`
|
|
86
|
+
subscription DustLedgerEvents($id: Int) {
|
|
87
|
+
dustLedgerEvents(id: $id) {
|
|
88
|
+
type: __typename
|
|
89
|
+
id
|
|
90
|
+
raw
|
|
91
|
+
maxId
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
`;var i2=N(()=>{e9=3n*60n*60n});import QQ from"ws";async function r4($,Z,Q=[]){let X=$.timeoutMs??XQ;return new Promise((z,Y)=>{let J=new QQ($.url),G=!1,q=(B)=>{if(!G)G=!0,B()},K=setTimeout(()=>{q(()=>{try{J.close()}catch{}Y(new Error(`RPC ${Z} timed out after ${X}ms`))})},X);J.on("open",()=>{try{J.send(JSON.stringify({jsonrpc:"2.0",id:1,method:Z,params:Q}))}catch(B){clearTimeout(K),q(()=>Y(B instanceof Error?B:new Error(String(B))))}}),J.on("message",(B)=>{clearTimeout(K),q(()=>{try{let H=JSON.parse(B.toString());try{J.close()}catch{}if(H.error)Y(new Error(`RPC ${Z}: ${H.error.message}`));else z(H.result)}catch(H){Y(H instanceof Error?H:new Error(String(H)))}})}),J.on("error",(B)=>{clearTimeout(K),q(()=>Y(B instanceof Error?B:new Error(String(B))))})})}var XQ=5000;var j$=()=>{};var o2={};m(o2,{getChainGenesisHash:()=>YQ,clearChainIdMemo:()=>JQ});async function YQ($){let Z=n2.get($);if(Z&&Date.now()-Z.fetchedAt<zQ)return Z.hash;try{let Q=await r4({url:$,timeoutMs:5000},"chain_getBlockHash",[0]);if(typeof Q!=="string"||!Q.startsWith("0x"))return null;return n2.set($,{hash:Q,fetchedAt:Date.now()}),Q}catch{return null}}function JQ(){n2.clear()}var n2,zQ=60000;var a2=N(()=>{j$();n2=new Map});var GZ={};m(GZ,{validateDustCacheChainId:()=>t2,saveDustCache:()=>s4,primeDustCache:()=>HQ,loadDustCache:()=>D4,getDustCachePath:()=>F$,dustPublicKeyHexFromSeed:()=>A4,dustPublicKeyHex:()=>A1,clearDustDirectCache:()=>N4});import{existsSync as M1,mkdirSync as GQ,writeFileSync as qQ,readFileSync as zZ,unlinkSync as P$,readdirSync as r2,renameSync as KQ}from"node:fs";import{join as R4,dirname as VQ}from"node:path";import{homedir as YZ}from"node:os";import{randomBytes as BQ}from"node:crypto";import*as R1 from"@midnight-ntwrk/ledger-v8";function A1($){return $.toString(16).padStart(64,"0")}function A4($){let Z=_4($),Q=R1.DustSecretKey.fromSeed(Z);return A1(Q.publicKey)}function s2($,Z){let Q=Z??R4(YZ(),B0,G4);return R4(Q,$)}function F$($,Z,Q){let X=Z.slice(0,20);return R4(s2($,Q),`dust-${X}.json`)}function D4($,Z,Q){let X=F$($,Z,Q);if(!M1(X))return null;try{let z=JSON.parse(zZ(X,"utf-8"));if(z.version!==JZ)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:R1.DustLocalState.deserialize(Y),lastAppliedEventId:z.lastAppliedEventId}}catch{return null}}function s4($,Z,Q,X,z,Y){let J={version:JZ,network:$,dustPublicKeyHex:Z,lastAppliedEventId:X,timestamp:new Date().toISOString(),dustState:Buffer.from(Q.serialize()).toString("hex"),...Y?{chainId:Y}:{}},G=F$($,Z,z),q=VQ(G);if(!M1(q))GQ(q,{recursive:!0,mode:N0});let K=G+`.tmp.${BQ(4).toString("hex")}`;try{qQ(K,JSON.stringify(J),{mode:E0}),KQ(K,G)}catch(B){try{P$(K)}catch{}throw B}}async function t2($,Z,Q){let{getChainGenesisHash:X}=await Promise.resolve().then(() => (a2(),o2)),z=await X(Z);if(!z)return[];let Y=s2($,Q);if(!M1(Y))return[];let J=[],G;try{G=r2(Y)}catch{return[]}for(let q of G){if(!q.startsWith("dust-")||!q.endsWith(".json"))continue;let K=R4(Y,q);try{let B=JSON.parse(zZ(K,"utf-8"));if(!B.chainId)continue;if(B.chainId!==z)P$(K),J.push(K)}catch{}}return J}function N4($,Z,Q){let X=Q??R4(YZ(),B0,G4);if($&&Z){let Y=F$($,Z,Q);try{P$(Y)}catch{}return}let z=(Y)=>{if(!M1(Y))return;try{for(let J of r2(Y))if(J.startsWith("dust-")&&J.endsWith(".json"))try{P$(R4(Y,J))}catch{}}catch{}};if($){z(s2($,Q));return}if(!M1(X))return;try{for(let Y of r2(X))z(R4(X,Y))}catch{}}async function HQ($,Z,Q,X={}){let z=_4($),Y=R1.DustSecretKey.fromSeed(z),J=A1(Y.publicKey),G=D4(Z,J),q=G?G.lastAppliedEventId+1:0,K=await O$(Y,Q,{initialState:G?.state,startFromId:q,onProgress:X.onProgress,signal:X.signal});if(K.lastAppliedEventId>=0)s4(Z,J,K.state,K.lastAppliedEventId);else if(!G)s4(Z,J,K.state,-1);return{eventCount:K.eventCount,lastAppliedEventId:K.lastAppliedEventId>=0?K.lastAppliedEventId:G?.lastAppliedEventId??-1,fromCache:G!==null}}var JZ=1;var D1=N(()=>{F1();i2()});import{ShieldedWallet as qZ}from"@midnight-ntwrk/wallet-sdk-shielded";import{UnshieldedWallet as KZ,createKeystore as WQ,PublicKey as UQ}from"@midnight-ntwrk/wallet-sdk-unshielded-wallet";import{DustWallet as VZ}from"@midnight-ntwrk/wallet-sdk-dust-wallet";import{WalletFacade as BZ}from"@midnight-ntwrk/wallet-sdk-facade";import*as t4 from"@midnight-ntwrk/ledger-v8";import{NetworkId as e2,InMemoryTransactionHistoryStorage as LQ,TransactionHistoryStorage as OQ}from"@midnight-ntwrk/wallet-sdk-abstractions";import*as M0 from"rxjs";async function H4($,Z,Q){let X=jQ[Z.networkId];if(X===void 0)throw new Error(`Unknown networkId: ${Z.networkId}`);d("facade",`Building facade for network ${Z.networkId}`),d("facade",`Node: ${Z.node}`),d("facade",`Indexer: ${Z.indexerWS}`),d("facade",`Proof server: ${Z.proofServer}`);let z=B$($),Y=p7($),J=_4($),G=t4.ZswapSecretKeys.fromSeed(z),q=t4.DustSecretKey.fromSeed(J),K=WQ(Y,X),B={networkId:X,indexerClientConnection:{indexerHttpUrl:Z.indexer,indexerWsUrl:Z.indexerWS},costParameters:{additionalFeeOverhead:W2,feeBlocksMargin:U2},txHistoryStorage:new LQ(OQ.TransactionHistoryCommonSchema),provingServerUrl:new URL(Z.proofServer),relayURL:new URL(Z.node)},H=()=>BZ.init({configuration:B,shielded:(W)=>qZ(W).startWithSecretKeys(G),unshielded:(W)=>KZ(W).startWithPublicKey(UQ.fromKeyStore(K)),dust:(W)=>VZ(W).startWithSecretKey(q,t4.LedgerParameters.initialParameters().dust)}),U=Q?PQ(Q,Z,q.publicKey):null,j=!1,L;if(U){d("facade","Restoring from cache...");try{L=await BZ.init({configuration:B,shielded:(W)=>qZ(W).restore(U.shielded),unshielded:(W)=>KZ(W).restore(U.unshielded),dust:(W)=>VZ(W).restore(U.dust)}),j=!0,d("facade","Cache restore successful")}catch(W){d("facade",`Cache restore failed: ${W.message}`),process.stderr.write(` Cache restore failed, building from scratch: ${W.message}
|
|
95
|
+
`),L=await H()}}else d("facade","No cache, building fresh"),L=await H();return{facade:L,keystore:K,zswapSecretKeys:G,dustSecretKey:q,restoredFromCache:j}}function PQ($,Z,Q){try{let X=Z.networkId.toLowerCase(),z=A1(Q),Y=D4(X,z);if(!Y)return $;let J=JSON.parse($.dust),G=J.offset!==void 0?Number(J.offset):-1;if(G>=Y.lastAppliedEventId)return d("facade",`Facade dust offset=${G} >= dust-direct offset=${Y.lastAppliedEventId}; skipping bridge`),$;return J.state=Buffer.from(Y.state.serialize()).toString("hex"),J.offset=Y.lastAppliedEventId.toString(),d("facade",`Bridged dust-direct cache (facade offset ${G} → dust-direct offset ${Y.lastAppliedEventId})`),{...$,dust:JSON.stringify(J)}}catch(X){return d("facade",`Dust-direct bridge skipped: ${X.message}`),$}}function $8($,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 J=$.dust?.state?.progress;if(J&&J.highestRelevantWalletIndex>0&&J.appliedIndex>=J.highestRelevantWalletIndex)z=!0;if(J&&J.highestRelevantWalletIndex===0&&J.appliedIndex===0&&Q)z=!0}catch{}}if(Z==="lite")return Q&&z;let Y=$.shielded?.state?.progress?.isStrictlyComplete()??!1;if(!Y&&Q)try{let J=$.shielded?.state?.progress;if(J&&J.highestRelevantWalletIndex===0&&J.appliedIndex===0)Y=!0}catch{}return Y&&Q&&z}function FQ($){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 W4($,Z={}){let{onProgress:Q,onSyncDetail:X,timeoutMs:z,syncMode:Y="full",requireStrictSync:J=!1}=Z,{facade:G,zswapSecretKeys:q,dustSecretKey:K}=$;d("sync","Starting facade (connecting to node and indexer)..."),await G.start(q,K),d("sync","Facade started, subscribing to state...");let B=z??L2;d("sync",`Sync timeout: ${B/1000}s, mode: ${Y}`);let H=!J&&$.restoredFromCache,U=Date.now();return new Promise((j,L)=>{let W=!1,P=0,O="",F=null,I=setTimeout(()=>{if(!W){if(d("sync",`Sync timed out after ${B/1000}s (${P} emissions)`),F)try{let _=F.unshielded?.progress;d("sync",` unshielded: applied=${_?.appliedId} highest=${_?.highestTransactionId} complete=${_?.isStrictlyComplete()}`);let v=F.dust?.state?.progress;if(d("sync",` dust: applied=${v?.appliedIndex} highest=${v?.highestRelevantWalletIndex} complete=${v?.isStrictlyComplete?.()} connected=${v?.isConnected}`),Y==="full"){let b=F.shielded?.state?.progress;d("sync",` shielded: complete=${b?.isStrictlyComplete()}`)}}catch{}L(new Error("Wallet sync timed out"))}},B);$.keepAlive=G.state().subscribe({next:(_)=>{if(W)return;if(P++,F=_,$.restoredFromCache){let b=_Q(_);if(b){W=!0,clearTimeout(I),d("sync",`Stale cache detected: ${b}`),L(new I$(b));return}}if(Q){let b=_.unshielded.progress;if(b){let c=Number(b.appliedId),k=Number(b.highestTransactionId);Q(Math.min(c,k),k)}}let v=[];try{if((Y==="full"||Y==="no-dust")&&!_.shielded?.state?.progress?.isStrictlyComplete())v.push("shielded");if(Y!=="no-dust"&&FQ(_))v.push("dust");if(!_.unshielded?.progress?.isStrictlyComplete())v.push("unshielded")}catch{}if(v.length>0){X?.(v.join(", "));let b=v.join(",");if(P===1||b!==O||P%100===0)d("sync",`Waiting on: ${v.join(", ")} (emission #${P})`),O=b}if($8(_,Y)){W=!0,clearTimeout(I),d("sync",`Sync complete after ${P} emissions`),j(_);return}if(H){let b=Date.now()-U;if(b>=IQ&&$8(_,"no-dust"))W=!0,clearTimeout(I),d("sync",`Sync resolved via cached-restore grace (${b}ms, ${P} emissions)`),j(_)}},error:(_)=>{if(!W)d("sync",`Sync error: ${_.message}`),clearTimeout(I),L(_)}})})}function _Q($){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 HZ($){let Z=(Q)=>{let X=Q.unshielded?.progress?.isStrictlyComplete()??!1,z=Q.dust?.state?.progress?.isStrictlyComplete()??!1;return X&&z};try{return await M0.firstValueFrom($.facade.state().pipe(M0.filter(Z),M0.timeout(15000)))}catch{return await M0.firstValueFrom($.facade.state())}}async function e4($,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 M0.firstValueFrom($.facade.state().pipe(M0.filter(Q),M0.timeout(Z)))}catch{return await M0.firstValueFrom($.facade.state())}}async function Z8($,Z="full"){return M0.firstValueFrom($.facade.state().pipe(M0.filter((Q)=>$8(Q,Z)),M0.timeout(O2)))}async function n0($){$.keepAlive?.unsubscribe(),await Promise.race([$.facade.stop(),new Promise((Z)=>setTimeout(Z,5000))])}function U0($){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 jQ,IQ=1e4,I$;var o0=N(()=>{F1();D1();B4();jQ={PreProd:e2.NetworkId.PreProd,Preview:e2.NetworkId.Preview,Undeployed:e2.NetworkId.Undeployed};I$=class I$ extends Error{code="STALE_CACHE";constructor($){super(`Cached wallet state is stale (from a previous chain). ${$}
|
|
96
|
+
Run: midnight cache clear --wallet <name> --network <name>
|
|
97
|
+
Or: midnight cache clear (wipe all caches)`);this.name="StaleCacheError"}}});function TQ($){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 _$($){let Z=$?.message?.toLowerCase()??"";return Z.includes("not enough dust")||Z.includes("dust generated")||Z.includes("insufficient funds")||Z.includes("no dust tokens")||TQ($)}function N1($){let Z=$?.message?.toLowerCase()??"";if($?._tag==="Wallet.InsufficientFunds")return!0;return Z==="insufficient funds"||Z.startsWith("insufficient funds")}var UZ={};m(UZ,{validateWalletCacheChainId:()=>X8,saveWalletCache:()=>S0,loadWalletCache:()=>Z4,getCachePath:()=>M$,clearWalletCache:()=>r0});import{existsSync as x1,mkdirSync as MQ,writeFileSync as RQ,readFileSync as WZ,unlinkSync as E1,readdirSync as T$,renameSync as AQ}from"node:fs";import{join as a0,dirname as DQ}from"node:path";import{homedir as Q8}from"node:os";import{randomBytes as NQ}from"node:crypto";function M$($,Z,Q){let X=Q??a0(Q8(),B0,G4),z=$.slice(0,20);return a0(X,Z,`${z}.json`)}function Z4($,Z,Q){let X=M$($,Z,Q);if(!x1(X))return null;try{let z=WZ(X,"utf-8"),Y=JSON.parse(z);if(Y.version!==e1)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 S0($,Z,Q,X,z){let[Y,J,G]=await Promise.all([Q.shielded.serializeState(),Q.unshielded.serializeState(),Q.dust.serializeState()]),q={version:e1,network:Z,address:$,timestamp:new Date().toISOString(),wallets:{shielded:Y,unshielded:J,dust:G},...z?{chainId:z}:{}},K=M$($,Z,X),B=DQ(K);if(!x1(B))MQ(B,{recursive:!0,mode:N0});let H=K+`.tmp.${NQ(4).toString("hex")}`;try{RQ(H,JSON.stringify(q),{mode:E0}),AQ(H,K)}catch(U){try{E1(H)}catch{}throw U}}async function X8($,Z,Q){let{getChainGenesisHash:X}=await Promise.resolve().then(() => (a2(),o2)),z=await X(Z);if(!z)return[];let Y=Q??a0(Q8(),B0,G4),J=a0(Y,$);if(!x1(J))return[];let G=[],q;try{q=T$(J)}catch{return[]}for(let K of q){if(!K.endsWith(".json"))continue;let B=a0(J,K);try{let H=JSON.parse(WZ(B,"utf-8"));if(!H.chainId)continue;if(H.chainId!==z)E1(B),G.push(B)}catch{}}return G}function r0($,Z,Q){let X=Q??a0(Q8(),B0,G4);if($&&Z){let z=M$($,Z,Q);try{E1(z)}catch{}return}if(Z){let z=a0(X,Z);if(!x1(z))return;try{for(let Y of T$(z))if(Y.endsWith(".json"))E1(a0(z,Y))}catch{}return}if(!x1(X))return;try{for(let z of T$(X)){let Y=a0(X,z);try{for(let J of T$(Y))if(J.endsWith(".json"))E1(a0(Y,J))}catch{}}}catch{}}var U4=()=>{};var jZ={};m(jZ,{setDefaultRepository:()=>wQ,defaultRepository:()=>j0,WalletDataRepository:()=>z8});import*as OZ from"@midnight-ntwrk/ledger-v8";class z8{now;fetchTip;fetchUnshielded;fetchDust;cacheDir;dustMemo=new Map;unshieldedMemo=new Map;tipMemo=new Map;constructor($={}){this.now=$.now??Date.now,this.fetchTip=$.fetchTip??bQ,this.fetchUnshielded=$.fetchUnshielded??SQ,this.fetchDust=$.fetchDust??kQ,this.cacheDir=$.cacheDir}async dust($,Z,Q={}){let X=R$(Z),z=A4($),Y=`${X}:${z}`;if(!Q.forceFresh){let U=await this.tryMemo(Y,Z,this.dustMemo,Q.signal);if(U)return{...U,fromCache:!0,eventsApplied:0}}await t2(X,z,Z.node);let J=Q.forceFresh?null:D4(X,z,this.cacheDir),G=J!==null,q=0,K=null;for(let U=0;U<xQ;U++){if(Q.signal?.aborted)throw new Error("Operation cancelled");let j=J?J.lastAppliedEventId+1:0;if(Q.onStatus?.(U>0?`Resuming dust from event ${j} (continuation ${U+1})…`:J?`Resuming dust from event ${j}…`:"Reading dust events…"),K=await this.fetchDust($,Z,{startFromId:j,initialState:J?.state,signal:Q.signal,onProgress:(L,W)=>{let P=Math.max(1,W+1-j);Q.onStatus?.(`Reading dust events… ${L}/${P}`)},onCheckpoint:(L,W)=>{try{s4(X,z,L,W,this.cacheDir)}catch{}}}),K.lastAppliedEventId>=0||!J){let L=K.lastAppliedEventId>=0?K.lastAppliedEventId:J?.lastAppliedEventId??-1;try{s4(X,z,K.state,L,this.cacheDir)}catch{}}if(q+=K.eventCount,!K.partial)break;J=D4(X,z,this.cacheDir)}if(!K)throw new Error("Dust sync exhausted retries without producing a result");let B={state:K.state,balance:K.balance,availableCoins:K.availableCoins,ownedUtxoCount:K.ownedUtxoCount,syncTime:K.syncTime,fromCache:G,eventsApplied:q,fetchedAt:this.now()},H=await this.getTip(Z,Q.signal);return this.dustMemo.set(Y,{value:B,fetchedAt:B.fetchedAt,tipAtFetch:H}),B}async unshielded($,Z,Q={}){let X=R$(Z),z=typeof $==="string"?$:y0($,X),Y=`${X}:${z}`;if(!Q.forceFresh){let K=await this.tryMemo(Y,Z,this.unshieldedMemo,Q.signal);if(K)return{...K,fromCache:!0}}let G={...await this.fetchUnshielded(z,Z,Q.onProgress),fromCache:!1,fetchedAt:this.now()},q=await this.getTip(Z,Q.signal);return this.unshieldedMemo.set(Y,{value:G,fetchedAt:G.fetchedAt,tipAtFetch:q}),G}async withFacade($,Z,Q,X={}){let z=X.syncMode??"full",Y=X.requireStrictSync??!0,J=Y&&!X.readOnly,G=R$(Z),q=y0($,G),K=Z.networkId!=="Undeployed",B=X.syncTimeoutMs??(K?vQ:CQ);if(await X8(q,G,Z.node),J&&!X.noPrime&&!X.forceFresh)try{await this.dust($,Z,{signal:X.signal,onStatus:X.onStatus})}catch(W){X.onSyncWarning?.("PrePrime",W.message)}let H,U,j=!1,L=async()=>{if(j||!H)return;j=!0;try{await n0(H)}catch{}};try{for(let O=1;O<=C1;O++){if(X.signal?.aborted)throw new Error("Operation cancelled");let F=X.forceFresh?null:Z4(q,G,this.cacheDir);H=await H4($,Z,F);try{U=await W4(H,{onProgress:X.onSyncProgress,onSyncDetail:X.onSyncDetail,timeoutMs:B,syncMode:z,requireStrictSync:Y});break}catch(I){if(X.signal?.aborted)throw new Error("Operation cancelled");if(I instanceof I$&&O<C1&&J){X.onStatus?.(`Cache is stale, clearing and rebuilding (attempt ${O+1}/${C1})...`),await n0(H).catch(()=>{}),H=void 0,r0(q,G,this.cacheDir),N4(G,A4($),this.cacheDir);try{await this.dust($,Z,{signal:X.signal,onStatus:X.onStatus})}catch{}continue}if(O<C1&&String(I?.message??"").includes("timed out")){try{await S0(q,G,H.facade,this.cacheDir)}catch{}X.onStatus?.(`Sync timed out, retrying (attempt ${O+1}/${C1})...`),await n0(H).catch(()=>{}),H=void 0;continue}throw I}}if(!H||!U)throw new Error("Sync failed: no facade state after retries");let W,P=0;while(!0){if(P++,X.signal?.aborted)throw new Error("Operation cancelled");try{W=await Q({bundle:H,state:U});break}catch(O){if(!(J&&P<LZ&&N1(O)))throw O;if(X.onStatus?.(`Refreshing wallet state (attempt ${P+1}/${LZ})...`),await n0(H).catch(()=>{}),H=void 0,await new Promise((_)=>setTimeout(_,yQ)),X.signal?.aborted)throw new Error("Operation cancelled");let I=Z4(q,G,this.cacheDir);H=await H4($,Z,I),U=await W4(H,{onProgress:X.onSyncProgress,onSyncDetail:X.onSyncDetail,timeoutMs:B,syncMode:z,requireStrictSync:Y})}}if(!X.skipAutoSave)try{await S0(q,G,H.facade,this.cacheDir)}catch{}if(!X.readOnly)this.invalidate({network:Z,seed:$});return W}finally{await L()}}invalidate($){let Z=R$($.network),Q=$.kinds??["dust","unshielded","facade"];if(Q.includes("dust"))if($.seed){let X=A4($.seed);this.dustMemo.delete(`${Z}:${X}`)}else this.deleteByPrefix(this.dustMemo,`${Z}:`);if(Q.includes("unshielded")){let X=$.address??($.seed?y0($.seed,Z):null);if(X)this.unshieldedMemo.delete(`${Z}:${X}`);else this.deleteByPrefix(this.unshieldedMemo,`${Z}:`)}if(Q.includes("facade")&&$.seed){let X=y0($.seed,Z);try{r0(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<EQ)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 bQ($){return r4({url:$.node,timeoutMs:3000},"chain_getBlockHash",[])}function SQ($,Z,Q){return o4($,Z.indexerWS,Q)}function kQ($,Z,Q){let X=_4($),z=OZ.DustSecretKey.fromSeed(X);return O$(z,Z.indexerWS,Q)}function R$($){let Z=$.networkId.toLowerCase();if(_0(Z))return Z;throw new Error(`Unsupported network: ${$.networkId}`)}function j0(){if(!A$)A$=new z8;return A$}function wQ($){A$=$}var EQ=5000,xQ=6,CQ=30000,vQ=120000,C1=3,LZ=5,yQ=5000,A$=null;var L4=N(()=>{C0();o0();D1();U4();T1();i2();j$();F1();i0()});function PZ($={}){let Z=Math.min(Math.max($.windowMs??15000,1000),60000),Q=[],X=null,z=(J)=>{let G=J-Z;while(Q.length>0&&Q[0].t<G)Q.shift()},Y=()=>{let J=Q[Q.length-1],G=Q[0],q=J.highest>0?Math.min(100,Math.round(J.applied/J.highest*100)):null,K=null,B=null;if(Q.length>=2){let H=J.applied-G.applied,U=J.t-G.t;if(U>=1000&&H>0){K=H/(U/1000);let j=Math.max(0,J.highest-J.applied);B=j>0?j/K:0}}return{applied:J.applied,highest:J.highest,percent:q,ratePerSec:K,etaSec:B}};return{sample(J){return Q.push(J),z(J.t),X=Y(),X},snapshot(){return X},reset(){Q.length=0,X=null}}}function FZ($,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 ${fQ($.etaSec)}`);if($.ratePerSec!==null&&$.ratePerSec>0)Q.push(`${Math.round($.ratePerSec)} evt/s`);return Q.join(" · ")}function fQ($){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 G8 from"@midnight-ntwrk/ledger-v8";import{MidnightBech32m as hQ,UnshieldedAddress as mQ}from"@midnight-ntwrk/wallet-sdk-address-format";import{NetworkId as Y8}from"@midnight-ntwrk/wallet-sdk-abstractions";function v1($){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 y1($){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 uQ($,Z){let Q=gQ[Z.networkId];if(Q===void 0)throw new Error(`Unknown networkId: ${Z.networkId}`);try{return hQ.parse($).decode(mQ,Q)}catch(X){throw new Error(`Invalid recipient address: ${X.message}
|
|
98
|
+
Expected a bech32m address for network "${Z.networkId}"`)}}function D$($){let Z=$<0n?-$:$,Q=Z/1000000000000000n,X=Z%1000000000000000n;return`${$<0n?"-":""}${Q}.${X.toString().padStart(15,"0").slice(0,6)}`}function J8($){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 P0(){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 pQ($,Z,Q){let X=new Date(Date.now()+w4*60*1000);await Promise.race([$.facade.dust.waitForSyncedState(),new Promise((K,B)=>setTimeout(()=>B(new Error("Insufficient funds: dust wallet sync timed out")),a1))]);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 J=$.keystore.signData(Y.signatureData(1)),G=await $.facade.dust.addDustGenerationSignature(z,J),q=await $.facade.finalizeTransaction(G);return await $.facade.submitTransaction(q)}async function cQ($,Z,Q,X){let z=Date.now(),Y=z+P2,J,G=!1,q=setInterval(()=>{if(G&&X){let K=J8(Date.now()-z);X(`Waiting for dust generation capacity (${K} elapsed, ~5 min on fresh wallets)...`)}},1000);try{while(Date.now()<Y)try{return await pQ($,Z,Q)}catch(K){if(J=K,_$(K)&&Date.now()+t1<Y){G=!0;let B=J8(Date.now()-z);X?.(`Waiting for dust generation capacity (${B} elapsed, ~5 min on fresh wallets)...`),await new Promise((H)=>setTimeout(H,t1));continue}throw K}throw J??new Error("Dust registration timed out")}finally{clearInterval(q)}}async function s0($,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),Y;if(z.length>0){Z?.(`Registering ${z.length} UTXO(s) for dust generation...`);let G=z.map((q)=>({...q.utxo,ctime:new Date(q.meta.ctime)}));Y=await cQ($,G,X.dust.address,Z)}else Z?.("UTXOs already registered, waiting for dust generation...");Z?.("Waiting for dust tokens...");let J=Date.now();while(Date.now()-J<r1){try{if((await Promise.race([$.facade.waitForSyncedState(),new Promise((q,K)=>setTimeout(()=>K(new Error("Poll sync timed out")),a1))])).dust.balance(new Date)>0n)return Z?.("Dust available"),{alreadyAvailable:!1,txHash:Y}}catch{}await new Promise((G)=>setTimeout(G,5000))}throw new Error("Timed out waiting for dust tokens. Try running: midnight dust register")}async function dQ($,Z,Q,X,z,Y,J){let G=Date.now(),q=G+r1,K,B=0;while(!0)try{if(K)await Z8($,"lite");let H=new Date(Date.now()+w4*60*1000);d("transfer","Building transfer transaction...");let U=await $.facade.transferTransaction([{type:"unshielded",outputs:[{amount:Q,receiverAddress:Z,type:G8.unshieldedToken().raw}]}],{shieldedSecretKeys:$.zswapSecretKeys,dustSecretKey:$.dustSecretKey},{ttl:H,payFees:!0});d("transfer","Signing recipe...");let j=await $.facade.signRecipe(U,(F)=>$.keystore.signData(F));d("transfer","Generating ZK proof..."),X?.();let L=await Promise.race([$.facade.finalizeRecipe(j),new Promise((F,I)=>{setTimeout(()=>I(new Error("ZK proof generation timed out")),K1)})]);d("transfer","Submitting transaction to node..."),z?.();let W=Date.now(),P=J?setInterval(()=>J(Date.now()-W),1000):void 0,O;try{O=await $.facade.submitTransaction(L)}finally{if(P)clearInterval(P)}return d("transfer",`Transaction submitted: ${O}`),O}catch(H){if(K=H,(H?.code===F2||H?.message?.includes("115")||H?.message?.toLowerCase().includes("stale"))&&++B<j2)continue;if(N1(H))throw H;if(_$(H)&&Date.now()<q){let j=J8(Date.now()-G);Y?.(`Dust insufficient, re-ensuring (${j} elapsed)...`);try{let L=await Z8($,"lite"),W=L.dust.balance(new Date);if(W>0n&&W<k4)throw new Error(`Insufficient dust for transaction fees.
|
|
99
|
+
Available: ${D$(W)} DUST, need ≥${D$(k4)} DUST.
|
|
100
|
+
Dust regenerates over time from registered NIGHT UTXOs.
|
|
101
|
+
Check status: midnight dust status`);await s0($,Y,L)}catch(L){if(String(L?.message).startsWith("Insufficient dust"))throw L;await new Promise((W)=>setTimeout(W,5000))}continue}throw H}}async function $1($){let{seedBuffer:Z,networkConfig:Q,recipientAddress:X,amountNight:z,signal:Y,onSync:J,onSyncDetail:G,onDust:q,onProving:K,onSubmitting:B,onSubmittingTick:H,onSyncWarning:U}=$,j=v1(z),L=uQ(X,Q),W=U0(U),P=P0();try{return await j0().withFacade(Z,Q,async({bundle:O,state:F})=>{if(Y?.aborted)throw new Error("Operation cancelled");d("transfer","Sync complete, checking balance...");let I=F.unshielded.balances[G8.unshieldedToken().raw]??0n,_=F.unshielded.availableCoins?.length??0;if(d("transfer",`Balance: ${Number(I)/o1} NIGHT, coins: ${_}`),I<j){let c=Number(I)/o1;throw new Error(`Insufficient balance: ${c.toFixed(6)} NIGHT available, ${z} NIGHT requested`)}let v=F.dust.balance(new Date);if(v>0n&&v<k4)throw new Error(`Insufficient dust for transaction fees.
|
|
102
|
+
Available: ${D$(v)} DUST, need ≥${D$(k4)} DUST.
|
|
103
|
+
Dust regenerates over time from registered NIGHT UTXOs.
|
|
104
|
+
Check status: midnight dust status`);if(Y?.aborted)throw new Error("Operation cancelled");if(d("transfer","Ensuring dust availability..."),await s0(O,q,F),d("transfer","Dust available"),Y?.aborted)throw new Error("Operation cancelled");return d("transfer","Building and submitting transaction..."),{txHash:await dQ(O,L,j,K,B,q,H),amountMicroNight:j}},{syncMode:"lite",requireStrictSync:!0,signal:Y,onStatus:q,onSyncProgress:J,onSyncDetail:G,onSyncWarning:U})}finally{P(),W()}}var gQ;var O4=N(()=>{o0();L4();B4();gQ={PreProd:Y8.NetworkId.PreProd,Preview:Y8.NetworkId.Preview,Undeployed:Y8.NetworkId.Undeployed}});function TZ(){return E4}function _Z($){return $.replace(/\x1b\[[0-9;]*m/g,"")}function l($){let Z=!1;if(!h0()){let G=$;process.stderr.write(`⠋ ${G}`);let q={update(K){G=K,process.stderr.write(`\r⠋ ${G}\x1B[K`)},stop(K){if(Z)return;Z=!0,E4=null,process.stderr.write(`\r✓ ${K??G}\x1B[K
|
|
105
|
+
`)},fail(K){if(Z)return;Z=!0,E4=null,process.stderr.write(`\r✗ ${K??G}\x1B[K
|
|
74
106
|
`)},log(K){process.stderr.write(`\r\x1B[K${K}
|
|
75
|
-
`),process.stderr.write(`⠋ ${G}`)}};return
|
|
107
|
+
`),process.stderr.write(`⠋ ${G}`)}};return E4=q,q}let Q=0,X=$,z=()=>{let G=f(IZ[Q]),K=(process.stderr.columns||80)-4,B=X;if(_Z(B).length>K)B=_Z(B).slice(0,K);process.stderr.write(`\r${G} ${B}\x1B[K`),Q=(Q+1)%IZ.length};z();let Y=setInterval(z,lQ),J={update(G){X=G},stop(G){if(Z)return;Z=!0,clearInterval(Y),E4=null;let q=G??X;process.stderr.write(`\r\x1B[32m✓\x1B[0m ${q}\x1B[K
|
|
108
|
+
`)},fail(G){if(Z)return;Z=!0,clearInterval(Y),E4=null;let q=G??X;process.stderr.write(`\r\x1B[31m✗\x1B[0m ${q}\x1B[K
|
|
76
109
|
`)},log(G){process.stderr.write(`\r\x1B[K${G}
|
|
77
|
-
`),z()}};return
|
|
78
|
-
`);
|
|
79
|
-
`);else process.stdout.write(`${
|
|
110
|
+
`),z()}};return E4=J,J}async function N$($,Z,Q){let X=Date.now(),z=(G)=>{let q=Math.floor(G/1000),K=Math.floor(q/60).toString().padStart(2,"0"),B=(q%60).toString().padStart(2,"0");return`${K}:${B}`},Y=()=>$.update(`${Z} ${z(Date.now()-X)}`);Y();let J=setInterval(Y,1000);try{return await Q}finally{clearInterval(J)}}var IZ,lQ=80,E4=null;var u0=N(()=>{IZ=["⠋","⠙","⠹","⠸","⠼","⠴","⠦","⠧","⠇","⠏"]});var q8={};m(q8,{default:()=>RZ});import*as MZ from"@midnight-ntwrk/ledger-v8";import{MidnightBech32m as iQ}from"@midnight-ntwrk/wallet-sdk-address-format";async function RZ($){if($.subcommand)return oQ($);return aQ($)}function nQ($){let Z=$.match(/^mn_addr_([a-z]+)1/);if(!Z)return null;let Q=Z[1];return _0(Q)?Q:null}async function oQ($){if(D0($))b0();let Z=$.subcommand;d("balance",`address=${Z}`);let Q=nQ(Z),X=D($,"network");if(X&&Q&&X!==Q)throw new Error(`Address belongs to ${Q} but --network is ${X}.
|
|
111
|
+
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}=V0({args:$});x0(Y,{proofServer:D($,"proof-server"),node:D($,"node"),indexerWS:D($,"indexer-ws")});let J=l(`Checking balance on ${z}...`);try{let G=await j0().unshielded(Z,Y,{forceFresh:x($,"no-cache"),onProgress:(q,K)=>{if(K>0){let B=Math.round(q/K*100);J.update(`Syncing transactions... ${B}%`)}}});if(J.stop(`Synced ${G.txCount} transactions`),x($,"json")){let q={};for(let[B,H]of G.balances){let U=_1(B)?"NIGHT":B;q[U]=_1(B)?I4(H):H.toString()}let K=q.NIGHT!==void 0?{NIGHT:q.NIGHT}:{};if(z4($)){y({network:z,balances:q,...K,utxoCount:G.utxoCount});return}y({address:Z,network:z,balances:q,...K,utxoCount:G.utxoCount,txCount:G.txCount});return}if(G.balances.size===0)process.stdout.write(`0
|
|
112
|
+
`);else for(let[q,K]of G.balances)process.stdout.write(`${_1(q)?"NIGHT":q}=${K}
|
|
80
113
|
`);if(process.stderr.write(`
|
|
81
|
-
`+
|
|
114
|
+
`+g("Balance")+`
|
|
82
115
|
|
|
83
|
-
`),process.stderr.write(
|
|
84
|
-
`),process.stderr.write(
|
|
85
|
-
`),process.stderr.write(
|
|
86
|
-
`),process.stderr.write(
|
|
116
|
+
`),process.stderr.write(A("Address",$0(Z))+`
|
|
117
|
+
`),process.stderr.write(A("Network",z)+`
|
|
118
|
+
`),process.stderr.write(A("UTXOs",G.utxoCount.toString())+`
|
|
119
|
+
`),process.stderr.write(A("Transactions",G.txCount.toString())+`
|
|
87
120
|
`),process.stderr.write(`
|
|
88
|
-
`),
|
|
89
|
-
`);else for(let[
|
|
90
|
-
`);else{let q
|
|
121
|
+
`),G.balances.size===0)process.stderr.write(` ${V("No balance found")}
|
|
122
|
+
`);else for(let[q,K]of G.balances)if(_1(q))process.stderr.write(A("NIGHT",M(Q$(K)))+`
|
|
123
|
+
`);else{let B=q.slice(0,8)+"…"+q.slice(-8);process.stderr.write(A(`Token ${B}`,M(K.toString()))+`
|
|
91
124
|
`)}process.stderr.write(`
|
|
92
|
-
`+
|
|
125
|
+
`+o()+`
|
|
126
|
+
|
|
127
|
+
`)}catch(G){throw J.fail("Failed"),G}}async function aQ($){if(D0($))b0();let{name:Z,config:Q}=V0({args:$});d("balance",`network=${Z} indexerWS=${Q.indexerWS}`);let X=z0(D($,"wallet"));d("balance",`wallet path=${X}`);let z=Y0(X),Y=Buffer.from(z.seed,"hex"),J=z.addresses[Z],G=z.shieldedAddresses?.[Z]??"";x0(Q,{proofServer:D($,"proof-server"),node:D($,"node"),indexerWS:D($,"indexer-ws")});let q=U0(),K=P0(),B=x($,"no-cache"),H=x($,"json"),U=MZ.unshieldedToken().raw,j=null;try{if(!H){if(process.stderr.write(`
|
|
128
|
+
`+g("Balance")+`
|
|
129
|
+
|
|
130
|
+
`),process.stderr.write(A("Address",$0(J))+`
|
|
131
|
+
`),G)process.stderr.write(A("Shielded",$0(G))+`
|
|
132
|
+
`);process.stderr.write(A("Network",Z)+`
|
|
133
|
+
`),process.stderr.write(`
|
|
134
|
+
`+M("Unshielded")+`
|
|
135
|
+
`)}if(!H)j=l("Checking unshielded...");d("balance",`phase=unshielded forceFresh=${B}`);let L=await j0().unshielded(J,Q,{forceFresh:B}),W=L.balances.get(U)??0n,P=L.utxoCount;if(d("balance",`unshielded fromCache=${L.fromCache} utxos=${P} txs=${L.txCount}`),j)if(j.stop("Unshielded ready"),j=null,W>0n)process.stderr.write(A(" NIGHT",M(Q$(W)))+`
|
|
136
|
+
`),process.stderr.write(A(" UTXOs",P.toString())+`
|
|
137
|
+
`);else process.stderr.write(` ${V("No unshielded balance")}
|
|
138
|
+
`);if(!H)process.stdout.write(`NIGHT=${W}
|
|
139
|
+
`),process.stderr.write(`
|
|
140
|
+
`+M("Shielded")+`
|
|
141
|
+
`);if(!H)j=l("Syncing shielded...");let O=PZ(),F=n4(Q.networkId),{liveShieldedAddrStr:I,shieldedBalance:_,shieldedCoins:v,pendingCoins:b}=await j0().withFacade(Y,Q,async({state:c})=>{let k=iQ.encode(F,c.shielded.address).asString();return M4(X,Z,k),{liveShieldedAddrStr:k,shieldedBalance:c.shielded.balances[U]??0n,shieldedCoins:c.shielded.availableCoins.length,pendingCoins:c.shielded.pendingCoins.length}},{syncMode:"no-dust",requireStrictSync:!1,readOnly:!0,forceFresh:B,onSyncProgress:(c,k)=>{if(!j)return;let J0=O.sample({applied:c,highest:k,t:Date.now()});j.update(FZ(J0,"Syncing shielded"))},onSyncDetail:(c)=>j?.update(`Syncing shielded (waiting on: ${c})`)});if(j)if(j.stop("Shielded ready"),j=null,_>0n)process.stderr.write(A(" NIGHT",M(Q$(_)))+`
|
|
142
|
+
`),process.stderr.write(A(" Coins",`${v} available, ${b} pending`)+`
|
|
143
|
+
`);else process.stderr.write(` ${V("No shielded balance")}
|
|
144
|
+
`);if(H){if(z4($)){y({network:Z,unshielded:{NIGHT:I4(W),utxoCount:P},shielded:{NIGHT:I4(_),availableCoins:v,pendingCoins:b}});return}y({address:J,shieldedAddress:I,network:Z,unshielded:{NIGHT:I4(W),utxoCount:P},shielded:{NIGHT:I4(_),availableCoins:v,pendingCoins:b}});return}process.stdout.write(`SHIELDED_NIGHT=${_}
|
|
145
|
+
`),process.stderr.write(`
|
|
146
|
+
`+o()+`
|
|
93
147
|
|
|
94
|
-
`)}catch(
|
|
148
|
+
`)}catch(L){throw j?.fail("Failed"),L}finally{K(),q()}}var K8=N(()=>{B4();C0();F0();H0();C0();L$();T1();L4();o0();O4();K0();u0()});var V8={};m(V8,{default:()=>AZ});async function AZ($){let Z=S4($,"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=D($,"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=T0({args:$}),J=y0(z,Y,X),G=`m/44'/2400'/0'/NightExternal/${X}`;if(x($,"json")){y({address:J,network:Y,index:X,path:G});return}process.stdout.write(J+`
|
|
95
149
|
`),process.stderr.write(`
|
|
96
|
-
`),process.stderr.write(
|
|
97
|
-
`),process.stderr.write(
|
|
98
|
-
`),process.stderr.write(
|
|
99
|
-
`),process.stderr.write(
|
|
100
|
-
`),process.stderr.write(
|
|
150
|
+
`),process.stderr.write(A("Network",Y)+`
|
|
151
|
+
`),process.stderr.write(A("Index",X.toString())+`
|
|
152
|
+
`),process.stderr.write(A("Address",$0(J))+`
|
|
153
|
+
`),process.stderr.write(A("Path",V(G))+`
|
|
154
|
+
`),process.stderr.write(o()+`
|
|
101
155
|
|
|
102
|
-
`)}var
|
|
156
|
+
`)}var B8=N(()=>{i0();H0();K0()});var H8={};m(H8,{default:()=>DZ});async function DZ($){let Z=T0({args:$}),Q=Buffer.from(J4,"hex"),X=y0(Q,Z);if(x($,"json")){y({address:X,network:Z});return}process.stdout.write(X+`
|
|
103
157
|
`),process.stderr.write(`
|
|
104
|
-
`),process.stderr.write(
|
|
105
|
-
`),process.stderr.write(
|
|
106
|
-
`),process.stderr.write(
|
|
107
|
-
`),process.stderr.write(
|
|
158
|
+
`),process.stderr.write(A("Network",Z)+`
|
|
159
|
+
`),process.stderr.write(A("Address",$0(X))+`
|
|
160
|
+
`),process.stderr.write(A("Seed",V("0x01 (genesis)"))+`
|
|
161
|
+
`),process.stderr.write(o()+`
|
|
108
162
|
|
|
109
|
-
`)}var
|
|
163
|
+
`)}var W8=N(()=>{i0();H0();K0()});var U8={};m(U8,{default:()=>EZ});import*as NZ from"@midnight-ntwrk/ledger-v8";function b1($,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 rQ($){return{readTime:b1($,"readTime",1000000000n),computeTime:b1($,"computeTime",1000000000n),blockUsage:b1($,"blockUsage",10000n),bytesWritten:b1($,"bytesWritten",10000n),bytesChurned:b1($,"bytesChurned",1000000n)}}async function EZ($){let Z=NZ.LedgerParameters.initialParameters(),Q=rQ(Z);if(x($,"json")){y(Q);return}for(let[X,z]of Object.entries(Q))process.stdout.write(`${X}=${z}
|
|
110
164
|
`);process.stderr.write(`
|
|
111
|
-
`+
|
|
165
|
+
`+g("Block Limits")+`
|
|
112
166
|
|
|
113
|
-
`),process.stderr.write(
|
|
167
|
+
`),process.stderr.write(V(" Derived from LedgerParameters.initialParameters()")+`
|
|
114
168
|
|
|
115
|
-
`);for(let[X,z]of Object.entries(Q)){let Y=
|
|
169
|
+
`);for(let[X,z]of Object.entries(Q)){let Y=sQ[X]??"";process.stderr.write(A(X,`${M(z.toLocaleString())} ${V(Y)}`)+`
|
|
116
170
|
`)}process.stderr.write(`
|
|
117
|
-
`+
|
|
118
|
-
`),process.stderr.write(
|
|
119
|
-
`),process.stderr.write(
|
|
120
|
-
|
|
121
|
-
`)}var
|
|
122
|
-
`)}var M7=!1;var u0=()=>{};import{HDWallet as qZ,Roles as p2}from"@midnight-ntwrk/wallet-sdk-hd";function u2($,Z){let Q=qZ.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 R7($){return u2($,p2.Zswap)}function D7($){return u2($,p2.NightExternal)}function A7($){return u2($,p2.Dust)}var N7=()=>{};import{ShieldedWallet as E7}from"@midnight-ntwrk/wallet-sdk-shielded";import{UnshieldedWallet as C7,createKeystore as UZ,PublicKey as VZ,InMemoryTransactionHistoryStorage as jZ}from"@midnight-ntwrk/wallet-sdk-unshielded-wallet";import{DustWallet as x7}from"@midnight-ntwrk/wallet-sdk-dust-wallet";import{WalletFacade as y7}from"@midnight-ntwrk/wallet-sdk-facade";import*as r0 from"@midnight-ntwrk/ledger-v7";import{NetworkId as m2}from"@midnight-ntwrk/wallet-sdk-abstractions";import*as Y0 from"rxjs";async function m0($,Z,Q){let X=BZ[Z.networkId];if(X===void 0)throw new Error(`Unknown networkId: ${Z.networkId}`);v("facade",`Building facade for network ${Z.networkId}`),v("facade",`Node: ${Z.node}`),v("facade",`Indexer: ${Z.indexerWS}`),v("facade",`Proof server: ${Z.proofServer}`);let z=R7($),Y=D7($),J=A7($),G=r0.ZswapSecretKeys.fromSeed(z),q=r0.DustSecretKey.fromSeed(J),K=UZ(Y,X),U={networkId:X,indexerClientConnection:{indexerHttpUrl:Z.indexer,indexerWsUrl:Z.indexerWS},costParameters:{additionalFeeOverhead:b$,feeBlocksMargin:S$},txHistoryStorage:new jZ,provingServerUrl:new URL(Z.proofServer),relayURL:new URL(Z.node)},W=()=>y7.init({configuration:U,shielded:(H)=>E7(H).startWithSecretKeys(G),unshielded:(H)=>C7(H).startWithPublicKey(VZ.fromKeyStore(K)),dust:(H)=>x7(H).startWithSecretKey(q,r0.LedgerParameters.initialParameters().dust)}),j=!1,V;if(Q){v("facade","Restoring from cache...");try{V=await y7.init({configuration:U,shielded:(H)=>E7(H).restore(Q.shielded),unshielded:(H)=>C7(H).restore(Q.unshielded),dust:(H)=>x7(H).restore(Q.dust)}),j=!0,v("facade","Cache restore successful")}catch(H){v("facade",`Cache restore failed: ${H.message}`),process.stderr.write(` Cache restore failed, building from scratch: ${H.message}
|
|
123
|
-
`),V=await W()}}else v("facade","No cache, building fresh"),V=await W();return{facade:V,keystore:K,zswapSecretKeys:G,dustSecretKey:q,restoredFromCache:j}}function b7($,Z="full"){let Q=$.unshielded?.progress?.isStrictlyComplete()??!1,X=$.dust?.state?.progress?.isStrictlyComplete()??!1;if(!X)try{let Y=$.dust?.state?.progress;if(Y&&Y.highestRelevantWalletIndex>0&&Y.appliedIndex>=Y.highestRelevantWalletIndex)X=!0}catch{}if(Z==="lite")return Q&&X;return($.shielded?.state?.progress?.isStrictlyComplete()??!1)&&Q&&X}function WZ($){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 g0($,Z={}){let{onProgress:Q,onSyncDetail:X,timeoutMs:z,syncMode:Y="full"}=Z,{facade:J,zswapSecretKeys:G,dustSecretKey:q}=$;v("sync","Starting facade (connecting to node and indexer)..."),await J.start(G,q),v("sync","Facade started, subscribing to state...");let K=z??v$;return v("sync",`Sync timeout: ${K/1000}s, mode: ${Y}`),new Promise((U,W)=>{let j=!1,V=0,H="",F=null,D=setTimeout(()=>{if(!j){if(v("sync",`Sync timed out after ${K/1000}s (${V} emissions)`),F)try{let P=F.unshielded?.progress;v("sync",` unshielded: applied=${P?.appliedId} highest=${P?.highestTransactionId} complete=${P?.isStrictlyComplete()}`);let M=F.dust?.state?.progress;if(v("sync",` dust: applied=${M?.appliedIndex} highest=${M?.highestRelevantWalletIndex} complete=${M?.isStrictlyComplete?.()} connected=${M?.isConnected}`),Y==="full"){let C=F.shielded?.state?.progress;v("sync",` shielded: complete=${C?.isStrictlyComplete()}`)}}catch{}W(new Error("Wallet sync timed out"))}},K);$.keepAlive=J.state().subscribe({next:(P)=>{if(j)return;if(V++,F=P,Q){let C=P.unshielded.progress;if(C){let A=Number(C.appliedId),k=Number(C.highestTransactionId);Q(Math.min(A,k),k)}}let M=[];try{if(Y==="full"&&!P.shielded?.state?.progress?.isStrictlyComplete())M.push("shielded");if(WZ(P))M.push("dust");if(!P.unshielded?.progress?.isStrictlyComplete())M.push("unshielded")}catch{}if(M.length>0){X?.(M.join(", "));let C=M.join(",");if(V===1||C!==H||V%100===0)v("sync",`Waiting on: ${M.join(", ")} (emission #${V})`),H=C}if(b7(P,Y))j=!0,clearTimeout(D),v("sync",`Sync complete after ${V} emissions`),U(P)},error:(P)=>{if(!j)v("sync",`Sync error: ${P.message}`),clearTimeout(D),W(P)}})})}async function g2($){let Z=(Q)=>{let X=Q.unshielded?.progress?.isStrictlyComplete()??!1,z=Q.dust?.state?.progress?.isStrictlyComplete()??!1;return X&&z};try{return await Y0.firstValueFrom($.facade.state().pipe(Y0.filter(Z),Y0.timeout(15000)))}catch{return await Y0.firstValueFrom($.facade.state())}}async function S7($,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 Y0.firstValueFrom($.facade.state().pipe(Y0.filter(Q),Y0.timeout(Z)))}catch{return await Y0.firstValueFrom($.facade.state())}}async function c2($,Z="full"){return Y0.firstValueFrom($.facade.state().pipe(Y0.filter((Q)=>b7(Q,Z)),Y0.timeout(w$)))}async function c0($){$.keepAlive?.unsubscribe(),await Promise.race([$.facade.stop(),new Promise((Z)=>setTimeout(Z,5000))])}function e0($){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 BZ;var u1=x(()=>{N7();u0();BZ={PreProd:m2.NetworkId.PreProd,Preview:m2.NetworkId.Preview,Undeployed:m2.NetworkId.Undeployed}});import{existsSync as g1,mkdirSync as HZ,writeFileSync as LZ,readFileSync as FZ,unlinkSync as m1,readdirSync as d2,renameSync as PZ}from"node:fs";import{join as d0,dirname as OZ}from"node:path";import{homedir as v7}from"node:os";import{randomBytes as IZ}from"node:crypto";function l2($,Z,Q){let X=Q??d0(v7(),J0,j2),z=$.slice(0,20);return d0(X,Z,`${z}.json`)}function l0($,Z,Q){let X=l2($,Z,Q);if(!g1(X))return null;try{let z=FZ(X,"utf-8"),Y=JSON.parse(z);if(Y.version!==V2)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 E0($,Z,Q,X){let[z,Y,J]=await Promise.all([Q.shielded.serializeState(),Q.unshielded.serializeState(),Q.dust.serializeState()]),G={version:V2,network:Z,address:$,timestamp:new Date().toISOString(),wallets:{shielded:z,unshielded:Y,dust:J}},q=l2($,Z,X),K=OZ(q);if(!g1(K))HZ(K,{recursive:!0,mode:P0});let U=q+`.tmp.${IZ(4).toString("hex")}`;try{LZ(U,JSON.stringify(G),{mode:x0}),PZ(U,q)}catch(W){try{m1(U)}catch{}throw W}}function c1($,Z,Q){let X=Q??d0(v7(),J0,j2);if($&&Z){let z=l2($,Z,Q);try{m1(z)}catch{}return}if(Z){let z=d0(X,Z);if(!g1(z))return;try{for(let Y of d2(z))if(Y.endsWith(".json"))m1(d0(z,Y))}catch{}return}if(!g1(X))return;try{for(let z of d2(X)){let Y=d0(X,z);try{for(let J of d2(Y))if(J.endsWith(".json"))m1(d0(Y,J))}catch{}}}catch{}}var F1=()=>{};import*as n2 from"@midnight-ntwrk/ledger-v7";import{MidnightBech32m as _Z,UnshieldedAddress as TZ}from"@midnight-ntwrk/wallet-sdk-address-format";import{NetworkId as i2}from"@midnight-ntwrk/wallet-sdk-abstractions";function RZ($){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 l1($){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 DZ($,Z){let Q=MZ[Z.networkId];if(Q===void 0)throw new Error(`Unknown networkId: ${Z.networkId}`);try{return _Z.parse($).decode(TZ,Q)}catch(X){throw new Error(`Invalid recipient address: ${X.message}
|
|
124
|
-
Expected a bech32m address for network "${Z.networkId}"`)}}function AZ($){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 k7($){let Z=$?.message?.toLowerCase()??"";return Z.includes("not enough dust")||Z.includes("dust generated")||Z.includes("insufficient funds")||Z.includes("no dust tokens")||AZ($)}function d1($){let Z=$<0n?-$:$,Q=Z/1000000000000000n,X=Z%1000000000000000n;return`${$<0n?"-":""}${Q}.${X.toString().padStart(15,"0").slice(0,6)}`}function o2($){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 P1(){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 NZ($,Z,Q){let X=new Date(Date.now()+z1*60*1000);await Promise.race([$.facade.dust.waitForSyncedState(),new Promise((K,U)=>setTimeout(()=>U(new Error("Insufficient funds: dust wallet sync timed out")),D1))]);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 J=$.keystore.signData(Y.signatureData(1)),G=await $.facade.dust.addDustGenerationSignature(z,J),q=await $.facade.finalizeTransaction(G);return await $.facade.submitTransaction(q)}async function EZ($,Z,Q,X){let z=Date.now(),Y=z+h$,J,G=!1,q=setInterval(()=>{if(G&&X){let K=o2(Date.now()-z);X(`Waiting for dust generation capacity (${K} elapsed, ~5 min on fresh wallets)...`)}},1000);try{while(Date.now()<Y)try{return await NZ($,Z,Q)}catch(K){if(J=K,k7(K)&&Date.now()+U2<Y){G=!0;let U=o2(Date.now()-z);X?.(`Waiting for dust generation capacity (${U} elapsed, ~5 min on fresh wallets)...`),await new Promise((W)=>setTimeout(W,U2));continue}throw K}throw J??new Error("Dust registration timed out")}finally{clearInterval(q)}}async function i1($,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),Y;if(z.length>0){Z?.(`Registering ${z.length} UTXO(s) for dust generation...`);let G=z.map((q)=>({...q.utxo,ctime:new Date(q.meta.ctime)}));Y=await EZ($,G,X.dust.address,Z)}else Z?.("UTXOs already registered, waiting for dust generation...");Z?.("Waiting for dust tokens...");let J=Date.now();while(Date.now()-J<K2){try{if((await Promise.race([$.facade.waitForSyncedState(),new Promise((q,K)=>setTimeout(()=>K(new Error("Poll sync timed out")),D1))])).dust.balance(new Date)>0n)return Z?.("Dust available"),{alreadyAvailable:!1,txHash:Y}}catch{}await new Promise((G)=>setTimeout(G,5000))}throw new Error("Timed out waiting for dust tokens. Try running: midnight dust register")}async function CZ($,Z,Q,X,z,Y){let J=Date.now(),G=J+K2,q,K=0;while(!0)try{if(q)await c2($,"lite");let U=new Date(Date.now()+z1*60*1000);v("transfer","Building transfer transaction...");let W=await $.facade.transferTransaction([{type:"unshielded",outputs:[{amount:Q,receiverAddress:Z,type:n2.unshieldedToken().raw}]}],{shieldedSecretKeys:$.zswapSecretKeys,dustSecretKey:$.dustSecretKey},{ttl:U,payFees:!0});v("transfer","Signing recipe...");let j=await $.facade.signRecipe(W,(F)=>$.keystore.signData(F));v("transfer","Generating ZK proof..."),X?.();let V=await Promise.race([$.facade.finalizeRecipe(j),new Promise((F,D)=>{setTimeout(()=>D(new Error("ZK proof generation timed out")),A1)})]);v("transfer","Submitting transaction to node..."),z?.();let H=await $.facade.submitTransaction(V);return v("transfer",`Transaction submitted: ${H}`),H}catch(U){if(q=U,(U?.code===p$||U?.message?.includes("115")||U?.message?.toLowerCase().includes("stale"))&&++K<f$)continue;if(k7(U)&&Date.now()<G){let j=o2(Date.now()-J);Y?.(`Dust insufficient, re-ensuring (${j} elapsed)...`);try{let V=await c2($,"lite"),H=V.dust.balance(new Date);if(H>0n&&H<X1)throw new Error(`Insufficient dust for transaction fees.
|
|
125
|
-
Available: ${d1(H)} DUST, need ≥${d1(X1)} DUST.
|
|
126
|
-
Dust regenerates over time from registered NIGHT UTXOs.
|
|
127
|
-
Check status: midnight dust status`);await i1($,Y,V)}catch(V){if(String(V?.message).startsWith("Insufficient dust"))throw V;await new Promise((H)=>setTimeout(H,5000))}continue}throw U}}async function o1($){let{seedBuffer:Z,networkConfig:Q,recipientAddress:X,amountNight:z,signal:Y,onSync:J,onSyncDetail:G,onDust:q,onProving:K,onSubmitting:U,onSyncWarning:W,noCache:j,walletAddress:V,networkName:H}=$,F=RZ(z),D=DZ(X,Q),P=e0(W),M=P1(),C=!j&&V&&H,A=C?l0(V,H):null;v("transfer","Building facade...");let k=await m0(Z,Q,A),$0=!1,n=async()=>{if(!$0){$0=!0;try{await c0(k)}catch{}}},B0=()=>{n()};Y?.addEventListener("abort",B0,{once:!0});try{let z0=Q.networkId!=="Undeployed",L0=z0?k$:D1,d=3,r;v("transfer",`Sync timeout: ${L0/1000}s (${z0?"remote":"local"} network)`);for(let O=1;O<=3;O++)try{r=await g0(k,{onProgress:J,onSyncDetail:G,timeoutMs:L0,syncMode:"lite"});break}catch(_){if(Y?.aborted)throw new Error("Operation cancelled");if(O<3&&String(_?.message).includes("timed out")){if(C)try{v("transfer","Saving partial sync progress to cache..."),await E0(V,H,k.facade)}catch{}q?.(`Sync timed out, retrying (attempt ${O+1}/3)...`),await c0(k).catch(()=>{});let b=C?l0(V,H):null;k=await m0(Z,Q,b);continue}throw _}if(Y?.aborted)throw new Error("Operation cancelled");v("transfer","Sync complete, checking balance...");let o=r.unshielded.balances[n2.unshieldedToken().raw]??0n;if(v("transfer",`Balance: ${Number(o)/G2} NIGHT`),o<F){let O=Number(o)/G2;throw new Error(`Insufficient balance: ${O.toFixed(6)} NIGHT available, ${z} NIGHT requested`)}if(Y?.aborted)throw new Error("Operation cancelled");v("transfer","Ensuring dust availability..."),await i1(k,q,r),v("transfer","Dust available");let I=r.dust.balance(new Date);if(I>0n&&I<X1)throw new Error(`Insufficient dust for transaction fees.
|
|
128
|
-
Available: ${d1(I)} DUST, need ≥${d1(X1)} DUST.
|
|
129
|
-
Dust regenerates over time from registered NIGHT UTXOs.
|
|
130
|
-
Check status: midnight dust status`);if(Y?.aborted)throw new Error("Operation cancelled");v("transfer","Building and submitting transaction...");let L=await CZ(k,D,F,K,U,q);if(C)try{await E0(V,H,k.facade)}catch{}return{txHash:L,amountMicroNight:F}}finally{Y?.removeEventListener("abort",B0),M(),P(),await n()}}var MZ;var O1=x(()=>{u1();F1();u0();MZ={PreProd:i2.NetworkId.PreProd,Preview:i2.NetworkId.Preview,Undeployed:i2.NetworkId.Undeployed}});var a2={};e(a2,{default:()=>w7});async function w7($,Z){let Q=$.subcommand;if(!Q)throw new Error(`Missing amount.
|
|
171
|
+
`+o()+`
|
|
172
|
+
`),process.stderr.write(V(" bytesWritten is typically the tightest constraint")+`
|
|
173
|
+
`),process.stderr.write(V(" for large contract deployments.")+`
|
|
174
|
+
|
|
175
|
+
`)}var sQ;var L8=N(()=>{K0();sQ={readTime:"picoseconds",computeTime:"picoseconds",blockUsage:"bytes",bytesWritten:"bytes",bytesChurned:"bytes"}});var E$={};m(E$,{default:()=>CZ});import*as xZ from"@midnight-ntwrk/ledger-v8";import{MidnightBech32m as tQ}from"@midnight-ntwrk/wallet-sdk-address-format";async function CZ($,Z){Y4($);let Q=$.subcommand;if(!Q)throw new Error(`Missing amount.
|
|
131
176
|
Usage: midnight airdrop <amount>
|
|
132
|
-
Example: midnight airdrop 1000`);let X=
|
|
177
|
+
Example: midnight airdrop 1000`);let X=y1(Q),z=Y0(z0(D($,"wallet"))),{name:Y,config:J}=V0({args:$});if(x0(J,{proofServer:D($,"proof-server"),node:D($,"node"),indexerWS:D($,"indexer-ws")}),Y!=="undeployed")throw new Error(`Airdrop is only available on the "undeployed" network (local devnet).
|
|
133
178
|
Current network: "${Y}"
|
|
134
|
-
On preprod/preview, use a faucet or transfer from another wallet.`);
|
|
135
|
-
`+
|
|
179
|
+
On preprod/preview, use a faucet or transfer from another wallet.`);if(x($,"shielded"))return $X($,z,X,Y,J,Z);return eQ($,z,X,Y,J,Z)}async function eQ($,Z,Q,X,z,Y){if(D0($))b0();let J=Z.addresses[X],G=Buffer.from(J4,"hex");process.stderr.write(`
|
|
180
|
+
`+g("Airdrop")+`
|
|
136
181
|
|
|
137
|
-
`),process.stderr.write(
|
|
138
|
-
`),process.stderr.write(
|
|
139
|
-
`),process.stderr.write(
|
|
140
|
-
`),process.stderr.write(
|
|
182
|
+
`),process.stderr.write(A("Network",X)+`
|
|
183
|
+
`),process.stderr.write(A("From",V("genesis (seed 0x01)"))+`
|
|
184
|
+
`),process.stderr.write(A("To",$0(J,!0))+`
|
|
185
|
+
`),process.stderr.write(A("Amount",M(Q+" NIGHT"))+`
|
|
141
186
|
`),process.stderr.write(`
|
|
142
|
-
`);let
|
|
187
|
+
`);let q=l("Starting genesis wallet...");try{let K=await $1({seedBuffer:G,networkConfig:z,recipientAddress:J,amountNight:Q,signal:Y,onSync(B,H){if(H>0){let U=Math.round(B/H*100);q.update(`Syncing genesis wallet... ${U}%`)}},onDust(B){q.update(`Dust: ${B}`)},onProving(){q.update("Generating ZK proof (this may take a few minutes)...")},onSubmitting(){q.update("Submitting and waiting for finalization (typically 12 to 30s)...")},onSubmittingTick(B){let H=Math.floor(B/1000),U=Math.floor(H/60).toString().padStart(2,"0"),j=(H%60).toString().padStart(2,"0");q.update(`Submitting and waiting for finalization... ${U}:${j} elapsed (typically 12 to 30s)`)},onSyncWarning(B,H){q.update(`Syncing genesis wallet... (${H}, retrying)`)}});if(q.stop("Transaction submitted"),x($,"json")){y({txHash:K.txHash,amount:Q,recipient:J,network:X});return}process.stdout.write(K.txHash+`
|
|
143
188
|
`),process.stderr.write(`
|
|
144
|
-
`+
|
|
189
|
+
`+K4(`Airdropped ${Q} NIGHT to your wallet`,K.txHash)+`
|
|
145
190
|
`),process.stderr.write(`
|
|
146
|
-
`+
|
|
147
|
-
`),process.stderr.write(
|
|
148
|
-
`),process.stderr.write(
|
|
149
|
-
`),process.stderr.write(
|
|
150
|
-
`),process.stderr.write(
|
|
191
|
+
`+o()+`
|
|
192
|
+
`),process.stderr.write(V(" Verify: midnight balance")+`
|
|
193
|
+
`),process.stderr.write(V(" Register dust: midnight dust register")+`
|
|
194
|
+
`),process.stderr.write(V(" Note: Dust generation takes a few minutes on a fresh wallet.")+`
|
|
195
|
+
`),process.stderr.write(V(" It will happen automatically on your first transfer.")+`
|
|
151
196
|
|
|
152
|
-
`)}catch(
|
|
197
|
+
`)}catch(K){if(q.fail("Failed"),K instanceof Error&&K.message.toLowerCase().includes("dust"))throw new Error(`${K.message}
|
|
153
198
|
|
|
154
199
|
On a fresh localnet, the minimum airdrop is ~1 NIGHT.
|
|
155
|
-
Try: midnight airdrop 1`);throw
|
|
200
|
+
Try: midnight airdrop 1`);throw K}}async function $X($,Z,Q,X,z,Y){let J=v1(Q);if(D0($))b0();let G=Buffer.from(Z.seed,"hex"),q=Buffer.from(J4,"hex"),K=n4(z.networkId),B=xZ.unshieldedToken().raw,H=H$(G),U=tQ.encode(K,H).asString();M4(z0(D($,"wallet")),X,U),process.stderr.write(`
|
|
201
|
+
`+g("Shielded Airdrop")+`
|
|
202
|
+
|
|
203
|
+
`),process.stderr.write(A("Network",X)+`
|
|
204
|
+
`),process.stderr.write(A("From",V("genesis shielded (seed 0x01)"))+`
|
|
205
|
+
`),process.stderr.write(A("To",$0(U,!0))+`
|
|
206
|
+
`),process.stderr.write(A("Amount",M(Q+" NIGHT (shielded)"))+`
|
|
207
|
+
`),process.stderr.write(`
|
|
208
|
+
`);let j=U0(),L=P0(),W=l("Syncing genesis wallet...");try{let P=await j0().withFacade(q,z,async({bundle:O,state:F})=>{let I=F.shielded.balances[B]??0n;if(I<J){let k=Number(I)/1e6;throw new Error(`Genesis wallet has insufficient shielded balance: ${k.toFixed(6)} NIGHT available, ${Q} NIGHT requested`)}if(W.update("Checking dust..."),await s0(O,(k)=>W.update(k)),Y?.aborted)throw new Error("Operation cancelled");W.update("Building shielded transaction...");let _=await O.facade.transferTransaction([{type:"shielded",outputs:[{type:B,amount:J,receiverAddress:H}]}],{shieldedSecretKeys:O.zswapSecretKeys,dustSecretKey:O.dustSecretKey},{ttl:new Date(Date.now()+3600000)});W.update("Signing...");let v=await O.facade.signRecipe(_,(k)=>O.keystore.signData(k));W.update("Generating ZK proof (this may take a few minutes)...");let b=await O.facade.finalizeRecipe(v),c=await N$(W,"Submitting and waiting for finalization (typically 12 to 30s)...",O.facade.submitTransaction(b));return String(c)},{syncMode:"full",requireStrictSync:!0,signal:Y,onStatus:(O)=>W.update(O),onSyncProgress:(O,F)=>{if(F>0){let I=Math.min(Math.round(O/F*100),100);W.update(I>=100?"Syncing genesis wallet...":`Syncing genesis wallet... ${I}%`)}},onSyncDetail:(O)=>W.update(`Syncing genesis wallet... (waiting on: ${O})`)});if(W.stop("Transaction submitted"),x($,"json")){y({txHash:P,amount:Q,shieldedAddress:U,network:X,type:"shielded"});return}process.stdout.write(P+`
|
|
209
|
+
`),process.stderr.write(`
|
|
210
|
+
`+K4(`Airdropped ${Q} shielded NIGHT to your wallet`,P)+`
|
|
211
|
+
`),process.stderr.write(`
|
|
212
|
+
`+o()+`
|
|
213
|
+
`),process.stderr.write(V(" Verify: midnight balance --shielded")+`
|
|
214
|
+
|
|
215
|
+
`)}catch(P){throw W.fail("Failed"),P}finally{L(),j()}}var x$=N(()=>{B4();F0();H0();C0();L$();i0();O4();o0();L4();K0();u0()});var O8={};m(O8,{default:()=>bZ});import*as yZ from"@midnight-ntwrk/ledger-v8";import{MidnightBech32m as vZ,ShieldedAddress as ZX}from"@midnight-ntwrk/wallet-sdk-address-format";async function bZ($,Z){Y4($);let Q=$.subcommand,X=$.positionals[0];if(!Q)throw new Error(`Missing recipient address.
|
|
156
216
|
Usage: midnight transfer <to> <amount>
|
|
157
|
-
Example: midnight transfer mn_addr_undeployed1... 100
|
|
217
|
+
Example: midnight transfer mn_addr_undeployed1... 100
|
|
218
|
+
Example: midnight transfer alice 100`);if(!X)throw new Error(`Missing amount.
|
|
158
219
|
Usage: midnight transfer <to> <amount>
|
|
159
|
-
Example: midnight transfer mn_addr_undeployed1... 100`);let z=
|
|
160
|
-
|
|
220
|
+
Example: midnight transfer mn_addr_undeployed1... 100`);let z=x($,"shielded"),Y=QX(Q,$,z);if(z)return zX($,Y,X,Z);return XX($,Y,X,Z)}function QX($,Z,Q){if($.startsWith("mn_addr_")||$.startsWith("mn_shield-addr_"))return $;let X=z0($),z=Y0(X),{name:Y}=V0({args:Z});if(Q){let J=z.shieldedAddresses?.[Y];if(!J)throw new Error(`Wallet "${$}" has no shielded address for network "${Y}".
|
|
221
|
+
Regenerate the wallet or run "midnight balance --shielded" first.`);return J}return z.addresses[Y]}async function XX($,Z,Q,X){let z=y1(Q),Y=z0(D($,"wallet")),J=Y0(Y),G=Buffer.from(J.seed,"hex"),{name:q,config:K}=V0({args:$}),B=J.addresses[q];if(x0(K,{proofServer:D($,"proof-server"),node:D($,"node"),indexerWS:D($,"indexer-ws")}),process.stderr.write(`
|
|
222
|
+
`+g("Transfer")+`
|
|
223
|
+
|
|
224
|
+
`),process.stderr.write(A("Network",q)+`
|
|
225
|
+
`),process.stderr.write(A("From",$0(B,!0))+`
|
|
226
|
+
`),process.stderr.write(A("To",$0(Z,!0))+`
|
|
227
|
+
`),process.stderr.write(A("Amount",M(z+" NIGHT"))+`
|
|
228
|
+
`),process.stderr.write(`
|
|
229
|
+
`),D0($))b0();let H=l("Starting wallet...");try{let U=await $1({seedBuffer:G,networkConfig:K,recipientAddress:Z,amountNight:z,signal:X,onSync(j,L){if(L>0){let W=Math.min(Math.round(j/L*100),100);H.update(W>=100?"Syncing wallet...":`Syncing wallet... ${W}%`)}},onSyncDetail(j){H.update(`Syncing wallet... (waiting on: ${j})`)},onDust(j){H.update(`Dust: ${j}`)},onProving(){H.update("Generating ZK proof (this may take a few minutes)...")},onSubmitting(){H.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"),P=(L%60).toString().padStart(2,"0");H.update(`Submitting and waiting for finalization... ${W}:${P} elapsed (typically 12 to 30s)`)},onSyncWarning(j,L){H.update(`Syncing wallet... (${L}, retrying)`)}});if(H.stop("Transaction submitted"),x($,"json")){y({txHash:U.txHash,amount:z,recipient:Z,network:q});return}process.stdout.write(U.txHash+`
|
|
230
|
+
`),process.stderr.write(`
|
|
231
|
+
`+K4(`Transferred ${z} NIGHT`,U.txHash)+`
|
|
232
|
+
`),process.stderr.write(`
|
|
233
|
+
`+o()+`
|
|
234
|
+
`),process.stderr.write(V(" Verify: midnight balance")+`
|
|
161
235
|
|
|
162
|
-
`),
|
|
163
|
-
`)
|
|
164
|
-
|
|
165
|
-
|
|
236
|
+
`)}catch(U){throw H.fail("Failed"),U}}async function zX($,Z,Q,X){let z=y1(Q),Y=v1(z),J=z0(D($,"wallet")),G=Y0(J),q=Buffer.from(G.seed,"hex"),{name:K,config:B}=V0({args:$}),H=G.addresses[K],U=n4(B.networkId),j=yZ.unshieldedToken().raw;x0(B,{proofServer:D($,"proof-server"),node:D($,"node"),indexerWS:D($,"indexer-ws")});let L;try{L=vZ.parse(Z).decode(ZX,U)}catch(F){throw new Error(`Invalid shielded address: ${F.message}
|
|
237
|
+
Expected a shielded address (mn_shield-addr_...) for network "${B.networkId}"`)}if(process.stderr.write(`
|
|
238
|
+
`+g("Shielded Transfer")+`
|
|
239
|
+
|
|
240
|
+
`),process.stderr.write(A("Network",K)+`
|
|
241
|
+
`),process.stderr.write(A("To",$0(Z,!0))+`
|
|
242
|
+
`),process.stderr.write(A("Amount",M(z+" NIGHT (shielded)"))+`
|
|
166
243
|
`),process.stderr.write(`
|
|
167
|
-
`)
|
|
244
|
+
`),D0($))b0();let W=U0(),P=P0(),O=l("Syncing wallet...");try{let F=await j0().withFacade(q,B,async({bundle:I,state:_})=>{let v=vZ.encode(U,_.shielded.address).asString();M4(J,K,v);let b=_.shielded.balances[j]??0n;if(b<Y){let i=Number(b)/1e6;throw new Error(`Insufficient shielded balance: ${i.toFixed(6)} NIGHT available, ${z} NIGHT requested.
|
|
245
|
+
Fund shielded balance: midnight airdrop ${z} --shielded`)}if(O.update("Checking dust..."),await s0(I,(i)=>O.update(i)),X?.aborted)throw new Error("Operation cancelled");O.update("Building shielded transaction...");let c=await I.facade.transferTransaction([{type:"shielded",outputs:[{type:j,amount:Y,receiverAddress:L}]}],{shieldedSecretKeys:I.zswapSecretKeys,dustSecretKey:I.dustSecretKey},{ttl:new Date(Date.now()+3600000)});O.update("Signing...");let k=await I.facade.signRecipe(c,(i)=>I.keystore.signData(i));O.update("Generating ZK proof (this may take a few minutes)...");let J0=await I.facade.finalizeRecipe(k),w=await N$(O,"Submitting and waiting for finalization (typically 12 to 30s)...",I.facade.submitTransaction(J0));return String(w)},{syncMode:"full",requireStrictSync:!0,signal:X,onStatus:(I)=>O.update(I),onSyncProgress:(I,_)=>{if(_>0){let v=Math.min(Math.round(I/_*100),100);O.update(v>=100?"Syncing wallet...":`Syncing wallet... ${v}%`)}},onSyncDetail:(I)=>O.update(`Syncing wallet... (waiting on: ${I})`)});if(O.stop("Transaction submitted"),x($,"json")){y({txHash:F,amount:z,recipient:Z,network:K,type:"shielded"});return}process.stdout.write(F+`
|
|
168
246
|
`),process.stderr.write(`
|
|
169
|
-
`+
|
|
247
|
+
`+K4(`Transferred ${z} shielded NIGHT`,F)+`
|
|
170
248
|
`),process.stderr.write(`
|
|
171
|
-
`+
|
|
172
|
-
`),process.stderr.write(
|
|
249
|
+
`+o()+`
|
|
250
|
+
`),process.stderr.write(V(" Verify: midnight balance --shielded")+`
|
|
173
251
|
|
|
174
|
-
`)}catch(
|
|
252
|
+
`)}catch(F){throw O.fail("Failed"),F}finally{P(),W()}}var j8=N(()=>{B4();F0();H0();C0();L$();O4();o0();L4();K0();u0()});var C$={};m(C$,{default:()=>SZ});async function SZ($,Z){let Q=$.subcommand;if(!Q||Q!=="register"&&Q!=="status")throw new t(`Missing or invalid subcommand.
|
|
175
253
|
Usage:
|
|
176
254
|
midnight dust register Register NIGHT UTXOs for dust generation
|
|
177
|
-
midnight dust status Check dust registration status`);let X=
|
|
178
|
-
`+
|
|
179
|
-
|
|
180
|
-
`),process.stderr.write(
|
|
181
|
-
|
|
182
|
-
`);let
|
|
183
|
-
`),
|
|
184
|
-
|
|
185
|
-
|
|
255
|
+
midnight dust status Check dust registration status`);let X=z0(D($,"wallet")),z=Y0(X),Y=Buffer.from(z.seed,"hex"),{name:J,config:G}=V0({args:$}),q=z.addresses[J];if(x0(G,{proofServer:D($,"proof-server"),node:D($,"node"),indexerWS:D($,"indexer-ws")}),D0($))b0();let K=x($,"json"),B=z4($);if(Q==="status"){let L=await YX(q,J,G.indexerWS,Z);if(!L.isRegistered){JX(J,L,K,B);return}let W=!x($,"no-cache");await qX(Y,G,L,K,B,W,Z);return}Y4($);let H={},U=U0((L,W)=>H.current?.(L,W)),j=P0();try{await GX(Y,G,K,Z,H)}finally{j(),U()}}async function YX($,Z,Q,X){process.stderr.write(`
|
|
256
|
+
`+g("Dust Status")+`
|
|
257
|
+
|
|
258
|
+
`),process.stderr.write(A("Network",Z)+`
|
|
259
|
+
|
|
260
|
+
`);let z=l(`Checking registration on ${Z}...`);try{let Y=await o4($,Q,(q,K)=>{if(K>0){let B=Math.min(Math.round(q/K*100),100);z.update(B>=100?"Checking registration...":`Checking registration... ${B}%`)}});if(X?.aborted)throw new Error("Operation cancelled");let{registeredUtxos:J,unregisteredUtxos:G}=Y;return z.stop("Registration checked"),{isRegistered:J>0,registeredUtxos:J,unregisteredUtxos:G}}catch(Y){throw z.fail("Failed"),Y}}function JX($,Z,Q,X){let{registeredUtxos:z,unregisteredUtxos:Y}=Z,J=z+Y;if(Q){if(X){y({network:$,registered:!1,registeredUtxos:z,unregisteredUtxos:Y});return}y({subcommand:"status",registered:!1,registeredUtxos:z,unregisteredUtxos:Y,network:$});return}if(process.stdout.write(`registered=no
|
|
261
|
+
`),process.stdout.write(`registered_utxos=${z}
|
|
262
|
+
`),process.stdout.write(`unregistered_utxos=${Y}
|
|
263
|
+
`),process.stderr.write(A("Registered",M("no"))+`
|
|
264
|
+
`),process.stderr.write(A("UTXOs",`${z} registered, ${Y} unregistered`)+`
|
|
265
|
+
`),J===0)process.stderr.write(`
|
|
266
|
+
`+V("No NIGHT UTXOs at this address. Fund the wallet first.")+`
|
|
186
267
|
`);else process.stderr.write(`
|
|
187
|
-
`+
|
|
268
|
+
`+V("Not generating dust. Run: midnight dust register")+`
|
|
269
|
+
`);process.stderr.write(`
|
|
270
|
+
`+o()+`
|
|
188
271
|
|
|
189
|
-
`)}
|
|
190
|
-
`+
|
|
272
|
+
`)}async function GX($,Z,Q,X,z){let Y=Z.networkId.toLowerCase();process.stderr.write(`
|
|
273
|
+
`+g("Dust Register")+`
|
|
191
274
|
|
|
192
|
-
`),process.stderr.write(
|
|
275
|
+
`),process.stderr.write(A("Network",Y)+`
|
|
193
276
|
|
|
194
|
-
`);let
|
|
195
|
-
`),process.stdout.write(`registered=${V}
|
|
196
|
-
`),process.stdout.write(`unregistered=${j.length}
|
|
197
|
-
`),process.stderr.write(T("NIGHT Balance",y(x1(H)))+`
|
|
198
|
-
`),process.stderr.write(T("Dust Balance",y(b1(K)))+`
|
|
199
|
-
`),process.stderr.write(T("Dust Available",U?"yes":"no")+`
|
|
200
|
-
`),process.stderr.write(T("Registered",V.toString()+" UTXO(s)")+`
|
|
201
|
-
`),process.stderr.write(T("Unregistered",j.length.toString()+" UTXO(s)")+`
|
|
277
|
+
`);let J=l("Syncing wallet...");if(z)z.current=(G,q)=>J.update(`Syncing wallet... (${q}, retrying)`);try{let{result:G,dustBal:q}=await j0().withFacade($,Z,async({bundle:K,state:B})=>{if(X?.aborted)throw new Error("Operation cancelled");J.update("Checking dust status...");let H=await s0(K,(j)=>J.update(j),B);if(X?.aborted)throw new Error("Operation cancelled");let U=await HZ(K);return{result:H,dustBal:U.dust.balance(new Date)}},{syncMode:"lite",requireStrictSync:!0,signal:X,onStatus:(K)=>J.update(K),onSyncProgress:(K,B)=>{if(B>0){let H=Math.min(Math.round(K/B*100),100);J.update(H>=100?"Syncing wallet...":`Syncing wallet... ${H}%`)}},onSyncDetail:(K)=>J.update(`Syncing wallet... (waiting on: ${K})`)});if(J.stop(G.alreadyAvailable?"Dust already available":"Dust registration complete"),Q){let K={subcommand:"register",dustBalance:W1(q)};if(G.txHash)K.txHash=G.txHash;y(K);return}process.stdout.write(q.toString()+`
|
|
202
278
|
`),process.stderr.write(`
|
|
203
|
-
`+
|
|
204
|
-
|
|
205
|
-
`)}catch(
|
|
206
|
-
`)
|
|
207
|
-
`)
|
|
208
|
-
`)
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
`)
|
|
212
|
-
`)
|
|
213
|
-
|
|
214
|
-
`)
|
|
279
|
+
`+K4(G.alreadyAvailable?`Dust tokens already available: ${X$(q)}`:`Dust tokens available: ${X$(q)}`)+`
|
|
280
|
+
|
|
281
|
+
`)}catch(G){throw J.fail("Failed"),G}}async function qX($,Z,Q,X,z,Y,J){let G=Z.networkId.toLowerCase(),q=l(`Reading dust events from ${G}...`);try{let K=await j0().dust($,Z,{forceFresh:!Y,signal:J,onStatus:(U)=>q.update(U)});q.stop(K.fromCache&&K.eventsApplied===0?"Cache up to date":"Dust events applied");let{registeredUtxos:B,unregisteredUtxos:H}=Q;if(X){if(z){y({network:G,registered:!0,registeredUtxos:B,unregisteredUtxos:H,dustBalance:W1(K.balance),dustAvailable:K.availableCoins>0});return}y({subcommand:"status",registered:!0,registeredUtxos:B,unregisteredUtxos:H,dustBalance:W1(K.balance),dustAvailable:K.availableCoins>0,eventsApplied:K.eventsApplied,ownedUtxos:K.ownedUtxoCount,cached:K.fromCache,network:G});return}if(process.stdout.write(`registered=yes
|
|
282
|
+
`),process.stdout.write(`dust=${K.balance}
|
|
283
|
+
`),process.stdout.write(`registered_utxos=${B}
|
|
284
|
+
`),process.stdout.write(`unregistered_utxos=${H}
|
|
285
|
+
`),process.stderr.write(A("Registered",M("yes"))+`
|
|
286
|
+
`),process.stderr.write(A("UTXOs",`${B} registered, ${H} unregistered`)+`
|
|
287
|
+
`),process.stderr.write(A("Dust Balance",M(X$(K.balance)))+`
|
|
288
|
+
`),process.stderr.write(A("Dust Available",K.availableCoins>0?"yes":"no")+`
|
|
289
|
+
`),process.stderr.write(A("Events applied",K.eventsApplied.toString()+(K.fromCache?" (delta)":""))+`
|
|
290
|
+
`),H>0)process.stderr.write(`
|
|
291
|
+
`+V(`${H} UTXO(s) not yet registered. Run: midnight dust register`)+`
|
|
292
|
+
`);process.stderr.write(`
|
|
293
|
+
`+o()+`
|
|
294
|
+
|
|
295
|
+
`)}catch(K){throw q.fail("Failed"),K}}var v$=N(()=>{l0();B4();F0();H0();C0();o0();O4();T1();L4();K0();u0()});var P8={};m(P8,{default:()=>kZ});async function kZ($){if($.subcommand!=="clear")throw new t("Usage: midnight cache clear [--network <name>] [--wallet <name|file>]");let Q=D($,"wallet"),X=D($,"network"),z=x($,"json");if(Q){let Y=z0(Q),J=Y0(Y),{name:G}=V0({args:$}),q=J.addresses[G];r0(q,G);let K=A4(Buffer.from(J.seed,"hex"));if(N4(G,K),z){y({action:"clear",scope:"wallet",wallet:Q,network:G});return}process.stderr.write(p("✓")+` Cache cleared for wallet "${Q}" on ${G}
|
|
296
|
+
`)}else if(X){if(r0(void 0,X),N4(X),z){y({action:"clear",scope:"network",network:X});return}process.stderr.write(p("✓")+` Cache cleared for network "${X}"
|
|
297
|
+
`)}else{if(r0(),N4(),z){y({action:"clear",scope:"all"});return}process.stderr.write(p("✓")+` Cache cleared
|
|
298
|
+
`)}}var F8=N(()=>{l0();U4();D1();F0();H0()});var I8={};m(I8,{default:()=>wZ});async function wZ($){let Z=$.subcommand;if(!Z||!["get","set","unset"].includes(Z))throw new t(`Usage: midnight config <get|set|unset> <key> [value]
|
|
299
|
+
Valid keys: ${V$().join(", ")}`);let Q=$.positionals[0];if(!Q)throw new t(`Missing config key.
|
|
300
|
+
Valid keys: ${V$().join(", ")}`);if(Z==="get"){let X=C2(Q);if(x($,"json")){y({action:"get",key:Q,value:X});return}process.stdout.write(X+`
|
|
301
|
+
`)}else if(Z==="unset"){y2(Q);let X=Q==="network"?"(default)":"(removed)";if(x($,"json")){y({action:"unset",key:Q,value:X});return}process.stderr.write(p("✓")+` ${Q} ${X}
|
|
302
|
+
`)}else{let X=$.positionals[1];if(X===void 0)throw new t(`Missing value for config set.
|
|
303
|
+
Usage: midnight config set ${Q} <value>`);if(v2(Q,X),x($,"json")){y({action:"set",key:Q,value:X});return}process.stderr.write(p("✓")+` ${Q} = ${X}
|
|
304
|
+
`)}}var _8=N(()=>{l0();p4()});import{execSync as S1}from"child_process";import{existsSync as fZ,mkdirSync as KX,readFileSync as VX,writeFileSync as hZ}from"fs";import{homedir as BX}from"os";import{join as M8}from"path";function S$(){try{return S1("docker compose version",{...b$,timeout:1e4}).trim()}catch{try{throw S1("docker --version",{...b$,timeout:5000}),new Error(`Docker Compose v2 is required.
|
|
215
305
|
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.
|
|
216
|
-
Install Docker from https://docs.docker.com/get-docker/`)}}}function
|
|
217
|
-
`),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:
|
|
306
|
+
Install Docker from https://docs.docker.com/get-docker/`)}}}function Z1(){if(fZ(T8)&&fZ(y$)&&VX(T8,"utf-8").trim()===mZ)return!1;return KX(R8,{recursive:!0,mode:N0}),hZ(y$,LX,"utf-8"),hZ(T8,mZ,"utf-8"),!0}function p0($){return S1(`docker compose -f "${y$}" ${$}`,b$)}function Q4(){try{let $=p0("ps --format json");if(!$.trim())return[];let Z=$.trim().split(`
|
|
307
|
+
`),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:OX[Y]??"",image:z.Image??""})}catch{}}return Q}catch{return[]}}function Q1($=120000,Z=3000){let Q=Date.now()+$;while(Date.now()<Q){let X=Q4();if(X.length===3&&X.every((Y)=>Y.state==="running")){if(X.every((J)=>J.health==="healthy"||J.health===""))return!0}S1(`sleep ${Z/1000}`,{timeout:Z+1000})}return!1}function A8(){return y$}function gZ(){let $=[];for(let Z of D8)try{S1(`docker rm -f "${Z}"`,{...b$,timeout:1e4}),$.push(Z)}catch{}return $}var mZ="3.0.1",HX="midnightntwrk/midnight-node:0.22.3",WX="midnightntwrk/indexer-standalone:4.0.1",UX="midnightntwrk/proof-server:8.0.3",R8,y$,T8,LX,b$,OX,D8;var k$=N(()=>{R8=M8(BX(),B0,_2),y$=M8(R8,"compose.yml"),T8=M8(R8,".version"),LX=`services:
|
|
218
308
|
proof-server:
|
|
219
|
-
image: '
|
|
309
|
+
image: '${UX}'
|
|
220
310
|
container_name: "proof-server"
|
|
221
311
|
ports:
|
|
222
312
|
- "6300:6300"
|
|
223
313
|
|
|
224
314
|
indexer:
|
|
225
|
-
image: '
|
|
315
|
+
image: '${WX}'
|
|
226
316
|
container_name: "indexer"
|
|
227
317
|
ports:
|
|
228
318
|
- '8088:8088'
|
|
@@ -244,7 +334,7 @@ Install Docker from https://docs.docker.com/get-docker/`)}}}function i7(){if(g7(
|
|
|
244
334
|
condition: service_started
|
|
245
335
|
|
|
246
336
|
node:
|
|
247
|
-
image: '
|
|
337
|
+
image: '${HX}'
|
|
248
338
|
container_name: "node"
|
|
249
339
|
ports:
|
|
250
340
|
- "9944:9944"
|
|
@@ -256,34 +346,38 @@ Install Docker from https://docs.docker.com/get-docker/`)}}}function i7(){if(g7(
|
|
|
256
346
|
start_period: 5s
|
|
257
347
|
environment:
|
|
258
348
|
CFG_PRESET: "dev"
|
|
259
|
-
`,
|
|
260
|
-
`)}async function
|
|
261
|
-
`);let Q=
|
|
262
|
-
`+
|
|
263
|
-
`);else
|
|
349
|
+
`,b$={encoding:"utf-8",timeout:30000};OX={node:"9944",indexer:"8088","proof-server":"6300"};D8=["node","indexer","proof-server"]});var uZ={};m(uZ,{waitForFirstBlock:()=>jX});async function jX($,Z={}){let Q=Z.timeoutMs??30000,X=Z.pollIntervalMs??1000,z=Date.now()+Q;while(Date.now()<z){try{let Y=await r4({url:$,timeoutMs:3000},"chain_getHeader"),J=typeof Y?.number==="string"?Y.number:"0x0",G=Number.parseInt(J,16);if(Number.isFinite(G)&&G>=1)return}catch{}await new Promise((Y)=>setTimeout(Y,X))}throw new Error(`Node did not produce a block within ${Q/1000}s`)}var pZ=N(()=>{j$()});var cZ={};m(cZ,{waitForAddressFunded:()=>PX});async function PX($,Z,Q={}){let X=Q.timeoutMs??30000,z=Q.pollIntervalMs??1000,Y=Date.now()+X;while(Date.now()<Y){try{if((await o4(Z,$)).utxoCount>0)return}catch{}await new Promise((J)=>setTimeout(J,z))}throw new Error(`Indexer did not report funds for ${Z.slice(0,20)}… within ${X/1000}s`)}var dZ=N(()=>{T1()});var lZ={};m(lZ,{waitForProofServerReady:()=>FX});async function FX($,Z={}){let Q=Z.timeoutMs??30000,X=Z.pollIntervalMs??1000,z=Z.requestTimeoutMs??2000,Y=Date.now()+Q,J=$.endsWith("/")?$:$+"/";while(Date.now()<Y){try{let G=new AbortController,q=setTimeout(()=>G.abort(),z);try{if((await fetch(J,{method:"GET",signal:G.signal})).ok)return}finally{clearTimeout(q)}}catch{}await new Promise((G)=>setTimeout(G,X))}throw new Error(`Proof server at ${J} did not respond within ${Q/1000}s`)}var N8={};m(N8,{default:()=>aZ});import{spawn as iZ}from"child_process";import{existsSync as IX,rmSync as _X}from"node:fs";import{resolve as TX}from"node:path";function MX($){return nZ.includes($)}function oZ($){let Z=[];for(let Q of $){let X=Q.state==="running"?p:n,z=Q.health?` (${Q.health})`:"",Y=Q.port?`:${Q.port}`:"",J=Q.image?` ${V(Q.image)}`:"";if(Z.push(` ${Q.name.padEnd(16)}${X(Q.state)}${V(z)}${V(Y)}`),J)Z.push(J)}return Z.join(`
|
|
350
|
+
`)}async function RX($){if(Z1())process.stderr.write(V(` Wrote compose.yml to ${A8()}`)+`
|
|
351
|
+
`);let Q=l("Starting local network...");try{if(p0("up -d"),Q.update("Waiting for services to be healthy..."),!Q1(120000))Q.stop(S("Services started but not all healthy yet")),process.stderr.write(`
|
|
352
|
+
`+V(" Tip: run ")+M("midnight localnet logs")+V(" to check for errors")+`
|
|
353
|
+
`);else{Q.update("Waiting for chain to produce first block...");try{let{waitForFirstBlock:Y}=await Promise.resolve().then(() => (pZ(),uZ)),{waitForAddressFunded:J}=await Promise.resolve().then(() => (dZ(),cZ)),{waitForProofServerReady:G}=await Promise.resolve().then(() => lZ),{getNetworkConfig:q}=await Promise.resolve().then(() => (C0(),h7)),{deriveUnshieldedAddress:K}=await Promise.resolve().then(() => (i0(),l7)),{GENESIS_SEED:B}=await Promise.resolve().then(() => Z$),H=q("undeployed");await Y(H.node,{timeoutMs:30000}),Q.update("Waiting for indexer to ingest genesis state...");let U=K(Buffer.from(B,"hex"),"undeployed");await J(H.indexerWS,U,{timeoutMs:30000}),Q.update("Waiting for proof server..."),await G(H.proofServer,{timeoutMs:30000}),Q.stop("Local network is running")}catch(Y){Q.stop(S("Services running but chain/indexer/proof-server not fully ready yet")),process.stderr.write(`
|
|
354
|
+
`+V(" Tip: give it a few more seconds then retry. Run ")+M("midnight localnet logs")+V(" if this persists.")+`
|
|
355
|
+
`)}}}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
|
|
264
356
|
`+`(likely from a previous midnight-local-network setup).
|
|
265
357
|
|
|
266
358
|
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.
|
|
267
|
-
`+"Check ports 9944, 8088, and 6300, then try again.")}throw z}let X=
|
|
268
|
-
`+
|
|
359
|
+
`+"Check ports 9944, 8088, and 6300, then try again.")}throw z}let X=Q4();if($){y({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(`
|
|
360
|
+
`+oZ(X)+`
|
|
269
361
|
`);for(let z of X)process.stdout.write(`${z.name}=${z.state}:${z.port}
|
|
270
362
|
`);process.stderr.write(`
|
|
271
|
-
`+
|
|
272
|
-
`)}async function
|
|
273
|
-
|
|
363
|
+
`+V(" Next: ")+M("midnight wallet generate dev --network undeployed")+`
|
|
364
|
+
`)}async function AX($){let Z=l("Stopping local network...");try{if(p0("stop"),Z.stop("Local network stopped (containers preserved)"),$){y({subcommand:"stop",status:"stopped"});return}}catch(Q){throw Z.fail("Failed to stop local network"),Q}}async function DX($){let Z=l("Tearing down local network...");try{p0("down --volumes");let{clearWalletCache:Q}=await Promise.resolve().then(() => (U4(),UZ)),{clearDustDirectCache:X}=await Promise.resolve().then(() => (D1(),GZ));Q(void 0,"undeployed"),X("undeployed");let z=NX();if(Z.stop("Local network removed (containers, networks, volumes, caches)"),$){y({subcommand:"down",status:"removed",cachesCleared:!0,dappLevelDbWiped:z});return}if(z)process.stderr.write(V(" Also cleared dapp leveldb at ./midnight-level-db (private state from old chain)")+`
|
|
365
|
+
`)}catch(Q){throw Z.fail("Failed to tear down local network"),Q}}function NX(){let $=TX(process.cwd(),"midnight-level-db");if(!IX($))return!1;try{return _X($,{recursive:!0,force:!0}),!0}catch{return!1}}async function EX($){let Z=Q4();if($){y({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(`
|
|
366
|
+
`+g("Localnet Status")+`
|
|
274
367
|
|
|
275
|
-
`),process.stderr.write(
|
|
276
|
-
`),process.stderr.write(
|
|
368
|
+
`),process.stderr.write(V(" No services running.")+`
|
|
369
|
+
`),process.stderr.write(V(" Run ")+M("midnight localnet up")+V(" to start.")+`
|
|
277
370
|
|
|
278
371
|
`);return}process.stderr.write(`
|
|
279
|
-
`+
|
|
372
|
+
`+g("Localnet Status")+`
|
|
280
373
|
|
|
281
|
-
`),process.stderr.write(
|
|
374
|
+
`),process.stderr.write(oZ(Z)+`
|
|
282
375
|
`),process.stderr.write(`
|
|
283
|
-
`+
|
|
376
|
+
`+o()+`
|
|
284
377
|
|
|
285
378
|
`);for(let Q of Z)process.stdout.write(`${Q.name}=${Q.state}:${Q.port}
|
|
286
|
-
`)}async function
|
|
379
|
+
`)}async function xX($){let Z=l("Removing conflicting containers...");try{try{p0("down")}catch{}let Q=gZ();if(Q.length>0)Z.stop(`Removed ${Q.length} container${Q.length>1?"s":""}: ${Q.join(", ")}`);else Z.stop("No conflicting containers found");if($){y({subcommand:"clean",status:"cleaned",removed:Q});return}}catch(Q){throw Z.fail("Failed to clean up"),Q}}async function CX($){let Z=A8(),Q=D($,"tail"),X=x($,"json");if(Q!==void 0||X){let Y=Q??"200",J=iZ("docker",["compose","-f",Z,"logs","--tail",Y,"--no-color"],{stdio:["ignore","pipe","pipe"]}),G="",q="";return J.stdout.on("data",(K)=>{G+=K.toString()}),J.stderr.on("data",(K)=>{q+=K.toString()}),new Promise((K,B)=>{J.on("close",(H)=>{if(H!==0&&H!==null){B(new Error(`docker compose logs exited with code ${H}: ${q.trim()}`));return}if(X)y({tail:Number(Y),lines:G.split(`
|
|
380
|
+
`).filter(Boolean)});else process.stdout.write(G);K()}),J.on("error",B)})}let z=iZ("docker",["compose","-f",Z,"logs","-f"],{stdio:"inherit"});return new Promise((Y,J)=>{z.on("close",(G)=>{if(G===0||G===130||G===null)Y();else J(new Error(`docker compose logs exited with code ${G}`))}),z.on("error",J)})}async function aZ($){let Z=$.subcommand;if(!Z||!MX(Z))throw new t(`Usage: midnight localnet <${nZ.join("|")}>
|
|
287
381
|
|
|
288
382
|
Subcommands:
|
|
289
383
|
up Start the local network
|
|
@@ -293,131 +387,941 @@ Subcommands:
|
|
|
293
387
|
logs Stream service logs
|
|
294
388
|
clean Remove conflicting containers
|
|
295
389
|
|
|
296
|
-
Example: midnight localnet up`);
|
|
297
|
-
`+
|
|
390
|
+
Example: midnight localnet up`);S$();let Q=x($,"json");switch(process.stderr.write(`
|
|
391
|
+
`+g("Localnet")+`
|
|
298
392
|
|
|
299
|
-
`),Z){case"up":return
|
|
300
|
-
Available: generate, list, use, info, remove
|
|
301
|
-
Run "midnight help wallet" for usage.`)}}async function
|
|
302
|
-
Usage: midnight wallet generate <name> [--network <name>]`);if(!
|
|
303
|
-
Wallet name must be a simple name (no path separators, .json suffix, or special characters).`);let Q=
|
|
304
|
-
Use --force to overwrite.`);let G,q;if(X!==void 0){let
|
|
393
|
+
`),Z){case"up":return RX(Q);case"stop":return AX(Q);case"down":return DX(Q);case"status":return EX(Q);case"logs":return CX($);case"clean":return xX(Q)}}var nZ;var E8=N(()=>{l0();k$();K0();u0();nZ=["up","stop","down","status","logs","clean"]});var w$={};m(w$,{default:()=>tZ});import*as sZ from"fs";import*as x8 from"path";import{homedir as vX}from"os";import{generateMnemonic as yX,mnemonicToSeedSync as rZ,mnemonicToEntropy as bX,validateMnemonic as SX}from"@scure/bip39";import{wordlist as C8}from"@scure/bip39/wordlists/english.js";async function tZ($){let Z=$.subcommand;switch(Z){case"generate":return kX($);case"list":case"ls":return wX($);case"use":return fX($);case"info":return hX($);case"remove":case"rm":return mX($);case"seed":return gX($);default:throw new t(`Unknown wallet subcommand: "${Z??"(none)"}"
|
|
394
|
+
Available: generate, list, use, info, remove, seed
|
|
395
|
+
Run "midnight help wallet" for usage.`)}}async function kX($){let Z=$.positionals[0];if(!Z)throw new Error(`Missing wallet name.
|
|
396
|
+
Usage: midnight wallet generate <name> [--network <name>]`);if(!q4(Z))throw new Error(`Invalid wallet name: "${Z}"
|
|
397
|
+
Wallet name must be a simple name (no path separators, .json suffix, or special characters).`);let Q=T0({args:$}),X=D($,"seed"),z=D($,"mnemonic");if(X!==void 0&&z!==void 0)throw new Error("Cannot specify both --seed and --mnemonic. Use one or the other.");let Y=x8.join(vX(),B0,F4),J=x8.join(Y,`${Z}.json`);if(sZ.existsSync(J)&&!x($,"force"))throw new Error(`Wallet "${Z}" already exists: ${J}
|
|
398
|
+
Use --force to overwrite.`);let G,q;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)");G=Buffer.from(W,"hex")}else if(z!==void 0){if(!SX(z,C8))throw new Error("Invalid BIP-39 mnemonic. Expected 12 or 24 words from the English wordlist.");q=z,G=Buffer.from(rZ(q))}else q=yX(C8,256),G=Buffer.from(rZ(q));let K=V4(G),B=d4(G),H=K[Q],U={seed:G.toString("hex"),addresses:K,shieldedAddresses:B,createdAt:new Date().toISOString()};if(q)U.mnemonic=q;let j=I1(U,J);if(l4(Z),x($,"json")){let W={name:Z,addresses:K,shieldedAddresses:B,activeAddress:H,activeNetwork:Q,seed:G.toString("hex"),file:j,createdAt:U.createdAt,active:!0};if(q)W.mnemonic=q;y(W);return}process.stdout.write(H+`
|
|
305
399
|
`),process.stderr.write(`
|
|
306
|
-
`+
|
|
307
|
-
|
|
308
|
-
`),process.stderr.write(
|
|
309
|
-
`),process.stderr.write(
|
|
310
|
-
`),process.stderr.write(
|
|
311
|
-
`)
|
|
312
|
-
`)
|
|
400
|
+
`+g("Wallet Generated")+`
|
|
401
|
+
|
|
402
|
+
`),process.stderr.write(A("Name",Z)+`
|
|
403
|
+
`),process.stderr.write(A("Network",Q)+`
|
|
404
|
+
`),process.stderr.write(A("Address",$0(H))+`
|
|
405
|
+
`);let L=B[Q];if(L)process.stderr.write(A("Shielded",$0(L))+`
|
|
406
|
+
`);if(process.stderr.write(A("File",j)+`
|
|
407
|
+
`),process.stderr.write(A("Active","yes")+`
|
|
313
408
|
`),process.stderr.write(`
|
|
314
|
-
`),q)process.stderr.write(
|
|
409
|
+
`),q)process.stderr.write(S(M(" MNEMONIC (save securely!):"))+`
|
|
315
410
|
`),process.stderr.write(` ${q}
|
|
316
411
|
|
|
317
|
-
`);process.stderr.write(
|
|
412
|
+
`);process.stderr.write(S(M(" SEED (hex):"))+`
|
|
318
413
|
`),process.stderr.write(` ${G.toString("hex")}
|
|
319
414
|
|
|
320
|
-
`),process.stderr.write(
|
|
321
|
-
`),process.stderr.write(
|
|
415
|
+
`),process.stderr.write(o()+`
|
|
416
|
+
`),process.stderr.write(V(" Next: midnight wallet list | midnight balance")+`
|
|
322
417
|
|
|
323
|
-
`),process.stderr.write(
|
|
324
|
-
`)}async function
|
|
418
|
+
`),process.stderr.write(p("✓")+` Wallet saved
|
|
419
|
+
`)}async function wX($){let Z=h2(),Q=T0({args:$});if(x($,"json")){if(z4($)){y({activeNetwork:Q,wallets:Z.map((G)=>({name:G.name,active:G.isActive,network:Q,address:G.addresses[Q]??null,shieldedAddress:G.shieldedAddresses?.[Q]??null}))});return}y({activeNetwork:Q,wallets:Z.map((G)=>({name:G.name,active:G.isActive,addresses:G.addresses,shieldedAddresses:G.shieldedAddresses,createdAt:G.createdAt,file:G.file}))});return}if(Z.length===0){process.stderr.write(`
|
|
325
420
|
No wallets found.
|
|
326
|
-
`),process.stderr.write(
|
|
421
|
+
`),process.stderr.write(V(" Create one: midnight wallet generate <name> --network <name>")+`
|
|
327
422
|
|
|
328
|
-
`);return}
|
|
329
|
-
`+
|
|
423
|
+
`);return}process.stderr.write(`
|
|
424
|
+
`+g("Wallets")+`
|
|
330
425
|
|
|
331
|
-
`);
|
|
426
|
+
`);let X=Math.max(...Z.map((G)=>G.name.length)),z=19,Y=19,J=" ".repeat(X-4);process.stderr.write(` ${V("name")}${J} ${V("unshielded".padEnd(z))} ${V("shielded")}
|
|
427
|
+
`);for(let G of Z){let q=G.isActive?p("●"):" ",K=G.isActive?M(f(G.name)):f(G.name),B=" ".repeat(X-G.name.length),H=G.addresses[Q]??"(unknown)",U=G.shieldedAddresses?.[Q],j=U?$0(U,!0):V("—".padEnd(Y));process.stderr.write(` ${q} ${K}${B} ${$0(H,!0)} ${j}
|
|
332
428
|
`)}process.stderr.write(`
|
|
333
|
-
`+
|
|
334
|
-
`),process.stderr.write(
|
|
429
|
+
`+o()+`
|
|
430
|
+
`),process.stderr.write(V(` ● = active wallet · network: ${Q} · details: midnight wallet info <name>`)+`
|
|
335
431
|
|
|
336
|
-
`)}async function
|
|
337
|
-
Usage: midnight wallet use <name>`);if(
|
|
338
|
-
`)}async function
|
|
432
|
+
`)}async function fX($){let Z=$.positionals[0];if(!Z)throw new Error(`Missing wallet name.
|
|
433
|
+
Usage: midnight wallet use <name>`);if(l4(Z),x($,"json")){y({wallet:Z,active:!0});return}process.stderr.write(p("✓")+` Active wallet set to "${Z}"
|
|
434
|
+
`)}async function hX($){let Z=$.positionals[0]??$4(),Q=z0(Z),X=Y0(Q),z=Z===$4(),Y=T0({args:$}),J=X.addresses[Y];if(x($,"json")){if(z4($)){y({name:Z,active:z,network:Y,address:J,shieldedAddress:X.shieldedAddresses?.[Y]??null});return}let q={name:Z,addresses:X.addresses,activeAddress:J,activeNetwork:Y,createdAt:X.createdAt,file:Q,active:z};if(X.shieldedAddresses)q.shieldedAddresses=X.shieldedAddresses;y(q);return}process.stdout.write(J+`
|
|
339
435
|
`),process.stderr.write(`
|
|
340
|
-
`+
|
|
341
|
-
|
|
342
|
-
`),
|
|
343
|
-
`)
|
|
344
|
-
`)
|
|
345
|
-
`)
|
|
346
|
-
`)
|
|
347
|
-
`)
|
|
436
|
+
`+g(`Wallet: ${Z}`)+`
|
|
437
|
+
|
|
438
|
+
`);let G=Object.keys(X.addresses);for(let q=0;q<G.length;q++){let K=G[q],B=K===Y,H=X.addresses[K],U=X.shieldedAddresses?.[K],j=B?M(f(K))+V(" (active)"):K;if(process.stderr.write(` ${j}
|
|
439
|
+
`),process.stderr.write(` ${V("unshielded")} ${$0(H)}
|
|
440
|
+
`),U)process.stderr.write(` ${V("shielded ")} ${$0(U)}
|
|
441
|
+
`);else process.stderr.write(` ${V("shielded ")} ${V("(unavailable)")}
|
|
442
|
+
`);if(q<G.length-1)process.stderr.write(`
|
|
443
|
+
`)}process.stderr.write(`
|
|
444
|
+
`+o()+`
|
|
445
|
+
|
|
446
|
+
`),process.stderr.write(` ${V("created")} ${X.createdAt}
|
|
447
|
+
`),process.stderr.write(` ${V("file ")} ${Q}
|
|
448
|
+
`),process.stderr.write(` ${V("active ")} ${z?p("yes"):"no"}
|
|
449
|
+
|
|
450
|
+
`)}async function mX($){let Z=$.positionals[0];if(!Z)throw new Error(`Missing wallet name.
|
|
451
|
+
Usage: midnight wallet remove <name>`);if(m2(Z),x($,"json")){y({wallet:Z,removed:!0});return}process.stderr.write(p("✓")+` Wallet "${Z}" removed
|
|
452
|
+
`)}async function gX($){let Z=$.positionals[0]??$4(),Q=z0(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=x($,"entropy"),J;if(Y){if(!z.mnemonic)throw new Error(`Wallet "${Z}" has no mnemonic on file — cannot derive BIP-39 entropy.
|
|
453
|
+
`+"The 32-byte entropy can only be computed from the original 24-word mnemonic.");J=Buffer.from(bX(z.mnemonic,C8)).toString("hex")}if(x($,"json")){let q={name:Z,seed:z.seed};if(z.mnemonic)q.mnemonic=z.mnemonic;if(J)q.entropy=J;y(q);return}if(process.stderr.write(`
|
|
454
|
+
`),process.stderr.write(S(M(" ⚠ WARNING: This will display your wallet seed and mnemonic."))+`
|
|
455
|
+
`),process.stderr.write(S(" Anyone with this seed can access your funds.")+`
|
|
456
|
+
`),process.stderr.write(S(" Never share it. Never paste it into a website.")+`
|
|
348
457
|
`),process.stderr.write(`
|
|
349
|
-
|
|
350
|
-
|
|
351
|
-
`)}
|
|
352
|
-
|
|
353
|
-
`)
|
|
354
|
-
|
|
355
|
-
`),process.stderr.write(
|
|
356
|
-
`);
|
|
357
|
-
`)
|
|
358
|
-
`)
|
|
359
|
-
|
|
360
|
-
|
|
361
|
-
`);
|
|
362
|
-
`),process.stderr.
|
|
363
|
-
|
|
364
|
-
|
|
365
|
-
`),J?.
|
|
366
|
-
`)
|
|
458
|
+
`),!await uX(" Show seed? (y/N) ")){process.stderr.write(V(" Cancelled.")+`
|
|
459
|
+
|
|
460
|
+
`);return}if(process.stdout.write(z.seed+`
|
|
461
|
+
`),process.stderr.write(`
|
|
462
|
+
`),process.stderr.write(A("Name",Z)+`
|
|
463
|
+
`),process.stderr.write(A("Seed (64-byte PBKDF2)",z.seed)+`
|
|
464
|
+
`),J)process.stderr.write(A("Entropy (32-byte BIP-39)",J)+`
|
|
465
|
+
`);if(z.mnemonic)process.stderr.write(A("Mnemonic",z.mnemonic)+`
|
|
466
|
+
`);if(process.stderr.write(A("File",Q)+`
|
|
467
|
+
`),J)process.stderr.write(`
|
|
468
|
+
`),process.stderr.write(V(" Note: seed and entropy derive DIFFERENT Midnight wallets from the")+`
|
|
469
|
+
`),process.stderr.write(V(" same mnemonic. Use the value your target project accepts.")+`
|
|
470
|
+
`);process.stderr.write(`
|
|
471
|
+
`)}function uX($){return new Promise((Z)=>{let X=X4("readline").createInterface({input:process.stdin,output:process.stderr});X.question($,(z)=>{X.close(),Z(z.trim().toLowerCase()==="y")})})}var f$=N(()=>{l0();i0();H0();F0();K0()});var X3={};m(X3,{default:()=>Q3});function y8($){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 v8($){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 h$($){let Z=`${B1}${$}`,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 lX($){let Z=D($,"network");if(Z)return Z;try{let{loadCliConfig:Q}=await Promise.resolve().then(() => (p4(),u7)),X=Q();if(X.network&&_0(X.network))return X.network}catch{}return}async function g$($,Z={}){let Q=new AbortController,X=setTimeout(()=>Q.abort(),dX),z=Date.now();try{return{response:await fetch($,{...Z,signal:Q.signal}),latencyMs:Date.now()-z}}finally{clearTimeout(X)}}async function iX($){let Z=new Date().toISOString();try{let{response:Q,latencyMs:X}=await g$($,{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>m$?"degraded":"up",latencyMs:X,lastChecked:Z,...Y!==void 0?{blockHeight:Y}:{},...X>m$?{notes:`Slow (${X}ms)`}:{}}}catch(Q){return{status:"down",latencyMs:0,lastChecked:Z,notes:Q.message}}}async function nX($){let Z=new Date().toISOString(),Q=$.replace(/^wss:/,"https:").replace(/^ws:/,"http:");try{let{response:X,latencyMs:z}=await g$(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 J=(await X.json())?.result;if(!J)return{status:"down",latencyMs:z,lastChecked:Z,notes:"No result in response"};let G=J.peers??0,q=J.isSyncing??!1,K="up",B;if(q)K="degraded",B="Node is syncing";else if(G===0)K="degraded",B="No peers";return{status:K,latencyMs:z,lastChecked:Z,peers:G,isSyncing:q,...B?{notes:B}:{}}}catch(X){return{status:"down",latencyMs:0,lastChecked:Z,notes:X.message}}}async function oX($){let Z=new Date().toISOString();if(!$)return{status:"unknown",latencyMs:0,lastChecked:Z,notes:"No faucet URL"};try{let{response:Q,latencyMs:X}=await g$($);if(!Q.ok)return{status:"down",latencyMs:X,lastChecked:Z,notes:`HTTP ${Q.status}`};return{status:X>$3?"degraded":"up",latencyMs:X,lastChecked:Z,...X>$3?{notes:`Slow (${X}ms)`}:{}}}catch(Q){return{status:"down",latencyMs:0,lastChecked:Z,notes:Q.message}}}async function aX($){let Z=new Date().toISOString();if(!$)return{status:"unknown",latencyMs:0,lastChecked:Z,notes:"URL not configured"};try{let{response:Q,latencyMs:X}=await g$($);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>m$?"degraded":"up",latencyMs:X,lastChecked:Z,...X>m$?{notes:`Slow (${X}ms)`}:{}}}catch(Q){return{status:"down",latencyMs:0,lastChecked:Z,notes:Q.message}}}async function Z3($){if(!_0($))return{};let Z=q$($),Q=pX[$];if(!Q)return{};let[X,z,Y,J]=await Promise.all([iX(Z.indexer),nX(Z.node),oX(Q.faucet),aX(Q.explorer)]);return{indexer:X,rpc:z,faucet:Y,explorer:J}}function rX($,Z,Q){process.stderr.write(`
|
|
472
|
+
`+M($)+`
|
|
473
|
+
`),process.stderr.write(V(" "+"─".repeat(62))+`
|
|
474
|
+
`);let X=new Set([...Object.keys(Q),...Z?Object.keys(Z):[]]);for(let z of X){let Y=Q[z],J=Z?.[z],G=(cX[z]??z).padEnd(18);if(Y){let q=eZ[Y.status]??V("—"),K=Y.latencyMs>0?V(` ${Y.latencyMs}ms`):"",B=Y.notes?V(` — ${Y.notes}`):"",H="";if(J&&J.status!==Y.status&&J.status!=="unknown")H=V(` (canary: ${J.status}${J.lastChecked?" "+y8(J.lastChecked):""})`);process.stderr.write(` ${t0(G)}${q}${K}${B}${H}
|
|
475
|
+
`)}else if(J){let q=eZ[J.status]??V("—"),K=J.latencyMs>0?V(` ${J.latencyMs}ms`):"",B=J.lastChecked?V(` (${y8(J.lastChecked)})`):"",H=J.notes?V(` — ${J.notes}`):"";process.stderr.write(` ${t0(G)}${q}${K}${B}${H}
|
|
476
|
+
`)}}}function sX($,Z){let Q=Z?$.filter((X)=>X.affects.toLowerCase().includes(Z.toLowerCase())):$;if(Q.length===0)return;process.stderr.write(`
|
|
477
|
+
`+g("Known Issues")+`
|
|
478
|
+
|
|
479
|
+
`);for(let X of Q)process.stderr.write(` ${n(X.id.padEnd(28))}${X.summary}
|
|
480
|
+
`),process.stderr.write(` ${" ".repeat(28)}${V(X.status)}
|
|
481
|
+
`)}async function Q3($){let Z=x($,"json"),Q=x($,"all"),X=x($,"watch"),z=await lX($)??"preprod",Y=l("Checking network status..."),J=null,G=null,q,K;if(Q)K=["preprod","preview"];else K=[z];let[B,...H]=await Promise.all([Promise.all([h$("/api/status"),h$("/api/issues")]).catch((W)=>{return q=W.message,null}),...K.map((W)=>Z3(W))]);if(B)[J,G]=B;Y.stop("Done");let U={};for(let W=0;W<K.length;W++)U[K[W]]=H[W];if(Z){let W={lastUpdated:J?.lastUpdated??null,dashboard:B1,canaryAvailable:J!==null,networks:{}};for(let O of K){let F=U[O]??{},I=J?.networks[O],_={...I,...F};W.networks[O]={overall:v8(_),live:F,canary:I??null}}if(G){let O=Q?void 0:K[0];W.issues=O?G.issues.filter((F)=>F.affects.toLowerCase().includes(O.toLowerCase())):G.issues}if(q)W.canaryError=q;y(W);let P=K.reduce((O,F)=>{let I={...J?.networks[F]??{},...U[F]??{}},_=v8(I);if(_==="down")return"down";if(_==="degraded"&&O!=="down")return"degraded";return O},"up");if(P==="down")process.exitCode=2;else if(P==="degraded")process.exitCode=1;return}let j=()=>{if(process.stderr.write(`
|
|
482
|
+
`+g("Midnight Network Status")+`
|
|
483
|
+
`),J?.lastUpdated)process.stderr.write(V(` Canary: ${y8(J.lastUpdated)}`)+" ");if(process.stderr.write(V("Live: just now")+`
|
|
484
|
+
`),q)process.stderr.write(V(` (Dashboard unreachable: ${q})`)+`
|
|
367
485
|
`);if(K.length===0){process.stderr.write(`
|
|
368
|
-
`+
|
|
369
|
-
|
|
370
|
-
`);return}for(let
|
|
371
|
-
`+
|
|
372
|
-
`),process.stderr.write(
|
|
373
|
-
|
|
374
|
-
`)};if(
|
|
375
|
-
`)}},30000);await new Promise(()=>{process.on("SIGINT",()=>{clearInterval(F),process.exit(0)})})}let H=K.reduce((F,D)=>{let P={...J?.networks[D]??{},...j[D]??{}},M=W$(P);if(M==="down")return"down";if(M==="degraded"&&F!=="down")return"degraded";return F},"up");if(H==="down")process.exitCode=2;else if(H==="degraded")process.exitCode=1}var $4,Z4,X9,Q4=15000,r1=5000,z9=1e4;var K9=x(()=>{T0();s();y0();$4={preprod:{faucet:"https://faucet.preprod.midnight.network/",explorer:"https://preprod.midnightexplorer.com/"},preview:{faucet:"https://faucet.preview.midnight.network/",explorer:null},undeployed:{faucet:"",explorer:""}},Z4={indexer:"Indexer",rpc:"RPC Node",faucet:"Faucet",explorer:"Explorer",chain:"Chain",dust:"Dust Generation",wallet:"Wallet Ops",dapp:"DApp Flow"},X9={up:u("UP"),down:c("DOWN"),degraded:y("\x1B[33mDEGRADED\x1B[0m"),unknown:B("—")}});var L4={};import{Server as U4}from"@modelcontextprotocol/sdk/server/index.js";import{StdioServerTransport as V4}from"@modelcontextprotocol/sdk/server/stdio.js";import{ListToolsRequestSchema as j4,CallToolRequestSchema as B4}from"@modelcontextprotocol/sdk/types.js";function j0($,Z,Q){let X={json:!0},z=[];for(let[Y,J]of Object.entries(Z)){if(J===void 0||J===null)continue;if(typeof J==="boolean"){if(J)X[Y]=!0}else X[Y]=String(J)}return{command:$,subcommand:Q,positionals:z,flags:X}}async function g($){let Z=W4[$];if(!Z)throw new Error(`Unknown command handler: ${$}`);return(await Z()).default}async function H4(){let $=new V4;await L$.connect($)}var W4,q9,L$;var U9=x(()=>{G7();H2();v1();W4={generate:()=>Promise.resolve().then(() => (E2(),N2)),info:()=>Promise.resolve().then(() => (x2(),C2)),balance:()=>Promise.resolve().then(() => (b2(),y2)),address:()=>Promise.resolve().then(() => (v2(),S2)),"genesis-address":()=>Promise.resolve().then(() => (w2(),k2)),"inspect-cost":()=>Promise.resolve().then(() => (h2(),f2)),airdrop:()=>Promise.resolve().then(() => (t2(),a2)),transfer:()=>Promise.resolve().then(() => (r2(),s2)),dust:()=>Promise.resolve().then(() => ($$(),e2)),cache:()=>Promise.resolve().then(() => (Q$(),Z$)),config:()=>Promise.resolve().then(() => (z$(),X$)),localnet:()=>Promise.resolve().then(() => (U$(),q$)),wallet:()=>Promise.resolve().then(() => (B$(),j$)),status:()=>Promise.resolve().then(() => (K9(),G9))};q9=[{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=j0("generate",$);if($.force==="true"||$.force===!0)Z.flags.force=!0;let Q=await g("generate");return m(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=j0("wallet",$,"generate");if(Q.positionals=[Z],delete Q.flags.name,$.force==="true"||$.force===!0)Q.flags.force=!0;let X=await g("wallet");return m(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 g("wallet");return m(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 g("wallet");return m(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 g("wallet");return m(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 g("wallet");return m(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=j0("info",$),Q=await g("info");return m(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=j0("balance",$,Z);delete Q.flags.address;let X=await g("balance");return m(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=j0("address",$),Q=await g("address");return m(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=j0("genesis-address",$),Q=await g("genesis-address");return m(Q,Z)}},{name:"midnight_inspect_cost",description:"Display current block limits derived from LedgerParameters",inputSchema:{type:"object",properties:{}},async handler(){let $=j0("inspect-cost",{}),Z=await g("inspect-cost");return m(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=j0("airdrop",$,Z);delete Q.flags.amount;let X=await g("airdrop");return m(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=j0("transfer",$,Z);X.positionals=[Q],delete X.flags.to,delete X.flags.amount;let z=await g("transfer");return m(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=j0("dust",$,"register"),Q=await g("dust");return m(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=j0("dust",$,"status"),Q=await g("dust");return m(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 g("config");return m(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 g("config");return m(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=j0("cache",$,"clear"),Q=await g("cache");return m(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 g("config");return m(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 g("localnet");return m(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 g("localnet");return m(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 g("localnet");return m(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 g("localnet");return m(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 g("localnet");return m(Z,$)}}],L$=new U4({name:"midnight-wallet-cli",version:k0},{capabilities:{tools:{}}});L$.setRequestHandler(j4,async()=>{return{tools:q9.map(($)=>({name:$.name,description:$.description,inputSchema:$.inputSchema}))}});L$.setRequestHandler(B4,async($)=>{let{name:Z,arguments:Q}=$.params,X=q9.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 Y=z instanceof Error?z:new Error(String(z)),{errorCode:J}=U1(Y);return{content:[{type:"text",text:JSON.stringify({error:!0,code:J,message:Y.message})}],isError:!0}}});H4().catch(($)=>{process.stderr.write(`MCP server error: ${$.message}
|
|
376
|
-
|
|
377
|
-
|
|
378
|
-
|
|
379
|
-
`)}
|
|
380
|
-
`),
|
|
381
|
-
`),
|
|
382
|
-
`)
|
|
383
|
-
`)
|
|
384
|
-
`)
|
|
385
|
-
`)
|
|
486
|
+
`+V("No network data available.")+`
|
|
487
|
+
|
|
488
|
+
`);return}for(let W of K)rX(W,J?.networks[W],U[W]??{});if(G){let W=Q?void 0:K[0];sX(G.issues,W)}process.stderr.write(`
|
|
489
|
+
`+o()+`
|
|
490
|
+
`),process.stderr.write(V(" Dashboard: ")+f(B1)+`
|
|
491
|
+
|
|
492
|
+
`)};if(j(),X){let W=setInterval(async()=>{try{let[P,...O]=await Promise.all([Promise.all([h$("/api/status"),h$("/api/issues")]).catch(()=>null),...K.map((F)=>Z3(F))]);if(P)[J,G]=P;for(let F=0;F<K.length;F++)U[K[F]]=O[F];process.stderr.write("\x1B[2J\x1B[H"),j()}catch{process.stderr.write(V(" Refresh failed — retrying in 30s")+`
|
|
493
|
+
`)}},30000);await new Promise(()=>{process.on("SIGINT",()=>{clearInterval(W),process.exit(0)})})}let L=K.reduce((W,P)=>{let O={...J?.networks[P]??{},...U[P]??{}},F=v8(O);if(F==="down")return"down";if(F==="degraded"&&W!=="down")return"degraded";return W},"up");if(L==="down")process.exitCode=2;else if(L==="degraded")process.exitCode=1}var pX,cX,eZ,dX=15000,m$=5000,$3=1e4;var z3=N(()=>{C0();K0();u0();pX={preprod:{faucet:"https://faucet.preprod.midnight.network/",explorer:"https://preprod.midnightexplorer.com/"},preview:{faucet:"https://faucet.preview.midnight.network/",explorer:null},undeployed:{faucet:"",explorer:""}},cX={indexer:"Indexer",rpc:"RPC Node",faucet:"Faucet",explorer:"Explorer",chain:"Chain",dust:"Dust Generation",wallet:"Wallet Ops",dapp:"DApp Flow"},eZ={up:p("UP"),down:n("DOWN"),degraded:M("\x1B[33mDEGRADED\x1B[0m"),unknown:V("—")}});var h8={};m(h8,{toJsonOutput:()=>f8,formatWitnessSignature:()=>w8,formatCompactType:()=>R0,formatCircuitSignature:()=>S8,formatCircuitFlags:()=>k8,findContractInfo:()=>j4});import{existsSync as b8,readFileSync as tX,readdirSync as eX}from"node:fs";import{join as X1,basename as $z}from"node:path";function j4($,Z){let Q=X1($,"compiler",u$);if(b8(Q))return J3(Q,$,[]);for(let X of Y3){let z=X?X1($,X,"managed"):X1($,"managed");if(!b8(z))continue;let Y;try{Y=eX(z).sort()}catch{continue}let J=Y.filter((B)=>b8(X1(z,B,"compiler",u$)));if(J.length===0)continue;let G;if(Z){if(!J.includes(Z))throw new Error(`Contract "${Z}" not found in ${z}
|
|
494
|
+
Available: ${J.join(", ")}`);G=Z}else G=J[0];let q=J.filter((B)=>B!==G),K=X1(z,G,"compiler",u$);return J3(K,X1(z,G),q)}throw new Error(`No compiled contract found in ${$}
|
|
495
|
+
Expected: managed/<name>/compiler/${u$}
|
|
496
|
+
Searched: ${Y3.map((X)=>X?`${X}/managed/`:"managed/").join(", ")}
|
|
497
|
+
Run "compact compile" first, or use --managed <path> to specify the managed directory.`)}function J3($,Z,Q){let X;try{X=JSON.parse(tX($,"utf-8"))}catch(Y){throw new Error(`Failed to parse ${$}: ${Y.message}`)}return{info:Zz(X,$,Z,Q),infoPath:$}}function Zz($,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",J=typeof z["language-version"]==="string"?z["language-version"]:"unknown",G=typeof z["runtime-version"]==="string"?z["runtime-version"]:"unknown",q=Array.isArray(z.circuits)?z.circuits:[],K=Array.isArray(z.witnesses)?z.witnesses:[];return{name:$z(Q),managedDir:Q,compilerVersion:Y,languageVersion:J,runtimeVersion:G,circuits:q,witnesses:K,siblings:X}}function R0($){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(R0).join(", ")}]`;case"Boolean":return"boolean";case"String":return"string";case"Vector":return $.types?.[0]?`${R0($.types[0])}[]`:"unknown[]";case"Map":if($.types&&$.types.length>=2)return`Map<${R0($.types[0])}, ${R0($.types[1])}>`;return"Map<unknown, unknown>";case"Set":return $.types?.[0]?`Set<${R0($.types[0])}>`:"Set<unknown>";case"Option":return $.types?.[0]?`${R0($.types[0])} | null`:"unknown | null";default:return $["type-name"]??"unknown"}}function S8($){let Z=$.arguments.map((Q)=>`${Q.name}: ${R0(Q.type)}`).join(", ");return`${$.name}(${Z})`}function k8($){if($.pure)return"pure";let Z=["impure"];if($.proof)Z.push("proof");return Z.join(", ")}function w8($){let Z=$.arguments.map((z)=>`${z.name}: ${R0(z.type)}`).join(", "),Q=R0($["result type"]),X=Q!=="void"?` → ${Q}`:"";return`${$.name}(${Z})${X}`}function f8($){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:R0(Q.type)})),returnType:R0(Z["result-type"])})),witnesses:$.witnesses.map((Z)=>({name:Z.name,arguments:Z.arguments.map((Q)=>({name:Q.name,type:R0(Q.type)})),returnType:R0(Z["result type"])}))}}var u$="contract-info.json",Y3;var k1=N(()=>{Y3=["","contract/src","contract","contracts/src","contracts","src"]});import*as G3 from"node:readline";function zz($){return Qz.has($)}function Yz($){return Xz.has($)}function Jz($){let Z=[],Q=$.dappName?`Request from "${$.dappName}"`:"DApp Request";if(Z.push(M(Q)),Z.push(""),Z.push(` ${V("Action:")} ${f($.method)}`),Z.push(` ${V("Network:")} ${$.network}`),$.details.length>0){Z.push("");for(let X of $.details)Z.push(` ${V(X.label+":")} ${X.value}`)}return Z.push(""),Z.push(` ${p("[A]pprove")} ${n("[R]eject")}`),z$(Z,"heavy")}async function q3($,Z={}){if(Z.approveAll)return process.stderr.write(V(` Auto-approved: ${$.method}`)+`
|
|
498
|
+
`),"approve";if(Z.autoApproveReads&&zz($.method))return process.stderr.write(V(` Auto-approved (read-only): ${$.method}`)+`
|
|
499
|
+
`),"approve";if(Z.autoApproveReads&&Yz($.method))return process.stderr.write(V(` Auto-approved (prep): ${$.method}`)+`
|
|
500
|
+
`),"approve";if(!process.stdin.isTTY)return process.stderr.write(n(" Cannot prompt for approval: stdin is not a TTY")+`
|
|
501
|
+
`),process.stderr.write(V(" Use --approve-all for non-interactive environments")+`
|
|
502
|
+
`),"reject";if(w1)return process.stderr.write(n(" Rejected: another approval prompt is active")+`
|
|
503
|
+
`),"reject";process.stderr.write(`
|
|
504
|
+
`+Jz($)+`
|
|
505
|
+
|
|
506
|
+
`),w1=!0;let Q;try{Q=G3.createInterface({input:process.stdin,output:process.stderr})}catch{return w1=!1,process.stderr.write(n(" Cannot create readline interface")+`
|
|
507
|
+
`),"reject"}return new Promise((X)=>{let z=(Y)=>{w1=!1,Q.close(),X(Y)};Q.on("close",()=>{if(w1)z("reject")}),Q.question(S(" Approve? [A/r] "),(Y)=>{let J=Y.trim().toLowerCase();if(J==="r"||J==="reject")z("reject");else z("approve")})})}var Qz,Xz,w1=!1;var K3=N(()=>{K0();Qz=new Set(["getShieldedBalances","getUnshieldedBalances","getDustBalance","getShieldedAddresses","getUnshieldedAddress","getDustAddress","getTxHistory","getConfiguration","getConnectionStatus"]),Xz=new Set(["balanceUnsealedTransaction","balanceSealedTransaction"])});import{WebSocketServer as Gz}from"ws";function Kz($){if(m8($))return qz[$.code]??-32603;return-32603}function m8($){return typeof $==="object"&&$!==null&&"type"in $&&$.type==="DAppConnectorAPIError"}function Vz($){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,J=z.message??"";Z.push(Y?`[${Y}] ${J}`:J),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 p$($,Z){if(typeof Z==="bigint")return Z.toString();return Z}function Bz($,Z){return Z}function Hz($){let Z;try{Z=JSON.parse($,Bz)}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 c$($){let{port:Z,handlers:Q,onConnect:X,onDisconnect:z,onRequest:Y,onResponse:J}=$,G=new Map,q=new Gz({port:Z,host:"127.0.0.1"});q.on("error",(B)=>{process.stderr.write(`WebSocket server error: ${B.message}
|
|
508
|
+
`)}),q.on("connection",(B)=>{let H=`conn_${++Wz}`,U={ws:B,id:H,connectedAt:new Date,authenticated:!1,requestCount:0,notify(W,P){if(B.readyState!==B.OPEN)return;let O={jsonrpc:"2.0",method:W,...P&&{params:P}};B.send(JSON.stringify(O,p$))}};G.set(H,U),X?.(U);let j=!1,L=()=>{if(j)return;j=!0,G.delete(H),z?.(U)};B.on("message",async(W)=>{let P=null,O,F,I=Date.now();try{O=Hz(W.toString("utf-8")),P=O.id,U.requestCount++,Y?.(U,O);let _=Q[O.method];if(!_){let k={jsonrpc:"2.0",id:O.id,error:{code:-32601,message:`Method not found: ${O.method}`}};B.send(JSON.stringify(k,p$));return}F={notify:U.notify.bind(U),connectionId:H,requestId:O.id,metadata:{}};let v=await _(O.params??{},F),b=Date.now()-I;if(O.method==="connect"&&v&&typeof v==="object")U.authenticated=!0,U.networkId=v.networkId;let c={jsonrpc:"2.0",id:O.id,result:v};B.send(JSON.stringify(c,p$)),J?.(U,O,b,v,void 0,F.metadata)}catch(_){let v=Date.now()-I,b=_?.rpcCode??Kz(_),c=_ instanceof Error?_.message:"Internal error",k=m8(_)?{type:_.type,code:_.code}:Vz(_),J0={jsonrpc:"2.0",id:P,error:{code:b,message:c,data:k}};if(B.send(JSON.stringify(J0,p$)),O){let w={message:c,code:m8(_)?_.code:void 0};J?.(U,O,v,void 0,w,F?.metadata)}}}),B.on("close",L),B.on("error",()=>{L();try{B.close()}catch{}})});async function K(){for(let B of G.values())B.ws.close(1001,"Server shutting down");return G.clear(),new Promise((B,H)=>{q.close((U)=>{if(U)H(U);else B()})})}return{wss:q,connections:G,close:K}}function k0($,Z){let Q=new Error(Z);return Q.type="DAppConnectorAPIError",Q.code=$,Q.reason=Z,Q}var qz,Wz=0;var d$=N(()=>{qz={Rejected:-32000,PermissionRejected:-32001,Disconnected:-32002,InvalidRequest:-32602,InternalError:-32603}});function V3($){let Z=[],Q,X;function z(){if(Q!==void 0&&X!==void 0){let G=Date.now()-X;Z.push({phase:Q,durationMs:G}),$?.onComplete?.(Q,G),Q=void 0,X=void 0}}function Y(G){z(),Q=G,X=Date.now(),$?.onStart?.(G)}function J(){return[...Z]}return{start:Y,complete:z,getTimings:J}}import{Transaction as B3}from"@midnight-ntwrk/ledger-v8";function Oz($){return Buffer.from($).toString("hex")}function l$($){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 U3($){return Oz($.serialize())}function L3($){return B3.deserialize(H3,W3,Lz,l$($))}function g8($){return B3.deserialize(H3,W3,Uz,l$($))}var H3="signature",W3="proof",Uz="binding",Lz="pre-binding";var O3=()=>{};import{Transaction as jz}from"@midnight-ntwrk/ledger-v8";function i$($,Z){let Q=[];Q.push({label:"Tx size",value:_z($.length/2)});let X=Pz($,Z);if(X){let z=Fz(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 Pz($,Z){let Q=Iz($),X=Z==="sealed"?[["signature","proof","binding"],["signature","proof","pre-binding"]]:[["signature","proof","pre-binding"],["signature","pre-proof","pre-binding"]];for(let[z,Y,J]of X)try{return jz.deserialize(z,Y,J,Q).toString(!0)}catch{}return null}function Fz($){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 J=$.match(/Call\s+\S+\s+(\w+)/);if(J)Z.circuits.push(J[1])}let Y=$.match(/ttl:\s*Timestamp\((\d+)\)/);if(Y){let J=parseInt(Y[1],10),G=new Date(J*1000);Z.ttl=G.toISOString().replace("T"," ").slice(0,19)}return Z}function Iz($){return new Uint8Array(Buffer.from($,"hex"))}function _z($){if($<1024)return`${$} B`;let Z=$/1024;if(Z<1024)return`${Z.toFixed(1)} KB`;return`${(Z/1024).toFixed(1)} MB`}var j3=()=>{};import{DustLocalState as Tz}from"@midnight-ntwrk/ledger-v8";import{CoreWallet as c0}from"@midnight-ntwrk/wallet-sdk-dust-wallet/v1";var x4,Mz,Rz,RW;var P3=N(()=>{x4=new Map,Mz=c0.applyEventsWithChanges;c0.applyEventsWithChanges=function $(Z,Q,X,z){let Y=Mz.call(c0,Z,Q,X,z),J=Array.isArray(Y)?Y[0]:Y;if(x4.size>0&&J?.pendingDust){let G=new Set(J.pendingDust.map((q)=>q.nullifier));for(let q of x4.keys())if(!G.has(q))x4.delete(q)}return Y};Rz=c0.spendCoins;c0.spendCoins=function $(Z,Q,X,z){let Y=null;try{Y=Z.state.serialize()}catch{}let J=Rz.call(c0,Z,Q,X,z),G=J[1]??J;if(Y&&G?.pendingDust){for(let q of G.pendingDust)if(!x4.has(q.nullifier))x4.set(q.nullifier,Y)}return J};RW=c0.revertTransaction;c0.revertTransaction=function $(Z,Q){let X=c0.pendingDustToMap(Z.pendingDust),z=[],Y=Q.intents;if(Y)for(let G of Y.values()){let q=G.dustActions?.spends;if(!q)continue;for(let K of q)if(X.has(K.oldNullifier))z.push(K.oldNullifier)}let J=Z.state;for(let G of z){let q=x4.get(G);if(q){try{J=Tz.deserialize(q)}catch{}x4.delete(G)}}return{...Z,state:J,pendingDust:Z.pendingDust.filter((G)=>!z.includes(G.nullifier))}};c0.applyFailed=c0.revertTransaction});import*as C4 from"rxjs";import{MidnightBech32m as u8,UnshieldedAddress as Az,ShieldedAddress as Dz}from"@midnight-ntwrk/wallet-sdk-address-format";import{NetworkId as p8}from"@midnight-ntwrk/wallet-sdk-abstractions";function Ez($){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 J=Y._tag,G=Y.message??"";if(J&&G)Z.push(`[${J}] ${G}`);else if(G)Z.push(G);else if(J)Z.push(`[${J}]`);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 J=JSON.stringify(Q);if(J&&J!=="{}")Z.push(J);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 n$($){let{bundle:Z,networkConfig:Q,approvalOptions:X,callbacks:z}=$,{facade:Y,keystore:J,zswapSecretKeys:G,dustSecretKey:q}=Z,K=Nz[Q.networkId];if(K===void 0)throw new Error(`Unknown networkId: ${Q.networkId}`);let B,H=Y.state().pipe(C4.filter((C)=>C.isSynced)).subscribe((C)=>{B=C});function U(){if(!B)throw k0("Disconnected","Wallet not synced yet");return B}let j={shieldedSecretKeys:G,dustSecretKey:q},L=new Map;function W(C,R,T){let E=L.get(C);if(!E)E=new Map,L.set(C,E);let u=setTimeout(async()=>{if(E.delete(R),E.size===0)L.delete(C);try{await Y.revert(T)}catch{}process.stderr.write(V(` abandoned tx reverted (${C})`)+`
|
|
509
|
+
`)},T2);E.set(R,{recipe:T,timer:u})}function P(C,R){let T=L.get(C);if(!T)return;let E=T.get(R);if(E)clearTimeout(E.timer),T.delete(R);if(T.size===0)L.delete(C)}async function O(C){let R=L.get(C);if(!R||R.size===0)return;L.delete(C);for(let[,T]of R){clearTimeout(T.timer);try{await Y.revert(T.recipe)}catch{}}process.stderr.write(V(` reverted ${R.size} pending tx(s) on disconnect`)+`
|
|
510
|
+
`)}function F(){return new Date(Date.now()+w4*60*1000)}async function I(C,R){R?.start("signing");let T=await Y.signRecipe(C,(h)=>J.signData(h));R?.start("proving");let E,u=await Promise.race([Y.finalizeRecipe(T),new Promise((h,s)=>{E=setTimeout(()=>s(new Error("ZK proof generation timed out")),K1)})]);return clearTimeout(E),R?.complete(),{hex:U3(u),finalized:u}}async function _(C,R=[],T){T?.notify("approval:pending",{method:C});let E=await q3({method:C,network:Q.networkId,details:R,dappName:T?.connectionId},X),u=E==="reject"?"rejected":"approved";if(T?.notify("approval:resolved",{method:C,result:u}),E==="reject")throw k0("Rejected","User rejected the request")}function v(C){return u8.encode(K,C).asString()}function b(C){let R={};for(let T of C){let E=T.kind;if(E!=="shielded"&&E!=="unshielded")throw k0("InvalidRequest",`Invalid output kind: "${E}" — must be "shielded" or "unshielded"`);if(!R[E])R[E]=[];let u=BigInt(T.value),h;if(E==="unshielded")h=u8.parse(T.recipient).decode(Az,K);else h=u8.parse(T.recipient).decode(Dz,K);R[E].push({type:T.type,receiverAddress:h,amount:u})}return Object.entries(R).map(([T,E])=>({type:T,outputs:E}))}function c(C,R){return V3({onStart:(T)=>{let E=R?.connectionId??"unknown";z?.onPhaseStart?.(E,C,T),R?.notify("progress",{method:C,phase:T,status:"started"})},onComplete:(T,E)=>{let u=R?.connectionId??"unknown";z?.onPhaseComplete?.(u,C,T,E),R?.notify("progress",{method:C,phase:T,status:"completed",durationMs:E})}})}function k(){if(!B)return!1;try{let C=B.dust;return C?.availableCoins?.length>0||C?.balance(new Date)>0n}catch{return!1}}async function J0(C){return new Promise((R)=>{if(k()){R(!0);return}let T=Y.state().pipe(C4.filter(()=>k()),C4.take(1),C4.timeout(C)).subscribe({next:()=>{T.unsubscribe(),R(!0)},error:()=>{T.unsubscribe(),R(!1)}})})}async function w(C){for(let R=1;R<=h4;R++)try{return await C()}catch(T){let E=String(T?.message??T??"");if(!(/no dust tokens/i.test(E)||/dust.*unavailable/i.test(E))||R===h4)throw T;if(process.stderr.write(V(` dust unavailable, waiting for recovery (${R}/${h4})... [${E.slice(0,60)}]`)+`
|
|
511
|
+
`),!await J0($$)&&R<h4)await new Promise((s)=>setTimeout(s,$$))}throw new Error("unreachable")}let i={connect:async(C)=>{let R=String(C.networkId??"");if(R.toLowerCase()!==Q.networkId.toLowerCase())throw k0("InvalidRequest",`Network mismatch: wallet is on ${Q.networkId}, requested ${R}`);return{networkId:Q.networkId}},getUnshieldedBalances:async()=>{return U().unshielded.balances},getShieldedBalances:async()=>{return U().shielded.balances},getDustBalance:async()=>{let R=U().dust.balance(new Date);return{cap:R,balance:R}},getUnshieldedAddress:async()=>{let C=U();return{unshieldedAddress:v(C.unshielded.address)}},getShieldedAddresses:async()=>{let R=U().shielded.address;return{shieldedAddress:v(R),shieldedCoinPublicKey:R.coinPublicKeyString(),shieldedEncryptionPublicKey:R.encryptionPublicKeyString()}},getDustAddress:async()=>{let C=U();return{dustAddress:v(C.dust.address)}},getTxHistory:async(C)=>{let R=U(),T=Number(C.pageNumber??0),E=Number(C.pageSize??20);try{let u=R.unshielded.transactionHistory,h=[],s=0,O0=T*E;for await(let X0 of u.getAll()){if(s>=O0+E)break;if(s>=O0)h.push({txHash:X0.hash,txStatus:X0.status==="SUCCESS"?{status:"finalized"}:{status:"pending"}});s++}return h}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:K}},makeTransfer:async(C,R)=>{let T=C.desiredOutputs;if(!Array.isArray(T)||T.length===0)throw k0("InvalidRequest","desiredOutputs must be a non-empty array");let E=c("makeTransfer",R),u=T.map((w0,f0)=>({label:`Output ${f0+1}`,value:`${w0.value} → ${String(w0.recipient).slice(0,20)}... (${w0.kind})`}));E.start("approve"),await _("makeTransfer",u,R),E.start("building");let h=b(T),s=C.options?.payFees??!0,O0=await w(()=>Y.transferTransaction(h,j,{ttl:F(),payFees:s})),{hex:X0,finalized:G0}=await I(O0,E);return W(R.connectionId,X0,O0),R.metadata.phases=E.getTimings(),{tx:X0}},submitTransaction:async(C,R)=>{let T=String(C.tx??"");if(!T)throw k0("InvalidRequest","tx is required");let E=c("submitTransaction",R);E.start("approve");try{await _("submitTransaction",i$(T,"sealed"),R)}catch(h){let O0=L.get(R.connectionId)?.get(T);if(O0)try{await Y.revert(O0.recipe)}catch{}throw P(R.connectionId,T),h}let u=g8(T);E.start("submitting");try{let h=await Y.submitTransaction(u);return E.complete(),P(R.connectionId,T),R.metadata.phases=E.getTimings(),{txHash:h}}catch(h){E.complete();let O0=L.get(R.connectionId)?.get(T);if(O0)try{await Y.revert(O0.recipe)}catch{}P(R.connectionId,T);let X0=Ez(h),G0=new Error(`Transaction submission failed: ${X0}`);throw G0.cause=h,G0}},balanceUnsealedTransaction:async(C,R)=>{let T=String(C.tx??"");if(!T)throw k0("InvalidRequest","tx is required");let E=c("balanceUnsealedTransaction",R);E.start("approve"),await _("balanceUnsealedTransaction",i$(T,"unsealed"),R),E.start("building");let u=L3(T),h=await w(()=>Y.balanceUnboundTransaction(u,j,{ttl:F()})),{hex:s,finalized:O0}=await I(h,E);return W(R.connectionId,s,h),R.metadata.phases=E.getTimings(),{tx:s}},balanceSealedTransaction:async(C,R)=>{let T=String(C.tx??"");if(!T)throw k0("InvalidRequest","tx is required");let E=c("balanceSealedTransaction",R);E.start("approve"),await _("balanceSealedTransaction",i$(T,"sealed"),R),E.start("building");let u=g8(T),h=await w(()=>Y.balanceFinalizedTransaction(u,j,{ttl:F()})),{hex:s,finalized:O0}=await I(h,E);return W(R.connectionId,s,h),R.metadata.phases=E.getTimings(),{tx:s}},makeIntent:async(C,R)=>{let{desiredInputs:T,desiredOutputs:E,options:u}=C;if(!u)throw k0("InvalidRequest","options is required for makeIntent");let h=c("makeIntent",R);h.start("approve"),await _("makeIntent",[],R),h.start("building");let s={};if(Array.isArray(T))for(let f0 of T){let A0=f0.kind;if(!s[A0])s[A0]={};s[A0][f0.type]=BigInt(f0.value)}let O0=b(E??[]),X0=await w(()=>Y.initSwap(s,O0,j,{ttl:F(),payFees:u.payFees??!0})),{hex:G0,finalized:w0}=await I(X0,h);return W(R.connectionId,G0,X0),R.metadata.phases=h.getTimings(),{tx:G0}},signData:async(C,R)=>{let T=String(C.data??""),E=C.options;if(!T||!E?.encoding)throw k0("InvalidRequest","data and options.encoding are required");if(E.keyType&&E.keyType!=="unshielded")throw k0("InvalidRequest",`Unsupported keyType: "${E.keyType}" — only "unshielded" is supported`);await _("signData",[{label:"Encoding",value:E.encoding},{label:"Data",value:T.length>64?T.slice(0,64)+"...":T}],R);let u;switch(E.encoding){case"hex":u=l$(T);break;case"base64":u=new Uint8Array(Buffer.from(T,"base64"));break;case"text":u=new Uint8Array(Buffer.from(T,"utf-8"));break;default:throw k0("InvalidRequest",`Unknown encoding: ${E.encoding}`)}let h=J.signData(u),s=J.getPublicKey();return{data:T,signature:String(h),verifyingKey:String(s)}},getProvingProvider:async()=>{return{provingProvider:"ready",proverServerUri:Q.proofServer}},hintUsage:async(C)=>{let R=C.methodNames??[];process.stderr.write(V(` DApp hints usage: ${R.join(", ")}`)+`
|
|
512
|
+
`)}};function e(){H.unsubscribe();for(let[,C]of L)for(let[,R]of C)clearTimeout(R.timer);L.clear()}function Q0(){for(let[,C]of L)if(C.size>0)return!0;return!1}return{handlers:i,revertPendingTxs:O,hasPendingTxs:Q0,dispose:e}}var Nz;var c8=N(()=>{K3();d$();O3();j3();P3();Nz={PreProd:p8.NetworkId.PreProd,Preview:p8.NetworkId.Preview,Undeployed:p8.NetworkId.Undeployed}});var T3={};m(T3,{startServeOrReuse:()=>d8,startServe:()=>_3,probeServeNetwork:()=>I3,isPortInUse:()=>F3});import{Socket as xz}from"node:net";function F3($){return new Promise((Z)=>{let Q=new xz;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 I3($){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 G=JSON.parse(Y.toString())?.result?.networkId;X.close(),Q(typeof G==="string"?G:null)}catch{X.close(),Q(null)}}),X.on("error",()=>{clearTimeout(z),Q(null)})})}async function d8($){let Z=$.port??m4,Q=($.network??"undeployed").toLowerCase(),X=$.onMessage??(()=>{});if(await F3(Z)){let z=await I3(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 _3($)}async function _3($){let Z=$.port??m4,Q=$.onMessage??(()=>{}),X=Y0(z0($.wallet)),z=Buffer.from(X.seed,"hex"),{name:Y,config:J}=V0({args:{command:"serve",subcommand:void 0,positionals:[],flags:$.network?{network:$.network}:{}}}),G=X.addresses[Y],q=U0((L,W)=>{Q(`SDK: ${W}`)}),K=P0();Q("Building wallet facade...");let B=Z4(G,Y),H=await H4(z,J,B);Q("Syncing wallet..."),await W4(H,{onProgress:(L,W)=>{if(W>0){let P=Math.min(Math.round(L/W*100),100);Q(`Syncing wallet... ${P}%`)}}}),Q("Wallet synced"),Q("Waiting for dust..."),await e4(H),Q("Dust ready");try{await S0(G,Y,H.facade)}catch{}let U=n$({bundle:H,networkConfig:J,approvalOptions:{approveAll:!0,autoApproveReads:!0},callbacks:{onPhaseStart:(L,W,P)=>{Q(`${P}...`)},onPhaseComplete:(L,W,P,O)=>{Q(`${P} ${Math.round(O)}ms`)}}}),j=c$({port:Z,handlers:U.handlers,onConnect:(L)=>{Q(`connected ${L.id}`)},onDisconnect:(L)=>{Q(`disconnected ${L.id}`),U.revertPendingTxs(L.id).catch(()=>{})},onRequest:(L,W)=>{Q(`${L.id} → ${W.method}`)},onResponse:(L,W,P,O,F)=>{if(F)Q(`${L.id} ← ${W.method} FAILED: ${F.message}`);else Q(`${L.id} ← ${W.method} (${Math.round(P)}ms)`)}});return Q(`Server ready on ws://localhost:${Z}`),{port:Z,async stop(){j.close(),U.dispose(),K(),q();try{await n0(H)}catch{}}}}var l8=N(()=>{F0();H0();o0();U4();O4();c8();d$()});var N3={};m(N3,{findWitnessSource:()=>D3,findWitnessFile:()=>i8,buildMissingWitnessError:()=>n8,WITNESS_SOURCE_CANDIDATES:()=>A3,WITNESS_FILE_CANDIDATES:()=>f1});import{existsSync as M3}from"node:fs";import{join as R3}from"node:path";function i8($){for(let Z of f1){let Q=R3($,Z);if(M3(Q))return Q}return null}function D3($){for(let Z of A3){let Q=R3($,Z);if(M3(Q))return Q}return null}function n8($){let{projectRoot:Z,witnessNames:Q}=$,X=D3(Z),z=[`Contract declares ${Q.length} witness(es): ${Q.join(", ")}`,"but no compiled witnesses module was found.","",`Searched (relative to ${Z}):`,...f1.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(`
|
|
513
|
+
`)}var f1,A3;var o$=N(()=>{f1=["dist/witnesses.js","src/witnesses.js","contract/dist/witnesses.js","contract/src/witnesses.js"],A3=["src/witnesses.ts","contract/src/witnesses.ts"]});function a$($){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(a$)}if($&&typeof $==="object"&&Object.getPrototypeOf($)===Object.prototype){let Z={};for(let Q in $)if(Object.prototype.hasOwnProperty.call($,Q))Z[Q]=a$($[Q]);return Z}return $}var s$={};m(s$,{runState:()=>m1,runDeploy:()=>a8,runCall:()=>r8});import{spawn as Cz}from"node:child_process";import{writeFileSync as vz,unlinkSync as E3,existsSync as r$,symlinkSync as x3}from"node:fs";import{join as h1,dirname as C3}from"node:path";import{fileURLToPath as yz}from"node:url";function bz(){let $=C3(yz(import.meta.url));for(let Z=0;Z<8;Z++){let Q=h1($,"node_modules");if(r$(Q))return Q;let X=C3($);if(X===$)break;$=X}return null}async function m1($){let Z=wz($),Q=await s8($.dappDir,Z,$.onMessage);try{return JSON.parse(Q)}catch{throw new Error(`State script returned unexpected output:
|
|
514
|
+
${Q}`)}}async function a8($){let Z=Sz($),Q=await s8($.dappDir,Z,$.onMessage);try{return JSON.parse(Q)}catch{throw new Error(`Deploy script returned unexpected output:
|
|
515
|
+
${Q}`)}}async function r8($){let Z=kz($),Q=await s8($.dappDir,Z,$.onMessage);try{return JSON.parse(Q)}catch{throw new Error(`Call script returned unexpected output:
|
|
516
|
+
${Q}`)}}function v3($){return`
|
|
517
|
+
import WebSocket from 'ws';
|
|
518
|
+
|
|
519
|
+
const RPC_URL = 'ws://127.0.0.1:${$}';
|
|
520
|
+
let rpcId = 0;
|
|
521
|
+
let rpcWs;
|
|
522
|
+
|
|
523
|
+
async function rpcConnect() {
|
|
524
|
+
return new Promise((resolve, reject) => {
|
|
525
|
+
rpcWs = new WebSocket(RPC_URL);
|
|
526
|
+
rpcWs.on('open', () => resolve(rpcWs));
|
|
527
|
+
rpcWs.on('error', (err) => reject(new Error('Cannot connect to mn serve at ' + RPC_URL + ': ' + err.message)));
|
|
528
|
+
setTimeout(() => reject(new Error('Timeout connecting to mn serve at ' + RPC_URL)), 5000);
|
|
529
|
+
});
|
|
530
|
+
}
|
|
531
|
+
|
|
532
|
+
function rpcCall(method, params = {}) {
|
|
533
|
+
return new Promise((resolve, reject) => {
|
|
534
|
+
const id = ++rpcId;
|
|
535
|
+
const timeout = setTimeout(() => reject(new Error('RPC timeout for ' + method)), 300000);
|
|
536
|
+
|
|
537
|
+
const handler = (data) => {
|
|
538
|
+
try {
|
|
539
|
+
const msg = JSON.parse(data.toString());
|
|
540
|
+
if (msg.id === id) {
|
|
541
|
+
clearTimeout(timeout);
|
|
542
|
+
rpcWs.removeListener('message', handler);
|
|
543
|
+
if (msg.error) reject(new Error(msg.error.message || JSON.stringify(msg.error)));
|
|
544
|
+
else resolve(msg.result);
|
|
545
|
+
}
|
|
546
|
+
} catch {}
|
|
547
|
+
};
|
|
548
|
+
|
|
549
|
+
rpcWs.on('message', handler);
|
|
550
|
+
rpcWs.send(JSON.stringify({ jsonrpc: '2.0', id, method, params }));
|
|
551
|
+
});
|
|
552
|
+
}
|
|
553
|
+
|
|
554
|
+
function rpcClose() {
|
|
555
|
+
if (rpcWs) rpcWs.close();
|
|
556
|
+
}
|
|
557
|
+
`}function y3(){return`
|
|
558
|
+
// Wallet provider backed by mn serve RPC
|
|
559
|
+
const walletProvider = {
|
|
560
|
+
getCoinPublicKey: () => {
|
|
561
|
+
// Will be populated after getShieldedAddresses
|
|
562
|
+
return walletState?.shieldedCoinPublicKey ?? '';
|
|
563
|
+
},
|
|
564
|
+
getEncryptionPublicKey: () => {
|
|
565
|
+
return walletState?.shieldedEncryptionPublicKey ?? '';
|
|
566
|
+
},
|
|
567
|
+
async balanceTx(tx, ttl) {
|
|
568
|
+
const { toHex } = await import('@midnight-ntwrk/midnight-js-utils');
|
|
569
|
+
const txHex = toHex(tx.serialize());
|
|
570
|
+
const result = await rpcCall('balanceUnsealedTransaction', { tx: txHex });
|
|
571
|
+
|
|
572
|
+
const { fromHex } = await import('@midnight-ntwrk/midnight-js-utils');
|
|
573
|
+
const { Transaction } = await import('@midnight-ntwrk/midnight-js-types');
|
|
574
|
+
const bytes = fromHex(result.tx);
|
|
575
|
+
return Transaction.deserialize('signature', 'proof', 'binding', bytes);
|
|
576
|
+
},
|
|
577
|
+
async submitTx(tx) {
|
|
578
|
+
const { toHex } = await import('@midnight-ntwrk/midnight-js-utils');
|
|
579
|
+
const txHex = toHex(tx.serialize());
|
|
580
|
+
await rpcCall('submitTransaction', { tx: txHex });
|
|
581
|
+
return tx.identifiers()[0];
|
|
582
|
+
},
|
|
583
|
+
};
|
|
584
|
+
|
|
585
|
+
// Get wallet state for coin/encryption public keys
|
|
586
|
+
const walletState = await rpcCall('getShieldedAddresses', {});
|
|
587
|
+
`}function b3($,Z){return`
|
|
588
|
+
import { pathToFileURL } from 'node:url';
|
|
589
|
+
import { resolve } from 'node:path';
|
|
590
|
+
import { CompiledContract } from '@midnight-ntwrk/compact-js';
|
|
591
|
+
|
|
592
|
+
const MANAGED_DIR = ${JSON.stringify(Z)};
|
|
593
|
+
|
|
594
|
+
// Load compiled contract class
|
|
595
|
+
const contractMod = await import(pathToFileURL(resolve(MANAGED_DIR, 'contract', 'index.js')).href);
|
|
596
|
+
|
|
597
|
+
// Try to load witnesses and private state factory.
|
|
598
|
+
// Both relative-to-cwd locations and contract/<...> locations are checked
|
|
599
|
+
// so mn dev (which chdirs into the contract sub-package) and mn contract
|
|
600
|
+
// deploy run from a workspace root both work.
|
|
601
|
+
const WITNESS_CANDIDATES = ${JSON.stringify(f1)};
|
|
602
|
+
let witnesses;
|
|
603
|
+
let createPrivateState;
|
|
604
|
+
for (const p of WITNESS_CANDIDATES) {
|
|
605
|
+
try {
|
|
606
|
+
const wMod = await import(pathToFileURL(resolve(p)).href);
|
|
607
|
+
if (wMod.witnesses) { witnesses = wMod.witnesses; }
|
|
608
|
+
// Look for createInitialPrivateState or create*PrivateState
|
|
609
|
+
for (const key of Object.keys(wMod)) {
|
|
610
|
+
if (typeof wMod[key] === 'function' && key.startsWith('create') && key.toLowerCase().includes('privatestate')) {
|
|
611
|
+
createPrivateState = wMod[key];
|
|
612
|
+
}
|
|
613
|
+
}
|
|
614
|
+
if (witnesses) break;
|
|
615
|
+
} catch {}
|
|
616
|
+
}
|
|
617
|
+
|
|
618
|
+
// Note: callers (commands/contract.ts, commands/dev.ts) preflight when the
|
|
619
|
+
// contract's compiler-info declares witnesses, so a missing module here means
|
|
620
|
+
// the contract truly has no witnesses (vacant is correct). The runtime
|
|
621
|
+
// warning still names every path we tried — useful when the SDK later
|
|
622
|
+
// complains about a specific witness name despite preflight passing.
|
|
623
|
+
if (!witnesses) {
|
|
624
|
+
process.stderr.write('Warning: No witnesses module found — using vacant witnesses. Searched:\\n');
|
|
625
|
+
for (const p of WITNESS_CANDIDATES) process.stderr.write(' - ' + p + '\\n');
|
|
626
|
+
}
|
|
627
|
+
|
|
628
|
+
// Generate initial private state.
|
|
629
|
+
// Uses a deterministic seed derived from the wallet so post/takeDown use the same key.
|
|
630
|
+
// The private state provider (leveldb) persists between calls, but initialPrivateState
|
|
631
|
+
// is the fallback when no stored state exists yet.
|
|
632
|
+
function makeInitialPrivateState() {
|
|
633
|
+
if (createPrivateState) {
|
|
634
|
+
// Derive a deterministic key from the wallet's coin public key.
|
|
635
|
+
// This ensures post and takeDown always use the same secret key
|
|
636
|
+
// for the same wallet, even across separate CLI invocations.
|
|
637
|
+
const cpk = walletState?.shieldedCoinPublicKey ?? '';
|
|
638
|
+
let secretKey;
|
|
639
|
+
if (cpk && cpk.length >= 64) {
|
|
640
|
+
// Use first 32 bytes of coin public key as seed
|
|
641
|
+
secretKey = new Uint8Array(32);
|
|
642
|
+
for (let i = 0; i < 32; i++) {
|
|
643
|
+
secretKey[i] = parseInt(cpk.substr(i * 2, 2), 16);
|
|
644
|
+
}
|
|
645
|
+
} else {
|
|
646
|
+
// Fallback: random (will break takeDown if state isn't persisted)
|
|
647
|
+
secretKey = new Uint8Array(32);
|
|
648
|
+
globalThis.crypto.getRandomValues(secretKey);
|
|
649
|
+
}
|
|
650
|
+
try { return createPrivateState(secretKey); } catch {}
|
|
651
|
+
try { return createPrivateState(); } catch {}
|
|
652
|
+
}
|
|
653
|
+
return {};
|
|
654
|
+
}
|
|
655
|
+
|
|
656
|
+
const compiled = witnesses
|
|
657
|
+
? CompiledContract.make(${JSON.stringify($)}, contractMod.Contract).pipe(
|
|
658
|
+
(c) => CompiledContract.withWitnesses(c, witnesses),
|
|
659
|
+
(c) => CompiledContract.withCompiledFileAssets(c, MANAGED_DIR),
|
|
660
|
+
)
|
|
661
|
+
: CompiledContract.make(${JSON.stringify($)}, contractMod.Contract).pipe(
|
|
662
|
+
CompiledContract.withVacantWitnesses,
|
|
663
|
+
(c) => CompiledContract.withCompiledFileAssets(c, MANAGED_DIR),
|
|
664
|
+
);
|
|
665
|
+
`}function S3($,Z,Q){return`
|
|
666
|
+
import { NodeZkConfigProvider } from '@midnight-ntwrk/midnight-js-node-zk-config-provider';
|
|
667
|
+
import { httpClientProofProvider } from '@midnight-ntwrk/midnight-js-http-client-proof-provider';
|
|
668
|
+
import { indexerPublicDataProvider } from '@midnight-ntwrk/midnight-js-indexer-public-data-provider';
|
|
669
|
+
import { levelPrivateStateProvider } from '@midnight-ntwrk/midnight-js-level-private-state-provider';
|
|
670
|
+
|
|
671
|
+
const zkConfigProvider = new NodeZkConfigProvider(MANAGED_DIR);
|
|
672
|
+
|
|
673
|
+
const providers = {
|
|
674
|
+
privateStateProvider: levelPrivateStateProvider({
|
|
675
|
+
privateStateStoreName: ${JSON.stringify(Q)},
|
|
676
|
+
privateStoragePasswordProvider: () => Promise.resolve('mn-contract-default-pwd-16ch'),
|
|
677
|
+
accountId: 'mn-contract-runner',
|
|
678
|
+
}),
|
|
679
|
+
publicDataProvider: indexerPublicDataProvider(
|
|
680
|
+
${JSON.stringify(Z.indexer)},
|
|
681
|
+
${JSON.stringify(Z.indexerWS??Z.indexer.replace("http","ws"))},
|
|
682
|
+
),
|
|
683
|
+
zkConfigProvider,
|
|
684
|
+
proofProvider: httpClientProofProvider(${JSON.stringify(Z.proofServer)}, zkConfigProvider),
|
|
685
|
+
walletProvider,
|
|
686
|
+
midnightProvider: walletProvider,
|
|
687
|
+
};
|
|
688
|
+
`}function Sz($){let Z=$.privateStateKey??`${$.contractName}PrivateState`,Q=JSON.stringify($.args??[]);return`
|
|
689
|
+
import { setNetworkId } from '@midnight-ntwrk/midnight-js-network-id';
|
|
690
|
+
setNetworkId(${JSON.stringify($.networkConfig.networkId.toLowerCase())});
|
|
691
|
+
|
|
692
|
+
${v3($.servePort)}
|
|
693
|
+
${b3($.contractName,$.managedDir)}
|
|
694
|
+
|
|
695
|
+
process.stderr.write('Connecting to mn serve...\\n');
|
|
696
|
+
await rpcConnect();
|
|
697
|
+
process.stderr.write('Connected\\n');
|
|
698
|
+
|
|
699
|
+
${y3()}
|
|
700
|
+
${S3($.managedDir,$.networkConfig,Z)}
|
|
701
|
+
|
|
702
|
+
import { deployContract } from '@midnight-ntwrk/midnight-js-contracts';
|
|
703
|
+
|
|
704
|
+
${k3}
|
|
705
|
+
const constructorArgs = (${Q}).map(coerceArg);
|
|
706
|
+
process.stderr.write('Deploying contract' + (constructorArgs.length ? ' with ' + constructorArgs.length + ' constructor arg(s)...' : '...') + '\\n');
|
|
707
|
+
const deployed = await deployContract(providers, {
|
|
708
|
+
compiledContract: compiled,
|
|
709
|
+
privateStateId: ${JSON.stringify(Z)},
|
|
710
|
+
initialPrivateState: makeInitialPrivateState(),
|
|
711
|
+
args: constructorArgs,
|
|
712
|
+
});
|
|
713
|
+
|
|
714
|
+
const address = deployed.deployTxData?.public?.contractAddress ?? 'unknown';
|
|
715
|
+
process.stderr.write('Deploy complete: ' + address + '\\n');
|
|
716
|
+
|
|
717
|
+
console.log(JSON.stringify({ contractAddress: address }));
|
|
718
|
+
rpcClose();
|
|
719
|
+
process.exit(0);
|
|
720
|
+
`}function kz($){let Z=$.privateStateKey??`${$.contractName}PrivateState`;return`
|
|
721
|
+
import { setNetworkId } from '@midnight-ntwrk/midnight-js-network-id';
|
|
722
|
+
setNetworkId(${JSON.stringify($.networkConfig.networkId.toLowerCase())});
|
|
723
|
+
|
|
724
|
+
${v3($.servePort)}
|
|
725
|
+
${b3($.contractName,$.managedDir)}
|
|
726
|
+
|
|
727
|
+
process.stderr.write('Connecting to mn serve...\\n');
|
|
728
|
+
await rpcConnect();
|
|
729
|
+
process.stderr.write('Connected\\n');
|
|
730
|
+
|
|
731
|
+
${y3()}
|
|
732
|
+
${S3($.managedDir,$.networkConfig,Z)}
|
|
733
|
+
|
|
734
|
+
import { findDeployedContract } from '@midnight-ntwrk/midnight-js-contracts';
|
|
735
|
+
|
|
736
|
+
process.stderr.write('Finding deployed contract...\\n');
|
|
737
|
+
const deployed = await findDeployedContract(providers, {
|
|
738
|
+
compiledContract: compiled,
|
|
739
|
+
contractAddress: ${JSON.stringify($.contractAddress)},
|
|
740
|
+
privateStateId: ${JSON.stringify(Z)},
|
|
741
|
+
initialPrivateState: makeInitialPrivateState(),
|
|
742
|
+
});
|
|
743
|
+
|
|
744
|
+
process.stderr.write('Calling ${$.circuit}...\\n');
|
|
745
|
+
${k3}
|
|
746
|
+
const args = (${JSON.stringify($.args??[])}).map(coerceArg);
|
|
747
|
+
await deployed.callTx.${$.circuit}(...args);
|
|
748
|
+
|
|
749
|
+
console.log(JSON.stringify({ status: 'success', circuit: ${JSON.stringify($.circuit)} }));
|
|
750
|
+
rpcClose();
|
|
751
|
+
process.exit(0);
|
|
752
|
+
`}function wz($){return`
|
|
753
|
+
import { setNetworkId } from '@midnight-ntwrk/midnight-js-network-id';
|
|
754
|
+
setNetworkId(${JSON.stringify($.networkConfig.networkId.toLowerCase())});
|
|
755
|
+
|
|
756
|
+
import { pathToFileURL } from 'node:url';
|
|
757
|
+
import { resolve } from 'node:path';
|
|
758
|
+
import { indexerPublicDataProvider } from '@midnight-ntwrk/midnight-js-indexer-public-data-provider';
|
|
759
|
+
|
|
760
|
+
const MANAGED_DIR = ${JSON.stringify($.managedDir)};
|
|
761
|
+
|
|
762
|
+
// Load the contract module which exports the ledger() function
|
|
763
|
+
const contractMod = await import(pathToFileURL(resolve(MANAGED_DIR, 'contract', 'index.js')).href);
|
|
764
|
+
|
|
765
|
+
if (typeof contractMod.ledger !== 'function') {
|
|
766
|
+
console.error('Contract module does not export a ledger() function');
|
|
767
|
+
process.exit(1);
|
|
768
|
+
}
|
|
769
|
+
|
|
770
|
+
process.stderr.write('Querying contract state...\\n');
|
|
771
|
+
const provider = indexerPublicDataProvider(
|
|
772
|
+
${JSON.stringify($.networkConfig.indexer)},
|
|
773
|
+
${JSON.stringify($.networkConfig.indexerWS??$.networkConfig.indexer.replace("http","ws"))},
|
|
774
|
+
);
|
|
775
|
+
|
|
776
|
+
const contractState = await provider.queryContractState(${JSON.stringify($.contractAddress)});
|
|
777
|
+
if (!contractState) {
|
|
778
|
+
console.error('No contract found at address ${$.contractAddress}');
|
|
779
|
+
process.exit(1);
|
|
780
|
+
}
|
|
781
|
+
|
|
782
|
+
process.stderr.write('Parsing ledger state...\\n');
|
|
783
|
+
// ledger() expects contractState.data (ChargedState), not the full ContractState
|
|
784
|
+
const state = contractMod.ledger(contractState.data ?? contractState);
|
|
785
|
+
|
|
786
|
+
// Extract scalar fields and map fields
|
|
787
|
+
const fields = {};
|
|
788
|
+
const maps = {};
|
|
789
|
+
|
|
790
|
+
for (const key of Object.keys(state)) {
|
|
791
|
+
const val = state[key];
|
|
792
|
+
|
|
793
|
+
// Check if it's a map-like (has size() and Symbol.iterator)
|
|
794
|
+
if (val && typeof val === 'object' && typeof val.size === 'function') {
|
|
795
|
+
try {
|
|
796
|
+
maps[key] = { size: val.size().toString() };
|
|
797
|
+
} catch {
|
|
798
|
+
maps[key] = { size: '?' };
|
|
799
|
+
}
|
|
800
|
+
} else if (typeof val === 'bigint') {
|
|
801
|
+
fields[key] = val.toString();
|
|
802
|
+
} else if (typeof val === 'string') {
|
|
803
|
+
fields[key] = val;
|
|
804
|
+
} else if (typeof val === 'boolean') {
|
|
805
|
+
fields[key] = String(val);
|
|
806
|
+
} else if (val instanceof Uint8Array) {
|
|
807
|
+
fields[key] = Array.from(val).map(b => b.toString(16).padStart(2, '0')).join('');
|
|
808
|
+
} else if (val !== undefined && val !== null) {
|
|
809
|
+
try { fields[key] = JSON.stringify(val); } catch { fields[key] = '?'; }
|
|
810
|
+
}
|
|
811
|
+
}
|
|
812
|
+
|
|
813
|
+
console.log(JSON.stringify({ fields, maps }));
|
|
814
|
+
process.exit(0);
|
|
815
|
+
`}function hz($){if(!o8)return[];let Z=[];if(!r$($)){try{x3(o8,$),Z.push($)}catch{}return Z}for(let Q of fz){let X=h1(o8,Q),z=h1($,Q);if(!r$(X)||r$(z))continue;try{x3(X,z),Z.push(z)}catch{}}return Z}async function s8($,Z,Q){let X=h1($,"node_modules"),z=hz(X),Y=h1($,`.mn-contract-${Date.now()}.mjs`);vz(Y,Z);let J=()=>{try{E3(Y)}catch{}for(let G of z.slice().reverse())try{E3(G)}catch{}};return new Promise((G,q)=>{let K=Cz("node",[Y],{cwd:$,stdio:["ignore","pipe","pipe"],env:{...process.env,NODE_NO_WARNINGS:"1"}}),B="",H="";K.stdout?.on("data",(U)=>{B+=U.toString()}),K.stderr?.on("data",(U)=>{let j=U.toString().trim();if(j)H+=j+`
|
|
816
|
+
`,Q?.(j)}),K.on("close",(U)=>{if(J(),U!==0){q(new Error(H.trim()||`Script exited with code ${U}`));return}G(B.trim())}),K.on("error",(U)=>{J(),q(U)})})}var o8,k3,fz;var g1=N(()=>{o$();o8=bz();k3=`
|
|
817
|
+
const coerceArg = (${a$.toString()});
|
|
818
|
+
`;fz=["@midnight-ntwrk","ws"]});var w3={};m(w3,{buildStateProvider:()=>gz});import{indexerPublicDataProvider as mz}from"@midnight-ntwrk/midnight-js-indexer-public-data-provider";function gz($){return mz($.indexer,$.indexerWS??$.indexer.replace("http","ws"))}var f3=()=>{};var $2={};m($2,{default:()=>m3});import{resolve as t$}from"node:path";function u1(){return{update(){},stop(){},fail(){},log(){}}}function uz($){return h3.includes($)}async function m3($,Z){let Q=$.subcommand;if(!Q||!uz(Q))throw new t(`Usage: midnight contract <${h3.join("|")}>
|
|
819
|
+
|
|
820
|
+
inspect Show circuits, witnesses, and types for a compiled contract
|
|
821
|
+
deploy Deploy a contract to the network
|
|
822
|
+
call Call a circuit on a deployed contract
|
|
823
|
+
state Read the ledger state of a deployed contract
|
|
824
|
+
|
|
825
|
+
Run "midnight help contract" for more info.`);switch(Q){case"inspect":return pz($);case"deploy":return cz($);case"call":return lz($);case"state":return iz($)}}function e$($){let Z=D($,"managed"),Q=D($,"path");return t$(Z??Q??process.cwd())}async function pz($){let Z=x($,"json"),Q=e$($),X=D($,"name"),{info:z}=j4(Q,X);if(Z){y(f8(z));return}if(process.stderr.write(`
|
|
826
|
+
`+g(`Contract: ${z.name}`)+`
|
|
827
|
+
|
|
828
|
+
`),process.stderr.write(A("Compiler",z.compilerVersion)+`
|
|
829
|
+
`),process.stderr.write(A("Language",z.languageVersion)+`
|
|
830
|
+
`),process.stderr.write(A("Runtime",z.runtimeVersion)+`
|
|
831
|
+
`),process.stderr.write(`
|
|
832
|
+
`+M(" Circuits")+`
|
|
833
|
+
`),z.circuits.length===0)process.stderr.write(V(" (none)")+`
|
|
834
|
+
`);else for(let Y of z.circuits){let J=S8(Y),G=k8(Y),q=Y.pure?f(G):S(G);process.stderr.write(` ${J} ${V("—")} ${q}
|
|
835
|
+
`)}if(process.stderr.write(`
|
|
836
|
+
`+M(" Witnesses")+`
|
|
837
|
+
`),z.witnesses.length===0)process.stderr.write(V(" (none)")+`
|
|
838
|
+
`);else for(let Y of z.witnesses)process.stderr.write(` ${w8(Y)}
|
|
839
|
+
`);if(z.siblings.length>0){process.stderr.write(`
|
|
840
|
+
`+M(" Other contracts in this project")+`
|
|
841
|
+
`);for(let Y of z.siblings)process.stderr.write(` ${f(Y)}
|
|
842
|
+
`);process.stderr.write(`
|
|
843
|
+
`+V(" Inspect a sibling: mn contract inspect --name <name>")+`
|
|
844
|
+
`)}process.stderr.write(`
|
|
845
|
+
`)}async function g3($,Z,Q){let{loadWalletConfig:X,resolveWalletPath:z}=await Promise.resolve().then(() => (F0(),o7)),{resolveNetwork:Y}=await Promise.resolve().then(() => (H0(),i4)),{defaultRepository:J}=await Promise.resolve().then(() => (L4(),jZ)),{NATIVE_TOKEN_TYPE:G}=await Promise.resolve().then(() => Z$),{config:q}=Y({args:{command:"contract",subcommand:void 0,positionals:[],flags:{network:$}}}),K=X(z(Q)),B=Buffer.from(K.seed,"hex"),H=K.addresses[$],U=Z?u1():l("Checking wallet..."),j=J();try{let W=(await j.unshielded(H,q)).balances.get(G)??0n;if(W===0n)throw U.fail("No NIGHT balance"),new Error(`Wallet has 0 NIGHT on ${$}.
|
|
846
|
+
|
|
847
|
+
Fund your wallet before deploying:
|
|
848
|
+
Address: ${H}
|
|
849
|
+
`+($==="undeployed"?` Run: midnight airdrop 1000
|
|
850
|
+
`:` Use the Midnight faucet or transfer from another wallet.
|
|
851
|
+
`));let P=await j.dust(B,q,{onStatus:(O)=>U.update(O)});if(P.balance===0n&&P.availableCoins===0)throw U.fail("No dust available"),new Error(`Wallet has no dust on ${$}. Dust is required to pay transaction fees.
|
|
852
|
+
|
|
853
|
+
Register for dust generation:
|
|
854
|
+
Run: midnight dust register --network ${$}
|
|
855
|
+
Then wait: midnight dust status --network ${$}
|
|
856
|
+
`);U.stop(`Wallet OK (${W} NIGHT, dust available)`)}catch(L){throw U.fail("Wallet check failed"),L}}async function u3($,Z,Q){let{startServeOrReuse:X}=await Promise.resolve().then(() => (l8(),T3)),z=Z?u1():l("Starting mn serve..."),Y=await X({network:$,wallet:Q,onMessage:(J)=>z.update(J)});return z.stop("mn serve ready (port "+Y.port+")"),{port:Y.port,stop:()=>Y.stop()}}async function cz($){let Z=x($,"json"),Q=t$(D($,"path")??process.cwd()),X=e$($),z=D($,"name"),{runDeploy:Y}=await Promise.resolve().then(() => (g1(),s$)),{resolveNetwork:J}=await Promise.resolve().then(() => (H0(),i4));if(!Z)process.stderr.write(`
|
|
857
|
+
`+g("Contract Deploy")+`
|
|
858
|
+
|
|
859
|
+
`);let{info:G}=j4(X,z),q=G.witnesses.map((W)=>W.name);if(q.length>0){let{findWitnessFile:W,buildMissingWitnessError:P}=await Promise.resolve().then(() => (o$(),N3));if(!W(Q))throw new Error(P({projectRoot:Q,witnessNames:q}))}let K=D($,"network")??"undeployed",{config:B}=J({args:{command:"contract",subcommand:"deploy",positionals:[],flags:{network:K}}}),H=D($,"args"),U=[];if(H)try{let W=JSON.parse(H);U=Array.isArray(W)?W:Object.values(W)}catch(W){throw new t(`Invalid --args JSON: ${W.message}`)}if(!Z){if(process.stderr.write(A("Contract",G.name)+`
|
|
860
|
+
`),process.stderr.write(A("Network",K)+`
|
|
861
|
+
`),U.length>0)process.stderr.write(A("Constructor args",String(U.length))+`
|
|
862
|
+
`);process.stderr.write(`
|
|
863
|
+
`)}await g3(K,Z,D($,"wallet"));let j=await u3(K,Z,D($,"wallet")),L=Z?u1():l("Deploying contract...");try{let W=await Y({dappDir:Q,networkConfig:B,managedDir:G.managedDir,contractName:G.name,servePort:j.port,args:U,onMessage:(P)=>L.update(P)});if(L.stop(p("✓")+" Contract deployed"),Z)y({subcommand:"deploy",contractName:G.name,address:W.contractAddress,network:K});else process.stderr.write(`
|
|
864
|
+
`),process.stderr.write(A("Address",W.contractAddress)+`
|
|
865
|
+
`),process.stderr.write(`
|
|
866
|
+
`+p(" ✓ Deploy successful")+`
|
|
867
|
+
`),dz(W.contractAddress,G.circuits,K),process.stdout.write(W.contractAddress+`
|
|
868
|
+
`)}catch(W){throw L.fail("Deploy failed"),W}finally{await j.stop()}}function dz($,Z,Q){let X=Q==="undeployed"?"":` --network ${Q}`;process.stderr.write(`
|
|
869
|
+
`+V(" Next:")+`
|
|
870
|
+
`),process.stderr.write(V(" mn contract state --address ")+f($)+V(X)+`
|
|
871
|
+
`);let z=Z.find((Y)=>Y.arguments.length===0);if(z)process.stderr.write(V(" mn contract call --address ")+f($)+V(" --circuit ")+f(z.name)+V(X)+`
|
|
872
|
+
`);process.stderr.write(`
|
|
873
|
+
`)}async function lz($){let Z=x($,"json"),Q=t$(D($,"path")??process.cwd()),X=e$($),z=D($,"name"),Y=S4($,"address","contract address"),J=S4($,"circuit","circuit name"),G=D($,"args"),{runCall:q}=await Promise.resolve().then(() => (g1(),s$)),{resolveNetwork:K}=await Promise.resolve().then(() => (H0(),i4));if(!Z)process.stderr.write(`
|
|
874
|
+
`+g("Contract Call")+`
|
|
875
|
+
|
|
876
|
+
`);let{info:B}=j4(X,z),H=[];if(G)try{let P=JSON.parse(G);H=Array.isArray(P)?P:Object.values(P)}catch(P){throw new t(`Invalid --args JSON: ${P.message}`)}let U=D($,"network")??"undeployed",{config:j}=K({args:{command:"contract",subcommand:"call",positionals:[],flags:{network:U}}});if(!Z)process.stderr.write(A("Contract",B.name)+`
|
|
877
|
+
`),process.stderr.write(A("Circuit",J)+`
|
|
878
|
+
`),process.stderr.write(A("Address",Y.slice(0,20)+"...")+`
|
|
879
|
+
|
|
880
|
+
`);await g3(U,Z,D($,"wallet"));let L=await u3(U,Z,D($,"wallet")),W=Z?u1():l(`Calling ${J}...`);try{let P=await q({dappDir:Q,networkConfig:j,managedDir:B.managedDir,contractName:B.name,contractAddress:Y,circuit:J,args:H,servePort:L.port,onMessage:(O)=>W.update(O)});if(W.stop(p("✓")+` ${J} called`),Z)y({subcommand:"call",contractName:B.name,circuit:J,address:Y,network:U,status:P.status});else process.stderr.write(`
|
|
881
|
+
`+p(" ✓ Circuit call successful")+`
|
|
882
|
+
|
|
883
|
+
`)}catch(P){throw W.fail("Call failed"),P}finally{await L.stop()}}async function iz($){let Z=x($,"json"),Q=t$(D($,"path")??process.cwd()),X=e$($),z=D($,"name"),Y=S4($,"address","contract address"),{resolveNetwork:J}=await Promise.resolve().then(() => (H0(),i4)),{runState:G}=await Promise.resolve().then(() => (g1(),s$)),q=D($,"network")??"undeployed",{config:K}=J({args:{command:"contract",subcommand:"state",positionals:[],flags:{network:q}}});if(!Z)process.stderr.write(`
|
|
884
|
+
`+g("Contract State")+`
|
|
885
|
+
|
|
886
|
+
`),process.stderr.write(A("Address",Y.slice(0,20)+"...")+`
|
|
887
|
+
`),process.stderr.write(A("Network",q)+`
|
|
888
|
+
|
|
889
|
+
`);let B=Z?u1():l("Querying contract state...");try{let{info:H}=j4(X,z),U=await G({dappDir:Q,networkConfig:K,managedDir:H.managedDir,contractName:H.name,contractAddress:Y,onMessage:(j)=>B.update(j)});if(B.stop(p("✓")+" State retrieved"),Z)y({subcommand:"state",address:Y,network:q,...U});else{let j=Object.keys(U.fields).length>0,L=Object.keys(U.maps).length>0;if(j||L){process.stderr.write(M(" Ledger State")+`
|
|
890
|
+
`);for(let[W,P]of Object.entries(U.fields))process.stderr.write(` ${W}: ${f(P)}
|
|
891
|
+
`);for(let[W,P]of Object.entries(U.maps))process.stderr.write(` ${W}: ${V(`Map (${P.size} entries)`)}
|
|
892
|
+
`)}else process.stderr.write(V(" (no ledger fields found)")+`
|
|
893
|
+
`);process.stderr.write(`
|
|
894
|
+
`)}}catch(H){B.stop(S("⚠")+" Parsed state unavailable, showing raw");let{buildStateProvider:U}=await Promise.resolve().then(() => (f3(),w3)),L=await U(K).queryContractState(Y);if(!L)throw new Error(`No contract found at address ${Y} on ${q}`);let W=L.data??L;if(Z)y({subcommand:"state",address:Y,network:q,raw:W});else process.stderr.write(M(" Raw State")+`
|
|
895
|
+
`),process.stderr.write(V(" (run from dApp root for parsed ledger fields)")+`
|
|
896
|
+
`),process.stderr.write(` ${JSON.stringify(W).slice(0,200)}
|
|
897
|
+
|
|
898
|
+
`)}}var h3;var Z2=N(()=>{l0();K0();u0();k1();h3=["inspect","deploy","call","state"]});import{existsSync as z1,readFileSync as p1,readdirSync as nz}from"node:fs";import{join as v4,resolve as oz}from"node:path";function Q2($){let Z=oz($??process.cwd()),Q=v4(Z,t8);if(!z1(Q))throw new Error(`No ${t8} found in ${Z}
|
|
899
|
+
Run this command from the root of a dApp project, or create a ${t8} file.`);let X;try{X=JSON.parse(p1(Q,"utf-8"))}catch(Y){throw new Error(`Failed to parse ${Q}: ${Y.message}`)}return{config:sz(X,Q),configPath:Q,dappDir:Z}}function e8($){let Z=v4($,az);if(!z1(Z))return[];let Q=nz(Z,{withFileTypes:!0}),X=[];for(let z of Q){if(!z.isDirectory())continue;let Y=v4(Z,z.name),J=v4(Y,"suite.json");if(!z1(J))continue;let G;try{G=JSON.parse(p1(J,"utf-8"))}catch(q){throw new Error(`Failed to parse ${J}: ${q.message}`)}X.push({suite:tz(G,J),suiteDir:Y})}return X}function i3($){let Z=v4($,"assertions.json");if(!z1(Z))return null;let Q;try{Q=JSON.parse(p1(Z,"utf-8"))}catch(X){throw new Error(`Failed to parse ${Z}: ${X.message}`)}return ez(Q,Z)}function n3($){let Z=v4($,"actions.json");if(!z1(Z))return null;let Q;try{Q=JSON.parse(p1(Z,"utf-8"))}catch(X){throw new Error(`Failed to parse ${Z}: ${X.message}`)}return $Y(Q,Z)}function o3($){let Z=v4($,"prompt.md");if(!z1(Z))return null;return p1(Z,"utf-8")}function sz($,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&&!p3.includes(Q.network))throw new Error(`${Z}: "network" must be one of: ${p3.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"||!rz.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 tz($,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(!c3.includes(Q.strategy))throw new Error(`${Z}: "strategy" must be one of: ${c3.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&&!d3.includes(Q.browserMode))throw new Error(`${Z}: "browserMode" must be one of: ${d3.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 ez($,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 $Y($,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"||!l3.includes(z.type))throw new Error(`${Z}: action "${z.id}" has invalid type "${z.type}". Valid: ${l3.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 t8="dapp.test.json",az="tests/suites",p3,c3,d3,rz,l3;var a3=N(()=>{p3=["undeployed","preprod","preview"],c3=["browser","cli","hybrid"],d3=["dom","script","vision","auto"],rz=/^(cache-clear|localnet-up|balance:\d+|dust|dust-register|dust-wait|mn-serve|build-and-serve)$/;l3=["contract-deploy","contract-call","contract-state","wallet-cmd"]});function r3(){let $=[];return{cleanups:$,addCleanup(Z){$.push(Z)}}}import{spawn as ZY}from"node:child_process";import{createWriteStream as QY}from"node:fs";import{join as XY}from"node:path";import zY from"node:http";import YY from"node:net";async function t3($){let{dappDir:Z,buildCmd:Q,buildDir:X=".",port:z,url:Y,logFile:J,timeoutMs:G=240000,onMessage:q=()=>{}}=$;if(await JY(z)){if(await s3(Y))return q(`Reusing existing dev server on port ${z} (URL responds 200)`),{port:z,child:{kill:()=>!0},stop(){}};q(`Port ${z} in use by an unresponsive process — killing it`),await GY(z),await new Promise((W)=>setTimeout(W,2000))}let K=XY(Z,X),B=QY(J,{flags:"a"}),H=ZY("bash",["-lc",Q],{cwd:K,stdio:["ignore","pipe","pipe"],detached:!1});H.stdout?.pipe(B),H.stderr?.pipe(B);let U=!1,j=null;H.on("exit",(W)=>{U=!0,j=W}),q(`Building (${Q})...`);let L=Date.now()+G;while(Date.now()<L){if(U)throw new Error(`Build process exited with code ${j} before the UI was ready.
|
|
900
|
+
Check the build log: ${J}`);if(await s3(Y))return q("UI ready"),{port:z,child:H,stop(){if(!U)H.kill("SIGTERM"),setTimeout(()=>{if(!U)H.kill("SIGKILL")},5000).unref();B.end()}};await new Promise((W)=>setTimeout(W,2000))}throw H.kill("SIGTERM"),B.end(),new Error(`UI did not start within ${Math.round(G/1000)}s at ${Y}
|
|
901
|
+
Check the build log: ${J}`)}function JY($){return new Promise((Z)=>{let Q=new YY.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 GY($){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 s3($){return new Promise((Z)=>{let Q=zY.get($,{timeout:3000},(X)=>{Z(X.statusCode===200),X.resume()});Q.on("error",()=>Z(!1)),Q.on("timeout",()=>{Q.destroy(),Z(!1)})})}var e3=()=>{};import*as X5 from"@midnight-ntwrk/ledger-v8";import{execSync as qY}from"node:child_process";import{join as KY}from"node:path";async function z5($,Z,Q,X){let z=[];for(let Y of $.prep){let J=Date.now();X.onStepStart(Y);try{await VY(Y,$,Z,Q,X);let G=Date.now()-J;z.push({step:Y,status:"pass",duration:G}),X.onStepComplete(Y,"pass",G)}catch(G){let q=Date.now()-J,K=G.message;throw z.push({step:Y,status:"fail",duration:q,error:K}),X.onStepComplete(Y,"fail",q,K),new Error(`Prep step "${Y}" failed: ${K}`)}}return z}async function VY($,Z,Q,X,z){if($==="cache-clear")return BY(Z);if($==="localnet-up")return UY(z);if($.startsWith("balance:")){let Y=parseInt($.split(":")[1],10);return LY(Y,Z,z)}if($==="dust"||$==="dust-register"||$==="dust-wait")return OY(Z,z);if($==="mn-serve")return jY(Z,X,z);if($==="build-and-serve")return PY(Z,Q,X,z);throw new Error(`Unknown prep step: ${$}`)}async function BY($){let Z=$.network??"undeployed";r0(void 0,Z)}function WY(){try{return qY("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 Z5($,Z){let Q=Date.now();while(Date.now()-Q<$){if(WY())return!0;Z?.("Waiting for indexer...");let{execSync:X}=X4("node:child_process");try{X("sleep 2",{timeout:3000})}catch{}}return!1}async function UY($){S$(),Z1();let Z=["node","indexer","proof-server"],Q=Q4();if(Q.length>=3&&Z.every((J)=>Q.some((G)=>G.name===J&&G.state==="running"))&&Q.every((J)=>!J.health||J.health==="healthy")){if(Z5(1e4,(J)=>$.onMessage(J))){$.onMessage("Localnet OK (3 services healthy, indexer responding)");return}$.onMessage("Localnet containers running but indexer not responding. Restarting...")}let z=Q.filter((J)=>J.state==="running").map((J)=>J.name),Y=Z.filter((J)=>!z.includes(J));if(Y.length>0)$.onMessage(`Localnet missing: ${Y.join(", ")}. Restarting clean...`);else $.onMessage("Localnet unhealthy. Restarting clean...");for(let J=1;J<=X2;J++){$.onMessage(`Attempt ${J}/${X2}: tearing down...`),p0("down"),$.onMessage(`Attempt ${J}/${X2}: starting...`),p0("up -d");let G=Q1(HY);if(G&&Z5($5,(U)=>$.onMessage(U))){$.onMessage("Localnet running (indexer verified)");return}if(G){$.onMessage(`Attempt ${J}: Docker healthy but indexer not responding after ${$5/1000}s`);continue}let q=Q4(),K=Z.filter((U)=>!q.some((j)=>j.name===U&&j.state==="running")),B=q.filter((U)=>U.state==="running"&&U.health&&U.health!=="healthy"),H=[];if(K.length>0)H.push(`not running: ${K.join(", ")}`);if(B.length>0)H.push(`unhealthy: ${B.map((U)=>U.name).join(", ")}`);$.onMessage(`Attempt ${J} failed: ${H.join("; ")||"timeout"}`)}throw new Error(`Localnet failed to start after ${X2} attempts`)}async function LY($,Z,Q){let X=Z.network??"undeployed",{config:z}=V0({args:{command:"test",subcommand:void 0,positionals:[],flags:{network:X}}}),Y=Y0(z0()),J=Buffer.from(Y.seed,"hex"),G=Y.addresses[X];Q.onMessage("Checking balance...");let q=U0((L,W)=>{Q.onMessage(`SDK: ${W}`)}),K=P0(),B=Z4(G,X),H=await H4(J,z,B);try{let L=await W4(H,{syncMode:"lite",onProgress:(O,F)=>{if(F>0){let I=Math.min(Math.round(O/F*100),100);Q.onMessage(`Syncing wallet... ${I}%`)}},onSyncDetail:(O)=>{Q.onMessage(`Syncing wallet... (${O})`)}}),W=X5.unshieldedToken().raw,P=L.unshielded.balances[W]??0n;if(P>0n){Q.onMessage(`Balance OK: ${P}`);try{await S0(G,X,H.facade)}catch{}return}}finally{K(),q();try{await n0(H)}catch{}}if(X!=="undeployed")throw new Error(`Wallet has 0 NIGHT on ${X}. Fund your wallet before running tests:
|
|
902
|
+
mn airdrop ${$} (if faucet available)
|
|
903
|
+
Or transfer NIGHT from another wallet.`);Q.onMessage(`Balance is 0. Airdropping ${$} NIGHT from genesis...`);let U=Buffer.from(J4,"hex"),j=y0(U,X);await $1({seedBuffer:U,networkConfig:z,recipientAddress:G,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 OY($,Z){let Q=$.network??"undeployed",{config:X}=V0({args:{command:"test",subcommand:void 0,positionals:[],flags:{network:Q}}}),z=Y0(z0()),Y=Buffer.from(z.seed,"hex"),J=z.addresses[Q],G=U0(),q=P0(),K=Z4(J,Q),B=await H4(Y,X,K);try{if(Z.onMessage("Syncing wallet..."),await W4(B,{syncMode:"lite"}),Z.onMessage("Ensuring dust..."),(await s0(B,(L)=>Z.onMessage(`Dust: ${L}`))).alreadyAvailable){Z.onMessage("Dust already available");try{await S0(J,Q,B.facade)}catch{}return}Z.onMessage("Dust registered. Waiting for coins to become available...");let U=await e4(B,Q5);if(!(()=>{try{let L=U.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:
|
|
904
|
+
mn dust register
|
|
905
|
+
mn dust status (check until dustAvailable: true)`);throw new Error(`Dust not available after ${Q5/1000}s. The chain may be slow — try again.`)}Z.onMessage("Dust is available");try{await S0(J,Q,B.facade)}catch{}}finally{q(),G();try{await n0(B)}catch{}}}async function jY($,Z,Q){Q.onMessage("Starting mn serve...");let X=await d8({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 PY($,Z,Q,X){if(!$.buildCmd){X.onMessage("No buildCmd in config, skipping build");return}let z=$.port??4173,Y=$.url??`http://localhost:${z}/`,J=KY(Z,"tests","results",`build_${Date.now()}.log`),G=await t3({dappDir:Z,buildCmd:$.buildCmd,buildDir:$.buildDir,port:z,url:Y,logFile:J,onMessage:(q)=>X.onMessage(`[build] ${q}`)});Q.buildHandle=G,Q.addCleanup(async()=>G.stop())}var HY=30000,X2=2,$5=30000,Q5=90000;var Y5=N(()=>{U4();F0();H0();k$();o0();U4();O4();i0();l8();e3()});import{execSync as J5,spawn as FY}from"node:child_process";import{createWriteStream as IY,existsSync as _Y,mkdirSync as TY,readFileSync as MY}from"node:fs";import{dirname as RY,join as AY}from"node:path";import{homedir as DY}from"node:os";function NY(){let $=AY(DY(),".claude.json");if(!_Y($))return!1;try{let Z=JSON.parse(MY($,"utf-8"));return Boolean(Z.mcpServers?.["chrome-devtools"])}catch{return!1}}function xY($){let Z;try{Z=J5(`ps -ax -o pid,command | grep "${EY}" | grep -v grep | awk '{print $1}'`,{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim()}catch{return}if(!Z)return;let Q=Z.split(`
|
|
906
|
+
`).filter(Boolean);$(`Found ${Q.length} orphan chrome-devtools-mcp Chrome process(es), killing`);for(let X of Q)try{J5(`kill ${X}`,{stdio:"pipe"})}catch{}}function $7($){let Z=$.browserMode??"vision";if(Z==="auto")return"vision";return Z}async function G5($){let{suite:Z,prompt:Q,dappDir:X,logFile:z,onMessage:Y=()=>{}}=$,J=(Z.timeout??600)*1000,G=$7(Z);if(G==="dom"||G==="script"){if(!NY())throw new Error(`Browser mode "${G}" requires the chrome-devtools MCP server, which is not registered in ~/.claude.json.
|
|
907
|
+
|
|
908
|
+
Install it:
|
|
909
|
+
claude mcp add chrome-devtools --scope user npx chrome-devtools-mcp@latest
|
|
910
|
+
|
|
911
|
+
Then re-run the suite. Alternatively, switch the suite to "browserMode": "vision" `+"(no MCP needed, but slower — uses screenshots).");xY(Y)}TY(RY(z),{recursive:!0});let q=IY(z),K=[];if(G==="vision")K.push("--chrome");if(K.push("--dangerously-skip-permissions"),Z.model)K.push("--model",Z.model);if(Z.effort)K.push("--effort",Z.effort);let B=Q;if(G==="dom")B=CY+Q;else if(G==="script")B=vY+Q;return K.push("-p",B),Y(`Launching Claude (mode: ${G}, model: ${Z.model??"default"}, timeout: ${Z.timeout??600}s)`),new Promise((H)=>{let U=FY("claude",K,{cwd:X,stdio:["inherit","pipe","inherit"]}),j=!1;U.stdout?.on("data",(W)=>{process.stdout.write(W),q.write(W)});let L=setTimeout(()=>{j=!0,Y(`Timeout: test exceeded ${Z.timeout??600}s`),U.kill("SIGTERM"),setTimeout(()=>{if(!U.killed)U.kill("SIGKILL")},1e4).unref()},J);U.on("close",(W)=>{clearTimeout(L),q.end(),H({exitCode:W??1,logFile:z,timedOut:j})}),U.on("error",(W)=>{clearTimeout(L),q.end(),Y(`Claude process error: ${W.message}`),H({exitCode:1,logFile:z,timedOut:!1})})})}var EY="chrome-devtools-mcp/chrome-profile",CY="## 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",vY="## 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 q5=()=>{};import{existsSync as yY,readFileSync as bY}from"node:fs";import SY from"node:net";async function K5($,Z){let Q=[];for(let X of $)try{let z=await kY(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 kY($,Z){switch($.type){case"balance-changed":return wY($,Z);case"process-exit-code":return fY($,Z);case"port-listening":return hY($);case"agent-report-no-failure":return gY($,Z);default:throw new Error(`Unsupported assertion type: "${$.type}" (available in v2)`)}}function wY($,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 fY($,Z){if(Z.processExitCode===void 0)throw new Error("Process exit code not available");let Q=$.params.code;return Z.processExitCode===Q}function hY($){let Z=$.params.port;return new Promise((Q)=>{let X=new SY.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 gY($,Z){let Q=$.params??{},X=Q.logPath??Z.agentLogPath;if(!X||!yY(X))return!0;let z;try{z=bY(X,"utf-8")}catch{return!0}let Y=pY(z),J=Q.markers??mY;for(let G of J)if(uY(Y,G))return!1;return!0}function uY($,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 pY($){return $.replace(/```[\s\S]*?```/g,"").replace(/`[^`]*`/g,"")}var mY;var V5=N(()=>{mY=["FAILED","❌","did not pass","test failed","overall: fail","result: fail"]});import{existsSync as cY,mkdirSync as dY,writeFileSync as lY,readFileSync as iY,readdirSync as nY}from"node:fs";import{join as z2}from"node:path";function H5($,Z){let Q=z2(Z,B5);dY(Q,{recursive:!0});let X=`${$.suite}_${$.timestamp.replace(/[:.]/g,"-")}.json`,z=z2(Q,X);return lY(z,JSON.stringify($,null,2)+`
|
|
912
|
+
`),z}function W5($,Z){let Q=Z7($,Z);return Q.length>0?Q[0]:null}function Z7($,Z){let Q=z2($,B5);if(!cY(Q))return[];let X=nY(Q).filter((Y)=>Y.endsWith(".json")).filter((Y)=>!Z||Y.startsWith(Z+"_")),z=[];for(let Y of X)try{let J=iY(z2(Q,Y),"utf-8");z.push(JSON.parse(J))}catch{}return z.sort((Y,J)=>J.timestamp.localeCompare(Y.timestamp)),z}var B5="tests/results";var U5=()=>{};import{execSync as oY}from"node:child_process";async function Q7($,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)aY($.buildHandle.port,Z)}function aY($,Z){try{oY(`osascript -e 'tell application "Google Chrome"
|
|
913
|
+
repeat with theWindow in every window
|
|
914
|
+
repeat with theTab in every tab of theWindow
|
|
915
|
+
if URL of theTab contains "localhost:${$}" then
|
|
916
|
+
close theTab
|
|
917
|
+
end if
|
|
918
|
+
end repeat
|
|
919
|
+
end repeat
|
|
920
|
+
end tell'`,{timeout:5000,stdio:"pipe"}),Z?.(`Closed Chrome tabs for localhost:${$}`)}catch{}}var L5=()=>{};var _5={};m(_5,{placeholderArgsFor:()=>Y2,placeholderArg:()=>X7,buildScaffold:()=>ZJ,buildPromptMarkdown:()=>I5});function X7($){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(X7);default:return null}}function Y2($){let Z={};for(let Q of $.arguments)Z[Q.name]=X7(Q.type);return Z}function ZJ($,Z){if((Z.strategy??"cli")==="browser"){if(!Z.browser)throw new Error("Browser strategy requires browser options (port, buildCmd, buildDir?, url?)");return XJ(Z,Z.browser)}return QJ($,Z)}function QJ($,Z){let Q=Z.suiteName??rY,X=Z.network??O5,z=Z.servePort??j5;return{dappConfig:{name:Z.contractName,network:X,prep:P5},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??tY},actions:{actions:zJ($)},assertions:{post:[F5(z)]},prompt:null,suiteName:Q}}function XJ($,Z){let Q=$.suiteName??sY,X=$.network??O5,z=$.servePort??j5,Y=Z.url??`http://localhost:${Z.port}/`,J={name:$.contractName,network:X,port:Z.port,buildCmd:Z.buildCmd,url:Y,prep:$J};if(Z.buildDir)J.buildDir=Z.buildDir;return{dappConfig:J,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??eY,...Z.browserMode?{browserMode:Z.browserMode}:{}},actions:null,assertions:{post:[{id:"claude-exit-ok",type:"process-exit-code",params:{code:0},expect:"pass"},F5(z),{id:"agent-no-failure",type:"agent-report-no-failure",params:{},expect:"pass"}]},prompt:I5($.contractName,Y),suiteName:Q}}function F5($){return{id:"serve-port-listening",type:"port-listening",params:{port:$},expect:"pass"}}function I5($,Z){return`You are running an automated E2E test of ${$}.
|
|
921
|
+
|
|
922
|
+
The UI is at ${Z} — open it in Chrome.
|
|
923
|
+
mn serve is running with --approve-all so contract calls will auto-approve.
|
|
924
|
+
|
|
925
|
+
> TODO: replace the steps below with the real user flow for your dApp.
|
|
926
|
+
> Keep them numbered — Claude follows numbered lists more reliably than prose.
|
|
927
|
+
> Use specific text Claude can find on screen ("WALLET OK", "Submit", etc.)
|
|
928
|
+
> rather than vague phrases ("the wallet button").
|
|
929
|
+
|
|
930
|
+
Follow these steps:
|
|
931
|
+
|
|
932
|
+
1. Open ${Z} in Chrome.
|
|
933
|
+
2. Wait for the wallet connection indicator to show success
|
|
934
|
+
(TODO: name the exact text/element your dApp shows on connect).
|
|
935
|
+
3. TODO: dApp-specific actions. Examples:
|
|
936
|
+
- Click the button labelled "Deploy".
|
|
937
|
+
- Fill the input named "amount" with 100.
|
|
938
|
+
- Press Enter to submit.
|
|
939
|
+
4. Wait for the on-chain confirmation.
|
|
940
|
+
Contract calls take 30–60 seconds because of ZK proof generation.
|
|
941
|
+
5. Verify the result on screen
|
|
942
|
+
(TODO: name what your dApp shows after success — a tx hash, a state change, etc.).
|
|
943
|
+
6. Report:
|
|
944
|
+
- Did each step succeed?
|
|
945
|
+
- What was the final on-screen state?
|
|
946
|
+
- Any errors?
|
|
947
|
+
|
|
948
|
+
If any step fails, report exactly what you see on screen and stop.
|
|
949
|
+
Take screenshots at each major step to document the test.
|
|
950
|
+
`}function zJ($){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=Y2(X);Z.push(z)}return Z.push({id:"check-final",type:"contract-state"}),Z}var O5="undeployed",rY="cli-default",sY="ui-default",tY=300,eY=600,j5=9932,P5,$J;var z7=N(()=>{P5=["cache-clear","localnet-up","balance:1000","dust","mn-serve"],$J=[...P5,"build-and-serve"]});var R5={};m(R5,{writeScaffold:()=>GJ});import{existsSync as M5,mkdirSync as YJ,readFileSync as JJ,writeFileSync as T5}from"node:fs";import{join as Y1}from"node:path";function GJ($,Z){let{dappDir:Q,force:X=!1}=Z,z=Y1(Q,"tests","suites",$.suiteName),Y=Y1(Q,"dapp.test.json"),J=[{path:Y1(z,"suite.json"),kind:"json",body:$.suite},{path:Y1(z,"assertions.json"),kind:"json",body:$.assertions}];if($.actions)J.push({path:Y1(z,"actions.json"),kind:"json",body:$.actions});if($.prompt)J.push({path:Y1(z,"prompt.md"),kind:"text",body:$.prompt});if(!X){let B=J.filter((H)=>M5(H.path)).map((H)=>H.path);if(B.length>0)throw new Error(`Suite "${$.suiteName}" already exists at ${z}.
|
|
951
|
+
Pick a different suite name (--suite <name>) or pass --force to overwrite.
|
|
952
|
+
Existing files:
|
|
953
|
+
`+B.map((H)=>` - ${H}`).join(`
|
|
954
|
+
`))}let G=qJ(Y,$,X);YJ(z,{recursive:!0});let q=[],K=[];if(G.action==="write")T5(Y,JSON.stringify($.dappConfig,null,2)+`
|
|
955
|
+
`),q.push(Y);else K.push(Y);for(let B of J){let H=B.kind==="json"?JSON.stringify(B.body,null,2)+`
|
|
956
|
+
`:B.body;T5(B.path,H),q.push(B.path)}return{written:q,preserved:K}}function qJ($,Z,Q){if(Q||!M5($))return{action:"write"};let X;try{X=JSON.parse(JJ($,"utf-8"))}catch(z){let Y=z instanceof Error?z.message:String(z);throw new Error(`${$} exists but is not valid JSON: ${Y}
|
|
957
|
+
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 A5=()=>{};var x5={};m(x5,{discoverScreensInDir:()=>Y7,discoverScreens:()=>OJ});import{existsSync as N5,readdirSync as KJ,readFileSync as VJ,statSync as BJ}from"node:fs";import{basename as HJ,extname as WJ,join as E5,relative as UJ}from"node:path";function OJ($){let Z=new Map;for(let Q of LJ){let X=E5($,Q);if(!N5(X))continue;J7(X,$,Z)}return[...Z.values()].sort((Q,X)=>Q.name.localeCompare(X.name))}function Y7($,Z=$){if(!N5($))return[];let Q=new Map;return J7($,Z,Q),[...Q.values()].sort((X,z)=>X.name.localeCompare(z.name))}function J7($,Z,Q){let X;try{X=KJ($)}catch{return}for(let z of X){let Y=E5($,z),J;try{J=BJ(Y)}catch{continue}if(J.isDirectory()){if(jJ.has(z.toLowerCase()))continue;J7(Y,Z,Q);continue}if(WJ(z)!==D5)continue;if(z.endsWith(".test.tsx")||z.endsWith(".spec.tsx"))continue;let G=HJ(z,D5);if(!PJ(G,Y))continue;let q=IJ(G);if(Q.has(q))continue;Q.set(q,{name:q,component:G,path:Y,relativePath:UJ(Z,Y)})}}function PJ($,Z){if(!/^[A-Z][A-Za-z0-9]*$/.test($))return!1;let Q;try{Q=VJ(Z,"utf-8")}catch{return!1}return FJ.test(Q)}function IJ($){return $.replace(/([a-z0-9])([A-Z])/g,"$1-$2").replace(/([A-Z]+)([A-Z][a-z])/g,"$1-$2").toLowerCase()}var LJ,D5=".tsx",jJ,FJ;var G7=N(()=>{LJ=["src/components","src/pages","src/screens","src/views"];jJ=new Set(["layout","layouts","shared","common","ui","icons","styles"]);FJ=/\bexport\s+(default\b|const\s+[A-Z]|function\s+[A-Z])/});var b5={};m(b5,{promptSuiteName:()=>xJ,promptStrategy:()=>RJ,promptScreen:()=>y5,promptGoal:()=>EJ,promptCircuit:()=>DJ,promptBrowserOptions:()=>AJ,promptBrowserMode:()=>v5,isInteractive:()=>MJ});import{existsSync as _J,statSync as TJ}from"node:fs";import*as C5 from"node:readline";function MJ(){return process.stdin.isTTY===!0}function d0($,Z){let Q=C5.createInterface({input:process.stdin,output:process.stderr}),X=Z!==void 0?V(` [${Z}]`):"";return new Promise((z)=>{Q.question(` ${$}${X}: `,(Y)=>{Q.close();let J=Y.trim();z(J===""&&Z!==void 0?Z:J)})})}async function RJ(){process.stderr.write(`
|
|
958
|
+
`+f("Test scaffold setup")+`
|
|
959
|
+
`);for(;;){let $=(await d0("Strategy? cli or ui","cli")).toLowerCase();if($==="cli")return"cli";if($==="ui"||$==="browser")return"browser";process.stderr.write(V(` Unknown strategy "${$}". Pick "cli" or "ui".
|
|
960
|
+
`))}}async function AJ($={}){process.stderr.write(`
|
|
961
|
+
`+V("Browser test scaffold needs your dApp UI details:")+`
|
|
962
|
+
`);let Z=$.port!==void 0?String($.port):await d0("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 d0("Build command","npm run dev");if(!X)throw new Error("Build command is required for browser strategy.");let z=$.buildDir!==void 0?$.buildDir:await d0("Build dir (blank for project root)",""),Y=z===""?void 0:z,J=$.url??await d0("URL",`http://localhost:${Q}/`),G=$.browserMode??await v5();return{port:Q,buildCmd:X,buildDir:Y,url:J,browserMode:G}}async function v5(){process.stderr.write(`
|
|
963
|
+
`+f("Browser mode — how should Claude perceive the page?")+`
|
|
964
|
+
`),process.stderr.write(` ${V("1.")} dom ${V("— accessibility tree (text). Fast. HTML / React / Vue UIs.")}
|
|
965
|
+
`),process.stderr.write(` ${V("2.")} vision ${V("— screenshots. Slow. Canvas games (no DOM).")}
|
|
966
|
+
`),process.stderr.write(` ${V("3.")} script ${V("— direct JS. Fastest. Needs dApp-side hooks (advanced).")}
|
|
967
|
+
`),process.stderr.write(V(` dom and script need chrome-devtools-mcp installed.
|
|
968
|
+
`));for(;;){let $=(await d0("Mode","dom")).toLowerCase().trim();if($==="1"||$==="dom")return"dom";if($==="2"||$==="vision")return"vision";if($==="3"||$==="script")return"script";process.stderr.write(S(` Pick 1 (dom), 2 (vision), or 3 (script).
|
|
969
|
+
`))}}async function DJ($){let Z=$.filter((Q)=>!Q.pure);if(Z.length===0){process.stderr.write(V(` No impure circuits in this contract — nothing to test as a write path.
|
|
970
|
+
`));return}process.stderr.write(`
|
|
971
|
+
`+f("Pick the circuit this suite will exercise:")+`
|
|
972
|
+
`);for(let Q=0;Q<Z.length;Q++)process.stderr.write(` ${V(`${Q+1}.`)} ${Z[Q].name}
|
|
973
|
+
`);process.stderr.write(V(` (or type "skip" to scaffold all circuits without AI)
|
|
974
|
+
`));for(;;){let Q=(await d0("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(V(` Pick a number 1–${Z.length}, the circuit name, or "skip".
|
|
975
|
+
`))}}async function y5($){if($.length===0)return NJ();process.stderr.write(`
|
|
976
|
+
`+f("Pick the screen this suite will test:")+`
|
|
977
|
+
`);for(let Z=0;Z<$.length;Z++)process.stderr.write(` ${V(`${Z+1}.`)} ${$[Z].name} ${V(`(${$[Z].relativePath})`)}
|
|
978
|
+
`);process.stderr.write(V(` (or type "skip" to scaffold a generic prompt.md without AI)
|
|
979
|
+
`));for(;;){let Z=(await d0("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(V(` Pick a number 1–${$.length}, the screen name, or "skip".
|
|
980
|
+
`))}}async function NJ(){process.stderr.write(V(`
|
|
981
|
+
Could not auto-discover screens (no src/components, src/pages, or src/screens under the build dir).
|
|
982
|
+
`));for(;;){let $=(await d0('Path to a .tsx file or a UI directory (or "skip")')).trim();if($.toLowerCase()==="skip"||$==="")return;if(!_J($)){process.stderr.write(V(` No such path: ${$}
|
|
983
|
+
`));continue}let Z;try{Z=TJ($)}catch{process.stderr.write(V(` Cannot stat: ${$}
|
|
984
|
+
`));continue}if(Z.isDirectory()){let Q=Y7($);if(Q.length===0){process.stderr.write(V(` No PascalCase .tsx components in ${$}
|
|
985
|
+
`));continue}return y5(Q)}if(!$.endsWith(".tsx")){process.stderr.write(V(` Not a .tsx file: ${$}
|
|
986
|
+
`));continue}return vJ($)}}async function EJ(){let $=(await d0("What does success look like? (one line, optional)")).trim();return $===""?void 0:$}async function xJ($){for(;;){let Z=(await d0("Suite name",$)).trim();if(CJ(Z))return Z;process.stderr.write(S(` Suite names must be kebab-case (a–z, 0–9, dashes, underscores). Try again.
|
|
987
|
+
`))}}function CJ($){return/^[a-z0-9][a-z0-9_-]{0,63}$/.test($)}function vJ($){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 S5=N(()=>{G7()});function f5($){let{contractName:Z,contractSummary:Q,contractSource:X,targetCircuit:z,goal:Y,witnessDependentCircuits:J}=$,G=J&&J.size>0?`
|
|
988
|
+
## Circuits NOT testable from a CLI suite
|
|
989
|
+
|
|
990
|
+
These circuits call witnesses that read from private state. In a
|
|
991
|
+
CLI test the private state is empty (no UI to populate it), so
|
|
992
|
+
they crash with "Cannot read properties of undefined". Do NOT
|
|
993
|
+
include them as actions, even as setup steps:
|
|
994
|
+
|
|
995
|
+
`+[...J.entries()].map(([B,H])=>` - ${B} (uses ${H.join(", ")})`).join(`
|
|
996
|
+
`)+`
|
|
997
|
+
${z&&J.has(z.name)?`
|
|
998
|
+
The user explicitly asked for "${z.name}" — try anyway,
|
|
999
|
+
`+`but expect a runtime witness-undefined error and call that out
|
|
1000
|
+
in the description so the user knows the suite needs the UI to
|
|
1001
|
+
populate state first.
|
|
1002
|
+
`:""}`:"",q=z?`## Target
|
|
1003
|
+
|
|
1004
|
+
Focus this suite on ONE circuit:
|
|
1005
|
+
|
|
1006
|
+
Circuit: ${z.name} (${z.pure?"pure":"impure"})
|
|
1007
|
+
Args:
|
|
1008
|
+
${z.arguments.length===0?" (takes no arguments)":z.arguments.map((B)=>` - ${B.name}: ${B.type["type-name"]}`).join(`
|
|
1009
|
+
`)}
|
|
1010
|
+
${$.startingArgs&&Object.keys($.startingArgs).length>0?`
|
|
1011
|
+
Starting args (REQUIRED — keep these exact field names; replace
|
|
1012
|
+
placeholder values with values that satisfy the goal):
|
|
1013
|
+
|
|
1014
|
+
\`\`\`json
|
|
1015
|
+
${JSON.stringify($.startingArgs,null,2)}
|
|
1016
|
+
\`\`\`
|
|
1017
|
+
|
|
1018
|
+
If the contract source declares Struct or Alias arg types, find the
|
|
1019
|
+
inner field names by reading the source below — do NOT invent them.
|
|
1020
|
+
For Struct args, the JSON object MUST have the inner fields (e.g.
|
|
1021
|
+
\`providerPk: { x: ..., y: ... }\`); leaving inner fields out crashes
|
|
1022
|
+
the runtime with "Cannot read properties of undefined".
|
|
1023
|
+
`:""}
|
|
1024
|
+
For circuits that need prior state (e.g. registerProvider before
|
|
1025
|
+
requestLoan), include the setup actions explicitly. Each setup action
|
|
1026
|
+
must follow the same arg-shape rules.`:`## Target
|
|
1027
|
+
|
|
1028
|
+
No specific circuit was selected — pick the impure circuit (or short
|
|
1029
|
+
chain of circuits) from the contract that best matches the goal below.
|
|
1030
|
+
Prefer the most representative "main path" circuit. If the goal
|
|
1031
|
+
implies setup (e.g. registerProvider before requestLoan), include the
|
|
1032
|
+
setup actions explicitly.`,K=Y?`Success criterion (from the user):
|
|
1033
|
+
> ${Y}
|
|
1034
|
+
`:`No specific success criterion was given. Pick a reasonable one and reflect it in the suite description.
|
|
1035
|
+
`;return`You are scaffolding a CLI test suite for the Midnight contract \`${Z}\`.
|
|
1036
|
+
Strategy is "cli" — no browser. The test will run via \`mn test run\`, which
|
|
1037
|
+
deploys the contract, executes the actions in order, and asserts on the
|
|
1038
|
+
final ledger state.
|
|
1039
|
+
|
|
1040
|
+
${q}
|
|
1041
|
+
${G}
|
|
1042
|
+
${K}
|
|
1043
|
+
|
|
1044
|
+
## Contract context
|
|
1045
|
+
|
|
1046
|
+
${Q}
|
|
1047
|
+
|
|
1048
|
+
${X?`
|
|
1049
|
+
--- contract source ---
|
|
1050
|
+
${X}
|
|
1051
|
+
--- end source ---
|
|
1052
|
+
`:""}
|
|
1053
|
+
|
|
1054
|
+
## What to produce
|
|
1055
|
+
|
|
1056
|
+
A JSON object with EXACTLY this shape (top-level keys: actions, assertions, description):
|
|
1057
|
+
|
|
1058
|
+
\`\`\`json
|
|
1059
|
+
{
|
|
1060
|
+
"description": "one short sentence",
|
|
1061
|
+
"actions": [
|
|
1062
|
+
{ "id": "deploy", "type": "contract-deploy" },
|
|
1063
|
+
{ "id": "<short-slug>", "type": "contract-call", "circuit": "<name>", "args": { ... } },
|
|
1064
|
+
{ "id": "<short-slug>", "type": "contract-state", "assert": { "<fieldOrMap>": { "<op>": <value> } } }
|
|
1065
|
+
],
|
|
1066
|
+
"assertions": {
|
|
1067
|
+
"post": [
|
|
1068
|
+
{ "id": "serve-port-listening", "type": "port-listening", "params": { "port": 9932 }, "expect": "pass" }
|
|
1069
|
+
]
|
|
1070
|
+
}
|
|
1071
|
+
}
|
|
1072
|
+
\`\`\`
|
|
1073
|
+
|
|
1074
|
+
Notes:
|
|
1075
|
+
- \`actions\` is a top-level ARRAY of action objects (not nested under another \`actions\` key).
|
|
1076
|
+
- Always start with \`{ "id": "deploy", "type": "contract-deploy" }\`.
|
|
1077
|
+
- End with a contract-state read so the final assertion runs on the result.
|
|
1078
|
+
- For circuits that need prior state (e.g. registerProvider before requestLoan),
|
|
1079
|
+
include the setup actions explicitly between deploy and the target call.
|
|
1080
|
+
|
|
1081
|
+
Action shape:
|
|
1082
|
+
{ "id": "<short-slug>", "type": "contract-deploy" }
|
|
1083
|
+
{ "id": "<short-slug>", "type": "contract-state", "assert": { "<fieldOrMap>": { "<op>": <value> } } }
|
|
1084
|
+
{ "id": "<short-slug>", "type": "contract-call", "circuit": "<name>", "args": { ... } }
|
|
1085
|
+
|
|
1086
|
+
Args coercion the runner applies (so JSON arg values can be):
|
|
1087
|
+
- number → BigInt for Uint<N>
|
|
1088
|
+
- "123n" string → BigInt for values beyond Number.MAX_SAFE_INTEGER
|
|
1089
|
+
- [0..255] int array → Uint8Array for Bytes<N>
|
|
1090
|
+
- object → recurses into Struct fields
|
|
1091
|
+
|
|
1092
|
+
${w5}
|
|
1093
|
+
|
|
1094
|
+
Now produce the JSON.`}function h5($){let{contractName:Z,contractSummary:Q,screenComponent:X,screenSource:z,relatedSources:Y,url:J,goal:G}=$,q=(Y??[]).map((H)=>`
|
|
1095
|
+
--- ${H.path} ---
|
|
1096
|
+
${H.source}
|
|
1097
|
+
--- end ${H.path} ---`).join(`
|
|
1098
|
+
`),K=X&&z?`## Target screen
|
|
1099
|
+
|
|
1100
|
+
Component: ${X}
|
|
1101
|
+
Served at: ${J}
|
|
1102
|
+
|
|
1103
|
+
## Screen source
|
|
1104
|
+
|
|
1105
|
+
\`\`\`tsx
|
|
1106
|
+
${z}
|
|
1107
|
+
\`\`\`
|
|
1108
|
+
${q?`
|
|
1109
|
+
## Imported components from the same UI
|
|
1110
|
+
${q}
|
|
1111
|
+
`:""}`:`## Target
|
|
1112
|
+
|
|
1113
|
+
No specific screen was selected — generate a generic Midnight dApp
|
|
1114
|
+
flow grounded in the contract circuits + the user's goal below.
|
|
1115
|
+
The dApp is served at ${J}.
|
|
1116
|
+
|
|
1117
|
+
Conventional patterns Midnight dApps follow that you can rely on:
|
|
1118
|
+
- A "Connect Wallet" button in the header.
|
|
1119
|
+
- After connect, a contract-address paste field labelled near
|
|
1120
|
+
"Contract address" with a "Connect →" or "Link" action.
|
|
1121
|
+
- Per-circuit forms with a labelled "Submit" / "Send" / "Run" action
|
|
1122
|
+
and progress feedback during ZK proof generation (30–90 s).
|
|
1123
|
+
- A history / state panel that updates after a successful tx.
|
|
1124
|
+
- Toast or inline error messages on failure.
|
|
1125
|
+
|
|
1126
|
+
Use generic-but-specific selectors (\`getByRole('button', { name: /connect wallet/i })\`)
|
|
1127
|
+
in the prompt steps, since you don't have the actual JSX.`,B=G?`Success criterion (from the user):
|
|
1128
|
+
> ${G}
|
|
1129
|
+
`: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?
|
|
1130
|
+
`:`No specific success criterion was given. Infer one from the contract circuits: what is the obvious "happy path" outcome the dApp exists to support?
|
|
1131
|
+
`;return`You are scaffolding a browser test for the Midnight dApp \`${Z}\`.
|
|
1132
|
+
The test runs via \`mn test run\` — Claude (you, in the test session)
|
|
1133
|
+
drives Chrome and follows a prompt.md file.
|
|
1134
|
+
|
|
1135
|
+
${K}
|
|
1136
|
+
|
|
1137
|
+
${B}
|
|
1138
|
+
|
|
1139
|
+
## Contract context (so you know what circuits the dApp calls)
|
|
1140
|
+
|
|
1141
|
+
${Q}
|
|
1142
|
+
|
|
1143
|
+
## What to produce
|
|
1144
|
+
|
|
1145
|
+
A JSON object with three keys:
|
|
1146
|
+
|
|
1147
|
+
- prompt: the markdown body of prompt.md the test runner will hand to
|
|
1148
|
+
Claude in the actual test session. Numbered steps, terse.
|
|
1149
|
+
${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."}
|
|
1150
|
+
Steps should:
|
|
1151
|
+
1. open ${J} in Chrome
|
|
1152
|
+
2. wait for any wallet/connection state the dApp needs
|
|
1153
|
+
3. perform the on-screen actions for the goal
|
|
1154
|
+
4. verify the success criterion on screen
|
|
1155
|
+
5. report pass/fail per step + final on-screen text
|
|
1156
|
+
Keep it under 25 lines.
|
|
1157
|
+
- assertions: { "post": [ ... ] } — at minimum
|
|
1158
|
+
{ "id": "claude-exit-ok", "type": "process-exit-code",
|
|
1159
|
+
"params": { "code": 0 }, "expect": "pass" },
|
|
1160
|
+
{ "id": "serve-port-listening", "type": "port-listening",
|
|
1161
|
+
"params": { "port": 9932 }, "expect": "pass" },
|
|
1162
|
+
and
|
|
1163
|
+
{ "id": "agent-no-failure", "type": "agent-report-no-failure",
|
|
1164
|
+
"params": {}, "expect": "pass" }
|
|
1165
|
+
(this last one parses your final report — without it, you
|
|
1166
|
+
writing "FAILED" still counts as a pass).
|
|
1167
|
+
Add others if the success state can be checked from the
|
|
1168
|
+
chain (e.g. ledger-field on a contract-state read).
|
|
1169
|
+
- description: one short sentence summarising what this suite verifies.
|
|
1170
|
+
|
|
1171
|
+
${w5}
|
|
1172
|
+
|
|
1173
|
+
Now produce the JSON.`}function q7($){let{name:Z,circuits:Q,witnesses:X}=$,z=[`Contract: ${Z}`];if(Q.length>0){z.push("","Circuits:");for(let Y of Q){let J=Y.arguments.length===0?"()":"("+Y.arguments.map((G)=>`${G.name}: ${G.type["type-name"]}`).join(", ")+")";z.push(` ${Y.pure?"pure ":"impure"} ${Y.name}${J}`)}}if(X.length>0){z.push("","Witnesses:");for(let Y of X){let J=Y.arguments.length===0?"()":"("+Y.arguments.map((G)=>`${G.name}: ${G.type["type-name"]}`).join(", ")+")";z.push(` ${Y.name}${J}`)}}return z.join(`
|
|
1174
|
+
`)}var k5="```json",w5;var m5=N(()=>{w5=`Output rules:
|
|
1175
|
+
- Reply with EXACTLY one ${"```json"} ... \`\`\` fenced code block.
|
|
1176
|
+
- Inside the fence: valid JSON, nothing else (no comments, no trailing text).
|
|
1177
|
+
- Fields you don't know about: omit them. Don't invent fields.
|
|
1178
|
+
- Use only the circuit / ledger field names that appear in the inputs below.
|
|
1179
|
+
- Be concise — fewer, well-named actions beat many vague ones.`});var g5={};m(g5,{analyzeWitnessDependencies:()=>c1});function c1($,Z){if(Z.length===0)return{byCircuit:new Map};let Q=SJ($),X=kJ(Q);if(X.size===0)return{byCircuit:new Map};let z=new Set(Z),Y=new Map,J=new Map;for(let[K,B]of X){Y.set(K,new Set),J.set(K,new Set);for(let H of fJ(B.body,B.argNames))if(z.has(H))Y.get(K).add(H);else if(X.has(H))J.get(K).add(H)}let G=!0;while(G){G=!1;for(let[K,B]of J){let H=Y.get(K);for(let U of B)for(let j of Y.get(U)??[])if(!H.has(j))H.add(j),G=!0}}let q=new Map;for(let[K,B]of X){if(!B.isCircuit)continue;let H=Y.get(K);if(H&&H.size>0)q.set(K,[...H].sort())}return{byCircuit:q}}function SJ($){return $.replace(bJ,"").replace(yJ,"")}function kJ($){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],J=X[3],G=X.index+X[0].length,q=$.indexOf("{",G);if(q<0)continue;let K=1,B=q+1;for(;B<$.length&&K>0;B++){let j=$[B];if(j==="{")K++;else if(j==="}")K--}if(K!==0)continue;let H=$.slice(q+1,B-1),U=wJ(J);Z.set(Y,{body:H,argNames:U,isCircuit:z==="circuit"})}return Z}function wJ($){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 fJ($,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(hJ.has(Y)||Z.has(Y))continue;Q.add(Y)}return Q}var yJ,bJ,hJ;var J2=N(()=>{yJ=/\/\/[^\n]*/g,bJ=/\/\*[\s\S]*?\*\//g;hJ=new Set(["if","for","while","switch","return","new","assert","disclose","transientHash","persistentHash","ownPublicKey","publicKey","Map","Set","Vector","Tuple","Option","Bytes","Field","Uint","Boolean"])});var r5={};m(r5,{isClaudeAvailable:()=>lJ,generateUiScaffoldWithAI:()=>sJ,generateCliScaffoldWithAI:()=>rJ,findContractSourcePath:()=>B7,claudeSubprocessRunner:()=>V7});import{execFileSync as mJ,spawn as gJ}from"node:child_process";import{existsSync as K7,readdirSync as uJ,readFileSync as c5}from"node:fs";import{dirname as J1,join as G2}from"node:path";function lJ(){try{return mJ("claude",["--version"],{stdio:"ignore",timeout:5000}),!0}catch{return!1}}function n5($){let Q=new RegExp("```json\\s*\\n([\\s\\S]*?)\\n```").exec($);if(!Q)throw new Error(`Claude response did not contain a ${k5} ... \`\`\` 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 iJ($,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=oJ($);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)"}`);nJ(z,Y)}return{actions:Q}}function nJ($,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((J)=>`${J.name}: ${J.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 oJ($){if(Array.isArray($.actions))return $.actions;let Z=$.actions;if(Z&&Array.isArray(Z.actions))return Z.actions;return null}function aJ($){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 rJ($,Z=V7){let Q=$.contractSourcePath&&K7($.contractSourcePath)?c5($.contractSourcePath,"utf-8"):void 0,X=$.contract.witnesses.map((O)=>O.name),z=Q?c1(Q,X).byCircuit:new Map,Y=$.targetCircuit?Y2($.targetCircuit):void 0,J=f5({contractName:$.contract.name,contractSummary:q7({name:$.contract.name,circuits:$.contract.circuits,witnesses:$.contract.witnesses}),contractSource:Q,targetCircuit:$.targetCircuit,startingArgs:Y,witnessDependentCircuits:z,goal:$.goal}),G=await Z(J),q=n5(G),K=iJ(q,$.contract),B=$.suiteName??($.targetCircuit?`cli-${$.targetCircuit.name.toLowerCase().replace(/_/g,"-")}`:`cli-${a5($.goal)??"ai"}`),H=$.network??d5,U=$.servePort??l5,j={name:$.contract.name,network:H,prep:i5},L={name:B,description:q.description,strategy:"cli",timeout:pJ},W=tJ(q.assertions,U),P=[...z.entries()].filter(([O])=>O!==$.targetCircuit?.name).map(([O,F])=>({name:O,reason:`uses witness ${F.join(", ")} (private state — UI-only)`}));return{dappConfig:j,suite:L,actions:K,assertions:W,prompt:null,suiteName:B,excludedCircuits:P.length>0?P:void 0}}async function sJ($,Z=V7){let Q=$.screen?c5($.screen.path,"utf-8"):void 0,X=h5({contractName:$.contract.name,contractSummary:q7({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=n5(z);aJ(Y);let J=$.suiteName??($.screen?`ui-${$.screen.name}`:`ui-${a5($.goal)??"ai"}`),G=$.network??d5,q=$.servePort??l5,K=$.url,B={name:$.contract.name,network:G,port:$.port,buildCmd:$.buildCmd,url:K,prep:dJ};if($.buildDir)B.buildDir=$.buildDir;let H={name:J,description:Y.description,strategy:"browser",timeout:cJ,...$.browserMode?{browserMode:$.browserMode}:{}},U=eJ(Y.assertions,q);return{dappConfig:B,suite:H,actions:null,assertions:U,prompt:Y.prompt,suiteName:J}}function o5($){let Z=$.params;return typeof Z?.port==="number"?Z.port:void 0}function tJ($,Z){if($.post.some((X)=>X.type==="port-listening"&&o5(X)===Z))return $;return{...$,post:[...$.post,{id:"serve-port-listening",type:"port-listening",params:{port:Z},expect:"pass"}]}}function eJ($,Z){let Q=$.post;if(!Q.some((J)=>J.type==="process-exit-code"))Q=[...Q,{id:"claude-exit-ok",type:"process-exit-code",params:{code:0},expect:"pass"}];if(!Q.some((J)=>J.type==="port-listening"&&o5(J)===Z))Q=[...Q,{id:"serve-port-listening",type:"port-listening",params:{port:Z},expect:"pass"}];if(!Q.some((J)=>J.type==="agent-report-no-failure"))Q=[...Q,{id:"agent-no-failure",type:"agent-report-no-failure",params:{},expect:"pass"}];return{...$,post:Q}}function a5($){if(!$)return;let Z=$.toLowerCase().replace(/[^a-z0-9]+/g,"-").replace(/^-+|-+$/g,"").slice(0,32);return Z.length>=2?Z:void 0}function B7($){let Z=$.split("/").pop(),Q=[J1($),G2(J1($),"src"),J1(J1($)),G2(J1(J1($)),"src")];for(let X of Q){if(!K7(X))continue;if(Z){let J=G2(X,`${Z}.compact`);if(K7(J))return J}let z;try{z=uJ(X)}catch{continue}let Y=z.find((J)=>J.endsWith(".compact"));if(Y)return G2(X,Y)}return}var d5="undeployed",l5=9932,pJ=300,cJ=600,i5,dJ,u5=300000,p5=4194304,V7=($)=>{return new Promise((Z,Q)=>{let X=gJ("claude",["--print"],{stdio:["pipe","pipe","pipe"]}),z="",Y="",J=0,G=!1,q=setTimeout(()=>{X.kill("SIGTERM"),Q(new Error(`claude --print timed out after ${u5/1000}s`))},u5);X.stdout?.on("data",(K)=>{if(J+=K.length,J>p5){G=!0,X.kill("SIGTERM");return}z+=K.toString()}),X.stderr?.on("data",(K)=>{Y+=K.toString()}),X.on("error",(K)=>{if(clearTimeout(q),K.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(K)}),X.on("exit",(K,B)=>{if(clearTimeout(q),G){Q(new Error(`claude --print response exceeded ${p5/1024/1024}MB; aborted.`));return}if(B==="SIGTERM")return;if(K!==0){let H=Y.trim().slice(0,300)||`(no stderr; exit ${K})`;Q(new Error(`claude --print exited with code ${K}: ${H}`));return}Z(z)}),X.stdin?.write($),X.stdin?.end()})};var H7=N(()=>{m5();z7();J2();i5=["cache-clear","localnet-up","balance:1000","dust","mn-serve"],dJ=[...i5,"build-and-serve"]});var Z6={};m(Z6,{runActions:()=>QG,diffState:()=>KG});import{existsSync as s5,readFileSync as t5,writeFileSync as $G,mkdirSync as ZG}from"node:fs";import{join as e5}from"node:path";async function QG($){let{actions:Z,config:Q,dappDir:X,suiteName:z,networkConfig:Y,servePort:J,redeploy:G,onActionStart:q,onActionComplete:K,onMessage:B}=$,H=[],U=Q.network??"undeployed",j;if(!G)j=VG(X,z,U);let{info:L}=j4(X),W=qG(L),P;for(let O of Z){q?.(O);let F=Date.now();if(P&&(O.type==="contract-call"||O.type==="contract-state")){let I={id:O.id,type:O.type,status:"skip",duration:0,message:`cascaded skip — depends on ${P}`};H.push(I),K?.(O,I);continue}try{let I=await XG(O,{dappDir:X,networkConfig:Y,managedDir:L.managedDir,contractName:L.name,servePort:J,contractAddress:j,witnessDeps:W,onMessage:B});if(I.contractAddress){if(j=I.contractAddress,BG(X,z,U,j),O.type==="contract-deploy")P=void 0}let _=Date.now()-F,v={...I,duration:_};if(H.push(v),K?.(O,v),I.status==="skip"&&O.type==="contract-call")P=`skipped action "${O.id}"`}catch(I){let _=Date.now()-F,v={id:O.id,type:O.type,status:"fail",duration:_,message:I.message};H.push(v),K?.(O,v);break}}return H}async function XG($,Z){switch($.type){case"contract-deploy":return zG($,Z);case"contract-call":return YG($,Z);case"contract-state":return JG($,Z);case"wallet-cmd":return GG($,Z);default:throw new Error(`Unknown action type: ${$.type}`)}}async function zG($,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 a8({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 YG($,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 m1({dappDir:Z.dappDir,networkConfig:Z.networkConfig,managedDir:Z.managedDir,contractName:Z.contractName,contractAddress:Z.contractAddress,onMessage:()=>{}})}catch{}let z=$.args?Object.values($.args):[];await r8({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 m1({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 JG($,Z){if(!Z.contractAddress)throw new Error(`Action "${$.id}": no contract address. Add a contract-deploy action first.`);let Q=await m1({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 J=Q.fields[z];if(J===void 0){let G=Q.maps[z];if(G)W7(z,BigInt(G.size),Y,X);else X.push(`Field "${z}" not found in ledger state`);continue}try{let G=BigInt(J);W7(z,G,Y,X)}catch{W7(z,J,Y,X)}}if(X.length>0)throw new Error(`State assertion failed:
|
|
1180
|
+
${X.join(`
|
|
1181
|
+
`)}`)}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 W7($,Z,Q,X){for(let[z,Y]of Object.entries(Q)){let J=typeof Y==="number"?BigInt(Y):Y;if(typeof Z==="bigint"&&(typeof J==="bigint"||typeof J==="number")){let G=BigInt(J);switch(z){case">":if(!(Z>G))X.push(`${$}: expected > ${G}, got ${Z}`);break;case">=":if(!(Z>=G))X.push(`${$}: expected >= ${G}, got ${Z}`);break;case"<":if(!(Z<G))X.push(`${$}: expected < ${G}, got ${Z}`);break;case"<=":if(!(Z<=G))X.push(`${$}: expected <= ${G}, got ${Z}`);break;case"==":if(Z!==G)X.push(`${$}: expected == ${G}, got ${Z}`);break;case"!=":if(Z===G)X.push(`${$}: expected != ${G}, got ${Z}`);break;default:X.push(`${$}: unknown operator "${z}"`)}}else{let G=String(J);switch(z){case"==":if(Z!==G)X.push(`${$}: expected == "${G}", got "${Z}"`);break;case"!=":if(Z===G)X.push(`${$}: expected != "${G}", got "${Z}"`);break;default:X.push(`${$}: operator "${z}" not supported for string values`)}}}}async function GG($,Z){return{id:$.id,type:$.type,status:"pass",duration:0,message:`wallet-cmd: ${$.cmd??"no command"} (not yet implemented)`}}function qG($){if($.witnesses.length===0)return new Map;let Z=B7($.managedDir);if(!Z||!s5(Z))return new Map;let Q=t5(Z,"utf-8"),X=$.witnesses.map((z)=>z.name);return c1(Q,X).byCircuit}function KG($,Z){if(!$||!Z)return[];let Q=[],X=new Set([...Object.keys($.fields),...Object.keys(Z.fields)]);for(let Y of X){let J=$.fields[Y]??"(absent)",G=Z.fields[Y]??"(absent)";if(J!==G)Q.push({field:Y,before:J,after:G})}let z=new Set([...Object.keys($.maps),...Object.keys(Z.maps)]);for(let Y of z){let J=$.maps[Y]?.size??"0",G=Z.maps[Y]?.size??"0";if(J!==G)Q.push({field:`${Y} (entries)`,before:J,after:G})}return Q}function $6($,Z){return e5($,"tests","results",`.contract-cache-${Z}.json`)}function VG($,Z,Q){let X=$6($,Z);if(!s5(X))return;try{let z=JSON.parse(t5(X,"utf-8"));if(z.network!==Q)return;return z.address}catch{return}}function BG($,Z,Q,X){let z=$6($,Z);ZG(e5($,"tests","results"),{recursive:!0});let Y={address:X,network:Q,timestamp:new Date().toISOString()};$G(z,JSON.stringify(Y,null,2)+`
|
|
1182
|
+
`)}var Q6=N(()=>{g1();k1();J2();H7()});var U7={};m(U7,{default:()=>Y6});import{join as X6}from"node:path";import{mkdirSync as HG}from"node:fs";function WG($){return z6.includes($)}async function Y6($,Z){let Q=$.subcommand;if(!Q||!WG(Q))throw new t(`Usage: midnight test <${z6.join("|")}>
|
|
1183
|
+
|
|
1184
|
+
create Generate dapp.test.json + a CLI test suite from the contract
|
|
1185
|
+
run Run test suites for the current dApp
|
|
1186
|
+
list List available test suites
|
|
1187
|
+
results Show latest test results
|
|
1188
|
+
|
|
1189
|
+
Run from the root of a dApp project containing dapp.test.json.`);let X=x($,"json");switch(Q){case"create":return PG($,X);case"run":return TG($,X,Z);case"list":return MG(X);case"results":return RG($,X)}}function UG($){let Z=($??"").toLowerCase();if(Z==="")return;if(Z==="cli"||Z==="browser")return Z;if(Z==="ui")return"browser";throw new t(`Unknown strategy "${$}". Use "cli" or "ui".`)}function LG($){return $==="preprod"||$==="preview"||$==="undeployed"?$:void 0}function OG($){let Z={},Q=D($,"port");if(Q!==void 0){let G=parseInt(Q,10);if(!Number.isFinite(G)||G<=0||G>65535)throw new t(`Invalid --port "${Q}" — must be 1–65535.`);Z.port=G}let X=D($,"build-cmd");if(X!==void 0)Z.buildCmd=X;let z=D($,"build-dir");if(z!==void 0)Z.buildDir=z;let Y=D($,"url");if(Y!==void 0)Z.url=Y;let J=D($,"browser-mode");if(J!==void 0){if(J!=="dom"&&J!=="vision"&&J!=="script"&&J!=="auto")throw new t(`Invalid --browser-mode "${J}" — must be dom, vision, or script.`);Z.browserMode=J}return Z}function jG($){if($.port===void 0||$.buildCmd===void 0)throw new t("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 PG($,Z){let{resolve:Q}=await import("node:path"),{findContractInfo:X}=await Promise.resolve().then(() => (k1(),h8)),{buildScaffold:z}=await Promise.resolve().then(() => (z7(),_5)),{writeScaffold:Y}=await Promise.resolve().then(() => (A5(),R5)),J=await Promise.resolve().then(() => (S5(),b5)),{writeJsonResult:G}=await Promise.resolve().then(() => A2),{isInteractive:q,promptStrategy:K,promptBrowserOptions:B,promptCircuit:H,promptScreen:U,promptGoal:j,promptSuiteName:L}=J,W=await Promise.resolve().then(() => (H7(),r5)),{discoverScreens:P}=await Promise.resolve().then(() => (G7(),x5)),O=Q(D($,"path")??process.cwd()),F=D($,"name"),I=D($,"suite"),_=LG(D($,"network")),v=x($,"force"),b=!Z&&q(),c=D($,"goal"),k=D($,"screen"),J0=x($,"no-ai"),i=UG(D($,"strategy"))??(b?await K():"cli"),e;if(i==="browser"){let h=OG($);e=b?await B(h):jG(h)}let{info:Q0}=X(O,F),C=!J0&&W.isClaudeAvailable(),R=!J0&&(c!==void 0||k!==void 0||b&&C),T=R?await FG({strategy:i,info:Q0,dappDir:O,browser:e,network:_,suiteName:I,goalFlag:c,screenFlag:k,interactive:b,jsonMode:Z,ai:W,promptCircuit:H,promptScreen:U,promptGoal:j,promptSuiteName:L,discoverScreens:P})??z(Q0.circuits,{contractName:Q0.name,suiteName:I,strategy:i,browser:e,network:_}):z(Q0.circuits,{contractName:Q0.name,suiteName:I,strategy:i,browser:e,network:_});if(b&&!I){let h=await L(T.suiteName);if(h!==T.suiteName)T.suiteName=h,T.suite.name=h}let E=Y(T,{dappDir:O,force:v});if(Z){G({subcommand:"create",contractName:Q0.name,suiteName:T.suiteName,strategy:i,aiAssisted:R&&T.suiteName!=="cli-default"&&T.suiteName!=="ui-default",written:E.written,excludedCircuits:T.excludedCircuits});return}process.stderr.write(`
|
|
1190
|
+
`+g(`Test scaffold: ${Q0.name}`)+`
|
|
1191
|
+
|
|
1192
|
+
`);for(let h of E.written)process.stderr.write(` ${p("✓")} ${h}
|
|
1193
|
+
`);let u=i==="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(T.excludedCircuits&&T.excludedCircuits.length>0){process.stderr.write(`
|
|
1194
|
+
`+V(" Skipped (not CLI-testable):")+`
|
|
1195
|
+
`);for(let h of T.excludedCircuits)process.stderr.write(` ${S("!")} ${h.name} ${V("— "+h.reason)}
|
|
1196
|
+
`);process.stderr.write(V(` These need the dApp UI to populate private state first.
|
|
1197
|
+
`)),process.stderr.write(V(` Cover them with a browser-strategy suite (mn test create --strategy ui).
|
|
1198
|
+
`))}process.stderr.write(`
|
|
1199
|
+
`+V(" Next:")+`
|
|
1200
|
+
`),process.stderr.write(V(" Edit ")+f(`tests/suites/${T.suiteName}/${u.file}`)+`
|
|
1201
|
+
`),process.stderr.write(V(" ")+V(u.hint)+`
|
|
1202
|
+
`),process.stderr.write(V(" Run ")+f(`mn test run --suite ${T.suiteName}`)+`
|
|
1203
|
+
`),process.stderr.write(V(" List ")+f("mn test list")+V(" (see every suite in this project)")+`
|
|
1204
|
+
|
|
1205
|
+
`)}async function FG($){try{if(!$.ai.isClaudeAvailable())return null;let Z=$.goalFlag??($.interactive?await $.promptGoal():void 0);if($.strategy==="cli")return await IG($,Z);return await _G($,Z)}catch(Z){return process.stderr.write(`
|
|
1206
|
+
${V("AI scaffold failed, falling back to deterministic:")} ${Z.message}
|
|
1207
|
+
`),null}}async function J6($,Z,Q){if(!Q)return Z();let X=l($);try{let z=await Z();return X.stop(),z}catch(z){throw X.fail("AI scaffold failed"),z}}async function IG($,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(() => (J2(),g5)),J=z(X,"utf-8"),G=$.info.witnesses.map((B)=>B.name),K=Y(J,G).byCircuit.get(Q.name);if(K&&K.length>0)process.stderr.write(`
|
|
1208
|
+
`+S("!")+` ${Q.name} calls ${K.join(", ")} — a witness that reads private state.
|
|
1209
|
+
`),process.stderr.write(V(` The CLI suite will deploy + run, but expect a "Cannot read properties of undefined" failure
|
|
1210
|
+
`)+V(` unless the dApp UI populates that state first. Consider --strategy ui instead.
|
|
1211
|
+
`))}return J6("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 _G($,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((J)=>J.name===$.screenFlag||J.component===$.screenFlag):$.interactive?await $.promptScreen(z):z[0];if(!Y&&!Z)return null;return J6("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 TG($,Z,Q){let{config:X,dappDir:z}=Q2(),Y=e8(z),J=D($,"suite"),G;if(J){if(G=Y.find((j)=>j.suite.name===J),!G){let j=Y.map((L)=>L.suite.name).join(", ")||"none found";throw new t(`Suite "${J}" not found. Available: ${j}`)}}else if(Y.length>0)G=Y[0];HG(X6(z,"tests","results"),{recursive:!0});let q=Date.now(),K=new Date().toISOString();if(!Z){if(process.stderr.write(`
|
|
1212
|
+
`+g(`E2E Test: ${X.name}`)+`
|
|
1213
|
+
|
|
1214
|
+
`),process.stderr.write(A("Network",X.network??"undeployed")+`
|
|
1215
|
+
`),X.port)process.stderr.write(A("Port",String(X.port))+`
|
|
1216
|
+
`);if(G){if(process.stderr.write(A("Suite",G.suite.name)+`
|
|
1217
|
+
`),process.stderr.write(A("Strategy",G.suite.strategy)+`
|
|
1218
|
+
`),G.suite.strategy==="browser"){let j=$7(G.suite);process.stderr.write(A("Browser Mode",j)+`
|
|
1219
|
+
`)}}process.stderr.write(`
|
|
1220
|
+
`)}let B=r3(),H=[],U=(j)=>{if(!Z)process.stderr.write(V(` [teardown] ${j}`)+`
|
|
1221
|
+
`)};Q?.addEventListener("abort",()=>Q7(B,U),{once:!0});try{if(!Z)process.stderr.write(M(` Prep
|
|
1222
|
+
`));let j=l("Starting prep..."),L=await z5(X,z,B,{onStepStart(w){j.update(`[prep] ${w}...`)},onStepComplete(w,i,e,Q0){if(i==="pass")j.stop(`${p("✓")} ${w} (${AG(e)})`);else j.stop(`${n("✗")} ${w}: ${Q0??"failed"}`);j=l("")},onMessage(w){j.update(w)}});if(j.stop(""),H.push(...L),!Z)process.stderr.write(`
|
|
1223
|
+
`);let W=0,P="";if(G){let w=G.suite;if(w.strategy==="browser"){let i=o3(G.suiteDir);if(!i)throw new Error(`No prompt.md found in ${G.suiteDir} — required for browser strategy`);if(P=X6(z,"tests","results",`${w.name}_${K.replace(/[:.]/g,"-")}_claude.log`),!Z)process.stderr.write(M(` Browser Test
|
|
1224
|
+
`)),process.stderr.write(V(` Claude will take over the terminal. Output is also logged.
|
|
1225
|
+
|
|
1226
|
+
`));let e=await G5({suite:w,prompt:i,dappDir:z,logFile:P,onMessage:(Q0)=>{if(!Z)process.stderr.write(V(` ${Q0}`)+`
|
|
1227
|
+
`)}});if(W=e.exitCode,!Z)if(process.stderr.write(`
|
|
1228
|
+
`),e.timedOut)process.stderr.write(` ${n("✗")} Browser test timed out
|
|
1229
|
+
`);else process.stderr.write(` ${W===0?p("✓"):n("✗")} Claude exited with code ${W}
|
|
1230
|
+
`)}else if(w.strategy==="cli"){let i=n3(G.suiteDir);if(!i)throw new Error(`No actions.json found in ${G.suiteDir} — required for CLI strategy`);if(!Z)process.stderr.write(M(` CLI Test
|
|
1231
|
+
`)),process.stderr.write(V(` Executing ${i.actions.length} action(s)
|
|
1232
|
+
|
|
1233
|
+
`));let{runActions:e,diffState:Q0}=await Promise.resolve().then(() => (Q6(),Z6)),{resolveNetwork:C}=await Promise.resolve().then(() => (H0(),i4)),R=X.network??"undeployed",{config:T}=C({args:{command:"test",subcommand:"run",positionals:[],flags:{network:R}}}),{DEFAULT_SERVE_PORT:E}=await Promise.resolve().then(() => Z$),u=B.serveHandle?.port??E,h=x($,"redeploy"),s=await e({actions:i.actions,config:X,dappDir:z,suiteName:w.name,networkConfig:T,servePort:u,redeploy:h,onActionStart:(X0)=>{if(!Z){let G0=l(`[${X0.id}] ${X0.type}${X0.circuit?` ${X0.circuit}`:""}...`);X0._spinner=G0}},onActionComplete:(X0,G0)=>{if(!Z){let w0=X0._spinner;if(w0)if(G0.status==="pass")w0.stop(`${p("✓")} ${X0.id}: ${G0.message??"pass"}`);else if(G0.status==="skip")w0.stop(`${S("⊘")} ${X0.id}: ${G0.message??"skipped"}`);else w0.stop(`${n("✗")} ${X0.id}: ${G0.message??"fail"}`);if(G0.stateBefore&&G0.stateAfter){let f0=Q0(G0.stateBefore,G0.stateAfter);if(f0.length>0)for(let A0 of f0)process.stderr.write(V(` ${A0.field}: ${A0.before} → `)+f(A0.after)+`
|
|
1234
|
+
`)}}},onMessage:(X0)=>{}});if(W=s.some((X0)=>X0.status==="fail")?1:0,!Z){process.stderr.write(`
|
|
1235
|
+
`);let X0=s.filter((A0)=>A0.status==="pass").length,G0=s.filter((A0)=>A0.status==="skip").length,w0=s.filter((A0)=>A0.status==="fail").length,f0=p(String(X0)+" passed");if(G0>0)f0+=`, ${S(String(G0)+" skipped")}`;if(w0>0)f0+=`, ${n(String(w0)+" failed")}`;process.stderr.write(` Actions: ${f0}
|
|
1236
|
+
`)}B._actionResults=s}}else if(!Z)process.stderr.write(V(` No test suites found — only prep was run
|
|
1237
|
+
`));let O={processExitCode:W,agentLogPath:P},F=[];if(G){let w=i3(G.suiteDir);if(w&&w.post.length>0){if(!Z)process.stderr.write(`
|
|
1238
|
+
`+M(` Assertions
|
|
1239
|
+
`));if(F=await K5(w.post,O),!Z)for(let i of F){let e=i.status==="pass"?p("✓"):n("✗"),Q0=i.message?V(` — ${i.message}`):"";process.stderr.write(` ${e} ${i.id}${Q0}
|
|
1240
|
+
`)}}}let I=Math.round((Date.now()-q)/1000),_=H.every((w)=>w.status==="pass"),v=F.every((w)=>w.status==="pass"),b=_&&v&&W===0?"pass":"fail",c=B._actionResults,k={id:`${X.name}_${K}`,dapp:X.name,suite:G?.suite.name??"prep-only",timestamp:K,duration:I,network:X.network??"undeployed",strategy:G?.suite.strategy??"none",model:G?.suite.model,status:b,prep:H,actions:c?.map((w)=>({id:w.id,type:w.type,status:w.status,duration:w.duration,message:w.message,contractAddress:w.contractAddress})),assertions:F,testOutput:P?{exitCode:W,logFile:P}:void 0},J0=H5(k,z);if(Z)y(k);else process.stderr.write(`
|
|
1241
|
+
`+o()+`
|
|
1242
|
+
`),process.stderr.write(` ${b==="pass"?p(M("PASS")):n(M("FAIL"))} — ${X.name} (${I}s)
|
|
1243
|
+
`),process.stderr.write(V(` Results: ${J0}`)+`
|
|
1244
|
+
|
|
1245
|
+
`);if(b==="fail")process.exitCode=1}finally{await Q7(B,U)}}function MG($){let{config:Z,dappDir:Q}=Q2(),X=e8(Q);if($){y({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(`
|
|
1246
|
+
`+g(`Tests: ${Z.name}`)+`
|
|
1247
|
+
|
|
1248
|
+
`),X.length===0)process.stderr.write(V(` No test suites found.
|
|
1249
|
+
`)),process.stderr.write(V(` Create tests/suites/<name>/suite.json to add one.
|
|
1250
|
+
`));else for(let{suite:z}of X)process.stderr.write(` ${M(f(z.name))}
|
|
1251
|
+
`),process.stderr.write(` ${V(z.description)}
|
|
1252
|
+
`),process.stderr.write(V(` strategy: ${z.strategy}`)+(z.timeout?V(`, timeout: ${z.timeout}s`):"")+`
|
|
1253
|
+
|
|
1254
|
+
`);process.stderr.write(`
|
|
1255
|
+
`)}function RG($,Z){let{config:Q,dappDir:X}=Q2(),z=x($,"all"),Y=D($,"suite"),J=z?Z7(X,Y):(()=>{let G=W5(X,Y);return G?[G]:[]})();if(Z){y(z?{dapp:Q.name,results:J}:J[0]??{});return}if(process.stderr.write(`
|
|
1256
|
+
`+g(`Results: ${Q.name}`)+`
|
|
1257
|
+
|
|
1258
|
+
`),J.length===0)process.stderr.write(V(` No test results found.
|
|
1259
|
+
`)),process.stderr.write(V(` Run "midnight test run" to generate results.
|
|
1260
|
+
`));else for(let G of J){let q=G.status==="pass"?p("✓"):n("✗"),K=G.status==="pass"?p(M("PASS")):n(M("FAIL"));process.stderr.write(` ${q} ${M(G.suite)} — ${K} (${G.duration}s)
|
|
1261
|
+
`),process.stderr.write(V(` ${G.timestamp} | ${G.network} | ${G.strategy}`)+`
|
|
1262
|
+
`);let B=G.prep.filter((U)=>U.status==="fail");if(B.length>0)process.stderr.write(n(` prep: ${B.length} failed`)+`
|
|
1263
|
+
`);let H=G.assertions.filter((U)=>U.status==="fail");if(H.length>0)process.stderr.write(n(` assertions: ${H.length} failed`)+`
|
|
1264
|
+
`);process.stderr.write(`
|
|
1265
|
+
`)}process.stderr.write(`
|
|
1266
|
+
`)}function AG($){if($<1000)return`${$}ms`;return`${($/1000).toFixed(1)}s`}var z6;var L7=N(()=>{l0();K0();u0();a3();Y5();q5();V5();U5();L5();z6=["run","list","results","create"]});var mG={};import{readFileSync as G6}from"node:fs";import{fileURLToPath as V6}from"node:url";import{Server as DG}from"@modelcontextprotocol/sdk/server/index.js";import{StdioServerTransport as NG}from"@modelcontextprotocol/sdk/server/stdio.js";import{ListToolsRequestSchema as EG,CallToolRequestSchema as xG,ListResourcesRequestSchema as CG,ReadResourceRequestSchema as vG}from"@modelcontextprotocol/sdk/types.js";function L0($,Z,Q){let X={json:!0},z=[];for(let[Y,J]of Object.entries(Z)){if(J===void 0||J===null)continue;let G=Y==="full"?q1:Y;if(typeof J==="boolean"){if(J)X[G]=!0}else X[G]=String(J)}return{command:$,subcommand:Q,positionals:z,flags:X}}async function Z0($){let Z=kG[$];if(!Z)throw new Error(`Unknown command handler: ${$}`);return(await Z()).default}function fG($,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)}`}}function j7($){return{...$,_serverVersion:g0}}async function K6($,Z){let Q=W6.find((X)=>X.name===$);if(!Q)return O7(new Error(`Unknown tool: ${$}`));try{let X=await Q.handler(Z);return{content:[{type:"text",text:JSON.stringify(j7(X),null,2)}]}}catch(X){return O7(X instanceof Error?X:new Error(String(X)))}}function O7($){let{errorCode:Z}=U1($);return{content:[{type:"text",text:JSON.stringify(j7({error:!0,code:Z,message:a7($.message)}))}],isError:!0}}async function hG(){let $=new NG;await d1.connect($)}var B6="midnight-wallet://skill/core",H6="midnight-wallet://skill/full",yG="midnight-wallet://skill",bG,SG,kG,W6,wG,q6,d1;var U6=N(()=>{W$();R2();j1();s7();bG=V6(new URL("../docs/SKILL-CORE.md",import.meta.url)),SG=V6(new URL("../docs/SKILL.md",import.meta.url));kG={generate:()=>Promise.resolve().then(() => (p2(),u2)),info:()=>Promise.resolve().then(() => (d2(),c2)),balance:()=>Promise.resolve().then(() => (K8(),q8)),address:()=>Promise.resolve().then(() => (B8(),V8)),"genesis-address":()=>Promise.resolve().then(() => (W8(),H8)),"inspect-cost":()=>Promise.resolve().then(() => (L8(),U8)),airdrop:()=>Promise.resolve().then(() => (x$(),E$)),transfer:()=>Promise.resolve().then(() => (j8(),O8)),dust:()=>Promise.resolve().then(() => (v$(),C$)),cache:()=>Promise.resolve().then(() => (F8(),P8)),config:()=>Promise.resolve().then(() => (_8(),I8)),localnet:()=>Promise.resolve().then(() => (E8(),N8)),wallet:()=>Promise.resolve().then(() => (f$(),w$)),status:()=>Promise.resolve().then(() => (z3(),X3)),contract:()=>Promise.resolve().then(() => (Z2(),$2)),test:()=>Promise.resolve().then(() => (L7(),U7))};W6=[{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=L0("wallet",$,"generate");if(Q.positionals=[Z],delete Q.flags.name,$.force==="true"||$.force===!0)Q.flags.force=!0;let X=await Z0("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[q1]=!0;let Q={command:"wallet",subcommand:"list",positionals:[],flags:Z},X=await Z0("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 Z0("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[q1]=!0;let z={command:"wallet",subcommand:"info",positionals:Z?[Z]:[],flags:X},Y=await Z0("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 Z0("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=L0("info",$),Q=await Z0("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=L0("balance",$,Z);delete Q.flags.address;let X=await Z0("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=L0("address",$),Q=await Z0("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=L0("genesis-address",$),Q=await Z0("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 $=L0("inspect-cost",{}),Z=await Z0("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=L0("airdrop",$,Z);delete Q.flags.amount;let X=await Z0("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=L0("transfer",$,Z);X.positionals=[Q],delete X.flags.to,delete X.flags.amount;let z=await Z0("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=L0("dust",$,"register"),Q=await Z0("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=L0("dust",$,"status"),Q=await Z0("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 Z0("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 Z0("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=L0("cache",$,"clear"),Q=await Z0("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 Z0("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 Z0("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 Z0("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 Z0("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 Z0("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 Z0("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=L0("contract",$,"inspect"),Q=await Z0("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=L0("contract",$,"state"),Q=await Z0("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=L0("contract",$,"deploy"),Q=await Z0("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=L0("contract",$,"call"),Q=await Z0("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=L0("test",$,"create"),Q=await Z0("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=L0("localnet",$,"logs");if(!Z.flags.tail)Z.flags.tail="200";let Q=await Z0("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")}}],wG=new Set(["midnight_transfer","midnight_contract_deploy","midnight_contract_call"]),q6=r7();d1=new DG({name:"midnight-wallet-cli",version:g0},{capabilities:{tools:{},resources:{}}});d1.setRequestHandler(CG,async()=>({resources:[{uri:B6,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:H6,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"}]}));d1.setRequestHandler(vG,async($)=>{let Z=$.params.uri;if(Z===B6){let Q=G6(bG,"utf-8");return{contents:[{uri:Z,mimeType:"text/markdown",text:Q}]}}if(Z===H6||Z===yG){let Q=G6(SG,"utf-8");return{contents:[{uri:Z,mimeType:"text/markdown",text:Q}]}}throw new Error(`Unknown resource: ${Z}`)});d1.setRequestHandler(EG,async()=>{return{tools:W6.map(($)=>({name:$.name,description:$.description,inputSchema:$.inputSchema,...$.annotations?{annotations:$.annotations}:{}}))}});d1.setRequestHandler(xG,async($)=>{let{name:Z,arguments:Q}=$.params,X=Q??{};if(Z==="midnight_confirm_operation"){let z=typeof X.token==="string"?X.token:"",Y=q6.redeem(z);if(!Y)return O7(new Error("Unknown or expired confirmation token. The first-step tool call may need to be re-issued."));return K6(Y.tool,Y.args)}if(wG.has(Z)){let z=q6.create({tool:Z,args:X,description:fG(Z,X)});return{content:[{type:"text",text:JSON.stringify(j7({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 K6(Z,X)});hG().catch(($)=>{process.stderr.write(`MCP server error: ${$.message}
|
|
1267
|
+
`),process.exit(1)})});function O6($){let Z=Math.max(0,Math.min(1,$)),Q=G1[0].length,X=Math.floor(Z*Q);return G1.map((z)=>z.slice(0,X).padEnd(Q))}function j6($){let Z=Math.max(0,Math.min(1,$)),Q=Math.floor($*100);if(Z>=1)return G1;return G1.map((X,z)=>Array.from(X).map((Y,J)=>{if(Y===" ")return" ";if(Z<=0)return K2(z+20,J,0);let G=((z+20)*131+J*997)%100/100;return Z>=G?Y:K2(z+20,J,Q)}).join(""))}function K2($,Z,Q){let X=($*131+Z*997+Q*7919)%65537/65537;return L6[Math.floor(X*L6.length)]}function P6($){let Z=Math.max(0,Math.min(1,$)),Q=q2.split(`
|
|
1268
|
+
`),X=Math.floor($*100);if(Z>=1)return q2;if(Z<=0)return Q.map((z,Y)=>Array.from(z).map((J,G)=>K2(Y,G,0)).join("")).join(`
|
|
1269
|
+
`);return Q.map((z,Y)=>Array.from(z).map((J,G)=>{if(J===" ")return" ";let q=(Y*131+G*997)%100/100;return Z>=q?J:K2(Y,G,X)}).join("")).join(`
|
|
1270
|
+
`)}var q2,G1,P7,L6;var F7=N(()=>{q2=[" ████████████ "," ███ ███ "," ███ ██ ███ "," ██ ██ "," ██ ██ ██ "," ██ ██ "," ██ ██ ██ "," ██ ██ "," ██ ██ "," ██ ██ "," ██ ██ "," ███ ███ "," ███ ███ "," ████████████ "].join(`
|
|
1271
|
+
`),G1=["█▄ ▄█ █ █▀▄ █▄ █ █ █▀▀ █ █ ▀█▀","█ █ █ █ █ █ █ ██ █ █ █ █▀█ █ ","▀ ▀ ▀ ▀▀ ▀ ▀ ▀ ▀▀▀ ▀ ▀ ▀ "];P7=[["wallet","Manage named wallets"],["info","Display wallet metadata"],["balance","Check balance"],["address","Derive address from seed"],["genesis-address","Show genesis address"],["inspect-cost","Display block limits"],["airdrop","Fund from genesis wallet"],["transfer","Send NIGHT tokens"],["dust","Manage dust (fee tokens)"],["contract","Inspect, deploy, call contracts"],["dev","Contract dev loop (watch + deploy)"],["config","Manage CLI config"],["cache","Manage wallet cache"],["localnet","Manage local network"],["serve","DApp Connector server"],["test","Run dApp E2E tests"],["help","Show command usage"],["manual","Full reference manual"]],L6=["░","▒","▓","█","·"," "]});function _7($,Z){return new Promise((Q,X)=>{if(Z?.aborted){Q();return}let z=setTimeout(Q,$);Z?.addEventListener("abort",()=>{clearTimeout(z),Q()},{once:!0})})}async function F6($,Z,Q){let z=q2.split(`
|
|
1272
|
+
`),Y=z.length,J=36,G=Math.max(Y,Z?.length??0),q=3,K=Q?.animated??!0;if(!h0()||!K){let W=h0();for(let P=0;P<G;P++){let O=P<Y?z[P]:"",F=W?m0(O).padEnd(45):O.padEnd(35),I=Z?.[P]??"",_=W&&P<3&&I?M(m0(I)):I;if(W)process.stderr.write(O?m0(O)+"\x1B[36G"+_+`
|
|
1273
|
+
`:"\x1B[36G"+_+`
|
|
1274
|
+
`);else process.stderr.write(F+_+`
|
|
1275
|
+
`)}return}function B(W,P){for(let O=0;O<G;O++){let F=O<W.length?m0(W[O]):"",I=P[O]??"";if(I)process.stderr.write(F+"\x1B[36G"+I+`\x1B[K
|
|
1276
|
+
`);else process.stderr.write(F+`\x1B[K
|
|
1277
|
+
`)}}function H(){process.stderr.write(`\x1B[${G}A`)}for(let W=0;W<=20;W++){if($?.aborted)break;let P=W/20,F=P6(P).split(`
|
|
1278
|
+
`);if(W>0)H();B(F,[]),await _7(I7,$)}let U=20;for(let W=0;W<=U;W++){if($?.aborted)break;let P=W/U,O=O6(P);H();let F=new Array(G).fill(null);for(let I=0;I<O.length;I++)F[I]=M(m0(O[I]));B(z,F),await _7(I7,$)}let j=12;for(let W=0;W<=j;W++){if($?.aborted)break;let P=W/j,O=j6(P);H();let F=new Array(G).fill(null);for(let I=0;I<O.length;I++)F[I]=M(m0(O[I]));B(z,F),await _7(I7,$)}H();let L=new Array(G).fill(null);if(Z)for(let W=0;W<Z.length;W++)if(W<3)L[W]=M(m0(Z[W]));else L[W]=Z[W];B(z,L)}var I7=80;var I6=N(()=>{F7()});import{existsSync as gG,writeFileSync as uG}from"node:fs";import{join as pG}from"node:path";import{tmpdir as cG}from"node:os";function dG(){return(process.env.TERM_SESSION_ID??process.env.ITERM_SESSION_ID??`pid-${process.ppid}`).replace(/[^a-zA-Z0-9-]/g,"_").slice(0,64)}function _6(){return pG(cG(),`mn-intro-${dG()}`)}function T6(){return gG(_6())}function M6(){try{uG(_6(),String(Date.now()))}catch{}}var R6=()=>{};var D6={};m(D6,{default:()=>A6,COMMAND_SPECS:()=>y4});function lG(){let $=[];for(let Z of G1)$.push(Z);$.push(""),$.push(M("Commands"));for(let[Z,Q]of P7)$.push(`${f(Z.padEnd(18))}${Q}`);return $.push(""),$.push(V("midnight (or mn) help <command>")),$.push(V("--json flag available on all commands")),$.push(V("midnight help --agent")+V(" AI & MCP reference")),$}function iG(){process.stderr.write(`
|
|
386
1279
|
Commands:
|
|
387
1280
|
|
|
388
|
-
`);for(let[$,Z]of
|
|
1281
|
+
`);for(let[$,Z]of P7)process.stderr.write(` ${$.padEnd(18)}${Z}
|
|
389
1282
|
`);process.stderr.write(`
|
|
390
1283
|
Usage: midnight <command> (or: mn <command>)
|
|
391
1284
|
`),process.stderr.write(` --json flag available on all commands
|
|
392
1285
|
`),process.stderr.write(` midnight help --agent AI & MCP reference
|
|
393
1286
|
|
|
394
|
-
`)}function
|
|
395
|
-
`+
|
|
1287
|
+
`)}function nG($){if(process.stdout.write(`
|
|
1288
|
+
`+g($.name)+`
|
|
396
1289
|
|
|
397
1290
|
`),process.stdout.write(` ${$.description}
|
|
398
1291
|
|
|
399
|
-
`),process.stdout.write(
|
|
1292
|
+
`),process.stdout.write(t0("Usage:")+`
|
|
400
1293
|
`),process.stdout.write(` ${$.usage}
|
|
401
1294
|
|
|
402
|
-
`),$.flags&&$.flags.length>0){process.stdout.write(
|
|
1295
|
+
`),$.flags&&$.flags.length>0){process.stdout.write(t0("Flags:")+`
|
|
403
1296
|
`);for(let Z of $.flags)process.stdout.write(` ${Z}
|
|
404
1297
|
`);process.stdout.write(`
|
|
405
|
-
`)}if($.examples&&$.examples.length>0){process.stdout.write(
|
|
406
|
-
`);for(let Z of $.examples)process.stdout.write(` ${
|
|
1298
|
+
`)}if($.examples&&$.examples.length>0){process.stdout.write(t0("Examples:")+`
|
|
1299
|
+
`);for(let Z of $.examples)process.stdout.write(` ${V("$")} ${Z}
|
|
407
1300
|
`);process.stdout.write(`
|
|
408
|
-
`)}}function
|
|
1301
|
+
`)}}function oG(){let $={cli:{name:O1,version:g0,description:k7,bin:["midnight","mn"]},globalFlags:[{name:"--json",description:"Output structured JSON to stdout (suppresses all stderr)"},{name:"--wallet <name|file>",description:"Wallet name or path"},{name:"--network <name>",description:"Override network (preprod, preview, undeployed)"},{name:"--version, -v",description:"Print CLI version"},{name:"--help, -h",description:"Show help"}],commands:y4.map((Z)=>({name:Z.name,description:Z.description,usage:Z.usage,flags:Z.flags,examples:Z.examples,jsonFields:Z.jsonFields}))};y($)}function aG(){let $=`
|
|
409
1302
|
MIDNIGHT CLI — AI Agent & MCP Reference
|
|
410
1303
|
========================================
|
|
411
1304
|
|
|
412
|
-
Version: ${
|
|
1305
|
+
Version: ${g0}
|
|
1306
|
+
|
|
1307
|
+
OVERVIEW
|
|
1308
|
+
────────
|
|
1309
|
+
|
|
1310
|
+
midnight-wallet-cli (mn) is a standalone CLI wallet for the Midnight
|
|
1311
|
+
blockchain. It manages wallets, balances, transfers (unshielded and
|
|
1312
|
+
shielded), dust fees, a DApp connector server, contract inspection,
|
|
1313
|
+
E2E testing, and a local devnet — all from the terminal.
|
|
1314
|
+
|
|
1315
|
+
Wallets are network-agnostic: one seed derives addresses for all three
|
|
1316
|
+
networks (undeployed, preprod, preview). Network is chosen at runtime.
|
|
413
1317
|
|
|
414
1318
|
STRUCTURED JSON OUTPUT
|
|
415
1319
|
──────────────────────
|
|
416
1320
|
|
|
417
|
-
Every command supports
|
|
1321
|
+
Every command supports --json. When passed:
|
|
418
1322
|
- stdout receives a single line of JSON
|
|
419
1323
|
- stderr is fully suppressed (no spinners, no formatting)
|
|
420
|
-
- Errors
|
|
1324
|
+
- Errors: {"error":true,"code":"...","message":"...","exitCode":N}
|
|
421
1325
|
|
|
422
1326
|
Usage: midnight <command> [args] --json
|
|
423
1327
|
|
|
@@ -426,13 +1330,12 @@ CAPABILITY MANIFEST
|
|
|
426
1330
|
|
|
427
1331
|
midnight help --json
|
|
428
1332
|
|
|
429
|
-
Outputs a machine-readable
|
|
430
|
-
|
|
431
|
-
for programmatic discovery of CLI capabilities.
|
|
1333
|
+
Outputs a machine-readable manifest with all commands, flags,
|
|
1334
|
+
examples, and JSON field schemas.
|
|
432
1335
|
|
|
433
1336
|
COMMANDS & JSON SCHEMAS
|
|
434
1337
|
───────────────────────
|
|
435
|
-
${
|
|
1338
|
+
${y4.filter((Z)=>Z.jsonFields).map((Z)=>{let Q=Object.entries(Z.jsonFields).map(([X,z])=>` ${X.padEnd(20)}${z}`).join(`
|
|
436
1339
|
`);return`
|
|
437
1340
|
${Z.name}
|
|
438
1341
|
${Z.usage}
|
|
@@ -440,11 +1343,115 @@ ${_1.filter((Z)=>Z.jsonFields).map((Z)=>{let Q=Object.entries(Z.jsonFields).map(
|
|
|
440
1343
|
${Q}`}).join(`
|
|
441
1344
|
`)}
|
|
442
1345
|
|
|
443
|
-
|
|
1346
|
+
SHIELDED TRANSACTIONS
|
|
1347
|
+
─────────────────────
|
|
1348
|
+
|
|
1349
|
+
Midnight supports private (shielded) transactions using zero-knowledge
|
|
1350
|
+
proofs. The CLI provides full shielded support:
|
|
1351
|
+
|
|
1352
|
+
Balance (shows both unshielded + shielded):
|
|
1353
|
+
midnight balance --json
|
|
1354
|
+
→ { address, shieldedAddress, network,
|
|
1355
|
+
unshielded: { NIGHT, utxoCount },
|
|
1356
|
+
shielded: { NIGHT, availableCoins, pendingCoins } }
|
|
1357
|
+
|
|
1358
|
+
Shielded airdrop (localnet only — genesis has 250M shielded NIGHT):
|
|
1359
|
+
midnight airdrop 100 --shielded --json
|
|
1360
|
+
→ { txHash, amount, shieldedAddress, network, type: "shielded" }
|
|
1361
|
+
|
|
1362
|
+
Shielded transfer (to shielded address or wallet name):
|
|
1363
|
+
midnight transfer alice 50 --shielded --json
|
|
1364
|
+
midnight transfer mn_shield-addr_... 50 --shielded --json
|
|
1365
|
+
→ { txHash, amount, recipient, network, type: "shielded" }
|
|
1366
|
+
|
|
1367
|
+
Positional address balance (unshielded only, fast GraphQL):
|
|
1368
|
+
midnight balance mn_addr_... --json
|
|
1369
|
+
→ { address, network, balances, utxoCount, txCount }
|
|
1370
|
+
|
|
1371
|
+
Note: there is no self-shielding. Shielded coins come from receiving
|
|
1372
|
+
transfers from wallets that already have shielded tokens.
|
|
1373
|
+
|
|
1374
|
+
WALLET NAME RESOLUTION
|
|
1375
|
+
──────────────────────
|
|
1376
|
+
|
|
1377
|
+
Transfer commands accept wallet names instead of full addresses:
|
|
1378
|
+
|
|
1379
|
+
midnight transfer alice 10 → resolves alice's unshielded address
|
|
1380
|
+
midnight transfer alice 10 --shielded → resolves alice's shielded address
|
|
1381
|
+
|
|
1382
|
+
Names are resolved from ~/.midnight/wallets/<name>.json. If the input
|
|
1383
|
+
starts with mn_addr_ or mn_shield-addr_, it's used as an address directly.
|
|
1384
|
+
|
|
1385
|
+
DAPP CONNECTOR
|
|
1386
|
+
──────────────
|
|
1387
|
+
|
|
1388
|
+
midnight serve [--port 9932] [--approve-all] [--network name]
|
|
1389
|
+
|
|
1390
|
+
Starts a WebSocket JSON-RPC server implementing the Midnight
|
|
1391
|
+
ConnectedAPI interface (same as the Lace browser wallet). Any DApp
|
|
1392
|
+
can connect to it — no browser extension needed.
|
|
1393
|
+
|
|
1394
|
+
- Port default: 9932, localhost only
|
|
1395
|
+
- Read operations: auto-approved
|
|
1396
|
+
- Write operations: terminal approval prompt (or --approve-all)
|
|
1397
|
+
- --no-auto-approve-reads: require approval for everything
|
|
1398
|
+
|
|
1399
|
+
DApp developers connect via the midnight-wallet-connector npm package:
|
|
1400
|
+
|
|
1401
|
+
npm install midnight-wallet-connector
|
|
1402
|
+
|
|
1403
|
+
import { createWalletClient } from 'midnight-wallet-connector';
|
|
1404
|
+
const wallet = await createWalletClient({
|
|
1405
|
+
url: 'ws://localhost:9932',
|
|
1406
|
+
networkId: 'undeployed', // or 'preprod', 'preview' — lowercase per @midnight-ntwrk/wallet-sdk-abstractions
|
|
1407
|
+
});
|
|
1408
|
+
const balances = await wallet.getUnshieldedBalances();
|
|
1409
|
+
|
|
1410
|
+
Reference DApp: https://github.com/nel349/midnight-starship
|
|
1411
|
+
|
|
1412
|
+
SMART CONTRACTS
|
|
1413
|
+
───────────────
|
|
1414
|
+
|
|
1415
|
+
Run these commands from the root of a dApp project that contains a
|
|
1416
|
+
compiled Compact contract (managed/ directory with .js and .d.ts files).
|
|
1417
|
+
|
|
1418
|
+
Inspect — show circuits, witnesses, types:
|
|
1419
|
+
midnight contract inspect [--path <dir>] [--json]
|
|
1420
|
+
→ { name, compilerVersion, circuits: [...], witnesses: [...] }
|
|
1421
|
+
|
|
1422
|
+
Deploy — deploy a contract to the network:
|
|
1423
|
+
midnight contract deploy [--network <name>] [--json]
|
|
1424
|
+
→ { contractName, address, network }
|
|
1425
|
+
Requires: funded wallet with dust. Auto-starts mn serve if needed.
|
|
1426
|
+
|
|
1427
|
+
Call — call a circuit on a deployed contract:
|
|
1428
|
+
midnight contract call --address <addr> --circuit <name> [--args '<json>'] [--json]
|
|
1429
|
+
→ { contractName, circuit, address, network, status }
|
|
1430
|
+
Example: midnight contract call --address abc123... --circuit post --args '["Hello!"]'
|
|
1431
|
+
|
|
1432
|
+
State — read ledger state of a deployed contract:
|
|
1433
|
+
midnight contract state --address <addr> [--network <name>] [--json]
|
|
1434
|
+
→ { address, network, fields: { key: value }, maps: { key: { size } } }
|
|
1435
|
+
|
|
1436
|
+
Full workflow example:
|
|
1437
|
+
cd my-dapp
|
|
1438
|
+
midnight contract inspect # see what circuits exist
|
|
1439
|
+
midnight contract deploy # deploy to localnet
|
|
1440
|
+
midnight contract call --address <addr> --circuit post --args '["Hello"]'
|
|
1441
|
+
midnight contract state --address <addr> # read on-chain state
|
|
1442
|
+
|
|
1443
|
+
E2E TESTING
|
|
444
1444
|
───────────
|
|
445
1445
|
|
|
446
|
-
|
|
447
|
-
|
|
1446
|
+
midnight test run [--suite <name>] [--json]
|
|
1447
|
+
midnight test list [--json]
|
|
1448
|
+
midnight test results [--all] [--json]
|
|
1449
|
+
|
|
1450
|
+
Runs E2E tests for Midnight dApps defined in dapp.test.json. Includes
|
|
1451
|
+
contract deployment, circuit calls, and state verification.
|
|
1452
|
+
|
|
1453
|
+
ERROR CODES
|
|
1454
|
+
───────────
|
|
448
1455
|
|
|
449
1456
|
Code Exit Meaning
|
|
450
1457
|
INVALID_ARGS 2 Missing or invalid arguments
|
|
@@ -458,163 +1465,307 @@ When --json is active and an error occurs, the output is:
|
|
|
458
1465
|
CANCELLED 7 Operation cancelled (SIGINT)
|
|
459
1466
|
UNKNOWN 1 Unclassified error
|
|
460
1467
|
|
|
461
|
-
EXIT CODES
|
|
462
|
-
──────────
|
|
463
|
-
|
|
464
|
-
0 Success
|
|
465
|
-
1 General error
|
|
466
|
-
2 Invalid arguments / usage
|
|
467
|
-
3 Wallet not found
|
|
468
|
-
4 Network / connection error
|
|
469
|
-
5 Insufficient balance
|
|
470
|
-
6 Transaction rejected
|
|
471
|
-
7 Operation cancelled (SIGINT)
|
|
472
|
-
|
|
473
1468
|
MCP SERVER
|
|
474
1469
|
──────────
|
|
475
1470
|
|
|
476
1471
|
The CLI includes an MCP (Model Context Protocol) server for native
|
|
477
|
-
AI agent integration.
|
|
478
|
-
|
|
1472
|
+
AI agent integration. Agents call typed tools directly via JSON-RPC
|
|
1473
|
+
over stdio — no shell spawning or output parsing.
|
|
479
1474
|
|
|
480
|
-
Setup
|
|
1475
|
+
Setup:
|
|
481
1476
|
|
|
482
|
-
Claude Code (.mcp.json
|
|
483
|
-
{
|
|
484
|
-
"mcpServers": {
|
|
485
|
-
"midnight-wallet": {
|
|
486
|
-
"type": "stdio",
|
|
487
|
-
"command": "midnight-wallet-mcp"
|
|
488
|
-
}
|
|
489
|
-
}
|
|
490
|
-
}
|
|
1477
|
+
Claude Code (.mcp.json):
|
|
1478
|
+
{ "mcpServers": { "midnight-wallet": { "command": "midnight-wallet-mcp" } } }
|
|
491
1479
|
|
|
492
|
-
|
|
1480
|
+
CLI: claude mcp add --transport stdio midnight-wallet -- midnight-wallet-mcp
|
|
493
1481
|
|
|
494
1482
|
Cursor (.cursor/mcp.json):
|
|
495
|
-
{
|
|
496
|
-
|
|
497
|
-
|
|
498
|
-
|
|
499
|
-
}
|
|
500
|
-
}
|
|
501
|
-
}
|
|
1483
|
+
{ "mcpServers": { "midnight-wallet": { "command": "midnight-wallet-mcp" } } }
|
|
1484
|
+
|
|
1485
|
+
VS Code (.vscode/mcp.json):
|
|
1486
|
+
{ "servers": { "midnight-wallet": { "type": "stdio", "command": "midnight-wallet-mcp" } } }
|
|
502
1487
|
|
|
503
1488
|
If not installed globally, use "command": "npx" with
|
|
504
|
-
"args": ["-y", "midnight-wallet-cli@latest", "--mcp"]
|
|
1489
|
+
"args": ["-y", "midnight-wallet-cli@latest", "--mcp"].
|
|
505
1490
|
|
|
506
|
-
AVAILABLE MCP TOOLS (
|
|
1491
|
+
AVAILABLE MCP TOOLS (31)
|
|
507
1492
|
────────────────────────
|
|
508
1493
|
|
|
509
|
-
|
|
1494
|
+
Wallet Management
|
|
510
1495
|
midnight_wallet_generate Create a named wallet name
|
|
511
1496
|
midnight_wallet_list List all wallets —
|
|
512
1497
|
midnight_wallet_use Set active wallet name
|
|
513
|
-
midnight_wallet_info Show wallet details
|
|
1498
|
+
midnight_wallet_info Show wallet details (incl. shielded address) —
|
|
514
1499
|
midnight_wallet_remove Remove a named wallet name
|
|
515
1500
|
midnight_generate Generate or restore a wallet (deprecated) —
|
|
1501
|
+
|
|
1502
|
+
Balance & Info
|
|
516
1503
|
midnight_info Display wallet metadata —
|
|
517
|
-
midnight_balance Check unshielded balance
|
|
1504
|
+
midnight_balance Check unshielded + shielded balance —
|
|
518
1505
|
midnight_address Derive address from seed seed
|
|
519
1506
|
midnight_genesis_address Genesis wallet address —
|
|
520
1507
|
midnight_inspect_cost Display block cost limits —
|
|
1508
|
+
|
|
1509
|
+
Transactions
|
|
521
1510
|
midnight_airdrop Fund wallet from genesis (undeployed only) amount
|
|
522
|
-
midnight_transfer Send NIGHT
|
|
1511
|
+
midnight_transfer Send NIGHT (two-step — returns pending token) to, amount
|
|
523
1512
|
midnight_dust_register Register UTXOs for dust generation —
|
|
524
1513
|
midnight_dust_status Check dust balance and registration —
|
|
1514
|
+
|
|
1515
|
+
Consent
|
|
1516
|
+
midnight_confirm_operation Redeem a pending token returned by a destructive token
|
|
1517
|
+
tool (step 2 of the confirmation flow)
|
|
1518
|
+
|
|
1519
|
+
Configuration
|
|
525
1520
|
midnight_config_get Read a config value key
|
|
526
1521
|
midnight_config_set Write a config value key, value
|
|
527
1522
|
midnight_config_unset Reset a config value to default key
|
|
528
1523
|
midnight_cache_clear Clear cached wallet sync state —
|
|
1524
|
+
|
|
1525
|
+
Local Network
|
|
529
1526
|
midnight_localnet_up Start local network (Docker) —
|
|
530
1527
|
midnight_localnet_stop Stop local network (preserves state) —
|
|
531
|
-
midnight_localnet_down Full teardown (
|
|
1528
|
+
midnight_localnet_down Full teardown (volumes + undeployed cache) —
|
|
532
1529
|
midnight_localnet_status Show service status and ports —
|
|
533
1530
|
midnight_localnet_clean Remove conflicting containers —
|
|
1531
|
+
midnight_localnet_logs Snapshot of recent logs from each service —
|
|
534
1532
|
|
|
535
|
-
|
|
536
|
-
|
|
1533
|
+
Contracts (Compact smart contracts)
|
|
1534
|
+
midnight_contract_inspect Read circuits, witnesses, ledger shape from path / managed / name
|
|
1535
|
+
compiled artifact (no chain access needed)
|
|
1536
|
+
midnight_contract_state Query a deployed contract's ledger state address, path
|
|
1537
|
+
midnight_contract_deploy Deploy a compiled contract (two-step pending token) path / managed
|
|
1538
|
+
midnight_contract_call Call a circuit on a deployed contract address, circuit
|
|
1539
|
+
(two-step pending token; args auto-coerce numbers
|
|
1540
|
+
and "123n" strings to BigInt, [0–255] to Uint8Array)
|
|
537
1541
|
|
|
538
|
-
|
|
1542
|
+
Test Framework
|
|
1543
|
+
midnight_test_create Generate a CLI or browser test scaffold from the path, strategy,
|
|
1544
|
+
compiled contract port, build-cmd
|
|
539
1545
|
|
|
540
|
-
|
|
541
|
-
|
|
1546
|
+
Optional params shared by wallet tools: wallet, network.
|
|
1547
|
+
All tools return JSON. Every response carries _serverVersion so a stale
|
|
1548
|
+
MCP server (CLI on disk says X, responses still say Y) is detectable.
|
|
1549
|
+
Errors: {error, code, message, _serverVersion}.
|
|
1550
|
+
|
|
1551
|
+
TOOL ANNOTATIONS (MCP safety hints)
|
|
1552
|
+
───────────────────────────────────
|
|
1553
|
+
|
|
1554
|
+
Every tool carries MCP-spec annotations so clients can apply safety policy
|
|
1555
|
+
without hardcoding per-tool rules:
|
|
1556
|
+
|
|
1557
|
+
readOnlyHint Safe to call without user consent (balance, info, list).
|
|
1558
|
+
destructiveHint Moves funds, deletes files, or tears down infra — treat
|
|
1559
|
+
as requiring user consent.
|
|
1560
|
+
idempotentHint Repeated calls with same args yield the same result.
|
|
1561
|
+
openWorldHint Touches the network / chain / Docker — can fail
|
|
1562
|
+
non-deterministically.
|
|
1563
|
+
|
|
1564
|
+
TWO-STEP CONFIRMATION FLOW (midnight_transfer)
|
|
1565
|
+
──────────────────────────────────────────────
|
|
1566
|
+
|
|
1567
|
+
midnight_transfer does NOT execute on the first call. It returns a pending
|
|
1568
|
+
token that the agent must show to the user for consent, then redeem via
|
|
1569
|
+
midnight_confirm_operation to actually execute.
|
|
1570
|
+
|
|
1571
|
+
Call 1: midnight_transfer({ to, amount, wallet, network })
|
|
1572
|
+
→ { pending: true, token, description, expiresAt, nextStep }
|
|
1573
|
+
Show: the description to the user verbatim.
|
|
1574
|
+
Call 2: midnight_confirm_operation({ token })
|
|
1575
|
+
→ actual transfer result (txHash, etc)
|
|
1576
|
+
|
|
1577
|
+
Tokens are single-use and expire after 5 minutes.
|
|
542
1578
|
|
|
543
|
-
|
|
544
|
-
|
|
545
|
-
3. midnight_airdrop → Fund wallet (amount: "1000")
|
|
546
|
-
4. midnight_dust_register → Register UTXOs for fee tokens
|
|
547
|
-
5. midnight_balance → Verify balance
|
|
548
|
-
6. midnight_transfer → Send tokens (to: "mn_addr_...", amount: "100")
|
|
549
|
-
7. midnight_dust_status → Check remaining dust for fees
|
|
1579
|
+
MCP RESOURCES (skill file)
|
|
1580
|
+
──────────────────────────
|
|
550
1581
|
|
|
551
|
-
|
|
552
|
-
────────────────
|
|
1582
|
+
The server exposes one MCP Resource:
|
|
553
1583
|
|
|
554
|
-
|
|
555
|
-
midnight
|
|
556
|
-
|
|
1584
|
+
uri midnight-wallet://skill
|
|
1585
|
+
name midnight-wallet skill
|
|
1586
|
+
mimeType text/markdown
|
|
557
1587
|
|
|
558
|
-
|
|
1588
|
+
Call resources/read on connect to fetch a conversational guide covering
|
|
1589
|
+
intent routing, canonical flows, safety rules, and error recovery. Ground
|
|
1590
|
+
responses in it instead of training-data guesses.
|
|
1591
|
+
|
|
1592
|
+
TYPICAL AGENT WORKFLOWS
|
|
1593
|
+
───────────────────────
|
|
1594
|
+
|
|
1595
|
+
Local development (undeployed):
|
|
1596
|
+
1. midnight_localnet_up → Start node, indexer, proof server
|
|
1597
|
+
2. midnight_wallet_generate → Create wallet (name: "alice")
|
|
1598
|
+
3. midnight_config_set → Set network (key: "network", value: "undeployed")
|
|
1599
|
+
4. midnight_airdrop → Fund wallet (amount: "1000")
|
|
1600
|
+
5. midnight_dust_register → Register for fee tokens
|
|
1601
|
+
6. midnight_balance → Check unshielded + shielded balance
|
|
1602
|
+
7. midnight_transfer → Returns pending token + description
|
|
1603
|
+
8. (show description to user, get explicit consent)
|
|
1604
|
+
9. midnight_confirm_operation → Redeem token → actual transfer runs
|
|
1605
|
+
10. midnight_dust_status → Check remaining dust
|
|
1606
|
+
|
|
1607
|
+
Shielded workflow:
|
|
1608
|
+
1. midnight_airdrop → Fund shielded (amount: "100", shielded: "true")
|
|
1609
|
+
2. midnight_balance → Shows both unshielded + shielded
|
|
1610
|
+
3. midnight_transfer → Returns pending token for shielded send
|
|
1611
|
+
4. (show + consent) → midnight_confirm_operation
|
|
1612
|
+
|
|
1613
|
+
Testnet (preprod/preview):
|
|
1614
|
+
1. midnight_wallet_generate → Create wallet
|
|
1615
|
+
2. midnight_config_set → Set network (key: "network", value: "preview")
|
|
1616
|
+
3. (fund via faucet: https://faucet.preview.midnight.network/)
|
|
1617
|
+
4. midnight_dust_register → Register for fees
|
|
1618
|
+
5. midnight_balance → Check balance
|
|
1619
|
+
6. midnight_transfer → midnight_confirm_operation → transfer
|
|
1620
|
+
|
|
1621
|
+
Contract development (CLI only — not an MCP tool):
|
|
1622
|
+
Run "mn dev" in a Compact project. It auto-starts localnet, provisions
|
|
1623
|
+
3 funded wallets (dev-alice/dev-bob/dev-carol), compiles on save, and
|
|
1624
|
+
accepts a "d" keystroke to deploy the current artifact with dev-alice.
|
|
1625
|
+
See "mn help dev".
|
|
1626
|
+
|
|
1627
|
+
EXAMPLE CLI COMMANDS
|
|
1628
|
+
────────────────────
|
|
1629
|
+
|
|
1630
|
+
# Generate wallet (all 3 network addresses + seed)
|
|
1631
|
+
midnight wallet generate alice --json
|
|
1632
|
+
# → {"name":"alice","addresses":{...},"activeAddress":"mn_addr_...","activeNetwork":"undeployed","seed":"...","file":"..."}
|
|
1633
|
+
|
|
1634
|
+
# Balance (full sync — unshielded + shielded)
|
|
559
1635
|
midnight balance --json
|
|
560
|
-
# → {"address":"mn_addr_...","network":"undeployed","
|
|
561
|
-
|
|
562
|
-
#
|
|
563
|
-
midnight
|
|
564
|
-
# → {"txHash":"...","amount":
|
|
565
|
-
|
|
566
|
-
#
|
|
567
|
-
midnight transfer
|
|
568
|
-
# → {"txHash":"...","amount":
|
|
569
|
-
|
|
570
|
-
Available commands: ${_1.map((z)=>z.name).join(", ")}`);O4(X);return}if(!process.stderr.isTTY){P4();return}let Q=F4();await H9(void 0,Q)}var _1;var O9=x(()=>{s();L9();P$();v1();_1=[{name:"wallet",description:"Manage named wallets (generate, list, use, info, remove)",usage:"midnight wallet <generate|list|use|info|remove> [args]",flags:["generate <name> Create a new named wallet and set it as active","list Show all wallets with active marker","use <name> Set active wallet","info [name] Show wallet details (active wallet if no name)","remove <name> Delete a wallet (refuses active or last wallet)","","generate flags:","--network <name> Network: preprod, preview, undeployed","--seed <hex> Restore from existing seed (64-char hex)",'--mnemonic "..." Restore from BIP-39 mnemonic (24 words)',"--force Overwrite existing wallet file"],examples:["midnight wallet generate alice --network preprod","midnight wallet list","midnight wallet use alice","midnight wallet info alice","midnight wallet remove bob"],jsonFields:{name:"Wallet name",address:"Wallet address (bech32m)",network:"Network name",active:"Whether this is the active wallet",wallets:"Array of wallet info objects (list only)"}},{name:"generate",description:'(Deprecated — use "midnight wallet generate <name>" instead)',usage:'midnight generate [--network <name>] [--seed <hex>] [--mnemonic "..."] [--output <file>] [--force]',flags:["--network <name> Network: preprod, preview, undeployed","--seed <hex> Restore from existing seed (64-char hex)",'--mnemonic "..." Restore from BIP-39 mnemonic (24 words)',"--output <file> Custom output path (default: ~/.midnight/wallet.json)","--force Overwrite existing wallet file"],examples:["midnight generate --network preprod"],jsonFields:{address:"Generated wallet address (bech32m)",network:"Network name",seed:"Hex-encoded 32-byte seed",mnemonic:"BIP-39 mnemonic (24 words, only if generated or provided)",file:"Path where wallet file was saved",createdAt:"ISO 8601 creation timestamp"}},{name:"info",description:"Display wallet address, network, creation date (no secrets shown)",usage:"midnight info [--wallet <name|file>]",flags:["--wallet <name|file> Wallet name or path"],examples:["midnight info","midnight info --wallet my-wallet.json"],jsonFields:{address:"Wallet address (bech32m)",network:"Network name",createdAt:"ISO 8601 creation timestamp",file:"Wallet file path"}},{name:"balance",description:"Check unshielded balance via GraphQL subscription",usage:"midnight balance [address] [--network <name>] [--indexer-ws <url>]",flags:["<address> Address to check (or reads from wallet file)","--network <name> Override network detection","--indexer-ws <url> Custom indexer WebSocket URL"],examples:["midnight balance","midnight balance mn_addr_preprod1...","midnight balance --network preprod"],jsonFields:{address:"Checked address (bech32m)",network:"Network name",balances:"Object mapping token type to balance string",utxoCount:"Number of UTXOs",txCount:"Number of transactions synced"}},{name:"address",description:"Derive and display an unshielded address from a seed",usage:"midnight address --seed <hex> [--network <name>] [--index <n>]",flags:["--seed <hex> Seed to derive from (required, 64-char hex)","--network <name> Network for address prefix (default: resolved)","--index <n> Key derivation index (default: 0)"],examples:["midnight address --seed 0123456789abcdef... --network preprod","midnight address --seed 0123456789abcdef... --index 1"],jsonFields:{address:"Derived address (bech32m)",network:"Network name",index:"Key derivation index",path:"BIP-44 derivation path"}},{name:"genesis-address",description:"Display the genesis wallet address (seed 0x01) for a network",usage:"midnight genesis-address [--network <name>]",flags:["--network <name> Network for address prefix (default: resolved)"],examples:["midnight genesis-address --network undeployed","midnight genesis-address --network preprod"],jsonFields:{address:"Genesis wallet address (bech32m)",network:"Network name"}},{name:"inspect-cost",description:"Display current block limits derived from LedgerParameters",usage:"midnight inspect-cost",examples:["midnight inspect-cost"],jsonFields:{readTime:"Read time limit (picoseconds)",computeTime:"Compute time limit (picoseconds)",blockUsage:"Block usage limit (bytes)",bytesWritten:"Bytes written limit (bytes)",bytesChurned:"Bytes churned limit (bytes)"}},{name:"airdrop",description:"Fund your wallet from the genesis wallet (undeployed network only)",usage:"midnight airdrop <amount> [--wallet <name|file>] [--no-cache]",flags:["<amount> Amount in NIGHT to airdrop","--wallet <name|file> Wallet name or path","--no-cache Bypass wallet state cache"],examples:["midnight airdrop 1000","midnight airdrop 0.5 --wallet my-wallet.json"],jsonFields:{txHash:"Transaction hash",amount:"Amount airdropped (NIGHT string)",recipient:"Recipient address (bech32m)",network:"Network name"}},{name:"transfer",description:"Send NIGHT tokens to another address",usage:"midnight transfer <to> <amount> [--wallet <name|file>] [--proof-server <url>] [--node <url>] [--indexer-ws <url>] [--no-cache]",flags:["<to> Recipient bech32m address","<amount> Amount in NIGHT to send","--wallet <name|file> Wallet name or path","--proof-server <url> Override proof server URL","--node <url> Override substrate node RPC URL","--indexer-ws <url> Override indexer WebSocket URL","--no-cache Bypass wallet state cache"],examples:["midnight transfer mn_addr_undeployed1... 100","midnight transfer mn_addr_preprod1... 50 --wallet my-wallet.json"],jsonFields:{txHash:"Transaction hash",amount:"Amount transferred (NIGHT string)",recipient:"Recipient address (bech32m)",network:"Network name"}},{name:"dust",description:"Register UTXOs for dust (fee token) generation or check status",usage:"midnight dust <register|status> [--wallet <name|file>] [--proof-server <url>] [--node <url>] [--indexer-ws <url>] [--no-cache]",flags:["register Register NIGHT UTXOs for dust generation","status Check dust registration status and balance","--wallet <name|file> Wallet name or path","--proof-server <url> Override proof server URL","--node <url> Override substrate node RPC URL","--indexer-ws <url> Override indexer WebSocket URL","--no-cache Bypass wallet state cache"],examples:["midnight dust register","midnight dust status"],jsonFields:{subcommand:"register or status",dustBalance:"Dust balance (raw bigint string)",registered:"Number of registered UTXOs (status only)",unregistered:"Number of unregistered UTXOs (status only)",nightBalance:"NIGHT balance (raw bigint string, status only)",dustAvailable:"Whether dust tokens are available (status only)",txHash:"Registration transaction hash (register only, if submitted)"}},{name:"config",description:"Manage persistent config (default network, endpoints, etc.)",usage:"midnight config <get|set|unset> <key> [value]",flags:["get <key> Read a config value","set <key> <value> Write a config value","unset <key> Reset a config value to its default","","Keys: network, wallet, proof-server, node, indexer-ws"],examples:["midnight config get network","midnight config set network preprod","midnight config set wallet alice","midnight config set proof-server http://localhost:6300","midnight config set node wss://rpc.preprod.midnight.network","midnight config set indexer-ws wss://indexer.preprod.midnight.network/api/v3/graphql/ws","midnight config unset proof-server"],jsonFields:{action:"get, set, or unset",key:"Config key name",value:"Config value"}},{name:"cache",description:"Manage wallet state cache (clear cached sync data)",usage:"midnight cache clear [--network <name>] [--wallet <name|file>]",flags:["clear Clear cached wallet state","--network <name> Only clear cache for this network","--wallet <name|file> Only clear cache for this wallet"],examples:["midnight cache clear","midnight cache clear --network preprod","midnight cache clear --wallet alice"],jsonFields:{action:"clear",scope:"all, network, or wallet",network:"Network name (when scoped)",wallet:"Wallet name (when scoped)"}},{name:"serve",description:"Start DApp Connector server over WebSocket JSON-RPC",usage:"midnight serve [--port <n>] [--wallet <name|file>] [--network <name>] [--proof-server <url>] [--node <url>] [--indexer-ws <url>] [--approve-all] [--no-auto-approve-reads] [--no-cache] [--json]",flags:["--port <n> Server port (default: 9932)","--wallet <name|file> Wallet name or path","--network <name> Override network detection","--proof-server <url> Override proof server URL","--node <url> Override substrate node RPC URL","--indexer-ws <url> Override indexer WebSocket URL","--approve-all Auto-approve all requests (reads + writes)","--no-auto-approve-reads Prompt for read methods too","--no-cache Bypass wallet state cache"],examples:["midnight serve","midnight serve --port 8080","midnight serve --approve-all"],jsonFields:{port:"Server port number",network:"Network name",address:"Wallet address (bech32m)",status:"Server status (listening)"}},{name:"localnet",description:"Manage a local Midnight network via Docker Compose",usage:"midnight localnet <up|stop|down|status|logs|clean>",flags:["up Start the local network (node, indexer, proof server)","stop Stop containers (preserves state for fast restart)","down Remove containers, networks, and volumes (full teardown)","status Show service status and ports","logs Stream service logs (Ctrl+C to stop)","clean Remove conflicting containers from other setups"],examples:["midnight localnet up","midnight localnet stop","midnight localnet status","midnight localnet down","midnight localnet clean"],jsonFields:{subcommand:"up, stop, down, status, or clean",services:"Array of { name, state, port, health? } (up/status only)",status:"Operation result message (stop/down/clean)",removed:"Array of removed container names (clean only)"}},{name:"help",description:"Show usage for all commands or a specific command",usage:"midnight help [command]",examples:["midnight help","midnight help balance"],jsonFields:{cli:"CLI metadata (name, version, description)",globalFlags:"Array of global flag descriptions",commands:"Array of command specs with jsonFields"}}]});import*as I9 from"node:readline";function R4($){return T4.has($)}function D4($){return M4.has($)}function A4($){let Z=[],Q=$.dappName?`Request from "${$.dappName}"`:"DApp Request";if(Z.push(y(Q)),Z.push(""),Z.push(` ${B("Action:")} ${t($.method)}`),Z.push(` ${B("Network:")} ${$.network}`),$.details.length>0){Z.push("");for(let X of $.details)Z.push(` ${B(X.label+":")} ${X.value}`)}return Z.push(""),Z.push(` ${u("[A]pprove")} ${c("[R]eject")}`),W2(Z,"heavy")}async function _9($,Z={}){if(Z.approveAll)return process.stderr.write(B(` Auto-approved: ${$.method}`)+`
|
|
571
|
-
|
|
572
|
-
|
|
573
|
-
|
|
574
|
-
|
|
575
|
-
|
|
576
|
-
|
|
577
|
-
|
|
578
|
-
|
|
579
|
-
|
|
580
|
-
|
|
581
|
-
|
|
582
|
-
|
|
583
|
-
|
|
584
|
-
|
|
585
|
-
|
|
586
|
-
|
|
587
|
-
|
|
588
|
-
|
|
589
|
-
|
|
590
|
-
|
|
591
|
-
|
|
592
|
-
|
|
1636
|
+
# → {"address":"mn_addr_...","shieldedAddress":"mn_shield-addr_...","network":"undeployed","unshielded":{"NIGHT":"1000.000000","utxoCount":1},"shielded":{"NIGHT":"10.000000","availableCoins":1,"pendingCoins":0}}
|
|
1637
|
+
|
|
1638
|
+
# Transfer by wallet name
|
|
1639
|
+
midnight transfer bob 100 --json
|
|
1640
|
+
# → {"txHash":"00ab...","amount":100,"recipient":"mn_addr_...","network":"undeployed"}
|
|
1641
|
+
|
|
1642
|
+
# Shielded transfer
|
|
1643
|
+
midnight transfer bob 50 --shielded --json
|
|
1644
|
+
# → {"txHash":"00cd...","amount":50,"recipient":"mn_shield-addr_...","network":"undeployed","type":"shielded"}
|
|
1645
|
+
|
|
1646
|
+
# Shielded airdrop (localnet only)
|
|
1647
|
+
midnight airdrop 100 --shielded --json
|
|
1648
|
+
# → {"txHash":"00ef...","amount":100,"shieldedAddress":"mn_shield-addr_...","network":"undeployed","type":"shielded"}
|
|
1649
|
+
|
|
1650
|
+
# Contract inspection
|
|
1651
|
+
midnight contract inspect --json
|
|
1652
|
+
# → {"name":"bboard","compilerVersion":"0.30.0","circuits":[{"name":"post",...}],"witnesses":[...]}
|
|
1653
|
+
|
|
1654
|
+
# Deploy contract (from dApp root directory)
|
|
1655
|
+
midnight contract deploy --json
|
|
1656
|
+
# → {"subcommand":"deploy","contractName":"bboard","address":"6cc5...","network":"undeployed"}
|
|
1657
|
+
|
|
1658
|
+
# Call a circuit
|
|
1659
|
+
midnight contract call --address 6cc5... --circuit post --args '["Hello from CLI!"]' --json
|
|
1660
|
+
# → {"subcommand":"call","circuit":"post","status":"success"}
|
|
1661
|
+
|
|
1662
|
+
# Read contract state
|
|
1663
|
+
midnight contract state --address 6cc5... --json
|
|
1664
|
+
# → {"subcommand":"state","fields":{"state":"1","message":"...","owner":"..."}}
|
|
1665
|
+
|
|
1666
|
+
# Start DApp connector
|
|
1667
|
+
midnight serve --network preview --approve-all
|
|
1668
|
+
# DApps connect at ws://localhost:9932
|
|
1669
|
+
`;process.stdout.write($)}async function A6($){if(x($,"agent")){aG();return}if(x($,"json")){oG();return}let Z=$.subcommand;if(Z){let G=y4.find((q)=>q.name===Z);if(!G)throw new t(`Unknown command: "${Z}"
|
|
1670
|
+
Available commands: ${y4.map((q)=>q.name).join(", ")}`);nG(G);return}if(!process.stderr.isTTY){iG();return}let Q=lG(),X=x($,"intro"),z=x($,"no-intro"),Y=T6(),J=X||!z&&!Y;if(await F6(void 0,Q,{animated:J}),J)M6()}var y4;var T7=N(()=>{K0();I6();F7();R6();l0();j1();y4=[{name:"wallet",description:"Manage named wallets (generate, list, use, info, remove, seed)",usage:"midnight wallet <generate|list|use|info|remove|seed> [args]",flags:["generate <name> Create a new named wallet and set it as active","list Show all wallets with active marker","use <name> Set active wallet","info [name] Show wallet details (active wallet if no name)","remove <name> Delete a wallet (refuses active or last wallet)","seed [name] Reveal the seed + mnemonic (prompts for confirmation)","","generate flags:","--network <name> Network: preprod, preview, undeployed","--seed <hex> Restore from existing seed (64-char hex)",'--mnemonic "..." Restore from BIP-39 mnemonic (24 words)',"--force Overwrite existing wallet file","","seed flags:","--entropy Also output the 32-byte BIP-39 entropy alongside the"," 64-byte PBKDF2 seed (use when another tool expects"," the shorter entropy format — NB: the two derive"," DIFFERENT Midnight wallets from the same mnemonic)","--json Print JSON (skips interactive confirmation)"],examples:["midnight wallet generate alice --network preprod","midnight wallet list","midnight wallet use alice","midnight wallet info alice","midnight wallet remove bob","midnight wallet seed alice --entropy"],jsonFields:{name:"Wallet name",address:"Wallet address (bech32m)",network:"Network name",active:"Whether this is the active wallet",wallets:"Array of wallet info objects (list only)"}},{name:"generate",description:'(Deprecated — use "midnight wallet generate <name>" instead)',usage:'midnight generate [--network <name>] [--seed <hex>] [--mnemonic "..."] [--output <file>] [--force]',flags:["--network <name> Network: preprod, preview, undeployed","--seed <hex> Restore from existing seed (64-char hex)",'--mnemonic "..." Restore from BIP-39 mnemonic (24 words)',"--output <file> Custom output path (default: ~/.midnight/wallet.json)","--force Overwrite existing wallet file"],examples:["midnight generate --network preprod"],jsonFields:{address:"Generated wallet address (bech32m)",network:"Network name",seed:"Hex-encoded 32-byte seed",mnemonic:"BIP-39 mnemonic (24 words, only if generated or provided)",file:"Path where wallet file was saved",createdAt:"ISO 8601 creation timestamp"}},{name:"info",description:"Display wallet address, network, creation date (no secrets shown)",usage:"midnight info [--wallet <name|file>]",flags:["--wallet <name|file> Wallet name or path"],examples:["midnight info","midnight info --wallet my-wallet.json"],jsonFields:{addresses:"Per-network unshielded addresses { preprod, preview, undeployed }",shieldedAddresses:"Per-network shielded addresses { preprod, preview, undeployed }",activeNetwork:"Currently active network",activeAddress:"Unshielded address for active network (also written to stdout)",createdAt:"ISO 8601 creation timestamp",file:"Wallet file path"}},{name:"balance",description:"Check unshielded + shielded balance (full wallet sync)",usage:"midnight balance [address] [--network <name>] [--indexer-ws <url>]",flags:["<address> Check a specific address (unshielded only, no wallet sync)","--network <name> Override network","--indexer-ws <url> Custom indexer WebSocket URL"],examples:["midnight balance","midnight balance mn_addr_preprod1..."],jsonFields:{address:"Unshielded address (bech32m)",shieldedAddress:"Shielded address",network:"Network name",unshielded:"{ NIGHT, utxoCount }",shielded:"{ NIGHT, availableCoins, pendingCoins }"}},{name:"address",description:"Derive and display an unshielded address from a seed",usage:"midnight address --seed <hex> [--network <name>] [--index <n>]",flags:["--seed <hex> Seed to derive from (required, 64-char hex)","--network <name> Network for address prefix (default: resolved)","--index <n> Key derivation index (default: 0)"],examples:["midnight address --seed 0123456789abcdef... --network preprod","midnight address --seed 0123456789abcdef... --index 1"],jsonFields:{address:"Derived address (bech32m)",network:"Network name",index:"Key derivation index",path:"BIP-44 derivation path"}},{name:"genesis-address",description:"Display the genesis wallet address (seed 0x01) for a network",usage:"midnight genesis-address [--network <name>]",flags:["--network <name> Network for address prefix (default: resolved)"],examples:["midnight genesis-address --network undeployed","midnight genesis-address --network preprod"],jsonFields:{address:"Genesis wallet address (bech32m)",network:"Network name"}},{name:"inspect-cost",description:"Display current block limits derived from LedgerParameters",usage:"midnight inspect-cost",examples:["midnight inspect-cost"],jsonFields:{readTime:"Read time limit (picoseconds)",computeTime:"Compute time limit (picoseconds)",blockUsage:"Block usage limit (bytes)",bytesWritten:"Bytes written limit (bytes)",bytesChurned:"Bytes churned limit (bytes)"}},{name:"airdrop",description:"Fund your wallet from the genesis wallet (undeployed network only)",usage:"midnight airdrop <amount> [--shielded] [--wallet <name|file>]",flags:["<amount> Amount in NIGHT to airdrop","--shielded Airdrop shielded NIGHT (from genesis shielded balance)","--wallet <name|file> Wallet name or path"],examples:["midnight airdrop 1000","midnight airdrop 100 --shielded"],jsonFields:{txHash:"Transaction hash",amount:"Amount airdropped (NIGHT string)",recipient:"Recipient address (unshielded airdrop)",shieldedAddress:"Shielded address (--shielded airdrop)",network:"Network name"}},{name:"transfer",description:"Send NIGHT tokens to another address (unshielded or --shielded)",usage:"midnight transfer <to> <amount> [--shielded] [--wallet <name|file>] [--proof-server <url>] [--node <url>] [--indexer-ws <url>]",flags:["<to> Recipient bech32m address (unshielded or shielded)","<amount> Amount in NIGHT to send","--shielded Send from shielded balance to a shielded address","--wallet <name|file> Wallet name or path","--proof-server <url> Override proof server URL","--node <url> Override substrate node RPC URL","--indexer-ws <url> Override indexer WebSocket URL"],examples:["midnight transfer mn_addr_undeployed1... 100","midnight transfer mn_shield-addr_undeployed1... 50 --shielded"],jsonFields:{txHash:"Transaction hash",amount:"Amount transferred (NIGHT string)",recipient:"Recipient address (bech32m)",network:"Network name"}},{name:"dust",description:"Register UTXOs for dust (fee token) generation or check status",usage:"midnight dust <register|status> [--wallet <name|file>] [--proof-server <url>] [--node <url>] [--indexer-ws <url>] [--no-cache (status only)]",flags:["register Register NIGHT UTXOs for dust generation","status Check dust registration; if registered, also shows dust balance","--wallet <name|file> Wallet name or path","--proof-server <url> Override proof server URL (register only)","--node <url> Override substrate node RPC URL (register only)","--indexer-ws <url> Override indexer WebSocket URL","--no-cache Bypass wallet state cache (status only)"],examples:["midnight dust register","midnight dust status"],jsonFields:{subcommand:"register or status",registered:"true if at least one NIGHT UTXO is registered for dust generation (status only)",registeredUtxos:"Number of registered UTXOs (status only)",unregisteredUtxos:"Number of unregistered UTXOs (status only)",dustBalance:"Dust balance (raw bigint string; status only when registered, and register)",nightBalance:"NIGHT balance (raw bigint string; status only when registered)",dustAvailable:"Whether dust tokens are available (status only when registered)",txHash:"Registration transaction hash (register only, if submitted)"}},{name:"config",description:"Manage persistent config (default network, endpoints, etc.)",usage:"midnight config <get|set|unset> <key> [value]",flags:["get <key> Read a config value","set <key> <value> Write a config value","unset <key> Reset a config value to its default","","Keys: network, wallet, proof-server, node, indexer-ws"],examples:["midnight config get network","midnight config set network preprod","midnight config set wallet alice","midnight config set proof-server http://localhost:6300","midnight config set node wss://rpc.preprod.midnight.network","midnight config set indexer-ws wss://indexer.preprod.midnight.network/api/v3/graphql/ws","midnight config unset proof-server"],jsonFields:{action:"get, set, or unset",key:"Config key name",value:"Config value"}},{name:"cache",description:"Manage wallet state cache (clear cached sync data)",usage:"midnight cache clear [--network <name>] [--wallet <name|file>]",flags:["clear Clear cached wallet state","--network <name> Only clear cache for this network","--wallet <name|file> Only clear cache for this wallet"],examples:["midnight cache clear","midnight cache clear --network preprod","midnight cache clear --wallet alice"],jsonFields:{action:"clear",scope:"all, network, or wallet",network:"Network name (when scoped)",wallet:"Wallet name (when scoped)"}},{name:"serve",description:"Start DApp Connector server over WebSocket JSON-RPC",usage:"midnight serve [--port <n>] [--wallet <name|file>] [--network <name>] [--proof-server <url>] [--node <url>] [--indexer-ws <url>] [--approve-all] [--no-auto-approve-reads] [--json]",flags:["--port <n> Server port (default: 9932)","--wallet <name|file> Wallet name or path","--network <name> Override network detection","--proof-server <url> Override proof server URL","--node <url> Override substrate node RPC URL","--indexer-ws <url> Override indexer WebSocket URL","--approve-all Auto-approve all requests (reads + writes)","--no-auto-approve-reads Prompt for read methods too"],examples:["midnight serve","midnight serve --port 8080","midnight serve --approve-all"],jsonFields:{port:"Server port number",network:"Network name",address:"Wallet address (bech32m)",status:"Server status (listening)"}},{name:"test",description:"Generate and run E2E test suites for Midnight dApps",usage:"midnight test <create|run|list|results> [options]",flags:["create Generate a test scaffold from the contract (CLI or browser strategy)","run Run test suites for the current dApp","list List available test suites","results Show latest test results","--suite <name> Select a specific test suite (run, create)","--all Show all results (with results subcommand)","--strategy cli|ui Scaffold strategy (create — interactive prompt if omitted)","--name <contract> Specific contract name in multi-contract projects (create)","--path <dir> dApp directory (default: cwd)","--network <name> Network for the generated suite (default: undeployed)","--port <n> Browser strategy: dev server port (create --strategy ui)","--build-cmd <cmd> Browser strategy: build/serve command","--build-dir <subdir> Browser strategy: subdirectory the build runs in (monorepo case)","--url <url> Browser strategy: full URL Claude opens (default http://localhost:<port>/)","--browser-mode <m> Browser strategy: dom (HTML UIs, fast — default), vision (canvas games), script (advanced)","--force Overwrite existing scaffold files (create)","--json Output structured JSON"],examples:["midnight test create","midnight test create --strategy cli --force",'midnight test create --strategy ui --port 4173 --build-cmd "npm run dev"',"midnight test run","midnight test run --suite e2e-gameplay","midnight test list","midnight test results --all --json"],jsonFields:{dapp:"DApp name from dapp.test.json",suite:"Test suite name",status:"Overall result: pass, fail, timeout, or error",duration:"Total duration in seconds",prep:"Array of prep step results",assertions:"Array of assertion results"}},{name:"dev",description:"Iteration loop for Compact contract development (localnet + compile-on-save + one-key deploy)",usage:"midnight dev [path]",flags:["[path] Project directory (default: cwd)","","Keystrokes (while running):"," d Deploy the current compiled artifact (dev-alice on undeployed)"," t Run the project's npm test script (test:dev preferred, then test)"," q Quit cleanly"," Ctrl+C Quit cleanly"],examples:["midnight dev","midnight dev ./contract"]},{name:"contract",description:"Inspect, deploy, call, and query Midnight smart contracts",usage:"midnight contract <inspect|deploy|call|state> [options]",flags:["inspect Show circuits, witnesses, and types","deploy Deploy a contract to the network","call Call a circuit on a deployed contract","state Read ledger state of a deployed contract","--address <addr> Contract address (call, state)","--circuit <name> Circuit to call (call)",`--args '<json>' JSON arguments — constructor args (deploy) or circuit args (call). Numbers auto-coerce to BigInt; "123n" strings parse as BigInt; arrays of 0–255 ints become Uint8Array`,"--network <name> Override network (default: undeployed)","--path <dir> dApp directory (all subcommands; default: cwd)","--managed <dir> Direct path to managed/<name>/ directory (all subcommands)","--name <contract> Specific contract when the project ships multiple (all subcommands)","--wallet <name> Wallet to use for the operation (deploy, call, state)","--json Output structured JSON"],examples:["midnight contract inspect","midnight contract deploy",`midnight contract deploy --args '{"deadlineSecs": 300}'`,`midnight contract call --address 0x123 --circuit submit_score --args '{"score": 100, "alias": "TEST"}'`,"midnight contract state --address 0x123","midnight contract deploy --json"],jsonFields:{name:"Contract name",address:"Deployed contract address (deploy, call)",compilerVersion:"Compact compiler version (inspect)",circuits:"Array of circuit definitions (inspect)",witnesses:"Array of witness definitions (inspect)",state:"Contract ledger state (state)"}},{name:"localnet",description:"Manage a local Midnight network via Docker Compose",usage:"midnight localnet <up|stop|down|status|logs|clean>",flags:["up Start the local network (node, indexer, proof server)","stop Stop containers (preserves state for fast restart)","down Full teardown: containers, networks, volumes, + undeployed wallet cache","status Show service status and ports","logs Stream service logs (Ctrl+C to stop)","clean Remove conflicting containers from other setups"],examples:["midnight localnet up","midnight localnet stop","midnight localnet status","midnight localnet down","midnight localnet clean"],jsonFields:{subcommand:"up, stop, down, status, or clean",services:"Array of { name, state, port, health? } (up/status only)",status:"Operation result message (stop/down/clean)",removed:"Array of removed container names (clean only)"}},{name:"help",description:"Show usage for all commands or a specific command",usage:"midnight help [command]",examples:["midnight help","midnight help balance"],jsonFields:{cli:"CLI metadata (name, version, description)",globalFlags:"Array of global flag descriptions",commands:"Array of command specs with jsonFields"}},{name:"manual",description:"Full reference manual (long form, paged through $PAGER)",usage:"midnight manual [--no-pager] [--raw] [--json]",flags:["--no-pager Print straight to stdout, do not pipe through less","--raw Strip ANSI colors (useful for piping to a file)","--json Emit the manual text wrapped in a JSON object"],examples:["midnight manual","midnight manual --no-pager | grep -i dust","midnight manual --raw > MANUAL.txt"],jsonFields:{name:"CLI package name",version:"CLI version",manual:"Manual text (with ANSI colors)"}}]});var C6={};m(C6,{default:()=>x6,buildManual:()=>V2});import{spawn as rG}from"node:child_process";function sG($){return`${M(N6)}
|
|
1671
|
+
${M(m0($))}
|
|
1672
|
+
${M(N6)}
|
|
1673
|
+
`}function P4($){return`
|
|
1674
|
+
${V(E6)}
|
|
1675
|
+
${M(m0($))}
|
|
1676
|
+
${V(E6)}
|
|
1677
|
+
`}function tG($){let Z=[];if(Z.push(`${f(M("▸ "+$.name))}${" ".repeat(Math.max(2,22-$.name.length))}${$.description}`),Z.push(""),Z.push(` ${M("USAGE")}`),Z.push(` ${$.usage}`),$.flags&&$.flags.length>0){Z.push(""),Z.push(` ${M("FLAGS")}`);for(let Q of $.flags){let X=Q.match(/^(\s*)(--[\w-]+(?:\s<[^>]+>)?)(\s+)(.*)$/);if(X)Z.push(` ${S(X[2])}${X[3]}${X[4]}`);else if(/^[A-Za-z][\w -]*:/.test(Q))Z.push(` ${V(Q)}`);else if(Q.trim()==="")Z.push("");else Z.push(` ${Q}`)}}if($.examples&&$.examples.length>0){Z.push(""),Z.push(` ${M("EXAMPLES")}`);for(let Q of $.examples)Z.push(` ${V("$")} ${V(Q)}`)}return Z.join(`
|
|
1678
|
+
`)}function V2(){let $=[];$.push(sG(`${O1} manual v${g0}`)),$.push(""),$.push(`${M("NAME")}`),$.push(" midnight (mn) — wallet and development CLI for the Midnight blockchain"),$.push(""),$.push(`${M("SYNOPSIS")}`),$.push(" mn <command> [subcommand] [options]"),$.push(" midnight <command> [subcommand] [options]"),$.push(""),$.push(`${M("DESCRIPTION")}`),$.push(" midnight-wallet-cli is a standalone CLI that handles wallet"),$.push(" operations, contract deployment, and a development loop, on three"),$.push(" networks (undeployed, preprod, preview). Every command speaks"),$.push(" JSON via --json for use with AI agents and scripts. An MCP"),$.push(" server ships with the package."),$.push(""),$.push(`${M("INSTALLATION")}`),$.push(` ${V("$")} ${V("npm install -g midnight-wallet-cli")}`),$.push(""),$.push(" Optional dependencies:"),$.push(" Compact toolchain: https://docs.midnight.network/develop"),$.push(" Docker: for the local development network"),$.push(P4("CONCEPTS")),$.push(` ${M("Networks")}`),$.push(` ${f("undeployed")} Local Docker network. Free dev tokens, fast iteration.`),$.push(` ${f("preprod")} Public testnet. Persistent, real ZK proofs.`),$.push(` ${f("preview")} Public testnet, smaller scale.`),$.push(""),$.push(` ${M("Tokens")}`),$.push(` ${S("NIGHT")} Native asset. Used for transfers and contract value.`),$.push(` ${S("DUST")} Fee token. Generated from registered NIGHT UTXOs.`),$.push(""),$.push(` ${M("Wallets")}`),$.push(` Stored at ${V("~/.midnight/wallets/<name>.json")}`),$.push(" Three networks share one seed; each gets its own derived address."),$.push(" A tip-aware cache makes repeated reads finish in seconds."),$.push(""),$.push(` ${M("Shielded vs Unshielded")}`),$.push(` ${S("Unshielded")} addresses (mn_addr_*) are like UTXOs.`),$.push(` ${S("Shielded")} addresses (mn_shield-addr_*) hide amount and recipient.`),$.push(P4("COMMANDS"));for(let Z of y4)$.push(tG(Z)),$.push("");return $.push(P4("COMMON FLOWS")),$.push(` ${M("First wallet on undeployed")}`),$.push(` ${V("$")} ${V("mn localnet up")}`),$.push(` ${V("$")} ${V("mn wallet generate alice")}`),$.push(` ${V("$")} ${V("mn airdrop 1000 --wallet alice")}`),$.push(` ${V("$")} ${V("mn dust register --wallet alice")}`),$.push(` ${V("$")} ${V("mn balance --wallet alice")}`),$.push(""),$.push(` ${M("First wallet on preprod")}`),$.push(` ${V("$")} ${V("mn wallet generate alice --network preprod")}`),$.push(` ${V("# fund from the preprod faucet:")}`),$.push(` ${V("# https://faucet.preprod.midnight.network/")}`),$.push(` ${V("$")} ${V("mn dust register --wallet alice --network preprod")}`),$.push(` ${V("$")} ${V("mn balance --wallet alice --network preprod")}`),$.push(""),$.push(` ${M("Deploy and call a contract on undeployed")}`),$.push(" Prerequisite: a compiled artifact at ./src/managed/<name>/"),$.push(""),$.push(` ${V("$")} ${V("mn contract deploy --wallet alice")}`),$.push(` ${V(" → Address: 64da9d71cb…")}`),$.push(` ${V("$")} ${V("mn contract state --address 64da9d71cb… --wallet alice")}`),$.push(` ${V(' → { "round": 0 }')}`),$.push(` ${V("$")} ${V("mn contract call --address 64da9d71cb… --circuit increment --wallet alice")}`),$.push(` ${V("$")} ${V("mn contract state --address 64da9d71cb… --wallet alice")}`),$.push(` ${V(' → { "round": 1 }')}`),$.push(""),$.push(" Same artifact on preprod — swap one flag:"),$.push(` ${V("$")} ${V("mn contract deploy --network preprod --wallet alice")}`),$.push(""),$.push(` ${M("Transfer NIGHT between wallets")}`),$.push(` ${V("$")} ${V("mn transfer bob 100 --wallet alice")}`),$.push(` ${V(" shielded variant:")}`),$.push(` ${V("$")} ${V("mn transfer bob 100 --shielded --wallet alice")}`),$.push(P4("CONFIGURATION")),$.push(` ${M("Files")}`),$.push(` ${V("~/.midnight/config.json")} Persistent CLI settings`),$.push(` ${V("~/.midnight/wallets/")} Per-wallet JSON files (mode 0600)`),$.push(` ${V("~/.midnight/cache/<network>/")} Wallet sync cache, per network`),$.push(""),$.push(` ${M("Config keys")}`),$.push(` ${S("network")} Default network when --network is omitted`),$.push(` ${S("wallet")} Default wallet when --wallet is omitted`),$.push(` ${S("proof-server")} Override proof server URL`),$.push(` ${S("node")} Override substrate RPC URL`),$.push(` ${S("indexer-ws")} Override indexer WebSocket URL`),$.push(""),$.push(` ${M("Compatibility alias")}`),$.push(` ${S("network-id")} Resolves transparently to ${S("network")}.`),$.push(""),$.push(` ${M("Examples")}`),$.push(` ${V("$")} ${V("mn config set network preprod")}`),$.push(` ${V("$")} ${V("mn config get network")}`),$.push(` ${V("$")} ${V("mn config unset proof-server")}`),$.push(P4("JSON OUTPUT")),$.push(` Every command accepts ${S("--json")} for structured output on stdout.`),$.push(" Stderr keeps the chrome (spinners, headers); stdout stays parseable."),$.push(""),$.push(` ${V("$")} ${V("mn balance <addr> --json | jq -r .NIGHT")}`),$.push(` ${V("$")} ${V("mn wallet info alice --json")}`),$.push(""),$.push(` Stable JSON shapes are documented in ${V("docs/JSON_CONTRACT.md")}.`),$.push(" We promise not to break those without a major version bump."),$.push(P4("EXIT CODES")),$.push(` ${eG("0")} Success`),$.push(` ${V("1")} Unknown error`),$.push(` ${$q("2")} Invalid arguments (usage error, see yellow box on stderr)`),$.push(` ${V("3")} Wallet not found`),$.push(` ${V("4")} Network error`),$.push(` ${V("5")} Insufficient balance / DUST_REQUIRED`),$.push(` ${V("6")} Transaction rejected / STALE_UTXO / PROOF_TIMEOUT`),$.push(` ${V("7")} Cancelled by user`),$.push(""),$.push(` ${M("Stable error code strings")} (in JSON ${S("--json")} error output):`),$.push(` ${S("DUST_REQUIRED")} No dust available to pay fees`),$.push(` ${S("STALE_UTXO")} UTXO consumed concurrently, retry`),$.push(` ${S("PROOF_TIMEOUT")} Proof generation exceeded deadline`),$.push(` ${S("PROOF_FAILURE")} Proof server rejected the proof`),$.push(` ${S("INVALID_DUST_PROOF")} Stale commitment tree, run ${V("mn cache clear")}`),$.push(` ${S("STALE_CACHE")} Local cache out of sync, run ${V("mn cache clear")}`),$.push(` ${S("SYNC_TIMEOUT")} Wallet sync exceeded its deadline`),$.push(P4("TROUBLESHOOTING")),$.push(` ${M('Sync hangs forever or "applied > highest" error')}`),$.push(" Stale cache after a chain reset. Recover with:"),$.push(` ${V("$")} ${V("mn cache clear --wallet <name>")}`),$.push(""),$.push(` ${M('"DUST_REQUIRED" or "no dust available" on transfer')}`),$.push(" Wallet has NIGHT but no fee token. Register dust:"),$.push(` ${V("$")} ${V("mn dust register --wallet <name>")}`),$.push(" Then wait until status shows availability:"),$.push(` ${V("$")} ${V("mn dust status --wallet <name>")}`),$.push(""),$.push(` ${M('"Cannot find package @midnight-ntwrk/..." on contract deploy')}`),$.push(" The CLI bundles its own SDK. Reinstall to pick up the bundle:"),$.push(` ${V("$")} ${V("npm install -g midnight-wallet-cli@latest")}`),$.push(""),$.push(` ${M('"witnesses module not found" on contract deploy')}`),$.push(" Your contract declares witnesses but witnesses.js isn't built."),$.push(" Build the TypeScript source first:"),$.push(` ${V("$")} ${V("npm run build")}`),$.push(""),$.push(` ${M('"mn serve already running on port 9932 for network X"')}`),$.push(" A leftover serve from a previous network is in the way."),$.push(` ${V("$")} ${V("pkill -f 'mn serve'")}`),$.push(P4("SEE ALSO")),$.push(` ${f("mn help")} Brief command list`),$.push(` ${f("mn help <command>")} Help for one command (flags and examples)`),$.push(` ${f("mn help --agent")} Comprehensive AI / MCP reference`),$.push(""),$.push(` Repository: ${V("github.com/nel349/midnight-wallet-cli")}`),$.push(` JSON contract: ${V("docs/JSON_CONTRACT.md")}`),$.push(""),$.join(`
|
|
1679
|
+
`)}function eG($){return`\x1B[32m${$}\x1B[0m`}function $q($){return`\x1B[33m${$}\x1B[0m`}async function Zq($,Z){if(!Z||!process.stdout.isTTY){process.stdout.write($);return}let Q=process.env.PAGER??"less -R",[X,...z]=Q.split(/\s+/);if(!X){process.stdout.write($);return}await new Promise((Y)=>{let J=rG(X,z,{stdio:["pipe","inherit","inherit"]});J.stdin.on("error",()=>{}),J.on("close",()=>Y()),J.on("error",()=>{process.stdout.write($),Y()}),J.stdin.write($),J.stdin.end()})}async function x6($){let Z=!x($,"no-pager");if(x($,"json")){let{writeJsonResult:Q}=await Promise.resolve().then(() => A2);Q({name:O1,version:g0,manual:V2()});return}if(x($,"raw")){process.stdout.write(V2().replace(/\x1b\[[0-9;]*m/g,""));return}await Zq(V2(),Z)}var N6,E6;var v6=N(()=>{T7();j1();N6="═".repeat(63),E6="─".repeat(64)});var b6={};m(b6,{default:()=>y6});async function y6($,Z){Y4($);let Q=D($,"port"),X=Q?parseInt(Q,10):m4;if(Number.isNaN(X)||X<1||X>65535)throw new Error(`Invalid port: "${Q}" — must be 1-65535`);let z=x($,"approve-all"),Y=z||!x($,"no-auto-approve-reads"),J=x($,"json"),G=Y0(z0(D($,"wallet"))),q=Buffer.from(G.seed,"hex"),{name:K,config:B}=V0({args:$}),H=G.addresses[K];x0(B,{proofServer:D($,"proof-server"),node:D($,"node"),indexerWS:D($,"indexer-ws")}),process.stderr.write(`
|
|
1680
|
+
`+g("DApp Connector Server")+`
|
|
1681
|
+
|
|
1682
|
+
`),process.stderr.write(A("Network",K)+`
|
|
1683
|
+
`),process.stderr.write(A("Address",$0(H,!0))+`
|
|
1684
|
+
`),process.stderr.write(A("Port",String(X))+`
|
|
1685
|
+
`),process.stderr.write(A("Auto-approve reads",z||Y?"yes":"no")+`
|
|
1686
|
+
`),process.stderr.write(A("Auto-approve writes",z?"yes":"no")+`
|
|
593
1687
|
`),process.stderr.write(`
|
|
594
|
-
`);let
|
|
595
|
-
`)}),
|
|
596
|
-
`)},onDisconnect:(
|
|
597
|
-
`),
|
|
598
|
-
`)},onResponse:(
|
|
599
|
-
`),
|
|
600
|
-
`)}else{let
|
|
601
|
-
`);let
|
|
1688
|
+
`);let U=U0((W,P)=>{let O=TZ();if(O)O.log(V(` SDK: ${P}`));else process.stderr.write(V(` SDK: ${P}`)+`
|
|
1689
|
+
`)}),j=P0();if(D0($))b0();let L=l("Syncing wallet...");await j0().withFacade(q,B,async({bundle:W})=>{L.stop("Wallet synced");let P=l("Waiting for dust..."),F=(await e4(W)).dust?.availableCoins?.length>0;P.stop(F?"Dust ready":"Dust not yet available (writes may fail)");try{await S0(H,K,W.facade)}catch{}if(Z?.aborted)throw new Error("Operation cancelled");let I,_;try{let v={approve:"Waiting for approval...",building:"Building transaction...",signing:"Signing...",proving:"Proving (ZK)...",submitting:"Submitting..."},b;if(_=n$({bundle:W,networkConfig:B,approvalOptions:{approveAll:z,autoApproveReads:Y},callbacks:{onPhaseStart:(k,J0,w)=>{if(w==="approve")return;let i=v[w]??`${w}...`;if(b)b.update(i);else b=l(i)},onPhaseComplete:(k,J0,w,i)=>{if(w==="approve")return;if(w==="proving"||w==="submitting")b?.stop(`${w} ${A7(i)}`),b=void 0}}}),I=c$({port:X,handlers:_.handlers,onConnect:(k)=>{process.stderr.write(V(` [${M7()}] `)+p("connected")+V(` ${k.id}`)+`
|
|
1690
|
+
`)},onDisconnect:(k)=>{process.stderr.write(V(` [${M7()}] `)+n("disconnected")+V(` ${k.id}`)+`
|
|
1691
|
+
`),_?.revertPendingTxs(k.id).catch(()=>{})},onRequest:(k,J0)=>{process.stderr.write(V(` [${M7()}] ${k.id} #${k.requestCount} → ${J0.method}`)+`
|
|
1692
|
+
`)},onResponse:(k,J0,w,i,e,Q0)=>{if(e&&b)b.fail("Failed"),b=void 0;let C=A7(w);if(e){let R=e.code==="Rejected"?"rejected by operator":e.message;if(process.stderr.write(` ${n("✗")} ${V(`${k.id} ← ${J0.method} (${C})`)} ${n(R)}
|
|
1693
|
+
`),e.code!=="Rejected"&&e.message)process.stderr.write(` ${V(" error:")} ${e.message}
|
|
1694
|
+
`)}else{let R=Q0?.phases,T=R&&R.length>0?": "+R.map((u)=>`${u.phase} ${A7(u.durationMs)}`).join(", "):"";process.stderr.write(` ${p("✓")} ${V(`${k.id} ← ${J0.method} (${C}${T})`)}
|
|
1695
|
+
`);let E=i?.txHash;if(J0.method==="submitTransaction"&&E)process.stderr.write(` ${V(" tx:")} ${f(String(E))}
|
|
602
1696
|
`)}}}),process.stderr.write(`
|
|
603
|
-
`+
|
|
604
|
-
`),process.stderr.write(" "+
|
|
605
|
-
`),process.stderr.write(
|
|
606
|
-
|
|
607
|
-
`),J)
|
|
608
|
-
`+
|
|
609
|
-
`),!
|
|
610
|
-
|
|
611
|
-
`)}catch(
|
|
612
|
-
|
|
1697
|
+
`+o()+`
|
|
1698
|
+
`),process.stderr.write(" "+M(f(`Server ready — listening on ws://localhost:${X}`))+`
|
|
1699
|
+
`),process.stderr.write(V(" Press Ctrl+C to stop")+`
|
|
1700
|
+
|
|
1701
|
+
`),J)y({port:X,network:K,address:H,status:"listening"});if(await new Promise((k)=>{if(Z)if(Z.aborted)k();else Z.addEventListener("abort",()=>k(),{once:!0})}),process.stderr.write(`
|
|
1702
|
+
`+V(" Shutting down...")+`
|
|
1703
|
+
`),!_.hasPendingTxs())try{await S0(H,K,W.facade)}catch{}try{await I.close()}catch{}_.dispose(),_=void 0,I=void 0,process.stderr.write(V(" Server stopped.")+`
|
|
1704
|
+
|
|
1705
|
+
`)}catch(v){if(L.fail("Failed"),I)try{await I.close()}catch{}throw _?.dispose(),v}},{syncMode:"full",requireStrictSync:!0,skipAutoSave:!0,signal:Z,onStatus:(W)=>L.update(W),onSyncProgress:(W,P)=>{if(P>0){let O=Math.min(Math.round(W/P*100),100);L.update(O>=100?"Syncing wallet...":`Syncing wallet... ${O}%`)}},onSyncDetail:(W)=>L.update(`Syncing wallet... (waiting on: ${W})`)}).finally(()=>{j(),U()})}function M7(){let $=new Date;return`${R7($.getHours())}:${R7($.getMinutes())}:${R7($.getSeconds())}`}function R7($){return $<10?"0"+$:String($)}function A7($){if($<1000)return`${$}ms`;return`${($/1000).toFixed(1)}s`}var S6=N(()=>{B4();F0();H0();C0();o0();U4();L4();O4();c8();d$();K0();u0()});import{existsSync as f6,readFileSync as Qq,readdirSync as Xq,statSync as zq}from"node:fs";import{dirname as Yq,join as D7,resolve as k6}from"node:path";function h6($){let Z=k6($),Q=new Set,X=new Set;for(let H of qq){let U=k6(Z,H);if(!f6(U))continue;p6(U,0,Q,X)}if(Q.size===0)throw new Error(`No .compact source files found under ${Z}
|
|
1706
|
+
Run "mn dev" from a Midnight project directory (expected .compact files in ./, ./src, ./contract, or ./contract/src).`);let z=Hq([...X],Z),Y=z?.dir??Z,J=z?.packageJson??g6(D7(Z,"package.json")),G=z?.script??m6(J),q=[...Q].filter((H)=>w6(H,Y)).sort(),K=[...X].filter((H)=>w6(H,Y)).sort(),B=Bq(J);return{projectRoot:Y,sourceFiles:q.length>0?q:[...Q].sort(),sourceDirs:K.length>0?K:[...X].sort(),compileScript:G,hasNpmCompileScript:G!==null,testScript:B,packageJson:J}}function Bq($){for(let Z of Gq)if(u6($,Z))return Z;return null}function w6($,Z){if($===Z)return!0;let Q=Z.endsWith("/")?Z:Z+"/";return $.startsWith(Q)}function Hq($,Z){for(let Q of $){let X=Q;while(!0){let z=D7(X,"package.json"),Y=g6(z),J=m6(Y);if(Y&&J)return{dir:X,packageJson:Y,script:J};if(X===Z)break;let G=Yq(X);if(G===X)break;X=G}}return null}function m6($){for(let Z of Jq)if(u6($,Z))return Z;return null}function g6($){if(!f6($))return null;try{let Z=JSON.parse(Qq($,"utf-8"));return typeof Z==="object"&&Z!==null&&!Array.isArray(Z)?Z:null}catch{return null}}function u6($,Z){if(!$)return!1;let Q=$.scripts;if(!Q||typeof Q!=="object")return!1;return typeof Q[Z]==="string"}function p6($,Z,Q,X){if(Z>Kq)return;let z;try{z=Xq($)}catch{return}for(let Y of z){if(Y.startsWith(".")||Vq.has(Y))continue;let J=D7($,Y),G=!1;try{G=zq(J).isDirectory()}catch{continue}if(G)p6(J,Z+1,Q,X);else if(Y.endsWith(".compact"))Q.add(J),X.add($)}}var Jq,Gq,qq,Kq=4,Vq;var c6=N(()=>{Jq=["compact","compile"],Gq=["test:dev","test"],qq=[".","contract/src","contract","src"],Vq=new Set(["node_modules",".git","dist","build","managed","target"])});import{spawn as Wq}from"node:child_process";async function d6($){let{bin:Z,args:Q,label:X}=Uq($),z=Date.now();return new Promise((Y,J)=>{let G=Wq(Z,Q,{cwd:$.project.projectRoot,env:process.env,stdio:["ignore","pipe","pipe"]}),q="",K="";G.stdout?.on("data",(H)=>{q+=H.toString()}),G.stderr?.on("data",(H)=>{K+=H.toString()});let B=()=>{G.kill("SIGTERM")};if($.signal)if($.signal.aborted)B();else $.signal.addEventListener("abort",B,{once:!0});G.on("error",(H)=>{$.signal?.removeEventListener("abort",B),J(new Error(`Failed to spawn "${X}": ${H.message}`))}),G.on("close",(H)=>{$.signal?.removeEventListener("abort",B),Y({success:H===0,durationMs:Date.now()-z,command:X,stdout:q,stderr:K,exitCode:H??-1})})})}function Uq($){if($.commandOverride){let{bin:Z,args:Q}=$.commandOverride;return{bin:Z,args:Q,label:[Z,...Q].join(" ")}}if($.project.compileScript){let Z=$.project.compileScript;return{bin:"npm",args:["run",Z,"--silent"],label:`npm run ${Z}`}}throw new Error(`No compile script found in package.json.
|
|
1707
|
+
Add one of:
|
|
1708
|
+
"compact": "compact compile src/<your>.compact src/managed/<your>"
|
|
1709
|
+
"compile": "compact compile src/<your>.compact src/managed/<your>"
|
|
1710
|
+
create-mn-app and midnight-starship templates ship with this script already wired.`)}var l6=()=>{};import{spawn as Lq}from"node:child_process";async function i6($){let{bin:Z,args:Q,label:X}=Oq($),z=Date.now();return new Promise((Y,J)=>{let G=Lq(Z,Q,{cwd:$.project.projectRoot,env:process.env,stdio:["ignore","inherit","inherit"]}),q=()=>{G.kill("SIGTERM")};if($.signal)if($.signal.aborted)q();else $.signal.addEventListener("abort",q,{once:!0});G.on("error",(K)=>{$.signal?.removeEventListener("abort",q),J(new Error(`Failed to spawn "${X}": ${K.message}`))}),G.on("close",(K)=>{$.signal?.removeEventListener("abort",q),Y({success:K===0,durationMs:Date.now()-z,command:X,exitCode:K??-1})})})}function Oq($){if($.commandOverride){let{bin:Q,args:X}=$.commandOverride;return{bin:Q,args:X,label:[Q,...X].join(" ")}}if(!$.project.testScript)throw new Error(`No test script found in package.json.
|
|
1711
|
+
Add a "test:dev" or "test" script that runs your contract tests
|
|
1712
|
+
(e.g. "test": "vitest run"), then press t again.`);let Z=$.project.testScript;return{bin:"npm",args:["run",Z,"--silent"],label:`npm run ${Z}`}}var n6=()=>{};import{watch as jq}from"node:fs";import{join as Pq}from"node:path";function o6($){let Z=$.debounceMs??300,Q=[],X=new Set,z=null,Y=()=>{if(z=null,X.size===0)return;let J=[...X];X.clear();try{Promise.resolve($.onChange(J)).catch((G)=>{$.onError?.(G instanceof Error?G:new Error(String(G)))})}catch(G){$.onError?.(G instanceof Error?G:new Error(String(G)))}};for(let J of $.dirs)try{let G=jq(J,{persistent:!0},(q,K)=>{if(!K)return;if(!K.endsWith($.extension))return;if(X.add(Pq(J,K)),z)clearTimeout(z);z=setTimeout(Y,Z)});G.on("error",(q)=>{$.onError?.(q instanceof Error?q:new Error(String(q)))}),Q.push(G)}catch(G){$.onError?.(G instanceof Error?G:new Error(String(G)))}return{stop:()=>{if(z)clearTimeout(z),z=null;for(let J of Q)try{J.close()}catch{}}}}var a6=()=>{};async function r6($){if(Iq())return $?.("Localnet already running"),{state:"already-running"};return $?.("Writing compose file"),Z1(),$?.("Starting localnet containers"),p0("up -d"),$?.("Waiting for services to be healthy"),{state:Q1(120000)?"started":"started-unhealthy"}}function Fq(){try{return Q4()}catch{return[]}}function Iq(){let $=Fq(),Z=new Map($.map((Q)=>[Q.name,Q]));for(let Q of D8){let X=Z.get(Q);if(!X)return!1;if(X.state!=="running")return!1;if(X.health&&X.health!=="healthy")return!1}return!0}var s6=N(()=>{k$()});import{existsSync as _q}from"node:fs";import{join as Tq}from"node:path";import{homedir as Mq}from"node:os";async function $9($){let Z=[],Q=Rq();try{for(let X of $.names){if($.signal?.throwIfAborted(),Z9(X)){$.onProgress?.(X,"done","reused"),Z.push({name:X,state:"reused"});continue}$.onProgress?.(X,"creating","created"),await Dq(X,$.signal),$.onProgress?.(X,"funding","created"),await Nq(X,$.amountNight,$.signal),$.onProgress?.(X,"dust","created"),await Eq(X,$.signal),$.onProgress?.(X,"done","created"),Z.push({name:X,state:"created"})}}finally{Aq(Q)}return Z}function Rq(){try{return $4()}catch{return null}}function Aq($){if(!$)return;if(!Z9($))return;try{l4($)}catch{}}function Z9($){let Z=Tq(Mq(),B0,F4,`${$}.json`);return _q(Z)}async function Dq($,Z){let Q={command:"wallet",subcommand:"generate",positionals:[$],flags:{network:"undeployed"}},{default:X}=await Promise.resolve().then(() => (f$(),w$));await a(X,Q,Z)}async function Nq($,Z,Q){let X={command:"airdrop",subcommand:String(Z),positionals:[],flags:{wallet:$,network:"undeployed"}},{default:z}=await Promise.resolve().then(() => (x$(),E$));await a(z,X,Q)}async function Eq($,Z){let Q={command:"dust",subcommand:"register",positionals:[],flags:{wallet:$,network:"undeployed"}},{default:X}=await Promise.resolve().then(() => (v$(),C$));await a(X,Q,Z)}var t6,e6=1000;var Q9=N(()=>{W$();F0();t6=["dev-alice","dev-bob","dev-carol"]});function X9($){if(!xq)return{stop(){}};let Z=process.stdin,Q=Z.isRaw;Z.setRawMode(!0),Z.resume(),Z.setEncoding("utf8");let X=!1,z=(Y)=>{if(X)return;if(Y==="\x03"){$.onInterrupt();return}let J=Y.toLowerCase(),G=$.actions[J];if(!G){$.onUnknown?.(Y);return}Promise.resolve().then(G).catch((q)=>{let K=q instanceof Error?q:new Error(String(q));if($.onError)$.onError(J,K);else process.stderr.write(`key handler error (${J}): ${K.message}
|
|
1713
|
+
`)})};return Z.on("data",z),{stop(){if(X)return;X=!0,Z.off("data",z);try{Z.setRawMode(Q)}catch{}}}}var xq;var z9=N(()=>{xq=typeof process.stdin.isTTY==="boolean"&&process.stdin.isTTY===!0});var V9={};m(V9,{default:()=>K9});import{resolve as Cq,relative as Y9}from"node:path";async function K9($,Z){let Q=Cq($.positionals[0]??process.cwd());process.stderr.write(`
|
|
1714
|
+
`+g("mn dev")+`
|
|
1715
|
+
|
|
1716
|
+
`);let X=h6(Q);if(!X.compileScript)throw new Error(`No compile script found in ${X.projectRoot}/package.json.
|
|
1717
|
+
mn dev looks for a "compact" or "compile" npm script that invokes the
|
|
1718
|
+
Compact toolchain. Add one of:
|
|
1719
|
+
"scripts": { "compact": "compact compile src/my.compact src/managed/my" }
|
|
1720
|
+
"scripts": { "compile": "compact compile src/my.compact src/managed/my" }
|
|
1721
|
+
create-mn-app and midnight-starship templates ship with this already wired.`);process.stderr.write(A("Project",X.projectRoot)+`
|
|
1722
|
+
`),process.stderr.write(A("Sources",`${X.sourceFiles.length} .compact file(s)`)+`
|
|
1723
|
+
`),process.stderr.write(A("Compile",`npm run ${X.compileScript}`)+`
|
|
1724
|
+
|
|
1725
|
+
`);let z=l("Checking localnet...");try{let F=await r6((_)=>z.update(_)),I={"already-running":"Localnet already running",started:"Localnet started","started-unhealthy":"Localnet started (some services not yet healthy)"}[F.state];z.stop(I)}catch(F){throw z.fail("Localnet setup failed"),F}let Y=l("Provisioning dev wallets...");try{let F=await $9({names:t6,amountNight:e6,signal:Z,onProgress:(b,c)=>{let k={creating:"generating",funding:"airdropping",dust:"registering dust",done:"ready"}[c];Y.update(`${b}: ${k}`)}}),I=F.filter((b)=>b.state==="created").length,_=F.length-I,v=[I?`${I} created`:"",_?`${_} reused`:""].filter(Boolean).join(", ");Y.stop(`Dev wallets: ${F.map((b)=>b.name).join(", ")} (${v})`)}catch(F){throw Y.fail("Wallet provisioning failed"),F}let J=null,G=!1,q=!1,K=!1,B=!1;if(J=await q9(X,Z),!J.success)process.stderr.write(V(`
|
|
1726
|
+
Fix the errors above and save — mn dev will recompile automatically.
|
|
1727
|
+
`));let H=async()=>{if(G||q)return;if(K){B=!0;return}K=!0;try{J=await q9(X,Z)}finally{if(K=!1,B)B=!1,H()}},U=o6({dirs:X.sourceDirs,extension:".compact",onChange:async(F)=>{let I=F.map((_)=>Y9(X.projectRoot,_)).join(", ");process.stderr.write(`
|
|
1728
|
+
`+V(` Changed: ${I}`)+`
|
|
1729
|
+
`),await H()},onError:(F)=>{process.stderr.write(n(" watch error: ")+F.message+`
|
|
1730
|
+
`)}});process.stderr.write(`
|
|
1731
|
+
`+o()+`
|
|
1732
|
+
`);let j=X.sourceDirs.map((F)=>Y9(X.projectRoot,F)||".").join(", ");process.stderr.write(V(" Watching ")+f(j)+V(" — save to recompile.")+`
|
|
1733
|
+
`);let L=X.testScript?M("[t]")+V(` test (${X.testScript}) `):"";process.stderr.write(V(" ")+M("[d]")+V(" deploy ")+L+M("[q]")+V(` quit
|
|
1734
|
+
|
|
1735
|
+
`));let W=new AbortController,P=()=>{if(!W.signal.aborted)W.abort()},O=X9({actions:{d:async()=>{if(G){process.stderr.write(V(` Deploy already in flight — ignoring key press.
|
|
1736
|
+
`));return}if(q){process.stderr.write(V(` Tests are running — wait for them to finish before deploying.
|
|
1737
|
+
`));return}if(!J?.success){process.stderr.write(S(` Cannot deploy — last compile did not succeed. Fix and save first.
|
|
1738
|
+
`));return}G=!0;try{await vq(X)}finally{G=!1}},t:async()=>{if(q){process.stderr.write(V(` Tests already in flight — ignoring key press.
|
|
1739
|
+
`));return}if(G){process.stderr.write(V(` Deploy in flight — wait for it to finish before running tests.
|
|
1740
|
+
`));return}if(!X.testScript){process.stderr.write(S(` No test script found. Add "test:dev" or "test" to package.json.
|
|
1741
|
+
`));return}if(!J?.success){process.stderr.write(S(` Cannot run tests — last compile did not succeed. Fix and save first.
|
|
1742
|
+
`));return}q=!0;try{await yq(X,W.signal)}finally{q=!1}},q:()=>P()},onInterrupt:P});await new Promise((F)=>{let I=()=>{O.stop(),U.stop(),process.stderr.write(`
|
|
1743
|
+
`+V(" mn dev stopped.")+`
|
|
1744
|
+
`),F()};if(W.signal.aborted){I();return}if(W.signal.addEventListener("abort",I,{once:!0}),Z?.aborted){P();return}Z?.addEventListener("abort",P,{once:!0})})}async function vq($){let Z=process.cwd();if(Z!==$.projectRoot)try{process.chdir($.projectRoot)}catch{}try{let Q=await bq($.projectRoot);if(Q.length>0&&!i8($.projectRoot)){let z=n8({projectRoot:$.projectRoot,witnessNames:Q});process.stderr.write(S(` ✗ Cannot auto-deploy:
|
|
1745
|
+
`));for(let Y of z.split(`
|
|
1746
|
+
`))process.stderr.write(V(` ${Y}
|
|
1747
|
+
`));return}let X=l(`Deploying with ${J9} on ${G9}...`);try{let{default:z}=await Promise.resolve().then(() => (Z2(),$2)),Y=await a(z,{command:"contract",subcommand:"deploy",positionals:[],flags:{wallet:J9,network:G9}});X.stop("Deployed");let J=typeof Y.address==="string"?Y.address:"(unknown)";process.stderr.write(` ${V("address")} ${f(J)}
|
|
1748
|
+
`)}catch(z){X.fail("Deploy failed");let Y=z instanceof Error?z.message:String(z);process.stderr.write(n(" "+Y)+`
|
|
1749
|
+
`)}}finally{if(process.cwd()!==Z)try{process.chdir(Z)}catch{}}}async function yq($,Z){process.stderr.write(V(`
|
|
1750
|
+
Running npm run ${$.testScript}...
|
|
1751
|
+
`));try{let Q=await i6({project:$,signal:Z});if(Q.success)process.stderr.write(` ${V("─")} Tests passed ${V(`(${B2(Q.durationMs)})`)}
|
|
1752
|
+
`);else process.stderr.write(` ${V("─")} ${n("Tests failed")} ${V(`(exit ${Q.exitCode}, ${B2(Q.durationMs)})`)}
|
|
1753
|
+
`)}catch(Q){let X=Q instanceof Error?Q.message:String(Q);process.stderr.write(n(" Test runner error: ")+X+`
|
|
1754
|
+
`)}}async function bq($){try{let{findContractInfo:Z}=await Promise.resolve().then(() => (k1(),h8)),{info:Q}=Z($);return Q.witnesses.map((X)=>X.name)}catch{return[]}}async function q9($,Z){let Q=l("Compiling...");try{let X=await d6({project:$,signal:Z});if(X.success)Q.stop(`Compiled ${V(`(${B2(X.durationMs)})`)}`);else if(Q.fail(`Compile failed ${V(`(${B2(X.durationMs)})`)}`),X.stderr.trim())process.stderr.write(V(" ─ stderr ─")+`
|
|
1755
|
+
`),process.stderr.write(S(Sq(X.stderr.trim()," "))+`
|
|
1756
|
+
`);return X}catch(X){Q.fail("Compile error");let z=X instanceof Error?X.message:String(X);return process.stderr.write(n(" "+z)+`
|
|
1757
|
+
`),{success:!1,durationMs:0,command:"",stdout:"",stderr:z,exitCode:-1}}}function B2($){if($<1000)return`${$}ms`;return`${($/1000).toFixed(2)}s`}function Sq($,Z){return $.split(`
|
|
1758
|
+
`).map((Q)=>Z+Q).join(`
|
|
1759
|
+
`)}var J9="dev-alice",G9="undeployed";var B9=N(()=>{c6();l6();n6();a6();s6();Q9();z9();o$();W$();K0();u0()});K0();R2();l0();j1();F0();if(process.argv.includes("--mcp"))await Promise.resolve().then(() => (U6(),mG));else{let Y=function(){X.abort(),setTimeout(()=>process.exit(130),5000).unref()},$=N7(),Z=x($,"json");if(x($,"version")||x($,"v"))process.stdout.write(g0+`
|
|
1760
|
+
`),process.exit(0);if(x($,"help")||x($,"h"))$.command="help";let Q=$.command??"help";g2();let X=new AbortController,{signal:z}=X;process.on("SIGINT",Y),process.on("SIGTERM",Y);async function J(){switch(Q){case"help":{let{default:q}=await Promise.resolve().then(() => (T7(),D6));return q($)}case"manual":{let{default:q}=await Promise.resolve().then(() => (v6(),C6));return q($)}case"wallet":{let{default:q}=await Promise.resolve().then(() => (f$(),w$));return q($)}case"generate":{process.stderr.write(`
|
|
613
1761
|
Note: "midnight generate" is deprecated.
|
|
614
1762
|
Use "midnight wallet generate <name>" instead.
|
|
615
1763
|
|
|
616
|
-
`);let{default:
|
|
617
|
-
Run "midnight help" to see available commands.`)}}let
|
|
618
|
-
`+
|
|
1764
|
+
`);let{default:q}=await Promise.resolve().then(() => (p2(),u2));return q($)}case"info":{let{default:q}=await Promise.resolve().then(() => (d2(),c2));return q($)}case"balance":{let{default:q}=await Promise.resolve().then(() => (K8(),q8));return q($)}case"address":{let{default:q}=await Promise.resolve().then(() => (B8(),V8));return q($)}case"genesis-address":{let{default:q}=await Promise.resolve().then(() => (W8(),H8));return q($)}case"inspect-cost":{let{default:q}=await Promise.resolve().then(() => (L8(),U8));return q($)}case"config":{let{default:q}=await Promise.resolve().then(() => (_8(),I8));return q($)}case"cache":{let{default:q}=await Promise.resolve().then(() => (F8(),P8));return q($)}case"airdrop":{let{default:q}=await Promise.resolve().then(() => (x$(),E$));return q($,z)}case"transfer":{let{default:q}=await Promise.resolve().then(() => (j8(),O8));return q($,z)}case"dust":{let{default:q}=await Promise.resolve().then(() => (v$(),C$));return q($,z)}case"localnet":{let{default:q}=await Promise.resolve().then(() => (E8(),N8));return q($)}case"status":throw new Error('The "status" command is currently being reworked. Check back soon.');case"serve":{let{default:q}=await Promise.resolve().then(() => (S6(),b6));return q($,z)}case"test":{let{default:q}=await Promise.resolve().then(() => (L7(),U7));return q($,z)}case"contract":{let{default:q}=await Promise.resolve().then(() => (Z2(),$2));return q($,z)}case"dev":{let{default:q}=await Promise.resolve().then(() => (B9(),V9));return q($,z)}default:throw new t(`Unknown command: "${Q}"
|
|
1765
|
+
Run "midnight help" to see available commands.`)}}let G=new Set(["airdrop","transfer","dust","balance","serve","test","contract","dev"]);J().then(()=>{if(G.has(Q))process.exit(0)}).catch((q)=>{let K=y7(q.message);if(b7(q)){if(Z)J$(q,"INVALID_ARGS",M2);else process.stderr.write(`
|
|
1766
|
+
`+v7(K,'Run "midnight manual" for the full reference, or "midnight help <command>" for one command.')+`
|
|
1767
|
+
|
|
1768
|
+
`);process.exit(M2)}else if(Z){let{exitCode:B,errorCode:H}=U1(q);J$(q,H,B),process.exit(B)}else{process.stderr.write(`
|
|
1769
|
+
`+C7(K,'Run "midnight help" for usage information.')+`
|
|
619
1770
|
|
|
620
|
-
`);let{exitCode:
|
|
1771
|
+
`);let{exitCode:B}=U1(q);process.exit(B)}})}
|