@heyputer/puter.js 2.0.1 → 2.0.3
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.
- package/dist/puter.js +4 -0
- package/package.json +4 -4
- package/src/bg.png +0 -0
- package/src/bg.webp +0 -0
- package/src/index.js +165 -165
- package/src/lib/APICallLogger.js +110 -0
- package/src/lib/EventListener.js +51 -0
- package/src/lib/RequestError.js +6 -0
- package/src/lib/filesystem/APIFS.js +73 -0
- package/src/lib/filesystem/CacheFS.js +243 -0
- package/src/lib/filesystem/PostMessageFS.js +40 -0
- package/src/lib/filesystem/definitions.js +39 -0
- package/src/lib/path.js +509 -0
- package/src/lib/polyfills/localStorage.js +92 -0
- package/src/lib/polyfills/xhrshim.js +233 -0
- package/src/lib/socket.io/socket.io.esm.min.js +7 -0
- package/src/lib/socket.io/socket.io.esm.min.js.map +1 -0
- package/src/lib/socket.io/socket.io.js +4385 -0
- package/src/lib/socket.io/socket.io.js.map +1 -0
- package/src/lib/socket.io/socket.io.min.js +7 -0
- package/src/lib/socket.io/socket.io.min.js.map +1 -0
- package/src/lib/socket.io/socket.io.msgpack.min.js +7 -0
- package/src/lib/socket.io/socket.io.msgpack.min.js.map +1 -0
- package/src/lib/utils.js +620 -0
- package/src/lib/xdrpc.js +104 -0
- package/src/modules/AI.js +680 -0
- package/src/modules/Apps.js +215 -0
- package/src/modules/Auth.js +171 -0
- package/src/modules/Debug.js +39 -0
- package/src/modules/Drivers.js +278 -0
- package/src/modules/FSItem.js +139 -0
- package/src/modules/FileSystem/index.js +187 -0
- package/src/modules/FileSystem/operations/copy.js +64 -0
- package/src/modules/FileSystem/operations/deleteFSEntry.js +59 -0
- package/src/modules/FileSystem/operations/getReadUrl.js +42 -0
- package/src/modules/FileSystem/operations/mkdir.js +62 -0
- package/src/modules/FileSystem/operations/move.js +75 -0
- package/src/modules/FileSystem/operations/read.js +46 -0
- package/src/modules/FileSystem/operations/readdir.js +102 -0
- package/src/modules/FileSystem/operations/rename.js +58 -0
- package/src/modules/FileSystem/operations/sign.js +103 -0
- package/src/modules/FileSystem/operations/space.js +40 -0
- package/src/modules/FileSystem/operations/stat.js +95 -0
- package/src/modules/FileSystem/operations/symlink.js +55 -0
- package/src/modules/FileSystem/operations/upload.js +440 -0
- package/src/modules/FileSystem/operations/write.js +65 -0
- package/src/modules/FileSystem/utils/getAbsolutePathForApp.js +21 -0
- package/src/modules/Hosting.js +138 -0
- package/src/modules/KV.js +301 -0
- package/src/modules/OS.js +95 -0
- package/src/modules/Perms.js +109 -0
- package/src/modules/PuterDialog.js +481 -0
- package/src/modules/Threads.js +75 -0
- package/src/modules/UI.js +1555 -0
- package/src/modules/Util.js +38 -0
- package/src/modules/Workers.js +120 -0
- package/src/modules/networking/PSocket.js +87 -0
- package/src/modules/networking/PTLS.js +100 -0
- package/src/modules/networking/PWispHandler.js +89 -0
- package/src/modules/networking/parsers.js +157 -0
- package/src/modules/networking/requests.js +282 -0
- package/src/safeLoadPuter.cjs +29 -0
- package/src/services/APIAccess.js +46 -0
- package/src/services/FSRelay.js +20 -0
- package/src/services/Filesystem.js +122 -0
- package/src/services/NoPuterYet.js +20 -0
- package/src/services/XDIncoming.js +44 -0
- package/index.d.ts +0 -479
|
@@ -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">✕</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">✕</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
|
+
}
|