@cloudflare/sandbox 0.0.0-0ac3cfa → 0.0.0-0b4cc05

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 (79) hide show
  1. package/CHANGELOG.md +191 -0
  2. package/Dockerfile +166 -9
  3. package/README.md +149 -50
  4. package/dist/chunk-BCJ7SF3Q.js +117 -0
  5. package/dist/chunk-BCJ7SF3Q.js.map +1 -0
  6. package/dist/chunk-BFVUNTP4.js +104 -0
  7. package/dist/chunk-BFVUNTP4.js.map +1 -0
  8. package/dist/chunk-EKSWCBCA.js +86 -0
  9. package/dist/chunk-EKSWCBCA.js.map +1 -0
  10. package/dist/chunk-U2M5GSMU.js +2220 -0
  11. package/dist/chunk-U2M5GSMU.js.map +1 -0
  12. package/dist/chunk-Z532A7QC.js +78 -0
  13. package/dist/chunk-Z532A7QC.js.map +1 -0
  14. package/dist/file-stream.d.ts +43 -0
  15. package/dist/file-stream.js +9 -0
  16. package/dist/file-stream.js.map +1 -0
  17. package/dist/index.d.ts +9 -0
  18. package/dist/index.js +55 -0
  19. package/dist/index.js.map +1 -0
  20. package/dist/interpreter.d.ts +33 -0
  21. package/dist/interpreter.js +8 -0
  22. package/dist/interpreter.js.map +1 -0
  23. package/dist/request-handler.d.ts +18 -0
  24. package/dist/request-handler.js +12 -0
  25. package/dist/request-handler.js.map +1 -0
  26. package/dist/sandbox-Cyuj5F-M.d.ts +579 -0
  27. package/dist/sandbox.d.ts +4 -0
  28. package/dist/sandbox.js +12 -0
  29. package/dist/sandbox.js.map +1 -0
  30. package/dist/security.d.ts +31 -0
  31. package/dist/security.js +13 -0
  32. package/dist/security.js.map +1 -0
  33. package/dist/sse-parser.d.ts +28 -0
  34. package/dist/sse-parser.js +11 -0
  35. package/dist/sse-parser.js.map +1 -0
  36. package/package.json +11 -9
  37. package/src/clients/base-client.ts +280 -0
  38. package/src/clients/command-client.ts +115 -0
  39. package/src/clients/file-client.ts +269 -0
  40. package/src/clients/git-client.ts +92 -0
  41. package/src/clients/index.ts +63 -0
  42. package/src/clients/interpreter-client.ts +329 -0
  43. package/src/clients/port-client.ts +105 -0
  44. package/src/clients/process-client.ts +177 -0
  45. package/src/clients/sandbox-client.ts +41 -0
  46. package/src/clients/types.ts +84 -0
  47. package/src/clients/utility-client.ts +94 -0
  48. package/src/errors/adapter.ts +180 -0
  49. package/src/errors/classes.ts +469 -0
  50. package/src/errors/index.ts +105 -0
  51. package/src/file-stream.ts +164 -0
  52. package/src/index.ts +83 -119
  53. package/src/interpreter.ts +159 -0
  54. package/src/request-handler.ts +170 -0
  55. package/src/sandbox.ts +897 -0
  56. package/src/security.ts +104 -0
  57. package/src/sse-parser.ts +143 -0
  58. package/startup.sh +3 -0
  59. package/tests/base-client.test.ts +328 -0
  60. package/tests/command-client.test.ts +407 -0
  61. package/tests/file-client.test.ts +643 -0
  62. package/tests/file-stream.test.ts +306 -0
  63. package/tests/git-client.test.ts +328 -0
  64. package/tests/port-client.test.ts +301 -0
  65. package/tests/process-client.test.ts +658 -0
  66. package/tests/sandbox.test.ts +465 -0
  67. package/tests/sse-parser.test.ts +290 -0
  68. package/tests/utility-client.test.ts +266 -0
  69. package/tests/wrangler.jsonc +35 -0
  70. package/tsconfig.json +9 -1
  71. package/vitest.config.ts +31 -0
  72. package/container_src/index.ts +0 -2906
  73. package/container_src/package.json +0 -9
  74. package/src/client.ts +0 -1950
  75. package/tests/client.example.ts +0 -308
  76. package/tests/connection-test.ts +0 -81
  77. package/tests/simple-test.ts +0 -81
  78. package/tests/test1.ts +0 -281
  79. package/tests/test2.ts +0 -929
package/src/client.ts DELETED
@@ -1,1950 +0,0 @@
1
- import type { DurableObject } from "cloudflare:workers";
2
- import type { Sandbox } from "./index";
3
-
4
- interface ExecuteRequest {
5
- command: string;
6
- args?: string[];
7
- }
8
-
9
- export interface ExecuteResponse {
10
- success: boolean;
11
- stdout: string;
12
- stderr: string;
13
- exitCode: number;
14
- command: string;
15
- args: string[];
16
- timestamp: string;
17
- }
18
-
19
- interface SessionResponse {
20
- sessionId: string;
21
- message: string;
22
- timestamp: string;
23
- }
24
-
25
- interface SessionListResponse {
26
- sessions: Array<{
27
- sessionId: string;
28
- hasActiveProcess: boolean;
29
- createdAt: string;
30
- }>;
31
- count: number;
32
- timestamp: string;
33
- }
34
-
35
- interface CommandsResponse {
36
- availableCommands: string[];
37
- timestamp: string;
38
- }
39
-
40
- interface GitCheckoutRequest {
41
- repoUrl: string;
42
- branch?: string;
43
- targetDir?: string;
44
- sessionId?: string;
45
- }
46
-
47
- export interface GitCheckoutResponse {
48
- success: boolean;
49
- stdout: string;
50
- stderr: string;
51
- exitCode: number;
52
- repoUrl: string;
53
- branch: string;
54
- targetDir: string;
55
- timestamp: string;
56
- }
57
-
58
- interface MkdirRequest {
59
- path: string;
60
- recursive?: boolean;
61
- sessionId?: string;
62
- }
63
-
64
- export interface MkdirResponse {
65
- success: boolean;
66
- stdout: string;
67
- stderr: string;
68
- exitCode: number;
69
- path: string;
70
- recursive: boolean;
71
- timestamp: string;
72
- }
73
-
74
- interface WriteFileRequest {
75
- path: string;
76
- content: string;
77
- encoding?: string;
78
- sessionId?: string;
79
- }
80
-
81
- export interface WriteFileResponse {
82
- success: boolean;
83
- exitCode: number;
84
- path: string;
85
- timestamp: string;
86
- }
87
-
88
- interface ReadFileRequest {
89
- path: string;
90
- encoding?: string;
91
- sessionId?: string;
92
- }
93
-
94
- export interface ReadFileResponse {
95
- success: boolean;
96
- exitCode: number;
97
- path: string;
98
- content: string;
99
- timestamp: string;
100
- }
101
-
102
- interface DeleteFileRequest {
103
- path: string;
104
- sessionId?: string;
105
- }
106
-
107
- export interface DeleteFileResponse {
108
- success: boolean;
109
- exitCode: number;
110
- path: string;
111
- timestamp: string;
112
- }
113
-
114
- interface RenameFileRequest {
115
- oldPath: string;
116
- newPath: string;
117
- sessionId?: string;
118
- }
119
-
120
- export interface RenameFileResponse {
121
- success: boolean;
122
- exitCode: number;
123
- oldPath: string;
124
- newPath: string;
125
- timestamp: string;
126
- }
127
-
128
- interface MoveFileRequest {
129
- sourcePath: string;
130
- destinationPath: string;
131
- sessionId?: string;
132
- }
133
-
134
- export interface MoveFileResponse {
135
- success: boolean;
136
- exitCode: number;
137
- sourcePath: string;
138
- destinationPath: string;
139
- timestamp: string;
140
- }
141
-
142
- interface PingResponse {
143
- message: string;
144
- timestamp: string;
145
- }
146
-
147
- interface StreamEvent {
148
- type: "command_start" | "output" | "command_complete" | "error";
149
- command?: string;
150
- args?: string[];
151
- stream?: "stdout" | "stderr";
152
- data?: string;
153
- message?: string;
154
- path?: string;
155
- oldPath?: string;
156
- newPath?: string;
157
- sourcePath?: string;
158
- destinationPath?: string;
159
- content?: string;
160
- success?: boolean;
161
- exitCode?: number;
162
- stdout?: string;
163
- stderr?: string;
164
- error?: string;
165
- timestamp?: string;
166
- }
167
-
168
- interface HttpClientOptions {
169
- stub?: Sandbox;
170
- baseUrl?: string;
171
- port?: number;
172
- onCommandStart?: (command: string, args: string[]) => void;
173
- onOutput?: (
174
- stream: "stdout" | "stderr",
175
- data: string,
176
- command: string
177
- ) => void;
178
- onCommandComplete?: (
179
- success: boolean,
180
- exitCode: number,
181
- stdout: string,
182
- stderr: string,
183
- command: string,
184
- args: string[]
185
- ) => void;
186
- onError?: (error: string, command?: string, args?: string[]) => void;
187
- onStreamEvent?: (event: StreamEvent) => void;
188
- }
189
-
190
- export class HttpClient {
191
- private baseUrl: string;
192
- private options: HttpClientOptions;
193
- private sessionId: string | null = null;
194
-
195
- constructor(options: HttpClientOptions = {}) {
196
- this.options = {
197
- ...options,
198
- };
199
- this.baseUrl = this.options.baseUrl!;
200
- }
201
-
202
- private async doFetch(
203
- path: string,
204
- options?: RequestInit
205
- ): Promise<Response> {
206
- const url = this.options.stub ? `stub:${path}` : `${this.baseUrl}${path}`;
207
- const method = options?.method || "GET";
208
-
209
- console.log(`[HTTP Client] Making ${method} request to ${url}`);
210
-
211
- try {
212
- let response: Response;
213
-
214
- if (this.options.stub) {
215
- response = await this.options.stub.containerFetch(path, options, this.options.port);
216
- } else {
217
- response = await fetch(this.baseUrl + path, options);
218
- }
219
-
220
- console.log(`[HTTP Client] Response: ${response.status} ${response.statusText}`);
221
-
222
- if (!response.ok) {
223
- console.error(`[HTTP Client] Request failed: ${method} ${url} - ${response.status} ${response.statusText}`);
224
- }
225
-
226
- return response;
227
- } catch (error) {
228
- console.error(`[HTTP Client] Request error: ${method} ${url}`, error);
229
- throw error;
230
- }
231
- }
232
- // Public methods to set event handlers
233
- setOnOutput(
234
- handler: (
235
- stream: "stdout" | "stderr",
236
- data: string,
237
- command: string
238
- ) => void
239
- ): void {
240
- this.options.onOutput = handler;
241
- }
242
-
243
- setOnCommandComplete(
244
- handler: (
245
- success: boolean,
246
- exitCode: number,
247
- stdout: string,
248
- stderr: string,
249
- command: string,
250
- args: string[]
251
- ) => void
252
- ): void {
253
- this.options.onCommandComplete = handler;
254
- }
255
-
256
- setOnStreamEvent(handler: (event: StreamEvent) => void): void {
257
- this.options.onStreamEvent = handler;
258
- }
259
-
260
- // Public getter methods
261
- getOnOutput():
262
- | ((stream: "stdout" | "stderr", data: string, command: string) => void)
263
- | undefined {
264
- return this.options.onOutput;
265
- }
266
-
267
- getOnCommandComplete():
268
- | ((
269
- success: boolean,
270
- exitCode: number,
271
- stdout: string,
272
- stderr: string,
273
- command: string,
274
- args: string[]
275
- ) => void)
276
- | undefined {
277
- return this.options.onCommandComplete;
278
- }
279
-
280
- getOnStreamEvent(): ((event: StreamEvent) => void) | undefined {
281
- return this.options.onStreamEvent;
282
- }
283
-
284
- async createSession(): Promise<string> {
285
- try {
286
- const response = await this.doFetch(`/api/session/create`, {
287
- headers: {
288
- "Content-Type": "application/json",
289
- },
290
- method: "POST",
291
- });
292
-
293
- if (!response.ok) {
294
- throw new Error(`HTTP error! status: ${response.status}`);
295
- }
296
-
297
- const data: SessionResponse = await response.json();
298
- this.sessionId = data.sessionId;
299
- console.log(`[HTTP Client] Created session: ${this.sessionId}`);
300
- return this.sessionId;
301
- } catch (error) {
302
- console.error("[HTTP Client] Error creating session:", error);
303
- throw error;
304
- }
305
- }
306
-
307
- async listSessions(): Promise<SessionListResponse> {
308
- try {
309
- const response = await this.doFetch(`/api/session/list`, {
310
- headers: {
311
- "Content-Type": "application/json",
312
- },
313
- method: "GET",
314
- });
315
-
316
- if (!response.ok) {
317
- throw new Error(`HTTP error! status: ${response.status}`);
318
- }
319
-
320
- const data: SessionListResponse = await response.json();
321
- console.log(`[HTTP Client] Listed ${data.count} sessions`);
322
- return data;
323
- } catch (error) {
324
- console.error("[HTTP Client] Error listing sessions:", error);
325
- throw error;
326
- }
327
- }
328
-
329
- async execute(
330
- command: string,
331
- args: string[] = [],
332
- sessionId?: string
333
- ): Promise<ExecuteResponse> {
334
- try {
335
- const targetSessionId = sessionId || this.sessionId;
336
-
337
- const response = await this.doFetch(`/api/execute`, {
338
- body: JSON.stringify({
339
- args,
340
- command,
341
- sessionId: targetSessionId,
342
- }),
343
- headers: {
344
- "Content-Type": "application/json",
345
- },
346
- method: "POST",
347
- });
348
-
349
- if (!response.ok) {
350
- const errorData = (await response.json().catch(() => ({}))) as {
351
- error?: string;
352
- };
353
- throw new Error(
354
- errorData.error || `HTTP error! status: ${response.status}`
355
- );
356
- }
357
-
358
- const data: ExecuteResponse = await response.json();
359
- console.log(
360
- `[HTTP Client] Command executed: ${command}, Success: ${data.success}`
361
- );
362
-
363
- // Call the callback if provided
364
- this.options.onCommandComplete?.(
365
- data.success,
366
- data.exitCode,
367
- data.stdout,
368
- data.stderr,
369
- data.command,
370
- data.args
371
- );
372
-
373
- return data;
374
- } catch (error) {
375
- console.error("[HTTP Client] Error executing command:", error);
376
- this.options.onError?.(
377
- error instanceof Error ? error.message : "Unknown error",
378
- command,
379
- args
380
- );
381
- throw error;
382
- }
383
- }
384
-
385
- async executeStream(
386
- command: string,
387
- args: string[] = [],
388
- sessionId?: string
389
- ): Promise<void> {
390
- try {
391
- const targetSessionId = sessionId || this.sessionId;
392
-
393
- const response = await this.doFetch(`/api/execute/stream`, {
394
- body: JSON.stringify({
395
- args,
396
- command,
397
- sessionId: targetSessionId,
398
- }),
399
- headers: {
400
- "Content-Type": "application/json",
401
- },
402
- method: "POST",
403
- });
404
-
405
- if (!response.ok) {
406
- const errorData = (await response.json().catch(() => ({}))) as {
407
- error?: string;
408
- };
409
- throw new Error(
410
- errorData.error || `HTTP error! status: ${response.status}`
411
- );
412
- }
413
-
414
- if (!response.body) {
415
- throw new Error("No response body for streaming request");
416
- }
417
-
418
- const reader = response.body.getReader();
419
- const decoder = new TextDecoder();
420
-
421
- try {
422
- while (true) {
423
- const { done, value } = await reader.read();
424
-
425
- if (done) {
426
- break;
427
- }
428
-
429
- const chunk = decoder.decode(value, { stream: true });
430
- const lines = chunk.split("\n");
431
-
432
- for (const line of lines) {
433
- if (line.startsWith("data: ")) {
434
- try {
435
- const eventData = line.slice(6); // Remove 'data: ' prefix
436
- const event: StreamEvent = JSON.parse(eventData);
437
-
438
- console.log(`[HTTP Client] Stream event: ${event.type}`);
439
- this.options.onStreamEvent?.(event);
440
-
441
- switch (event.type) {
442
- case "command_start":
443
- console.log(
444
- `[HTTP Client] Command started: ${
445
- event.command
446
- } ${event.args?.join(" ")}`
447
- );
448
- this.options.onCommandStart?.(
449
- event.command!,
450
- event.args || []
451
- );
452
- break;
453
-
454
- case "output":
455
- console.log(`[${event.stream}] ${event.data}`);
456
- this.options.onOutput?.(
457
- event.stream!,
458
- event.data!,
459
- event.command!
460
- );
461
- break;
462
-
463
- case "command_complete":
464
- console.log(
465
- `[HTTP Client] Command completed: ${event.command}, Success: ${event.success}, Exit code: ${event.exitCode}`
466
- );
467
- this.options.onCommandComplete?.(
468
- event.success!,
469
- event.exitCode!,
470
- event.stdout!,
471
- event.stderr!,
472
- event.command!,
473
- event.args || []
474
- );
475
- break;
476
-
477
- case "error":
478
- console.error(
479
- `[HTTP Client] Command error: ${event.error}`
480
- );
481
- this.options.onError?.(
482
- event.error!,
483
- event.command,
484
- event.args
485
- );
486
- break;
487
- }
488
- } catch (parseError) {
489
- console.warn(
490
- "[HTTP Client] Failed to parse stream event:",
491
- parseError
492
- );
493
- }
494
- }
495
- }
496
- }
497
- } finally {
498
- reader.releaseLock();
499
- }
500
- } catch (error) {
501
- console.error("[HTTP Client] Error in streaming execution:", error);
502
- this.options.onError?.(
503
- error instanceof Error ? error.message : "Unknown error",
504
- command,
505
- args
506
- );
507
- throw error;
508
- }
509
- }
510
-
511
- async gitCheckout(
512
- repoUrl: string,
513
- branch: string = "main",
514
- targetDir?: string,
515
- sessionId?: string
516
- ): Promise<GitCheckoutResponse> {
517
- try {
518
- const targetSessionId = sessionId || this.sessionId;
519
-
520
- const response = await this.doFetch(`/api/git/checkout`, {
521
- body: JSON.stringify({
522
- branch,
523
- repoUrl,
524
- sessionId: targetSessionId,
525
- targetDir,
526
- }),
527
- headers: {
528
- "Content-Type": "application/json",
529
- },
530
- method: "POST",
531
- });
532
-
533
- if (!response.ok) {
534
- const errorData = (await response.json().catch(() => ({}))) as {
535
- error?: string;
536
- };
537
- throw new Error(
538
- errorData.error || `HTTP error! status: ${response.status}`
539
- );
540
- }
541
-
542
- const data: GitCheckoutResponse = await response.json();
543
- console.log(
544
- `[HTTP Client] Git checkout completed: ${repoUrl}, Success: ${data.success}, Target: ${data.targetDir}`
545
- );
546
-
547
- return data;
548
- } catch (error) {
549
- console.error("[HTTP Client] Error in git checkout:", error);
550
- throw error;
551
- }
552
- }
553
-
554
- async gitCheckoutStream(
555
- repoUrl: string,
556
- branch: string = "main",
557
- targetDir?: string,
558
- sessionId?: string
559
- ): Promise<void> {
560
- try {
561
- const targetSessionId = sessionId || this.sessionId;
562
-
563
- const response = await this.doFetch(`/api/git/checkout/stream`, {
564
- body: JSON.stringify({
565
- branch,
566
- repoUrl,
567
- sessionId: targetSessionId,
568
- targetDir,
569
- }),
570
- headers: {
571
- "Content-Type": "application/json",
572
- },
573
- method: "POST",
574
- });
575
-
576
- if (!response.ok) {
577
- const errorData = (await response.json().catch(() => ({}))) as {
578
- error?: string;
579
- };
580
- throw new Error(
581
- errorData.error || `HTTP error! status: ${response.status}`
582
- );
583
- }
584
-
585
- if (!response.body) {
586
- throw new Error("No response body for streaming request");
587
- }
588
-
589
- const reader = response.body.getReader();
590
- const decoder = new TextDecoder();
591
-
592
- try {
593
- while (true) {
594
- const { done, value } = await reader.read();
595
-
596
- if (done) {
597
- break;
598
- }
599
-
600
- const chunk = decoder.decode(value, { stream: true });
601
- const lines = chunk.split("\n");
602
-
603
- for (const line of lines) {
604
- if (line.startsWith("data: ")) {
605
- try {
606
- const eventData = line.slice(6); // Remove 'data: ' prefix
607
- const event: StreamEvent = JSON.parse(eventData);
608
-
609
- console.log(
610
- `[HTTP Client] Git checkout stream event: ${event.type}`
611
- );
612
- this.options.onStreamEvent?.(event);
613
-
614
- switch (event.type) {
615
- case "command_start":
616
- console.log(
617
- `[HTTP Client] Git checkout started: ${
618
- event.command
619
- } ${event.args?.join(" ")}`
620
- );
621
- this.options.onCommandStart?.(
622
- event.command!,
623
- event.args || []
624
- );
625
- break;
626
-
627
- case "output":
628
- console.log(`[${event.stream}] ${event.data}`);
629
- this.options.onOutput?.(
630
- event.stream!,
631
- event.data!,
632
- event.command!
633
- );
634
- break;
635
-
636
- case "command_complete":
637
- console.log(
638
- `[HTTP Client] Git checkout completed: ${event.command}, Success: ${event.success}, Exit code: ${event.exitCode}`
639
- );
640
- this.options.onCommandComplete?.(
641
- event.success!,
642
- event.exitCode!,
643
- event.stdout!,
644
- event.stderr!,
645
- event.command!,
646
- event.args || []
647
- );
648
- break;
649
-
650
- case "error":
651
- console.error(
652
- `[HTTP Client] Git checkout error: ${event.error}`
653
- );
654
- this.options.onError?.(
655
- event.error!,
656
- event.command,
657
- event.args
658
- );
659
- break;
660
- }
661
- } catch (parseError) {
662
- console.warn(
663
- "[HTTP Client] Failed to parse git checkout stream event:",
664
- parseError
665
- );
666
- }
667
- }
668
- }
669
- }
670
- } finally {
671
- reader.releaseLock();
672
- }
673
- } catch (error) {
674
- console.error("[HTTP Client] Error in streaming git checkout:", error);
675
- this.options.onError?.(
676
- error instanceof Error ? error.message : "Unknown error",
677
- "git clone",
678
- [branch, repoUrl, targetDir || ""]
679
- );
680
- throw error;
681
- }
682
- }
683
-
684
- async mkdir(
685
- path: string,
686
- recursive: boolean = false,
687
- sessionId?: string
688
- ): Promise<MkdirResponse> {
689
- try {
690
- const targetSessionId = sessionId || this.sessionId;
691
-
692
- const response = await this.doFetch(`/api/mkdir`, {
693
- body: JSON.stringify({
694
- path,
695
- recursive,
696
- sessionId: targetSessionId,
697
- }),
698
- headers: {
699
- "Content-Type": "application/json",
700
- },
701
- method: "POST",
702
- });
703
-
704
- if (!response.ok) {
705
- const errorData = (await response.json().catch(() => ({}))) as {
706
- error?: string;
707
- };
708
- throw new Error(
709
- errorData.error || `HTTP error! status: ${response.status}`
710
- );
711
- }
712
-
713
- const data: MkdirResponse = await response.json();
714
- console.log(
715
- `[HTTP Client] Directory created: ${path}, Success: ${data.success}, Recursive: ${data.recursive}`
716
- );
717
-
718
- return data;
719
- } catch (error) {
720
- console.error("[HTTP Client] Error creating directory:", error);
721
- throw error;
722
- }
723
- }
724
-
725
- async mkdirStream(
726
- path: string,
727
- recursive: boolean = false,
728
- sessionId?: string
729
- ): Promise<void> {
730
- try {
731
- const targetSessionId = sessionId || this.sessionId;
732
-
733
- const response = await this.doFetch(`/api/mkdir/stream`, {
734
- body: JSON.stringify({
735
- path,
736
- recursive,
737
- sessionId: targetSessionId,
738
- }),
739
- headers: {
740
- "Content-Type": "application/json",
741
- },
742
- method: "POST",
743
- });
744
-
745
- if (!response.ok) {
746
- const errorData = (await response.json().catch(() => ({}))) as {
747
- error?: string;
748
- };
749
- throw new Error(
750
- errorData.error || `HTTP error! status: ${response.status}`
751
- );
752
- }
753
-
754
- if (!response.body) {
755
- throw new Error("No response body for streaming request");
756
- }
757
-
758
- const reader = response.body.getReader();
759
- const decoder = new TextDecoder();
760
-
761
- try {
762
- while (true) {
763
- const { done, value } = await reader.read();
764
-
765
- if (done) {
766
- break;
767
- }
768
-
769
- const chunk = decoder.decode(value, { stream: true });
770
- const lines = chunk.split("\n");
771
-
772
- for (const line of lines) {
773
- if (line.startsWith("data: ")) {
774
- try {
775
- const eventData = line.slice(6); // Remove 'data: ' prefix
776
- const event: StreamEvent = JSON.parse(eventData);
777
-
778
- console.log(`[HTTP Client] Mkdir stream event: ${event.type}`);
779
- this.options.onStreamEvent?.(event);
780
-
781
- switch (event.type) {
782
- case "command_start":
783
- console.log(
784
- `[HTTP Client] Mkdir started: ${
785
- event.command
786
- } ${event.args?.join(" ")}`
787
- );
788
- this.options.onCommandStart?.(
789
- event.command!,
790
- event.args || []
791
- );
792
- break;
793
-
794
- case "output":
795
- console.log(`[${event.stream}] ${event.data}`);
796
- this.options.onOutput?.(
797
- event.stream!,
798
- event.data!,
799
- event.command!
800
- );
801
- break;
802
-
803
- case "command_complete":
804
- console.log(
805
- `[HTTP Client] Mkdir completed: ${event.command}, Success: ${event.success}, Exit code: ${event.exitCode}`
806
- );
807
- this.options.onCommandComplete?.(
808
- event.success!,
809
- event.exitCode!,
810
- event.stdout!,
811
- event.stderr!,
812
- event.command!,
813
- event.args || []
814
- );
815
- break;
816
-
817
- case "error":
818
- console.error(`[HTTP Client] Mkdir error: ${event.error}`);
819
- this.options.onError?.(
820
- event.error!,
821
- event.command,
822
- event.args
823
- );
824
- break;
825
- }
826
- } catch (parseError) {
827
- console.warn(
828
- "[HTTP Client] Failed to parse mkdir stream event:",
829
- parseError
830
- );
831
- }
832
- }
833
- }
834
- }
835
- } finally {
836
- reader.releaseLock();
837
- }
838
- } catch (error) {
839
- console.error("[HTTP Client] Error in streaming mkdir:", error);
840
- this.options.onError?.(
841
- error instanceof Error ? error.message : "Unknown error",
842
- "mkdir",
843
- recursive ? ["-p", path] : [path]
844
- );
845
- throw error;
846
- }
847
- }
848
-
849
- async writeFile(
850
- path: string,
851
- content: string,
852
- encoding: string = "utf-8",
853
- sessionId?: string
854
- ): Promise<WriteFileResponse> {
855
- try {
856
- const targetSessionId = sessionId || this.sessionId;
857
-
858
- const response = await this.doFetch(`/api/write`, {
859
- body: JSON.stringify({
860
- content,
861
- encoding,
862
- path,
863
- sessionId: targetSessionId,
864
- }),
865
- headers: {
866
- "Content-Type": "application/json",
867
- },
868
- method: "POST",
869
- });
870
-
871
- if (!response.ok) {
872
- const errorData = (await response.json().catch(() => ({}))) as {
873
- error?: string;
874
- };
875
- throw new Error(
876
- errorData.error || `HTTP error! status: ${response.status}`
877
- );
878
- }
879
-
880
- const data: WriteFileResponse = await response.json();
881
- console.log(
882
- `[HTTP Client] File written: ${path}, Success: ${data.success}`
883
- );
884
-
885
- return data;
886
- } catch (error) {
887
- console.error("[HTTP Client] Error writing file:", error);
888
- throw error;
889
- }
890
- }
891
-
892
- async writeFileStream(
893
- path: string,
894
- content: string,
895
- encoding: string = "utf-8",
896
- sessionId?: string
897
- ): Promise<void> {
898
- try {
899
- const targetSessionId = sessionId || this.sessionId;
900
-
901
- const response = await this.doFetch(`/api/write/stream`, {
902
- body: JSON.stringify({
903
- content,
904
- encoding,
905
- path,
906
- sessionId: targetSessionId,
907
- }),
908
- headers: {
909
- "Content-Type": "application/json",
910
- },
911
- method: "POST",
912
- });
913
-
914
- if (!response.ok) {
915
- const errorData = (await response.json().catch(() => ({}))) as {
916
- error?: string;
917
- };
918
- throw new Error(
919
- errorData.error || `HTTP error! status: ${response.status}`
920
- );
921
- }
922
-
923
- if (!response.body) {
924
- throw new Error("No response body for streaming request");
925
- }
926
-
927
- const reader = response.body.getReader();
928
- const decoder = new TextDecoder();
929
-
930
- try {
931
- while (true) {
932
- const { done, value } = await reader.read();
933
-
934
- if (done) {
935
- break;
936
- }
937
-
938
- const chunk = decoder.decode(value, { stream: true });
939
- const lines = chunk.split("\n");
940
-
941
- for (const line of lines) {
942
- if (line.startsWith("data: ")) {
943
- try {
944
- const eventData = line.slice(6); // Remove 'data: ' prefix
945
- const event: StreamEvent = JSON.parse(eventData);
946
-
947
- console.log(
948
- `[HTTP Client] Write file stream event: ${event.type}`
949
- );
950
- this.options.onStreamEvent?.(event);
951
-
952
- switch (event.type) {
953
- case "command_start":
954
- console.log(
955
- `[HTTP Client] Write file started: ${event.path}`
956
- );
957
- this.options.onCommandStart?.("write", [
958
- path,
959
- content,
960
- encoding,
961
- ]);
962
- break;
963
-
964
- case "output":
965
- console.log(`[output] ${event.message}`);
966
- this.options.onOutput?.("stdout", event.message!, "write");
967
- break;
968
-
969
- case "command_complete":
970
- console.log(
971
- `[HTTP Client] Write file completed: ${event.path}, Success: ${event.success}`
972
- );
973
- this.options.onCommandComplete?.(
974
- event.success!,
975
- 0,
976
- "",
977
- "",
978
- "write",
979
- [path, content, encoding]
980
- );
981
- break;
982
-
983
- case "error":
984
- console.error(
985
- `[HTTP Client] Write file error: ${event.error}`
986
- );
987
- this.options.onError?.(event.error!, "write", [
988
- path,
989
- content,
990
- encoding,
991
- ]);
992
- break;
993
- }
994
- } catch (parseError) {
995
- console.warn(
996
- "[HTTP Client] Failed to parse write file stream event:",
997
- parseError
998
- );
999
- }
1000
- }
1001
- }
1002
- }
1003
- } finally {
1004
- reader.releaseLock();
1005
- }
1006
- } catch (error) {
1007
- console.error("[HTTP Client] Error in streaming write file:", error);
1008
- this.options.onError?.(
1009
- error instanceof Error ? error.message : "Unknown error",
1010
- "write",
1011
- [path, content, encoding]
1012
- );
1013
- throw error;
1014
- }
1015
- }
1016
-
1017
- async readFile(
1018
- path: string,
1019
- encoding: string = "utf-8",
1020
- sessionId?: string
1021
- ): Promise<ReadFileResponse> {
1022
- try {
1023
- const targetSessionId = sessionId || this.sessionId;
1024
-
1025
- const response = await this.doFetch(`/api/read`, {
1026
- body: JSON.stringify({
1027
- encoding,
1028
- path,
1029
- sessionId: targetSessionId,
1030
- }),
1031
- headers: {
1032
- "Content-Type": "application/json",
1033
- },
1034
- method: "POST",
1035
- });
1036
-
1037
- if (!response.ok) {
1038
- const errorData = (await response.json().catch(() => ({}))) as {
1039
- error?: string;
1040
- };
1041
- throw new Error(
1042
- errorData.error || `HTTP error! status: ${response.status}`
1043
- );
1044
- }
1045
-
1046
- const data: ReadFileResponse = await response.json();
1047
- console.log(
1048
- `[HTTP Client] File read: ${path}, Success: ${data.success}, Content length: ${data.content.length}`
1049
- );
1050
-
1051
- return data;
1052
- } catch (error) {
1053
- console.error("[HTTP Client] Error reading file:", error);
1054
- throw error;
1055
- }
1056
- }
1057
-
1058
- async readFileStream(
1059
- path: string,
1060
- encoding: string = "utf-8",
1061
- sessionId?: string
1062
- ): Promise<void> {
1063
- try {
1064
- const targetSessionId = sessionId || this.sessionId;
1065
-
1066
- const response = await this.doFetch(`/api/read/stream`, {
1067
- body: JSON.stringify({
1068
- encoding,
1069
- path,
1070
- sessionId: targetSessionId,
1071
- }),
1072
- headers: {
1073
- "Content-Type": "application/json",
1074
- },
1075
- method: "POST",
1076
- });
1077
-
1078
- if (!response.ok) {
1079
- const errorData = (await response.json().catch(() => ({}))) as {
1080
- error?: string;
1081
- };
1082
- throw new Error(
1083
- errorData.error || `HTTP error! status: ${response.status}`
1084
- );
1085
- }
1086
-
1087
- if (!response.body) {
1088
- throw new Error("No response body for streaming request");
1089
- }
1090
-
1091
- const reader = response.body.getReader();
1092
- const decoder = new TextDecoder();
1093
-
1094
- try {
1095
- while (true) {
1096
- const { done, value } = await reader.read();
1097
-
1098
- if (done) {
1099
- break;
1100
- }
1101
-
1102
- const chunk = decoder.decode(value, { stream: true });
1103
- const lines = chunk.split("\n");
1104
-
1105
- for (const line of lines) {
1106
- if (line.startsWith("data: ")) {
1107
- try {
1108
- const eventData = line.slice(6); // Remove 'data: ' prefix
1109
- const event: StreamEvent = JSON.parse(eventData);
1110
-
1111
- console.log(
1112
- `[HTTP Client] Read file stream event: ${event.type}`
1113
- );
1114
- this.options.onStreamEvent?.(event);
1115
-
1116
- switch (event.type) {
1117
- case "command_start":
1118
- console.log(
1119
- `[HTTP Client] Read file started: ${event.path}`
1120
- );
1121
- this.options.onCommandStart?.("read", [path, encoding]);
1122
- break;
1123
-
1124
- case "command_complete":
1125
- console.log(
1126
- `[HTTP Client] Read file completed: ${
1127
- event.path
1128
- }, Success: ${event.success}, Content length: ${
1129
- event.content?.length || 0
1130
- }`
1131
- );
1132
- this.options.onCommandComplete?.(
1133
- event.success!,
1134
- 0,
1135
- event.content || "",
1136
- "",
1137
- "read",
1138
- [path, encoding]
1139
- );
1140
- break;
1141
-
1142
- case "error":
1143
- console.error(
1144
- `[HTTP Client] Read file error: ${event.error}`
1145
- );
1146
- this.options.onError?.(event.error!, "read", [
1147
- path,
1148
- encoding,
1149
- ]);
1150
- break;
1151
- }
1152
- } catch (parseError) {
1153
- console.warn(
1154
- "[HTTP Client] Failed to parse read file stream event:",
1155
- parseError
1156
- );
1157
- }
1158
- }
1159
- }
1160
- }
1161
- } finally {
1162
- reader.releaseLock();
1163
- }
1164
- } catch (error) {
1165
- console.error("[HTTP Client] Error in streaming read file:", error);
1166
- this.options.onError?.(
1167
- error instanceof Error ? error.message : "Unknown error",
1168
- "read",
1169
- [path, encoding]
1170
- );
1171
- throw error;
1172
- }
1173
- }
1174
-
1175
- async deleteFile(
1176
- path: string,
1177
- sessionId?: string
1178
- ): Promise<DeleteFileResponse> {
1179
- try {
1180
- const targetSessionId = sessionId || this.sessionId;
1181
-
1182
- const response = await this.doFetch(`/api/delete`, {
1183
- body: JSON.stringify({
1184
- path,
1185
- sessionId: targetSessionId,
1186
- }),
1187
- headers: {
1188
- "Content-Type": "application/json",
1189
- },
1190
- method: "POST",
1191
- });
1192
-
1193
- if (!response.ok) {
1194
- const errorData = (await response.json().catch(() => ({}))) as {
1195
- error?: string;
1196
- };
1197
- throw new Error(
1198
- errorData.error || `HTTP error! status: ${response.status}`
1199
- );
1200
- }
1201
-
1202
- const data: DeleteFileResponse = await response.json();
1203
- console.log(
1204
- `[HTTP Client] File deleted: ${path}, Success: ${data.success}`
1205
- );
1206
-
1207
- return data;
1208
- } catch (error) {
1209
- console.error("[HTTP Client] Error deleting file:", error);
1210
- throw error;
1211
- }
1212
- }
1213
-
1214
- async deleteFileStream(path: string, sessionId?: string): Promise<void> {
1215
- try {
1216
- const targetSessionId = sessionId || this.sessionId;
1217
-
1218
- const response = await this.doFetch(`/api/delete/stream`, {
1219
- body: JSON.stringify({
1220
- path,
1221
- sessionId: targetSessionId,
1222
- }),
1223
- headers: {
1224
- "Content-Type": "application/json",
1225
- },
1226
- method: "POST",
1227
- });
1228
-
1229
- if (!response.ok) {
1230
- const errorData = (await response.json().catch(() => ({}))) as {
1231
- error?: string;
1232
- };
1233
- throw new Error(
1234
- errorData.error || `HTTP error! status: ${response.status}`
1235
- );
1236
- }
1237
-
1238
- if (!response.body) {
1239
- throw new Error("No response body for streaming request");
1240
- }
1241
-
1242
- const reader = response.body.getReader();
1243
- const decoder = new TextDecoder();
1244
-
1245
- try {
1246
- while (true) {
1247
- const { done, value } = await reader.read();
1248
-
1249
- if (done) {
1250
- break;
1251
- }
1252
-
1253
- const chunk = decoder.decode(value, { stream: true });
1254
- const lines = chunk.split("\n");
1255
-
1256
- for (const line of lines) {
1257
- if (line.startsWith("data: ")) {
1258
- try {
1259
- const eventData = line.slice(6); // Remove 'data: ' prefix
1260
- const event: StreamEvent = JSON.parse(eventData);
1261
-
1262
- console.log(
1263
- `[HTTP Client] Delete file stream event: ${event.type}`
1264
- );
1265
- this.options.onStreamEvent?.(event);
1266
-
1267
- switch (event.type) {
1268
- case "command_start":
1269
- console.log(
1270
- `[HTTP Client] Delete file started: ${event.path}`
1271
- );
1272
- this.options.onCommandStart?.("delete", [path]);
1273
- break;
1274
-
1275
- case "command_complete":
1276
- console.log(
1277
- `[HTTP Client] Delete file completed: ${event.path}, Success: ${event.success}`
1278
- );
1279
- this.options.onCommandComplete?.(
1280
- event.success!,
1281
- 0,
1282
- "",
1283
- "",
1284
- "delete",
1285
- [path]
1286
- );
1287
- break;
1288
-
1289
- case "error":
1290
- console.error(
1291
- `[HTTP Client] Delete file error: ${event.error}`
1292
- );
1293
- this.options.onError?.(event.error!, "delete", [path]);
1294
- break;
1295
- }
1296
- } catch (parseError) {
1297
- console.warn(
1298
- "[HTTP Client] Failed to parse delete file stream event:",
1299
- parseError
1300
- );
1301
- }
1302
- }
1303
- }
1304
- }
1305
- } finally {
1306
- reader.releaseLock();
1307
- }
1308
- } catch (error) {
1309
- console.error("[HTTP Client] Error in streaming delete file:", error);
1310
- this.options.onError?.(
1311
- error instanceof Error ? error.message : "Unknown error",
1312
- "delete",
1313
- [path]
1314
- );
1315
- throw error;
1316
- }
1317
- }
1318
-
1319
- async renameFile(
1320
- oldPath: string,
1321
- newPath: string,
1322
- sessionId?: string
1323
- ): Promise<RenameFileResponse> {
1324
- try {
1325
- const targetSessionId = sessionId || this.sessionId;
1326
-
1327
- const response = await this.doFetch(`/api/rename`, {
1328
- body: JSON.stringify({
1329
- newPath,
1330
- oldPath,
1331
- sessionId: targetSessionId,
1332
- }),
1333
- headers: {
1334
- "Content-Type": "application/json",
1335
- },
1336
- method: "POST",
1337
- });
1338
-
1339
- if (!response.ok) {
1340
- const errorData = (await response.json().catch(() => ({}))) as {
1341
- error?: string;
1342
- };
1343
- throw new Error(
1344
- errorData.error || `HTTP error! status: ${response.status}`
1345
- );
1346
- }
1347
-
1348
- const data: RenameFileResponse = await response.json();
1349
- console.log(
1350
- `[HTTP Client] File renamed: ${oldPath} -> ${newPath}, Success: ${data.success}`
1351
- );
1352
-
1353
- return data;
1354
- } catch (error) {
1355
- console.error("[HTTP Client] Error renaming file:", error);
1356
- throw error;
1357
- }
1358
- }
1359
-
1360
- async renameFileStream(
1361
- oldPath: string,
1362
- newPath: string,
1363
- sessionId?: string
1364
- ): Promise<void> {
1365
- try {
1366
- const targetSessionId = sessionId || this.sessionId;
1367
-
1368
- const response = await this.doFetch(`/api/rename/stream`, {
1369
- body: JSON.stringify({
1370
- newPath,
1371
- oldPath,
1372
- sessionId: targetSessionId,
1373
- }),
1374
- headers: {
1375
- "Content-Type": "application/json",
1376
- },
1377
- method: "POST",
1378
- });
1379
-
1380
- if (!response.ok) {
1381
- const errorData = (await response.json().catch(() => ({}))) as {
1382
- error?: string;
1383
- };
1384
- throw new Error(
1385
- errorData.error || `HTTP error! status: ${response.status}`
1386
- );
1387
- }
1388
-
1389
- if (!response.body) {
1390
- throw new Error("No response body for streaming request");
1391
- }
1392
-
1393
- const reader = response.body.getReader();
1394
- const decoder = new TextDecoder();
1395
-
1396
- try {
1397
- while (true) {
1398
- const { done, value } = await reader.read();
1399
-
1400
- if (done) {
1401
- break;
1402
- }
1403
-
1404
- const chunk = decoder.decode(value, { stream: true });
1405
- const lines = chunk.split("\n");
1406
-
1407
- for (const line of lines) {
1408
- if (line.startsWith("data: ")) {
1409
- try {
1410
- const eventData = line.slice(6); // Remove 'data: ' prefix
1411
- const event: StreamEvent = JSON.parse(eventData);
1412
-
1413
- console.log(
1414
- `[HTTP Client] Rename file stream event: ${event.type}`
1415
- );
1416
- this.options.onStreamEvent?.(event);
1417
-
1418
- switch (event.type) {
1419
- case "command_start":
1420
- console.log(
1421
- `[HTTP Client] Rename file started: ${event.oldPath} -> ${event.newPath}`
1422
- );
1423
- this.options.onCommandStart?.("rename", [oldPath, newPath]);
1424
- break;
1425
-
1426
- case "command_complete":
1427
- console.log(
1428
- `[HTTP Client] Rename file completed: ${event.oldPath} -> ${event.newPath}, Success: ${event.success}`
1429
- );
1430
- this.options.onCommandComplete?.(
1431
- event.success!,
1432
- 0,
1433
- "",
1434
- "",
1435
- "rename",
1436
- [oldPath, newPath]
1437
- );
1438
- break;
1439
-
1440
- case "error":
1441
- console.error(
1442
- `[HTTP Client] Rename file error: ${event.error}`
1443
- );
1444
- this.options.onError?.(event.error!, "rename", [
1445
- oldPath,
1446
- newPath,
1447
- ]);
1448
- break;
1449
- }
1450
- } catch (parseError) {
1451
- console.warn(
1452
- "[HTTP Client] Failed to parse rename file stream event:",
1453
- parseError
1454
- );
1455
- }
1456
- }
1457
- }
1458
- }
1459
- } finally {
1460
- reader.releaseLock();
1461
- }
1462
- } catch (error) {
1463
- console.error("[HTTP Client] Error in streaming rename file:", error);
1464
- this.options.onError?.(
1465
- error instanceof Error ? error.message : "Unknown error",
1466
- "rename",
1467
- [oldPath, newPath]
1468
- );
1469
- throw error;
1470
- }
1471
- }
1472
-
1473
- async moveFile(
1474
- sourcePath: string,
1475
- destinationPath: string,
1476
- sessionId?: string
1477
- ): Promise<MoveFileResponse> {
1478
- try {
1479
- const targetSessionId = sessionId || this.sessionId;
1480
-
1481
- const response = await this.doFetch(`/api/move`, {
1482
- body: JSON.stringify({
1483
- destinationPath,
1484
- sessionId: targetSessionId,
1485
- sourcePath,
1486
- }),
1487
- headers: {
1488
- "Content-Type": "application/json",
1489
- },
1490
- method: "POST",
1491
- });
1492
-
1493
- if (!response.ok) {
1494
- const errorData = (await response.json().catch(() => ({}))) as {
1495
- error?: string;
1496
- };
1497
- throw new Error(
1498
- errorData.error || `HTTP error! status: ${response.status}`
1499
- );
1500
- }
1501
-
1502
- const data: MoveFileResponse = await response.json();
1503
- console.log(
1504
- `[HTTP Client] File moved: ${sourcePath} -> ${destinationPath}, Success: ${data.success}`
1505
- );
1506
-
1507
- return data;
1508
- } catch (error) {
1509
- console.error("[HTTP Client] Error moving file:", error);
1510
- throw error;
1511
- }
1512
- }
1513
-
1514
- async moveFileStream(
1515
- sourcePath: string,
1516
- destinationPath: string,
1517
- sessionId?: string
1518
- ): Promise<void> {
1519
- try {
1520
- const targetSessionId = sessionId || this.sessionId;
1521
-
1522
- const response = await this.doFetch(`/api/move/stream`, {
1523
- body: JSON.stringify({
1524
- destinationPath,
1525
- sessionId: targetSessionId,
1526
- sourcePath,
1527
- }),
1528
- headers: {
1529
- "Content-Type": "application/json",
1530
- },
1531
- method: "POST",
1532
- });
1533
-
1534
- if (!response.ok) {
1535
- const errorData = (await response.json().catch(() => ({}))) as {
1536
- error?: string;
1537
- };
1538
- throw new Error(
1539
- errorData.error || `HTTP error! status: ${response.status}`
1540
- );
1541
- }
1542
-
1543
- if (!response.body) {
1544
- throw new Error("No response body for streaming request");
1545
- }
1546
-
1547
- const reader = response.body.getReader();
1548
- const decoder = new TextDecoder();
1549
-
1550
- try {
1551
- while (true) {
1552
- const { done, value } = await reader.read();
1553
-
1554
- if (done) {
1555
- break;
1556
- }
1557
-
1558
- const chunk = decoder.decode(value, { stream: true });
1559
- const lines = chunk.split("\n");
1560
-
1561
- for (const line of lines) {
1562
- if (line.startsWith("data: ")) {
1563
- try {
1564
- const eventData = line.slice(6); // Remove 'data: ' prefix
1565
- const event: StreamEvent = JSON.parse(eventData);
1566
-
1567
- console.log(
1568
- `[HTTP Client] Move file stream event: ${event.type}`
1569
- );
1570
- this.options.onStreamEvent?.(event);
1571
-
1572
- switch (event.type) {
1573
- case "command_start":
1574
- console.log(
1575
- `[HTTP Client] Move file started: ${event.sourcePath} -> ${event.destinationPath}`
1576
- );
1577
- this.options.onCommandStart?.("move", [
1578
- sourcePath,
1579
- destinationPath,
1580
- ]);
1581
- break;
1582
-
1583
- case "command_complete":
1584
- console.log(
1585
- `[HTTP Client] Move file completed: ${event.sourcePath} -> ${event.destinationPath}, Success: ${event.success}`
1586
- );
1587
- this.options.onCommandComplete?.(
1588
- event.success!,
1589
- 0,
1590
- "",
1591
- "",
1592
- "move",
1593
- [sourcePath, destinationPath]
1594
- );
1595
- break;
1596
-
1597
- case "error":
1598
- console.error(
1599
- `[HTTP Client] Move file error: ${event.error}`
1600
- );
1601
- this.options.onError?.(event.error!, "move", [
1602
- sourcePath,
1603
- destinationPath,
1604
- ]);
1605
- break;
1606
- }
1607
- } catch (parseError) {
1608
- console.warn(
1609
- "[HTTP Client] Failed to parse move file stream event:",
1610
- parseError
1611
- );
1612
- }
1613
- }
1614
- }
1615
- }
1616
- } finally {
1617
- reader.releaseLock();
1618
- }
1619
- } catch (error) {
1620
- console.error("[HTTP Client] Error in streaming move file:", error);
1621
- this.options.onError?.(
1622
- error instanceof Error ? error.message : "Unknown error",
1623
- "move",
1624
- [sourcePath, destinationPath]
1625
- );
1626
- throw error;
1627
- }
1628
- }
1629
-
1630
- async ping(): Promise<string> {
1631
- try {
1632
- const response = await this.doFetch(`/api/ping`, {
1633
- headers: {
1634
- "Content-Type": "application/json",
1635
- },
1636
- method: "GET",
1637
- });
1638
-
1639
- if (!response.ok) {
1640
- throw new Error(`HTTP error! status: ${response.status}`);
1641
- }
1642
-
1643
- const data: PingResponse = await response.json();
1644
- console.log(`[HTTP Client] Ping response: ${data.message}`);
1645
- return data.timestamp;
1646
- } catch (error) {
1647
- console.error("[HTTP Client] Error pinging server:", error);
1648
- throw error;
1649
- }
1650
- }
1651
-
1652
- async getCommands(): Promise<string[]> {
1653
- try {
1654
- const response = await fetch(`${this.baseUrl}/api/commands`, {
1655
- headers: {
1656
- "Content-Type": "application/json",
1657
- },
1658
- method: "GET",
1659
- });
1660
-
1661
- if (!response.ok) {
1662
- throw new Error(`HTTP error! status: ${response.status}`);
1663
- }
1664
-
1665
- const data: CommandsResponse = await response.json();
1666
- console.log(
1667
- `[HTTP Client] Available commands: ${data.availableCommands.length}`
1668
- );
1669
- return data.availableCommands;
1670
- } catch (error) {
1671
- console.error("[HTTP Client] Error getting commands:", error);
1672
- throw error;
1673
- }
1674
- }
1675
-
1676
- getSessionId(): string | null {
1677
- return this.sessionId;
1678
- }
1679
-
1680
- setSessionId(sessionId: string): void {
1681
- this.sessionId = sessionId;
1682
- }
1683
-
1684
- clearSession(): void {
1685
- this.sessionId = null;
1686
- }
1687
- }
1688
-
1689
- // Example usage and utility functions
1690
- export function createClient(options?: HttpClientOptions): HttpClient {
1691
- return new HttpClient(options);
1692
- }
1693
-
1694
- // Convenience function for quick command execution
1695
- export async function quickExecute(
1696
- command: string,
1697
- args: string[] = [],
1698
- options?: HttpClientOptions
1699
- ): Promise<ExecuteResponse> {
1700
- const client = createClient(options);
1701
- await client.createSession();
1702
-
1703
- try {
1704
- return await client.execute(command, args);
1705
- } finally {
1706
- client.clearSession();
1707
- }
1708
- }
1709
-
1710
- // Convenience function for quick streaming command execution
1711
- export async function quickExecuteStream(
1712
- command: string,
1713
- args: string[] = [],
1714
- options?: HttpClientOptions
1715
- ): Promise<void> {
1716
- const client = createClient(options);
1717
- await client.createSession();
1718
-
1719
- try {
1720
- await client.executeStream(command, args);
1721
- } finally {
1722
- client.clearSession();
1723
- }
1724
- }
1725
-
1726
- // Convenience function for quick git checkout
1727
- export async function quickGitCheckout(
1728
- repoUrl: string,
1729
- branch: string = "main",
1730
- targetDir?: string,
1731
- options?: HttpClientOptions
1732
- ): Promise<GitCheckoutResponse> {
1733
- const client = createClient(options);
1734
- await client.createSession();
1735
-
1736
- try {
1737
- return await client.gitCheckout(repoUrl, branch, targetDir);
1738
- } finally {
1739
- client.clearSession();
1740
- }
1741
- }
1742
-
1743
- // Convenience function for quick directory creation
1744
- export async function quickMkdir(
1745
- path: string,
1746
- recursive: boolean = false,
1747
- options?: HttpClientOptions
1748
- ): Promise<MkdirResponse> {
1749
- const client = createClient(options);
1750
- await client.createSession();
1751
-
1752
- try {
1753
- return await client.mkdir(path, recursive);
1754
- } finally {
1755
- client.clearSession();
1756
- }
1757
- }
1758
-
1759
- // Convenience function for quick streaming git checkout
1760
- export async function quickGitCheckoutStream(
1761
- repoUrl: string,
1762
- branch: string = "main",
1763
- targetDir?: string,
1764
- options?: HttpClientOptions
1765
- ): Promise<void> {
1766
- const client = createClient(options);
1767
- await client.createSession();
1768
-
1769
- try {
1770
- await client.gitCheckoutStream(repoUrl, branch, targetDir);
1771
- } finally {
1772
- client.clearSession();
1773
- }
1774
- }
1775
-
1776
- // Convenience function for quick streaming directory creation
1777
- export async function quickMkdirStream(
1778
- path: string,
1779
- recursive: boolean = false,
1780
- options?: HttpClientOptions
1781
- ): Promise<void> {
1782
- const client = createClient(options);
1783
- await client.createSession();
1784
-
1785
- try {
1786
- await client.mkdirStream(path, recursive);
1787
- } finally {
1788
- client.clearSession();
1789
- }
1790
- }
1791
-
1792
- // Convenience function for quick file writing
1793
- export async function quickWriteFile(
1794
- path: string,
1795
- content: string,
1796
- encoding: string = "utf-8",
1797
- options?: HttpClientOptions
1798
- ): Promise<WriteFileResponse> {
1799
- const client = createClient(options);
1800
- await client.createSession();
1801
-
1802
- try {
1803
- return await client.writeFile(path, content, encoding);
1804
- } finally {
1805
- client.clearSession();
1806
- }
1807
- }
1808
-
1809
- // Convenience function for quick streaming file writing
1810
- export async function quickWriteFileStream(
1811
- path: string,
1812
- content: string,
1813
- encoding: string = "utf-8",
1814
- options?: HttpClientOptions
1815
- ): Promise<void> {
1816
- const client = createClient(options);
1817
- await client.createSession();
1818
-
1819
- try {
1820
- await client.writeFileStream(path, content, encoding);
1821
- } finally {
1822
- client.clearSession();
1823
- }
1824
- }
1825
-
1826
- // Convenience function for quick file reading
1827
- export async function quickReadFile(
1828
- path: string,
1829
- encoding: string = "utf-8",
1830
- options?: HttpClientOptions
1831
- ): Promise<ReadFileResponse> {
1832
- const client = createClient(options);
1833
- await client.createSession();
1834
-
1835
- try {
1836
- return await client.readFile(path, encoding);
1837
- } finally {
1838
- client.clearSession();
1839
- }
1840
- }
1841
-
1842
- // Convenience function for quick streaming file reading
1843
- export async function quickReadFileStream(
1844
- path: string,
1845
- encoding: string = "utf-8",
1846
- options?: HttpClientOptions
1847
- ): Promise<void> {
1848
- const client = createClient(options);
1849
- await client.createSession();
1850
-
1851
- try {
1852
- await client.readFileStream(path, encoding);
1853
- } finally {
1854
- client.clearSession();
1855
- }
1856
- }
1857
-
1858
- // Convenience function for quick file deletion
1859
- export async function quickDeleteFile(
1860
- path: string,
1861
- options?: HttpClientOptions
1862
- ): Promise<DeleteFileResponse> {
1863
- const client = createClient(options);
1864
- await client.createSession();
1865
-
1866
- try {
1867
- return await client.deleteFile(path);
1868
- } finally {
1869
- client.clearSession();
1870
- }
1871
- }
1872
-
1873
- // Convenience function for quick streaming file deletion
1874
- export async function quickDeleteFileStream(
1875
- path: string,
1876
- options?: HttpClientOptions
1877
- ): Promise<void> {
1878
- const client = createClient(options);
1879
- await client.createSession();
1880
-
1881
- try {
1882
- await client.deleteFileStream(path);
1883
- } finally {
1884
- client.clearSession();
1885
- }
1886
- }
1887
-
1888
- // Convenience function for quick file renaming
1889
- export async function quickRenameFile(
1890
- oldPath: string,
1891
- newPath: string,
1892
- options?: HttpClientOptions
1893
- ): Promise<RenameFileResponse> {
1894
- const client = createClient(options);
1895
- await client.createSession();
1896
-
1897
- try {
1898
- return await client.renameFile(oldPath, newPath);
1899
- } finally {
1900
- client.clearSession();
1901
- }
1902
- }
1903
-
1904
- // Convenience function for quick streaming file renaming
1905
- export async function quickRenameFileStream(
1906
- oldPath: string,
1907
- newPath: string,
1908
- options?: HttpClientOptions
1909
- ): Promise<void> {
1910
- const client = createClient(options);
1911
- await client.createSession();
1912
-
1913
- try {
1914
- await client.renameFileStream(oldPath, newPath);
1915
- } finally {
1916
- client.clearSession();
1917
- }
1918
- }
1919
-
1920
- // Convenience function for quick file moving
1921
- export async function quickMoveFile(
1922
- sourcePath: string,
1923
- destinationPath: string,
1924
- options?: HttpClientOptions
1925
- ): Promise<MoveFileResponse> {
1926
- const client = createClient(options);
1927
- await client.createSession();
1928
-
1929
- try {
1930
- return await client.moveFile(sourcePath, destinationPath);
1931
- } finally {
1932
- client.clearSession();
1933
- }
1934
- }
1935
-
1936
- // Convenience function for quick streaming file moving
1937
- export async function quickMoveFileStream(
1938
- sourcePath: string,
1939
- destinationPath: string,
1940
- options?: HttpClientOptions
1941
- ): Promise<void> {
1942
- const client = createClient(options);
1943
- await client.createSession();
1944
-
1945
- try {
1946
- await client.moveFileStream(sourcePath, destinationPath);
1947
- } finally {
1948
- client.clearSession();
1949
- }
1950
- }