@heyputer/puter.js 2.0.1 → 2.0.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (74) hide show
  1. package/APACHE_LICENSE.txt +201 -0
  2. package/doc/devlog.md +49 -0
  3. package/index.d.ts +6 -6
  4. package/package.json +2 -6
  5. package/src/bg.png +0 -0
  6. package/src/bg.webp +0 -0
  7. package/src/entry.js +9 -0
  8. package/src/lib/APICallLogger.js +110 -0
  9. package/src/lib/EventListener.js +51 -0
  10. package/src/lib/RequestError.js +6 -0
  11. package/src/lib/filesystem/APIFS.js +73 -0
  12. package/src/lib/filesystem/CacheFS.js +243 -0
  13. package/src/lib/filesystem/PostMessageFS.js +40 -0
  14. package/src/lib/filesystem/definitions.js +39 -0
  15. package/src/lib/path.js +509 -0
  16. package/src/lib/polyfills/localStorage.js +92 -0
  17. package/src/lib/polyfills/xhrshim.js +233 -0
  18. package/src/lib/socket.io/socket.io.esm.min.js +7 -0
  19. package/src/lib/socket.io/socket.io.esm.min.js.map +1 -0
  20. package/src/lib/socket.io/socket.io.js +4385 -0
  21. package/src/lib/socket.io/socket.io.js.map +1 -0
  22. package/src/lib/socket.io/socket.io.min.js +7 -0
  23. package/src/lib/socket.io/socket.io.min.js.map +1 -0
  24. package/src/lib/socket.io/socket.io.msgpack.min.js +7 -0
  25. package/src/lib/socket.io/socket.io.msgpack.min.js.map +1 -0
  26. package/src/lib/utils.js +620 -0
  27. package/src/lib/xdrpc.js +104 -0
  28. package/src/modules/AI.js +680 -0
  29. package/src/modules/Apps.js +215 -0
  30. package/src/modules/Auth.js +171 -0
  31. package/src/modules/Debug.js +39 -0
  32. package/src/modules/Drivers.js +278 -0
  33. package/src/modules/FSItem.js +139 -0
  34. package/src/modules/FileSystem/index.js +187 -0
  35. package/src/modules/FileSystem/operations/copy.js +64 -0
  36. package/src/modules/FileSystem/operations/deleteFSEntry.js +59 -0
  37. package/src/modules/FileSystem/operations/getReadUrl.js +42 -0
  38. package/src/modules/FileSystem/operations/mkdir.js +62 -0
  39. package/src/modules/FileSystem/operations/move.js +75 -0
  40. package/src/modules/FileSystem/operations/read.js +46 -0
  41. package/src/modules/FileSystem/operations/readdir.js +102 -0
  42. package/src/modules/FileSystem/operations/rename.js +58 -0
  43. package/src/modules/FileSystem/operations/sign.js +103 -0
  44. package/src/modules/FileSystem/operations/space.js +40 -0
  45. package/src/modules/FileSystem/operations/stat.js +95 -0
  46. package/src/modules/FileSystem/operations/symlink.js +55 -0
  47. package/src/modules/FileSystem/operations/upload.js +440 -0
  48. package/src/modules/FileSystem/operations/write.js +65 -0
  49. package/src/modules/FileSystem/utils/getAbsolutePathForApp.js +21 -0
  50. package/src/modules/Hosting.js +138 -0
  51. package/src/modules/KV.js +301 -0
  52. package/src/modules/OS.js +95 -0
  53. package/src/modules/Perms.js +109 -0
  54. package/src/modules/PuterDialog.js +481 -0
  55. package/src/modules/Threads.js +75 -0
  56. package/src/modules/UI.js +1555 -0
  57. package/src/modules/Util.js +38 -0
  58. package/src/modules/Workers.js +120 -0
  59. package/src/modules/networking/PSocket.js +87 -0
  60. package/src/modules/networking/PTLS.js +100 -0
  61. package/src/modules/networking/PWispHandler.js +89 -0
  62. package/src/modules/networking/parsers.js +157 -0
  63. package/src/modules/networking/requests.js +282 -0
  64. package/src/services/APIAccess.js +46 -0
  65. package/src/services/FSRelay.js +20 -0
  66. package/src/services/Filesystem.js +122 -0
  67. package/src/services/NoPuterYet.js +20 -0
  68. package/src/services/XDIncoming.js +44 -0
  69. package/test/ai.test.js +214 -0
  70. package/test/fs.test.js +798 -0
  71. package/test/index.html +1183 -0
  72. package/test/kv.test.js +548 -0
  73. package/test/txt2speech.test.js +178 -0
  74. package/webpack.config.js +25 -0
@@ -0,0 +1,481 @@
1
+ class PuterDialog extends (globalThis.HTMLElement || Object) { // It will fall back to only extending Object in environments without a DOM
2
+ /**
3
+ * Detects if the current page is loaded using the file:// protocol.
4
+ * @returns {boolean} True if using file:// protocol, false otherwise.
5
+ */
6
+ isUsingFileProtocol = ()=>{
7
+ return window.location.protocol === 'file:';
8
+ }
9
+
10
+
11
+ constructor(resolve, reject) {
12
+ super();
13
+ this.reject = reject;
14
+ this.resolve = resolve;
15
+ this.popupLaunched = false; // Track if popup was successfully launched
16
+
17
+ /**
18
+ * Detects if there's a recent user activation that would allow popup opening
19
+ * @returns {boolean} True if user activation is available, false otherwise.
20
+ */
21
+ this.hasUserActivation = () => {
22
+ // Modern browsers support navigator.userActivation
23
+ if (navigator.userActivation) {
24
+ return navigator.userActivation.hasBeenActive && navigator.userActivation.isActive;
25
+ }
26
+
27
+ // Fallback: try to detect user activation by attempting to open a popup
28
+ // This is a bit hacky but works as a fallback
29
+ try {
30
+ const testPopup = window.open('', '_blank', 'width=1,height=1,left=-1000,top=-1000');
31
+ if (testPopup) {
32
+ testPopup.close();
33
+ return true;
34
+ }
35
+ return false;
36
+ } catch (e) {
37
+ return false;
38
+ }
39
+ }
40
+
41
+ /**
42
+ * Launches the authentication popup window
43
+ * @returns {Window|null} The popup window reference or null if failed
44
+ */
45
+ this.launchPopup = () => {
46
+ try {
47
+ let w = 600;
48
+ let h = 400;
49
+ let title = 'Puter';
50
+ var left = (screen.width/2)-(w/2);
51
+ var top = (screen.height/2)-(h/2);
52
+ const popup = window.open(
53
+ puter.defaultGUIOrigin + '/?embedded_in_popup=true&request_auth=true' + (window.crossOriginIsolated ? '&cross_origin_isolated=true' : ''),
54
+ title,
55
+ 'toolbar=no, location=no, directories=no, status=no, menubar=no, scrollbars=no, resizable=no, copyhistory=no, width='+w+', height='+h+', top='+top+', left='+left
56
+ );
57
+ return popup;
58
+ } catch (e) {
59
+ console.error('Failed to open popup:', e);
60
+ return null;
61
+ }
62
+ }
63
+
64
+ this.attachShadow({ mode: 'open' });
65
+
66
+ let h;
67
+ // Dialog
68
+ h = `
69
+ <style>
70
+ dialog{
71
+ background: transparent;
72
+ border: none;
73
+ box-shadow: none;
74
+ outline: none;
75
+ }
76
+ .puter-dialog-content {
77
+ border: 1px solid #e8e8e8;
78
+ border-radius: 8px;
79
+ padding: 20px;
80
+ background: white;
81
+ box-shadow: 0 0 9px 1px rgb(0 0 0 / 21%);
82
+ padding: 80px 20px;
83
+ -webkit-font-smoothing: antialiased;
84
+ color: #575762;
85
+ position: relative;
86
+ background-image: url('');
87
+ background-repeat: no-repeat;
88
+ background-position: center center;
89
+ background-size: 100% 100%;
90
+ background-color: #fff;
91
+ }
92
+
93
+ dialog * {
94
+ max-width: 500px;
95
+ font-family: "Helvetica Neue", HelveticaNeue, Helvetica, Arial, sans-serif;
96
+ }
97
+
98
+ dialog p.about{
99
+ text-align: center;
100
+ font-size: 17px;
101
+ padding: 10px 30px;
102
+ font-weight: 400;
103
+ -webkit-font-smoothing: antialiased;
104
+ color: #404048;
105
+ box-sizing: border-box;
106
+ }
107
+
108
+ dialog .buttons{
109
+ display: flex;
110
+ justify-content: center;
111
+ align-items: center;
112
+ flex-wrap: wrap;
113
+ margin-top: 20px;
114
+ text-align: center;
115
+ margin-bottom: 20px;
116
+ }
117
+
118
+ .launch-auth-popup-footnote{
119
+ font-size: 11px;
120
+ color: #666;
121
+ margin-top: 10px;
122
+ /* footer at the bottom */
123
+ position: absolute;
124
+ left: 0;
125
+ right: 0;
126
+ bottom: 10px;
127
+ text-align: center;
128
+ margin: 0 10px;
129
+ }
130
+
131
+ dialog .close-btn{
132
+ position: absolute;
133
+ right: 15px;
134
+ top: 10px;
135
+ font-size: 17px;
136
+ color: #8a8a8a;
137
+ cursor: pointer;
138
+ }
139
+
140
+ dialog .close-btn:hover{
141
+ color: #000;
142
+ }
143
+
144
+ /* ------------------------------------
145
+ Button
146
+ ------------------------------------*/
147
+
148
+ dialog .button {
149
+ color: #666666;
150
+ background-color: #eeeeee;
151
+ border-color: #eeeeee;
152
+ font-size: 14px;
153
+ text-decoration: none;
154
+ text-align: center;
155
+ line-height: 40px;
156
+ height: 35px;
157
+ padding: 0 30px;
158
+ margin: 0;
159
+ display: inline-block;
160
+ appearance: none;
161
+ cursor: pointer;
162
+ border: none;
163
+ -webkit-box-sizing: border-box;
164
+ -moz-box-sizing: border-box;
165
+ box-sizing: border-box;
166
+ border-color: #b9b9b9;
167
+ border-style: solid;
168
+ border-width: 1px;
169
+ line-height: 35px;
170
+ background: -webkit-gradient(linear, left top, left bottom, from(#f6f6f6), to(#e1e1e1));
171
+ background: linear-gradient(#f6f6f6, #e1e1e1);
172
+ -webkit-box-shadow: inset 0px 1px 0px rgb(255 255 255 / 30%), 0 1px 2px rgb(0 0 0 / 15%);
173
+ box-shadow: inset 0px 1px 0px rgb(255 255 255 / 30%), 0 1px 2px rgb(0 0 0 / 15%);
174
+ border-radius: 4px;
175
+ outline: none;
176
+ -webkit-font-smoothing: antialiased;
177
+ }
178
+
179
+ dialog .button:focus-visible {
180
+ border-color: rgb(118 118 118);
181
+ }
182
+
183
+ dialog .button:active, dialog .button.active, dialog .button.is-active, dialog .button.has-open-contextmenu {
184
+ text-decoration: none;
185
+ background-color: #eeeeee;
186
+ border-color: #cfcfcf;
187
+ color: #a9a9a9;
188
+ -webkit-transition-duration: 0s;
189
+ transition-duration: 0s;
190
+ -webkit-box-shadow: inset 0 1px 3px rgb(0 0 0 / 20%);
191
+ box-shadow: inset 0px 2px 3px rgb(0 0 0 / 36%), 0px 1px 0px white;
192
+ }
193
+
194
+ dialog .button.disabled, dialog .button.is-disabled, dialog .button:disabled {
195
+ top: 0 !important;
196
+ background: #EEE !important;
197
+ border: 1px solid #DDD !important;
198
+ text-shadow: 0 1px 1px white !important;
199
+ color: #CCC !important;
200
+ cursor: default !important;
201
+ appearance: none !important;
202
+ pointer-events: none;
203
+ }
204
+
205
+ dialog .button-action.disabled, dialog .button-action.is-disabled, dialog .button-action:disabled {
206
+ background: #55a975 !important;
207
+ border: 1px solid #60ab7d !important;
208
+ text-shadow: none !important;
209
+ color: #CCC !important;
210
+ }
211
+
212
+ dialog .button-primary.disabled, dialog .button-primary.is-disabled, dialog .button-primary:disabled {
213
+ background: #8fc2e7 !important;
214
+ border: 1px solid #98adbd !important;
215
+ text-shadow: none !important;
216
+ color: #f5f5f5 !important;
217
+ }
218
+
219
+ dialog .button-block {
220
+ width: 100%;
221
+ }
222
+
223
+ dialog .button-primary {
224
+ border-color: #088ef0;
225
+ background: -webkit-gradient(linear, left top, left bottom, from(#34a5f8), to(#088ef0));
226
+ background: linear-gradient(#34a5f8, #088ef0);
227
+ color: white;
228
+ }
229
+
230
+ dialog .button-danger {
231
+ border-color: #f00808;
232
+ background: -webkit-gradient(linear, left top, left bottom, from(#f83434), to(#f00808));
233
+ background: linear-gradient(#f83434, #f00808);
234
+ color: white;
235
+ }
236
+
237
+ dialog .button-primary:active, dialog .button-primary.active, dialog .button-primary.is-active, dialog .button-primary-flat:active, dialog .button-primary-flat.active, dialog .button-primary-flat.is-active {
238
+ background-color: #2798eb;
239
+ border-color: #2798eb;
240
+ color: #bedef5;
241
+ }
242
+
243
+ dialog .button-action {
244
+ border-color: #08bf4e;
245
+ background: -webkit-gradient(linear, left top, left bottom, from(#29d55d), to(#1ccd60));
246
+ background: linear-gradient(#29d55d, #1ccd60);
247
+ color: white;
248
+ }
249
+
250
+ dialog .button-action:active, dialog .button-action.active, dialog .button-action.is-active, dialog .button-action-flat:active, dialog .button-action-flat.active, dialog .button-action-flat.is-active {
251
+ background-color: #27eb41;
252
+ border-color: #27eb41;
253
+ color: #bef5ca;
254
+ }
255
+
256
+ dialog .button-giant {
257
+ font-size: 28px;
258
+ height: 70px;
259
+ line-height: 70px;
260
+ padding: 0 70px;
261
+ }
262
+
263
+ dialog .button-jumbo {
264
+ font-size: 24px;
265
+ height: 60px;
266
+ line-height: 60px;
267
+ padding: 0 60px;
268
+ }
269
+
270
+ dialog .button-large {
271
+ font-size: 20px;
272
+ height: 50px;
273
+ line-height: 50px;
274
+ padding: 0 50px;
275
+ }
276
+
277
+ dialog .button-normal {
278
+ font-size: 16px;
279
+ height: 40px;
280
+ line-height: 38px;
281
+ padding: 0 40px;
282
+ }
283
+
284
+ dialog .button-small {
285
+ height: 30px;
286
+ line-height: 29px;
287
+ padding: 0 30px;
288
+ }
289
+
290
+ dialog .button-tiny {
291
+ font-size: 9.6px;
292
+ height: 24px;
293
+ line-height: 24px;
294
+ padding: 0 24px;
295
+ }
296
+
297
+ #launch-auth-popup{
298
+ margin-left: 10px;
299
+ width: 200px;
300
+ font-weight: 500;
301
+ font-size: 15px;
302
+ }
303
+ dialog .button-auth{
304
+ margin-bottom: 10px;
305
+ }
306
+ dialog a, dialog a:visited{
307
+ color: rgb(0 69 238);
308
+ text-decoration: none;
309
+ }
310
+ dialog a:hover{
311
+ text-decoration: underline;
312
+ }
313
+
314
+ @media (max-width:480px) {
315
+ .puter-dialog-content{
316
+ padding: 50px 20px;
317
+ }
318
+ dialog .buttons{
319
+ flex-direction: column-reverse;
320
+ }
321
+ dialog p.about{
322
+ padding: 10px 0;
323
+ }
324
+ dialog .button-auth{
325
+ width: 100% !important;
326
+ margin:0 !important;
327
+ margin-bottom: 10px !important;
328
+ }
329
+ }
330
+ .error-container h1 {
331
+ color: #e74c3c;
332
+ font-size: 20px;
333
+ text-align: center;
334
+ }
335
+
336
+ .puter-dialog-content a:focus{
337
+ outline: none;
338
+ }
339
+ </style>`;
340
+ // Error message for unsupported protocol
341
+ if(window.location.protocol === 'file:'){
342
+ h += `<dialog>
343
+ <div class="puter-dialog-content" style="padding: 20px 40px; background:white !important; font-size: 15px;">
344
+ <span class="close-btn">&#x2715</span>
345
+ <div class="error-container">
346
+ <h1>Puter.js Error: Unsupported Protocol</h1>
347
+ <p>It looks like you've opened this file directly in your browser (using the <code style="font-family: monospace;">file:///</code> protocol) which is not supported by Puter.js for security reasons.</p>
348
+ <p>To view this content properly, you need to serve it through a web server. Here are some options:</p>
349
+ <ul>
350
+ <li>Use a local development server (e.g., Python's built-in server or Node.js http-server)</li>
351
+ <li>Upload the files to a web hosting service</li>
352
+ <li>Use a local server application like XAMPP or MAMP</li>
353
+ </ul>
354
+ <p class="help-text">If you're not familiar with these options, consider reaching out to your development team or IT support for assistance.</p>
355
+ </div>
356
+ <p style="margin-top: 30px; border-top: 1px solid #eee; padding-top: 10px; text-align: center; font-size:13px;">
357
+ <a href="https://docs.puter.com" target="_blank">Docs</a><span style="margin:10px; color: #CCC;">|</span>
358
+ <a href="https://github.com/heyPuter/puter/" target="_blank">Github</a><span style="margin:10px; color: #CCC;">|</span>
359
+ <a href="https://discord.com/invite/PQcx7Teh8u" target="_blank">Discord</a>
360
+ </p>
361
+ </div>
362
+ </dialog>`;
363
+ }else{
364
+ h += `<dialog>
365
+ <div class="puter-dialog-content">
366
+ <span class="close-btn">&#x2715</span>
367
+ <a href="https://puter.com" target="_blank" style="border:none; outline:none; display: block; width: 70px; height: 70px; margin: 0 auto; border-radius: 4px;"><img style="display: block; width: 70px; height: 70px; margin: 0 auto; border-radius: 4px;" src=""/></a>
368
+ <p class="about">This website uses Puter to bring you safe, secure, and private AI and Cloud features.</p>
369
+ <div class="buttons">
370
+ <button class="button button-auth" id="launch-auth-popup-cancel">Cancel</button>
371
+ <button class="button button-primary button-auth" id="launch-auth-popup" style="margin-left:10px;">Continue</button>
372
+ </div>
373
+ <p style="text-align: center; margin-top: -15px; font-size: 14px;">Powered by <a href="https://developer.puter.com/?utm_source=sdk-splash" target="_blank">Puter</a></p>
374
+ <p class="launch-auth-popup-footnote">By clicking 'Continue' you agree to Puter's <a href="https://puter.com/terms" target="_blank">Terms of Service</a> and <a href="https://puter.com/privacy" target="_blank">Privacy Policy</a>.</p>
375
+ </div>
376
+ </dialog>`;
377
+ }
378
+
379
+
380
+ this.shadowRoot.innerHTML = h;
381
+
382
+ // Event listener for the 'message' event
383
+ this.messageListener = async (event) => {
384
+ if (event.data.msg === 'puter.token') {
385
+ this.close();
386
+ // Set the authToken property
387
+ puter.setAuthToken(event.data.token);
388
+ // update appID
389
+ puter.setAppID(event.data.app_uid);
390
+ // Remove the event listener to avoid memory leaks
391
+ window.removeEventListener('message', this.messageListener);
392
+
393
+ puter.puterAuthState.authGranted = true;
394
+ // Resolve the promise
395
+ this.resolve();
396
+
397
+ // Call onAuth callback
398
+ if(puter.onAuth && typeof puter.onAuth === 'function'){
399
+ puter.getUser().then((user) => {
400
+ puter.onAuth(user)
401
+ });
402
+ }
403
+
404
+ puter.puterAuthState.isPromptOpen = false;
405
+ // Resolve or reject any waiting promises.
406
+ if (puter.puterAuthState.resolver) {
407
+ if (puter.puterAuthState.authGranted) {
408
+ puter.puterAuthState.resolver.resolve();
409
+ } else {
410
+ puter.puterAuthState.resolver.reject();
411
+ }
412
+ puter.puterAuthState.resolver = null;
413
+ };
414
+ }
415
+ };
416
+
417
+ }
418
+
419
+ // Optional: Handle dialog cancellation as rejection
420
+ cancelListener = () => {
421
+ this.close();
422
+ window.removeEventListener('message', this.messageListener);
423
+ puter.puterAuthState.authGranted = false;
424
+ puter.puterAuthState.isPromptOpen = false;
425
+
426
+ // Reject the promise with an error message indicating user cancellation.
427
+ // This ensures that the calling code's catch block will be triggered.
428
+ this.reject(new Error('User cancelled the authentication'));
429
+
430
+ // If there's a resolver set, use it to reject the waiting promise as well.
431
+ if (puter.puterAuthState.resolver) {
432
+ puter.puterAuthState.resolver.reject(new Error('User cancelled the authentication'));
433
+ puter.puterAuthState.resolver = null;
434
+ }
435
+ };
436
+
437
+ connectedCallback() {
438
+ // Add event listener to the button
439
+ this.shadowRoot.querySelector('#launch-auth-popup')?.addEventListener('click', ()=>{
440
+ let w = 600;
441
+ let h = 400;
442
+ let title = 'Puter';
443
+ var left = (screen.width/2)-(w/2);
444
+ var top = (screen.height/2)-(h/2);
445
+ window.open(puter.defaultGUIOrigin + '/?embedded_in_popup=true&request_auth=true' + (window.crossOriginIsolated ? '&cross_origin_isolated=true' : ''),
446
+ title,
447
+ 'toolbar=no, location=no, directories=no, status=no, menubar=no, scrollbars=no, resizable=no, copyhistory=no, width='+w+', height='+h+', top='+top+', left='+left);
448
+ })
449
+
450
+ // Add the event listener to the window object
451
+ window.addEventListener('message', this.messageListener);
452
+
453
+ // Add event listeners for cancel and close buttons
454
+ this.shadowRoot.querySelector('#launch-auth-popup-cancel')?.addEventListener('click', this.cancelListener);
455
+ this.shadowRoot.querySelector('.close-btn')?.addEventListener('click',this.cancelListener);
456
+ }
457
+
458
+ open() {
459
+ if(this.hasUserActivation()){
460
+ let w = 600;
461
+ let h = 400;
462
+ let title = 'Puter';
463
+ var left = (screen.width/2)-(w/2);
464
+ var top = (screen.height/2)-(h/2);
465
+ window.open(puter.defaultGUIOrigin + '/?embedded_in_popup=true&request_auth=true' + (window.crossOriginIsolated ? '&cross_origin_isolated=true' : ''),
466
+ title,
467
+ 'toolbar=no, location=no, directories=no, status=no, menubar=no, scrollbars=no, resizable=no, copyhistory=no, width='+w+', height='+h+', top='+top+', left='+left);
468
+ }
469
+ else{
470
+ this.shadowRoot.querySelector('dialog').showModal();
471
+ }
472
+ }
473
+
474
+ close() {
475
+ this.shadowRoot.querySelector('dialog').close();
476
+ }
477
+ }
478
+ if (PuterDialog.__proto__ === globalThis.HTMLElement)
479
+ customElements.define('puter-dialog', PuterDialog);
480
+
481
+ export default PuterDialog;
@@ -0,0 +1,75 @@
1
+ import { RequestError } from "../lib/RequestError.js";
2
+
3
+ export default class Threads {
4
+ constructor (context) {
5
+ this.authToken = context.authToken;
6
+ this.APIOrigin = context.APIOrigin;
7
+ }
8
+ setAuthToken (authToken) {
9
+ this.authToken = authToken;
10
+ }
11
+ setAPIOrigin (APIOrigin) {
12
+ this.APIOrigin = APIOrigin;
13
+ }
14
+ async req_ (method, route, body) {
15
+ const resp = await fetch(
16
+ this.APIOrigin + route, {
17
+ method,
18
+ headers: {
19
+ Authorization: `Bearer ${this.authToken}`,
20
+ ...(body ? { 'Content-Type': 'application/json' } : {}),
21
+ },
22
+ ...(body ? { body: JSON.stringify(body) } : {}),
23
+ }
24
+ );
25
+ if ( ! resp.ok ) {
26
+ const resp_data = await resp.json();
27
+ const err = new RequestError(resp_data.message);
28
+ err.response = resp_data;
29
+ throw err;
30
+ }
31
+ return await resp.json();
32
+ }
33
+
34
+ async create (spec, parent) {
35
+ if ( typeof spec === 'string' ) spec = { text: spec };
36
+ return await this.req_('POST', '/threads/create', {
37
+ ...spec,
38
+ ...(parent ? { parent } : {}),
39
+ });
40
+ }
41
+
42
+ async edit (uid, spec = {}) {
43
+ if ( typeof spec === 'string' ) spec = { text: spec };
44
+ await this.req_('PUT', '/threads/edit/' + encodeURIComponent(uid), {
45
+ ...spec,
46
+ });
47
+ }
48
+
49
+ async delete (uid) {
50
+ await this.req_('DELETE', '/threads/' + encodeURIComponent(uid));
51
+ }
52
+
53
+ async list (uid, page, options) {
54
+ return await this.req_('POST',
55
+ '/threads/list/' + encodeURIComponent(uid) + '/' + page,
56
+ options ?? {},
57
+ );
58
+ }
59
+
60
+ async subscribe (uid, callback) {
61
+ puter.fs.socket.emit('thread.sub-request', { uid });
62
+
63
+ // socket.io, which we use unfortunatelly, doesn't handle
64
+ // wildcard events, so we have to just put them all here.
65
+ const events = [
66
+ 'post', 'edit', 'delete', 'child-edit', 'child-delete',
67
+ ];
68
+
69
+ for ( const event of events ) {
70
+ puter.fs.socket.on(`thread.${event}`, (data) => {
71
+ if ( data.subscription === uid ) callback(event, data);
72
+ });
73
+ }
74
+ }
75
+ }