@cloudflare/sandbox 0.0.0-db09b4d → 0.0.0-dc66e8e

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.
@@ -1,62 +1,44 @@
1
- import { spawn } from "node:child_process";
2
- import { mkdir, rename, unlink, writeFile } from "node:fs/promises";
3
- import { dirname } from "node:path";
1
+ import { randomBytes } from "node:crypto";
4
2
  import { serve } from "bun";
5
-
6
- interface ExecuteRequest {
7
- command: string;
8
- args?: string[];
9
- }
10
-
11
- interface GitCheckoutRequest {
12
- repoUrl: string;
13
- branch?: string;
14
- targetDir?: string;
15
- sessionId?: string;
16
- }
17
-
18
- interface MkdirRequest {
19
- path: string;
20
- recursive?: boolean;
21
- sessionId?: string;
22
- }
23
-
24
- interface WriteFileRequest {
25
- path: string;
26
- content: string;
27
- encoding?: string;
28
- sessionId?: string;
29
- }
30
-
31
- interface DeleteFileRequest {
32
- path: string;
33
- sessionId?: string;
34
- }
35
-
36
- interface RenameFileRequest {
37
- oldPath: string;
38
- newPath: string;
39
- sessionId?: string;
40
- }
41
-
42
- interface MoveFileRequest {
43
- sourcePath: string;
44
- destinationPath: string;
45
- sessionId?: string;
46
- }
47
-
48
- interface SessionData {
49
- sessionId: string;
50
- activeProcess: any | null;
51
- createdAt: Date;
52
- }
3
+ import { handleExecuteRequest, handleStreamingExecuteRequest } from "./handler/exec";
4
+ import {
5
+ handleDeleteFileRequest,
6
+ handleMkdirRequest,
7
+ handleMoveFileRequest,
8
+ handleReadFileRequest,
9
+ handleRenameFileRequest,
10
+ handleWriteFileRequest,
11
+ } from "./handler/file";
12
+ import { handleGitCheckoutRequest } from "./handler/git";
13
+ import {
14
+ handleExposePortRequest,
15
+ handleGetExposedPortsRequest,
16
+ handleProxyRequest,
17
+ handleUnexposePortRequest,
18
+ } from "./handler/ports";
19
+ import {
20
+ handleGetProcessLogsRequest,
21
+ handleGetProcessRequest,
22
+ handleKillAllProcessesRequest,
23
+ handleKillProcessRequest,
24
+ handleListProcessesRequest,
25
+ handleStartProcessRequest,
26
+ handleStreamProcessLogsRequest,
27
+ } from "./handler/process";
28
+ import type { ProcessRecord, SessionData } from "./types";
53
29
 
54
30
  // In-memory session storage (in production, you'd want to use a proper database)
55
31
  const sessions = new Map<string, SessionData>();
56
32
 
57
- // Generate a unique session ID
33
+ // In-memory storage for exposed ports
34
+ const exposedPorts = new Map<number, { name?: string; exposedAt: Date }>();
35
+
36
+ // In-memory process storage - cleared on container restart
37
+ const processes = new Map<string, ProcessRecord>();
38
+
39
+ // Generate a unique session ID using cryptographically secure randomness
58
40
  function generateSessionId(): string {
59
- return `session_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`;
41
+ return `session_${Date.now()}_${randomBytes(6).toString('hex')}`;
60
42
  }
61
43
 
62
44
  // Clean up old sessions (older than 1 hour)
@@ -78,6 +60,8 @@ const server = serve({
78
60
  const url = new URL(req.url);
79
61
  const pathname = url.pathname;
80
62
 
63
+ console.log(`[Container] Incoming ${req.method} request to ${pathname}`);
64
+
81
65
  // Handle CORS
82
66
  const corsHeaders = {
83
67
  "Access-Control-Allow-Headers": "Content-Type, Authorization",
@@ -87,11 +71,13 @@ const server = serve({
87
71
 
88
72
  // Handle preflight requests
89
73
  if (req.method === "OPTIONS") {
74
+ console.log(`[Container] Handling CORS preflight for ${pathname}`);
90
75
  return new Response(null, { headers: corsHeaders, status: 200 });
91
76
  }
92
77
 
93
78
  try {
94
79
  // Handle different routes
80
+ console.log(`[Container] Processing ${req.method} ${pathname}`);
95
81
  switch (pathname) {
96
82
  case "/":
97
83
  return new Response("Hello from Bun server! 🚀", {
@@ -101,51 +87,6 @@ const server = serve({
101
87
  },
102
88
  });
103
89
 
104
- case "/api/hello":
105
- return new Response(
106
- JSON.stringify({
107
- message: "Hello from API!",
108
- timestamp: new Date().toISOString(),
109
- }),
110
- {
111
- headers: {
112
- "Content-Type": "application/json",
113
- ...corsHeaders,
114
- },
115
- }
116
- );
117
-
118
- case "/api/users":
119
- if (req.method === "GET") {
120
- return new Response(
121
- JSON.stringify([
122
- { id: 1, name: "Alice" },
123
- { id: 2, name: "Bob" },
124
- { id: 3, name: "Charlie" },
125
- ]),
126
- {
127
- headers: {
128
- "Content-Type": "application/json",
129
- ...corsHeaders,
130
- },
131
- }
132
- );
133
- } else if (req.method === "POST") {
134
- return new Response(
135
- JSON.stringify({
136
- message: "User created successfully",
137
- method: "POST",
138
- }),
139
- {
140
- headers: {
141
- "Content-Type": "application/json",
142
- ...corsHeaders,
143
- },
144
- }
145
- );
146
- }
147
- break;
148
-
149
90
  case "/api/session/create":
150
91
  if (req.method === "POST") {
151
92
  const sessionId = generateSessionId();
@@ -202,13 +143,13 @@ const server = serve({
202
143
 
203
144
  case "/api/execute":
204
145
  if (req.method === "POST") {
205
- return handleExecuteRequest(req, corsHeaders);
146
+ return handleExecuteRequest(sessions, req, corsHeaders);
206
147
  }
207
148
  break;
208
149
 
209
150
  case "/api/execute/stream":
210
151
  if (req.method === "POST") {
211
- return handleStreamingExecuteRequest(req, corsHeaders);
152
+ return handleStreamingExecuteRequest(sessions, req, corsHeaders);
212
153
  }
213
154
  break;
214
155
 
@@ -263,84 +204,114 @@ const server = serve({
263
204
 
264
205
  case "/api/git/checkout":
265
206
  if (req.method === "POST") {
266
- return handleGitCheckoutRequest(req, corsHeaders);
207
+ return handleGitCheckoutRequest(sessions, req, corsHeaders);
267
208
  }
268
209
  break;
269
210
 
270
- case "/api/git/checkout/stream":
211
+ case "/api/mkdir":
271
212
  if (req.method === "POST") {
272
- return handleStreamingGitCheckoutRequest(req, corsHeaders);
213
+ return handleMkdirRequest(sessions, req, corsHeaders);
273
214
  }
274
215
  break;
275
216
 
276
- case "/api/mkdir":
217
+ case "/api/write":
277
218
  if (req.method === "POST") {
278
- return handleMkdirRequest(req, corsHeaders);
219
+ return handleWriteFileRequest(req, corsHeaders);
279
220
  }
280
221
  break;
281
222
 
282
- case "/api/mkdir/stream":
223
+ case "/api/read":
283
224
  if (req.method === "POST") {
284
- return handleStreamingMkdirRequest(req, corsHeaders);
225
+ return handleReadFileRequest(req, corsHeaders);
285
226
  }
286
227
  break;
287
228
 
288
- case "/api/write":
229
+ case "/api/delete":
289
230
  if (req.method === "POST") {
290
- return handleWriteFileRequest(req, corsHeaders);
231
+ return handleDeleteFileRequest(req, corsHeaders);
291
232
  }
292
233
  break;
293
234
 
294
- case "/api/write/stream":
235
+ case "/api/rename":
295
236
  if (req.method === "POST") {
296
- return handleStreamingWriteFileRequest(req, corsHeaders);
237
+ return handleRenameFileRequest(req, corsHeaders);
297
238
  }
298
239
  break;
299
240
 
300
- case "/api/delete":
241
+ case "/api/move":
301
242
  if (req.method === "POST") {
302
- return handleDeleteFileRequest(req, corsHeaders);
243
+ return handleMoveFileRequest(req, corsHeaders);
303
244
  }
304
245
  break;
305
246
 
306
- case "/api/delete/stream":
247
+ case "/api/expose-port":
307
248
  if (req.method === "POST") {
308
- return handleStreamingDeleteFileRequest(req, corsHeaders);
249
+ return handleExposePortRequest(exposedPorts, req, corsHeaders);
309
250
  }
310
251
  break;
311
252
 
312
- case "/api/rename":
313
- if (req.method === "POST") {
314
- return handleRenameFileRequest(req, corsHeaders);
253
+ case "/api/unexpose-port":
254
+ if (req.method === "DELETE") {
255
+ return handleUnexposePortRequest(exposedPorts, req, corsHeaders);
315
256
  }
316
257
  break;
317
258
 
318
- case "/api/rename/stream":
319
- if (req.method === "POST") {
320
- return handleStreamingRenameFileRequest(req, corsHeaders);
259
+ case "/api/exposed-ports":
260
+ if (req.method === "GET") {
261
+ return handleGetExposedPortsRequest(exposedPorts, req, corsHeaders);
321
262
  }
322
263
  break;
323
264
 
324
- case "/api/move":
265
+ case "/api/process/start":
325
266
  if (req.method === "POST") {
326
- return handleMoveFileRequest(req, corsHeaders);
267
+ return handleStartProcessRequest(processes, req, corsHeaders);
327
268
  }
328
269
  break;
329
270
 
330
- case "/api/move/stream":
331
- if (req.method === "POST") {
332
- return handleStreamingMoveFileRequest(req, corsHeaders);
271
+ case "/api/process/list":
272
+ if (req.method === "GET") {
273
+ return handleListProcessesRequest(processes, req, corsHeaders);
274
+ }
275
+ break;
276
+
277
+ case "/api/process/kill-all":
278
+ if (req.method === "DELETE") {
279
+ return handleKillAllProcessesRequest(processes, req, corsHeaders);
333
280
  }
334
281
  break;
335
282
 
336
283
  default:
284
+ // Handle dynamic routes for individual processes
285
+ if (pathname.startsWith("/api/process/")) {
286
+ const segments = pathname.split('/');
287
+ if (segments.length >= 4) {
288
+ const processId = segments[3];
289
+ const action = segments[4]; // Optional: logs, stream, etc.
290
+
291
+ if (!action && req.method === "GET") {
292
+ return handleGetProcessRequest(processes, req, corsHeaders, processId);
293
+ } else if (!action && req.method === "DELETE") {
294
+ return handleKillProcessRequest(processes, req, corsHeaders, processId);
295
+ } else if (action === "logs" && req.method === "GET") {
296
+ return handleGetProcessLogsRequest(processes, req, corsHeaders, processId);
297
+ } else if (action === "stream" && req.method === "GET") {
298
+ return handleStreamProcessLogsRequest(processes, req, corsHeaders, processId);
299
+ }
300
+ }
301
+ }
302
+ // Check if this is a proxy request for an exposed port
303
+ if (pathname.startsWith("/proxy/")) {
304
+ return handleProxyRequest(exposedPorts, req, corsHeaders);
305
+ }
306
+
307
+ console.log(`[Container] Route not found: ${pathname}`);
337
308
  return new Response("Not Found", {
338
309
  headers: corsHeaders,
339
310
  status: 404,
340
311
  });
341
312
  }
342
313
  } catch (error) {
343
- console.error("[Server] Error handling request:", error);
314
+ console.error(`[Container] Error handling ${req.method} ${pathname}:`, error);
344
315
  return new Response(
345
316
  JSON.stringify({
346
317
  error: "Internal server error",
@@ -356,2265 +327,35 @@ const server = serve({
356
327
  );
357
328
  }
358
329
  },
330
+ hostname: "0.0.0.0",
359
331
  port: 3000,
360
- } as any);
361
-
362
- async function handleExecuteRequest(
363
- req: Request,
364
- corsHeaders: Record<string, string>
365
- ): Promise<Response> {
366
- try {
367
- const body = (await req.json()) as ExecuteRequest & { sessionId?: string };
368
- const { command, args = [], sessionId } = body;
369
-
370
- if (!command || typeof command !== "string") {
371
- return new Response(
372
- JSON.stringify({
373
- error: "Command is required and must be a string",
374
- }),
375
- {
376
- headers: {
377
- "Content-Type": "application/json",
378
- ...corsHeaders,
379
- },
380
- status: 400,
381
- }
382
- );
383
- }
384
-
385
- // Basic safety check - prevent dangerous commands
386
- const dangerousCommands = [
387
- "rm",
388
- "rmdir",
389
- "del",
390
- "format",
391
- "shutdown",
392
- "reboot",
393
- ];
394
- const lowerCommand = command.toLowerCase();
395
-
396
- if (
397
- dangerousCommands.some((dangerous) => lowerCommand.includes(dangerous))
398
- ) {
399
- return new Response(
400
- JSON.stringify({
401
- error: "Dangerous command not allowed",
402
- }),
403
- {
404
- headers: {
405
- "Content-Type": "application/json",
406
- ...corsHeaders,
407
- },
408
- status: 400,
409
- }
410
- );
411
- }
412
-
413
- console.log(`[Server] Executing command: ${command} ${args.join(" ")}`);
414
-
415
- const result = await executeCommand(command, args, sessionId);
416
-
417
- return new Response(
418
- JSON.stringify({
419
- args,
420
- command,
421
- exitCode: result.exitCode,
422
- stderr: result.stderr,
423
- stdout: result.stdout,
424
- success: result.success,
425
- timestamp: new Date().toISOString(),
426
- }),
427
- {
428
- headers: {
429
- "Content-Type": "application/json",
430
- ...corsHeaders,
431
- },
432
- }
433
- );
434
- } catch (error) {
435
- console.error("[Server] Error in handleExecuteRequest:", error);
436
- return new Response(
437
- JSON.stringify({
438
- error: "Failed to execute command",
439
- message: error instanceof Error ? error.message : "Unknown error",
440
- }),
441
- {
442
- headers: {
443
- "Content-Type": "application/json",
444
- ...corsHeaders,
445
- },
446
- status: 500,
447
- }
448
- );
449
- }
450
- }
451
-
452
- async function handleStreamingExecuteRequest(
453
- req: Request,
454
- corsHeaders: Record<string, string>
455
- ): Promise<Response> {
456
- try {
457
- const body = (await req.json()) as ExecuteRequest & { sessionId?: string };
458
- const { command, args = [], sessionId } = body;
459
-
460
- if (!command || typeof command !== "string") {
461
- return new Response(
462
- JSON.stringify({
463
- error: "Command is required and must be a string",
464
- }),
465
- {
466
- headers: {
467
- "Content-Type": "application/json",
468
- ...corsHeaders,
469
- },
470
- status: 400,
471
- }
472
- );
473
- }
474
-
475
- // Basic safety check - prevent dangerous commands
476
- const dangerousCommands = [
477
- "rm",
478
- "rmdir",
479
- "del",
480
- "format",
481
- "shutdown",
482
- "reboot",
483
- ];
484
- const lowerCommand = command.toLowerCase();
485
-
486
- if (
487
- dangerousCommands.some((dangerous) => lowerCommand.includes(dangerous))
488
- ) {
489
- return new Response(
490
- JSON.stringify({
491
- error: "Dangerous command not allowed",
492
- }),
493
- {
494
- headers: {
495
- "Content-Type": "application/json",
496
- ...corsHeaders,
497
- },
498
- status: 400,
499
- }
500
- );
501
- }
502
-
503
- console.log(
504
- `[Server] Executing streaming command: ${command} ${args.join(" ")}`
505
- );
506
-
507
- const stream = new ReadableStream({
508
- start(controller) {
509
- const child = spawn(command, args, {
510
- shell: true,
511
- stdio: ["pipe", "pipe", "pipe"],
512
- });
513
-
514
- // Store the process reference for cleanup if sessionId is provided
515
- if (sessionId && sessions.has(sessionId)) {
516
- const session = sessions.get(sessionId)!;
517
- session.activeProcess = child;
518
- }
519
-
520
- let stdout = "";
521
- let stderr = "";
522
-
523
- // Send command start event
524
- controller.enqueue(
525
- new TextEncoder().encode(
526
- `data: ${JSON.stringify({
527
- args,
528
- command,
529
- timestamp: new Date().toISOString(),
530
- type: "command_start",
531
- })}\n\n`
532
- )
533
- );
534
-
535
- child.stdout?.on("data", (data) => {
536
- const output = data.toString();
537
- stdout += output;
538
-
539
- // Send real-time output
540
- controller.enqueue(
541
- new TextEncoder().encode(
542
- `data: ${JSON.stringify({
543
- command,
544
- data: output,
545
- stream: "stdout",
546
- type: "output",
547
- })}\n\n`
548
- )
549
- );
550
- });
551
-
552
- child.stderr?.on("data", (data) => {
553
- const output = data.toString();
554
- stderr += output;
555
-
556
- // Send real-time error output
557
- controller.enqueue(
558
- new TextEncoder().encode(
559
- `data: ${JSON.stringify({
560
- command,
561
- data: output,
562
- stream: "stderr",
563
- type: "output",
564
- })}\n\n`
565
- )
566
- );
567
- });
568
-
569
- child.on("close", (code) => {
570
- // Clear the active process reference
571
- if (sessionId && sessions.has(sessionId)) {
572
- const session = sessions.get(sessionId)!;
573
- session.activeProcess = null;
574
- }
575
-
576
- console.log(
577
- `[Server] Command completed: ${command}, Exit code: ${code}`
578
- );
579
-
580
- // Send command completion event
581
- controller.enqueue(
582
- new TextEncoder().encode(
583
- `data: ${JSON.stringify({
584
- args,
585
- command,
586
- exitCode: code,
587
- stderr,
588
- stdout,
589
- success: code === 0,
590
- timestamp: new Date().toISOString(),
591
- type: "command_complete",
592
- })}\n\n`
593
- )
594
- );
595
-
596
- controller.close();
597
- });
598
-
599
- child.on("error", (error) => {
600
- // Clear the active process reference
601
- if (sessionId && sessions.has(sessionId)) {
602
- const session = sessions.get(sessionId)!;
603
- session.activeProcess = null;
604
- }
605
-
606
- controller.enqueue(
607
- new TextEncoder().encode(
608
- `data: ${JSON.stringify({
609
- args,
610
- command,
611
- error: error.message,
612
- type: "error",
613
- })}\n\n`
614
- )
615
- );
616
-
617
- controller.close();
618
- });
619
- },
620
- });
621
-
622
- return new Response(stream, {
623
- headers: {
624
- "Cache-Control": "no-cache",
625
- Connection: "keep-alive",
626
- "Content-Type": "text/event-stream",
627
- ...corsHeaders,
628
- },
629
- });
630
- } catch (error) {
631
- console.error("[Server] Error in handleStreamingExecuteRequest:", error);
632
- return new Response(
633
- JSON.stringify({
634
- error: "Failed to execute streaming command",
635
- message: error instanceof Error ? error.message : "Unknown error",
636
- }),
637
- {
638
- headers: {
639
- "Content-Type": "application/json",
640
- ...corsHeaders,
641
- },
642
- status: 500,
643
- }
644
- );
645
- }
646
- }
647
-
648
- async function handleGitCheckoutRequest(
649
- req: Request,
650
- corsHeaders: Record<string, string>
651
- ): Promise<Response> {
652
- try {
653
- const body = (await req.json()) as GitCheckoutRequest;
654
- const { repoUrl, branch = "main", targetDir, sessionId } = body;
655
-
656
- if (!repoUrl || typeof repoUrl !== "string") {
657
- return new Response(
658
- JSON.stringify({
659
- error: "Repository URL is required and must be a string",
660
- }),
661
- {
662
- headers: {
663
- "Content-Type": "application/json",
664
- ...corsHeaders,
665
- },
666
- status: 400,
667
- }
668
- );
669
- }
670
-
671
- // Validate repository URL format
672
- const urlPattern =
673
- /^(https?:\/\/|git@|ssh:\/\/).*\.git$|^https?:\/\/.*\/.*$/;
674
- if (!urlPattern.test(repoUrl)) {
675
- return new Response(
676
- JSON.stringify({
677
- error: "Invalid repository URL format",
678
- }),
679
- {
680
- headers: {
681
- "Content-Type": "application/json",
682
- ...corsHeaders,
683
- },
684
- status: 400,
685
- }
686
- );
687
- }
688
-
689
- // Generate target directory if not provided
690
- const checkoutDir =
691
- targetDir ||
692
- `repo_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`;
693
-
694
- console.log(
695
- `[Server] Checking out repository: ${repoUrl} to ${checkoutDir}`
696
- );
697
-
698
- const result = await executeGitCheckout(
699
- repoUrl,
700
- branch,
701
- checkoutDir,
702
- sessionId
703
- );
704
-
705
- return new Response(
706
- JSON.stringify({
707
- branch,
708
- exitCode: result.exitCode,
709
- repoUrl,
710
- stderr: result.stderr,
711
- stdout: result.stdout,
712
- success: result.success,
713
- targetDir: checkoutDir,
714
- timestamp: new Date().toISOString(),
715
- }),
716
- {
717
- headers: {
718
- "Content-Type": "application/json",
719
- ...corsHeaders,
720
- },
721
- }
722
- );
723
- } catch (error) {
724
- console.error("[Server] Error in handleGitCheckoutRequest:", error);
725
- return new Response(
726
- JSON.stringify({
727
- error: "Failed to checkout repository",
728
- message: error instanceof Error ? error.message : "Unknown error",
729
- }),
730
- {
731
- headers: {
732
- "Content-Type": "application/json",
733
- ...corsHeaders,
734
- },
735
- status: 500,
736
- }
737
- );
738
- }
739
- }
740
-
741
- async function handleStreamingGitCheckoutRequest(
742
- req: Request,
743
- corsHeaders: Record<string, string>
744
- ): Promise<Response> {
745
- try {
746
- const body = (await req.json()) as GitCheckoutRequest;
747
- const { repoUrl, branch = "main", targetDir, sessionId } = body;
748
-
749
- if (!repoUrl || typeof repoUrl !== "string") {
750
- return new Response(
751
- JSON.stringify({
752
- error: "Repository URL is required and must be a string",
753
- }),
754
- {
755
- headers: {
756
- "Content-Type": "application/json",
757
- ...corsHeaders,
758
- },
759
- status: 400,
760
- }
761
- );
762
- }
763
-
764
- // Validate repository URL format
765
- const urlPattern =
766
- /^(https?:\/\/|git@|ssh:\/\/).*\.git$|^https?:\/\/.*\/.*$/;
767
- if (!urlPattern.test(repoUrl)) {
768
- return new Response(
769
- JSON.stringify({
770
- error: "Invalid repository URL format",
771
- }),
772
- {
773
- headers: {
774
- "Content-Type": "application/json",
775
- ...corsHeaders,
776
- },
777
- status: 400,
778
- }
779
- );
780
- }
781
-
782
- // Generate target directory if not provided
783
- const checkoutDir =
784
- targetDir ||
785
- `repo_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`;
786
-
787
- console.log(
788
- `[Server] Checking out repository: ${repoUrl} to ${checkoutDir}`
789
- );
790
-
791
- const stream = new ReadableStream({
792
- start(controller) {
793
- const child = spawn(
794
- "git",
795
- ["clone", "-b", branch, repoUrl, checkoutDir],
796
- {
797
- shell: true,
798
- stdio: ["pipe", "pipe", "pipe"],
799
- }
800
- );
801
-
802
- // Store the process reference for cleanup if sessionId is provided
803
- if (sessionId && sessions.has(sessionId)) {
804
- const session = sessions.get(sessionId)!;
805
- session.activeProcess = child;
806
- }
807
-
808
- let stdout = "";
809
- let stderr = "";
810
-
811
- // Send command start event
812
- controller.enqueue(
813
- new TextEncoder().encode(
814
- `data: ${JSON.stringify({
815
- args: [branch, repoUrl, checkoutDir],
816
- command: "git clone",
817
- timestamp: new Date().toISOString(),
818
- type: "command_start",
819
- })}\n\n`
820
- )
821
- );
822
-
823
- child.stdout?.on("data", (data) => {
824
- const output = data.toString();
825
- stdout += output;
826
-
827
- // Send real-time output
828
- controller.enqueue(
829
- new TextEncoder().encode(
830
- `data: ${JSON.stringify({
831
- command: "git clone",
832
- data: output,
833
- stream: "stdout",
834
- type: "output",
835
- })}\n\n`
836
- )
837
- );
838
- });
839
-
840
- child.stderr?.on("data", (data) => {
841
- const output = data.toString();
842
- stderr += output;
843
-
844
- // Send real-time error output
845
- controller.enqueue(
846
- new TextEncoder().encode(
847
- `data: ${JSON.stringify({
848
- command: "git clone",
849
- data: output,
850
- stream: "stderr",
851
- type: "output",
852
- })}\n\n`
853
- )
854
- );
855
- });
856
-
857
- child.on("close", (code) => {
858
- // Clear the active process reference
859
- if (sessionId && sessions.has(sessionId)) {
860
- const session = sessions.get(sessionId)!;
861
- session.activeProcess = null;
862
- }
863
-
864
- console.log(
865
- `[Server] Command completed: git clone, Exit code: ${code}`
866
- );
867
-
868
- // Send command completion event
869
- controller.enqueue(
870
- new TextEncoder().encode(
871
- `data: ${JSON.stringify({
872
- args: [branch, repoUrl, checkoutDir],
873
- command: "git clone",
874
- exitCode: code,
875
- stderr,
876
- stdout,
877
- success: code === 0,
878
- timestamp: new Date().toISOString(),
879
- type: "command_complete",
880
- })}\n\n`
881
- )
882
- );
883
-
884
- controller.close();
885
- });
886
-
887
- child.on("error", (error) => {
888
- // Clear the active process reference
889
- if (sessionId && sessions.has(sessionId)) {
890
- const session = sessions.get(sessionId)!;
891
- session.activeProcess = null;
892
- }
893
-
894
- controller.enqueue(
895
- new TextEncoder().encode(
896
- `data: ${JSON.stringify({
897
- args: [branch, repoUrl, checkoutDir],
898
- command: "git clone",
899
- error: error.message,
900
- type: "error",
901
- })}\n\n`
902
- )
903
- );
904
-
905
- controller.close();
906
- });
907
- },
908
- });
909
-
910
- return new Response(stream, {
911
- headers: {
912
- "Cache-Control": "no-cache",
913
- Connection: "keep-alive",
914
- "Content-Type": "text/event-stream",
915
- ...corsHeaders,
916
- },
917
- });
918
- } catch (error) {
919
- console.error(
920
- "[Server] Error in handleStreamingGitCheckoutRequest:",
921
- error
922
- );
923
- return new Response(
924
- JSON.stringify({
925
- error: "Failed to checkout repository",
926
- message: error instanceof Error ? error.message : "Unknown error",
927
- }),
928
- {
929
- headers: {
930
- "Content-Type": "application/json",
931
- ...corsHeaders,
932
- },
933
- status: 500,
934
- }
935
- );
936
- }
937
- }
938
-
939
- async function handleMkdirRequest(
940
- req: Request,
941
- corsHeaders: Record<string, string>
942
- ): Promise<Response> {
943
- try {
944
- const body = (await req.json()) as MkdirRequest;
945
- const { path, recursive = false, sessionId } = body;
946
-
947
- if (!path || typeof path !== "string") {
948
- return new Response(
949
- JSON.stringify({
950
- error: "Path is required and must be a string",
951
- }),
952
- {
953
- headers: {
954
- "Content-Type": "application/json",
955
- ...corsHeaders,
956
- },
957
- status: 400,
958
- }
959
- );
960
- }
961
-
962
- // Basic safety check - prevent dangerous paths
963
- const dangerousPatterns = [
964
- /^\/$/, // Root directory
965
- /^\/etc/, // System directories
966
- /^\/var/, // System directories
967
- /^\/usr/, // System directories
968
- /^\/bin/, // System directories
969
- /^\/sbin/, // System directories
970
- /^\/boot/, // System directories
971
- /^\/dev/, // System directories
972
- /^\/proc/, // System directories
973
- /^\/sys/, // System directories
974
- /^\/tmp\/\.\./, // Path traversal attempts
975
- /\.\./, // Path traversal attempts
976
- ];
977
-
978
- if (dangerousPatterns.some((pattern) => pattern.test(path))) {
979
- return new Response(
980
- JSON.stringify({
981
- error: "Dangerous path not allowed",
982
- }),
983
- {
984
- headers: {
985
- "Content-Type": "application/json",
986
- ...corsHeaders,
987
- },
988
- status: 400,
989
- }
990
- );
991
- }
992
-
993
- console.log(
994
- `[Server] Creating directory: ${path} (recursive: ${recursive})`
995
- );
996
-
997
- const result = await executeMkdir(path, recursive, sessionId);
998
-
999
- return new Response(
1000
- JSON.stringify({
1001
- exitCode: result.exitCode,
1002
- path,
1003
- recursive,
1004
- stderr: result.stderr,
1005
- stdout: result.stdout,
1006
- success: result.success,
1007
- timestamp: new Date().toISOString(),
1008
- }),
1009
- {
1010
- headers: {
1011
- "Content-Type": "application/json",
1012
- ...corsHeaders,
1013
- },
1014
- }
1015
- );
1016
- } catch (error) {
1017
- console.error("[Server] Error in handleMkdirRequest:", error);
1018
- return new Response(
1019
- JSON.stringify({
1020
- error: "Failed to create directory",
1021
- message: error instanceof Error ? error.message : "Unknown error",
1022
- }),
1023
- {
1024
- headers: {
1025
- "Content-Type": "application/json",
1026
- ...corsHeaders,
1027
- },
1028
- status: 500,
1029
- }
1030
- );
1031
- }
1032
- }
1033
-
1034
- async function handleStreamingMkdirRequest(
1035
- req: Request,
1036
- corsHeaders: Record<string, string>
1037
- ): Promise<Response> {
1038
- try {
1039
- const body = (await req.json()) as MkdirRequest;
1040
- const { path, recursive = false, sessionId } = body;
1041
-
1042
- if (!path || typeof path !== "string") {
1043
- return new Response(
1044
- JSON.stringify({
1045
- error: "Path is required and must be a string",
1046
- }),
1047
- {
1048
- headers: {
1049
- "Content-Type": "application/json",
1050
- ...corsHeaders,
1051
- },
1052
- status: 400,
1053
- }
1054
- );
1055
- }
1056
-
1057
- // Basic safety check - prevent dangerous paths
1058
- const dangerousPatterns = [
1059
- /^\/$/, // Root directory
1060
- /^\/etc/, // System directories
1061
- /^\/var/, // System directories
1062
- /^\/usr/, // System directories
1063
- /^\/bin/, // System directories
1064
- /^\/sbin/, // System directories
1065
- /^\/boot/, // System directories
1066
- /^\/dev/, // System directories
1067
- /^\/proc/, // System directories
1068
- /^\/sys/, // System directories
1069
- /^\/tmp\/\.\./, // Path traversal attempts
1070
- /\.\./, // Path traversal attempts
1071
- ];
1072
-
1073
- if (dangerousPatterns.some((pattern) => pattern.test(path))) {
1074
- return new Response(
1075
- JSON.stringify({
1076
- error: "Dangerous path not allowed",
1077
- }),
1078
- {
1079
- headers: {
1080
- "Content-Type": "application/json",
1081
- ...corsHeaders,
1082
- },
1083
- status: 400,
1084
- }
1085
- );
1086
- }
1087
-
1088
- console.log(
1089
- `[Server] Creating directory: ${path} (recursive: ${recursive})`
1090
- );
1091
-
1092
- const stream = new ReadableStream({
1093
- start(controller) {
1094
- const args = recursive ? ["-p", path] : [path];
1095
- const child = spawn("mkdir", args, {
1096
- shell: true,
1097
- stdio: ["pipe", "pipe", "pipe"],
1098
- });
1099
-
1100
- // Store the process reference for cleanup if sessionId is provided
1101
- if (sessionId && sessions.has(sessionId)) {
1102
- const session = sessions.get(sessionId)!;
1103
- session.activeProcess = child;
1104
- }
1105
-
1106
- let stdout = "";
1107
- let stderr = "";
1108
-
1109
- // Send command start event
1110
- controller.enqueue(
1111
- new TextEncoder().encode(
1112
- `data: ${JSON.stringify({
1113
- args,
1114
- command: "mkdir",
1115
- timestamp: new Date().toISOString(),
1116
- type: "command_start",
1117
- })}\n\n`
1118
- )
1119
- );
1120
-
1121
- child.stdout?.on("data", (data) => {
1122
- const output = data.toString();
1123
- stdout += output;
1124
-
1125
- // Send real-time output
1126
- controller.enqueue(
1127
- new TextEncoder().encode(
1128
- `data: ${JSON.stringify({
1129
- command: "mkdir",
1130
- data: output,
1131
- stream: "stdout",
1132
- type: "output",
1133
- })}\n\n`
1134
- )
1135
- );
1136
- });
1137
-
1138
- child.stderr?.on("data", (data) => {
1139
- const output = data.toString();
1140
- stderr += output;
1141
-
1142
- // Send real-time error output
1143
- controller.enqueue(
1144
- new TextEncoder().encode(
1145
- `data: ${JSON.stringify({
1146
- command: "mkdir",
1147
- data: output,
1148
- stream: "stderr",
1149
- type: "output",
1150
- })}\n\n`
1151
- )
1152
- );
1153
- });
1154
-
1155
- child.on("close", (code) => {
1156
- // Clear the active process reference
1157
- if (sessionId && sessions.has(sessionId)) {
1158
- const session = sessions.get(sessionId)!;
1159
- session.activeProcess = null;
1160
- }
1161
-
1162
- console.log(`[Server] Command completed: mkdir, Exit code: ${code}`);
1163
-
1164
- // Send command completion event
1165
- controller.enqueue(
1166
- new TextEncoder().encode(
1167
- `data: ${JSON.stringify({
1168
- args,
1169
- command: "mkdir",
1170
- exitCode: code,
1171
- stderr,
1172
- stdout,
1173
- success: code === 0,
1174
- timestamp: new Date().toISOString(),
1175
- type: "command_complete",
1176
- })}\n\n`
1177
- )
1178
- );
1179
-
1180
- controller.close();
1181
- });
1182
-
1183
- child.on("error", (error) => {
1184
- // Clear the active process reference
1185
- if (sessionId && sessions.has(sessionId)) {
1186
- const session = sessions.get(sessionId)!;
1187
- session.activeProcess = null;
1188
- }
1189
-
1190
- controller.enqueue(
1191
- new TextEncoder().encode(
1192
- `data: ${JSON.stringify({
1193
- args,
1194
- command: "mkdir",
1195
- error: error.message,
1196
- type: "error",
1197
- })}\n\n`
1198
- )
1199
- );
1200
-
1201
- controller.close();
1202
- });
1203
- },
1204
- });
1205
-
1206
- return new Response(stream, {
1207
- headers: {
1208
- "Cache-Control": "no-cache",
1209
- Connection: "keep-alive",
1210
- "Content-Type": "text/event-stream",
1211
- ...corsHeaders,
1212
- },
1213
- });
1214
- } catch (error) {
1215
- console.error("[Server] Error in handleStreamingMkdirRequest:", error);
1216
- return new Response(
1217
- JSON.stringify({
1218
- error: "Failed to create directory",
1219
- message: error instanceof Error ? error.message : "Unknown error",
1220
- }),
1221
- {
1222
- headers: {
1223
- "Content-Type": "application/json",
1224
- ...corsHeaders,
1225
- },
1226
- status: 500,
1227
- }
1228
- );
1229
- }
1230
- }
1231
-
1232
- async function handleWriteFileRequest(
1233
- req: Request,
1234
- corsHeaders: Record<string, string>
1235
- ): Promise<Response> {
1236
- try {
1237
- const body = (await req.json()) as WriteFileRequest;
1238
- const { path, content, encoding = "utf-8", sessionId } = body;
1239
-
1240
- if (!path || typeof path !== "string") {
1241
- return new Response(
1242
- JSON.stringify({
1243
- error: "Path is required and must be a string",
1244
- }),
1245
- {
1246
- headers: {
1247
- "Content-Type": "application/json",
1248
- ...corsHeaders,
1249
- },
1250
- status: 400,
1251
- }
1252
- );
1253
- }
1254
-
1255
- // Basic safety check - prevent dangerous paths
1256
- const dangerousPatterns = [
1257
- /^\/$/, // Root directory
1258
- /^\/etc/, // System directories
1259
- /^\/var/, // System directories
1260
- /^\/usr/, // System directories
1261
- /^\/bin/, // System directories
1262
- /^\/sbin/, // System directories
1263
- /^\/boot/, // System directories
1264
- /^\/dev/, // System directories
1265
- /^\/proc/, // System directories
1266
- /^\/sys/, // System directories
1267
- /^\/tmp\/\.\./, // Path traversal attempts
1268
- /\.\./, // Path traversal attempts
1269
- ];
1270
-
1271
- if (dangerousPatterns.some((pattern) => pattern.test(path))) {
1272
- return new Response(
1273
- JSON.stringify({
1274
- error: "Dangerous path not allowed",
1275
- }),
1276
- {
1277
- headers: {
1278
- "Content-Type": "application/json",
1279
- ...corsHeaders,
1280
- },
1281
- status: 400,
1282
- }
1283
- );
1284
- }
1285
-
1286
- console.log(
1287
- `[Server] Writing file: ${path} (content length: ${content.length})`
1288
- );
1289
-
1290
- const result = await executeWriteFile(path, content, encoding, sessionId);
1291
-
1292
- return new Response(
1293
- JSON.stringify({
1294
- exitCode: result.exitCode,
1295
- path,
1296
- success: result.success,
1297
- timestamp: new Date().toISOString(),
1298
- }),
1299
- {
1300
- headers: {
1301
- "Content-Type": "application/json",
1302
- ...corsHeaders,
1303
- },
1304
- }
1305
- );
1306
- } catch (error) {
1307
- console.error("[Server] Error in handleWriteFileRequest:", error);
1308
- return new Response(
1309
- JSON.stringify({
1310
- error: "Failed to write file",
1311
- message: error instanceof Error ? error.message : "Unknown error",
1312
- }),
1313
- {
1314
- headers: {
1315
- "Content-Type": "application/json",
1316
- ...corsHeaders,
1317
- },
1318
- status: 500,
1319
- }
1320
- );
1321
- }
1322
- }
1323
-
1324
- async function handleStreamingWriteFileRequest(
1325
- req: Request,
1326
- corsHeaders: Record<string, string>
1327
- ): Promise<Response> {
1328
- try {
1329
- const body = (await req.json()) as WriteFileRequest;
1330
- const { path, content, encoding = "utf-8", sessionId } = body;
1331
-
1332
- if (!path || typeof path !== "string") {
1333
- return new Response(
1334
- JSON.stringify({
1335
- error: "Path is required and must be a string",
1336
- }),
1337
- {
1338
- headers: {
1339
- "Content-Type": "application/json",
1340
- ...corsHeaders,
1341
- },
1342
- status: 400,
1343
- }
1344
- );
1345
- }
1346
-
1347
- // Basic safety check - prevent dangerous paths
1348
- const dangerousPatterns = [
1349
- /^\/$/, // Root directory
1350
- /^\/etc/, // System directories
1351
- /^\/var/, // System directories
1352
- /^\/usr/, // System directories
1353
- /^\/bin/, // System directories
1354
- /^\/sbin/, // System directories
1355
- /^\/boot/, // System directories
1356
- /^\/dev/, // System directories
1357
- /^\/proc/, // System directories
1358
- /^\/sys/, // System directories
1359
- /^\/tmp\/\.\./, // Path traversal attempts
1360
- /\.\./, // Path traversal attempts
1361
- ];
1362
-
1363
- if (dangerousPatterns.some((pattern) => pattern.test(path))) {
1364
- return new Response(
1365
- JSON.stringify({
1366
- error: "Dangerous path not allowed",
1367
- }),
1368
- {
1369
- headers: {
1370
- "Content-Type": "application/json",
1371
- ...corsHeaders,
1372
- },
1373
- status: 400,
1374
- }
1375
- );
1376
- }
1377
-
1378
- console.log(
1379
- `[Server] Writing file (streaming): ${path} (content length: ${content.length})`
1380
- );
1381
-
1382
- const stream = new ReadableStream({
1383
- start(controller) {
1384
- (async () => {
1385
- try {
1386
- // Send command start event
1387
- controller.enqueue(
1388
- new TextEncoder().encode(
1389
- `data: ${JSON.stringify({
1390
- path,
1391
- timestamp: new Date().toISOString(),
1392
- type: "command_start",
1393
- })}\n\n`
1394
- )
1395
- );
1396
-
1397
- // Ensure the directory exists
1398
- const dir = dirname(path);
1399
- if (dir !== ".") {
1400
- await mkdir(dir, { recursive: true });
1401
-
1402
- // Send directory creation event
1403
- controller.enqueue(
1404
- new TextEncoder().encode(
1405
- `data: ${JSON.stringify({
1406
- message: `Created directory: ${dir}`,
1407
- type: "output",
1408
- })}\n\n`
1409
- )
1410
- );
1411
- }
1412
-
1413
- // Write the file
1414
- await writeFile(path, content, {
1415
- encoding: encoding as BufferEncoding,
1416
- });
1417
-
1418
- console.log(`[Server] File written successfully: ${path}`);
1419
-
1420
- // Send command completion event
1421
- controller.enqueue(
1422
- new TextEncoder().encode(
1423
- `data: ${JSON.stringify({
1424
- path,
1425
- success: true,
1426
- timestamp: new Date().toISOString(),
1427
- type: "command_complete",
1428
- })}\n\n`
1429
- )
1430
- );
1431
-
1432
- controller.close();
1433
- } catch (error) {
1434
- console.error(`[Server] Error writing file: ${path}`, error);
1435
-
1436
- controller.enqueue(
1437
- new TextEncoder().encode(
1438
- `data: ${JSON.stringify({
1439
- error:
1440
- error instanceof Error ? error.message : "Unknown error",
1441
- path,
1442
- type: "error",
1443
- })}\n\n`
1444
- )
1445
- );
1446
-
1447
- controller.close();
1448
- }
1449
- })();
1450
- },
1451
- });
1452
-
1453
- return new Response(stream, {
1454
- headers: {
1455
- "Cache-Control": "no-cache",
1456
- Connection: "keep-alive",
1457
- "Content-Type": "text/event-stream",
1458
- ...corsHeaders,
1459
- },
1460
- });
1461
- } catch (error) {
1462
- console.error("[Server] Error in handleStreamingWriteFileRequest:", error);
1463
- return new Response(
1464
- JSON.stringify({
1465
- error: "Failed to write file",
1466
- message: error instanceof Error ? error.message : "Unknown error",
1467
- }),
1468
- {
1469
- headers: {
1470
- "Content-Type": "application/json",
1471
- ...corsHeaders,
1472
- },
1473
- status: 500,
1474
- }
1475
- );
1476
- }
1477
- }
1478
-
1479
- async function handleDeleteFileRequest(
1480
- req: Request,
1481
- corsHeaders: Record<string, string>
1482
- ): Promise<Response> {
1483
- try {
1484
- const body = (await req.json()) as DeleteFileRequest;
1485
- const { path, sessionId } = body;
1486
-
1487
- if (!path || typeof path !== "string") {
1488
- return new Response(
1489
- JSON.stringify({
1490
- error: "Path is required and must be a string",
1491
- }),
1492
- {
1493
- headers: {
1494
- "Content-Type": "application/json",
1495
- ...corsHeaders,
1496
- },
1497
- status: 400,
1498
- }
1499
- );
1500
- }
1501
-
1502
- // Basic safety check - prevent dangerous paths
1503
- const dangerousPatterns = [
1504
- /^\/$/, // Root directory
1505
- /^\/etc/, // System directories
1506
- /^\/var/, // System directories
1507
- /^\/usr/, // System directories
1508
- /^\/bin/, // System directories
1509
- /^\/sbin/, // System directories
1510
- /^\/boot/, // System directories
1511
- /^\/dev/, // System directories
1512
- /^\/proc/, // System directories
1513
- /^\/sys/, // System directories
1514
- /^\/tmp\/\.\./, // Path traversal attempts
1515
- /\.\./, // Path traversal attempts
1516
- ];
1517
-
1518
- if (dangerousPatterns.some((pattern) => pattern.test(path))) {
1519
- return new Response(
1520
- JSON.stringify({
1521
- error: "Dangerous path not allowed",
1522
- }),
1523
- {
1524
- headers: {
1525
- "Content-Type": "application/json",
1526
- ...corsHeaders,
1527
- },
1528
- status: 400,
1529
- }
1530
- );
1531
- }
1532
-
1533
- console.log(`[Server] Deleting file: ${path}`);
1534
-
1535
- const result = await executeDeleteFile(path, sessionId);
1536
-
1537
- return new Response(
1538
- JSON.stringify({
1539
- exitCode: result.exitCode,
1540
- path,
1541
- success: result.success,
1542
- timestamp: new Date().toISOString(),
1543
- }),
1544
- {
1545
- headers: {
1546
- "Content-Type": "application/json",
1547
- ...corsHeaders,
1548
- },
1549
- }
1550
- );
1551
- } catch (error) {
1552
- console.error("[Server] Error in handleDeleteFileRequest:", error);
1553
- return new Response(
1554
- JSON.stringify({
1555
- error: "Failed to delete file",
1556
- message: error instanceof Error ? error.message : "Unknown error",
1557
- }),
1558
- {
1559
- headers: {
1560
- "Content-Type": "application/json",
1561
- ...corsHeaders,
1562
- },
1563
- status: 500,
1564
- }
1565
- );
1566
- }
1567
- }
1568
-
1569
- async function handleStreamingDeleteFileRequest(
1570
- req: Request,
1571
- corsHeaders: Record<string, string>
1572
- ): Promise<Response> {
1573
- try {
1574
- const body = (await req.json()) as DeleteFileRequest;
1575
- const { path, sessionId } = body;
1576
-
1577
- if (!path || typeof path !== "string") {
1578
- return new Response(
1579
- JSON.stringify({
1580
- error: "Path is required and must be a string",
1581
- }),
1582
- {
1583
- headers: {
1584
- "Content-Type": "application/json",
1585
- ...corsHeaders,
1586
- },
1587
- status: 400,
1588
- }
1589
- );
1590
- }
1591
-
1592
- // Basic safety check - prevent dangerous paths
1593
- const dangerousPatterns = [
1594
- /^\/$/, // Root directory
1595
- /^\/etc/, // System directories
1596
- /^\/var/, // System directories
1597
- /^\/usr/, // System directories
1598
- /^\/bin/, // System directories
1599
- /^\/sbin/, // System directories
1600
- /^\/boot/, // System directories
1601
- /^\/dev/, // System directories
1602
- /^\/proc/, // System directories
1603
- /^\/sys/, // System directories
1604
- /^\/tmp\/\.\./, // Path traversal attempts
1605
- /\.\./, // Path traversal attempts
1606
- ];
1607
-
1608
- if (dangerousPatterns.some((pattern) => pattern.test(path))) {
1609
- return new Response(
1610
- JSON.stringify({
1611
- error: "Dangerous path not allowed",
1612
- }),
1613
- {
1614
- headers: {
1615
- "Content-Type": "application/json",
1616
- ...corsHeaders,
1617
- },
1618
- status: 400,
1619
- }
1620
- );
1621
- }
1622
-
1623
- console.log(`[Server] Deleting file (streaming): ${path}`);
1624
-
1625
- const stream = new ReadableStream({
1626
- start(controller) {
1627
- (async () => {
1628
- try {
1629
- // Send command start event
1630
- controller.enqueue(
1631
- new TextEncoder().encode(
1632
- `data: ${JSON.stringify({
1633
- path,
1634
- timestamp: new Date().toISOString(),
1635
- type: "command_start",
1636
- })}\n\n`
1637
- )
1638
- );
1639
-
1640
- // Delete the file
1641
- await executeDeleteFile(path, sessionId);
1642
-
1643
- console.log(`[Server] File deleted successfully: ${path}`);
1644
-
1645
- // Send command completion event
1646
- controller.enqueue(
1647
- new TextEncoder().encode(
1648
- `data: ${JSON.stringify({
1649
- path,
1650
- success: true,
1651
- timestamp: new Date().toISOString(),
1652
- type: "command_complete",
1653
- })}\n\n`
1654
- )
1655
- );
1656
-
1657
- controller.close();
1658
- } catch (error) {
1659
- console.error(`[Server] Error deleting file: ${path}`, error);
1660
-
1661
- controller.enqueue(
1662
- new TextEncoder().encode(
1663
- `data: ${JSON.stringify({
1664
- error:
1665
- error instanceof Error ? error.message : "Unknown error",
1666
- path,
1667
- type: "error",
1668
- })}\n\n`
1669
- )
1670
- );
1671
-
1672
- controller.close();
1673
- }
1674
- })();
1675
- },
1676
- });
1677
-
1678
- return new Response(stream, {
1679
- headers: {
1680
- "Cache-Control": "no-cache",
1681
- Connection: "keep-alive",
1682
- "Content-Type": "text/event-stream",
1683
- ...corsHeaders,
1684
- },
1685
- });
1686
- } catch (error) {
1687
- console.error("[Server] Error in handleStreamingDeleteFileRequest:", error);
1688
- return new Response(
1689
- JSON.stringify({
1690
- error: "Failed to delete file",
1691
- message: error instanceof Error ? error.message : "Unknown error",
1692
- }),
1693
- {
1694
- headers: {
1695
- "Content-Type": "application/json",
1696
- ...corsHeaders,
1697
- },
1698
- status: 500,
1699
- }
1700
- );
1701
- }
1702
- }
1703
-
1704
- async function handleRenameFileRequest(
1705
- req: Request,
1706
- corsHeaders: Record<string, string>
1707
- ): Promise<Response> {
1708
- try {
1709
- const body = (await req.json()) as RenameFileRequest;
1710
- const { oldPath, newPath, sessionId } = body;
1711
-
1712
- if (!oldPath || typeof oldPath !== "string") {
1713
- return new Response(
1714
- JSON.stringify({
1715
- error: "Old path is required and must be a string",
1716
- }),
1717
- {
1718
- headers: {
1719
- "Content-Type": "application/json",
1720
- ...corsHeaders,
1721
- },
1722
- status: 400,
1723
- }
1724
- );
1725
- }
1726
-
1727
- if (!newPath || typeof newPath !== "string") {
1728
- return new Response(
1729
- JSON.stringify({
1730
- error: "New path is required and must be a string",
1731
- }),
1732
- {
1733
- headers: {
1734
- "Content-Type": "application/json",
1735
- ...corsHeaders,
1736
- },
1737
- status: 400,
1738
- }
1739
- );
1740
- }
1741
-
1742
- // Basic safety check - prevent dangerous paths
1743
- const dangerousPatterns = [
1744
- /^\/$/, // Root directory
1745
- /^\/etc/, // System directories
1746
- /^\/var/, // System directories
1747
- /^\/usr/, // System directories
1748
- /^\/bin/, // System directories
1749
- /^\/sbin/, // System directories
1750
- /^\/boot/, // System directories
1751
- /^\/dev/, // System directories
1752
- /^\/proc/, // System directories
1753
- /^\/sys/, // System directories
1754
- /^\/tmp\/\.\./, // Path traversal attempts
1755
- /\.\./, // Path traversal attempts
1756
- ];
1757
-
1758
- if (
1759
- dangerousPatterns.some(
1760
- (pattern) => pattern.test(oldPath) || pattern.test(newPath)
1761
- )
1762
- ) {
1763
- return new Response(
1764
- JSON.stringify({
1765
- error: "Dangerous path not allowed",
1766
- }),
1767
- {
1768
- headers: {
1769
- "Content-Type": "application/json",
1770
- ...corsHeaders,
1771
- },
1772
- status: 400,
1773
- }
1774
- );
1775
- }
1776
-
1777
- console.log(`[Server] Renaming file: ${oldPath} -> ${newPath}`);
1778
-
1779
- const result = await executeRenameFile(oldPath, newPath, sessionId);
1780
-
1781
- return new Response(
1782
- JSON.stringify({
1783
- exitCode: result.exitCode,
1784
- newPath,
1785
- oldPath,
1786
- success: result.success,
1787
- timestamp: new Date().toISOString(),
1788
- }),
1789
- {
1790
- headers: {
1791
- "Content-Type": "application/json",
1792
- ...corsHeaders,
1793
- },
1794
- }
1795
- );
1796
- } catch (error) {
1797
- console.error("[Server] Error in handleRenameFileRequest:", error);
1798
- return new Response(
1799
- JSON.stringify({
1800
- error: "Failed to rename file",
1801
- message: error instanceof Error ? error.message : "Unknown error",
1802
- }),
1803
- {
1804
- headers: {
1805
- "Content-Type": "application/json",
1806
- ...corsHeaders,
1807
- },
1808
- status: 500,
1809
- }
1810
- );
1811
- }
1812
- }
1813
-
1814
- async function handleStreamingRenameFileRequest(
1815
- req: Request,
1816
- corsHeaders: Record<string, string>
1817
- ): Promise<Response> {
1818
- try {
1819
- const body = (await req.json()) as RenameFileRequest;
1820
- const { oldPath, newPath, sessionId } = body;
1821
-
1822
- if (!oldPath || typeof oldPath !== "string") {
1823
- return new Response(
1824
- JSON.stringify({
1825
- error: "Old path is required and must be a string",
1826
- }),
1827
- {
1828
- headers: {
1829
- "Content-Type": "application/json",
1830
- ...corsHeaders,
1831
- },
1832
- status: 400,
1833
- }
1834
- );
1835
- }
1836
-
1837
- if (!newPath || typeof newPath !== "string") {
1838
- return new Response(
1839
- JSON.stringify({
1840
- error: "New path is required and must be a string",
1841
- }),
1842
- {
1843
- headers: {
1844
- "Content-Type": "application/json",
1845
- ...corsHeaders,
1846
- },
1847
- status: 400,
1848
- }
1849
- );
1850
- }
1851
-
1852
- // Basic safety check - prevent dangerous paths
1853
- const dangerousPatterns = [
1854
- /^\/$/, // Root directory
1855
- /^\/etc/, // System directories
1856
- /^\/var/, // System directories
1857
- /^\/usr/, // System directories
1858
- /^\/bin/, // System directories
1859
- /^\/sbin/, // System directories
1860
- /^\/boot/, // System directories
1861
- /^\/dev/, // System directories
1862
- /^\/proc/, // System directories
1863
- /^\/sys/, // System directories
1864
- /^\/tmp\/\.\./, // Path traversal attempts
1865
- /\.\./, // Path traversal attempts
1866
- ];
1867
-
1868
- if (
1869
- dangerousPatterns.some(
1870
- (pattern) => pattern.test(oldPath) || pattern.test(newPath)
1871
- )
1872
- ) {
1873
- return new Response(
1874
- JSON.stringify({
1875
- error: "Dangerous path not allowed",
1876
- }),
1877
- {
1878
- headers: {
1879
- "Content-Type": "application/json",
1880
- ...corsHeaders,
1881
- },
1882
- status: 400,
1883
- }
1884
- );
1885
- }
1886
-
1887
- console.log(`[Server] Renaming file (streaming): ${oldPath} -> ${newPath}`);
1888
-
1889
- const stream = new ReadableStream({
1890
- start(controller) {
1891
- (async () => {
1892
- try {
1893
- // Send command start event
1894
- controller.enqueue(
1895
- new TextEncoder().encode(
1896
- `data: ${JSON.stringify({
1897
- newPath,
1898
- oldPath,
1899
- timestamp: new Date().toISOString(),
1900
- type: "command_start",
1901
- })}\n\n`
1902
- )
1903
- );
1904
-
1905
- // Rename the file
1906
- await executeRenameFile(oldPath, newPath, sessionId);
1907
-
1908
- console.log(
1909
- `[Server] File renamed successfully: ${oldPath} -> ${newPath}`
1910
- );
1911
-
1912
- // Send command completion event
1913
- controller.enqueue(
1914
- new TextEncoder().encode(
1915
- `data: ${JSON.stringify({
1916
- newPath,
1917
- oldPath,
1918
- success: true,
1919
- timestamp: new Date().toISOString(),
1920
- type: "command_complete",
1921
- })}\n\n`
1922
- )
1923
- );
1924
-
1925
- controller.close();
1926
- } catch (error) {
1927
- console.error(
1928
- `[Server] Error renaming file: ${oldPath} -> ${newPath}`,
1929
- error
1930
- );
1931
-
1932
- controller.enqueue(
1933
- new TextEncoder().encode(
1934
- `data: ${JSON.stringify({
1935
- error:
1936
- error instanceof Error ? error.message : "Unknown error",
1937
- newPath,
1938
- oldPath,
1939
- type: "error",
1940
- })}\n\n`
1941
- )
1942
- );
1943
-
1944
- controller.close();
1945
- }
1946
- })();
1947
- },
1948
- });
1949
-
1950
- return new Response(stream, {
1951
- headers: {
1952
- "Cache-Control": "no-cache",
1953
- Connection: "keep-alive",
1954
- "Content-Type": "text/event-stream",
1955
- ...corsHeaders,
1956
- },
1957
- });
1958
- } catch (error) {
1959
- console.error("[Server] Error in handleStreamingRenameFileRequest:", error);
1960
- return new Response(
1961
- JSON.stringify({
1962
- error: "Failed to rename file",
1963
- message: error instanceof Error ? error.message : "Unknown error",
1964
- }),
1965
- {
1966
- headers: {
1967
- "Content-Type": "application/json",
1968
- ...corsHeaders,
1969
- },
1970
- status: 500,
1971
- }
1972
- );
1973
- }
1974
- }
1975
-
1976
- async function handleMoveFileRequest(
1977
- req: Request,
1978
- corsHeaders: Record<string, string>
1979
- ): Promise<Response> {
1980
- try {
1981
- const body = (await req.json()) as MoveFileRequest;
1982
- const { sourcePath, destinationPath, sessionId } = body;
1983
-
1984
- if (!sourcePath || typeof sourcePath !== "string") {
1985
- return new Response(
1986
- JSON.stringify({
1987
- error: "Source path is required and must be a string",
1988
- }),
1989
- {
1990
- headers: {
1991
- "Content-Type": "application/json",
1992
- ...corsHeaders,
1993
- },
1994
- status: 400,
1995
- }
1996
- );
1997
- }
1998
-
1999
- if (!destinationPath || typeof destinationPath !== "string") {
2000
- return new Response(
2001
- JSON.stringify({
2002
- error: "Destination path is required and must be a string",
2003
- }),
2004
- {
2005
- headers: {
2006
- "Content-Type": "application/json",
2007
- ...corsHeaders,
2008
- },
2009
- status: 400,
2010
- }
2011
- );
2012
- }
2013
-
2014
- // Basic safety check - prevent dangerous paths
2015
- const dangerousPatterns = [
2016
- /^\/$/, // Root directory
2017
- /^\/etc/, // System directories
2018
- /^\/var/, // System directories
2019
- /^\/usr/, // System directories
2020
- /^\/bin/, // System directories
2021
- /^\/sbin/, // System directories
2022
- /^\/boot/, // System directories
2023
- /^\/dev/, // System directories
2024
- /^\/proc/, // System directories
2025
- /^\/sys/, // System directories
2026
- /^\/tmp\/\.\./, // Path traversal attempts
2027
- /\.\./, // Path traversal attempts
2028
- ];
2029
-
2030
- if (
2031
- dangerousPatterns.some(
2032
- (pattern) => pattern.test(sourcePath) || pattern.test(destinationPath)
2033
- )
2034
- ) {
2035
- return new Response(
2036
- JSON.stringify({
2037
- error: "Dangerous path not allowed",
2038
- }),
2039
- {
2040
- headers: {
2041
- "Content-Type": "application/json",
2042
- ...corsHeaders,
2043
- },
2044
- status: 400,
2045
- }
2046
- );
2047
- }
2048
-
2049
- console.log(`[Server] Moving file: ${sourcePath} -> ${destinationPath}`);
2050
-
2051
- const result = await executeMoveFile(
2052
- sourcePath,
2053
- destinationPath,
2054
- sessionId
2055
- );
2056
-
2057
- return new Response(
2058
- JSON.stringify({
2059
- destinationPath,
2060
- exitCode: result.exitCode,
2061
- sourcePath,
2062
- success: result.success,
2063
- timestamp: new Date().toISOString(),
2064
- }),
2065
- {
2066
- headers: {
2067
- "Content-Type": "application/json",
2068
- ...corsHeaders,
2069
- },
2070
- }
2071
- );
2072
- } catch (error) {
2073
- console.error("[Server] Error in handleMoveFileRequest:", error);
2074
- return new Response(
2075
- JSON.stringify({
2076
- error: "Failed to move file",
2077
- message: error instanceof Error ? error.message : "Unknown error",
2078
- }),
2079
- {
2080
- headers: {
2081
- "Content-Type": "application/json",
2082
- ...corsHeaders,
2083
- },
2084
- status: 500,
2085
- }
2086
- );
2087
- }
2088
- }
2089
-
2090
- async function handleStreamingMoveFileRequest(
2091
- req: Request,
2092
- corsHeaders: Record<string, string>
2093
- ): Promise<Response> {
2094
- try {
2095
- const body = (await req.json()) as MoveFileRequest;
2096
- const { sourcePath, destinationPath, sessionId } = body;
2097
-
2098
- if (!sourcePath || typeof sourcePath !== "string") {
2099
- return new Response(
2100
- JSON.stringify({
2101
- error: "Source path is required and must be a string",
2102
- }),
2103
- {
2104
- headers: {
2105
- "Content-Type": "application/json",
2106
- ...corsHeaders,
2107
- },
2108
- status: 400,
2109
- }
2110
- );
2111
- }
2112
-
2113
- if (!destinationPath || typeof destinationPath !== "string") {
2114
- return new Response(
2115
- JSON.stringify({
2116
- error: "Destination path is required and must be a string",
2117
- }),
2118
- {
2119
- headers: {
2120
- "Content-Type": "application/json",
2121
- ...corsHeaders,
2122
- },
2123
- status: 400,
2124
- }
2125
- );
2126
- }
2127
-
2128
- // Basic safety check - prevent dangerous paths
2129
- const dangerousPatterns = [
2130
- /^\/$/, // Root directory
2131
- /^\/etc/, // System directories
2132
- /^\/var/, // System directories
2133
- /^\/usr/, // System directories
2134
- /^\/bin/, // System directories
2135
- /^\/sbin/, // System directories
2136
- /^\/boot/, // System directories
2137
- /^\/dev/, // System directories
2138
- /^\/proc/, // System directories
2139
- /^\/sys/, // System directories
2140
- /^\/tmp\/\.\./, // Path traversal attempts
2141
- /\.\./, // Path traversal attempts
2142
- ];
2143
-
2144
- if (
2145
- dangerousPatterns.some(
2146
- (pattern) => pattern.test(sourcePath) || pattern.test(destinationPath)
2147
- )
2148
- ) {
2149
- return new Response(
2150
- JSON.stringify({
2151
- error: "Dangerous path not allowed",
2152
- }),
2153
- {
2154
- headers: {
2155
- "Content-Type": "application/json",
2156
- ...corsHeaders,
2157
- },
2158
- status: 400,
2159
- }
2160
- );
2161
- }
2162
-
2163
- console.log(
2164
- `[Server] Moving file (streaming): ${sourcePath} -> ${destinationPath}`
2165
- );
2166
-
2167
- const stream = new ReadableStream({
2168
- start(controller) {
2169
- (async () => {
2170
- try {
2171
- // Send command start event
2172
- controller.enqueue(
2173
- new TextEncoder().encode(
2174
- `data: ${JSON.stringify({
2175
- destinationPath,
2176
- sourcePath,
2177
- timestamp: new Date().toISOString(),
2178
- type: "command_start",
2179
- })}\n\n`
2180
- )
2181
- );
2182
-
2183
- // Move the file
2184
- await executeMoveFile(sourcePath, destinationPath, sessionId);
2185
-
2186
- console.log(
2187
- `[Server] File moved successfully: ${sourcePath} -> ${destinationPath}`
2188
- );
2189
-
2190
- // Send command completion event
2191
- controller.enqueue(
2192
- new TextEncoder().encode(
2193
- `data: ${JSON.stringify({
2194
- destinationPath,
2195
- sourcePath,
2196
- success: true,
2197
- timestamp: new Date().toISOString(),
2198
- type: "command_complete",
2199
- })}\n\n`
2200
- )
2201
- );
2202
-
2203
- controller.close();
2204
- } catch (error) {
2205
- console.error(
2206
- `[Server] Error moving file: ${sourcePath} -> ${destinationPath}`,
2207
- error
2208
- );
2209
-
2210
- controller.enqueue(
2211
- new TextEncoder().encode(
2212
- `data: ${JSON.stringify({
2213
- destinationPath,
2214
- error:
2215
- error instanceof Error ? error.message : "Unknown error",
2216
- sourcePath,
2217
- type: "error",
2218
- })}\n\n`
2219
- )
2220
- );
2221
-
2222
- controller.close();
2223
- }
2224
- })();
2225
- },
2226
- });
2227
-
2228
- return new Response(stream, {
2229
- headers: {
2230
- "Cache-Control": "no-cache",
2231
- Connection: "keep-alive",
2232
- "Content-Type": "text/event-stream",
2233
- ...corsHeaders,
2234
- },
2235
- });
2236
- } catch (error) {
2237
- console.error("[Server] Error in handleStreamingMoveFileRequest:", error);
2238
- return new Response(
2239
- JSON.stringify({
2240
- error: "Failed to move file",
2241
- message: error instanceof Error ? error.message : "Unknown error",
2242
- }),
2243
- {
2244
- headers: {
2245
- "Content-Type": "application/json",
2246
- ...corsHeaders,
2247
- },
2248
- status: 500,
2249
- }
2250
- );
2251
- }
2252
- }
2253
-
2254
- function executeCommand(
2255
- command: string,
2256
- args: string[],
2257
- sessionId?: string
2258
- ): Promise<{
2259
- success: boolean;
2260
- stdout: string;
2261
- stderr: string;
2262
- exitCode: number;
2263
- }> {
2264
- return new Promise((resolve, reject) => {
2265
- const child = spawn(command, args, {
2266
- shell: true,
2267
- stdio: ["pipe", "pipe", "pipe"],
2268
- });
2269
-
2270
- // Store the process reference for cleanup if sessionId is provided
2271
- if (sessionId && sessions.has(sessionId)) {
2272
- const session = sessions.get(sessionId)!;
2273
- session.activeProcess = child;
2274
- }
2275
-
2276
- let stdout = "";
2277
- let stderr = "";
2278
-
2279
- child.stdout?.on("data", (data) => {
2280
- stdout += data.toString();
2281
- });
2282
-
2283
- child.stderr?.on("data", (data) => {
2284
- stderr += data.toString();
2285
- });
2286
-
2287
- child.on("close", (code) => {
2288
- // Clear the active process reference
2289
- if (sessionId && sessions.has(sessionId)) {
2290
- const session = sessions.get(sessionId)!;
2291
- session.activeProcess = null;
2292
- }
2293
-
2294
- console.log(`[Server] Command completed: ${command}, Exit code: ${code}`);
2295
-
2296
- resolve({
2297
- exitCode: code || 0,
2298
- stderr,
2299
- stdout,
2300
- success: code === 0,
2301
- });
2302
- });
2303
-
2304
- child.on("error", (error) => {
2305
- // Clear the active process reference
2306
- if (sessionId && sessions.has(sessionId)) {
2307
- const session = sessions.get(sessionId)!;
2308
- session.activeProcess = null;
2309
- }
2310
-
2311
- reject(error);
2312
- });
2313
- });
2314
- }
2315
-
2316
- function executeGitCheckout(
2317
- repoUrl: string,
2318
- branch: string,
2319
- targetDir: string,
2320
- sessionId?: string
2321
- ): Promise<{
2322
- success: boolean;
2323
- stdout: string;
2324
- stderr: string;
2325
- exitCode: number;
2326
- }> {
2327
- return new Promise((resolve, reject) => {
2328
- // First, clone the repository
2329
- const cloneChild = spawn(
2330
- "git",
2331
- ["clone", "-b", branch, repoUrl, targetDir],
2332
- {
2333
- shell: true,
2334
- stdio: ["pipe", "pipe", "pipe"],
2335
- }
2336
- );
2337
-
2338
- // Store the process reference for cleanup if sessionId is provided
2339
- if (sessionId && sessions.has(sessionId)) {
2340
- const session = sessions.get(sessionId)!;
2341
- session.activeProcess = cloneChild;
2342
- }
2343
-
2344
- let stdout = "";
2345
- let stderr = "";
2346
-
2347
- cloneChild.stdout?.on("data", (data) => {
2348
- stdout += data.toString();
2349
- });
2350
-
2351
- cloneChild.stderr?.on("data", (data) => {
2352
- stderr += data.toString();
2353
- });
2354
-
2355
- cloneChild.on("close", (code) => {
2356
- // Clear the active process reference
2357
- if (sessionId && sessions.has(sessionId)) {
2358
- const session = sessions.get(sessionId)!;
2359
- session.activeProcess = null;
2360
- }
2361
-
2362
- if (code === 0) {
2363
- console.log(
2364
- `[Server] Repository cloned successfully: ${repoUrl} to ${targetDir}`
2365
- );
2366
- resolve({
2367
- exitCode: code || 0,
2368
- stderr,
2369
- stdout,
2370
- success: true,
2371
- });
2372
- } else {
2373
- console.error(
2374
- `[Server] Failed to clone repository: ${repoUrl}, Exit code: ${code}`
2375
- );
2376
- resolve({
2377
- exitCode: code || 1,
2378
- stderr,
2379
- stdout,
2380
- success: false,
2381
- });
2382
- }
2383
- });
2384
-
2385
- cloneChild.on("error", (error) => {
2386
- // Clear the active process reference
2387
- if (sessionId && sessions.has(sessionId)) {
2388
- const session = sessions.get(sessionId)!;
2389
- session.activeProcess = null;
2390
- }
2391
-
2392
- console.error(`[Server] Error cloning repository: ${repoUrl}`, error);
2393
- reject(error);
2394
- });
2395
- });
2396
- }
2397
-
2398
- function executeMkdir(
2399
- path: string,
2400
- recursive: boolean,
2401
- sessionId?: string
2402
- ): Promise<{
2403
- success: boolean;
2404
- stdout: string;
2405
- stderr: string;
2406
- exitCode: number;
2407
- }> {
2408
- return new Promise((resolve, reject) => {
2409
- const args = recursive ? ["-p", path] : [path];
2410
- const mkdirChild = spawn("mkdir", args, {
2411
- shell: true,
2412
- stdio: ["pipe", "pipe", "pipe"],
2413
- });
2414
-
2415
- // Store the process reference for cleanup if sessionId is provided
2416
- if (sessionId && sessions.has(sessionId)) {
2417
- const session = sessions.get(sessionId)!;
2418
- session.activeProcess = mkdirChild;
2419
- }
2420
-
2421
- let stdout = "";
2422
- let stderr = "";
2423
-
2424
- mkdirChild.stdout?.on("data", (data) => {
2425
- stdout += data.toString();
2426
- });
2427
-
2428
- mkdirChild.stderr?.on("data", (data) => {
2429
- stderr += data.toString();
2430
- });
2431
-
2432
- mkdirChild.on("close", (code) => {
2433
- // Clear the active process reference
2434
- if (sessionId && sessions.has(sessionId)) {
2435
- const session = sessions.get(sessionId)!;
2436
- session.activeProcess = null;
2437
- }
2438
-
2439
- if (code === 0) {
2440
- console.log(`[Server] Directory created successfully: ${path}`);
2441
- resolve({
2442
- exitCode: code || 0,
2443
- stderr,
2444
- stdout,
2445
- success: true,
2446
- });
2447
- } else {
2448
- console.error(
2449
- `[Server] Failed to create directory: ${path}, Exit code: ${code}`
2450
- );
2451
- resolve({
2452
- exitCode: code || 1,
2453
- stderr,
2454
- stdout,
2455
- success: false,
2456
- });
2457
- }
2458
- });
2459
-
2460
- mkdirChild.on("error", (error) => {
2461
- // Clear the active process reference
2462
- if (sessionId && sessions.has(sessionId)) {
2463
- const session = sessions.get(sessionId)!;
2464
- session.activeProcess = null;
2465
- }
2466
-
2467
- console.error(`[Server] Error creating directory: ${path}`, error);
2468
- reject(error);
2469
- });
2470
- });
2471
- }
2472
-
2473
- function executeWriteFile(
2474
- path: string,
2475
- content: string,
2476
- encoding: string,
2477
- sessionId?: string
2478
- ): Promise<{
2479
- success: boolean;
2480
- exitCode: number;
2481
- }> {
2482
- return new Promise((resolve, reject) => {
2483
- (async () => {
2484
- try {
2485
- // Ensure the directory exists
2486
- const dir = dirname(path);
2487
- if (dir !== ".") {
2488
- await mkdir(dir, { recursive: true });
2489
- }
2490
-
2491
- // Write the file
2492
- await writeFile(path, content, {
2493
- encoding: encoding as BufferEncoding,
2494
- });
2495
-
2496
- console.log(`[Server] File written successfully: ${path}`);
2497
- resolve({
2498
- exitCode: 0,
2499
- success: true,
2500
- });
2501
- } catch (error) {
2502
- console.error(`[Server] Error writing file: ${path}`, error);
2503
- reject(error);
2504
- }
2505
- })();
2506
- });
2507
- }
2508
-
2509
- function executeDeleteFile(
2510
- path: string,
2511
- sessionId?: string
2512
- ): Promise<{
2513
- success: boolean;
2514
- exitCode: number;
2515
- }> {
2516
- return new Promise((resolve, reject) => {
2517
- (async () => {
2518
- try {
2519
- // Delete the file
2520
- await unlink(path);
2521
-
2522
- console.log(`[Server] File deleted successfully: ${path}`);
2523
- resolve({
2524
- exitCode: 0,
2525
- success: true,
2526
- });
2527
- } catch (error) {
2528
- console.error(`[Server] Error deleting file: ${path}`, error);
2529
- reject(error);
2530
- }
2531
- })();
2532
- });
2533
- }
2534
-
2535
- function executeRenameFile(
2536
- oldPath: string,
2537
- newPath: string,
2538
- sessionId?: string
2539
- ): Promise<{
2540
- success: boolean;
2541
- exitCode: number;
2542
- }> {
2543
- return new Promise((resolve, reject) => {
2544
- (async () => {
2545
- try {
2546
- // Rename the file
2547
- await rename(oldPath, newPath);
2548
-
2549
- console.log(
2550
- `[Server] File renamed successfully: ${oldPath} -> ${newPath}`
2551
- );
2552
- resolve({
2553
- exitCode: 0,
2554
- success: true,
2555
- });
2556
- } catch (error) {
2557
- console.error(
2558
- `[Server] Error renaming file: ${oldPath} -> ${newPath}`,
2559
- error
2560
- );
2561
- reject(error);
2562
- }
2563
- })();
2564
- });
2565
- }
2566
-
2567
- function executeMoveFile(
2568
- sourcePath: string,
2569
- destinationPath: string,
2570
- sessionId?: string
2571
- ): Promise<{
2572
- success: boolean;
2573
- exitCode: number;
2574
- }> {
2575
- return new Promise((resolve, reject) => {
2576
- (async () => {
2577
- try {
2578
- // Move the file
2579
- await rename(sourcePath, destinationPath);
2580
-
2581
- console.log(
2582
- `[Server] File moved successfully: ${sourcePath} -> ${destinationPath}`
2583
- );
2584
- resolve({
2585
- exitCode: 0,
2586
- success: true,
2587
- });
2588
- } catch (error) {
2589
- console.error(
2590
- `[Server] Error moving file: ${sourcePath} -> ${destinationPath}`,
2591
- error
2592
- );
2593
- reject(error);
2594
- }
2595
- })();
2596
- });
2597
- }
332
+ // We don't need this, but typescript complains
333
+ websocket: { async message() { } },
334
+ });
2598
335
 
2599
- console.log(`🚀 Bun server running on http://localhost:${server.port}`);
336
+ console.log(`🚀 Bun server running on http://0.0.0.0:${server.port}`);
2600
337
  console.log(`📡 HTTP API endpoints available:`);
2601
338
  console.log(` POST /api/session/create - Create a new session`);
2602
339
  console.log(` GET /api/session/list - List all sessions`);
2603
340
  console.log(` POST /api/execute - Execute a command (non-streaming)`);
2604
341
  console.log(` POST /api/execute/stream - Execute a command (streaming)`);
2605
342
  console.log(` POST /api/git/checkout - Checkout a git repository`);
2606
- console.log(
2607
- ` POST /api/git/checkout/stream - Checkout a git repository (streaming)`
2608
- );
2609
343
  console.log(` POST /api/mkdir - Create a directory`);
2610
- console.log(` POST /api/mkdir/stream - Create a directory (streaming)`);
2611
344
  console.log(` POST /api/write - Write a file`);
2612
- console.log(` POST /api/write/stream - Write a file (streaming)`);
345
+ console.log(` POST /api/read - Read a file`);
2613
346
  console.log(` POST /api/delete - Delete a file`);
2614
- console.log(` POST /api/delete/stream - Delete a file (streaming)`);
2615
347
  console.log(` POST /api/rename - Rename a file`);
2616
- console.log(` POST /api/rename/stream - Rename a file (streaming)`);
2617
348
  console.log(` POST /api/move - Move a file`);
2618
- console.log(` POST /api/move/stream - Move a file (streaming)`);
349
+ console.log(` POST /api/expose-port - Expose a port for external access`);
350
+ console.log(` DELETE /api/unexpose-port - Unexpose a port`);
351
+ console.log(` GET /api/exposed-ports - List exposed ports`);
352
+ console.log(` POST /api/process/start - Start a background process`);
353
+ console.log(` GET /api/process/list - List all processes`);
354
+ console.log(` GET /api/process/{id} - Get process status`);
355
+ console.log(` DELETE /api/process/{id} - Kill a process`);
356
+ console.log(` GET /api/process/{id}/logs - Get process logs`);
357
+ console.log(` GET /api/process/{id}/stream - Stream process logs (SSE)`);
358
+ console.log(` DELETE /api/process/kill-all - Kill all processes`);
359
+ console.log(` GET /proxy/{port}/* - Proxy requests to exposed ports`);
2619
360
  console.log(` GET /api/ping - Health check`);
2620
361
  console.log(` GET /api/commands - List available commands`);