@getverbal/cli 0.1.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 +21 -0
- package/README.md +36 -0
- package/dist/auth/browser-auth.d.ts +6 -0
- package/dist/auth/browser-auth.d.ts.map +1 -0
- package/dist/auth/browser-auth.js +202 -0
- package/dist/auth/browser-auth.js.map +1 -0
- package/dist/auth/credentials.d.ts +6 -0
- package/dist/auth/credentials.d.ts.map +1 -0
- package/dist/auth/credentials.js +78 -0
- package/dist/auth/credentials.js.map +1 -0
- package/dist/cli.d.ts +3 -0
- package/dist/cli.d.ts.map +1 -0
- package/dist/cli.js +66 -0
- package/dist/cli.js.map +1 -0
- package/dist/commands/init.d.ts +2 -0
- package/dist/commands/init.d.ts.map +1 -0
- package/dist/commands/init.js +188 -0
- package/dist/commands/init.js.map +1 -0
- package/dist/commands/logout.d.ts +2 -0
- package/dist/commands/logout.d.ts.map +1 -0
- package/dist/commands/logout.js +17 -0
- package/dist/commands/logout.js.map +1 -0
- package/dist/commands/status.d.ts +2 -0
- package/dist/commands/status.d.ts.map +1 -0
- package/dist/commands/status.js +43 -0
- package/dist/commands/status.js.map +1 -0
- package/dist/commands/uninstall.d.ts +2 -0
- package/dist/commands/uninstall.d.ts.map +1 -0
- package/dist/commands/uninstall.js +43 -0
- package/dist/commands/uninstall.js.map +1 -0
- package/dist/configure/claude-code.d.ts +7 -0
- package/dist/configure/claude-code.d.ts.map +1 -0
- package/dist/configure/claude-code.js +11 -0
- package/dist/configure/claude-code.js.map +1 -0
- package/dist/configure/claude-desktop.d.ts +8 -0
- package/dist/configure/claude-desktop.d.ts.map +1 -0
- package/dist/configure/claude-desktop.js +28 -0
- package/dist/configure/claude-desktop.js.map +1 -0
- package/dist/configure/codex.d.ts +7 -0
- package/dist/configure/codex.d.ts.map +1 -0
- package/dist/configure/codex.js +12 -0
- package/dist/configure/codex.js.map +1 -0
- package/dist/configure/cursor.d.ts +7 -0
- package/dist/configure/cursor.d.ts.map +1 -0
- package/dist/configure/cursor.js +12 -0
- package/dist/configure/cursor.js.map +1 -0
- package/dist/configure/index.d.ts +30 -0
- package/dist/configure/index.d.ts.map +1 -0
- package/dist/configure/index.js +149 -0
- package/dist/configure/index.js.map +1 -0
- package/dist/detect/claude-code.d.ts +3 -0
- package/dist/detect/claude-code.d.ts.map +1 -0
- package/dist/detect/claude-code.js +82 -0
- package/dist/detect/claude-code.js.map +1 -0
- package/dist/detect/claude-desktop.d.ts +3 -0
- package/dist/detect/claude-desktop.d.ts.map +1 -0
- package/dist/detect/claude-desktop.js +89 -0
- package/dist/detect/claude-desktop.js.map +1 -0
- package/dist/detect/codex.d.ts +3 -0
- package/dist/detect/codex.d.ts.map +1 -0
- package/dist/detect/codex.js +64 -0
- package/dist/detect/codex.js.map +1 -0
- package/dist/detect/cursor.d.ts +3 -0
- package/dist/detect/cursor.d.ts.map +1 -0
- package/dist/detect/cursor.js +81 -0
- package/dist/detect/cursor.js.map +1 -0
- package/dist/detect/index.d.ts +3 -0
- package/dist/detect/index.d.ts.map +1 -0
- package/dist/detect/index.js +28 -0
- package/dist/detect/index.js.map +1 -0
- package/dist/types.d.ts +18 -0
- package/dist/types.d.ts.map +1 -0
- package/dist/types.js +2 -0
- package/dist/types.js.map +1 -0
- package/dist/verify.d.ts +7 -0
- package/dist/verify.d.ts.map +1 -0
- package/dist/verify.js +40 -0
- package/dist/verify.js.map +1 -0
- package/package.json +42 -0
package/LICENSE
ADDED
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
MIT License
|
|
2
|
+
|
|
3
|
+
Copyright (c) 2025 Verbal Inc.
|
|
4
|
+
|
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
6
|
+
of this software and associated documentation files (the "Software"), to deal
|
|
7
|
+
in the Software without restriction, including without limitation the rights
|
|
8
|
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
9
|
+
copies of the Software, and to permit persons to whom the Software is
|
|
10
|
+
furnished to do so, subject to the following conditions:
|
|
11
|
+
|
|
12
|
+
The above copyright notice and this permission notice shall be included in all
|
|
13
|
+
copies or substantial portions of the Software.
|
|
14
|
+
|
|
15
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
16
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
17
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
18
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
19
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
20
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
21
|
+
SOFTWARE.
|
package/README.md
ADDED
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
# getverbal
|
|
2
|
+
|
|
3
|
+
Set up Verbal AI spend tracking in one command.
|
|
4
|
+
|
|
5
|
+
## Quick Start
|
|
6
|
+
|
|
7
|
+
```sh
|
|
8
|
+
npx getverbal init
|
|
9
|
+
```
|
|
10
|
+
|
|
11
|
+
That's it. Verbal auto-detects your AI tools, authenticates via browser, and configures the MCP server for usage tracking.
|
|
12
|
+
|
|
13
|
+
## What It Does
|
|
14
|
+
|
|
15
|
+
`getverbal init` scans your machine for supported AI tools — Claude Code, Claude Desktop, Cursor, and Codex — then authenticates your Verbal account via browser and injects the Verbal MCP server configuration into each detected tool. After setup, every LLM call is tracked automatically.
|
|
16
|
+
|
|
17
|
+
## Commands
|
|
18
|
+
|
|
19
|
+
| Command | Description |
|
|
20
|
+
|-------------|--------------------------------------------------------|
|
|
21
|
+
| `init` | Authenticate and configure all detected AI tools |
|
|
22
|
+
| `status` | Show current configuration and connection status |
|
|
23
|
+
| `logout` | Remove stored credentials |
|
|
24
|
+
| `uninstall` | Remove Verbal from all AI tools and delete credentials |
|
|
25
|
+
|
|
26
|
+
## Requirements
|
|
27
|
+
|
|
28
|
+
- Node.js >= 18
|
|
29
|
+
|
|
30
|
+
## More Info
|
|
31
|
+
|
|
32
|
+
[https://getverbal.ai](https://getverbal.ai)
|
|
33
|
+
|
|
34
|
+
## License
|
|
35
|
+
|
|
36
|
+
MIT
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"browser-auth.d.ts","sourceRoot":"","sources":["../../src/auth/browser-auth.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,aAAa,CAAC;AAE/C,MAAM,WAAW,kBAAkB;IACjC,MAAM,EAAE,MAAM,CAAC;CAChB;AA4HD,wBAAsB,WAAW,CAAC,OAAO,EAAE,kBAAkB,GAAG,OAAO,CAAC,WAAW,CAAC,CA2GnF"}
|
|
@@ -0,0 +1,202 @@
|
|
|
1
|
+
import { createServer } from 'node:http';
|
|
2
|
+
import { randomBytes } from 'node:crypto';
|
|
3
|
+
const PORT_START = 9876;
|
|
4
|
+
const PORT_END = 9900;
|
|
5
|
+
const TIMEOUT_MS = 120_000;
|
|
6
|
+
async function openBrowser(url) {
|
|
7
|
+
const { exec } = await import('node:child_process');
|
|
8
|
+
const platform = process.platform;
|
|
9
|
+
const cmd = platform === 'darwin' ? 'open' : platform === 'win32' ? 'start' : 'xdg-open';
|
|
10
|
+
exec(`${cmd} "${url}"`);
|
|
11
|
+
}
|
|
12
|
+
async function findAvailablePort() {
|
|
13
|
+
for (let port = PORT_START; port <= PORT_END; port++) {
|
|
14
|
+
const available = await new Promise((resolve) => {
|
|
15
|
+
const probe = createServer();
|
|
16
|
+
probe.once('error', (err) => {
|
|
17
|
+
if (err.code === 'EADDRINUSE') {
|
|
18
|
+
resolve(false);
|
|
19
|
+
}
|
|
20
|
+
else {
|
|
21
|
+
resolve(false);
|
|
22
|
+
}
|
|
23
|
+
});
|
|
24
|
+
probe.once('listening', () => {
|
|
25
|
+
probe.close(() => resolve(true));
|
|
26
|
+
});
|
|
27
|
+
probe.listen(port, '127.0.0.1');
|
|
28
|
+
});
|
|
29
|
+
if (available)
|
|
30
|
+
return port;
|
|
31
|
+
}
|
|
32
|
+
throw new Error(`No available port found in range ${PORT_START}-${PORT_END}. Please free up a port and try again.`);
|
|
33
|
+
}
|
|
34
|
+
async function exchangeCode(apiUrl, code) {
|
|
35
|
+
const res = await fetch(`${apiUrl}/api/v1/cli/exchange`, {
|
|
36
|
+
method: 'POST',
|
|
37
|
+
headers: { 'Content-Type': 'application/json' },
|
|
38
|
+
body: JSON.stringify({ code }),
|
|
39
|
+
});
|
|
40
|
+
if (!res.ok) {
|
|
41
|
+
const body = await res.json().catch(() => ({}));
|
|
42
|
+
throw new Error(body.error ?? `Exchange failed: HTTP ${res.status}`);
|
|
43
|
+
}
|
|
44
|
+
const data = await res.json();
|
|
45
|
+
return {
|
|
46
|
+
api_key: data.api_key,
|
|
47
|
+
org_id: data.org_id,
|
|
48
|
+
user_email: data.user_email,
|
|
49
|
+
api_url: data.api_url,
|
|
50
|
+
ingest_url: data.ingest_url,
|
|
51
|
+
created_at: new Date().toISOString(),
|
|
52
|
+
};
|
|
53
|
+
}
|
|
54
|
+
const WAITING_HTML = `<!DOCTYPE html>
|
|
55
|
+
<html lang="en">
|
|
56
|
+
<head>
|
|
57
|
+
<meta charset="UTF-8">
|
|
58
|
+
<title>Verbal CLI Authentication</title>
|
|
59
|
+
<style>
|
|
60
|
+
body { font-family: system-ui, sans-serif; display: flex; align-items: center; justify-content: center; min-height: 100vh; margin: 0; background: #f5f5f5; }
|
|
61
|
+
.card { background: white; padding: 2rem; border-radius: 8px; box-shadow: 0 2px 8px rgba(0,0,0,0.1); text-align: center; max-width: 400px; }
|
|
62
|
+
h1 { font-size: 1.25rem; color: #333; margin-bottom: 0.5rem; }
|
|
63
|
+
p { color: #666; font-size: 0.95rem; }
|
|
64
|
+
</style>
|
|
65
|
+
</head>
|
|
66
|
+
<body>
|
|
67
|
+
<div class="card">
|
|
68
|
+
<h1>Waiting for authentication...</h1>
|
|
69
|
+
<p>Please complete the sign-in in the browser window that was opened.</p>
|
|
70
|
+
</div>
|
|
71
|
+
</body>
|
|
72
|
+
</html>`;
|
|
73
|
+
const SUCCESS_HTML = `<!DOCTYPE html>
|
|
74
|
+
<html lang="en">
|
|
75
|
+
<head>
|
|
76
|
+
<meta charset="UTF-8">
|
|
77
|
+
<title>Verbal CLI — Authenticated</title>
|
|
78
|
+
<style>
|
|
79
|
+
body { font-family: system-ui, sans-serif; display: flex; align-items: center; justify-content: center; min-height: 100vh; margin: 0; background: #f5f5f5; }
|
|
80
|
+
.card { background: white; padding: 2rem; border-radius: 8px; box-shadow: 0 2px 8px rgba(0,0,0,0.1); text-align: center; max-width: 400px; }
|
|
81
|
+
h1 { font-size: 1.25rem; color: #22c55e; margin-bottom: 0.5rem; }
|
|
82
|
+
p { color: #666; font-size: 0.95rem; }
|
|
83
|
+
</style>
|
|
84
|
+
</head>
|
|
85
|
+
<body>
|
|
86
|
+
<div class="card">
|
|
87
|
+
<h1>Authentication complete!</h1>
|
|
88
|
+
<p>You can close this tab and return to your terminal.</p>
|
|
89
|
+
</div>
|
|
90
|
+
</body>
|
|
91
|
+
</html>`;
|
|
92
|
+
const FAILURE_HTML = `<!DOCTYPE html>
|
|
93
|
+
<html lang="en">
|
|
94
|
+
<head>
|
|
95
|
+
<meta charset="UTF-8">
|
|
96
|
+
<title>Verbal CLI — Authentication Failed</title>
|
|
97
|
+
<style>
|
|
98
|
+
body { font-family: system-ui, sans-serif; display: flex; align-items: center; justify-content: center; min-height: 100vh; margin: 0; background: #f5f5f5; }
|
|
99
|
+
.card { background: white; padding: 2rem; border-radius: 8px; box-shadow: 0 2px 8px rgba(0,0,0,0.1); text-align: center; max-width: 400px; }
|
|
100
|
+
h1 { font-size: 1.25rem; color: #ef4444; margin-bottom: 0.5rem; }
|
|
101
|
+
p { color: #666; font-size: 0.95rem; }
|
|
102
|
+
</style>
|
|
103
|
+
</head>
|
|
104
|
+
<body>
|
|
105
|
+
<div class="card">
|
|
106
|
+
<h1>Authentication failed.</h1>
|
|
107
|
+
<p>Please run <code>npx getverbal init</code> again.</p>
|
|
108
|
+
</div>
|
|
109
|
+
</body>
|
|
110
|
+
</html>`;
|
|
111
|
+
export async function browserAuth(options) {
|
|
112
|
+
const { apiUrl } = options;
|
|
113
|
+
// Step 1: Generate a 256-bit random state parameter for CSRF protection
|
|
114
|
+
const state = randomBytes(32).toString('hex');
|
|
115
|
+
// Step 2: Find an available port
|
|
116
|
+
const port = await findAvailablePort();
|
|
117
|
+
// Step 3 & 5-9: Start HTTP server and wait for callback
|
|
118
|
+
return new Promise((resolve, reject) => {
|
|
119
|
+
let settled = false;
|
|
120
|
+
let timeoutId;
|
|
121
|
+
const server = createServer((req, res) => {
|
|
122
|
+
const url = new URL(req.url ?? '/', `http://127.0.0.1:${port}`);
|
|
123
|
+
if (req.method !== 'GET') {
|
|
124
|
+
res.writeHead(405, { 'Content-Type': 'text/plain' });
|
|
125
|
+
res.end('Method Not Allowed');
|
|
126
|
+
return;
|
|
127
|
+
}
|
|
128
|
+
if (url.pathname === '/') {
|
|
129
|
+
res.writeHead(200, { 'Content-Type': 'text/html; charset=utf-8' });
|
|
130
|
+
res.end(WAITING_HTML);
|
|
131
|
+
return;
|
|
132
|
+
}
|
|
133
|
+
if (url.pathname === '/callback') {
|
|
134
|
+
const code = url.searchParams.get('code');
|
|
135
|
+
const receivedState = url.searchParams.get('state');
|
|
136
|
+
// Step 6: Validate state (CSRF protection)
|
|
137
|
+
if (!code || receivedState !== state) {
|
|
138
|
+
res.writeHead(400, { 'Content-Type': 'text/html; charset=utf-8' });
|
|
139
|
+
res.end(FAILURE_HTML);
|
|
140
|
+
if (!settled) {
|
|
141
|
+
settled = true;
|
|
142
|
+
if (timeoutId !== undefined)
|
|
143
|
+
clearTimeout(timeoutId);
|
|
144
|
+
server.close(() => {
|
|
145
|
+
reject(new Error('Authentication failed: invalid state or missing code.'));
|
|
146
|
+
});
|
|
147
|
+
}
|
|
148
|
+
return;
|
|
149
|
+
}
|
|
150
|
+
// State is valid — send success response immediately so the browser gets feedback
|
|
151
|
+
res.writeHead(200, { 'Content-Type': 'text/html; charset=utf-8' });
|
|
152
|
+
res.end(SUCCESS_HTML);
|
|
153
|
+
if (!settled) {
|
|
154
|
+
settled = true;
|
|
155
|
+
if (timeoutId !== undefined)
|
|
156
|
+
clearTimeout(timeoutId);
|
|
157
|
+
// Step 7: Exchange code for credentials, then close server
|
|
158
|
+
exchangeCode(apiUrl, code)
|
|
159
|
+
.then((credentials) => {
|
|
160
|
+
server.close(() => resolve(credentials));
|
|
161
|
+
})
|
|
162
|
+
.catch((err) => {
|
|
163
|
+
server.close(() => {
|
|
164
|
+
const message = err instanceof Error ? err.message : String(err);
|
|
165
|
+
reject(new Error(`Code exchange failed: ${message}`));
|
|
166
|
+
});
|
|
167
|
+
});
|
|
168
|
+
}
|
|
169
|
+
return;
|
|
170
|
+
}
|
|
171
|
+
// All other routes — 404
|
|
172
|
+
res.writeHead(404, { 'Content-Type': 'text/plain' });
|
|
173
|
+
res.end('Not Found');
|
|
174
|
+
});
|
|
175
|
+
server.on('error', (err) => {
|
|
176
|
+
if (!settled) {
|
|
177
|
+
settled = true;
|
|
178
|
+
if (timeoutId !== undefined)
|
|
179
|
+
clearTimeout(timeoutId);
|
|
180
|
+
reject(new Error(`Server error: ${err.message}`));
|
|
181
|
+
}
|
|
182
|
+
});
|
|
183
|
+
// Step 3: Start server on 127.0.0.1 only
|
|
184
|
+
server.listen(port, '127.0.0.1', () => {
|
|
185
|
+
// Step 8 (timeout): Close server if no callback within 120 seconds
|
|
186
|
+
timeoutId = setTimeout(() => {
|
|
187
|
+
if (!settled) {
|
|
188
|
+
settled = true;
|
|
189
|
+
server.close(() => {
|
|
190
|
+
reject(new Error('Authentication timed out after 120 seconds. Please run `npx getverbal init` again.'));
|
|
191
|
+
});
|
|
192
|
+
}
|
|
193
|
+
}, TIMEOUT_MS);
|
|
194
|
+
// Step 4: Open browser
|
|
195
|
+
const authUrl = `${apiUrl}/cli/auth?state=${state}&port=${port}`;
|
|
196
|
+
openBrowser(authUrl).catch(() => {
|
|
197
|
+
// Non-fatal: user may open browser manually
|
|
198
|
+
});
|
|
199
|
+
});
|
|
200
|
+
});
|
|
201
|
+
}
|
|
202
|
+
//# sourceMappingURL=browser-auth.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"browser-auth.js","sourceRoot":"","sources":["../../src/auth/browser-auth.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
import type { Credentials } from '../types.js';
|
|
2
|
+
export declare function getCredentialsDir(): string;
|
|
3
|
+
export declare function readCredentials(): Promise<Credentials | null>;
|
|
4
|
+
export declare function writeCredentials(credentials: Credentials): Promise<void>;
|
|
5
|
+
export declare function clearCredentials(): Promise<void>;
|
|
6
|
+
//# sourceMappingURL=credentials.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"credentials.d.ts","sourceRoot":"","sources":["../../src/auth/credentials.ts"],"names":[],"mappings":"AAIA,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,aAAa,CAAC;AAK/C,wBAAgB,iBAAiB,IAAI,MAAM,CAM1C;AAMD,wBAAsB,eAAe,IAAI,OAAO,CAAC,WAAW,GAAG,IAAI,CAAC,CAkBnE;AAED,wBAAsB,gBAAgB,CAAC,WAAW,EAAE,WAAW,GAAG,OAAO,CAAC,IAAI,CAAC,CA4B9E;AAED,wBAAsB,gBAAgB,IAAI,OAAO,CAAC,IAAI,CAAC,CAYtD"}
|
|
@@ -0,0 +1,78 @@
|
|
|
1
|
+
import { readFileSync, writeFileSync, rmSync, mkdirSync, chmodSync, renameSync } from 'node:fs';
|
|
2
|
+
import { join } from 'node:path';
|
|
3
|
+
import { homedir } from 'node:os';
|
|
4
|
+
import { randomBytes } from 'node:crypto';
|
|
5
|
+
const APP_DIR = 'getverbal';
|
|
6
|
+
const CREDENTIALS_FILE = 'credentials.json';
|
|
7
|
+
export function getCredentialsDir() {
|
|
8
|
+
const xdgConfigHome = process.env['XDG_CONFIG_HOME'];
|
|
9
|
+
if (xdgConfigHome && xdgConfigHome.trim() !== '') {
|
|
10
|
+
return join(xdgConfigHome, APP_DIR);
|
|
11
|
+
}
|
|
12
|
+
return join(homedir(), '.config', APP_DIR);
|
|
13
|
+
}
|
|
14
|
+
function getCredentialsPath() {
|
|
15
|
+
return join(getCredentialsDir(), CREDENTIALS_FILE);
|
|
16
|
+
}
|
|
17
|
+
export async function readCredentials() {
|
|
18
|
+
const credPath = getCredentialsPath();
|
|
19
|
+
try {
|
|
20
|
+
const content = readFileSync(credPath, 'utf-8');
|
|
21
|
+
const parsed = JSON.parse(content);
|
|
22
|
+
return parsed;
|
|
23
|
+
}
|
|
24
|
+
catch (err) {
|
|
25
|
+
const nodeErr = err;
|
|
26
|
+
// File doesn't exist — not an error condition
|
|
27
|
+
if (nodeErr.code === 'ENOENT') {
|
|
28
|
+
return null;
|
|
29
|
+
}
|
|
30
|
+
// Malformed JSON or other read error — treat as missing to avoid crashing CLI
|
|
31
|
+
if (err instanceof SyntaxError) {
|
|
32
|
+
return null;
|
|
33
|
+
}
|
|
34
|
+
throw err;
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
export async function writeCredentials(credentials) {
|
|
38
|
+
const dir = getCredentialsDir();
|
|
39
|
+
const credPath = getCredentialsPath();
|
|
40
|
+
// Create directory (and parents) if needed
|
|
41
|
+
mkdirSync(dir, { recursive: true });
|
|
42
|
+
const content = JSON.stringify(credentials, null, 2);
|
|
43
|
+
// Atomic write: write to temp file then rename
|
|
44
|
+
const tmpPath = join(dir, `.credentials-${randomBytes(8).toString('hex')}.tmp`);
|
|
45
|
+
try {
|
|
46
|
+
writeFileSync(tmpPath, content, { encoding: 'utf-8', mode: 0o600 });
|
|
47
|
+
// Ensure permissions are set correctly (in case umask altered them)
|
|
48
|
+
chmodSync(tmpPath, 0o600);
|
|
49
|
+
renameSync(tmpPath, credPath);
|
|
50
|
+
}
|
|
51
|
+
catch (err) {
|
|
52
|
+
// Clean up temp file if rename failed
|
|
53
|
+
try {
|
|
54
|
+
rmSync(tmpPath, { force: true });
|
|
55
|
+
}
|
|
56
|
+
catch {
|
|
57
|
+
// Ignore cleanup error
|
|
58
|
+
}
|
|
59
|
+
throw err;
|
|
60
|
+
}
|
|
61
|
+
// Ensure final file has correct permissions
|
|
62
|
+
chmodSync(credPath, 0o600);
|
|
63
|
+
}
|
|
64
|
+
export async function clearCredentials() {
|
|
65
|
+
const credPath = getCredentialsPath();
|
|
66
|
+
try {
|
|
67
|
+
rmSync(credPath, { force: true });
|
|
68
|
+
}
|
|
69
|
+
catch (err) {
|
|
70
|
+
const nodeErr = err;
|
|
71
|
+
if (nodeErr.code === 'ENOENT') {
|
|
72
|
+
// File already gone — not an error
|
|
73
|
+
return;
|
|
74
|
+
}
|
|
75
|
+
throw err;
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
//# sourceMappingURL=credentials.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"credentials.js","sourceRoot":"","sources":["../../src/auth/credentials.ts"],"names":[],"mappings":"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"}
|
package/dist/cli.d.ts
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"cli.d.ts","sourceRoot":"","sources":["../src/cli.ts"],"names":[],"mappings":""}
|
package/dist/cli.js
ADDED
|
@@ -0,0 +1,66 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
import pc from 'picocolors';
|
|
3
|
+
import { runInit } from './commands/init.js';
|
|
4
|
+
import { runStatus } from './commands/status.js';
|
|
5
|
+
import { runLogout } from './commands/logout.js';
|
|
6
|
+
import { runUninstall } from './commands/uninstall.js';
|
|
7
|
+
const VERSION = '0.1.0';
|
|
8
|
+
const HELP = `
|
|
9
|
+
${pc.bold('getverbal')} — Set up Verbal AI spend tracking in one command
|
|
10
|
+
|
|
11
|
+
${pc.bold('Usage:')}
|
|
12
|
+
getverbal <command>
|
|
13
|
+
|
|
14
|
+
${pc.bold('Commands:')}
|
|
15
|
+
init Authenticate and configure all detected AI tools
|
|
16
|
+
status Show current configuration and connection status
|
|
17
|
+
logout Remove stored credentials
|
|
18
|
+
uninstall Remove Verbal from all AI tools and delete credentials
|
|
19
|
+
|
|
20
|
+
${pc.bold('Options:')}
|
|
21
|
+
--help, -h Show this help message
|
|
22
|
+
--version, -v Show version number
|
|
23
|
+
|
|
24
|
+
${pc.bold('Examples:')}
|
|
25
|
+
npx getverbal init
|
|
26
|
+
getverbal status
|
|
27
|
+
getverbal logout
|
|
28
|
+
`.trim();
|
|
29
|
+
async function main() {
|
|
30
|
+
const command = process.argv[2];
|
|
31
|
+
if (!command || command === '--help' || command === '-h') {
|
|
32
|
+
console.log(HELP);
|
|
33
|
+
process.exit(0);
|
|
34
|
+
}
|
|
35
|
+
if (command === '--version' || command === '-v') {
|
|
36
|
+
console.log(VERSION);
|
|
37
|
+
process.exit(0);
|
|
38
|
+
}
|
|
39
|
+
try {
|
|
40
|
+
switch (command) {
|
|
41
|
+
case 'init':
|
|
42
|
+
await runInit();
|
|
43
|
+
break;
|
|
44
|
+
case 'status':
|
|
45
|
+
await runStatus();
|
|
46
|
+
break;
|
|
47
|
+
case 'logout':
|
|
48
|
+
await runLogout();
|
|
49
|
+
break;
|
|
50
|
+
case 'uninstall':
|
|
51
|
+
await runUninstall();
|
|
52
|
+
break;
|
|
53
|
+
default:
|
|
54
|
+
console.error(pc.red(`Unknown command: ${command}`));
|
|
55
|
+
console.error(`Run ${pc.cyan('getverbal --help')} to see available commands.`);
|
|
56
|
+
process.exit(1);
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
catch (err) {
|
|
60
|
+
const message = err instanceof Error ? err.message : String(err);
|
|
61
|
+
console.error(pc.red(`Error: ${message}`));
|
|
62
|
+
process.exit(1);
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
main();
|
|
66
|
+
//# sourceMappingURL=cli.js.map
|
package/dist/cli.js.map
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"cli.js","sourceRoot":"","sources":["../src/cli.ts"],"names":[],"mappings":";AACA,OAAO,EAAE,MAAM,YAAY,CAAC;AAC5B,OAAO,EAAE,OAAO,EAAE,MAAM,oBAAoB,CAAC;AAC7C,OAAO,EAAE,SAAS,EAAE,MAAM,sBAAsB,CAAC;AACjD,OAAO,EAAE,SAAS,EAAE,MAAM,sBAAsB,CAAC;AACjD,OAAO,EAAE,YAAY,EAAE,MAAM,yBAAyB,CAAC;AAEvD,MAAM,OAAO,GAAG,OAAO,CAAC;AAExB,MAAM,IAAI,GAAG;EACX,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC;;EAEpB,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC;;;EAGjB,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC;;;;;;EAMpB,EAAE,CAAC,IAAI,CAAC,UAAU,CAAC;;;;EAInB,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC;;;;CAIrB,CAAC,IAAI,EAAE,CAAC;AAET,KAAK,UAAU,IAAI;IACjB,MAAM,OAAO,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAEhC,IAAI,CAAC,OAAO,IAAI,OAAO,KAAK,QAAQ,IAAI,OAAO,KAAK,IAAI,EAAE,CAAC;QACzD,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAClB,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IAED,IAAI,OAAO,KAAK,WAAW,IAAI,OAAO,KAAK,IAAI,EAAE,CAAC;QAChD,OAAO,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;QACrB,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IAED,IAAI,CAAC;QACH,QAAQ,OAAO,EAAE,CAAC;YAChB,KAAK,MAAM;gBACT,MAAM,OAAO,EAAE,CAAC;gBAChB,MAAM;YACR,KAAK,QAAQ;gBACX,MAAM,SAAS,EAAE,CAAC;gBAClB,MAAM;YACR,KAAK,QAAQ;gBACX,MAAM,SAAS,EAAE,CAAC;gBAClB,MAAM;YACR,KAAK,WAAW;gBACd,MAAM,YAAY,EAAE,CAAC;gBACrB,MAAM;YACR;gBACE,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC,GAAG,CAAC,oBAAoB,OAAO,EAAE,CAAC,CAAC,CAAC;gBACrD,OAAO,CAAC,KAAK,CAAC,OAAO,EAAE,CAAC,IAAI,CAAC,kBAAkB,CAAC,6BAA6B,CAAC,CAAC;gBAC/E,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QACpB,CAAC;IACH,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,MAAM,OAAO,GAAG,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;QACjE,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC,GAAG,CAAC,UAAU,OAAO,EAAE,CAAC,CAAC,CAAC;QAC3C,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;AACH,CAAC;AAED,IAAI,EAAE,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"init.d.ts","sourceRoot":"","sources":["../../src/commands/init.ts"],"names":[],"mappings":"AAgEA,wBAAsB,OAAO,IAAI,OAAO,CAAC,IAAI,CAAC,CAqK7C"}
|
|
@@ -0,0 +1,188 @@
|
|
|
1
|
+
import pc from 'picocolors';
|
|
2
|
+
import { createInterface } from 'node:readline/promises';
|
|
3
|
+
import { existsSync, readFileSync } from 'node:fs';
|
|
4
|
+
import { join } from 'node:path';
|
|
5
|
+
import { browserAuth } from '../auth/browser-auth.js';
|
|
6
|
+
import { readCredentials, writeCredentials } from '../auth/credentials.js';
|
|
7
|
+
import { detectAll } from '../detect/index.js';
|
|
8
|
+
import { configureAll } from '../configure/index.js';
|
|
9
|
+
import { verifyConnection } from '../verify.js';
|
|
10
|
+
// ---------------------------------------------------------------------------
|
|
11
|
+
// Helpers
|
|
12
|
+
// ---------------------------------------------------------------------------
|
|
13
|
+
async function confirm(rl, message, defaultYes) {
|
|
14
|
+
const hint = defaultYes ? '(Y/n)' : '(y/N)';
|
|
15
|
+
const answer = await rl.question(`${message} ${hint} `);
|
|
16
|
+
const trimmed = answer.trim().toLowerCase();
|
|
17
|
+
if (trimmed === '')
|
|
18
|
+
return defaultYes;
|
|
19
|
+
return trimmed === 'y' || trimmed === 'yes';
|
|
20
|
+
}
|
|
21
|
+
/**
|
|
22
|
+
* Best-effort check — only matches exact lines, not glob patterns like *.json
|
|
23
|
+
* or **\/.mcp.json. A false negative (missing the pattern) just means we skip
|
|
24
|
+
* the "already gitignored" hint, which is acceptable.
|
|
25
|
+
*/
|
|
26
|
+
function isInGitignore(filename) {
|
|
27
|
+
const gitignorePath = join(process.cwd(), '.gitignore');
|
|
28
|
+
if (!existsSync(gitignorePath))
|
|
29
|
+
return false;
|
|
30
|
+
try {
|
|
31
|
+
const content = readFileSync(gitignorePath, 'utf-8');
|
|
32
|
+
return content.split('\n').some((line) => line.trim() === filename);
|
|
33
|
+
}
|
|
34
|
+
catch {
|
|
35
|
+
return false;
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
function toolDisplayName(tool) {
|
|
39
|
+
switch (tool) {
|
|
40
|
+
case 'claude-code':
|
|
41
|
+
return 'Claude Code';
|
|
42
|
+
case 'claude-desktop':
|
|
43
|
+
return 'Claude Desktop';
|
|
44
|
+
case 'cursor':
|
|
45
|
+
return 'Cursor';
|
|
46
|
+
case 'codex':
|
|
47
|
+
return 'Codex';
|
|
48
|
+
default: {
|
|
49
|
+
const _exhaustive = tool;
|
|
50
|
+
return String(_exhaustive);
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
// ---------------------------------------------------------------------------
|
|
55
|
+
// Main command
|
|
56
|
+
// ---------------------------------------------------------------------------
|
|
57
|
+
export async function runInit() {
|
|
58
|
+
// ── 1. Banner ─────────────────────────────────────────────────────────────
|
|
59
|
+
console.log();
|
|
60
|
+
console.log(pc.bold('Verbal — AI Spend Tracker'));
|
|
61
|
+
console.log();
|
|
62
|
+
const rl = createInterface({ input: process.stdin, output: process.stdout });
|
|
63
|
+
let credentials = null;
|
|
64
|
+
try {
|
|
65
|
+
// ── 2. Check for existing credentials ──────────────────────────────────
|
|
66
|
+
const existing = await readCredentials();
|
|
67
|
+
if (existing !== null) {
|
|
68
|
+
console.log(pc.dim(`Already authenticated as ${pc.bold(existing.user_email)}.`));
|
|
69
|
+
const reauth = await confirm(rl, 'Re-authenticate?', false);
|
|
70
|
+
if (!reauth) {
|
|
71
|
+
credentials = existing;
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
// ── 3. Browser auth (if not reusing existing) ──────────────────────────
|
|
75
|
+
if (credentials === null) {
|
|
76
|
+
console.log(pc.cyan('Opening browser to sign up or log in...'));
|
|
77
|
+
console.log(pc.dim('If your browser does not open, visit:'), pc.underline('https://www.getverbal.ai/cli/auth'));
|
|
78
|
+
console.log();
|
|
79
|
+
try {
|
|
80
|
+
const creds = await browserAuth({ apiUrl: 'https://www.getverbal.ai' });
|
|
81
|
+
await writeCredentials(creds);
|
|
82
|
+
console.log(pc.green(`✓ Authenticated as ${pc.bold(creds.user_email)}`));
|
|
83
|
+
console.log();
|
|
84
|
+
credentials = creds;
|
|
85
|
+
}
|
|
86
|
+
catch (err) {
|
|
87
|
+
const message = err instanceof Error ? err.message : String(err);
|
|
88
|
+
console.error(pc.red(`✗ Authentication failed: ${message}`));
|
|
89
|
+
process.exit(1);
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
// ── 4. Detect tools ────────────────────────────────────────────────────
|
|
93
|
+
console.log(pc.bold('Detecting AI tools...'));
|
|
94
|
+
const detections = await detectAll();
|
|
95
|
+
for (const d of detections) {
|
|
96
|
+
const name = toolDisplayName(d.tool);
|
|
97
|
+
if (d.detected) {
|
|
98
|
+
console.log(pc.green(` ✓ ${name}`) + pc.dim(` (${d.details})`));
|
|
99
|
+
}
|
|
100
|
+
else {
|
|
101
|
+
console.log(pc.dim(` ✗ ${name} (not found)`));
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
console.log();
|
|
105
|
+
const detected = detections.filter((d) => d.detected);
|
|
106
|
+
if (detected.length === 0) {
|
|
107
|
+
console.log(pc.yellow('⚠ No supported AI tools detected. Skipping configuration.'));
|
|
108
|
+
console.log();
|
|
109
|
+
}
|
|
110
|
+
// ── 5. Per-tool confirmation ────────────────────────────────────────────
|
|
111
|
+
const confirmed = [];
|
|
112
|
+
for (const d of detected) {
|
|
113
|
+
const name = toolDisplayName(d.tool);
|
|
114
|
+
if (d.existingConfig) {
|
|
115
|
+
console.log(pc.dim(` ${name} is already configured.`));
|
|
116
|
+
const overwrite = await confirm(rl, ` Overwrite ${name} config?`, false);
|
|
117
|
+
if (overwrite) {
|
|
118
|
+
confirmed.push(d);
|
|
119
|
+
}
|
|
120
|
+
else {
|
|
121
|
+
console.log(pc.dim(` Skipping ${name}.`));
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
else {
|
|
125
|
+
const configure = await confirm(rl, ` Configure ${name}?`, true);
|
|
126
|
+
if (configure) {
|
|
127
|
+
confirmed.push(d);
|
|
128
|
+
}
|
|
129
|
+
else {
|
|
130
|
+
console.log(pc.dim(` Skipping ${name}.`));
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
if (confirmed.length > 0) {
|
|
135
|
+
console.log();
|
|
136
|
+
}
|
|
137
|
+
// ── 6. Configure confirmed tools ───────────────────────────────────────
|
|
138
|
+
if (confirmed.length > 0) {
|
|
139
|
+
console.log(pc.bold('Configuring tools...'));
|
|
140
|
+
const results = await configureAll(confirmed, credentials);
|
|
141
|
+
for (const r of results) {
|
|
142
|
+
const d = confirmed.find((det) => det.tool === r.tool);
|
|
143
|
+
const name = toolDisplayName(r.tool);
|
|
144
|
+
const configPath = d?.configPath ?? '';
|
|
145
|
+
if (r.success) {
|
|
146
|
+
console.log(pc.green(` ✓ ${name}`) + (configPath ? pc.dim(` ${configPath}`) : ''));
|
|
147
|
+
}
|
|
148
|
+
else {
|
|
149
|
+
console.log(pc.red(` ✗ ${name}: ${r.error ?? 'unknown error'}`));
|
|
150
|
+
}
|
|
151
|
+
}
|
|
152
|
+
console.log();
|
|
153
|
+
}
|
|
154
|
+
// ── 7. Verify connection ───────────────────────────────────────────────
|
|
155
|
+
process.stdout.write(pc.dim('Verifying connection...'));
|
|
156
|
+
try {
|
|
157
|
+
await verifyConnection(credentials);
|
|
158
|
+
console.log(' ' + pc.green('✓ test event received'));
|
|
159
|
+
}
|
|
160
|
+
catch (err) {
|
|
161
|
+
const message = err instanceof Error ? err.message : String(err);
|
|
162
|
+
console.log();
|
|
163
|
+
console.log(pc.yellow(`⚠ Verification warning: ${message}`));
|
|
164
|
+
}
|
|
165
|
+
console.log();
|
|
166
|
+
// ── 8. Post-install warnings ───────────────────────────────────────────
|
|
167
|
+
const mcpJsonPath = join(process.cwd(), '.mcp.json');
|
|
168
|
+
if (existsSync(mcpJsonPath) && !isInGitignore('.mcp.json')) {
|
|
169
|
+
console.log(pc.yellow('⚠ .mcp.json contains your API key. Add it to .gitignore to avoid leaking credentials:'));
|
|
170
|
+
console.log(pc.dim(' echo ".mcp.json" >> .gitignore'));
|
|
171
|
+
console.log();
|
|
172
|
+
}
|
|
173
|
+
const needsRestart = confirmed.some((d) => d.tool === 'claude-desktop' || d.tool === 'cursor');
|
|
174
|
+
if (needsRestart) {
|
|
175
|
+
console.log(pc.yellow('⚠ Restart Claude Desktop and Cursor to activate Verbal.'));
|
|
176
|
+
console.log();
|
|
177
|
+
}
|
|
178
|
+
// ── 9. Done ────────────────────────────────────────────────────────────
|
|
179
|
+
console.log(pc.bold('Done!') +
|
|
180
|
+
pc.dim(' Dashboard → ') +
|
|
181
|
+
pc.cyan('https://www.getverbal.ai/dashboard'));
|
|
182
|
+
console.log();
|
|
183
|
+
}
|
|
184
|
+
finally {
|
|
185
|
+
rl.close();
|
|
186
|
+
}
|
|
187
|
+
}
|
|
188
|
+
//# sourceMappingURL=init.js.map
|