fastbrowser_cli 1.0.2

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.
Files changed (101) hide show
  1. package/README.md +0 -0
  2. package/dist/contrib/fastweb-cli/fastweb-cli.d.ts +3 -0
  3. package/dist/contrib/fastweb-cli/fastweb-cli.d.ts.map +1 -0
  4. package/dist/contrib/fastweb-cli/fastweb-cli.js +151 -0
  5. package/dist/contrib/fastweb-cli/fastweb-cli.js.map +1 -0
  6. package/dist/contrib/fastweb-cli/http-client.d.ts +7 -0
  7. package/dist/contrib/fastweb-cli/http-client.d.ts.map +1 -0
  8. package/dist/contrib/fastweb-cli/http-client.js +51 -0
  9. package/dist/contrib/fastweb-cli/http-client.js.map +1 -0
  10. package/dist/contrib/fastweb-http-server/fastweb-http-server.d.ts +3 -0
  11. package/dist/contrib/fastweb-http-server/fastweb-http-server.d.ts.map +1 -0
  12. package/dist/contrib/fastweb-http-server/fastweb-http-server.js +82 -0
  13. package/dist/contrib/fastweb-http-server/fastweb-http-server.js.map +1 -0
  14. package/dist/contrib/fastweb-http-server/routes.d.ts +6 -0
  15. package/dist/contrib/fastweb-http-server/routes.d.ts.map +1 -0
  16. package/dist/contrib/fastweb-http-server/routes.js +41 -0
  17. package/dist/contrib/fastweb-http-server/routes.js.map +1 -0
  18. package/dist/contrib/fastweb-http-server/tool-schemas.d.ts +63 -0
  19. package/dist/contrib/fastweb-http-server/tool-schemas.d.ts.map +1 -0
  20. package/dist/contrib/fastweb-http-server/tool-schemas.js +61 -0
  21. package/dist/contrib/fastweb-http-server/tool-schemas.js.map +1 -0
  22. package/dist/fastweb_cli/fastweb_cli.d.ts +3 -0
  23. package/dist/fastweb_cli/fastweb_cli.d.ts.map +1 -0
  24. package/dist/fastweb_cli/fastweb_cli.js +254 -0
  25. package/dist/fastweb_cli/fastweb_cli.js.map +1 -0
  26. package/dist/fastweb_cli/http-client.d.ts +7 -0
  27. package/dist/fastweb_cli/http-client.d.ts.map +1 -0
  28. package/dist/fastweb_cli/http-client.js +51 -0
  29. package/dist/fastweb_cli/http-client.js.map +1 -0
  30. package/dist/fastweb_cli/libs/http-client.d.ts +7 -0
  31. package/dist/fastweb_cli/libs/http-client.d.ts.map +1 -0
  32. package/dist/fastweb_cli/libs/http-client.js +51 -0
  33. package/dist/fastweb_cli/libs/http-client.js.map +1 -0
  34. package/dist/fastweb_cli/libs/server-manager.d.ts +12 -0
  35. package/dist/fastweb_cli/libs/server-manager.d.ts.map +1 -0
  36. package/dist/fastweb_cli/libs/server-manager.js +194 -0
  37. package/dist/fastweb_cli/libs/server-manager.js.map +1 -0
  38. package/dist/fastweb_http_server/fastweb_http_server.d.ts +3 -0
  39. package/dist/fastweb_http_server/fastweb_http_server.d.ts.map +1 -0
  40. package/dist/fastweb_http_server/fastweb_http_server.js +82 -0
  41. package/dist/fastweb_http_server/fastweb_http_server.js.map +1 -0
  42. package/dist/fastweb_http_server/libs/routes.d.ts +6 -0
  43. package/dist/fastweb_http_server/libs/routes.d.ts.map +1 -0
  44. package/dist/fastweb_http_server/libs/routes.js +41 -0
  45. package/dist/fastweb_http_server/libs/routes.js.map +1 -0
  46. package/dist/fastweb_http_server/libs/tool-schemas.d.ts +72 -0
  47. package/dist/fastweb_http_server/libs/tool-schemas.d.ts.map +1 -0
  48. package/dist/fastweb_http_server/libs/tool-schemas.js +65 -0
  49. package/dist/fastweb_http_server/libs/tool-schemas.js.map +1 -0
  50. package/dist/fastweb_http_server/routes.d.ts +6 -0
  51. package/dist/fastweb_http_server/routes.d.ts.map +1 -0
  52. package/dist/fastweb_http_server/routes.js +41 -0
  53. package/dist/fastweb_http_server/routes.js.map +1 -0
  54. package/dist/fastweb_http_server/tool-schemas.d.ts +63 -0
  55. package/dist/fastweb_http_server/tool-schemas.d.ts.map +1 -0
  56. package/dist/fastweb_http_server/tool-schemas.js +61 -0
  57. package/dist/fastweb_http_server/tool-schemas.js.map +1 -0
  58. package/dist/fastweb_mcp/fastweb_mcp.d.ts +4 -0
  59. package/dist/fastweb_mcp/fastweb_mcp.d.ts.map +1 -0
  60. package/dist/fastweb_mcp/fastweb_mcp.js +417 -0
  61. package/dist/fastweb_mcp/fastweb_mcp.js.map +1 -0
  62. package/dist/fastweb_mcp/libs/mcp_client.d.ts +120 -0
  63. package/dist/fastweb_mcp/libs/mcp_client.d.ts.map +1 -0
  64. package/dist/fastweb_mcp/libs/mcp_client.js +83 -0
  65. package/dist/fastweb_mcp/libs/mcp_client.js.map +1 -0
  66. package/dist/fastweb_mcp/libs/mcp_proxy.d.ts +10 -0
  67. package/dist/fastweb_mcp/libs/mcp_proxy.d.ts.map +1 -0
  68. package/dist/fastweb_mcp/libs/mcp_proxy.js +45 -0
  69. package/dist/fastweb_mcp/libs/mcp_proxy.js.map +1 -0
  70. package/dist/fastweb_mcp/libs/schemas.d.ts +28 -0
  71. package/dist/fastweb_mcp/libs/schemas.d.ts.map +1 -0
  72. package/dist/fastweb_mcp/libs/schemas.js +38 -0
  73. package/dist/fastweb_mcp/libs/schemas.js.map +1 -0
  74. package/dist/src/fastweb_mcp.d.ts +17 -0
  75. package/dist/src/fastweb_mcp.d.ts.map +1 -0
  76. package/dist/src/fastweb_mcp.js +342 -0
  77. package/dist/src/fastweb_mcp.js.map +1 -0
  78. package/dist/src/libs/mcp_client.d.ts +120 -0
  79. package/dist/src/libs/mcp_client.d.ts.map +1 -0
  80. package/dist/src/libs/mcp_client.js +83 -0
  81. package/dist/src/libs/mcp_client.js.map +1 -0
  82. package/dist/src/libs/mcp_proxy.d.ts +10 -0
  83. package/dist/src/libs/mcp_proxy.d.ts.map +1 -0
  84. package/dist/src/libs/mcp_proxy.js +45 -0
  85. package/dist/src/libs/mcp_proxy.js.map +1 -0
  86. package/docs/brainstorm_scrap_by_ai.md +32 -0
  87. package/docs/feature_support_cli.md +86 -0
  88. package/package.json +36 -0
  89. package/src/fastweb_cli/fastweb_cli.ts +306 -0
  90. package/src/fastweb_cli/libs/http-client.ts +50 -0
  91. package/src/fastweb_cli/libs/server-manager.ts +210 -0
  92. package/src/fastweb_http_server/fastweb_http_server.ts +97 -0
  93. package/src/fastweb_http_server/libs/routes.ts +44 -0
  94. package/src/fastweb_http_server/libs/tool-schemas.ts +107 -0
  95. package/src/fastweb_mcp/fastweb_mcp.ts +475 -0
  96. package/src/fastweb_mcp/libs/mcp_client.ts +117 -0
  97. package/src/fastweb_mcp/libs/mcp_proxy.ts +48 -0
  98. package/src/fastweb_mcp/libs/schemas.ts +44 -0
  99. package/tmp/.claude/settings.local.json +7 -0
  100. package/tmp/.claude/skills/fastweb/SKILL.md +214 -0
  101. package/tsconfig.json +40 -0
@@ -0,0 +1,306 @@
1
+ #!/usr/bin/env node
2
+
3
+ // npm imports
4
+ import { Command } from 'commander';
5
+
6
+ // local imports
7
+ import { HttpClient } from './libs/http-client.js';
8
+ import { ServerManager } from './libs/server-manager.js';
9
+ import type { QuerySelectorInput, QuerySelectorFirstInput, QuerySelectorsAllRequest, QuerySelectorRequest } from '../fastweb_http_server/libs/tool-schemas.js';
10
+
11
+
12
+ ///////////////////////////////////////////////////////////////////////////////
13
+ ///////////////////////////////////////////////////////////////////////////////
14
+ //
15
+ ///////////////////////////////////////////////////////////////////////////////
16
+ ///////////////////////////////////////////////////////////////////////////////
17
+
18
+ type GlobalOpts = {
19
+ server?: string;
20
+ autostart?: boolean;
21
+ };
22
+
23
+ function getServerFromCmd(cmd: Command): string {
24
+ const globalOpts = cmd.optsWithGlobals<GlobalOpts>();
25
+ return HttpClient.getServerUrl(globalOpts.server);
26
+ }
27
+
28
+ function getAutostartFromCmd(cmd: Command): boolean {
29
+ const globalOpts = cmd.optsWithGlobals<GlobalOpts>();
30
+ return globalOpts.autostart !== false;
31
+ }
32
+
33
+ async function runTool(cmd: Command, routeName: string, body: unknown): Promise<void> {
34
+ const server = getServerFromCmd(cmd);
35
+ try {
36
+ if (getAutostartFromCmd(cmd) === true) {
37
+ await ServerManager.ensureRunning(server);
38
+ }
39
+ const response = await HttpClient.postTool(server, routeName, body);
40
+ HttpClient.printResponse(response);
41
+ } catch (err) {
42
+ const message = err instanceof Error ? err.message : String(err);
43
+ console.error(`fastweb-cli error: ${message}`);
44
+ process.exit(1);
45
+ }
46
+ }
47
+
48
+ ///////////////////////////////////////////////////////////////////////////////
49
+ ///////////////////////////////////////////////////////////////////////////////
50
+ //
51
+ ///////////////////////////////////////////////////////////////////////////////
52
+ ///////////////////////////////////////////////////////////////////////////////
53
+
54
+ function buildQuerySelectorsBody(opts: {
55
+ selector?: string[];
56
+ limit?: string;
57
+ withAncestors?: boolean;
58
+ selectorsJson?: string;
59
+ }): QuerySelectorsAllRequest {
60
+ if (opts.selectorsJson !== undefined && opts.selectorsJson !== '') {
61
+ let parsed: unknown;
62
+ try {
63
+ parsed = JSON.parse(opts.selectorsJson);
64
+ } catch (err) {
65
+ throw new Error(`--selectors-json is not valid JSON: ${(err as Error).message}`);
66
+ }
67
+ if (Array.isArray(parsed) === false) {
68
+ throw new Error('--selectors-json must be a JSON array');
69
+ }
70
+ return { selectors: parsed as QuerySelectorInput[] };
71
+ }
72
+
73
+ const selectorList = opts.selector ?? [];
74
+ if (selectorList.length === 0) {
75
+ throw new Error('At least one --selector or --selectors-json is required');
76
+ }
77
+
78
+ const limit = opts.limit === undefined ? 0 : Number.parseInt(opts.limit, 10);
79
+ if (Number.isNaN(limit) === true) {
80
+ throw new Error(`Invalid --limit: ${opts.limit}`);
81
+ }
82
+ const withAncestors = opts.withAncestors !== false;
83
+
84
+ const selectors: QuerySelectorInput[] = selectorList.map((selector) => ({
85
+ selector,
86
+ limit,
87
+ withAncestors,
88
+ }));
89
+ return { selectors };
90
+ }
91
+
92
+ function buildQuerySelectorFirstBody(opts: {
93
+ selector?: string[];
94
+ withAncestors?: boolean;
95
+ selectorsJson?: string;
96
+ }): QuerySelectorRequest {
97
+ if (opts.selectorsJson !== undefined && opts.selectorsJson !== '') {
98
+ let parsed: unknown;
99
+ try {
100
+ parsed = JSON.parse(opts.selectorsJson);
101
+ } catch (err) {
102
+ throw new Error(`--selectors-json is not valid JSON: ${(err as Error).message}`);
103
+ }
104
+ if (Array.isArray(parsed) === false) {
105
+ throw new Error('--selectors-json must be a JSON array');
106
+ }
107
+ return { selectors: parsed as QuerySelectorFirstInput[] };
108
+ }
109
+
110
+ const selectorList = opts.selector ?? [];
111
+ if (selectorList.length === 0) {
112
+ throw new Error('At least one --selector or --selectors-json is required');
113
+ }
114
+
115
+ const withAncestors = opts.withAncestors !== false;
116
+
117
+ const selectors: QuerySelectorFirstInput[] = selectorList.map((selector) => ({
118
+ selector,
119
+ withAncestors,
120
+ }));
121
+ return { selectors };
122
+ }
123
+
124
+ ///////////////////////////////////////////////////////////////////////////////
125
+ ///////////////////////////////////////////////////////////////////////////////
126
+ //
127
+ ///////////////////////////////////////////////////////////////////////////////
128
+ ///////////////////////////////////////////////////////////////////////////////
129
+
130
+ async function main(): Promise<void> {
131
+ const program = new Command();
132
+ program
133
+ .name('fastweb-cli')
134
+ .description('CLI client for fastweb-http-server2')
135
+ .option('--server <url>', 'fastweb-http-server URL (default: env FASTWEB_SERVER or http://localhost:8787)')
136
+ .option('--autostart', 'Auto-start the server before a command if it is not running', true)
137
+ .option('--no-autostart', 'Do not auto-start the server before a command');
138
+
139
+ const serverCmd = program
140
+ .command('server')
141
+ .description('Manage the fastweb HTTP server');
142
+
143
+ serverCmd
144
+ .command('start')
145
+ .description('Start the fastweb HTTP server as a detached daemon')
146
+ .action(async (_opts, cmd: Command) => {
147
+ const server = getServerFromCmd(cmd);
148
+ try {
149
+ await ServerManager.start(server);
150
+ } catch (err) {
151
+ const message = err instanceof Error ? err.message : String(err);
152
+ console.error(`fastweb-cli error: ${message}`);
153
+ process.exit(1);
154
+ }
155
+ });
156
+
157
+ serverCmd
158
+ .command('stop')
159
+ .description('Stop the fastweb HTTP server')
160
+ .action(async (_opts, cmd: Command) => {
161
+ const server = getServerFromCmd(cmd);
162
+ try {
163
+ await ServerManager.stop(server);
164
+ } catch (err) {
165
+ const message = err instanceof Error ? err.message : String(err);
166
+ console.error(`fastweb-cli error: ${message}`);
167
+ process.exit(1);
168
+ }
169
+ });
170
+
171
+ serverCmd
172
+ .command('status')
173
+ .description('Report whether the fastweb HTTP server is running')
174
+ .action(async (_opts, cmd: Command) => {
175
+ const server = getServerFromCmd(cmd);
176
+ const state = await ServerManager.status(server);
177
+ console.log(`fastweb server at ${server}: ${state}`);
178
+ if (state === 'stopped') process.exit(1);
179
+ });
180
+
181
+ program
182
+ .command('list_pages')
183
+ .description('List all open browser pages')
184
+ .action(async (_opts, cmd: Command) => {
185
+ await runTool(cmd, 'list_pages', {});
186
+ });
187
+
188
+ program
189
+ .command('new_page')
190
+ .description('Open a new browser page')
191
+ .requiredOption('--url <url>', 'URL to open')
192
+ .action(async (opts: { url: string }, cmd: Command) => {
193
+ await runTool(cmd, 'new_page', { url: opts.url });
194
+ });
195
+
196
+ program
197
+ .command('close_page')
198
+ .description('Close a page by its id')
199
+ .requiredOption('--page-id <number>', 'The page id to close')
200
+ .action(async (opts: { pageId: string }, cmd: Command) => {
201
+ const pageId = Number.parseInt(opts.pageId, 10);
202
+ if (Number.isNaN(pageId) === true) {
203
+ console.error(`Invalid --page-id: ${opts.pageId}`);
204
+ process.exit(1);
205
+ }
206
+ await runTool(cmd, 'close_page', { pageId });
207
+ });
208
+
209
+ program
210
+ .command('navigate_page')
211
+ .description('Navigate the current page to a URL')
212
+ .requiredOption('--url <url>', 'URL to navigate to')
213
+ .action(async (opts: { url: string }, cmd: Command) => {
214
+ await runTool(cmd, 'navigate_page', { url: opts.url });
215
+ });
216
+
217
+ program
218
+ .command('click')
219
+ .description('Click an element by its accessibility selector')
220
+ .requiredOption('-s, --selector <selector>', 'Accessibility selector (e.g. "#1_3" or \'button[name="Submit"]\')')
221
+ .action(async (opts: { selector: string }, cmd: Command) => {
222
+ await runTool(cmd, 'click', { selector: opts.selector });
223
+ });
224
+
225
+ program
226
+ .command('fill_form')
227
+ .description('Fill a form field by its accessibility selector')
228
+ .requiredOption('-s, --selector <selector>', 'Accessibility selector (e.g. "#1_3" or \'textbox[name="Email"]\')')
229
+ .requiredOption('--value <value>', 'Value to fill')
230
+ .action(async (opts: { selector: string; value: string }, cmd: Command) => {
231
+ await runTool(cmd, 'fill_form', {
232
+ elements: [{ selector: opts.selector, value: opts.value }],
233
+ });
234
+ });
235
+
236
+ program
237
+ .command('query_selectors_all')
238
+ .description('Query the accessibility tree with CSS-like selectors')
239
+ .option('-s, --selector <selector>', 'CSS-like selector (repeatable)', (value: string, prev: string[] = []) => {
240
+ prev.push(value);
241
+ return prev;
242
+ })
243
+ .option('--limit <number>', 'Max nodes per selector (0 = unlimited)', '0')
244
+ .option('--with-ancestors', 'Include ancestor nodes', true)
245
+ .option('--no-with-ancestors', 'Exclude ancestor nodes')
246
+ .option('--selectors-json <json>', 'JSON array of {selector,limit,withAncestors} for per-selector control')
247
+ .action(async (opts: {
248
+ selector?: string[];
249
+ limit?: string;
250
+ withAncestors?: boolean;
251
+ selectorsJson?: string;
252
+ }, cmd: Command) => {
253
+ let body: QuerySelectorsAllRequest;
254
+ try {
255
+ body = buildQuerySelectorsBody(opts);
256
+ } catch (err) {
257
+ console.error(`fastweb-cli error: ${(err as Error).message}`);
258
+ process.exit(1);
259
+ }
260
+ await runTool(cmd, 'query_selectors_all', body);
261
+ });
262
+
263
+ program
264
+ .command('query_selectors')
265
+ .description('Query the accessibility tree with CSS-like selectors and, for each, return the first matching node')
266
+ .option('-s, --selector <selector>', 'CSS-like selector (repeatable)', (value: string, prev: string[] = []) => {
267
+ prev.push(value);
268
+ return prev;
269
+ })
270
+ .option('--with-ancestors', 'Include ancestor nodes', true)
271
+ .option('--no-with-ancestors', 'Exclude ancestor nodes')
272
+ .option('--selectors-json <json>', 'JSON array of {selector,withAncestors} for per-selector control')
273
+ .action(async (opts: {
274
+ selector?: string[];
275
+ withAncestors?: boolean;
276
+ selectorsJson?: string;
277
+ }, cmd: Command) => {
278
+ let body: QuerySelectorRequest;
279
+ try {
280
+ body = buildQuerySelectorFirstBody(opts);
281
+ } catch (err) {
282
+ console.error(`fastweb-cli error: ${(err as Error).message}`);
283
+ process.exit(1);
284
+ }
285
+ await runTool(cmd, 'query_selectors', body);
286
+ });
287
+
288
+ program
289
+ .command('take_snapshot')
290
+ .description('Take an accessibility-tree snapshot of the current page')
291
+ .action(async (_opts, cmd: Command) => {
292
+ await runTool(cmd, 'take_snapshot', {});
293
+ });
294
+
295
+ program
296
+ .command('press_keys')
297
+ .description('Press a sequence of keys')
298
+ .requiredOption('--keys <keys>', "Comma-separated keys. E.g. 'Hello, Tab, Enter'")
299
+ .action(async (opts: { keys: string }, cmd: Command) => {
300
+ await runTool(cmd, 'press_keys', { keys: opts.keys });
301
+ });
302
+
303
+ await program.parseAsync(process.argv);
304
+ }
305
+
306
+ void main();
@@ -0,0 +1,50 @@
1
+ // local imports
2
+ import { ToolResponseSchema, type ToolResponse } from '../../fastweb_http_server/libs/tool-schemas.js';
3
+
4
+ ///////////////////////////////////////////////////////////////////////////////
5
+ ///////////////////////////////////////////////////////////////////////////////
6
+ //
7
+ ///////////////////////////////////////////////////////////////////////////////
8
+ ///////////////////////////////////////////////////////////////////////////////
9
+
10
+ export class HttpClient {
11
+ static getServerUrl(overrideUrl: string | undefined): string {
12
+ if (overrideUrl !== undefined && overrideUrl !== '') return overrideUrl;
13
+ const envUrl = process.env.FASTWEB_SERVER;
14
+ if (envUrl !== undefined && envUrl !== '') return envUrl;
15
+ return 'http://localhost:8787';
16
+ }
17
+
18
+ static async postTool(serverUrl: string, routeName: string, body: unknown): Promise<ToolResponse> {
19
+ const url = `${serverUrl.replace(/\/+$/, '')}/tools/${routeName}`;
20
+ const response = await fetch(url, {
21
+ method: 'POST',
22
+ headers: { 'content-type': 'application/json' },
23
+ body: JSON.stringify(body ?? {}),
24
+ });
25
+
26
+ const responseText = await response.text();
27
+ if (response.ok === false) {
28
+ throw new Error(`HTTP ${response.status} ${response.statusText}: ${responseText}`);
29
+ }
30
+
31
+ let parsedJson: unknown;
32
+ try {
33
+ parsedJson = JSON.parse(responseText);
34
+ } catch (err) {
35
+ throw new Error(`Invalid JSON from server: ${responseText}`);
36
+ }
37
+
38
+ const parsed = ToolResponseSchema.safeParse(parsedJson);
39
+ if (parsed.success === false) {
40
+ throw new Error(`Unexpected response shape: ${JSON.stringify(parsed.error.flatten())}`);
41
+ }
42
+ return parsed.data;
43
+ }
44
+
45
+ static printResponse(response: ToolResponse): void {
46
+ for (const part of response.content) {
47
+ console.log(part.text);
48
+ }
49
+ }
50
+ }
@@ -0,0 +1,210 @@
1
+ // node imports
2
+ import Fs from 'node:fs';
3
+ import Os from 'node:os';
4
+ import Path from 'node:path';
5
+ import { spawn } from 'node:child_process';
6
+
7
+ ///////////////////////////////////////////////////////////////////////////////
8
+ ///////////////////////////////////////////////////////////////////////////////
9
+ //
10
+ ///////////////////////////////////////////////////////////////////////////////
11
+ ///////////////////////////////////////////////////////////////////////////////
12
+
13
+ type PidFile = {
14
+ pid: number;
15
+ port: number;
16
+ startedAt: string;
17
+ };
18
+
19
+ const STATE_DIR = Path.join(Os.homedir(), '.fastweb_cli');
20
+ const PID_FILE = Path.join(STATE_DIR, 'server.json');
21
+ const LOG_FILE = Path.join(STATE_DIR, 'server.log');
22
+
23
+ ///////////////////////////////////////////////////////////////////////////////
24
+ ///////////////////////////////////////////////////////////////////////////////
25
+ //
26
+ ///////////////////////////////////////////////////////////////////////////////
27
+ ///////////////////////////////////////////////////////////////////////////////
28
+
29
+ export class ServerManager {
30
+ static async status(serverUrl: string): Promise<'running' | 'stopped'> {
31
+ const base = serverUrl.replace(/\/+$/, '');
32
+ try {
33
+ const response = await fetch(`${base}/health`, {
34
+ method: 'GET',
35
+ signal: AbortSignal.timeout(500),
36
+ });
37
+ if (response.ok === false) return 'stopped';
38
+ const payload = await response.json() as { ok?: unknown };
39
+ return payload.ok === true ? 'running' : 'stopped';
40
+ } catch {
41
+ return 'stopped';
42
+ }
43
+ }
44
+
45
+ static async ensureRunning(serverUrl: string): Promise<void> {
46
+ const state = await ServerManager.status(serverUrl);
47
+ if (state === 'running') return;
48
+
49
+ if (ServerManager.isLocalUrl(serverUrl) === false) {
50
+ throw new Error(`fastweb server at ${serverUrl} is not reachable and cannot be auto-started (non-local URL)`);
51
+ }
52
+ await ServerManager.start(serverUrl);
53
+ }
54
+
55
+ static async start(serverUrl: string): Promise<void> {
56
+ const existing = await ServerManager.status(serverUrl);
57
+ if (existing === 'running') {
58
+ console.error(`fastweb server already running at ${serverUrl}`);
59
+ return;
60
+ }
61
+
62
+ if (ServerManager.isLocalUrl(serverUrl) === false) {
63
+ throw new Error(`Refusing to start: ${serverUrl} is not a local URL`);
64
+ }
65
+
66
+ const port = ServerManager.parsePort(serverUrl);
67
+ const entryPath = Path.resolve(__dirname, '..', '..', 'fastweb_http_server', 'fastweb_http_server.ts');
68
+ const packageRoot = Path.resolve(__dirname, '..', '..', '..');
69
+
70
+ Fs.mkdirSync(STATE_DIR, { recursive: true });
71
+ const logFd = Fs.openSync(LOG_FILE, 'a');
72
+
73
+ const child = spawn('npx', ['tsx', entryPath, '--port', String(port)], {
74
+ detached: true,
75
+ stdio: ['ignore', logFd, logFd],
76
+ cwd: packageRoot,
77
+ env: process.env,
78
+ });
79
+ child.unref();
80
+
81
+ const pid = child.pid;
82
+ if (pid === undefined) {
83
+ Fs.closeSync(logFd);
84
+ throw new Error('Failed to spawn fastweb-http-server (no pid)');
85
+ }
86
+
87
+ const pidFile: PidFile = {
88
+ pid,
89
+ port,
90
+ startedAt: new Date().toISOString(),
91
+ };
92
+ Fs.writeFileSync(PID_FILE, JSON.stringify(pidFile, null, 2));
93
+
94
+ const deadline = Date.now() + 10_000;
95
+ while (Date.now() < deadline) {
96
+ const state = await ServerManager.status(serverUrl);
97
+ if (state === 'running') {
98
+ Fs.closeSync(logFd);
99
+ console.error(`fastweb server started (pid=${pid}, port=${port})`);
100
+ return;
101
+ }
102
+ await ServerManager.sleep(500);
103
+ }
104
+
105
+ Fs.closeSync(logFd);
106
+ const tail = ServerManager.readLogTail(50);
107
+ throw new Error(`fastweb server did not become healthy within 10s. Log tail:\n${tail}`);
108
+ }
109
+
110
+ static async stop(serverUrl: string): Promise<void> {
111
+ if (Fs.existsSync(PID_FILE) === false) {
112
+ console.error('no server pid file; nothing to stop');
113
+ return;
114
+ }
115
+
116
+ let pidFile: PidFile;
117
+ try {
118
+ const raw = Fs.readFileSync(PID_FILE, 'utf8');
119
+ pidFile = JSON.parse(raw) as PidFile;
120
+ } catch (err) {
121
+ throw new Error(`Failed to read pid file ${PID_FILE}: ${(err as Error).message}`);
122
+ }
123
+
124
+ const pid = pidFile.pid;
125
+ try {
126
+ process.kill(pid, 'SIGTERM');
127
+ } catch (err) {
128
+ const code = (err as NodeJS.ErrnoException).code;
129
+ if (code !== 'ESRCH') {
130
+ throw err;
131
+ }
132
+ }
133
+
134
+ const deadline = Date.now() + 5_000;
135
+ while (Date.now() < deadline) {
136
+ if (ServerManager.isAlive(pid) === false) break;
137
+ await ServerManager.sleep(200);
138
+ }
139
+
140
+ if (ServerManager.isAlive(pid) === true) {
141
+ try {
142
+ process.kill(pid, 'SIGKILL');
143
+ } catch {
144
+ // best effort
145
+ }
146
+ }
147
+
148
+ try {
149
+ Fs.unlinkSync(PID_FILE);
150
+ } catch {
151
+ // best effort
152
+ }
153
+
154
+ // Best-effort: ensure the HTTP server is actually down from the caller's perspective.
155
+ const state = await ServerManager.status(serverUrl);
156
+ if (state === 'running') {
157
+ console.error(`warning: process ${pid} killed but ${serverUrl} still responds to /health`);
158
+ return;
159
+ }
160
+ console.error('fastweb server stopped');
161
+ }
162
+
163
+ private static isLocalUrl(serverUrl: string): boolean {
164
+ try {
165
+ const parsed = new URL(serverUrl);
166
+ const host = parsed.hostname;
167
+ return host === 'localhost' || host === '127.0.0.1' || host === '::1' || host === '0.0.0.0';
168
+ } catch {
169
+ return false;
170
+ }
171
+ }
172
+
173
+ private static parsePort(serverUrl: string): number {
174
+ try {
175
+ const parsed = new URL(serverUrl);
176
+ if (parsed.port !== '') {
177
+ const port = Number.parseInt(parsed.port, 10);
178
+ if (Number.isNaN(port) === false) return port;
179
+ }
180
+ } catch {
181
+ // fall through
182
+ }
183
+ return 8787;
184
+ }
185
+
186
+ private static isAlive(pid: number): boolean {
187
+ try {
188
+ process.kill(pid, 0);
189
+ return true;
190
+ } catch (err) {
191
+ const code = (err as NodeJS.ErrnoException).code;
192
+ if (code === 'EPERM') return true;
193
+ return false;
194
+ }
195
+ }
196
+
197
+ private static readLogTail(maxLines: number): string {
198
+ try {
199
+ const content = Fs.readFileSync(LOG_FILE, 'utf8');
200
+ const lines = content.split('\n');
201
+ return lines.slice(Math.max(0, lines.length - maxLines)).join('\n');
202
+ } catch {
203
+ return '(log unavailable)';
204
+ }
205
+ }
206
+
207
+ private static sleep(ms: number): Promise<void> {
208
+ return new Promise((resolve) => setTimeout(resolve, ms));
209
+ }
210
+ }
@@ -0,0 +1,97 @@
1
+ #!/usr/bin/env node
2
+
3
+ // node imports
4
+ import Path from 'node:path';
5
+
6
+ // npm imports
7
+ import { Command } from 'commander';
8
+ import express from 'express';
9
+
10
+ // local imports
11
+ import { McpClient } from '../fastweb_mcp/libs/mcp_client.js';
12
+ import { Routes } from './libs/routes.js';
13
+
14
+ ///////////////////////////////////////////////////////////////////////////////
15
+ ///////////////////////////////////////////////////////////////////////////////
16
+ //
17
+ ///////////////////////////////////////////////////////////////////////////////
18
+ ///////////////////////////////////////////////////////////////////////////////
19
+
20
+ class MainHelper {
21
+ static async commandStart({
22
+ port,
23
+ verbose = false,
24
+ }: {
25
+ port: number;
26
+ verbose?: boolean;
27
+ }): Promise<void> {
28
+ // Spawn fastweb-mcp as a subprocess and hold a persistent MCP client to it.
29
+ const fastwebMcpEntry = Path.resolve(__dirname, '..', 'fastweb_mcp', 'fastweb_mcp.ts');
30
+ const mcpClient = new McpClient({
31
+ name: 'fastweb-http-server',
32
+ version: '1.0.0',
33
+ transport: {
34
+ type: 'stdio',
35
+ command: 'npx',
36
+ args: ['tsx', fastwebMcpEntry, 'mcp_server'],
37
+ },
38
+ });
39
+
40
+ console.error('Connecting to fastweb-mcp ...');
41
+ await mcpClient.connect();
42
+ console.error('MCP client connected');
43
+
44
+ if (verbose) {
45
+ const tools = await mcpClient.listTools();
46
+ console.error('Tools available in fastweb-mcp:');
47
+ for (const tool of tools) {
48
+ console.error(`- ${tool.name}`);
49
+ }
50
+ }
51
+
52
+ const app = express();
53
+ app.use(express.json({ limit: '2mb' }));
54
+ Routes.register(app, mcpClient);
55
+
56
+ const server = app.listen(port, () => {
57
+ console.error(`fastweb-http-server listening on http://localhost:${port}`);
58
+ });
59
+
60
+ const shutdown = async (signal: string): Promise<void> => {
61
+ console.error(`Received ${signal}, shutting down ...`);
62
+ server.close();
63
+ await mcpClient.close();
64
+ process.exit(0);
65
+ };
66
+ process.on('SIGINT', () => { void shutdown('SIGINT'); });
67
+ process.on('SIGTERM', () => { void shutdown('SIGTERM'); });
68
+ }
69
+ }
70
+
71
+ ///////////////////////////////////////////////////////////////////////////////
72
+ ///////////////////////////////////////////////////////////////////////////////
73
+ //
74
+ ///////////////////////////////////////////////////////////////////////////////
75
+ ///////////////////////////////////////////////////////////////////////////////
76
+
77
+ async function main(): Promise<void> {
78
+ const program = new Command();
79
+
80
+ program
81
+ .name('fastweb-http-server')
82
+ .description('Persistent HTTP server fronting fastweb-mcp')
83
+ .option('-p, --port <number>', 'Port to listen on', '8787')
84
+ .option('-v, --verbose', 'Enable verbose logging')
85
+ .action(async (opts: { port: string; verbose?: boolean }) => {
86
+ const port = Number.parseInt(opts.port, 10);
87
+ if (Number.isNaN(port) === true) {
88
+ console.error(`Invalid --port: ${opts.port}`);
89
+ process.exit(1);
90
+ }
91
+ await MainHelper.commandStart({ port, verbose: opts.verbose });
92
+ });
93
+
94
+ await program.parseAsync(process.argv);
95
+ }
96
+
97
+ void main();