@heyputer/puter.js 1.0.1 → 2.0.1

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/README.md +14 -43
  2. package/index.d.ts +479 -0
  3. package/package.json +13 -3
  4. package/APACHE_LICENSE.txt +0 -201
  5. package/doc/devlog.md +0 -49
  6. package/src/bg.png +0 -0
  7. package/src/bg.webp +0 -0
  8. package/src/lib/APICallLogger.js +0 -110
  9. package/src/lib/EventListener.js +0 -51
  10. package/src/lib/RequestError.js +0 -6
  11. package/src/lib/filesystem/APIFS.js +0 -73
  12. package/src/lib/filesystem/CacheFS.js +0 -243
  13. package/src/lib/filesystem/PostMessageFS.js +0 -40
  14. package/src/lib/filesystem/definitions.js +0 -39
  15. package/src/lib/path.js +0 -509
  16. package/src/lib/polyfills/localStorage.js +0 -92
  17. package/src/lib/polyfills/xhrshim.js +0 -233
  18. package/src/lib/socket.io/socket.io.esm.min.js +0 -7
  19. package/src/lib/socket.io/socket.io.esm.min.js.map +0 -1
  20. package/src/lib/socket.io/socket.io.js +0 -4385
  21. package/src/lib/socket.io/socket.io.js.map +0 -1
  22. package/src/lib/socket.io/socket.io.min.js +0 -7
  23. package/src/lib/socket.io/socket.io.min.js.map +0 -1
  24. package/src/lib/socket.io/socket.io.msgpack.min.js +0 -7
  25. package/src/lib/socket.io/socket.io.msgpack.min.js.map +0 -1
  26. package/src/lib/utils.js +0 -620
  27. package/src/lib/xdrpc.js +0 -104
  28. package/src/modules/AI.js +0 -680
  29. package/src/modules/Apps.js +0 -215
  30. package/src/modules/Auth.js +0 -171
  31. package/src/modules/Debug.js +0 -39
  32. package/src/modules/Drivers.js +0 -278
  33. package/src/modules/FSItem.js +0 -139
  34. package/src/modules/FileSystem/index.js +0 -187
  35. package/src/modules/FileSystem/operations/copy.js +0 -64
  36. package/src/modules/FileSystem/operations/deleteFSEntry.js +0 -59
  37. package/src/modules/FileSystem/operations/getReadUrl.js +0 -42
  38. package/src/modules/FileSystem/operations/mkdir.js +0 -62
  39. package/src/modules/FileSystem/operations/move.js +0 -75
  40. package/src/modules/FileSystem/operations/read.js +0 -46
  41. package/src/modules/FileSystem/operations/readdir.js +0 -102
  42. package/src/modules/FileSystem/operations/rename.js +0 -58
  43. package/src/modules/FileSystem/operations/sign.js +0 -103
  44. package/src/modules/FileSystem/operations/space.js +0 -40
  45. package/src/modules/FileSystem/operations/stat.js +0 -95
  46. package/src/modules/FileSystem/operations/symlink.js +0 -55
  47. package/src/modules/FileSystem/operations/upload.js +0 -440
  48. package/src/modules/FileSystem/operations/write.js +0 -65
  49. package/src/modules/FileSystem/utils/getAbsolutePathForApp.js +0 -21
  50. package/src/modules/Hosting.js +0 -138
  51. package/src/modules/KV.js +0 -301
  52. package/src/modules/OS.js +0 -95
  53. package/src/modules/Perms.js +0 -109
  54. package/src/modules/PuterDialog.js +0 -481
  55. package/src/modules/Threads.js +0 -75
  56. package/src/modules/UI.js +0 -1555
  57. package/src/modules/Util.js +0 -38
  58. package/src/modules/Workers.js +0 -120
  59. package/src/modules/networking/PSocket.js +0 -87
  60. package/src/modules/networking/PTLS.js +0 -100
  61. package/src/modules/networking/PWispHandler.js +0 -89
  62. package/src/modules/networking/parsers.js +0 -157
  63. package/src/modules/networking/requests.js +0 -282
  64. package/src/services/APIAccess.js +0 -46
  65. package/src/services/FSRelay.js +0 -20
  66. package/src/services/Filesystem.js +0 -122
  67. package/src/services/NoPuterYet.js +0 -20
  68. package/src/services/XDIncoming.js +0 -44
  69. package/test/ai.test.js +0 -214
  70. package/test/fs.test.js +0 -798
  71. package/test/index.html +0 -1183
  72. package/test/kv.test.js +0 -548
  73. package/test/txt2speech.test.js +0 -178
  74. package/webpack.config.js +0 -25
@@ -1,481 +0,0 @@
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;
@@ -1,75 +0,0 @@
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
- }