@secrecy/lib 1.0.0-dev.54 → 1.0.0-dev.57

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.
@@ -0,0 +1,1637 @@
1
+ import _classPrivateFieldLooseBase from "@babel/runtime/helpers/classPrivateFieldLooseBase";
2
+ import _classPrivateFieldLooseKey from "@babel/runtime/helpers/classPrivateFieldLooseKey";
3
+
4
+ /* eslint-disable @typescript-eslint/naming-convention */
5
+ import axios from "axios";
6
+ import ky from "ky";
7
+ import { encryptName } from "../index.js";
8
+ import { nodesCache, filesCache } from "../cache.js";
9
+ import { secretstreamKeygen } from "../crypto/file.js";
10
+ import { decryptCryptoBox } from "../crypto/index.js";
11
+ import { encryptCryptoBox } from "../crypto/index.js";
12
+ import { compress, uncompress } from "../minify/index.js";
13
+ import { sodium } from "../sodium.js";
14
+ import { enumerate, chunks, promiseAllLimit, concatenate } from "../utils/utils.js";
15
+ import { md5 } from "../worker/md5.js";
16
+ import { decrypt, encrypt } from "../worker/sodium.js";
17
+ import { gqlFileToExternal } from "./convert/file.js";
18
+ import { gqlNodeFullToInternalFull, gqlNodeToExternal, gqlNodeToExternalNodeFull, internalNodeToNode } from "./convert/node.js";
19
+ import { fileSelector, nodeFullSelector, nodeSelector } from "./types/selectors.js"; // import { md5 } from "../worker/index.js";
20
+ // import { firstValueFrom, of } from "rxjs";
21
+
22
+ var _client = /*#__PURE__*/_classPrivateFieldLooseKey("client");
23
+
24
+ var _keys = /*#__PURE__*/_classPrivateFieldLooseKey("keys");
25
+
26
+ var _thunder = /*#__PURE__*/_classPrivateFieldLooseKey("thunder");
27
+
28
+ export class SecrecyCloudClient {
29
+ constructor(client, keys, thunder) {
30
+ Object.defineProperty(this, _client, {
31
+ writable: true,
32
+ value: void 0
33
+ });
34
+ Object.defineProperty(this, _keys, {
35
+ writable: true,
36
+ value: void 0
37
+ });
38
+ Object.defineProperty(this, _thunder, {
39
+ writable: true,
40
+ value: void 0
41
+ });
42
+
43
+ this.perNode = async (nodeId, publicKey) => {
44
+ var _node$access;
45
+
46
+ let node = nodesCache.get(nodeId);
47
+
48
+ if (!node) {
49
+ await this.node({
50
+ id: nodeId
51
+ });
52
+ node = nodesCache.get(nodeId);
53
+
54
+ if (!node) {
55
+ return null;
56
+ }
57
+ }
58
+
59
+ const nameKey = (_node$access = node.access) == null ? void 0 : _node$access.nameKey;
60
+
61
+ if (!nameKey) {
62
+ return null;
63
+ }
64
+
65
+ return {
66
+ id: node.id,
67
+ nameKey: sodium.to_hex(encryptCryptoBox(sodium.from_hex(nameKey), publicKey, _classPrivateFieldLooseBase(this, _keys)[_keys].privateKey)),
68
+ files: "history" in node ? node.history.map(f => ({
69
+ id: f.id,
70
+ key: sodium.to_hex(encryptCryptoBox(sodium.from_hex(f.key), publicKey, _classPrivateFieldLooseBase(this, _keys)[_keys].privateKey))
71
+ })) : []
72
+ };
73
+ };
74
+
75
+ _classPrivateFieldLooseBase(this, _client)[_client] = client;
76
+ _classPrivateFieldLooseBase(this, _keys)[_keys] = keys;
77
+ _classPrivateFieldLooseBase(this, _thunder)[_thunder] = thunder;
78
+ }
79
+
80
+ async addFileToHistory(_ref) {
81
+ let {
82
+ fileId,
83
+ nodeId
84
+ } = _ref;
85
+ const {
86
+ addFileToHistory
87
+ } = await _classPrivateFieldLooseBase(this, _thunder)[_thunder]("mutation")({
88
+ addFileToHistory: [{
89
+ fileId,
90
+ nodeId
91
+ }, {
92
+ "...on ErrorAccessDenied": {
93
+ __typename: true,
94
+ message: true
95
+ },
96
+ "...on ErrorNotExist": {
97
+ __typename: true,
98
+ message: true
99
+ },
100
+ "...on AddFileToHistoryResponse": {
101
+ __typename: true,
102
+ addFileToHistory: nodeFullSelector
103
+ }
104
+ }]
105
+ });
106
+
107
+ if (!addFileToHistory) {
108
+ return null;
109
+ }
110
+
111
+ if (addFileToHistory.__typename === "ErrorAccessDenied") {
112
+ return addFileToHistory;
113
+ }
114
+
115
+ if (addFileToHistory.__typename === "ErrorNotExist") {
116
+ return addFileToHistory;
117
+ }
118
+
119
+ const node = await gqlNodeFullToInternalFull(addFileToHistory.addFileToHistory, _classPrivateFieldLooseBase(this, _keys)[_keys]);
120
+ const file = node.history.find(f => f.id === fileId);
121
+
122
+ if (file) {
123
+ const users = node.users.filter(_ref2 => {
124
+ let [u] = _ref2;
125
+ return u.publicKey !== _classPrivateFieldLooseBase(this, _keys)[_keys].publicKey;
126
+ });
127
+ const input = {
128
+ fileId: file.id,
129
+ users: users.map(_ref3 => {
130
+ let [u] = _ref3;
131
+ return {
132
+ id: u.id,
133
+ key: sodium.to_hex(encryptCryptoBox(sodium.from_hex(file.key), _classPrivateFieldLooseBase(this, _keys)[_keys].publicKey, _classPrivateFieldLooseBase(this, _keys)[_keys].privateKey))
134
+ };
135
+ })
136
+ };
137
+ await _classPrivateFieldLooseBase(this, _thunder)[_thunder]("mutation")({
138
+ shareFileInHistory: [{
139
+ input,
140
+ nodeId
141
+ }, {
142
+ "...on ErrorAccessDenied": {
143
+ __typename: true,
144
+ message: true
145
+ },
146
+ "...on ErrorNotFound": {
147
+ __typename: true,
148
+ message: true
149
+ },
150
+ "...on ShareFileInHistoryResponse": {
151
+ __typename: true,
152
+ shareFileInHistory: true
153
+ }
154
+ }]
155
+ });
156
+ }
157
+
158
+ const result = internalNodeToNode(node);
159
+ return {
160
+ __typename: "SuccessResponse",
161
+ data: result
162
+ };
163
+ }
164
+
165
+ async uploadFile(_ref4) {
166
+ let {
167
+ file,
168
+ encryptProgress,
169
+ uploadProgress,
170
+ signal
171
+ } = _ref4;
172
+ const fileKey = secretstreamKeygen();
173
+ const fileBuffer = file instanceof File ? new Uint8Array(await file.arrayBuffer()) : file;
174
+ const compressed = compress(fileBuffer);
175
+ const {
176
+ data: encryptedFile,
177
+ md5: md5File,
178
+ md5Encrypted
179
+ } = await encrypt(fileKey, compressed, encryptProgress, signal);
180
+ const encryptedFileKey = encryptCryptoBox(fileKey, _classPrivateFieldLooseBase(this, _keys)[_keys].publicKey, _classPrivateFieldLooseBase(this, _keys)[_keys].privateKey);
181
+ const {
182
+ uploadFile
183
+ } = await _classPrivateFieldLooseBase(this, _thunder)[_thunder]("mutation", {
184
+ scalars: {
185
+ Json: {
186
+ encode: e => JSON.stringify(e),
187
+ decode: e => JSON.parse(e)
188
+ },
189
+ DateTime: {
190
+ decode: e => new Date(e),
191
+ encode: e => e.toISOString()
192
+ },
193
+ BigInt: {
194
+ decode: e => BigInt(e),
195
+ encode: e => e.toString()
196
+ }
197
+ }
198
+ })({
199
+ uploadFile: [{
200
+ fileSize: encryptedFile.byteLength,
201
+ fileSizeBefore: fileBuffer.byteLength,
202
+ fileKey: sodium.to_hex(encryptedFileKey),
203
+ md5Encrypted,
204
+ md5: md5File
205
+ }, {
206
+ "...on ErrorAccessDenied": {
207
+ __typename: true,
208
+ message: true
209
+ },
210
+ "...on ErrorLimit": {
211
+ __typename: true,
212
+ message: true
213
+ },
214
+ "...on ErrorNotFound": {
215
+ __typename: true,
216
+ message: true
217
+ },
218
+ "...on UploadFileResponse": {
219
+ __typename: true,
220
+ uploadFile: {
221
+ fileId: true,
222
+ filePartSize: true,
223
+ parts: {
224
+ fields: true,
225
+ order: true,
226
+ url: true
227
+ }
228
+ }
229
+ }
230
+ }]
231
+ });
232
+
233
+ if (!uploadFile) {
234
+ return null;
235
+ }
236
+
237
+ if (uploadFile.__typename === "ErrorAccessDenied") {
238
+ return uploadFile;
239
+ }
240
+
241
+ if (uploadFile.__typename === "ErrorLimit") {
242
+ return uploadFile;
243
+ }
244
+
245
+ if (uploadFile.__typename === "ErrorNotFound") {
246
+ return uploadFile;
247
+ }
248
+
249
+ if (!uploadFile.uploadFile) {
250
+ return null;
251
+ }
252
+
253
+ uploadProgress == null ? void 0 : uploadProgress({
254
+ total: encryptedFile.byteLength,
255
+ current: 0,
256
+ percent: 0
257
+ });
258
+
259
+ if (uploadFile.uploadFile.parts.length === 0) {
260
+ uploadProgress == null ? void 0 : uploadProgress({
261
+ total: encryptedFile.byteLength,
262
+ current: encryptedFile.byteLength,
263
+ percent: 1
264
+ });
265
+ return {
266
+ __typename: "SuccessResponse",
267
+ data: uploadFile.uploadFile.fileId
268
+ };
269
+ }
270
+
271
+ const uploadPartEnded = async (md5, order) => {
272
+ if (!uploadFile.uploadFile) {
273
+ return null;
274
+ }
275
+
276
+ const {
277
+ uploadFilePartEnd
278
+ } = await _classPrivateFieldLooseBase(this, _thunder)[_thunder]("mutation")({
279
+ uploadFilePartEnd: [{
280
+ fileId: uploadFile.uploadFile.fileId,
281
+ md5,
282
+ order
283
+ }, {
284
+ "...on ErrorAccessDenied": {
285
+ __typename: true,
286
+ message: true
287
+ },
288
+ "...on UploadFilePartEndResponse": {
289
+ __typename: true,
290
+ uploadFilePartEnd: true
291
+ }
292
+ }]
293
+ });
294
+
295
+ if (!uploadFilePartEnd) {
296
+ return null;
297
+ }
298
+
299
+ if (uploadFilePartEnd.__typename === "ErrorAccessDenied") {
300
+ return uploadFilePartEnd;
301
+ }
302
+
303
+ return {
304
+ __typename: "SuccessResponse",
305
+ data: uploadFilePartEnd.uploadFilePartEnd
306
+ };
307
+ };
308
+
309
+ const uploadEnded = async () => {
310
+ if (!uploadFile.uploadFile) {
311
+ return null;
312
+ }
313
+
314
+ const {
315
+ uploadFileEnd
316
+ } = await _classPrivateFieldLooseBase(this, _thunder)[_thunder]("mutation")({
317
+ uploadFileEnd: [{
318
+ fileId: uploadFile.uploadFile.fileId
319
+ }, {
320
+ "...on ErrorAccessDenied": {
321
+ __typename: true,
322
+ message: true
323
+ },
324
+ "...on ErrorNotFound": {
325
+ __typename: true,
326
+ message: true
327
+ },
328
+ "...on UploadFileEndResponse": {
329
+ __typename: true,
330
+ uploadFileEnd: true
331
+ }
332
+ }]
333
+ });
334
+
335
+ if (!uploadFileEnd) {
336
+ return null;
337
+ }
338
+
339
+ if (uploadFileEnd.__typename === "ErrorAccessDenied") {
340
+ return uploadFileEnd;
341
+ }
342
+
343
+ if (uploadFileEnd.__typename === "ErrorNotFound") {
344
+ return uploadFileEnd;
345
+ }
346
+
347
+ if (!uploadFileEnd.uploadFileEnd) {
348
+ return null;
349
+ }
350
+
351
+ return {
352
+ __typename: "SuccessResponse",
353
+ data: uploadFileEnd.uploadFileEnd
354
+ };
355
+ };
356
+
357
+ const chunkParts = new Array();
358
+
359
+ for (const [index, chunk] of enumerate(chunks(encryptedFile, Number(uploadFile.uploadFile.filePartSize)))) {
360
+ chunkParts.push({
361
+ order: index + 1,
362
+ data: chunk,
363
+ md5: await md5(chunk)
364
+ });
365
+ }
366
+
367
+ const progressParts = {};
368
+
369
+ const onProgress = (part, progressEvent) => {
370
+ progressParts[part] = progressEvent;
371
+ const current = Object.values(progressParts).reduce((prv, cur) => prv + cur.loaded, 0);
372
+ uploadProgress == null ? void 0 : uploadProgress({
373
+ percent: current / encryptedFile.byteLength,
374
+ total: encryptedFile.byteLength,
375
+ current
376
+ });
377
+ };
378
+
379
+ const byPart = async part => {
380
+ if (!uploadFile.uploadFile) {
381
+ return;
382
+ }
383
+
384
+ const formData = new FormData();
385
+ const chunk = chunkParts.find(p => p.order === part.order);
386
+
387
+ if (!chunk) {
388
+ return;
389
+ }
390
+
391
+ for (const [key, value] of Object.entries(part.fields)) {
392
+ formData.append(key, value);
393
+ }
394
+
395
+ formData.append("file", new Blob([chunk.data]), uploadFile.uploadFile.fileId + "-" + chunk.order);
396
+ await axios.post(part.url, formData, {
397
+ onUploadProgress: progressEvent => onProgress(part.order, progressEvent),
398
+ signal
399
+ });
400
+ await uploadPartEnded(chunk.md5, chunk.order); // if ((e as any).response.status === 0) {
401
+ // // TODO https://github.com/sindresorhus/ky/issues/305
402
+ // } else {
403
+ // throw e;
404
+ // }
405
+ };
406
+
407
+ if (!uploadFile.uploadFile) {
408
+ return null;
409
+ }
410
+
411
+ await promiseAllLimit(3, uploadFile.uploadFile.parts.map(p => () => byPart(p)));
412
+ const result = await uploadEnded();
413
+
414
+ if (!result) {
415
+ return null;
416
+ }
417
+
418
+ if (result.__typename === "ErrorAccessDenied") {
419
+ return result;
420
+ }
421
+
422
+ if (result.__typename === "ErrorNotFound") {
423
+ return result;
424
+ }
425
+
426
+ return {
427
+ __typename: "SuccessResponse",
428
+ data: result.data
429
+ };
430
+ }
431
+
432
+ async uploadFileInCloud(_ref5) {
433
+ let {
434
+ file,
435
+ name,
436
+ nodeId,
437
+ encryptProgress,
438
+ uploadProgress,
439
+ signal
440
+ } = _ref5;
441
+ const fileId = await this.uploadFile({
442
+ file,
443
+ encryptProgress,
444
+ uploadProgress,
445
+ signal
446
+ });
447
+
448
+ if (!fileId) {
449
+ return null;
450
+ }
451
+
452
+ if (fileId.__typename === "ErrorAccessDenied") {
453
+ return fileId;
454
+ }
455
+
456
+ if (fileId.__typename === "ErrorLimit") {
457
+ return fileId;
458
+ }
459
+
460
+ if (fileId.__typename === "ErrorNotFound") {
461
+ return fileId;
462
+ }
463
+
464
+ const result = await this.saveInCloud({
465
+ fileId: fileId.data,
466
+ name,
467
+ nodeId
468
+ });
469
+
470
+ if (!result) {
471
+ return null;
472
+ }
473
+
474
+ if (result.__typename === "ErrorAccessDenied") {
475
+ return result;
476
+ }
477
+
478
+ if (result.__typename === "ErrorBasic") {
479
+ return result;
480
+ }
481
+
482
+ if (result.__typename === "ErrorLimit") {
483
+ return result;
484
+ }
485
+
486
+ if (result.__typename === "ErrorNotFound") {
487
+ return result;
488
+ }
489
+
490
+ if (result.__typename === "ErrorNotExist") {
491
+ return result;
492
+ }
493
+
494
+ return {
495
+ __typename: "SuccessResponse",
496
+ data: result.data
497
+ };
498
+ }
499
+
500
+ async deletedNodes() {
501
+ const {
502
+ deletedNodes
503
+ } = await _classPrivateFieldLooseBase(this, _thunder)[_thunder]("query")({
504
+ deletedNodes: {
505
+ "...on ErrorAccessDenied": {
506
+ __typename: true,
507
+ message: true
508
+ },
509
+ "...on DeletedNodesResponse": {
510
+ __typename: true,
511
+ deletedNodes: nodeSelector
512
+ }
513
+ }
514
+ });
515
+
516
+ if (!deletedNodes) {
517
+ return null;
518
+ }
519
+
520
+ if (deletedNodes.__typename === "ErrorAccessDenied") {
521
+ return deletedNodes;
522
+ }
523
+
524
+ const nodes = new Array();
525
+
526
+ for (const node of deletedNodes.deletedNodes) {
527
+ nodes.push(await gqlNodeToExternal(node, _classPrivateFieldLooseBase(this, _keys)[_keys]));
528
+ }
529
+
530
+ return {
531
+ __typename: "SuccessResponse",
532
+ data: nodes
533
+ };
534
+ }
535
+
536
+ async sharedNodes() {
537
+ const {
538
+ sharedNodes
539
+ } = await _classPrivateFieldLooseBase(this, _thunder)[_thunder]("query")({
540
+ sharedNodes: {
541
+ "...on ErrorAccessDenied": {
542
+ __typename: true,
543
+ message: true
544
+ },
545
+ "...on SharedNodesResponse": {
546
+ __typename: true,
547
+ sharedNodes: nodeSelector
548
+ }
549
+ }
550
+ });
551
+
552
+ if (!sharedNodes) {
553
+ return null;
554
+ }
555
+
556
+ if (sharedNodes.__typename === "ErrorAccessDenied") {
557
+ return sharedNodes;
558
+ }
559
+
560
+ const nodes = new Array();
561
+
562
+ for (const folder of sharedNodes.sharedNodes) {
563
+ nodes.push(await gqlNodeToExternal(folder, _classPrivateFieldLooseBase(this, _keys)[_keys]));
564
+ }
565
+
566
+ return {
567
+ __typename: "SuccessResponse",
568
+ data: nodes
569
+ };
570
+ }
571
+
572
+ async nodesSharedWithMe(type) {
573
+ const {
574
+ nodesSharedWithMe
575
+ } = await _classPrivateFieldLooseBase(this, _thunder)[_thunder]("query")({
576
+ nodesSharedWithMe: [{
577
+ type
578
+ }, {
579
+ "...on ErrorAccessDenied": {
580
+ __typename: true,
581
+ message: true
582
+ },
583
+ "...on NodesSharedWithMeResponse": {
584
+ __typename: true,
585
+ nodesSharedWithMe: nodeSelector
586
+ }
587
+ }]
588
+ });
589
+
590
+ if (!nodesSharedWithMe) {
591
+ return null;
592
+ }
593
+
594
+ if (nodesSharedWithMe.__typename === "ErrorAccessDenied") {
595
+ return nodesSharedWithMe;
596
+ }
597
+
598
+ const nodes = new Array();
599
+
600
+ for (const folder of nodesSharedWithMe.nodesSharedWithMe) {
601
+ nodes.push(await gqlNodeToExternal(folder, _classPrivateFieldLooseBase(this, _keys)[_keys]));
602
+ }
603
+
604
+ return {
605
+ __typename: "SuccessResponse",
606
+ data: nodes
607
+ };
608
+ }
609
+
610
+ async deleteNodeSharing(_ref6) {
611
+ let {
612
+ nodeId,
613
+ userId
614
+ } = _ref6;
615
+ const {
616
+ deleteNodeSharing
617
+ } = await _classPrivateFieldLooseBase(this, _thunder)[_thunder]("mutation")({
618
+ deleteNodeSharing: [{
619
+ nodeId,
620
+ userId
621
+ }, {
622
+ "...on ErrorAccessDenied": {
623
+ __typename: true,
624
+ message: true
625
+ },
626
+ "...on DeleteNodeSharingResponse": {
627
+ __typename: true,
628
+ deleteNodeSharing: true
629
+ }
630
+ }]
631
+ });
632
+
633
+ if (!deleteNodeSharing) {
634
+ return null;
635
+ }
636
+
637
+ if (deleteNodeSharing.__typename === "ErrorAccessDenied") {
638
+ return deleteNodeSharing;
639
+ }
640
+
641
+ return {
642
+ __typename: "SuccessResponse",
643
+ data: deleteNodeSharing.deleteNodeSharing
644
+ };
645
+ }
646
+
647
+ async duplicateNode(_ref7) {
648
+ var _node$access2;
649
+
650
+ let {
651
+ nodeId,
652
+ folderId,
653
+ customName
654
+ } = _ref7;
655
+ let node = nodesCache.get(nodeId);
656
+
657
+ if (!node) {
658
+ await this.node({
659
+ id: nodeId
660
+ });
661
+ node = nodesCache.get(nodeId);
662
+
663
+ if (!node) {
664
+ throw new Error("Node (" + nodeId + ") does not exists");
665
+ }
666
+ }
667
+
668
+ if (!((_node$access2 = node.access) != null && _node$access2.nameKey)) {
669
+ throw new Error("Can't have access to node " + nodeId);
670
+ }
671
+
672
+ customName = customName ? await encryptName(customName, node.access.nameKey) : null;
673
+ const {
674
+ duplicateNode
675
+ } = await _classPrivateFieldLooseBase(this, _thunder)[_thunder]("mutation")({
676
+ duplicateNode: [{
677
+ nodeId,
678
+ folderId,
679
+ customName
680
+ }, {
681
+ "...on ErrorAccessDenied": {
682
+ __typename: true,
683
+ message: true
684
+ },
685
+ "...on ErrorNotFound": {
686
+ __typename: true,
687
+ message: true
688
+ },
689
+ "...on DuplicateNodeResponse": {
690
+ __typename: true,
691
+ duplicateNode: true
692
+ }
693
+ }]
694
+ });
695
+
696
+ if (!duplicateNode) {
697
+ return null;
698
+ }
699
+
700
+ if (duplicateNode.__typename === "ErrorAccessDenied") {
701
+ return duplicateNode;
702
+ }
703
+
704
+ if (duplicateNode.__typename === "ErrorNotFound") {
705
+ return duplicateNode;
706
+ }
707
+
708
+ return {
709
+ __typename: "SuccessResponse",
710
+ data: duplicateNode.duplicateNode
711
+ };
712
+ }
713
+
714
+ async deleteNodeCloudTrash(_ref8) {
715
+ var _deleteNodeCloudTrash;
716
+
717
+ let {
718
+ ids
719
+ } = _ref8;
720
+ const {
721
+ deleteNodeCloudTrash
722
+ } = await _classPrivateFieldLooseBase(this, _thunder)[_thunder]("mutation")({
723
+ deleteNodeCloudTrash: [{
724
+ ids
725
+ }, {
726
+ "...on ErrorAccessDenied": {
727
+ __typename: true,
728
+ message: true
729
+ },
730
+ "...on DeleteNodeCloudTrashResponse": {
731
+ __typename: true,
732
+ deleteNodeCloudTrash: true
733
+ }
734
+ }]
735
+ });
736
+
737
+ if (!deleteNodeCloudTrash) {
738
+ return null;
739
+ }
740
+
741
+ if (deleteNodeCloudTrash.__typename === "ErrorAccessDenied") {
742
+ return deleteNodeCloudTrash;
743
+ }
744
+
745
+ return {
746
+ __typename: "SuccessResponse",
747
+ data: (_deleteNodeCloudTrash = deleteNodeCloudTrash.deleteNodeCloudTrash) != null ? _deleteNodeCloudTrash : false
748
+ };
749
+ }
750
+
751
+ async createFolder(_ref9) {
752
+ var _folder$parent$users$, _folder$parent, _folder$parent$users;
753
+
754
+ let {
755
+ name,
756
+ parentFolderId
757
+ } = _ref9;
758
+ const key = secretstreamKeygen();
759
+ const encryptedName = await encryptName(name, sodium.to_hex(key));
760
+ const encryptedKey = encryptCryptoBox(key, _classPrivateFieldLooseBase(this, _keys)[_keys].publicKey, _classPrivateFieldLooseBase(this, _keys)[_keys].privateKey);
761
+ const {
762
+ createFolder
763
+ } = await _classPrivateFieldLooseBase(this, _thunder)[_thunder]("mutation")({
764
+ createFolder: [{
765
+ name: encryptedName,
766
+ parentFolderId,
767
+ key: sodium.to_hex(encryptedKey)
768
+ }, {
769
+ "...on ErrorAccessDenied": {
770
+ __typename: true,
771
+ message: true
772
+ },
773
+ "...on ErrorNotExist": {
774
+ __typename: true,
775
+ message: true
776
+ },
777
+ "...on CreateFolderResponse": {
778
+ __typename: true,
779
+ createFolder: nodeFullSelector
780
+ }
781
+ }]
782
+ });
783
+
784
+ if (!createFolder) {
785
+ //throw new Error(`Can't create folder`);
786
+ return null;
787
+ }
788
+
789
+ if (createFolder.__typename === "ErrorAccessDenied") {
790
+ return createFolder;
791
+ }
792
+
793
+ if (createFolder.__typename === "ErrorNotExist") {
794
+ return createFolder;
795
+ }
796
+
797
+ if (!createFolder.createFolder) {
798
+ return null;
799
+ }
800
+
801
+ const folder = await gqlNodeToExternalNodeFull(createFolder.createFolder, _classPrivateFieldLooseBase(this, _keys)[_keys]);
802
+ const users = (_folder$parent$users$ = (_folder$parent = folder.parent) == null ? void 0 : (_folder$parent$users = _folder$parent.users) == null ? void 0 : _folder$parent$users.filter(_ref10 => {
803
+ let [u] = _ref10;
804
+ return u.publicKey !== _classPrivateFieldLooseBase(this, _keys)[_keys].publicKey;
805
+ })) != null ? _folder$parent$users$ : [];
806
+
807
+ if (users.length) {
808
+ await Promise.all(users.map(_ref11 => {
809
+ let [u, rights] = _ref11;
810
+ return this.shareNode({
811
+ nodeId: folder.id,
812
+ rights,
813
+ userId: u.id
814
+ });
815
+ }));
816
+ }
817
+
818
+ return {
819
+ __typename: "SuccessResponse",
820
+ data: folder
821
+ };
822
+ }
823
+
824
+ async node(_temp) {
825
+ let {
826
+ id,
827
+ deleted
828
+ } = _temp === void 0 ? {} : _temp;
829
+ const {
830
+ node
831
+ } = await _classPrivateFieldLooseBase(this, _thunder)[_thunder]("query")({
832
+ node: [{
833
+ deleted,
834
+ id
835
+ }, {
836
+ "...on ErrorAccessDenied": {
837
+ __typename: true,
838
+ message: true
839
+ },
840
+ "...on NodeResponse": {
841
+ __typename: true,
842
+ node: nodeFullSelector
843
+ }
844
+ }]
845
+ });
846
+
847
+ if (!node) {
848
+ return null;
849
+ }
850
+
851
+ if (node.__typename === "ErrorAccessDenied") {
852
+ return node;
853
+ }
854
+
855
+ if (!node.node) {
856
+ return null;
857
+ }
858
+
859
+ const result = await gqlNodeToExternalNodeFull(node.node, _classPrivateFieldLooseBase(this, _keys)[_keys]);
860
+ return {
861
+ __typename: "SuccessResponse",
862
+ data: result
863
+ };
864
+ }
865
+
866
+ async file(_ref12) {
867
+ let {
868
+ id
869
+ } = _ref12;
870
+ const {
871
+ file
872
+ } = await _classPrivateFieldLooseBase(this, _thunder)[_thunder]("query")({
873
+ file: [{
874
+ id
875
+ }, {
876
+ "...on ErrorAccessDenied": {
877
+ __typename: true,
878
+ message: true
879
+ },
880
+ "...on FileQueryResponse": {
881
+ __typename: true,
882
+ file: fileSelector
883
+ }
884
+ }]
885
+ });
886
+
887
+ if (!file) {
888
+ return null;
889
+ }
890
+
891
+ if (file.__typename === "ErrorAccessDenied") {
892
+ return file;
893
+ }
894
+
895
+ if (!file.file) {
896
+ return null;
897
+ }
898
+
899
+ const result = gqlFileToExternal(file.file, _classPrivateFieldLooseBase(this, _keys)[_keys]);
900
+ return {
901
+ __typename: "SuccessResponse",
902
+ data: result
903
+ };
904
+ }
905
+
906
+ async shareNode(_ref13) {
907
+ var _shareNodeFinish$shar;
908
+
909
+ let {
910
+ nodeId,
911
+ userId,
912
+ rights
913
+ } = _ref13;
914
+ const user = await _classPrivateFieldLooseBase(this, _client)[_client].user({
915
+ userId,
916
+ withPublicKey: true
917
+ });
918
+
919
+ if (!user) {
920
+ return user;
921
+ }
922
+
923
+ if (user.__typename === "ErrorNotFound") {
924
+ return user;
925
+ }
926
+
927
+ const {
928
+ shareNode
929
+ } = await _classPrivateFieldLooseBase(this, _thunder)[_thunder]("mutation")({
930
+ shareNode: [{
931
+ nodeId,
932
+ userId
933
+ }, {
934
+ "...on ErrorAccessDenied": {
935
+ __typename: true,
936
+ message: true
937
+ },
938
+ "...on ErrorNotFound": {
939
+ __typename: true,
940
+ message: true
941
+ },
942
+ "...on ShareNodeResponse": {
943
+ __typename: true,
944
+ nodes: true
945
+ }
946
+ }]
947
+ });
948
+
949
+ if (!shareNode) {
950
+ return null;
951
+ }
952
+
953
+ if (shareNode.__typename === "ErrorAccessDenied") {
954
+ return shareNode;
955
+ }
956
+
957
+ if (shareNode.__typename === "ErrorNotFound") {
958
+ return shareNode;
959
+ }
960
+
961
+ const shareNodes = {
962
+ nodes: []
963
+ };
964
+
965
+ for (const id of shareNode.nodes) {
966
+ const nameKey = await this.perNode(id, user.data.publicKey);
967
+
968
+ if (nameKey) {
969
+ shareNodes.nodes.push(nameKey);
970
+ }
971
+ }
972
+
973
+ const {
974
+ shareNodeFinish
975
+ } = await _classPrivateFieldLooseBase(this, _thunder)[_thunder]("mutation")({
976
+ shareNodeFinish: [{
977
+ rights,
978
+ userId,
979
+ shareNodes
980
+ }, {
981
+ "...on ErrorAccessDenied": {
982
+ __typename: true,
983
+ message: true
984
+ },
985
+ "...on ErrorNotFound": {
986
+ __typename: true,
987
+ message: true
988
+ },
989
+ "...on ShareNodeFinishResponse": {
990
+ __typename: true,
991
+ shareNodeFinish: true
992
+ }
993
+ }]
994
+ });
995
+
996
+ if (!shareNodeFinish) {
997
+ return null;
998
+ }
999
+
1000
+ if (shareNodeFinish.__typename === "ErrorAccessDenied") {
1001
+ return shareNodeFinish;
1002
+ }
1003
+
1004
+ if (shareNodeFinish.__typename === "ErrorNotFound") {
1005
+ return shareNodeFinish;
1006
+ }
1007
+
1008
+ return {
1009
+ __typename: "SuccessResponse",
1010
+ data: (_shareNodeFinish$shar = shareNodeFinish.shareNodeFinish) != null ? _shareNodeFinish$shar : false
1011
+ };
1012
+ }
1013
+
1014
+ async updateNode(_ref14) {
1015
+ var _node$access3, _node$access4;
1016
+
1017
+ let {
1018
+ nodeId,
1019
+ name,
1020
+ isFavorite,
1021
+ deletedAt
1022
+ } = _ref14;
1023
+ let node = nodesCache.get(nodeId);
1024
+
1025
+ if (!node) {
1026
+ await this.node({
1027
+ id: nodeId
1028
+ });
1029
+ node = nodesCache.get(nodeId);
1030
+
1031
+ if (!node) {
1032
+ throw "Can't find Node " + nodeId;
1033
+ }
1034
+ }
1035
+
1036
+ if (!((_node$access3 = node.access) != null && _node$access3.nameKey)) {
1037
+ throw new Error("Can't have access to node " + nodeId);
1038
+ }
1039
+
1040
+ name = name ? (_node$access4 = node.access) != null && _node$access4.nameKey ? await encryptName(name, node.access.nameKey) : name : null;
1041
+ const {
1042
+ updateNode
1043
+ } = await _classPrivateFieldLooseBase(this, _thunder)[_thunder]("mutation")({
1044
+ updateNode: [{
1045
+ nodeId,
1046
+ name,
1047
+ isFavorite,
1048
+ deletedAt
1049
+ }, {
1050
+ "...on ErrorAccessDenied": {
1051
+ __typename: true,
1052
+ message: true
1053
+ },
1054
+ "...on ErrorNotExist": {
1055
+ __typename: true,
1056
+ message: true
1057
+ },
1058
+ "...on UpdateNodeResponse": {
1059
+ __typename: true,
1060
+ updateNode: nodeFullSelector
1061
+ }
1062
+ }]
1063
+ });
1064
+
1065
+ if (!updateNode) {
1066
+ return null;
1067
+ }
1068
+
1069
+ if (updateNode.__typename === "ErrorAccessDenied") {
1070
+ return updateNode;
1071
+ }
1072
+
1073
+ if (updateNode.__typename === "ErrorNotExist") {
1074
+ return updateNode;
1075
+ }
1076
+
1077
+ if (!updateNode.updateNode) {
1078
+ return null;
1079
+ }
1080
+
1081
+ const result = await gqlNodeToExternalNodeFull(updateNode.updateNode, _classPrivateFieldLooseBase(this, _keys)[_keys]);
1082
+ return {
1083
+ __typename: "SuccessResponse",
1084
+ data: result
1085
+ };
1086
+ }
1087
+
1088
+ async fileContent(_ref15) {
1089
+ let {
1090
+ fileId,
1091
+ onDownloadProgress,
1092
+ progressDecrypt,
1093
+ signal
1094
+ } = _ref15;
1095
+ const {
1096
+ fileContent
1097
+ } = await _classPrivateFieldLooseBase(this, _thunder)[_thunder]("query", {
1098
+ scalars: {
1099
+ Bytes: {
1100
+ decode: e => {
1101
+ console.log(e, typeof e);
1102
+ return Uint8Array.from([1, 2, 3]);
1103
+ },
1104
+ encode: e => e.toString()
1105
+ }
1106
+ }
1107
+ })({
1108
+ fileContent: [{
1109
+ fileId
1110
+ }, {
1111
+ "...on ErrorAccessDenied": {
1112
+ __typename: true,
1113
+ message: true
1114
+ },
1115
+ "...on ErrorBasic": {
1116
+ __typename: true,
1117
+ message: true
1118
+ },
1119
+ "...on FileContentResponse": {
1120
+ __typename: true,
1121
+ file: {
1122
+ "...on FileContentCloud": {
1123
+ __typename: true,
1124
+ parts: {
1125
+ contentUrl: true,
1126
+ order: true,
1127
+ md5: true
1128
+ },
1129
+ key: true,
1130
+ publicKey: true,
1131
+ totalSize: true,
1132
+ md5: true,
1133
+ md5Encrypted: true
1134
+ },
1135
+ "...on FileContentLite": {
1136
+ __typename: true,
1137
+ content: true,
1138
+ id: true,
1139
+ key: true,
1140
+ publicKey: true,
1141
+ md5: true,
1142
+ md5Encrypted: true,
1143
+ totalSize: true
1144
+ },
1145
+ "...on FileContentReceivedMail": {
1146
+ __typename: true,
1147
+ maybeParts: {
1148
+ contentUrl: true,
1149
+ order: true,
1150
+ md5: true
1151
+ },
1152
+ maybeContent: true,
1153
+ key: true,
1154
+ senderPublicKey: true,
1155
+ totalSize: true,
1156
+ md5: true,
1157
+ md5Encrypted: true
1158
+ },
1159
+ "...on FileContentSentMail": {
1160
+ __typename: true,
1161
+ maybeParts: {
1162
+ contentUrl: true,
1163
+ order: true,
1164
+ md5: true
1165
+ },
1166
+ maybeContent: true,
1167
+ key: true,
1168
+ totalSize: true,
1169
+ md5: true,
1170
+ md5Encrypted: true
1171
+ }
1172
+ }
1173
+ }
1174
+ }]
1175
+ });
1176
+
1177
+ if (!fileContent) {
1178
+ return null;
1179
+ }
1180
+
1181
+ if (fileContent.__typename === "ErrorAccessDenied") {
1182
+ return fileContent;
1183
+ }
1184
+
1185
+ if (fileContent.__typename === "ErrorBasic") {
1186
+ return fileContent;
1187
+ }
1188
+
1189
+ const file = fileContent.file;
1190
+
1191
+ if (!file) {
1192
+ return null;
1193
+ }
1194
+
1195
+ const progressParts = {};
1196
+
1197
+ const onProgress = (part, progressEvent) => {
1198
+ progressParts[part] = progressEvent;
1199
+ const transferredBytes = Object.values(progressParts).reduce((prv, cur) => prv + cur.transferredBytes, 0);
1200
+ const totalBytes = Number(file.totalSize);
1201
+ onDownloadProgress == null ? void 0 : onDownloadProgress({
1202
+ percent: transferredBytes / totalBytes,
1203
+ totalBytes,
1204
+ transferredBytes
1205
+ });
1206
+ };
1207
+
1208
+ const encryptedContentFromParts = async fileParts => {
1209
+ const parts = new Array();
1210
+
1211
+ const byPart = async part => {
1212
+ const buf = new Uint8Array(await ky.get(part.contentUrl, {
1213
+ timeout: false,
1214
+ onDownloadProgress: pr => onProgress(part.order, pr),
1215
+ signal: signal
1216
+ }).arrayBuffer());
1217
+ const md5Part = await md5(buf);
1218
+
1219
+ if (md5Part !== part.md5) {
1220
+ throw new Error("Invalid md5 for part " + part.order + " of file " + fileId);
1221
+ }
1222
+
1223
+ parts.push({
1224
+ data: buf,
1225
+ order: part.order
1226
+ });
1227
+ };
1228
+
1229
+ await promiseAllLimit(3, fileParts.map(p => () => byPart(p)));
1230
+ return concatenate(...parts.sort((a, b) => a.order - b.order).map(p => p.data));
1231
+ };
1232
+
1233
+ const finalize = async encryptedContent => {
1234
+ // const md5Encrypted = await firstValueFrom(md5(of(encryptedContent)));
1235
+ const md5Encrypted = await md5(encryptedContent);
1236
+
1237
+ if (md5Encrypted !== file.md5Encrypted) {
1238
+ throw new Error("Encrypted content does not match");
1239
+ }
1240
+
1241
+ const key = decryptCryptoBox(sodium.from_hex(file.key), file.__typename === "FileContentReceivedMail" ? file.senderPublicKey : file.__typename === "FileContentCloud" ? file.publicKey : _classPrivateFieldLooseBase(this, _keys)[_keys].publicKey, _classPrivateFieldLooseBase(this, _keys)[_keys].privateKey);
1242
+ const src = await decrypt(key, encryptedContent, progressDecrypt, signal); // const md5Content = await firstValueFrom(md5(of(src)));
1243
+
1244
+ const md5Content = await md5(src);
1245
+
1246
+ if (md5Content !== file.md5) {
1247
+ throw new Error("Content does not match");
1248
+ }
1249
+
1250
+ return uncompress(src);
1251
+ };
1252
+
1253
+ const encryptedContent = file.__typename === "FileContentLite" ? file.content : file.__typename === "FileContentCloud" ? await encryptedContentFromParts(file.parts) : file.maybeContent ? file.maybeContent : file.maybeParts ? await encryptedContentFromParts(file.maybeParts) : null;
1254
+
1255
+ if (!encryptedContent) {
1256
+ return null;
1257
+ }
1258
+
1259
+ return {
1260
+ __typename: "SuccessResponse",
1261
+ data: await finalize(encryptedContent)
1262
+ };
1263
+ }
1264
+
1265
+ async deleteFile(_ref16) {
1266
+ let {
1267
+ fileId,
1268
+ nodeId
1269
+ } = _ref16;
1270
+ const {
1271
+ deleteFile
1272
+ } = await _classPrivateFieldLooseBase(this, _thunder)[_thunder]("mutation")({
1273
+ deleteFile: [{
1274
+ fileId,
1275
+ nodeId
1276
+ }, {
1277
+ "...on ErrorAccessDenied": {
1278
+ __typename: true,
1279
+ message: true
1280
+ },
1281
+ "...on ErrorNotExist": {
1282
+ __typename: true,
1283
+ message: true
1284
+ },
1285
+ "...on DeleteFileResponse": {
1286
+ __typename: true,
1287
+ deleteFile: true
1288
+ }
1289
+ }]
1290
+ });
1291
+
1292
+ if (!deleteFile) {
1293
+ return null;
1294
+ }
1295
+
1296
+ if (deleteFile.__typename === "ErrorAccessDenied") {
1297
+ return deleteFile;
1298
+ }
1299
+
1300
+ if (deleteFile.__typename === "ErrorNotExist") {
1301
+ return deleteFile;
1302
+ }
1303
+
1304
+ return {
1305
+ __typename: "SuccessResponse",
1306
+ data: deleteFile.deleteFile
1307
+ };
1308
+ }
1309
+
1310
+ async deleteNode(_ref17) {
1311
+ let {
1312
+ nodeId
1313
+ } = _ref17;
1314
+ const {
1315
+ deleteNode
1316
+ } = await _classPrivateFieldLooseBase(this, _thunder)[_thunder]("mutation")({
1317
+ deleteNode: [{
1318
+ id: nodeId
1319
+ }, {
1320
+ "...on ErrorAccessDenied": {
1321
+ __typename: true,
1322
+ message: true
1323
+ },
1324
+ "...on DeleteNodeResponse": {
1325
+ __typename: true,
1326
+ deleteNode: true
1327
+ }
1328
+ }]
1329
+ });
1330
+
1331
+ if (!deleteNode) {
1332
+ return null;
1333
+ }
1334
+
1335
+ if (deleteNode.__typename === "ErrorAccessDenied") {
1336
+ return deleteNode;
1337
+ }
1338
+
1339
+ return {
1340
+ __typename: "SuccessResponse",
1341
+ data: deleteNode.deleteNode
1342
+ };
1343
+ }
1344
+
1345
+ async emptyTrash() {
1346
+ const {
1347
+ emptyCloudTrash
1348
+ } = await _classPrivateFieldLooseBase(this, _thunder)[_thunder]("mutation")({
1349
+ emptyCloudTrash: {
1350
+ "...on ErrorAccessDenied": {
1351
+ __typename: true,
1352
+ message: true
1353
+ },
1354
+ "...on EmptyCloudTrashResponse": {
1355
+ __typename: true,
1356
+ emptyCloudTrash: true
1357
+ }
1358
+ }
1359
+ });
1360
+
1361
+ if (!emptyCloudTrash) {
1362
+ return null;
1363
+ }
1364
+
1365
+ if (emptyCloudTrash.__typename === "ErrorAccessDenied") {
1366
+ return emptyCloudTrash;
1367
+ }
1368
+
1369
+ if (!emptyCloudTrash.emptyCloudTrash) {
1370
+ return null;
1371
+ }
1372
+
1373
+ return {
1374
+ __typename: "SuccessResponse",
1375
+ data: emptyCloudTrash.emptyCloudTrash
1376
+ };
1377
+ }
1378
+
1379
+ async recoverNode(id) {
1380
+ var _recoverNode$recoverN;
1381
+
1382
+ const {
1383
+ recoverNode
1384
+ } = await _classPrivateFieldLooseBase(this, _thunder)[_thunder]("mutation")({
1385
+ recoverNode: [{
1386
+ id
1387
+ }, {
1388
+ "...on ErrorAccessDenied": {
1389
+ __typename: true,
1390
+ message: true
1391
+ },
1392
+ "...on ErrorNotExist": {
1393
+ __typename: true,
1394
+ message: true
1395
+ },
1396
+ "...on RecoverNodeResponse": {
1397
+ __typename: true,
1398
+ recoverNode: true
1399
+ }
1400
+ }]
1401
+ });
1402
+
1403
+ if (!recoverNode) {
1404
+ return null;
1405
+ }
1406
+
1407
+ if (recoverNode.__typename === "ErrorAccessDenied") {
1408
+ return recoverNode;
1409
+ }
1410
+
1411
+ if (recoverNode.__typename === "ErrorNotExist") {
1412
+ return recoverNode;
1413
+ }
1414
+
1415
+ return {
1416
+ __typename: "SuccessResponse",
1417
+ data: (_recoverNode$recoverN = recoverNode.recoverNode) != null ? _recoverNode$recoverN : false
1418
+ };
1419
+ }
1420
+
1421
+ async moveNodes(_ref18) {
1422
+ var _moveNodes$moveNodes;
1423
+
1424
+ let {
1425
+ nodeIds,
1426
+ parentNodeId
1427
+ } = _ref18;
1428
+ const {
1429
+ moveNodes
1430
+ } = await _classPrivateFieldLooseBase(this, _thunder)[_thunder]("mutation")({
1431
+ moveNodes: [{
1432
+ nodeIds,
1433
+ parentNodeId
1434
+ }, {
1435
+ "...on ErrorAccessDenied": {
1436
+ __typename: true,
1437
+ message: true
1438
+ },
1439
+ "...on MoveNodesResponse": {
1440
+ __typename: true,
1441
+ moveNodes: true
1442
+ }
1443
+ }]
1444
+ });
1445
+
1446
+ if (!moveNodes) {
1447
+ return null;
1448
+ }
1449
+
1450
+ if (moveNodes.__typename === "ErrorAccessDenied") {
1451
+ return moveNodes;
1452
+ }
1453
+
1454
+ return {
1455
+ __typename: "SuccessResponse",
1456
+ data: (_moveNodes$moveNodes = moveNodes.moveNodes) != null ? _moveNodes$moveNodes : false
1457
+ };
1458
+ }
1459
+
1460
+ async saveInCloud(_ref19) {
1461
+ var _node$parent;
1462
+
1463
+ let {
1464
+ fileId,
1465
+ name,
1466
+ nodeId
1467
+ } = _ref19;
1468
+
1469
+ if (nodeId && !nodesCache.has(nodeId)) {
1470
+ await this.node({
1471
+ id: nodeId
1472
+ });
1473
+
1474
+ if (!nodesCache.has(nodeId)) {
1475
+ return {
1476
+ __typename: "ErrorBasic",
1477
+ message: "The node " + nodeId + " does not exists"
1478
+ };
1479
+ }
1480
+ }
1481
+
1482
+ let key = "";
1483
+ const file = filesCache.get(fileId);
1484
+
1485
+ if (!file) {
1486
+ var _filesCache$get;
1487
+
1488
+ await this.file({
1489
+ id: fileId
1490
+ });
1491
+ const file = (_filesCache$get = filesCache.get(fileId)) != null ? _filesCache$get : null;
1492
+
1493
+ if (!file) {
1494
+ const receivedMails = await _classPrivateFieldLooseBase(this, _client)[_client].mail.receivedMails();
1495
+
1496
+ if (!receivedMails) {
1497
+ return null;
1498
+ }
1499
+
1500
+ if (receivedMails.__typename !== "SuccessResponse") {
1501
+ return null;
1502
+ }
1503
+
1504
+ const mail = receivedMails.data.find(m => m.files.some(f => f.id === fileId));
1505
+
1506
+ if (!mail) {
1507
+ return {
1508
+ __typename: "ErrorBasic",
1509
+ message: "Can't find mail with the file " + fileId
1510
+ };
1511
+ }
1512
+
1513
+ const fileMail = mail.files.find(f => f.id === fileId);
1514
+
1515
+ if (!fileMail) {
1516
+ return {
1517
+ __typename: "ErrorBasic",
1518
+ message: "Can't find mail with the file " + fileId
1519
+ };
1520
+ }
1521
+
1522
+ const fileKey = decryptCryptoBox(sodium.from_hex(fileMail.key), mail.sender.publicKey, _classPrivateFieldLooseBase(this, _keys)[_keys].privateKey);
1523
+ key = sodium.to_hex(fileKey);
1524
+ } else {
1525
+ key = file.key;
1526
+ }
1527
+ } else {
1528
+ key = file.key;
1529
+ }
1530
+
1531
+ if (key === "") {
1532
+ return {
1533
+ __typename: "ErrorBasic",
1534
+ message: "Unexpected error 3"
1535
+ };
1536
+ }
1537
+
1538
+ key = sodium.to_hex(encryptCryptoBox(sodium.from_hex(key), _classPrivateFieldLooseBase(this, _keys)[_keys].publicKey, _classPrivateFieldLooseBase(this, _keys)[_keys].privateKey));
1539
+ const nameKey = secretstreamKeygen();
1540
+ const encryptedName = await encryptName(name, sodium.to_hex(nameKey));
1541
+ const encryptedNameKey = sodium.to_hex(encryptCryptoBox(nameKey, _classPrivateFieldLooseBase(this, _keys)[_keys].publicKey, _classPrivateFieldLooseBase(this, _keys)[_keys].privateKey));
1542
+ const {
1543
+ saveInCloud
1544
+ } = await _classPrivateFieldLooseBase(this, _thunder)[_thunder]("mutation")({
1545
+ saveInCloud: [{
1546
+ fileId,
1547
+ key,
1548
+ nodeId,
1549
+ filename: encryptedName,
1550
+ nameKey: encryptedNameKey
1551
+ }, {
1552
+ "...on ErrorAccessDenied": {
1553
+ __typename: true,
1554
+ message: true
1555
+ },
1556
+ "...on ErrorNotFound": {
1557
+ __typename: true,
1558
+ message: true
1559
+ },
1560
+ "...on ErrorLimit": {
1561
+ __typename: true,
1562
+ message: true
1563
+ },
1564
+ "...on ErrorBasic": {
1565
+ __typename: true,
1566
+ message: true
1567
+ },
1568
+ "...on ErrorNotExist": {
1569
+ __typename: true,
1570
+ message: true
1571
+ },
1572
+ "...on SaveInCloudResponse": {
1573
+ __typename: true,
1574
+ saveInCloud: nodeFullSelector
1575
+ }
1576
+ }]
1577
+ });
1578
+
1579
+ if (!saveInCloud) {
1580
+ return null;
1581
+ }
1582
+
1583
+ if (saveInCloud.__typename === "ErrorAccessDenied") {
1584
+ return saveInCloud;
1585
+ }
1586
+
1587
+ if (saveInCloud.__typename === "ErrorNotFound") {
1588
+ return saveInCloud;
1589
+ }
1590
+
1591
+ if (saveInCloud.__typename === "ErrorLimit") {
1592
+ return saveInCloud;
1593
+ }
1594
+
1595
+ if (saveInCloud.__typename === "ErrorBasic") {
1596
+ return saveInCloud;
1597
+ }
1598
+
1599
+ if (saveInCloud.__typename === "ErrorNotExist") {
1600
+ return saveInCloud;
1601
+ }
1602
+
1603
+ if (!saveInCloud.saveInCloud) {
1604
+ return null;
1605
+ }
1606
+
1607
+ const node = await gqlNodeToExternalNodeFull(saveInCloud.saveInCloud, _classPrivateFieldLooseBase(this, _keys)[_keys]);
1608
+ const me = (_node$parent = node.parent) == null ? void 0 : _node$parent.users.find(_ref20 => {
1609
+ let [u] = _ref20;
1610
+ return u.publicKey === _classPrivateFieldLooseBase(this, _keys)[_keys].publicKey;
1611
+ });
1612
+
1613
+ if (me && ["admin", "write"].includes(me[1])) {
1614
+ var _node$parent$users$fi, _node$parent2;
1615
+
1616
+ const others = (_node$parent$users$fi = (_node$parent2 = node.parent) == null ? void 0 : _node$parent2.users.filter(_ref21 => {
1617
+ let [u] = _ref21;
1618
+ return u.publicKey !== _classPrivateFieldLooseBase(this, _keys)[_keys].publicKey;
1619
+ })) != null ? _node$parent$users$fi : [];
1620
+ await Promise.all(others.map(_ref22 => {
1621
+ let [u, rights] = _ref22;
1622
+ return this.shareNode({
1623
+ nodeId: node.id,
1624
+ rights,
1625
+ userId: u.id
1626
+ });
1627
+ }));
1628
+ }
1629
+
1630
+ return {
1631
+ __typename: "SuccessResponse",
1632
+ data: node
1633
+ };
1634
+ }
1635
+
1636
+ }
1637
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["axios","ky","encryptName","nodesCache","filesCache","secretstreamKeygen","decryptCryptoBox","encryptCryptoBox","compress","uncompress","sodium","enumerate","chunks","promiseAllLimit","concatenate","md5","decrypt","encrypt","gqlFileToExternal","gqlNodeFullToInternalFull","gqlNodeToExternal","gqlNodeToExternalNodeFull","internalNodeToNode","fileSelector","nodeFullSelector","nodeSelector","SecrecyCloudClient","constructor","client","keys","thunder","perNode","nodeId","publicKey","node","get","id","nameKey","access","to_hex","from_hex","privateKey","files","history","map","f","key","addFileToHistory","fileId","__typename","message","file","find","users","filter","u","input","shareFileInHistory","result","data","uploadFile","encryptProgress","uploadProgress","signal","fileKey","fileBuffer","File","Uint8Array","arrayBuffer","compressed","encryptedFile","md5File","md5Encrypted","encryptedFileKey","scalars","Json","encode","e","JSON","stringify","decode","parse","DateTime","Date","toISOString","BigInt","toString","fileSize","byteLength","fileSizeBefore","filePartSize","parts","fields","order","url","total","current","percent","length","uploadPartEnded","uploadFilePartEnd","uploadEnded","uploadFileEnd","chunkParts","Array","index","chunk","Number","push","progressParts","onProgress","part","progressEvent","Object","values","reduce","prv","cur","loaded","byPart","formData","FormData","p","value","entries","append","Blob","post","onUploadProgress","uploadFileInCloud","name","saveInCloud","deletedNodes","nodes","sharedNodes","folder","nodesSharedWithMe","type","deleteNodeSharing","userId","duplicateNode","folderId","customName","Error","deleteNodeCloudTrash","ids","createFolder","parentFolderId","encryptedName","encryptedKey","parent","Promise","all","rights","shareNode","deleted","user","withPublicKey","shareNodes","shareNodeFinish","updateNode","isFavorite","deletedAt","fileContent","onDownloadProgress","progressDecrypt","Bytes","console","log","from","contentUrl","totalSize","content","maybeParts","maybeContent","senderPublicKey","transferredBytes","totalBytes","encryptedContentFromParts","fileParts","buf","timeout","pr","md5Part","sort","a","b","finalize","encryptedContent","src","md5Content","deleteFile","deleteNode","emptyTrash","emptyCloudTrash","recoverNode","moveNodes","nodeIds","parentNodeId","has","receivedMails","mail","m","some","fileMail","sender","encryptedNameKey","filename","me","includes","others"],"sources":["../../src/client/SecrecyCloudClient.ts"],"sourcesContent":["/* eslint-disable @typescript-eslint/naming-convention */\nimport axios from \"axios\";\nimport ky from \"ky\";\nimport type {\n  DownloadProgress,\n  ProgressCallback,\n  SecrecyClient,\n  SuccessResponse\n} from \"../index.js\";\nimport { encryptName } from \"../index.js\";\nimport { nodesCache, filesCache } from \"../cache.js\";\nimport { secretstreamKeygen } from \"../crypto/file.js\";\nimport type { KeyPair } from \"../crypto/index.js\";\nimport { decryptCryptoBox } from \"../crypto/index.js\";\nimport { encryptCryptoBox } from \"../crypto/index.js\";\nimport type {\n  ErrorAccessDenied,\n  ErrorBasic,\n  ErrorLimit,\n  ErrorNotExist,\n  ErrorNotFound\n} from \"../error.js\";\nimport { compress, uncompress } from \"../minify/index.js\";\nimport { sodium } from \"../sodium.js\";\nimport {\n  enumerate,\n  chunks,\n  promiseAllLimit,\n  concatenate\n} from \"../utils/utils.js\";\nimport { md5 } from \"../worker/md5.js\";\nimport { decrypt, encrypt } from \"../worker/sodium.js\";\nimport type { NodeType, Rights, Thunder } from \"../zeus/index.js\";\nimport { gqlFileToExternal } from \"./convert/file.js\";\nimport {\n  gqlNodeFullToInternalFull,\n  gqlNodeToExternal,\n  gqlNodeToExternalNodeFull,\n  internalNodeToNode\n} from \"./convert/node.js\";\nimport type {\n  FileContentPart,\n  FileMetadata,\n  FilePartResponse,\n  NameKeyInput,\n  Node,\n  NodeFull,\n  ShareFileInHistoryInput,\n  ShareNodesInput\n} from \"./types/index.js\";\nimport {\n  fileSelector,\n  nodeFullSelector,\n  nodeSelector\n} from \"./types/selectors.js\";\n// import { md5 } from \"../worker/index.js\";\n// import { firstValueFrom, of } from \"rxjs\";\n\nexport class SecrecyCloudClient {\n  #client: SecrecyClient;\n\n  #keys: KeyPair;\n\n  #thunder: ReturnType<typeof Thunder>;\n\n  constructor(\n    client: SecrecyClient,\n    keys: KeyPair,\n    thunder: ReturnType<typeof Thunder>\n  ) {\n    this.#client = client;\n    this.#keys = keys;\n    this.#thunder = thunder;\n  }\n\n  async addFileToHistory({\n    fileId,\n    nodeId\n  }: {\n    fileId: string;\n    nodeId: string;\n  }): Promise<\n    SuccessResponse<Node> | ErrorAccessDenied | ErrorNotExist | null\n  > {\n    const { addFileToHistory } = await this.#thunder(\"mutation\")({\n      addFileToHistory: [\n        {\n          fileId,\n          nodeId\n        },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on ErrorNotExist\": {\n            __typename: true,\n            message: true\n          },\n          \"...on AddFileToHistoryResponse\": {\n            __typename: true,\n            addFileToHistory: nodeFullSelector\n          }\n        }\n      ]\n    });\n\n    if (!addFileToHistory) {\n      return null;\n    }\n\n    if (addFileToHistory.__typename === \"ErrorAccessDenied\") {\n      return addFileToHistory;\n    }\n\n    if (addFileToHistory.__typename === \"ErrorNotExist\") {\n      return addFileToHistory;\n    }\n\n    const node = await gqlNodeFullToInternalFull(\n      addFileToHistory.addFileToHistory,\n      this.#keys\n    );\n    const file = node.history.find(f => f.id === fileId);\n    if (file) {\n      const users = node.users.filter(\n        ([u]) => u.publicKey !== this.#keys.publicKey\n      );\n      const input: ShareFileInHistoryInput = {\n        fileId: file.id,\n        users: users.map(([u]) => ({\n          id: u.id,\n          key: sodium.to_hex(\n            encryptCryptoBox(\n              sodium.from_hex(file.key),\n              this.#keys.publicKey,\n              this.#keys.privateKey\n            )\n          )\n        }))\n      };\n\n      await this.#thunder(\"mutation\")({\n        shareFileInHistory: [\n          { input, nodeId },\n          {\n            \"...on ErrorAccessDenied\": {\n              __typename: true,\n              message: true\n            },\n            \"...on ErrorNotFound\": {\n              __typename: true,\n              message: true\n            },\n            \"...on ShareFileInHistoryResponse\": {\n              __typename: true,\n              shareFileInHistory: true\n            }\n          }\n        ]\n      });\n    }\n    const result = internalNodeToNode(node);\n    return {\n      __typename: \"SuccessResponse\",\n      data: result\n    };\n  }\n\n  async uploadFile({\n    file,\n    encryptProgress,\n    uploadProgress,\n    signal\n  }: {\n    file: globalThis.File | Uint8Array;\n    encryptProgress?: ProgressCallback;\n    uploadProgress?: ProgressCallback;\n    signal?: AbortSignal;\n  }): Promise<\n    | SuccessResponse<string>\n    | ErrorAccessDenied\n    | ErrorLimit\n    | ErrorNotFound\n    | null\n  > {\n    const fileKey = secretstreamKeygen();\n    const fileBuffer =\n      file instanceof File ? new Uint8Array(await file.arrayBuffer()) : file;\n    const compressed = compress(fileBuffer);\n\n    const {\n      data: encryptedFile,\n      md5: md5File,\n      md5Encrypted\n    } = await encrypt(fileKey, compressed, encryptProgress, signal);\n\n    const encryptedFileKey = encryptCryptoBox(\n      fileKey,\n      this.#keys.publicKey,\n      this.#keys.privateKey\n    );\n\n    const { uploadFile } = await this.#thunder(\"mutation\", {\n      scalars: {\n        Json: {\n          encode: (e: unknown) => JSON.stringify(e),\n          decode: (e: unknown) => JSON.parse(e as string)\n        },\n        DateTime: {\n          decode: (e: unknown) => new Date(e as string),\n          encode: (e: unknown) => (e as Date).toISOString()\n        },\n        BigInt: {\n          decode: (e: unknown) => BigInt(e as string),\n          encode: (e: unknown) => (e as bigint).toString()\n        }\n      }\n    })({\n      uploadFile: [\n        {\n          fileSize: encryptedFile.byteLength,\n          fileSizeBefore: fileBuffer.byteLength,\n          fileKey: sodium.to_hex(encryptedFileKey),\n          md5Encrypted,\n          md5: md5File\n        },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on ErrorLimit\": {\n            __typename: true,\n            message: true\n          },\n          \"...on ErrorNotFound\": {\n            __typename: true,\n            message: true\n          },\n          \"...on UploadFileResponse\": {\n            __typename: true,\n            uploadFile: {\n              fileId: true,\n              filePartSize: true,\n              parts: {\n                fields: true,\n                order: true,\n                url: true\n              }\n            }\n          }\n        }\n      ]\n    });\n\n    if (!uploadFile) {\n      return null;\n    }\n\n    if (uploadFile.__typename === \"ErrorAccessDenied\") {\n      return uploadFile;\n    }\n\n    if (uploadFile.__typename === \"ErrorLimit\") {\n      return uploadFile;\n    }\n\n    if (uploadFile.__typename === \"ErrorNotFound\") {\n      return uploadFile;\n    }\n\n    if (!uploadFile.uploadFile) {\n      return null;\n    }\n\n    uploadProgress?.({\n      total: encryptedFile.byteLength,\n      current: 0,\n      percent: 0\n    });\n\n    if (uploadFile.uploadFile.parts.length === 0) {\n      uploadProgress?.({\n        total: encryptedFile.byteLength,\n        current: encryptedFile.byteLength,\n        percent: 1\n      });\n\n      return {\n        __typename: \"SuccessResponse\",\n        data: uploadFile.uploadFile.fileId\n      };\n    }\n\n    const uploadPartEnded = async (\n      md5: string,\n      order: number\n    ): Promise<SuccessResponse<boolean> | ErrorAccessDenied | null> => {\n      if (!uploadFile.uploadFile) {\n        return null;\n      }\n      const { uploadFilePartEnd } = await this.#thunder(\"mutation\")({\n        uploadFilePartEnd: [\n          {\n            fileId: uploadFile.uploadFile.fileId,\n            md5,\n            order\n          },\n          {\n            \"...on ErrorAccessDenied\": {\n              __typename: true,\n              message: true\n            },\n            \"...on UploadFilePartEndResponse\": {\n              __typename: true,\n              uploadFilePartEnd: true\n            }\n          }\n        ]\n      });\n\n      if (!uploadFilePartEnd) {\n        return null;\n      }\n\n      if (uploadFilePartEnd.__typename === \"ErrorAccessDenied\") {\n        return uploadFilePartEnd;\n      }\n\n      return {\n        __typename: \"SuccessResponse\",\n        data: uploadFilePartEnd.uploadFilePartEnd\n      };\n    };\n\n    const uploadEnded = async (): Promise<\n      SuccessResponse<string> | ErrorAccessDenied | ErrorNotFound | null\n    > => {\n      if (!uploadFile.uploadFile) {\n        return null;\n      }\n      const { uploadFileEnd } = await this.#thunder(\"mutation\")({\n        uploadFileEnd: [\n          {\n            fileId: uploadFile.uploadFile.fileId\n          },\n          {\n            \"...on ErrorAccessDenied\": {\n              __typename: true,\n              message: true\n            },\n            \"...on ErrorNotFound\": {\n              __typename: true,\n              message: true\n            },\n            \"...on UploadFileEndResponse\": {\n              __typename: true,\n              uploadFileEnd: true\n            }\n          }\n        ]\n      });\n\n      if (!uploadFileEnd) {\n        return null;\n      }\n\n      if (uploadFileEnd.__typename === \"ErrorAccessDenied\") {\n        return uploadFileEnd;\n      }\n\n      if (uploadFileEnd.__typename === \"ErrorNotFound\") {\n        return uploadFileEnd;\n      }\n\n      if (!uploadFileEnd.uploadFileEnd) {\n        return null;\n      }\n\n      return {\n        __typename: \"SuccessResponse\",\n        data: uploadFileEnd.uploadFileEnd\n      };\n    };\n\n    const chunkParts = new Array<{\n      order: number;\n      data: Uint8Array;\n      md5: string;\n    }>();\n\n    for (const [index, chunk] of enumerate(\n      chunks(encryptedFile, Number(uploadFile.uploadFile.filePartSize))\n    )) {\n      chunkParts.push({\n        order: index + 1,\n        data: chunk,\n        md5: await md5(chunk)\n      });\n    }\n\n    const progressParts: Record<number, ProgressEvent> = {};\n    const onProgress = (part: number, progressEvent: ProgressEvent): void => {\n      progressParts[part] = progressEvent;\n      const current = Object.values(progressParts).reduce(\n        (prv, cur) => prv + cur.loaded,\n        0\n      );\n      uploadProgress?.({\n        percent: current / encryptedFile.byteLength,\n        total: encryptedFile.byteLength,\n        current\n      });\n    };\n\n    const byPart = async (part: FilePartResponse): Promise<void> => {\n      if (!uploadFile.uploadFile) {\n        return;\n      }\n      const formData = new FormData();\n      const chunk = chunkParts.find(p => p.order === part.order);\n      if (!chunk) {\n        return;\n      }\n      for (const [key, value] of Object.entries(part.fields)) {\n        formData.append(key, value);\n      }\n      formData.append(\n        \"file\",\n        new Blob([chunk.data]),\n        `${uploadFile.uploadFile.fileId}-${chunk.order}`\n      );\n\n      await axios.post(part.url, formData, {\n        onUploadProgress: progressEvent =>\n          onProgress(part.order, progressEvent),\n        signal\n      });\n\n      await uploadPartEnded(chunk.md5, chunk.order);\n      // if ((e as any).response.status === 0) {\n      //   // TODO https://github.com/sindresorhus/ky/issues/305\n      // } else {\n      //   throw e;\n      // }\n    };\n\n    if (!uploadFile.uploadFile) {\n      return null;\n    }\n\n    await promiseAllLimit(\n      3,\n      uploadFile.uploadFile.parts.map(p => (): Promise<void> => byPart(p))\n    );\n\n    const result = await uploadEnded();\n\n    if (!result) {\n      return null;\n    }\n\n    if (result.__typename === \"ErrorAccessDenied\") {\n      return result;\n    }\n\n    if (result.__typename === \"ErrorNotFound\") {\n      return result;\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: result.data\n    };\n  }\n\n  async uploadFileInCloud({\n    file,\n    name,\n    nodeId,\n    encryptProgress,\n    uploadProgress,\n    signal\n  }: {\n    file: globalThis.File | Uint8Array;\n    name: string;\n    nodeId?: string;\n    encryptProgress?: ProgressCallback;\n    uploadProgress?: ProgressCallback;\n    signal?: AbortSignal;\n  }): Promise<\n    | SuccessResponse<NodeFull>\n    | ErrorAccessDenied\n    | ErrorLimit\n    | ErrorNotFound\n    | ErrorBasic\n    | ErrorNotExist\n    | null\n  > {\n    const fileId = await this.uploadFile({\n      file,\n      encryptProgress,\n      uploadProgress,\n      signal\n    });\n\n    if (!fileId) {\n      return null;\n    }\n\n    if (fileId.__typename === \"ErrorAccessDenied\") {\n      return fileId;\n    }\n\n    if (fileId.__typename === \"ErrorLimit\") {\n      return fileId;\n    }\n\n    if (fileId.__typename === \"ErrorNotFound\") {\n      return fileId;\n    }\n\n    const result = await this.saveInCloud({\n      fileId: fileId.data,\n      name,\n      nodeId\n    });\n\n    if (!result) {\n      return null;\n    }\n\n    if (result.__typename === \"ErrorAccessDenied\") {\n      return result;\n    }\n\n    if (result.__typename === \"ErrorBasic\") {\n      return result;\n    }\n\n    if (result.__typename === \"ErrorLimit\") {\n      return result;\n    }\n\n    if (result.__typename === \"ErrorNotFound\") {\n      return result;\n    }\n\n    if (result.__typename === \"ErrorNotExist\") {\n      return result;\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: result.data\n    };\n  }\n\n  async deletedNodes(): Promise<\n    SuccessResponse<Node[]> | ErrorAccessDenied | null\n  > {\n    const { deletedNodes } = await this.#thunder(\"query\")({\n      deletedNodes: {\n        \"...on ErrorAccessDenied\": {\n          __typename: true,\n          message: true\n        },\n        \"...on DeletedNodesResponse\": {\n          __typename: true,\n          deletedNodes: nodeSelector\n        }\n      }\n    });\n\n    if (!deletedNodes) {\n      return null;\n    }\n\n    if (deletedNodes.__typename === \"ErrorAccessDenied\") {\n      return deletedNodes;\n    }\n\n    const nodes = new Array<Node>();\n\n    for (const node of deletedNodes.deletedNodes) {\n      nodes.push(await gqlNodeToExternal(node, this.#keys));\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: nodes\n    };\n  }\n\n  async sharedNodes(): Promise<\n    SuccessResponse<Node[]> | ErrorAccessDenied | null\n  > {\n    const { sharedNodes } = await this.#thunder(\"query\")({\n      sharedNodes: {\n        \"...on ErrorAccessDenied\": {\n          __typename: true,\n          message: true\n        },\n        \"...on SharedNodesResponse\": {\n          __typename: true,\n          sharedNodes: nodeSelector\n        }\n      }\n    });\n\n    if (!sharedNodes) {\n      return null;\n    }\n\n    if (sharedNodes.__typename === \"ErrorAccessDenied\") {\n      return sharedNodes;\n    }\n\n    const nodes = new Array<Node>();\n\n    for (const folder of sharedNodes.sharedNodes) {\n      nodes.push(await gqlNodeToExternal(folder, this.#keys));\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: nodes\n    };\n  }\n\n  async nodesSharedWithMe(\n    type?: NodeType | null | undefined\n  ): Promise<SuccessResponse<Node[]> | ErrorAccessDenied | null> {\n    const { nodesSharedWithMe } = await this.#thunder(\"query\")({\n      nodesSharedWithMe: [\n        {\n          type\n        },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on NodesSharedWithMeResponse\": {\n            __typename: true,\n            nodesSharedWithMe: nodeSelector\n          }\n        }\n      ]\n    });\n\n    if (!nodesSharedWithMe) {\n      return null;\n    }\n\n    if (nodesSharedWithMe.__typename === \"ErrorAccessDenied\") {\n      return nodesSharedWithMe;\n    }\n\n    const nodes = new Array<Node>();\n\n    for (const folder of nodesSharedWithMe.nodesSharedWithMe) {\n      nodes.push(await gqlNodeToExternal(folder, this.#keys));\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: nodes\n    };\n  }\n\n  async deleteNodeSharing({\n    nodeId,\n    userId\n  }: {\n    nodeId: string;\n    userId: string;\n  }): Promise<SuccessResponse<boolean> | ErrorAccessDenied | null> {\n    const { deleteNodeSharing } = await this.#thunder(\"mutation\")({\n      deleteNodeSharing: [\n        {\n          nodeId,\n          userId\n        },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on DeleteNodeSharingResponse\": {\n            __typename: true,\n            deleteNodeSharing: true\n          }\n        }\n      ]\n    });\n\n    if (!deleteNodeSharing) {\n      return null;\n    }\n\n    if (deleteNodeSharing.__typename === \"ErrorAccessDenied\") {\n      return deleteNodeSharing;\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: deleteNodeSharing.deleteNodeSharing\n    };\n  }\n\n  async duplicateNode({\n    nodeId,\n    folderId,\n    customName\n  }: {\n    nodeId: string;\n    folderId?: string | null | undefined;\n    customName?: string | null | undefined;\n  }): Promise<\n    SuccessResponse<boolean> | ErrorAccessDenied | ErrorNotFound | null\n  > {\n    let node = nodesCache.get(nodeId);\n    if (!node) {\n      await this.node({ id: nodeId });\n      node = nodesCache.get(nodeId);\n      if (!node) {\n        throw new Error(`Node (${nodeId}) does not exists`);\n      }\n    }\n    if (!node.access?.nameKey) {\n      throw new Error(`Can't have access to node ${nodeId}`);\n    }\n    customName = customName\n      ? await encryptName(customName, node.access.nameKey)\n      : null;\n\n    const { duplicateNode } = await this.#thunder(\"mutation\")({\n      duplicateNode: [\n        {\n          nodeId,\n          folderId,\n          customName\n        },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on ErrorNotFound\": {\n            __typename: true,\n            message: true\n          },\n          \"...on DuplicateNodeResponse\": {\n            __typename: true,\n            duplicateNode: true\n          }\n        }\n      ]\n    });\n\n    if (!duplicateNode) {\n      return null;\n    }\n\n    if (duplicateNode.__typename === \"ErrorAccessDenied\") {\n      return duplicateNode;\n    }\n\n    if (duplicateNode.__typename === \"ErrorNotFound\") {\n      return duplicateNode;\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: duplicateNode.duplicateNode\n    };\n  }\n\n  async deleteNodeCloudTrash({\n    ids\n  }: {\n    ids: Array<string>;\n  }): Promise<SuccessResponse<boolean> | ErrorAccessDenied | null> {\n    const { deleteNodeCloudTrash } = await this.#thunder(\"mutation\")({\n      deleteNodeCloudTrash: [\n        {\n          ids\n        },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on DeleteNodeCloudTrashResponse\": {\n            __typename: true,\n            deleteNodeCloudTrash: true\n          }\n        }\n      ]\n    });\n\n    if (!deleteNodeCloudTrash) {\n      return null;\n    }\n\n    if (deleteNodeCloudTrash.__typename === \"ErrorAccessDenied\") {\n      return deleteNodeCloudTrash;\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: deleteNodeCloudTrash.deleteNodeCloudTrash ?? false\n    };\n  }\n\n  async createFolder({\n    name,\n    parentFolderId\n  }: {\n    name: string;\n    parentFolderId?: string | null;\n  }): Promise<\n    SuccessResponse<NodeFull> | ErrorAccessDenied | ErrorNotExist | null\n  > {\n    const key = secretstreamKeygen();\n    const encryptedName = await encryptName(name, sodium.to_hex(key));\n    const encryptedKey = encryptCryptoBox(\n      key,\n      this.#keys.publicKey,\n      this.#keys.privateKey\n    );\n    const { createFolder } = await this.#thunder(\"mutation\")({\n      createFolder: [\n        {\n          name: encryptedName,\n          parentFolderId,\n          key: sodium.to_hex(encryptedKey)\n        },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on ErrorNotExist\": {\n            __typename: true,\n            message: true\n          },\n          \"...on CreateFolderResponse\": {\n            __typename: true,\n            createFolder: nodeFullSelector\n          }\n        }\n      ]\n    });\n    if (!createFolder) {\n      //throw new Error(`Can't create folder`);\n      return null;\n    }\n\n    if (createFolder.__typename === \"ErrorAccessDenied\") {\n      return createFolder;\n    }\n\n    if (createFolder.__typename === \"ErrorNotExist\") {\n      return createFolder;\n    }\n\n    if (!createFolder.createFolder) {\n      return null;\n    }\n\n    const folder = await gqlNodeToExternalNodeFull(\n      createFolder.createFolder,\n      this.#keys\n    );\n\n    const users =\n      folder.parent?.users?.filter(\n        ([u]) => u.publicKey !== this.#keys.publicKey\n      ) ?? [];\n\n    if (users.length) {\n      await Promise.all(\n        users.map(([u, rights]) =>\n          this.shareNode({\n            nodeId: folder.id,\n            rights,\n            userId: u.id\n          })\n        )\n      );\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: folder\n    };\n  }\n\n  async node({\n    id,\n    deleted\n  }: {\n    id?: string | null | undefined;\n    deleted?: boolean | null | undefined;\n  } = {}): Promise<SuccessResponse<NodeFull> | ErrorAccessDenied | null> {\n    const { node } = await this.#thunder(\"query\")({\n      node: [\n        { deleted, id },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on NodeResponse\": {\n            __typename: true,\n            node: nodeFullSelector\n          }\n        }\n      ]\n    });\n    if (!node) {\n      return null;\n    }\n\n    if (node.__typename === \"ErrorAccessDenied\") {\n      return node;\n    }\n    if (!node.node) {\n      return null;\n    }\n\n    const result = await gqlNodeToExternalNodeFull(node.node, this.#keys);\n    return {\n      __typename: \"SuccessResponse\",\n      data: result\n    };\n  }\n\n  async file({\n    id\n  }: {\n    id: string;\n  }): Promise<SuccessResponse<FileMetadata> | ErrorAccessDenied | null> {\n    const { file } = await this.#thunder(\"query\")({\n      file: [\n        {\n          id\n        },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on FileQueryResponse\": {\n            __typename: true,\n            file: fileSelector\n          }\n        }\n      ]\n    });\n    if (!file) {\n      return null;\n    }\n\n    if (file.__typename === \"ErrorAccessDenied\") {\n      return file;\n    }\n\n    if (!file.file) {\n      return null;\n    }\n\n    const result = gqlFileToExternal(file.file, this.#keys);\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: result\n    };\n  }\n\n  async shareNode({\n    nodeId,\n    userId,\n    rights\n  }: {\n    nodeId: string;\n    userId: string;\n    rights: Rights;\n  }): Promise<\n    SuccessResponse<boolean> | ErrorAccessDenied | ErrorNotFound | null\n  > {\n    const user = await this.#client.user({ userId, withPublicKey: true });\n\n    if (!user) {\n      return user;\n    }\n\n    if (user.__typename === \"ErrorNotFound\") {\n      return user;\n    }\n\n    const { shareNode } = await this.#thunder(\"mutation\")({\n      shareNode: [\n        { nodeId, userId },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on ErrorNotFound\": {\n            __typename: true,\n            message: true\n          },\n          \"...on ShareNodeResponse\": {\n            __typename: true,\n            nodes: true\n          }\n        }\n      ]\n    });\n\n    if (!shareNode) {\n      return null;\n    }\n\n    if (shareNode.__typename === \"ErrorAccessDenied\") {\n      return shareNode;\n    }\n\n    if (shareNode.__typename === \"ErrorNotFound\") {\n      return shareNode;\n    }\n\n    const shareNodes: ShareNodesInput = {\n      nodes: []\n    };\n\n    for (const id of shareNode.nodes) {\n      const nameKey = await this.perNode(id, user.data.publicKey);\n      if (nameKey) {\n        shareNodes.nodes.push(nameKey);\n      }\n    }\n\n    const { shareNodeFinish } = await this.#thunder(\"mutation\")({\n      shareNodeFinish: [\n        {\n          rights,\n          userId,\n          shareNodes\n        },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on ErrorNotFound\": {\n            __typename: true,\n            message: true\n          },\n          \"...on ShareNodeFinishResponse\": {\n            __typename: true,\n            shareNodeFinish: true\n          }\n        }\n      ]\n    });\n\n    if (!shareNodeFinish) {\n      return null;\n    }\n\n    if (shareNodeFinish.__typename === \"ErrorAccessDenied\") {\n      return shareNodeFinish;\n    }\n\n    if (shareNodeFinish.__typename === \"ErrorNotFound\") {\n      return shareNodeFinish;\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: shareNodeFinish.shareNodeFinish ?? false\n    };\n  }\n\n  async updateNode({\n    nodeId,\n    name,\n    isFavorite,\n    deletedAt\n  }: {\n    nodeId: string;\n    name?: string | null | undefined;\n    isFavorite?: boolean | null | undefined;\n    deletedAt?: Date | null | undefined;\n  }): Promise<\n    SuccessResponse<NodeFull> | ErrorAccessDenied | ErrorNotExist | null\n  > {\n    let node = nodesCache.get(nodeId);\n    if (!node) {\n      await this.node({ id: nodeId });\n      node = nodesCache.get(nodeId);\n      if (!node) {\n        throw `Can't find Node ${nodeId}`;\n      }\n    }\n\n    if (!node.access?.nameKey) {\n      throw new Error(`Can't have access to node ${nodeId}`);\n    }\n\n    name = name\n      ? node.access?.nameKey\n        ? await encryptName(name, node.access.nameKey)\n        : name\n      : null;\n\n    const { updateNode } = await this.#thunder(\"mutation\")({\n      updateNode: [\n        {\n          nodeId,\n          name,\n          isFavorite,\n          deletedAt\n        },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on ErrorNotExist\": {\n            __typename: true,\n            message: true\n          },\n          \"...on UpdateNodeResponse\": {\n            __typename: true,\n            updateNode: nodeFullSelector\n          }\n        }\n      ]\n    });\n\n    if (!updateNode) {\n      return null;\n    }\n\n    if (updateNode.__typename === \"ErrorAccessDenied\") {\n      return updateNode;\n    }\n\n    if (updateNode.__typename === \"ErrorNotExist\") {\n      return updateNode;\n    }\n\n    if (!updateNode.updateNode) {\n      return null;\n    }\n\n    const result = await gqlNodeToExternalNodeFull(\n      updateNode.updateNode,\n      this.#keys\n    );\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: result\n    };\n  }\n\n  async fileContent({\n    fileId,\n    onDownloadProgress,\n    progressDecrypt,\n    signal\n  }: {\n    fileId: string;\n    onDownloadProgress?: (progress: DownloadProgress) => void;\n    progressDecrypt?: ProgressCallback;\n    signal?: AbortSignal;\n  }): Promise<\n    SuccessResponse<Uint8Array> | ErrorAccessDenied | ErrorBasic | null\n  > {\n    const { fileContent } = await this.#thunder(\"query\", {\n      scalars: {\n        Bytes: {\n          decode: (e: unknown) => {\n            console.log(e, typeof e);\n            return Uint8Array.from([1, 2, 3]);\n          },\n          encode: (e: unknown) => (e as Uint8Array).toString()\n        }\n      }\n    })({\n      fileContent: [\n        { fileId },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on ErrorBasic\": {\n            __typename: true,\n            message: true\n          },\n          \"...on FileContentResponse\": {\n            __typename: true,\n            file: {\n              \"...on FileContentCloud\": {\n                __typename: true,\n                parts: {\n                  contentUrl: true,\n                  order: true,\n                  md5: true\n                },\n                key: true,\n                publicKey: true,\n                totalSize: true,\n                md5: true,\n                md5Encrypted: true\n              },\n              \"...on FileContentLite\": {\n                __typename: true,\n                content: true,\n                id: true,\n                key: true,\n                publicKey: true,\n                md5: true,\n                md5Encrypted: true,\n                totalSize: true\n              },\n              \"...on FileContentReceivedMail\": {\n                __typename: true,\n                maybeParts: {\n                  contentUrl: true,\n                  order: true,\n                  md5: true\n                },\n                maybeContent: true,\n                key: true,\n                senderPublicKey: true,\n                totalSize: true,\n                md5: true,\n                md5Encrypted: true\n              },\n              \"...on FileContentSentMail\": {\n                __typename: true,\n                maybeParts: {\n                  contentUrl: true,\n                  order: true,\n                  md5: true\n                },\n                maybeContent: true,\n                key: true,\n                totalSize: true,\n                md5: true,\n                md5Encrypted: true\n              }\n            }\n          }\n        }\n      ]\n    });\n\n    if (!fileContent) {\n      return null;\n    }\n\n    if (fileContent.__typename === \"ErrorAccessDenied\") {\n      return fileContent;\n    }\n\n    if (fileContent.__typename === \"ErrorBasic\") {\n      return fileContent;\n    }\n\n    const file = fileContent.file;\n    if (!file) {\n      return null;\n    }\n\n    const progressParts: Record<number, DownloadProgress> = {};\n    const onProgress = (\n      part: number,\n      progressEvent: DownloadProgress\n    ): void => {\n      progressParts[part] = progressEvent;\n      const transferredBytes = Object.values(progressParts).reduce(\n        (prv, cur) => prv + cur.transferredBytes,\n        0\n      );\n      const totalBytes = Number(file.totalSize);\n      onDownloadProgress?.({\n        percent: transferredBytes / totalBytes,\n        totalBytes,\n        transferredBytes\n      });\n    };\n\n    const encryptedContentFromParts = async (\n      fileParts: FileContentPart[]\n    ): Promise<Uint8Array> => {\n      const parts = new Array<{ data: Uint8Array; order: number }>();\n\n      const byPart = async (part: FileContentPart): Promise<void> => {\n        const buf = new Uint8Array(\n          await ky\n            .get(part.contentUrl, {\n              timeout: false,\n              onDownloadProgress: pr => onProgress(part.order, pr),\n              signal: signal\n            })\n            .arrayBuffer()\n        );\n        const md5Part = await md5(buf);\n        if (md5Part !== part.md5) {\n          throw new Error(\n            `Invalid md5 for part ${part.order} of file ${fileId}`\n          );\n        }\n        parts.push({\n          data: buf,\n          order: part.order\n        });\n      };\n\n      await promiseAllLimit(\n        3,\n        fileParts.map(p => (): Promise<void> => byPart(p))\n      );\n\n      return concatenate(\n        ...parts.sort((a, b) => a.order - b.order).map(p => p.data)\n      );\n    };\n\n    const finalize = async (\n      encryptedContent: Uint8Array\n    ): Promise<Uint8Array> => {\n      // const md5Encrypted = await firstValueFrom(md5(of(encryptedContent)));\n      const md5Encrypted = await md5(encryptedContent);\n\n      if (md5Encrypted !== file.md5Encrypted) {\n        throw new Error(`Encrypted content does not match`);\n      }\n\n      const key = decryptCryptoBox(\n        sodium.from_hex(file.key),\n        file.__typename === \"FileContentReceivedMail\"\n          ? file.senderPublicKey\n          : file.__typename === \"FileContentCloud\"\n          ? file.publicKey\n          : this.#keys.publicKey,\n        this.#keys.privateKey\n      );\n\n      const src = await decrypt(key, encryptedContent, progressDecrypt, signal);\n\n      // const md5Content = await firstValueFrom(md5(of(src)));\n      const md5Content = await md5(src);\n\n      if (md5Content !== file.md5) {\n        throw new Error(`Content does not match`);\n      }\n\n      return uncompress(src);\n    };\n\n    const encryptedContent =\n      file.__typename === \"FileContentLite\"\n        ? file.content\n        : file.__typename === \"FileContentCloud\"\n        ? await encryptedContentFromParts(file.parts)\n        : file.maybeContent\n        ? file.maybeContent\n        : file.maybeParts\n        ? await encryptedContentFromParts(file.maybeParts)\n        : null;\n\n    if (!encryptedContent) {\n      return null;\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: await finalize(encryptedContent)\n    };\n  }\n\n  async deleteFile({\n    fileId,\n    nodeId\n  }: {\n    fileId: string;\n    nodeId: string;\n  }): Promise<\n    SuccessResponse<boolean> | ErrorAccessDenied | ErrorNotExist | null\n  > {\n    const { deleteFile } = await this.#thunder(\"mutation\")({\n      deleteFile: [\n        {\n          fileId,\n          nodeId\n        },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on ErrorNotExist\": {\n            __typename: true,\n            message: true\n          },\n          \"...on DeleteFileResponse\": {\n            __typename: true,\n            deleteFile: true\n          }\n        }\n      ]\n    });\n\n    if (!deleteFile) {\n      return null;\n    }\n\n    if (deleteFile.__typename === \"ErrorAccessDenied\") {\n      return deleteFile;\n    }\n\n    if (deleteFile.__typename === \"ErrorNotExist\") {\n      return deleteFile;\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: deleteFile.deleteFile\n    };\n  }\n\n  async deleteNode({\n    nodeId\n  }: {\n    nodeId: string;\n  }): Promise<SuccessResponse<boolean> | ErrorAccessDenied | null> {\n    const { deleteNode } = await this.#thunder(\"mutation\")({\n      deleteNode: [\n        {\n          id: nodeId\n        },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on DeleteNodeResponse\": {\n            __typename: true,\n            deleteNode: true\n          }\n        }\n      ]\n    });\n\n    if (!deleteNode) {\n      return null;\n    }\n\n    if (deleteNode.__typename === \"ErrorAccessDenied\") {\n      return deleteNode;\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: deleteNode.deleteNode\n    };\n  }\n\n  async emptyTrash(): Promise<\n    SuccessResponse<boolean> | ErrorAccessDenied | null\n  > {\n    const { emptyCloudTrash } = await this.#thunder(\"mutation\")({\n      emptyCloudTrash: {\n        \"...on ErrorAccessDenied\": {\n          __typename: true,\n          message: true\n        },\n        \"...on EmptyCloudTrashResponse\": {\n          __typename: true,\n          emptyCloudTrash: true\n        }\n      }\n    });\n\n    if (!emptyCloudTrash) {\n      return null;\n    }\n\n    if (emptyCloudTrash.__typename === \"ErrorAccessDenied\") {\n      return emptyCloudTrash;\n    }\n\n    if (!emptyCloudTrash.emptyCloudTrash) {\n      return null;\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: emptyCloudTrash.emptyCloudTrash\n    };\n  }\n\n  async recoverNode(\n    id: string\n  ): Promise<\n    SuccessResponse<boolean> | ErrorAccessDenied | ErrorNotExist | null\n  > {\n    const { recoverNode } = await this.#thunder(\"mutation\")({\n      recoverNode: [\n        { id },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on ErrorNotExist\": {\n            __typename: true,\n            message: true\n          },\n          \"...on RecoverNodeResponse\": {\n            __typename: true,\n            recoverNode: true\n          }\n        }\n      ]\n    });\n\n    if (!recoverNode) {\n      return null;\n    }\n\n    if (recoverNode.__typename === \"ErrorAccessDenied\") {\n      return recoverNode;\n    }\n\n    if (recoverNode.__typename === \"ErrorNotExist\") {\n      return recoverNode;\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: recoverNode.recoverNode ?? false\n    };\n  }\n\n  async moveNodes({\n    nodeIds,\n    parentNodeId\n  }: {\n    nodeIds: string[];\n    parentNodeId?: string | null | undefined;\n  }): Promise<SuccessResponse<boolean> | ErrorAccessDenied | null> {\n    const { moveNodes } = await this.#thunder(\"mutation\")({\n      moveNodes: [\n        {\n          nodeIds,\n          parentNodeId\n        },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on MoveNodesResponse\": {\n            __typename: true,\n            moveNodes: true\n          }\n        }\n      ]\n    });\n\n    if (!moveNodes) {\n      return null;\n    }\n\n    if (moveNodes.__typename === \"ErrorAccessDenied\") {\n      return moveNodes;\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: moveNodes.moveNodes ?? false\n    };\n  }\n\n  async saveInCloud({\n    fileId,\n    name,\n    nodeId\n  }: {\n    fileId: string;\n    name: string;\n    nodeId?: string;\n  }): Promise<\n    | SuccessResponse<NodeFull>\n    | ErrorAccessDenied\n    | ErrorBasic\n    | ErrorLimit\n    | ErrorNotFound\n    | ErrorNotExist\n    | null\n  > {\n    if (nodeId && !nodesCache.has(nodeId)) {\n      await this.node({ id: nodeId });\n      if (!nodesCache.has(nodeId)) {\n        return {\n          __typename: \"ErrorBasic\",\n          message: `The node ${nodeId} does not exists`\n        };\n      }\n    }\n\n    let key = \"\";\n\n    const file = filesCache.get(fileId);\n\n    if (!file) {\n      await this.file({ id: fileId });\n      const file = filesCache.get(fileId) ?? null;\n      if (!file) {\n        const receivedMails = await this.#client.mail.receivedMails();\n        if (!receivedMails) {\n          return null;\n        }\n        if (receivedMails.__typename !== \"SuccessResponse\") {\n          return null;\n        }\n        const mail = receivedMails.data.find(m =>\n          m.files.some(f => f.id === fileId)\n        );\n\n        if (!mail) {\n          return {\n            __typename: \"ErrorBasic\",\n            message: `Can't find mail with the file ${fileId}`\n          };\n        }\n\n        const fileMail = mail.files.find(f => f.id === fileId);\n\n        if (!fileMail) {\n          return {\n            __typename: \"ErrorBasic\",\n            message: `Can't find mail with the file ${fileId}`\n          };\n        }\n\n        const fileKey = decryptCryptoBox(\n          sodium.from_hex(fileMail.key),\n          mail.sender.publicKey,\n          this.#keys.privateKey\n        );\n\n        key = sodium.to_hex(fileKey);\n      } else {\n        key = file.key;\n      }\n    } else {\n      key = file.key;\n    }\n\n    if (key === \"\") {\n      return {\n        __typename: \"ErrorBasic\",\n        message: \"Unexpected error 3\"\n      };\n    }\n\n    key = sodium.to_hex(\n      encryptCryptoBox(\n        sodium.from_hex(key),\n        this.#keys.publicKey,\n        this.#keys.privateKey\n      )\n    );\n\n    const nameKey = secretstreamKeygen();\n    const encryptedName = await encryptName(name, sodium.to_hex(nameKey));\n    const encryptedNameKey = sodium.to_hex(\n      encryptCryptoBox(nameKey, this.#keys.publicKey, this.#keys.privateKey)\n    );\n\n    const { saveInCloud } = await this.#thunder(\"mutation\")({\n      saveInCloud: [\n        {\n          fileId,\n          key,\n          nodeId,\n          filename: encryptedName,\n          nameKey: encryptedNameKey\n        },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on ErrorNotFound\": {\n            __typename: true,\n            message: true\n          },\n          \"...on ErrorLimit\": {\n            __typename: true,\n            message: true\n          },\n          \"...on ErrorBasic\": {\n            __typename: true,\n            message: true\n          },\n          \"...on ErrorNotExist\": {\n            __typename: true,\n            message: true\n          },\n          \"...on SaveInCloudResponse\": {\n            __typename: true,\n            saveInCloud: nodeFullSelector\n          }\n        }\n      ]\n    });\n\n    if (!saveInCloud) {\n      return null;\n    }\n\n    if (saveInCloud.__typename === \"ErrorAccessDenied\") {\n      return saveInCloud;\n    }\n\n    if (saveInCloud.__typename === \"ErrorNotFound\") {\n      return saveInCloud;\n    }\n    if (saveInCloud.__typename === \"ErrorLimit\") {\n      return saveInCloud;\n    }\n\n    if (saveInCloud.__typename === \"ErrorBasic\") {\n      return saveInCloud;\n    }\n\n    if (saveInCloud.__typename === \"ErrorNotExist\") {\n      return saveInCloud;\n    }\n\n    if (!saveInCloud.saveInCloud) {\n      return null;\n    }\n\n    const node = await gqlNodeToExternalNodeFull(\n      saveInCloud.saveInCloud,\n      this.#keys\n    );\n\n    const me = node.parent?.users.find(\n      ([u]) => u.publicKey === this.#keys.publicKey\n    );\n\n    if (me && [\"admin\", \"write\"].includes(me[1])) {\n      const others =\n        node.parent?.users.filter(\n          ([u]) => u.publicKey !== this.#keys.publicKey\n        ) ?? [];\n\n      await Promise.all(\n        others.map(([u, rights]) =>\n          this.shareNode({\n            nodeId: node.id,\n            rights,\n            userId: u.id\n          })\n        )\n      );\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: node\n    };\n  }\n\n  private perNode = async (\n    nodeId: string,\n    publicKey: string\n  ): Promise<NameKeyInput | null> => {\n    let node = nodesCache.get(nodeId);\n\n    if (!node) {\n      await this.node({ id: nodeId });\n      node = nodesCache.get(nodeId);\n\n      if (!node) {\n        return null;\n      }\n    }\n\n    const nameKey = node.access?.nameKey;\n\n    if (!nameKey) {\n      return null;\n    }\n\n    return {\n      id: node.id,\n      nameKey: sodium.to_hex(\n        encryptCryptoBox(\n          sodium.from_hex(nameKey),\n          publicKey,\n          this.#keys.privateKey\n        )\n      ),\n      files:\n        \"history\" in node\n          ? node.history.map(f => ({\n              id: f.id,\n              key: sodium.to_hex(\n                encryptCryptoBox(\n                  sodium.from_hex(f.key),\n                  publicKey,\n                  this.#keys.privateKey\n                )\n              )\n            }))\n          : []\n    };\n  };\n}\n"],"mappings":"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"}