@visa/cli 1.7.0-rc.1 → 1.8.0-rc.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/dist/cli.js +51 -47
- package/dist/mcp-server/index.js +22 -21
- package/package.json +2 -2
package/dist/cli.js
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
|
-
"use strict";var
|
|
1
|
+
"use strict";var ur=Object.create;var Ot=Object.defineProperty;var dr=Object.getOwnPropertyDescriptor;var fr=Object.getOwnPropertyNames;var pr=Object.getPrototypeOf,gr=Object.prototype.hasOwnProperty;var mr=(e,t)=>()=>(t||e((t={exports:{}}).exports,t),t.exports);var hr=(e,t,n,r)=>{if(t&&typeof t=="object"||typeof t=="function")for(let s of fr(t))!gr.call(e,s)&&s!==n&&Ot(e,s,{get:()=>t[s],enumerable:!(r=dr(t,s))||r.enumerable});return e};var g=(e,t,n)=>(n=e!=null?ur(pr(e)):{},hr(t||!e||!e.__esModule?Ot(n,"default",{value:e,enumerable:!0}):n,e));var nt=mr((Bs,Rr)=>{Rr.exports={name:"@visa/cli",version:"1.8.0-rc.0",description:"AI-powered payments for Claude Code",bin:{"visa-cli":"./bin/visa-cli.js"},scripts:{build:"tsc --noEmit && node esbuild.config.js",dev:"tsc --watch",start:"node dist/mcp-server/index.js",test:"jest --config jest.config.js","test:unit":"jest --config jest.config.js","test:unit:watch":"jest --config jest.config.js --watch","test:unit:coverage":"jest --config jest.config.js --coverage","test:smoke":"VISA_AUTH_URL=https://auth.visacli.sh jest --config jest.smoke.config.js","test:integration":"jest --config jest.integration.config.js","test:e2e":"jest --config jest.e2e.config.js","test:catalog-e2e":"jest --config jest.catalog-e2e.config.js","test:all":"npm run test:unit && npm run test:integration && npm run test:e2e",prepublishOnly:"npm run build && npm test",lint:"eslint src/**/*.ts",format:'prettier --write "src/**/*.ts"',"format:check":'prettier --check "src/**/*.ts"'},keywords:["visa","checkout","mcp","ai-agent","payments","click-to-pay","usdc","stablecoin"],author:"Visa Crypto Labs",license:"SEE LICENSE IN LICENSE",dependencies:{"@modelcontextprotocol/sdk":"^1.0.0",commander:"^12.1.0",zod:"^3.23.0"},devDependencies:{"@visa-cli/tools":"workspace:*","@changesets/changelog-git":"^0.2.1","@changesets/cli":"^2.31.0","@types/jest":"^30.0.0","@types/node":"^25.6.0","@typescript-eslint/eslint-plugin":"^8.59.0","@typescript-eslint/parser":"^8.59.0","@types/express":"^5.0.0",esbuild:"^0.27.4",express:"^4.21.0",eslint:"^10.0.2","eslint-config-prettier":"^10.1.8",jest:"^29.7.0",prettier:"^3.8.3","ts-jest":"^29.2.0",typescript:"^5.7.0"},engines:{node:">=18.0.0"},mcpName:"io.github.visa-crypto-labs/visa-cli",files:["bin/visa-cli.js","dist/","install.ps1","native/visa-keychain.m","server.json","README.md","LICENSE"]}});var tr=require("commander"),Je=g(require("crypto")),nr=g(require("fs")),H=g(require("os")),be=g(require("path")),rr=g(require("readline")),sr=require("child_process"),or=require("util");var Ce=require("child_process"),Nt=require("util"),z=g(require("fs")),ke=g(require("os")),Qe=g(require("path")),B=(0,Nt.promisify)(Ce.execFile),et=Qe.join(ke.homedir(),".visa-mcp"),we=Qe.join(et,"session-token"),M="visa-cli",J="session-token",_e="rc-access",yr=5e3;async function Sr(){try{let{stdout:e}=await B("security",["find-generic-password","-s",M,"-a",J,"-w"],{timeout:5e3});return e.trim()||null}catch{return null}}async function vr(e){try{try{await B("security",["delete-generic-password","-s",M,"-a",J],{timeout:5e3})}catch{}return await B("security",["add-generic-password","-s",M,"-a",J,"-w",e],{timeout:5e3}),!0}catch{return!1}}async function br(){try{await B("security",["delete-generic-password","-s",M,"-a",J],{timeout:5e3})}catch{}}async function wr(){if(!tt())return null;try{let{stdout:e}=await B("secret-tool",["lookup","service",M,"account",J],{timeout:5e3});return e.trim()||null}catch{return null}}async function _r(e){if(!tt())return!1;try{let t=(0,Ce.execFile)("secret-tool",["store","--label",`${M} ${J}`,"service",M,"account",J]);return t.stdin?(t.stdin.write(e),t.stdin.end(),await Promise.race([new Promise((n,r)=>{t.on("exit",s=>s===0?n():r(new Error(`secret-tool exited ${s}`))),t.on("error",r)}),new Promise((n,r)=>setTimeout(()=>{t.kill(),r(new Error("secret-tool timed out"))},yr))]),!0):!1}catch{return!1}}async function Cr(){if(tt())try{await B("secret-tool",["clear","service",M,"account",J],{timeout:5e3})}catch{}}function tt(){return!!process.env.DBUS_SESSION_BUS_ADDRESS}async function kr(){try{let{stdout:e}=await B("security",["find-generic-password","-s",M,"-a",_e,"-w"],{timeout:5e3});return e.trim()||null}catch{return null}}async function $r(e){try{try{await B("security",["delete-generic-password","-s",M,"-a",_e],{timeout:5e3})}catch{}await B("security",["add-generic-password","-s",M,"-a",_e,"-w",e],{timeout:5e3})}catch{}}async function xr(){try{await B("security",["delete-generic-password","-s",M,"-a",_e],{timeout:5e3})}catch{}}function Xe(){try{return z.readFileSync(we,"utf-8").trim()||null}catch{return null}}function Dt(e){z.mkdirSync(et,{recursive:!0,mode:448}),z.writeFileSync(we,e,{mode:384}),process.platform==="win32"&&Er(we)}function Ze(){try{z.unlinkSync(we)}catch{}}function Er(e){try{let t=ke.userInfo().username;(0,Ce.execFile)("icacls",[e,"/inheritance:r","/grant:r",`${t}:F`],{timeout:5e3},n=>{n&&console.error(`[visa-cli] icacls ACL restriction failed: ${n.message}`)})}catch(t){console.error(`[visa-cli] Failed to invoke icacls: ${t instanceof Error?t.message:String(t)}`)}}function Ye(){switch(process.platform){case"darwin":return{get:Sr,store:vr,delete:br};case"linux":return{get:wr,store:_r,delete:Cr};default:return{get:async()=>Xe(),store:async e=>{try{return Dt(e),!0}catch{return!1}},delete:async()=>Ze()}}}var C=class{static async getSessionToken(){if(process.env.VISA_MOCK_KEYCHAIN==="true")return Promise.resolve("mock-session-token-for-testing");let t=Ye(),n=await t.get();if(n)return n;let r=Xe();return r?(await t.store(r),r):null}static async saveSessionToken(t){if(process.env.VISA_MOCK_KEYCHAIN==="true")return;let n=Ye();if(await n.store(t)){if(await n.get()===t){Ze();return}await n.delete()}if(Dt(t),Xe()!==t)throw new Error("Failed to persist session token. "+(process.platform==="darwin"?'Check Keychain Access permissions for "visa-cli".':`Ensure ${et} is writable.`))}static async getRcAccessToken(){return process.env.VISA_MOCK_KEYCHAIN==="true"?"mock-rc-token-for-testing":kr()}static async saveRcAccessToken(t){process.env.VISA_MOCK_KEYCHAIN!=="true"&&await $r(t)}static async deleteSessionToken(){if(process.env.VISA_MOCK_KEYCHAIN==="true")return;await Ye().delete(),Ze()}static async clearAll(){await this.deleteSessionToken(),await xr()}};var $e=g(require("crypto")),xe=g(require("tty")),Ee=g(require("fs"));var Q="6820f6e91b762e645c9bf020c0d3673bb99d4a25a824880c0d548e10bb9bc7b1";function Pr(e){return/-rc\.|-beta\./.test(e)}function rt(e){return $e.createHash("sha256").update(e.trim()).digest("hex")}function Ut(e){return Q==="SKIP"?!0:$e.timingSafeEqual(Buffer.from(rt(e)),Buffer.from(Q))}function Tr(e){return new Promise((t,n)=>{let r=Ee.openSync("/dev/tty","r+"),s=new xe.ReadStream(r),o=new xe.WriteStream(r),i=()=>{try{s.destroy()}catch{}try{o.destroy()}catch{}try{Ee.closeSync(r)}catch{}};o.write(e),s.setRawMode(!0),s.resume(),s.setEncoding("utf8");let a="";s.on("data",c=>{c==="\r"||c===`
|
|
2
2
|
`?(o.write(`
|
|
3
3
|
`),i(),t(a)):c===""?(o.write(`
|
|
4
|
-
`),i(),n(new Error("Cancelled"))):c==="\x7F"||c==="\b"?a.length>0&&(a=a.slice(0,-1),o.write("\b \b")):(a+=c,o.write("\u2022"))})})}var
|
|
4
|
+
`),i(),n(new Error("Cancelled"))):c==="\x7F"||c==="\b"?a.length>0&&(a=a.slice(0,-1),o.write("\b \b")):(a+=c,o.write("\u2022"))})})}var Ar=`
|
|
5
5
|
\u2588\u2588\u2557 \u2588\u2588\u2557\u2588\u2588\u2557\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2557 \u2588\u2588\u2588\u2588\u2588\u2557
|
|
6
6
|
\u2588\u2588\u2551 \u2588\u2588\u2551\u2588\u2588\u2551\u2588\u2588\u2554\u2550\u2550\u2550\u2550\u255D\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2557
|
|
7
7
|
\u2588\u2588\u2551 \u2588\u2588\u2551\u2588\u2588\u2551\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2557\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2551
|
|
@@ -10,26 +10,26 @@
|
|
|
10
10
|
\u255A\u2550\u2550\u2550\u255D \u255A\u2550\u255D\u255A\u2550\u2550\u2550\u2550\u2550\u2550\u255D\u255A\u2550\u255D \u255A\u2550\u255D
|
|
11
11
|
|
|
12
12
|
This is a Release Candidate build. Access is restricted to Visa employees.
|
|
13
|
-
`;async function
|
|
14
|
-
`),process.exit(1)}let s=await C.getRcAccessToken();if(s&&(Q==="SKIP"||s===Q))return;console.log(
|
|
13
|
+
`;async function Mt(e={}){let t=e.version??nt().version;if(!Pr(t))return;let n=process.env.VISA_RC_CODE;if(n&&Ut(n)){await C.saveRcAccessToken(rt(n));return}if(e.isMcp??!1){let i=await C.getRcAccessToken();if(i&&(Q==="SKIP"||i===Q))return;process.stderr.write(`[visa-cli] RC build requires access. Run: visa-cli setup
|
|
14
|
+
`),process.exit(1)}let s=await C.getRcAccessToken();if(s&&(Q==="SKIP"||s===Q))return;console.log(Ar);let o=3;for(let i=1;i<=o;i++){let a;try{a=await Tr(" Enter RC access code: ")}catch{process.exit(1)}if(Ut(a)){await C.saveRcAccessToken(rt(a)),console.log(`
|
|
15
15
|
Access granted. Welcome.
|
|
16
16
|
`);return}i<o&&console.log(`
|
|
17
17
|
Invalid code. ${o-i} attempt(s) remaining.
|
|
18
18
|
`)}console.log(`
|
|
19
19
|
Invalid code. Contact your team lead.
|
|
20
|
-
`),process.exit(1)}var
|
|
21
|
-
`)}function
|
|
22
|
-
`))n(` ${c}`);n("");let i=
|
|
23
|
-
`),
|
|
24
|
-
${
|
|
25
|
-
`);l>=0&&(a=a.slice(l+1))}}return a}catch{return null}finally{if(t!==void 0)try{
|
|
26
|
-
`),n=[],r=new Set;for(let s=t.length-1;s>=0&&n.length<
|
|
27
|
-
`)}function ee(e){return e?e.replace(/\u001B\][^\u0007]*(?:\u0007|\u001B\\)/g,"").replace(/\u001B\[[0-?]*[ -/]*[@-~]/g,"").replace(/[\u0000-\u0008\u000B-\u001F\u007F-\u009F]/g,"").replace(/\s+/g," ").trim():""}function
|
|
28
|
-
`;process.stderr.write(s),
|
|
20
|
+
`),process.exit(1)}var jt=require("child_process");function Ir(e=process.env,t=process.platform){return e.VISA_CLI_NO_BROWSER==="1"||e.VISA_CLI_NO_BROWSER==="true"?{headless:!0,reason:"VISA_CLI_NO_BROWSER is set"}:e.CI==="true"||e.CI==="1"?{headless:!0,reason:"CI environment detected"}:e.SSH_CONNECTION||e.SSH_TTY?{headless:!0,reason:"SSH session detected"}:t==="linux"&&!e.DISPLAY&&!e.WAYLAND_DISPLAY?{headless:!0,reason:"Linux with no $DISPLAY or $WAYLAND_DISPLAY"}:{headless:!1}}function Lr(e){let n=e.length+4;return[`\u250C${"\u2500".repeat(n)}\u2510`,`\u2502${" ".repeat(2)}${e}${" ".repeat(2)}\u2502`,`\u2514${"\u2500".repeat(n)}\u2518`].join(`
|
|
21
|
+
`)}function Or(e,t=process.platform){return t==="darwin"?{cmd:"open",args:[e]}:t==="win32"?{cmd:"cmd",args:["/c","start","",e]}:t==="linux"?{cmd:"xdg-open",args:[e]}:null}async function Ht(e,t={}){let n=t.log??(c=>console.log(c)),r=t.env??process.env,s=t.platform??process.platform,o=t.spawn??((c,l,u)=>{(0,jt.execFile)(c,l,f=>u(f))});n(""),n(" Sign in to Visa CLI by opening this URL in your browser:"),n("");for(let c of Lr(e).split(`
|
|
22
|
+
`))n(` ${c}`);n("");let i=Ir(r,s);if(i.headless){n(` (${i.reason} \u2014 skipping browser auto-open.)`),n(" Open the URL above on any device with a browser. The CLI will"),n(" continue waiting for you to complete sign-in."),n("");return}let a=Or(e,s);if(!a){n(` No known browser-open command for platform "${s}".`),n(" Open the URL above manually to continue."),n("");return}await new Promise(c=>{o(a.cmd,a.args,l=>{l?(n(` Could not open browser automatically (${l.message}).`),n(" Open the URL above manually to continue."),n("")):(n(" Opened browser. Waiting for you to sign in..."),n("")),c()})})}var Ft=g(require("crypto")),P=g(require("fs")),Re=g(require("path"));function Bt(e,t){P.mkdirSync(Re.dirname(e),{recursive:!0});let n=`${e}.${process.pid}.${Ft.randomBytes(8).toString("hex")}.tmp`;try{P.writeFileSync(n,JSON.stringify(t,null,2)+`
|
|
23
|
+
`),P.renameSync(n,e)}catch(r){try{P.unlinkSync(n)}catch{}throw r}}function Vt(e){return`'${e.replace(/'/g,"'\\''")}'`}var Kt="# visa-cli-hud-v1";function Nr(e,t){let n=e??process.execPath,r=t??process.argv[1]??"",s=r?Re.resolve(r):"";return`${s?`${Vt(n)} ${Vt(s)} statusline`:"visa-cli statusline"} ${Kt}`}function qt(e){return typeof e!="string"?!1:e.includes(Kt)?!0:e.includes("visa-cli")&&e.includes("statusline")}function st(e,t=Nr){let n={},r=!1;if(P.existsSync(e)){r=!0;try{n=JSON.parse(P.readFileSync(e,"utf-8"))}catch(s){return{installed:"malformed-json",message:`~/.claude/settings.json is not valid JSON (${s.message}). Fix the file manually, then run: visa-cli hud enable`}}}if(n.statusLine){let s=typeof n.statusLine=="object"?n.statusLine.command:"";return qt(s)?{installed:"already-visa",message:"Visa HUD already registered in ~/.claude/settings.json."}:{installed:"other-hud-present",message:"Another HUD is already configured (keeping it). To switch to Visa HUD, edit ~/.claude/settings.json \u2192 statusLine.command"}}try{return n.statusLine={type:"command",command:t()},Bt(e,n),{installed:"new",message:`Visa HUD registered in ~/.claude/settings.json${r?"":" (new file)"}. Restart Claude Code to see it pinned below the input.`}}catch(s){return{installed:"error",message:`Failed to write settings: ${s.message}. Enable manually later with: visa-cli hud enable`}}}function Wt(e){if(!P.existsSync(e))return{removed:!1,message:"No ~/.claude/settings.json found."};let t;try{t=JSON.parse(P.readFileSync(e,"utf-8"))}catch(r){return{removed:!1,message:`~/.claude/settings.json is not valid JSON: ${r.message}`}}if(!t.statusLine)return{removed:!1,message:"No statusLine configured."};let n=typeof t.statusLine=="object"?t.statusLine.command:"";return qt(n)?(delete t.statusLine,Bt(e,t),{removed:!0,message:"Visa HUD removed from ~/.claude/settings.json. Restart Claude Code to take effect."}):{removed:!1,message:"statusLine is owned by another tool \u2014 leaving it alone."}}var W=g(require("fs")),Y=g(require("path")),Dr=50,Ur=64*1024,Gt=10,ot=80;function Mr(){let e=(process.env.COLORTERM??"").toLowerCase();if(e==="truecolor"||e==="24bit")return!0;let t=(process.env.TERM??"").toLowerCase();return t.includes("truecolor")||t.includes("24bit")}var it=Mr(),S={reset:"\x1B[0m",visaBlue:it?"\x1B[38;2;20;52;203m":"\x1B[38;5;27m",visaBlueSoft:it?"\x1B[38;2;97;126;229m":"\x1B[38;5;111m",visaGold:it?"\x1B[38;2;247;182;0m":"\x1B[38;5;220m",green:"\x1B[38;5;48m",dim:"\x1B[2m"};function b(e,t){return e.length===0||process.env.NO_COLOR?e:`${t}${e}${S.reset}`}function Jt(e){return e.replace(/\u001B\[[0-?]*[ -/]*[@-~]/g,"").replace(/[\u0000-\u001F\u007F]/g,"").length}async function Xt(e=process.stdin){return e.isTTY?null:new Promise(t=>{let n=[],r=!1,s=()=>{e.removeListener("data",i),e.removeListener("end",a),e.removeListener("error",c),clearTimeout(l)},o=u=>{r||(r=!0,s(),t(u))},i=u=>{n.push(typeof u=="string"?Buffer.from(u):u)},a=()=>{let u=Buffer.concat(n).toString("utf-8").trim();if(!u)return o(null);try{let f=JSON.parse(u);o(f&&typeof f=="object"?f:null)}catch{o(null)}},c=()=>o(null),l=setTimeout(()=>{n.length>0?a():o(null)},Dr);e.on("data",i),e.on("end",a),e.on("error",c)})}function Zt(e){let n=Math.round(Math.max(0,Math.min(1,e))*10),r="\u2588".repeat(n),s="\u2591".repeat(10-n);return`${b(r||"",S.visaBlueSoft)}${b(s||"",S.dim)}`}function zt(e){return!Number.isFinite(e)||e<0?"0":e>=1e6?`${(e/1e6).toFixed(1)}M`:e>=1e3?`${Math.round(e/1e3)}k`:String(Math.round(e))}function jr(e){let t=` ${b("\u2502",S.dim)} `,n=ee(e.model?.display_name)||"claude",r=e.cwd?ee(Y.basename(e.cwd)):"",s=zr(e.workspace),o=r||s,i=s&&s!==o?` ${b(`@${s}`,S.visaBlueSoft)}`:"",a=[b(`[${n}]`,S.visaBlueSoft)];o&&a.push(`${b(o,S.visaGold)}${i}`);let c=a.join(t),l=Number(e.context_window?.context_window_size??0),u=typeof e.context_window?.used_percentage=="number"?e.context_window.used_percentage:typeof e.context_window?.remaining_percentage=="number"?100-e.context_window.remaining_percentage:null,f=e.context_window?.current_usage,d=[f?.input_tokens,f?.output_tokens,f?.cache_creation_input_tokens,f?.cache_read_input_tokens].filter(G=>typeof G=="number"&&Number.isFinite(G)),m=d.length>0?d.reduce((G,ze)=>G+ze,0):null,v=Number(e.context_window?.total_input_tokens??m??f?.input_tokens??0),k=u!==null?Math.max(0,Math.min(1,u/100)):l>0?Math.max(0,Math.min(1,v/l)):0,L=u!==null&&l>0?Math.round(k*l):v,$;if(l>0||u!==null){let G=`${String(Math.round(k*100)).padStart(2," ")}%`,ze=l>0?` ${b(`(${zt(L)}/${zt(l)})`,S.dim)}`:"";$=`${b("Context",S.dim)} ${Zt(k)} ${b(G,S.green)}${ze}`}else $=`${b("Context",S.dim)} ${b("-",S.dim)}`;let D=Vr(e.rate_limits),U=[$];D&&U.push(D);let R=U.join(t);return`${c}
|
|
24
|
+
${R}`}function Hr(e){if(typeof e!="number"||e<=0)return"";let n=(e>1e12?e:e*1e3)-Date.now();if(n<=0||n>=10080*60*1e3)return"";let r=Math.floor(n/6e4),s=Math.floor(r/60),o=r%60;return s>0?`${s}h ${o}m`:`${o}m`}function Yt(e,t){if(!e||typeof e.used_percentage!="number")return null;let n=Math.max(0,Math.min(100,e.used_percentage)),r=n/100,s=Zt(r),o=n>=90?"\x1B[38;5;196m":n>=70?S.visaGold:S.green,i=`${String(Math.round(n)).padStart(2," ")}%`,a=Hr(e.resets_at),c=a?` ${b(`(${a} / ${t})`,S.dim)}`:` ${b(`(${t})`,S.dim)}`;return`${s} ${b(i,o)}${c}`}function Vr(e){if(!e)return null;let t=Yt(e.five_hour,"5h"),n=typeof e.seven_day?.used_percentage=="number"?e.seven_day.used_percentage:null,r=n!==null&&n>=90?Yt(e.seven_day,"7d"):null;if(!t&&!r)return null;let s=` ${b("\u2502",S.dim)} `,o=[t,r].filter(Boolean);return`${b("Usage",S.dim)} ${o.join(s)}`}function Fr(e){let t;try{let n=W.statSync(e);if(!n.isFile())return null;let r=n.size;if(r===0)return null;let s=Math.min(r,Ur),o=r-s;t=W.openSync(e,"r");let i=Buffer.alloc(s);W.readSync(t,i,0,s,o);let a=i.toString("utf-8");if(o>0){let c=Buffer.alloc(1);if(W.readSync(t,c,0,1,o-1),c[0]!==10){let l=a.indexOf(`
|
|
25
|
+
`);l>=0&&(a=a.slice(l+1))}}return a}catch{return null}finally{if(t!==void 0)try{W.closeSync(t)}catch{}}}function Br(e){if(!e||typeof e!="object")return;let t=e,n=["file_path","path","notebook_path","pattern","command","url"];for(let r of n){let s=t[r];if(typeof s=="string"&&s.length>0){if(r.endsWith("_path")||r==="path"){let o=Y.basename(Y.dirname(s)),i=Y.basename(s);return o&&o!=="."&&o!=="/"?`${o}/${i}`:i}return s}}}function Kr(e){let t=e.split(`
|
|
26
|
+
`),n=[],r=new Set;for(let s=t.length-1;s>=0&&n.length<Gt*3;s-=1){let o=t[s].trim();if(!o)continue;let i;try{i=JSON.parse(o)}catch{continue}if(!i||typeof i!="object")continue;let c=i.message;if(!c||typeof c!="object")continue;let l=c.content;if(Array.isArray(l))for(let u of l){if(!u||typeof u!="object")continue;let f=u;if(f.type==="tool_result"&&typeof f.tool_use_id=="string")r.add(f.tool_use_id);else if(f.type==="tool_use"&&typeof f.name=="string"){let d=typeof f.id=="string"?f.id:"",m=d.length>0?!r.has(d):!1;n.push({name:f.name,target:Br(f.input),running:m})}}}return n.reverse(),n.slice(Math.max(0,n.length-Gt))}function qr(e){let t=[];for(let n of e){let r=t[t.length-1];r&&!r.running&&!n.running&&r.name===n.name&&r.target===void 0&&n.target===void 0?r.count+=1:t.push({...n,count:1})}return t}function Wr(e){let t=e.running?b("\u25D0",S.visaGold):b("\u2713",S.green),n=e.running?S.visaGold:S.visaBlueSoft,r=b(ee(e.name),n),s=e.target?ee(e.target):"",o=s?`${b(":",S.dim)} ${b(s,S.dim)}`:"",i=e.count>1?` ${b(`\xD7${e.count}`,S.dim)}`:"";return o?`${t} ${r}${o}${i}`:`${t} ${r}${i}`}function Gr(e){let t=` ${b("\u2502",S.dim)} `,n=Jt(t),r=e.slice();for(;r.length>0;){let c=r.join(t);if(Jt(c)<=ot)return c;r=r.slice(1)}if(e.length===0)return"";let s="\u2026",i=e[e.length-1].replace(/\u001B\[[0-?]*[ -/]*[@-~]/g,""),a=i.length>ot-n-1?i.slice(0,ot-n-1)+s:i;return`${b(s,S.dim)}${t}${a}`}async function Jr(e){if(!e||typeof e!="string")return null;let t=Fr(e);if(!t)return null;let n;try{n=Kr(t)}catch{return null}if(n.length===0)return null;let s=qr(n).map(Wr);return Gr(s)||null}async function Qt(e,t){let n=[e];if(t){let r=jr(t);if(r&&n.push(r),t.transcript_path){let s=await Jr(t.transcript_path);s&&n.push(s)}}return n.join(`
|
|
27
|
+
`)}function ee(e){return e?e.replace(/\u001B\][^\u0007]*(?:\u0007|\u001B\\)/g,"").replace(/\u001B\[[0-?]*[ -/]*[@-~]/g,"").replace(/[\u0000-\u0008\u000B-\u001F\u007F-\u009F]/g,"").replace(/\s+/g," ").trim():""}function zr(e){if(typeof e=="string")return ee(Y.basename(e));if(!e||typeof e!="object")return"";let t=typeof e.current_dir=="string"?e.current_dir:typeof e.path=="string"?e.path:"";return ee(t?Y.basename(t):e.name)}async function en(e,t){let n=t?.timeoutMs??3e4,r=new AbortController,s=setTimeout(()=>r.abort(),n);try{let{timeoutMs:o,...i}=t??{};return await fetch(e,{...i,signal:r.signal})}finally{clearTimeout(s)}}var Yr=/^(\d+)\.(\d+)\.(\d+)(?:-([0-9A-Za-z.-]+))?(?:\+[0-9A-Za-z.-]+)?$/;function nn(e,t){let n=tn(e),r=tn(t);if(!n||!r)return!1;for(let s=0;s<3;s++)if(n.main[s]!==r.main[s])return n.main[s]>r.main[s];return n.pre&&!r.pre?!1:!n.pre&&r.pre?!0:!n.pre&&!r.pre?!1:Xr(n.pre,r.pre)>0}function tn(e){if(typeof e!="string")return null;let n=e.trim().replace(/^v/,"").match(Yr);return n?{main:[Number(n[1]),Number(n[2]),Number(n[3])],pre:n[4]??null}:null}function Xr(e,t){let n=e.split("."),r=t.split("."),s=Math.max(n.length,r.length);for(let o=0;o<s;o++){if(o>=n.length)return-1;if(o>=r.length)return 1;let i=n[o],a=r[o],c=/^\d+$/.test(i),l=/^\d+$/.test(a);if(c&&l){let u=Number(i)-Number(a);if(u!==0)return u}else{if(c)return-1;if(l)return 1;if(i<a)return-1;if(i>a)return 1}}return 0}function K(){return!!(rn(process.env.VISA_CLI_NO_UPDATE_CHECK)||rn(process.env.CI)||process.env.NODE_ENV==="test")}function rn(e){if(e===void 0)return!1;let t=e.trim().toLowerCase();return!(t===""||t==="0"||t==="false"||t==="no"||t==="off")}var at="1.8.0-rc.0",re=class{constructor(t){this.getSessionToken=t;this.baseUrl=process.env.VISA_AUTH_URL||"https://auth.visacli.sh"}getSessionToken;baseUrl;lastSignals={};parseServerSignals(t){if(this.lastSignals={},!K()){let r=t.headers.get("X-Latest-Version"),s=t.headers.get("X-Update-Message");r&&nn(r,at)&&(this.lastSignals.updateAvailable={version:r,message:s||`Update available: v${r}. Run: npm install -g @visa/cli && visa-cli setup`})}let n=t.headers.get("X-Feedback-Prompt");if(n)try{this.lastSignals.feedbackPrompt=JSON.parse(n)}catch{}}getClientVersion(){return at}async request(t,n,r,s,o){let i=await this.getSessionToken();if(!i)throw new Error("Not logged in. Sign up at https://visacli.sh or run: visa-cli setup");let a={Authorization:`Bearer ${i}`};o&&(t==="GET"?a["X-User-Context"]=o.replace(/[\r\n\0]/g," ").slice(0,1e3):r={...r||{},user_context:o}),r&&(a["Content-Type"]="application/json");let c;try{c=await en(`${this.baseUrl}${n}`,{method:t,headers:{...a,"X-Visa-CLI-Version":at},body:r?JSON.stringify(r):void 0,timeoutMs:s})}catch(u){throw u.name==="AbortError"||u.message?.includes("aborted")?new Error("The request timed out. The server may be under heavy load. Please try again."):new Error("Cannot reach the Visa CLI server. Check your internet connection and try again.")}if(this.parseServerSignals(c),c.status===401)throw new Error("Your session has expired. Run: visa-cli setup");if(c.status===429){let u=c.headers.get("Retry-After")||"3";throw new Error(`Rate limited \u2014 wait ${u}s. Tip: use the batch tool to combine multiple requests into one.`)}if(c.status===503)throw new Error("Visa CLI is temporarily unavailable. Check https://visacli.sh for status.");let l;try{l=await c.json()}catch{throw c.status===500?new Error(`Server error on ${n}. Try again or check https://visacli.sh for status.`):new Error(`Unexpected response from ${n}. Try again.`)}if(!c.ok)throw c.status===500?new Error(`Server error on ${n}. Try again or check https://visacli.sh for status.`):new Error(l?.error||`Request failed (${c.status}). Try again.`);return l}async pay(t,n){return this.request("POST","/v1/pay",t,void 0,n)}async shortcut(t,n,r,s){return this.request("POST",`/v1/shortcuts/${encodeURIComponent(t)}`,n,r,s)}async batch(t,n,r){return this.request("POST","/v1/batch",t,n,r)}async catalogSearch(t,n){let r=new URLSearchParams;t&&r.set("q",t),n&&r.set("category",n);let s=r.toString();return this.request("GET",`/v1/catalog${s?`?${s}`:""}`)}async catalogTool(t){try{return await this.request("GET",`/v1/catalog/${encodeURIComponent(t)}`)}catch{return null}}async paymentPreview(t,n){return this.request("POST","/v1/payment-preview",t,void 0,n)}async getStatus(t){return this.request("GET","/v1/status",void 0,void 0,t)}async getTransactions(t){return this.request("GET","/v1/transactions",void 0,void 0,t)}async updateSpendingControls(t,n){return this.request("POST","/v1/spending-controls",t,void 0,n)}async removeCard(t,n,r){return this.request("DELETE",`/v1/cards/${encodeURIComponent(String(t))}`,n,void 0,r)}async setDefaultCard(t,n,r){return this.request("POST",`/v1/cards/${encodeURIComponent(String(t))}/default`,n,void 0,r)}async getAttestationChallenge(){return this.request("GET","/v1/attestation-challenge")}async registerAttestationKey(t){return this.request("POST","/v1/attestation-key",{publicKey:t})}async setBiometricPreference(t,n){return this.request("POST","/v1/biometric-preference",{...t,confirm:!0},void 0,n)}async logout(t,n){return this.request("POST","/v1/logout",t,void 0,n)}async feedback(t,n,r){return this.request("POST","/v1/feedback",{message:t,...n&&{transaction_id:n}},void 0,r)}async feedSubmit(t){return this.request("POST","/v1/feed",t)}async feedList(t){let n=new URLSearchParams;t?.tab&&n.set("tab",t.tab),t?.limit&&n.set("limit",String(t.limit)),t?.offset&&n.set("offset",String(t.offset));let r=n.toString();return this.request("GET",`/v1/feed${r?"?"+r:""}`)}async feedVote(t,n){return this.request("POST",`/v1/feed/${encodeURIComponent(t)}/vote`,{direction:n})}async feedApprove(t){return this.request("POST",`/v1/feed/${encodeURIComponent(t)}/approve`)}async feedDelete(t){return this.request("DELETE",`/v1/feed/${encodeURIComponent(t)}`)}async feedPending(){return this.request("GET","/v1/feed/pending")}async submitFeedback(t,n,r){return this.request("POST","/v1/feedback",{message:t,...n&&{transaction_id:n}},void 0,r)}async getFeedback(t,n){let r=new URLSearchParams;t&&r.set("limit",String(t));let s=r.toString();return this.request("GET",`/v1/feedback${s?"?"+s:""}`,void 0,void 0,n)}async submitRatedFeedback(t){return this.request("POST","/v1/feedback",t)}};var ft=require("child_process"),un=require("util"),dn=g(require("crypto")),x=g(require("fs")),fn=g(require("os")),q=g(require("path"));var T=g(require("fs")),ut=g(require("path")),sn=g(require("os")),lt=ut.join(sn.homedir(),".visa-mcp"),le=ut.join(lt,"mcp-server.log"),Zr=5*1024*1024,ct=null;function Qr(){T.existsSync(lt)||T.mkdirSync(lt,{recursive:!0,mode:448})}function es(){if(!ct){if(Qr(),T.existsSync(le)&&T.statSync(le).size>Zr){let t=le+".1";T.existsSync(t)&&T.unlinkSync(t),T.renameSync(le,t)}ct=T.createWriteStream(le,{flags:"a"})}return ct}function Pe(e,...t){let n=new Date().toISOString(),r=t.map(o=>typeof o=="string"?o:JSON.stringify(o,null,2)).join(" "),s=`[${n}] [${e}] ${r}
|
|
28
|
+
`;process.stderr.write(s),es().write(s)}var se={debug:(...e)=>Pe("DEBUG",...e),info:(...e)=>Pe("INFO",...e),warn:(...e)=>Pe("WARN",...e),error:(...e)=>Pe("ERROR",...e)};var oe=(0,un.promisify)(ft.execFile),Ie=q.join(fn.homedir(),".visa-mcp","bin"),te=q.join(Ie,"Visa CLI"),ts=q.join(__dirname,"..","native"),on="5",an=q.join(Ie,"visa-keychain.version"),cn=q.join(Ie,"visa-keychain.sha256");function ln(e){let t=x.readFileSync(e);return dn.createHash("sha256").update(t).digest("hex")}async function pn(){try{if(x.readFileSync(an,"utf-8").trim()===on&&x.existsSync(te)){let r=x.readFileSync(cn,"utf-8").trim();if(ln(te)!==r)se.warn("binary:hash-mismatch",{message:"Binary hash mismatch \u2014 possible tampering detected. Recompiling from source."}),x.unlinkSync(te);else return te}}catch{}let e=q.join(ts,"visa-keychain.m");if(x.existsSync(e)||(e=q.resolve(__dirname,"..","..","native","visa-keychain.m")),x.existsSync(e)||(e=q.resolve(__dirname,"..","native","visa-keychain.m")),!x.existsSync(e))throw new Error("visa-keychain.m source not found. Reinstall Visa CLI.");x.mkdirSync(Ie,{recursive:!0,mode:448});try{await oe("clang",["-framework","Security","-framework","LocalAuthentication","-framework","Foundation","-framework","AppKit","-o",te,e],{timeout:3e4})}catch(n){throw n.code==="ENOENT"?new Error("Xcode Command Line Tools required. Install: xcode-select --install"):n}let t=ln(te);return x.writeFileSync(cn,t,{mode:384}),x.writeFileSync(an,on,{mode:384}),te}async function gn(e){let t=await pn(),n;try{n=(await oe(t,e,{timeout:6e4})).stdout}catch(o){n=o.stdout||"";let i=n.trim();throw i.startsWith("ERROR:")?new Error(i.slice(6)):new Error(o.stderr?.trim()||o.message||"Unknown error")}let r=n.trim();if(r.startsWith("OK:"))return r.slice(3);if(r==="OK")return;let s=r.startsWith("ERROR:")?r.slice(6):"Unknown error";throw new Error(s)}var dt=null;function A(){return process.env.VISA_MOCK_TOUCHID==="true"?!0:process.platform!=="darwin"?!1:dt!==null?dt:(dt=!0,!0)}var Te="visa-cli",Ae="attestation-key";async function ns(e){try{await oe("security",["delete-generic-password","-s",Te,"-a",Ae],{timeout:5e3})}catch{}await oe("security",["add-generic-password","-s",Te,"-a",Ae,"-w",e],{timeout:5e3})}async function rs(){try{let{stdout:e}=await oe("security",["find-generic-password","-s",Te,"-a",Ae,"-w"],{timeout:5e3});return e.trim()||null}catch{return null}}async function Le(){let e=await gn(["generate-key"]);if(!e)throw new Error("Key generation returned no output");let t=e.indexOf(":");if(t<0)throw new Error("Unexpected generate-key output format");let n=e.slice(0,t),r=e.slice(t+1);return await ns(n),r}async function pt(e,t){if(process.env.VISA_MOCK_TOUCHID==="true")return Promise.resolve("mock-ecdsa-signature-for-testing");let n=await rs();if(!n)throw new Error("Attestation key not found. Run setup to generate a new key.");let r=await pn(),s=["sign",e];return t&&s.push(t),new Promise((o,i)=>{let a=(0,ft.execFile)(r,s,{timeout:6e4},(c,l)=>{let u=(l||"").trim();if(c){u.startsWith("ERROR:")?i(new Error(u.slice(6))):i(new Error(c.stderr?.trim()||c.message||"Unknown error"));return}u.startsWith("OK:")?o(u.slice(3)):i(new Error(u.startsWith("ERROR:")?u.slice(6):"Unknown error"))});a.stdin.write(n),a.stdin.end()})}async function mn(){try{await oe("security",["delete-generic-password","-s",Te,"-a",Ae],{timeout:5e3})}catch{}try{await gn(["delete-key"])}catch{}}function hn(e,t=process.stderr){if(K()||!e?.updateAvailable)return!1;let{message:n}=e.updateAvailable;return n?(t.write(`
|
|
29
29
|
\x1B[33m\u2191 ${n}\x1B[0m
|
|
30
|
-
`),!0):!1}function
|
|
30
|
+
`),!0):!1}function Sn(e,t,n,r){let o=us(e.spendingControls).dailyLimit,i=Math.max(0,ue(e.dailySpent)),a=o>0?Math.min(o,Math.max(0,ue(e.dailyRemaining??o-i))):0,c=o>0?Math.min(1,i/o):0,u=(Array.isArray(t)?t:[]).filter(is),f=u.slice(0,3),d=as(u,3),m=(e.cards??[]).slice(0,3),v=r.latestVersion?mt(r.latestVersion):"",k=r.updateCheckDisabled?"update checks disabled":v?`update available: v${v}`:"up to date",L=Math.round(c*100),$=X(e.status,e.enrolled?"approved":"not enrolled"),U=["VISA CLI",`Status: ${e.enrolled?"Visa ready":"Visa setup needed"} | account: ${$} | touch id: ${n?"ready":"unavailable"}`,`Version: v${r.currentVersion} | ${k}`,`Spend meter: ${yn(c)} ${String(L).padStart(3," ")}% | remaining ${ie(a)}/day | daily cap ${ie(o)}`,"","Spend",` Remaining ${ie(a)} / ${ie(o)}`,` Usage ${yn(c)} ${L}%`,` Spent today ${ie(i)}`,` Attestation key ${e.hasAttestationKey?"registered":"missing"}`,"","Cards",...m.length>0?m.map(R=>` ${ls(R)}`):[" No cards enrolled"],"","Last 3 services",...d.length>0?d.map((R,G)=>` ${G+1}. ${R}`):[" No paid services yet"],"","Recent receipts",...f.length>0?f.map(R=>` ${cs(R)}`):[" No receipts yet"]];return r.updateMessage&&U.push("",`Update: ${mt(r.updateMessage)}`),`${U.join(`
|
|
31
31
|
`)}
|
|
32
|
-
`}function
|
|
32
|
+
`}function ss(e){return e.tool_name!=null}function os(e){return ue(e.amount)===0&&e.status==="failed"}function is(e){return ss(e)&&!os(e)}function as(e,t=1/0){let n=new Set,r=[];for(let s of e){if(r.length>=t)break;let o=X(s.merchant_name,"Unknown merchant"),i=X(s.tool_name,"unknown_tool"),a=`${o} :: ${i}`;n.has(a)||(n.add(a),r.push(`${o} [${i}]`))}return r}function cs(e){let t=X(e.merchant_name,"Unknown merchant"),n=X(e.tool_name,"unknown_tool"),r=ie(ue(e.amount)),s=X(e.status,"unknown"),o=ds(e.created_at);return`${r} ${s.padEnd(9)} ${t} [${n}] ${o}`}function ls(e){let t=X(e.brand?.toUpperCase(),"CARD"),n=e.isDefault?" default":"";return`${Number.isInteger(e.id)?`#${e.id} `:""}${t} ****${e.last4}${n}`}function us(e){return{dailyLimit:ue(e?.daily_limit??e?.dailyLimit??0)}}function ue(e){let t=Number(typeof e=="string"?e:e??0);return Number.isFinite(t)?t:0}function ie(e){return`$${e.toFixed(2)}`}function yn(e,t=20){let n=Math.max(0,Math.min(1,e)),r=Math.round(n*t);return`[${"\u2588".repeat(r)}${"\u2591".repeat(t-r)}]`}function X(e,t){let n=mt(e??"").trim();return n.length>0?n:t}function ds(e){if(!e)return"unknown time";let t=new Date(e);return Number.isNaN(t.getTime())?X(e,"unknown time"):t.toISOString().slice(0,16).replace("T"," ")}function mt(e){return e.replace(/\u001B\][^\u0007]*(?:\u0007|\u001B\\)/g,"").replace(/\u001B\[[0-?]*[ -/]*[@-~]/g,"").replace(/[\u0000-\u0008\u000B-\u001F\u007F-\u009F]/g,"").replace(/\s+/g," ")}var y=g(require("fs")),Me=g(require("os")),w=g(require("path")),bn=require("child_process"),fs=2,Oe="# >>> visa-cli shell hud v2 >>>",Ne="# <<< visa-cli shell hud v2 <<<",ps="# >>> visa-cli shell hud >>>",gs="# <<< visa-cli shell hud <<<",wn=3e4,ms=3e4;function vt(){try{return w.join(Rn(),".visa-cli")}catch{return w.join(Me.tmpdir(),".visa-cli")}}function fe(){return w.join(vt(),"shell-hud.json")}function je(){return w.join(vt(),"shell-hud.line")}function _n(){return w.join(vt(),"shell-hud.lock")}function He(e){let t=e??process.env.SHELL;if(!t)return null;let n=w.basename(t.replace(/\\/g,"/")).toLowerCase();return n==="zsh"?"zsh":n==="bash"?"bash":n==="pwsh"||n==="pwsh.exe"||n==="powershell"||n==="powershell.exe"?"powershell":null}function Ve(e){let t=Rn();if(e==="zsh")return w.join(t,".zshrc");if(e==="bash")return w.join(t,".bashrc");let n=(process.platform==="win32","PowerShell");return w.join(t,"Documents",n,"Microsoft.PowerShell_profile.ps1")}function hs(e){let t="$HOME/.visa-cli/shell-hud.line",n=_s(e),r=Cs(e);if(e==="zsh")return`${Oe}
|
|
33
33
|
_visa_cli_shell_hud_precmd() {
|
|
34
34
|
setopt localoptions no_bg_nice
|
|
35
35
|
if [[ -f "${t}" ]]; then
|
|
@@ -46,7 +46,7 @@ autoload -Uz add-zsh-hook
|
|
|
46
46
|
if [[ -z "\${precmd_functions[(r)_visa_cli_shell_hud_precmd]}" ]]; then
|
|
47
47
|
add-zsh-hook precmd _visa_cli_shell_hud_precmd
|
|
48
48
|
fi
|
|
49
|
-
${
|
|
49
|
+
${Ne}`;if(e==="powershell"){let s="(Join-Path $HOME '.visa-cli/shell-hud.line')";return`${Oe}
|
|
50
50
|
if (-not (Test-Path Function:\\global:__visa_cli_original_prompt)) {
|
|
51
51
|
$function:global:__visa_cli_original_prompt = if (Test-Path Function:\\prompt) { $function:prompt } else { { '' } }
|
|
52
52
|
}
|
|
@@ -65,7 +65,7 @@ function global:prompt {
|
|
|
65
65
|
}
|
|
66
66
|
& $function:global:__visa_cli_original_prompt
|
|
67
67
|
}
|
|
68
|
-
${
|
|
68
|
+
${Ne}`}return`${Oe}
|
|
69
69
|
__visa_cli_shell_hud_precmd() {
|
|
70
70
|
if [ -f "${t}" ]; then
|
|
71
71
|
cat "${t}"
|
|
@@ -82,34 +82,35 @@ case ";$PROMPT_COMMAND;" in
|
|
|
82
82
|
*";__visa_cli_shell_hud_precmd;"*) ;;
|
|
83
83
|
*) PROMPT_COMMAND="__visa_cli_shell_hud_precmd\${PROMPT_COMMAND:+;$PROMPT_COMMAND}" ;;
|
|
84
84
|
esac
|
|
85
|
-
${
|
|
85
|
+
${Ne}`}function Cn(e){let t=e;for(let[n,r]of[[Oe,Ne],[ps,gs]])t=t.replace(new RegExp(`\\n?${vn(n)}[\\s\\S]*?${vn(r)}\\n?`,"g"),"");return t.trimEnd()}function ys(e,t){let n=Cn(e).trimEnd(),r=hs(t);return n.length>0?`${n}
|
|
86
86
|
|
|
87
87
|
${r}
|
|
88
88
|
`:`${r}
|
|
89
|
-
`}function
|
|
90
|
-
`:""),{removed:!0,shell:e,rcPath:t,message:`Removed persistent shell HUD from ${t}.`})}catch(n){return{removed:!1,shell:e,rcPath:t,message:`Failed to remove persistent shell HUD from ${t}: ${
|
|
91
|
-
`),
|
|
92
|
-
`)}catch{}}function
|
|
93
|
-
|
|
94
|
-
`),
|
|
95
|
-
`),
|
|
89
|
+
`}function kn(e=He()){if(!e)return{installed:!1,changed:!1,shell:null,message:"Shell HUD auto-install skipped: supported shells are zsh, bash, and PowerShell."};let t;try{t=Ve(e);let n=y.existsSync(t)?y.readFileSync(t,"utf-8"):"",r=ys(n,e),s=r!==n;return s&&Ue(t,r),{installed:!0,changed:s,shell:e,rcPath:t,message:s?`Persistent shell HUD installed in ${t}. Open a new terminal to start seeing it. Disable it any time with: visa-cli shell-hud disable`:`Persistent shell HUD already installed in ${t}. Disable it any time with: visa-cli shell-hud disable`}}catch(n){return{installed:!1,changed:!1,shell:e,rcPath:t,message:`Failed to install persistent shell HUD in ${t}: ${Pn(n)}`}}}function $n(e=He()){if(!e)return{removed:!1,shell:null,message:"Shell HUD uninstall skipped: supported shells are zsh, bash, and PowerShell."};let t;try{if(t=Ve(e),!y.existsSync(t))return{removed:!1,shell:e,rcPath:t,message:`No ${e} rc file found at ${t}.`};let n=y.readFileSync(t,"utf-8"),r=Cn(n);return r===n.trimEnd()?{removed:!1,shell:e,rcPath:t,message:`Persistent shell HUD was not installed in ${t}.`}:(Ue(t,r.length>0?`${r}
|
|
90
|
+
`:""),{removed:!0,shell:e,rcPath:t,message:`Removed persistent shell HUD from ${t}.`})}catch(n){return{removed:!1,shell:e,rcPath:t,message:`Failed to remove persistent shell HUD from ${t}: ${Pn(n)}`}}}function bt(e,t){let n=De(t.currentVersion),r=` ${j("\u2502",O.dim)} `;if(!e.enrolled)return`${j("VISA CLI",O.visaBlue)} ${j(`v${n}`,O.visaBlueSoft)}${r}${j("setup needed",O.visaGold)}`;let s=ht(e.spendingControls?.daily_limit??e.spendingControls?.dailyLimit),o=Math.max(0,ht(e.dailySpent)),i=s>0?Math.min(s,Math.max(0,ht(e.dailyRemaining??s-o))):0,a=s>0?Math.max(0,Math.min(1,o/s)):0,c=bs(e),l=s>0?`${vs(a)} ${j(`${yt(i)} left today`,O.green)} ${j(`(${yt(o)}/${yt(s)}/day)`,O.dim)}`:j("no spend limit",O.dim);return`${j("VISA CLI",O.visaBlue)} ${j(`v${n}`,O.visaBlueSoft)}${r}${j(c,O.visaGold)}${r}${l}`}function xn(){let e=En();return!!e&&Date.now()-e.renderedAt<=wn}function wt(){let e=En();if(e&&Date.now()-e.renderedAt<=wn||(Ss(),e?.line))return e.line;try{let t=je();if(y.existsSync(t))return y.readFileSync(t,"utf-8").trimEnd()}catch{}return"VISA | loading spend HUD\u2026"}function _t(e){try{let t=w.dirname(fe());y.mkdirSync(t,{recursive:!0});let n=De(e),r={hudVersion:fs,renderedAt:Date.now(),line:n};Ue(fe(),JSON.stringify(r)+`
|
|
91
|
+
`),Ue(je(),n+`
|
|
92
|
+
`)}catch{}}function de(){try{y.unlinkSync(_n())}catch{}}function En(){let e=fe();if(!y.existsSync(e))return null;try{return JSON.parse(y.readFileSync(e,"utf-8"))}catch{return null}}function Ss(){let e=_n();try{if(y.mkdirSync(w.dirname(e),{recursive:!0}),!ks(e))return;let t=process.argv[1]?w.resolve(process.argv[1]):"";if(!t){de();return}let n=(0,bn.spawn)(process.execPath,[t,"shell-hud","refresh"],{detached:!0,stdio:"ignore",env:{...process.env,VISA_CLI_SHELL_HUD_BACKGROUND:"1"}});n.once("error",de),n.unref()}catch{de()}}function ht(e){let t=Number(typeof e=="string"?e:e??0);return Number.isFinite(t)?t:0}function De(e){return e.replace(/\u001B\][^\u0007]*(?:\u0007|\u001B\\)/g,"").replace(/\u001B\[[0-?]*[ -/]*[@-~]/g,"").replace(/\u001B[P_^][^\u001B]*(?:\u001B\\|\u0007)/g,"").replace(/[\u0000-\u0008\u000B-\u001F\u007F-\u009F]/g,"").replace(/\s+/g," ").trim()}function vs(e){let n=Math.round(Math.max(0,Math.min(1,e))*10),r="\u2588".repeat(n),s="\u2591".repeat(10-n);return`${j(r||"",O.green)}${j(s||"",O.dim)}`}function yt(e){return`$${e.toFixed(2)}`}function bs(e){let t=e.cards?.find(a=>a.isDefault)??e.cards?.[0];if(!t)return"card none";let n=typeof t.brand=="string"?t.brand:"card",r=typeof t.last4=="string"?t.last4:"????",s=De(n.toUpperCase()),o=De(r);return`${t.isDefault?"default":"active"} ${s} ****${o}`}function ws(){let e=(process.env.COLORTERM??"").toLowerCase();if(e==="truecolor"||e==="24bit")return!0;let t=(process.env.TERM??"").toLowerCase();return t.includes("truecolor")||t.includes("24bit")}var St=ws(),O={reset:"\x1B[0m",visaBlue:St?"\x1B[38;2;20;52;203m":"\x1B[38;5;27m",visaBlueSoft:St?"\x1B[38;2;97;126;229m":"\x1B[38;5;111m",visaGold:St?"\x1B[38;2;247;182;0m":"\x1B[38;5;220m",green:"\x1B[38;5;48m",dim:"\x1B[2m"};function j(e,t){return e.length===0||process.env.NO_COLOR?e:`${t}${e}${O.reset}`}function vn(e){return e.replace(/[.*+?^${}()|[\]\\]/g,"\\$&")}function Rn(){let e=Me.homedir();if(!e||!w.isAbsolute(e))throw new Error("unable to determine a valid home directory");return w.resolve(e)}function _s(e){return e==="powershell"?"& visa-cli shell-hud refresh":"visa-cli shell-hud refresh"}function Cs(e){return e==="powershell"?"Get-Command visa-cli -ErrorAction SilentlyContinue -CommandType Application":"command -v visa-cli >/dev/null 2>&1"}function ks(e){for(let t=0;t<2;t+=1){let n;try{return n=y.openSync(e,"wx"),y.writeFileSync(n,String(Date.now())),!0}catch(r){if(r?.code!=="EEXIST")return!1;try{let s=y.statSync(e);if(Date.now()-s.mtimeMs<=ms)return!1;y.unlinkSync(e)}catch{}}finally{n!==void 0&&y.closeSync(n)}}return!1}function Pn(e){return e instanceof Error?e.message:"unknown file system error"}function Ue(e,t){y.mkdirSync(w.dirname(e),{recursive:!0});let n=`${e}.${process.pid}.${Date.now()}.tmp`;y.writeFileSync(n,t),y.renameSync(n,e)}async function Ct(e,t){try{return await t()}catch(n){if(n.message==="Invalid signature"&&A()){se.warn("attestation:key-mismatch",{action:"reregistering"});try{let r=await Le();await e.registerAttestationKey(r),se.info("attestation:key-reregistered")}catch(r){throw se.error("attestation:reregister-failure",{error:r.message}),n}return await t()}throw n}}var Z=g(require("fs")),In=g(require("path"));var Fe=g(require("fs")),Tn=g(require("path")),An=g(require("os"));var kt=Tn.join(An.homedir(),".visa-mcp"),pe=class{static ensureConfigDir(){Fe.existsSync(kt)||Fe.mkdirSync(kt,{recursive:!0,mode:448})}static getConfigDir(){return kt}static TOOL_STATES={login:!0,get_status:!0,get_cards:!0,add_card:!0,pay:!0,transaction_history:!0,update_spending_controls:!0,enroll_device:!1,verify_otp:!1,reset:!0,batch:!0,generate_x402_image:!1,browser_launch:!1,browser_navigate:!1,browser_snapshot:!1,browser_click:!1,browser_type:!1,browser_scroll:!1,generate_music_card:!1,generate_image_card:!0,query_onchain_prices_card:!0,generate_music_tempo_card:!0,check_music_status_tempo_card:!0,generate_image_fast_card:!0,pxlwall_card:!1,generate_video_tempo_card:!0};static loadToolStates(){return{...this.TOOL_STATES}}static getDisabledTools(){let t=new Set;for(let[n,r]of Object.entries(this.TOOL_STATES))r||t.add(n);return t}static isToolDisabled(t){return this.TOOL_STATES[t]===!1}};var $s="settings.json";function Be(){return In.join(pe.getConfigDir(),$s)}var ge={"auth.serverUrl":{type:"string",description:"Auth server base URL. Override for staging / self-hosted backends.",requiresRestart:!0,validate:e=>{if(typeof e!="string")throw new Error("auth.serverUrl must be a string");let t;try{t=new URL(e)}catch{throw new Error(`auth.serverUrl must be a valid URL (got: ${JSON.stringify(e)})`)}if(t.protocol!=="https:"&&t.protocol!=="http:")throw new Error(`auth.serverUrl must use http or https (got: ${t.protocol})`)}},"ui.suppressBrowser":{type:"boolean",description:"When true, the CLI/MCP server stops auto-opening result URLs in your browser."},"ui.suppressFeed":{type:"boolean",description:"When true, generated images/music are not auto-submitted to the public Made-with-Visa feed."},"tools.meta":{type:"boolean",description:"Show category meta-tools (generate_image, generate_music, ...). Restart required.",requiresRestart:!0},"tools.specific":{type:"boolean",description:"Show hardcoded per-merchant tools (generate_image_card, query_onchain_prices_card, ...). Restart required.",requiresRestart:!0},"tools.discover":{type:"boolean",description:"Show the dynamic-catalog tools (discover_tools, execute_tool). Restart required.",requiresRestart:!0}};function Ke(){let e=Be();if(!Z.existsSync(e))return{};try{let t=Z.readFileSync(e,"utf-8"),n=JSON.parse(t);return!n||typeof n!="object"||Array.isArray(n)?{}:n}catch{return{}}}function Ln(e){pe.ensureConfigDir();let t=Be(),n=`${t}.tmp`,r=JSON.stringify(e,null,2)+`
|
|
93
|
+
`;Z.writeFileSync(n,r,{mode:384}),Z.renameSync(n,t)}function On(e){let t=Ke()[e];return typeof t=="string"?t:void 0}function Nn(e){let t=Ke()[e];if(typeof t=="boolean")return t;if(t==="true")return!0;if(t==="false")return!1}var ae=class extends Error{constructor(t){let n=Object.keys(ge).sort().join(", ");super(`Unknown setting "${t}". Settable keys: ${n}. For server-controlled values (biometric.*, spending.*) use the dedicated tools (biometric_on/off, update_spending_controls).`),this.name="UnknownSettingKeyError"}},ce=class extends Error{constructor(t){let n="";t.startsWith("biometric.")?n="biometric_on / biometric_off":t.startsWith("spending.")?n="update_spending_controls":t.startsWith("cards.")?n="add_card / remove_card / set_default_card":t.startsWith("account.")&&(n="login / reset"),super(`"${t}" is a server-controlled value and cannot be set via config set. `+(n?`Use ${n} instead.`:"No client-side override is supported.")),this.name="ServerOnlySettingError"}},Dn=["biometric.","spending.","account.","cards.","biometric"];function Un(e,t){if(Dn.some(o=>e.startsWith(o)))throw new ce(e);let n=ge[e];if(!n)throw new ae(e);let r;if(n.type==="boolean")if(typeof t=="boolean")r=t;else if(typeof t=="string"){let o=t.toLowerCase();if(o==="true")r=!0;else if(o==="false")r=!1;else throw new Error(`${e} expects true or false (got: ${JSON.stringify(t)})`)}else throw new Error(`${e} expects a boolean (got: ${typeof t})`);else{if(typeof t!="string"||t.length===0)throw new Error(`${e} expects a non-empty string`);r=t}n.validate&&n.validate(r);let s=Ke();return s[e]=r,Ln(s),{key:e,value:r,requiresRestart:!!n.requiresRestart,path:Be()}}function Mn(e){if(Dn.some(s=>e.startsWith(s)))throw new ce(e);let t=ge[e];if(!t)throw new ae(e);let n=Ke(),r=e in n;return r&&(delete n[e],Ln(n)),{key:e,removed:r,requiresRestart:r&&!!t.requiresRestart,path:Be()}}var jn="1.8.0-rc.0";function $t(e,t){return t?{kind:"env",var:e}:{kind:"default"}}function xs(e,t,n){let r=process.env[e];if(r!==void 0&&r!=="")return{value:r,source:{kind:"env",var:e}};let s=On(t);return s!==void 0?{value:s,source:{kind:"settings"}}:{value:n,source:{kind:"default"}}}function me(e,t,n){let r=process.env[e];if(r!==void 0)return{value:n==="opt-in"?r==="true":r!=="false",source:{kind:"env",var:e}};let s=Nn(t);return s!==void 0?{value:s,source:{kind:"settings"}}:{value:n!=="opt-in",source:{kind:"default"}}}function N(e){return e==null?"\u2014":e?"yes":"no"}function qe(e){return e==null?"\u2014":`$${e.toFixed(2)}`}async function Hn(e){let t=null,n=null;try{t=await e.api.getStatus()}catch($){n=$?.message||"unknown error"}let r=[],s=xs("VISA_AUTH_URL","auth.serverUrl","https://auth.visacli.sh");r.push({key:"auth.serverUrl",value:s.value,formatted:s.value,source:s.source,hint:s.source.kind==="default"?"Persist with `visa-cli config set auth.serverUrl <url>` (or set VISA_AUTH_URL for one-off overrides).":void 0}),r.push({key:"account.enrolled",value:t?.enrolled??null,formatted:N(t?.enrolled),source:t?{kind:"server"}:{kind:"unknown",reason:n||"offline"}}),t?.githubUser&&r.push({key:"account.githubUser",value:t.githubUser,formatted:t.githubUser,source:{kind:"server"}});let o=t?t.attestationRequired!==!1:void 0;r.push({key:"biometric.required",value:o,formatted:N(o),source:t?{kind:"server"}:{kind:"unknown",reason:n||"offline"},hint:o===!1?"Touch ID prompts are suppressed. Re-enable with `visa-cli biometric on`.":void 0}),r.push({key:"biometric.keyRegistered",value:t?.hasAttestationKey??null,formatted:N(t?.hasAttestationKey),source:t?{kind:"server"}:{kind:"unknown",reason:n||"offline"}}),r.push({key:"biometric.deviceAvailable",value:A(),formatted:N(A()),source:{kind:"device"}});let i=t?.spendingControls,a=i?i.max_transaction_amount??i.maxTransactionAmount??null:null,c=i?i.daily_limit??i.dailyLimit??null:null;r.push({key:"spending.maxPerTxn",value:a,formatted:qe(a),source:t?{kind:"server"}:{kind:"unknown",reason:n||"offline"}}),r.push({key:"spending.dailyLimit",value:c,formatted:qe(c),source:t?{kind:"server"}:{kind:"unknown",reason:n||"offline"}}),r.push({key:"spending.dailySpent",value:t?.dailySpent??null,formatted:qe(t?.dailySpent),source:t?{kind:"server"}:{kind:"unknown",reason:n||"offline"}}),r.push({key:"spending.dailyRemaining",value:t?.dailyRemaining??null,formatted:qe(t?.dailyRemaining),source:t?{kind:"server"}:{kind:"unknown",reason:n||"offline"}}),r.push({key:"cards.count",value:t?.cardCount??null,formatted:t?.cardCount!=null?String(t.cardCount):"\u2014",source:t?{kind:"server"}:{kind:"unknown",reason:n||"offline"}});let l=t?.cards?.find($=>$.isDefault)??t?.cards?.[0];l&&r.push({key:"cards.default",value:{brand:l.brand??null,last4:l.last4},formatted:`${l.brand||"card"} \u2022\u2022\u2022\u2022 ${l.last4}`,source:{kind:"server"}});let u=me("VISA_SUPPRESS_BROWSER","ui.suppressBrowser","opt-in");r.push({key:"ui.suppressBrowser",value:u.value,formatted:N(u.value),source:u.source,hint:"Persist with `visa-cli config set ui.suppressBrowser true` to stop auto-opening result URLs."});let f=me("VISA_SUPPRESS_FEED","ui.suppressFeed","opt-in");r.push({key:"ui.suppressFeed",value:f.value,formatted:N(f.value),source:f.source});let d=K(),m;process.env.VISA_CLI_NO_UPDATE_CHECK?m={kind:"env",var:"VISA_CLI_NO_UPDATE_CHECK"}:process.env.CI?m={kind:"env",var:"CI"}:process.env.NODE_ENV==="test"?m={kind:"env",var:"NODE_ENV"}:m={kind:"default"},r.push({key:"ui.updateCheck",value:!d,formatted:N(!d),source:m});let v=me("VISA_META_TOOLS","tools.meta","opt-out");r.push({key:"tools.meta",value:v.value,formatted:N(v.value),source:v.source,hint:"Persist with `visa-cli config set tools.meta false`. Restart Claude Code for changes to take effect."});let k=me("VISA_SPECIFIC_TOOLS","tools.specific","opt-out");r.push({key:"tools.specific",value:k.value,formatted:N(k.value),source:k.source});let L=me("VISA_DISCOVER_TOOLS","tools.discover","opt-out");if(r.push({key:"tools.discover",value:L.value,formatted:N(L.value),source:L.source}),r.push({key:"client.version",value:jn,formatted:jn,source:{kind:"default"}}),e.includeDev){let $=process.env.VISA_MOCK_KEYCHAIN;r.push({key:"dev.mockKeychain",value:$==="true",formatted:N($==="true"),source:$t("VISA_MOCK_KEYCHAIN",!!$)});let D=process.env.VISA_MOCK_TOUCHID;r.push({key:"dev.mockTouchid",value:D==="true",formatted:N(D==="true"),source:$t("VISA_MOCK_TOUCHID",!!D)});let U=process.env.VISA_CLI_DEBUG;r.push({key:"dev.debug",value:!!U,formatted:N(!!U),source:$t("VISA_CLI_DEBUG",!!U)})}return{entries:r,statusError:n}}function Es(e){switch(e.kind){case"default":return"default";case"env":return`env ${e.var}`;case"settings":return"settings.json";case"server":return"server";case"device":return"device";case"unset":return"unset";case"unknown":return`unknown (${e.reason})`}}function Vn(e,t={}){if(e.length===0)return"";let n=Math.max(...e.map(o=>o.key.length)),r=Math.max(...e.map(o=>o.formatted.length)),s=[];for(let o of e){let i=o.key.padEnd(n+2),a=o.formatted.padEnd(r+2);s.push(`${i}${a}(${Es(o.source)})`),t.verbose&&o.hint&&s.push(` ${"\u21B3".padStart(n)} ${o.hint}`)}return s.join(`
|
|
94
|
+
`)}function Fn(e){return JSON.stringify({config:e.entries.map(t=>({key:t.key,value:t.value,source:t.source,hint:t.hint})),statusError:e.statusError},null,2)}var I=class extends Error{constructor(t){super(t),this.name="PayValidationError"}},Bn=["GET","POST"];function Kn(e){let t;try{t=new URL(e)}catch{throw new I(`Invalid URL: ${e}. Expected a fully-qualified http(s) URL.`)}if(t.protocol!=="http:"&&t.protocol!=="https:")throw new I(`Unsupported URL scheme "${t.protocol}". Only http and https are allowed.`);return t}function qn(e){let t=(e??"GET").toUpperCase();if(!Bn.includes(t))throw new I(`Unsupported HTTP method "${e}". Supported: ${Bn.join(", ")}.`);return t}function Wn(e){if(e!==void 0){try{JSON.parse(e)}catch(t){throw new I(`--body is not valid JSON: ${t?.message??"parse error"}`)}return e}}function Gn(e){if(!e||typeof e!="object")throw new I("Merchant returned no payment preview.");let t=e;if(typeof t.amount!="number"||!Number.isFinite(t.amount)||t.amount<=0)throw new I("Could not determine payment amount from merchant.");if(typeof t.merchantName!="string"||t.merchantName.trim().length===0)throw new I("Merchant returned an empty merchant name.");if(t.merchantName.length>200)throw new I(`Merchant name too long (${t.merchantName.length} chars).`);if(typeof t.currency!="string"||t.currency.trim().length===0)throw new I("Merchant returned an empty currency.");if(t.currency.length>10)throw new I(`Currency code too long (${t.currency.length} chars).`);return{amount:t.amount,currency:t.currency,merchantName:t.merchantName}}var h=g(require("fs")),p=g(require("path")),Jn=g(require("os")),_=Jn.homedir(),Rs=["Claude_","Anthropic.ClaudeDesktop_"];function zn(){return process.env.APPDATA||p.join(_,"AppData","Roaming")}function Ps(){return process.env.LOCALAPPDATA||p.join(_,"AppData","Local")}function Yn(){if(process.platform!=="win32")return;let e=p.join(Ps(),"Packages");if(h.existsSync(e))try{let t=h.readdirSync(e,"utf-8");for(let n of Rs){let r=t.filter(s=>s.startsWith(n)).sort()[0];if(r)return p.join(e,r,"LocalCache","Roaming","Claude")}}catch{return}}function Xn(){if(process.platform==="win32"){let e=Yn();return e?p.join(e,"claude_desktop_config.json"):p.join(zn(),"Claude","claude_desktop_config.json")}return p.join(_,"Library","Application Support","Claude","claude_desktop_config.json")}function Zn(){if(process.platform!=="win32")return[p.join(_,"Library","Application Support","Claude")];let e=[p.join(zn(),"Claude")],t=Yn();return t&&e.push(t),e}function We(e){return e.id==="claude-desktop"?Xn():e.globalConfigPath}function Ts(e){return e.id==="claude-desktop"?Zn():e.detectPaths}var V=[{id:"claude",displayName:"Claude Code",globalConfigPath:p.join(_,".claude.json"),configKey:"mcpServers",detectPaths:[p.join(_,".claude.json")],postInstallHint:"Restart Claude Code or run /mcp to connect."},{id:"claude-desktop",displayName:"Claude Desktop",globalConfigPath:Xn(),configKey:"mcpServers",detectPaths:Zn(),postInstallHint:"Restart the Claude desktop app to connect."},{id:"cursor",displayName:"Cursor",globalConfigPath:p.join(_,".cursor","mcp.json"),configKey:"mcpServers",detectPaths:[p.join(_,".cursor")],postInstallHint:"Restart Cursor to connect."},{id:"windsurf",displayName:"Windsurf",globalConfigPath:p.join(_,".codeium","windsurf","mcp_config.json"),configKey:"mcpServers",detectPaths:[p.join(_,".codeium","windsurf")],postInstallHint:"Restart Windsurf to connect."},{id:"cline",displayName:"Cline",globalConfigPath:p.join(_,".vscode","mcp.json"),configKey:"mcpServers",detectPaths:[p.join(_,".vscode","extensions","saoudrizwan.claude-dev-*")],postInstallHint:"Restart VS Code to connect."},{id:"roo-code",displayName:"Roo Code",globalConfigPath:p.join(_,".config","Roo","mcp_settings.json"),configKey:"mcpServers",detectPaths:[p.join(_,".vscode","extensions","RooVeterinaryInc.roo-cline-*")],postInstallHint:"Restart VS Code to connect."},{id:"copilot",displayName:"VS Code Copilot",globalConfigPath:p.join(_,".vscode","mcp.json"),configKey:"servers",detectPaths:[p.join(_,".vscode")],postInstallHint:"Restart VS Code to connect."},{id:"zed",displayName:"Zed",globalConfigPath:p.join(_,".config","zed","settings.json"),configKey:"context_servers",detectPaths:[p.join(_,".config","zed")],postInstallHint:"Restart Zed to connect.",buildEntry:e=>({source:"custom",...e})},{id:"codex",displayName:"Codex",globalConfigPath:p.join(_,".codex","config.toml"),configKey:"mcp_servers",configFormat:"toml",detectPaths:[p.join(_,".codex")],postInstallHint:"Restart Codex to connect."}];function xt(e){return V.find(t=>t.id===e)}function he(e){return Ts(e).some(t=>{if(t.includes("*")){let n=p.dirname(t),r=p.basename(t).replaceAll("*","");if(!h.existsSync(n))return!1;try{return h.readdirSync(n).some(s=>s.startsWith(r))}catch{return!1}}return h.existsSync(t)})}function Qn(){return{command:"node",args:[p.resolve(__dirname,"mcp-server/index.js")]}}function Et(e,t){return t==="project"?"json":e.configFormat??"json"}function As(e){if(e=e.trim(),e==="true")return!0;if(e==="false")return!1;if(e.startsWith('"')&&e.endsWith('"'))return e.slice(1,-1);if(e.startsWith("[")&&e.endsWith("]")){let n=e.slice(1,-1).trim();return n.length===0?[]:n.split(",").map(r=>r.trim()).filter(Boolean).map(r=>r.startsWith('"')&&r.endsWith('"')?r.slice(1,-1):r)}let t=Number(e);return isNaN(t)?e:t}function Rt(e,t){let n=`[mcp_servers.${t}]`,r=e.findIndex(o=>o.trim()===n);if(r===-1)return;let s=r+1;for(;s<e.length&&!e[s].trim().startsWith("[");)s++;return{start:r,end:s}}function Pt(e,t){let n=e.split(`
|
|
95
|
+
`),r=Rt(n,t);if(!r)return;let s={};for(let o=r.start+1;o<r.end;o++){let i=n[o].trim().match(/^(\w+)\s*=\s*(.+)$/);i&&(s[i[1]]=As(i[2]))}return Object.keys(s).length>0?s:void 0}function Is(e,t,n){let r=`[${n.args.map(c=>`"${c}"`).join(", ")}]`,s=[`[mcp_servers.${t}]`,`command = "${n.command}"`,`args = ${r}`],o=e.split(`
|
|
96
|
+
`),i=Rt(o,t);if(i){o.splice(i.start,i.end-i.start,...s);let c=o.join(`
|
|
96
97
|
`);return c.endsWith(`
|
|
97
98
|
`)?c:c+`
|
|
98
99
|
`}let a=e.trimEnd();return a+(a.length>0?`
|
|
99
100
|
|
|
100
101
|
`:"")+s.join(`
|
|
101
102
|
`)+`
|
|
102
|
-
`}function
|
|
103
|
-
`),r=
|
|
103
|
+
`}function Ls(e,t){let n=e.split(`
|
|
104
|
+
`),r=Rt(n,t);return r?(n.splice(r.start,r.end-r.start),n.join(`
|
|
104
105
|
`).replace(/\n{3,}/g,`
|
|
105
106
|
|
|
106
|
-
`)):e}function
|
|
107
|
-
`)}return{installed:!0,configPath:n,message:e.postInstallHint}}function
|
|
108
|
-
`),{removed:!0,configPath:n})}function
|
|
109
|
-
${r}: ${n(e.currentPath)}`}function
|
|
110
|
-
Step 2: Checking authentication...`);let i=await C.getSessionToken();if(i)try{await new re(()=>Promise.resolve(i)).getStatus(),console.log(" Already authenticated.")}catch(d){let m=d instanceof Error?d.message:"";m.includes("session has expired")||m.includes("Not logged in")?(console.log(" Existing session expired \u2014 re-authenticating..."),await C.clearAll(),i=null):console.log(` Couldn't verify session (${m||"unknown error"}) \u2014 continuing with existing token.`)}if(i||(console.log(" No session found. Opening browser for GitHub login..."),i=await new Promise(async(d,m)=>{let v=
|
|
111
|
-
Step 3: Setting up authentication...`),!A())console.log(" Not macOS \u2014 skipping biometric setup.");else{try{await
|
|
112
|
-
Step 4: Enable the Visa spend HUD?`),console.log(" Pins your spend + card info below the Claude Code input, so"),console.log(" you can see your spend, card, and session usage in real time.");let d;if(e.yes?(d=!0,console.log(" (--yes) Enabling Visa HUD.")):process.stdin.isTTY?d=await
|
|
107
|
+
`)):e}function ye(e,t="global"){let n=t==="project"?p.join(process.cwd(),".mcp.json"):We(e),r=p.dirname(n);h.existsSync(r)||h.mkdirSync(r,{recursive:!0});let s=Et(e,t),o=Qn();if(s==="toml"){let i=h.existsSync(n)?h.readFileSync(n,"utf-8"):"",a=Is(i,"visa-cli",o);h.writeFileSync(n,a)}else{let i={};if(h.existsSync(n))try{i=JSON.parse(h.readFileSync(n,"utf-8"))}catch{i={}}i[e.configKey]=i[e.configKey]||{},i[e.configKey]["visa-cli"]=e.buildEntry?e.buildEntry(o):o,h.writeFileSync(n,JSON.stringify(i,null,2)+`
|
|
108
|
+
`)}return{installed:!0,configPath:n,message:e.postInstallHint}}function Tt(e,t="global"){let n=t==="project"?p.join(process.cwd(),".mcp.json"):We(e);if(!h.existsSync(n))return{removed:!1,configPath:n};if(Et(e,t)==="toml"){let i=h.readFileSync(n,"utf-8");return Pt(i,"visa-cli")?(h.writeFileSync(n,Ls(i,"visa-cli")),{removed:!0,configPath:n}):{removed:!1,configPath:n}}let s;try{s=JSON.parse(h.readFileSync(n,"utf-8"))}catch{return{removed:!1,configPath:n}}let o=s[e.configKey];return!o||!o["visa-cli"]?{removed:!1,configPath:n}:(delete o["visa-cli"],h.writeFileSync(n,JSON.stringify(s,null,2)+`
|
|
109
|
+
`),{removed:!0,configPath:n})}function er(e,t="global"){let n=t==="project"?p.join(process.cwd(),".mcp.json"):We(e);if(!h.existsSync(n))return!1;if(Et(e,t)==="toml")try{let s=h.readFileSync(n,"utf-8");return!!Pt(s,"visa-cli")}catch{return!1}try{return!!JSON.parse(h.readFileSync(n,"utf-8"))?.[e.configKey]?.["visa-cli"]}catch{return!1}}function Os(e){if(!e||typeof e!="object")return;let t=e;if(t.command!=="node"||!Array.isArray(t.args)||t.args.length===0)return;let n=t.args[t.args.length-1];if(!(typeof n!="string"||n.length===0))return n}function Ns(e,t){if(e===t)return!0;let n=p.resolve(e),r=p.resolve(t);if(n===r)return!0;try{let s=h.realpathSync(n),o=h.realpathSync(r);return s===o}catch{return!1}}function Se(){let e=Qn(),t=e.args[e.args.length-1],n=[];for(let r of V){let s=We(r);if(!h.existsSync(s))continue;let o=r.configFormat??"json",i;if(o==="toml")try{let l=h.readFileSync(s,"utf-8");i=Pt(l,"visa-cli")}catch{continue}else{let l;try{l=JSON.parse(h.readFileSync(s,"utf-8"))}catch{continue}i=l?.[r.configKey]?.["visa-cli"]}if(!i)continue;let a=Os(i);if(!a||Ns(a,t))continue;let c=h.existsSync(a)?"mismatch":"missing";n.push({client:r,configPath:s,currentPath:a,expectedPath:t,staleReason:c})}return n}function At(e){return{configPath:ye(e.client,"global").configPath}}var Ds=(0,or.promisify)(sr.execFile);function Us(e,t){return process.stdin.isTTY?new Promise(n=>{let r=rr.createInterface({input:process.stdin,output:process.stdout});r.question(e,s=>{r.close();let o=s.trim().toLowerCase();if(o===""){n(t);return}n(o==="y"||o==="yes")})}):Promise.resolve(t)}function Ms(e){let t=H.homedir(),n=s=>s.replace(t,"~"),r=e.staleReason==="missing"?"path missing on disk":"path mismatch";return` \u2022 ${e.client.displayName} (${n(e.configPath)})
|
|
110
|
+
${r}: ${n(e.currentPath)}`}function ir(e,t){if(e.length===0){console.log(`${t} \u2713 All MCP client configs are up to date.`);return}console.log(`${t} Found ${e.length} stale MCP config ${e.length===1?"entry":"entries"}:`);for(let n of e)console.log(Ms(n))}var E=new tr.Command,Ge=null,ve=!1;function F(){return Ge=new re(()=>C.getSessionToken()),Ge}E.name("visa-cli").description("Visa CLI - AI payment orchestration").version(nt().version);E.hook("preAction",async()=>{await Mt()});E.command("setup").description("Register MCP server, authenticate, and generate attestation key").option("--check","Scan MCP client configs for stale visa-cli entries and exit without making changes").option("--yes","Accept all prompts (enable HUD etc) without asking \u2014 for CI/agent use").option("--no-hud","Skip the HUD install prompt (default: ask)").action(async e=>{try{if(e.check){let d=Se();ir(d,"MCP config check:"),d.length>0&&(console.log("\nRun `visa-cli setup` (or `visa-cli install --repair`) to rewrite these entries."),process.exit(1));return}console.log("Step 1: Registering MCP server...");let t=Se(),n=new Map(t.map(d=>[d.client.id,d])),r=new Set;for(let d of V)if(he(d)){let m=ye(d),v=n.get(d.id),k=v?` \u2014 repaired stale ${v.staleReason} entry`:"";console.log(` \u2713 ${d.displayName} (${m.configPath.replace(H.homedir(),"~")})${k}`),v&&r.add(d.id)}let s=t.filter(d=>!r.has(d.client.id));for(let d of s)At(d),console.log(` \u2713 ${d.client.displayName} (${d.configPath.replace(H.homedir(),"~")}) \u2014 repaired stale ${d.staleReason} entry`);let o=r.size+s.length;console.log(o===0?" \u2713 MCP config verified \u2014 nothing to repair.":` \u2713 Repaired ${o} stale MCP config ${o===1?"entry":"entries"}.`),console.log(`
|
|
111
|
+
Step 2: Checking authentication...`);let i=await C.getSessionToken();if(i)try{await new re(()=>Promise.resolve(i)).getStatus(),console.log(" Already authenticated.")}catch(d){let m=d instanceof Error?d.message:"";m.includes("session has expired")||m.includes("Not logged in")?(console.log(" Existing session expired \u2014 re-authenticating..."),await C.clearAll(),i=null):console.log(` Couldn't verify session (${m||"unknown error"}) \u2014 continuing with existing token.`)}if(i||(console.log(" No session found. Opening browser for GitHub login..."),i=await new Promise(async(d,m)=>{let v=Je.randomBytes(16).toString("hex"),k=`https://auth.visacli.sh/login?state=${v}`;await Ht(k);let L=3e4,$=300*1e3,D=Date.now()+$;for(;Date.now()<D;)try{let U=await globalThis.fetch("https://auth.visacli.sh/v1/auth-status",{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({state:v,timeout:L}),signal:AbortSignal.timeout(L+5e3)});if(!U.ok)continue;let R=await U.json();if(R.status==="pending")continue;if(R.status==="expired"){m(new Error("Session expired. Please run setup again."));return}if(R.status==="complete"&&R.sessionToken){console.log(` Signed in as ${R.user}.`),d(R.sessionToken);return}}catch{}m(new Error("Login timed out after 5 minutes. Please run setup again."))}),await C.saveSessionToken(i),console.log(" Session token saved.")),console.log(`
|
|
112
|
+
Step 3: Setting up authentication...`),!A())console.log(" Not macOS \u2014 skipping biometric setup.");else{try{await Ds("clang",["--version"])}catch{console.error(" Xcode Command Line Tools are required for payment authentication."),console.error(" Install them by running: xcode-select --install"),console.error(" Then re-run: visa-cli setup"),process.exit(1)}try{let d=await Le();console.log(" Attestation key generated."),await F().registerAttestationKey(d),console.log(" Attestation key registered with server.")}catch(d){console.log(` Skipped: ${d.message}`)}}let a=be.join(H.homedir(),".claude","settings.json"),c=nr.existsSync(be.join(H.homedir(),".claude.json")),l=e.hud===!1;if(c&&!l){console.log(`
|
|
113
|
+
Step 4: Enable the Visa spend HUD?`),console.log(" Pins your spend + card info below the Claude Code input, so"),console.log(" you can see your spend, card, and session usage in real time.");let d;if(e.yes?(d=!0,console.log(" (--yes) Enabling Visa HUD.")):process.stdin.isTTY?d=await Us(" Enable Visa HUD? [Y/n] ",!0):(d=!1,console.log(" Non-interactive shell \u2014 skipping prompt. Enable with: visa-cli hud enable")),d){let m=st(a),v=m.installed==="new"||m.installed==="already-visa"||m.installed==="other-hud-present"?" \u2713 ":" Skipped: ";console.log(`${v}${m.message}`);try{let k=F(),L=await k.getStatus(),$={currentVersion:k.getClientVersion(),latestVersion:k.lastSignals?.updateAvailable?.version,updateMessage:k.lastSignals?.updateAvailable?.message,updateCheckDisabled:K()},D=bt(L,$);_t(D),console.log(` Preview: ${D.split(`
|
|
113
114
|
`)[0]}`)}catch{}}else(e.yes===!1||process.stdin.isTTY)&&console.log(" Skipped. Enable any time with: visa-cli hud enable")}let u="\x1B[1m",f="\x1B[0m";console.log(`
|
|
114
115
|
\u2588\u2588\u2557 \u2588\u2588\u2557\u2588\u2588\u2557\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2557 \u2588\u2588\u2588\u2588\u2588\u2557
|
|
115
116
|
\u2588\u2588\u2551 \u2588\u2588\u2551\u2588\u2588\u2551\u2588\u2588\u2554\u2550\u2550\u2550\u2550\u255D\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2557
|
|
@@ -128,20 +129,23 @@ Step 4: Enable the Visa spend HUD?`),console.log(" Pins your spend + card info
|
|
|
128
129
|
${u}Verify:${f} visa-cli status
|
|
129
130
|
${u}HUD:${f} Restart Claude Code to see your spend pinned below the input
|
|
130
131
|
${u}Docs:${f} https://visacli.sh
|
|
131
|
-
`)}catch(t){console.error("Error:",t.message),process.exit(1)}});
|
|
132
|
+
`)}catch(t){console.error("Error:",t.message),process.exit(1)}});E.command("install [client]").description("Register MCP server with an AI client (claude, cursor, windsurf, cline, zed, ...)").option("--all","Install for all detected clients").option("--list","Show supported clients and install status").option("--check","Scan MCP client configs for stale visa-cli entries and exit without making changes").option("--repair","Repair stale MCP client configs without re-running the full setup flow").option("--scope <scope>","Install scope: global or project","global").action(async(e,t)=>{try{if(t.check){let o=Se();ir(o,"MCP config check:"),o.length>0&&process.exit(1);return}if(t.repair){let o=Se();if(o.length===0){console.log("\u2713 MCP config verified \u2014 nothing to repair.");return}for(let i of o)At(i),console.log(` \u2713 ${i.client.displayName} (${i.configPath.replace(H.homedir(),"~")}) \u2014 repaired stale ${i.staleReason} entry`);console.log(`
|
|
132
133
|
Repaired ${o.length} stale MCP config ${o.length===1?"entry":"entries"}.`);return}if(t.list){console.log(`
|
|
133
134
|
\x1B[1mSupported MCP Clients\x1B[0m
|
|
134
|
-
`),console.log(` ${"Client".padEnd(18)} ${"Detected".padEnd(10)} ${"Installed".padEnd(11)} Config Path`),console.log(` ${"\u2500".repeat(18)} ${"\u2500".repeat(10)} ${"\u2500".repeat(11)} ${"\u2500".repeat(40)}`);for(let a of V){let c=
|
|
135
|
-
Supported clients: ${V.map(o=>o.id).join(", ")}`),process.exit(1));let r=
|
|
136
|
-
Supported clients: ${V.map(o=>o.id).join(", ")}`),process.exit(1));let r=
|
|
137
|
-
`);let a=(await s.getStatus()).attestationRequired!==!1;a&&!A()&&console.warn("Warning: Touch ID unavailable on this system \u2014 payment will proceed without biometric attestation.");let c=
|
|
138
|
-
Result URLs:`);for(let u of l.receipt.urls)console.log(` ${u}`)}}else console.error(`Payment failed: ${l.message||"Unknown error"}`),process.exit(1)}catch(n){n instanceof I?console.error(`Error: ${n.message}`):console.error("Error:",n.message),process.exit(1)}});
|
|
139
|
-
`)}return{transactions:[]}});process.stdout.write(
|
|
140
|
-
`);try{await F().logout(),console.log(" Server session invalidated.")}catch{console.log(" Server logout skipped (no active session).")}if(await C.clearAll(),console.log(" Keychain credentials cleared."),A())try{await
|
|
141
|
-
Reset complete.`)}catch(e){console.error("Error:",e.message),process.exit(1)}});
|
|
135
|
+
`),console.log(` ${"Client".padEnd(18)} ${"Detected".padEnd(10)} ${"Installed".padEnd(11)} Config Path`),console.log(` ${"\u2500".repeat(18)} ${"\u2500".repeat(10)} ${"\u2500".repeat(11)} ${"\u2500".repeat(40)}`);for(let a of V){let c=he(a),l=er(a),u=c?"Yes":"No",f=l?"Yes":"No",d=a.globalConfigPath.replace(H.homedir(),"~");console.log(` ${a.displayName.padEnd(18)} ${u.padEnd(10)} ${f.padEnd(11)} ${d}`)}console.log("");return}let n=t.scope==="project"?"project":"global";if(t.all){let o=[],i=[];for(let a of V){if(!he(a)){i.push(a.displayName);continue}ye(a,n),o.push(a.displayName)}o.length>0&&console.log(`Installed for: ${o.join(", ")}.`),i.length>0&&console.log(`Skipped: ${i.map(a=>`${a} (not detected)`).join(", ")}.`),o.length===0&&i.length===0&&console.log("No supported clients found.");return}e||(console.error("Usage: visa-cli install <client>"),console.error(" visa-cli install --all"),console.error(" visa-cli install --list"),console.error(`
|
|
136
|
+
Supported clients: ${V.map(o=>o.id).join(", ")}`),process.exit(1));let r=xt(e);r||(console.error(`Unknown client: ${e}`),console.error(`Supported clients: ${V.map(o=>o.id).join(", ")}`),process.exit(1)),n==="global"&&!he(r)&&(console.error(`${r.displayName} not detected on this machine.`),console.error(`Expected: ${r.detectPaths.join(", ")}`),process.exit(1));let s=ye(r,n);console.log(`Registered visa-cli MCP server in ${s.configPath}`),console.log(s.message)}catch(n){console.error("Error:",n.message),process.exit(1)}});E.command("uninstall [client]").description("Remove MCP server from an AI client").option("--all","Remove from all clients").option("--scope <scope>","Uninstall scope: global or project","global").action(async(e,t)=>{try{let n=t.scope==="project"?"project":"global";if(t.all){let o=[];for(let i of V)Tt(i,n).removed&&o.push(i.displayName);o.length>0?console.log(`Removed visa-cli from: ${o.join(", ")}.`):console.log("visa-cli was not installed in any client.");return}e||(console.error("Usage: visa-cli uninstall <client>"),console.error(" visa-cli uninstall --all"),console.error(`
|
|
137
|
+
Supported clients: ${V.map(o=>o.id).join(", ")}`),process.exit(1));let r=xt(e);r||(console.error(`Unknown client: ${e}`),console.error(`Supported clients: ${V.map(o=>o.id).join(", ")}`),process.exit(1));let s=Tt(r,n);s.removed?console.log(`Removed visa-cli from ${s.configPath}`):console.log(`visa-cli was not installed for ${r.displayName}.`)}catch(n){console.error("Error:",n.message),process.exit(1)}});E.command("pay <url>").description("Pay a merchant URL (amount auto-detected from HTTP 402 response)").option("-m, --method <method>","HTTP method (GET or POST)","GET").option("-b, --body <json>","JSON request body for POST endpoints").action(async(e,t)=>{try{Kn(e);let n=qn(t.method),r=Wn(t.body),s=new re(()=>C.getSessionToken());console.log(`Checking payment for ${e}...`);let o=Gn(await s.paymentPreview({url:e}));console.log(` Merchant: ${o.merchantName}`),console.log(` Amount: $${o.amount.toFixed(2)} ${o.currency}`),console.log(` Rail: auto-detected
|
|
138
|
+
`);let a=(await s.getStatus()).attestationRequired!==!1;a&&!A()&&console.warn("Warning: Touch ID unavailable on this system \u2014 payment will proceed without biometric attestation.");let c=Je.randomUUID(),l=await Ct(s,async()=>{let u;if(a&&A())try{let{nonce:f}=await s.getAttestationChallenge(),d=Buffer.from(JSON.stringify({nonce:f,amount:o.amount,merchant:o.merchantName,context:e})).toString("base64");u={signature:await pt(d,`pay $${o.amount.toFixed(2)} to ${o.merchantName}`),nonce:f,amount:o.amount,merchant:o.merchantName}}catch(f){throw new Error(`Touch ID confirmation failed: ${f?.message||"user cancelled or biometric error"}`)}return s.pay({url:e,method:n,body:r,attestation:u,idempotencyKey:c})});if(l.success){if(console.log(`Payment complete: $${(l.amount??o.amount).toFixed(2)} \u2192 ${l.merchantName??o.merchantName}`),l.receipt?.urls?.length){console.log(`
|
|
139
|
+
Result URLs:`);for(let u of l.receipt.urls)console.log(` ${u}`)}}else console.error(`Payment failed: ${l.message||"Unknown error"}`),process.exit(1)}catch(n){n instanceof I?console.error(`Error: ${n.message}`):console.error("Error:",n.message),process.exit(1)}});E.command("status").description("Check enrollment, cards, wallet, and spending controls").action(async()=>{ve=!1;try{let e=F(),t=await e.getStatus(),n={currentVersion:e.getClientVersion(),latestVersion:e.lastSignals.updateAvailable?.version,updateMessage:e.lastSignals.updateAvailable?.message,updateCheckDisabled:K()},r=await e.getTransactions().catch(s=>{if(process.env.VISA_CLI_DEBUG){let o=s instanceof Error?s.message:String(s);process.stderr.write(`[visa-cli] getTransactions failed (HUD will omit): ${o}
|
|
140
|
+
`)}return{transactions:[]}});process.stdout.write(Sn(t,Array.isArray(r?.transactions)?r.transactions:[],A(),n)),ve=!0}catch(e){ve=!1,console.error("Error:",e.message),process.exit(1)}});E.command("reset").description("Log out and clear all credentials").action(async()=>{try{console.log(`Resetting Visa CLI...
|
|
141
|
+
`);try{await F().logout(),console.log(" Server session invalidated.")}catch{console.log(" Server logout skipped (no active session).")}if(await C.clearAll(),console.log(" Keychain credentials cleared."),A())try{await mn(),console.log(" Secure Enclave key deleted.")}catch{console.log(" No Secure Enclave key to delete.")}console.log(`
|
|
142
|
+
Reset complete.`)}catch(e){console.error("Error:",e.message),process.exit(1)}});E.command("feedback").description("Submit feedback about Visa CLI").argument("[message]","Your feedback message").action(async e=>{(!e||e.trim().length===0)&&(console.log('Usage: visa-cli feedback "your message"'),process.exit(1));try{await C.getSessionToken()||(console.error("Not logged in. Run visa-cli setup first."),process.exit(1)),await F().feedback(e.trim()),console.log("Feedback submitted. Thanks!")}catch(t){console.error("Error:",t.message),process.exit(1)}});var It=E.command("config").description("Inspect the CLI configuration");It.command("list").description("Show resolved config values (env vars, server state, defaults) with their source").option("--json","Output as JSON for scripting / agent consumption").option("--dev","Include developer/test hooks (VISA_MOCK_*, VISA_CLI_DEBUG)").option("--verbose","Show one-line hints beneath entries that have them").action(async e=>{try{let t=F(),n=await Hn({api:t,includeDev:!!e.dev});if(e.json){console.log(Fn(n));return}n.statusError&&(console.error(`Warning: could not reach auth server (${n.statusError}).`),console.error("Server-sourced entries show '\u2014'. Log in with `visa-cli setup` if you expected values here."),console.error("")),console.log(Vn(n.entries,{verbose:!!e.verbose}))}catch(t){console.error("Error:",t.message),process.exit(1)}});function js(){return Object.entries(ge).map(([e,t])=>` ${e.padEnd(22)} (${t.type}) ${t.description}`).join(`
|
|
143
|
+
`)}It.command("set <key> <value>").description("Persist a CLI setting to ~/.visa-mcp/settings.json").addHelpText("after",`
|
|
144
|
+
Settable keys:
|
|
145
|
+
${js()}`).action((e,t)=>{try{let n=Un(e,t);console.log(`Saved ${n.key}=${JSON.stringify(n.value)} \u2192 ${n.path}`),n.requiresRestart&&console.log("Restart Claude Code (or your MCP-enabled client) for the change to take effect.")}catch(n){n instanceof ae||n instanceof ce?console.error(`Error: ${n.message}`):console.error(`Error: ${n.message}`),process.exit(1)}});It.command("unset <key>").description("Remove a CLI setting from ~/.visa-mcp/settings.json (falls back to env var or default)").action(e=>{try{let t=Mn(e);t.removed?(console.log(`Removed ${t.key} from ${t.path}`),t.requiresRestart&&console.log("Restart Claude Code (or your MCP-enabled client) for the change to take effect.")):console.log(`No-op: ${t.key} was not set in ${t.path}.`)}catch(t){console.error(`Error: ${t.message}`),process.exit(1)}});var Lt=E.command("biometric").description("Manage Touch ID / biometric attestation enforcement");Lt.command("status").description("Show current biometric enforcement state").action(async()=>{try{await C.getSessionToken()||(console.error("Not logged in. Run visa-cli setup first."),process.exit(1));let n=await F().getStatus(),r=n.attestationRequired!==!1,s=!!n.hasAttestationKey,o=A();console.log(`Server policy: Touch ID ${r?"REQUIRED":"NOT required"}`),console.log(`Attestation key registered: ${s?"yes":"no"}`),console.log(`Touch ID available on this device: ${o?"yes":"no"}`)}catch(e){console.error("Error:",e.message),process.exit(1)}});Lt.command("on").description("Require Touch ID for payments (security upgrade \u2014 no Touch ID needed)").action(async()=>{try{await C.getSessionToken()||(console.error("Not logged in. Run visa-cli setup first."),process.exit(1));let t=F(),n=await t.setBiometricPreference({required:!0});n.success||(console.error(`Failed: ${n.error||"unknown error"}`),process.exit(1)),console.log("Touch ID is now REQUIRED for payments."),(await t.getStatus()).hasAttestationKey||console.warn("Note: no attestation key is registered yet. Run `visa-cli setup` to enroll Touch ID.")}catch(e){console.error("Error:",e.message),process.exit(1)}});Lt.command("off").description("Disable Touch ID requirement (security downgrade \u2014 one Touch ID confirmation required)").action(async()=>{try{await C.getSessionToken()||(console.error("Not logged in. Run visa-cli setup first."),process.exit(1));let t=F(),n=await t.getStatus();if(n.attestationRequired===!1){console.log("Touch ID is already disabled.");return}n.hasAttestationKey&&!A()&&(console.error("Touch ID is unavailable on this device but the server has a registered key."),console.error("Disable Touch ID from a device that can sign, or contact support."),process.exit(1));let r=await Ct(t,async()=>{let s;if(n.hasAttestationKey&&A())try{let{nonce:o}=await t.getAttestationChallenge(),i=0,a="",c=Buffer.from(JSON.stringify({nonce:o,amount:i,merchant:a,context:"biometric-preference"})).toString("base64");s={signature:await pt(c,"disable Touch ID requirement"),nonce:o,amount:i,merchant:a}}catch(o){throw new Error(`Touch ID confirmation failed: ${o?.message||"cancelled"}`)}return t.setBiometricPreference({required:!1,attestation:s})});r.success||(console.error(`Failed: ${r.error||"unknown error"}`),process.exit(1)),console.log("Touch ID is no longer required for payments.")}catch(e){console.error("Error:",e.message),process.exit(1)}});var ne=E.command("shell-hud").description("Manage the persistent Visa HUD shown in your shell prompt");function ar(){let e=kn();e.installed||(console.error(e.message),process.exit(1)),console.log(e.message)}function cr(){let e=$n();e.removed||(console.error(e.message),process.exit(1)),console.log(e.message)}ne.command("install").description("Install the persistent shell HUD into your zsh or bash rc file").action(ar);ne.command("enable").description("Enable the persistent shell HUD").action(ar);ne.command("uninstall").description("Remove the persistent shell HUD from your shell rc file").action(cr);ne.command("disable").description("Disable the persistent shell HUD").action(cr);ne.command("segment").description("Print the cached shell HUD segment").action(()=>{process.stdout.write(`${wt()}
|
|
142
146
|
`)});ne.command("doctor").description("Diagnose shell HUD installation and connectivity").action(async()=>{let{existsSync:e,readFileSync:t}=await import("fs"),n=!0,r=(f,d,m)=>{console.log(` ${d?"\u2713":"\u2717"} ${f}: ${m}`),d||(n=!1)};console.log(`Shell HUD Doctor
|
|
143
|
-
`);let s=
|
|
147
|
+
`);let s=He();if(r("Shell detected",!!s,s??"none (zsh, bash, or PowerShell required)"),s){let f=Ve(s),d=e(f);if(r("RC file exists",d,f.replace(H.homedir(),"~")),d){let v=t(f,"utf-8").includes("visa-cli shell hud v2");r("HUD block installed",v,v?"found in rc file":"missing \u2014 run: visa-cli shell-hud install")}}let o=fe(),i=e(o);if(r("Cache file",i,i?o.replace(H.homedir(),"~"):"missing \u2014 HUD has not refreshed yet"),i)try{let f=JSON.parse(t(o,"utf-8")),d=Date.now()-(f.renderedAt??0),m=Math.round(d/1e3),v=m<=30;console.log(` ${v?"\u2713":"\u26A0"} Cache freshness: ${m}s old${v?"":" (stale \u2014 will refresh on next prompt)"}`)}catch{r("Cache readable",!1,"corrupt JSON")}let a=je(),c=e(a),l="missing";if(c)try{l=t(a,"utf-8").trim().slice(0,80)}catch{l="unreadable"}r("Line file",c,l);let u=!1;try{u=!!await C.getSessionToken()}catch{}if(r("Auth token",u,u?"found in keychain":"missing \u2014 run: visa-cli setup"),u)try{await F().getStatus(),r("API connectivity",!0,"GET /v1/status OK")}catch(f){let d=f instanceof Error?f.message:"unknown error";r("API connectivity",!1,d)}console.log(n?`
|
|
144
148
|
All checks passed.`:`
|
|
145
|
-
Some checks failed \u2014 see above.`),n||process.exit(1)});ne.command("refresh").description("Refresh the shell HUD cache (no-op if cache is fresh unless --force)").option("--force","Bypass the cache freshness gate and always refresh").action(async e=>{if(!(!e.force&&
|
|
146
|
-
`)[0],r=await
|
|
147
|
-
`)});var
|
|
149
|
+
Some checks failed \u2014 see above.`),n||process.exit(1)});ne.command("refresh").description("Refresh the shell HUD cache (no-op if cache is fresh unless --force)").option("--force","Bypass the cache freshness gate and always refresh").action(async e=>{if(!(!e.force&&xn()))try{let t=F(),n=await t.getStatus(),r={currentVersion:t.getClientVersion(),latestVersion:t.lastSignals.updateAvailable?.version,updateMessage:t.lastSignals.updateAvailable?.message,updateCheckDisabled:K()},s=bt(n,r);_t(s)}catch{}finally{de()}});E.command("statusline").description("Output the multi-line Visa HUD for Claude Code statusLine integration").action(async()=>{let e=await Xt(),n=wt().split(`
|
|
150
|
+
`)[0],r=await Qt(n,e);process.stdout.write(`${r}
|
|
151
|
+
`)});var lr=E.command("hud").description("Manage the Visa HUD pinned below Claude Code input");lr.command("enable").description("Register Visa HUD as the Claude Code statusLine").action(()=>{let e=be.join(H.homedir(),".claude","settings.json"),t=st(e);console.log(t.message),(t.installed==="error"||t.installed==="malformed-json")&&process.exit(1)});lr.command("disable").description("Remove Visa HUD from Claude Code statusLine (leaves other tools untouched)").action(()=>{let e=be.join(H.homedir(),".claude","settings.json"),t=Wt(e);console.log(t.message),t.removed||process.exit(1)});E.hook("postAction",()=>{if(ve){ve=!1;return}Ge&&hn(Ge.lastSignals)});E.parse();
|