@secrecy/lib 1.0.0-dev.9 → 1.0.0
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/README.md +1 -1
- package/{lib → dist}/BaseClient.d.ts +62 -69
- package/dist/BaseClient.js +1124 -0
- package/{lib → dist}/PopupTools.d.ts +1 -1
- package/dist/PopupTools.js +212 -0
- package/{lib → dist}/ZeusThunder.d.ts +0 -0
- package/dist/ZeusThunder.js +79 -0
- package/dist/cache.d.ts +6 -0
- package/dist/cache.js +4 -0
- package/dist/client/admin/index.d.ts +11 -0
- package/dist/client/admin/index.js +110 -0
- package/{lib → dist}/client/convert/file.d.ts +3 -3
- package/dist/client/convert/file.js +34 -0
- package/{lib → dist}/client/convert/mail.d.ts +0 -0
- package/dist/client/convert/mail.js +46 -0
- package/dist/client/convert/node.d.ts +9 -0
- package/dist/client/convert/node.js +101 -0
- package/{lib → dist}/client/helpers.d.ts +1 -1
- package/dist/client/helpers.js +114 -0
- package/dist/client/index.d.ts +164 -0
- package/dist/client/index.js +3206 -0
- package/{lib → dist}/client/storage.d.ts +0 -0
- package/dist/client/storage.js +12 -0
- package/dist/client/types/File.d.ts +14 -0
- package/dist/client/types/File.js +3 -0
- package/{lib → dist}/client/types/Inputs.d.ts +2 -7
- package/dist/client/types/Inputs.js +3 -0
- package/dist/client/types/Node.d.ts +49 -0
- package/dist/client/types/Node.js +3 -0
- package/{lib → dist}/client/types/UserAppNotifications.d.ts +1 -1
- package/dist/client/types/UserAppNotifications.js +3 -0
- package/{lib → dist}/client/types/UserAppSettings.d.ts +0 -1
- package/dist/client/types/UserAppSettings.js +3 -0
- package/{lib → dist}/client/types/index.d.ts +7 -13
- package/dist/client/types/index.js +8 -0
- package/dist/client/types/selectors.d.ts +476 -0
- package/dist/client/types/selectors.js +125 -0
- package/{lib → dist}/crypto/file.d.ts +0 -0
- package/dist/crypto/file.js +210 -0
- package/{lib → dist}/crypto/index.d.ts +0 -0
- package/dist/crypto/index.js +47 -0
- package/dist/error.d.ts +30 -0
- package/dist/error.js +3 -0
- package/{lib → dist}/index.d.ts +3 -3
- package/dist/index.js +10 -0
- package/{lib → dist}/minify/index.d.ts +0 -0
- package/dist/minify/index.js +28 -0
- package/{lib → dist}/minify/lz4.d.ts +0 -0
- package/dist/minify/lz4.js +627 -0
- package/{lib → dist}/sodium.d.ts +0 -0
- package/dist/sodium.js +6 -0
- package/{lib → dist}/utils/store-buddy.d.ts +0 -0
- package/dist/utils/store-buddy.js +65 -0
- package/{lib → dist}/utils/time.d.ts +0 -0
- package/dist/utils/time.js +14 -0
- package/{lib → dist/utils}/utils.d.ts +0 -0
- package/dist/utils/utils.js +57 -0
- package/{lib → dist}/worker/__mock__/sodium.worker.d.ts +0 -0
- package/dist/worker/__mock__/sodium.worker.js +49 -0
- package/{lib → dist}/worker/md5.d.ts +0 -0
- package/dist/worker/md5.js +25 -0
- package/{lib → dist}/worker/sodium.d.ts +0 -0
- package/dist/worker/sodium.js +120 -0
- package/{lib → dist}/worker/workerCodes.d.ts +0 -0
- package/dist/worker/workerCodes.js +3 -0
- package/{lib → dist}/zeus/const.d.ts +5 -0
- package/dist/zeus/const.js +1230 -0
- package/dist/zeus/index.d.ts +4069 -0
- package/dist/zeus/index.js +657 -0
- package/package.json +64 -57
- package/lib/BaseClient.js +0 -1332
- package/lib/PopupTools.js +0 -213
- package/lib/ZeusThunder.js +0 -115
- package/lib/cache.d.ts +0 -7
- package/lib/cache.js +0 -5
- package/lib/client/admin/index.d.ts +0 -10
- package/lib/client/admin/index.js +0 -145
- package/lib/client/convert/file.js +0 -39
- package/lib/client/convert/folder.d.ts +0 -8
- package/lib/client/convert/folder.js +0 -264
- package/lib/client/convert/mail.js +0 -46
- package/lib/client/convert/vFile.d.ts +0 -5
- package/lib/client/convert/vFile.js +0 -164
- package/lib/client/helpers.js +0 -116
- package/lib/client/index.d.ts +0 -169
- package/lib/client/index.js +0 -3803
- package/lib/client/storage.js +0 -12
- package/lib/client/types/File.d.ts +0 -21
- package/lib/client/types/File.js +0 -2
- package/lib/client/types/FilesOnUsersOnApplications.d.ts +0 -9
- package/lib/client/types/FilesOnUsersOnApplications.js +0 -2
- package/lib/client/types/Folder.d.ts +0 -68
- package/lib/client/types/Folder.js +0 -7
- package/lib/client/types/Inputs.js +0 -2
- package/lib/client/types/UserAppNotifications.js +0 -2
- package/lib/client/types/UserAppSettings.js +0 -2
- package/lib/client/types/VFile.d.ts +0 -62
- package/lib/client/types/VFile.js +0 -4
- package/lib/client/types/index.js +0 -9
- package/lib/client/types/queries.d.ts +0 -535
- package/lib/client/types/queries.js +0 -192
- package/lib/crypto/file.js +0 -291
- package/lib/crypto/index.js +0 -37
- package/lib/index.js +0 -41
- package/lib/minify/index.js +0 -28
- package/lib/minify/lz4.js +0 -633
- package/lib/sodium.js +0 -28
- package/lib/utils/store-buddy.js +0 -69
- package/lib/utils/time.js +0 -22
- package/lib/utils.js +0 -188
- package/lib/worker/__mock__/sodium.worker.js +0 -57
- package/lib/worker/md5.js +0 -43
- package/lib/worker/sodium.js +0 -155
- package/lib/worker/workerCodes.js +0 -3
- package/lib/zeus/const.js +0 -1671
- package/lib/zeus/index.d.ts +0 -33302
- package/lib/zeus/index.js +0 -558
|
@@ -0,0 +1,3206 @@
|
|
|
1
|
+
import _defineProperty from "@babel/runtime/helpers/defineProperty";
|
|
2
|
+
import _classPrivateFieldLooseBase from "@babel/runtime/helpers/classPrivateFieldLooseBase";
|
|
3
|
+
import _classPrivateFieldLooseKey from "@babel/runtime/helpers/classPrivateFieldLooseKey";
|
|
4
|
+
|
|
5
|
+
function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); enumerableOnly && (symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; })), keys.push.apply(keys, symbols); } return keys; }
|
|
6
|
+
|
|
7
|
+
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = null != arguments[i] ? arguments[i] : {}; i % 2 ? ownKeys(Object(source), !0).forEach(function (key) { _defineProperty(target, key, source[key]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)) : ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } return target; }
|
|
8
|
+
|
|
9
|
+
/* eslint-disable @typescript-eslint/naming-convention */
|
|
10
|
+
import ky from "ky";
|
|
11
|
+
import axios from "axios";
|
|
12
|
+
import { BaseClient } from "../BaseClient.js";
|
|
13
|
+
import { nodesCache, filesCache, usersCache } from "../cache.js";
|
|
14
|
+
import { secretstreamKeygen, encryptSecretstream } from "../crypto/file.js";
|
|
15
|
+
import { compress, uncompress } from "../minify/index.js";
|
|
16
|
+
import { gqlNodeToExternal, gqlNodeToExternalNodeFull, gqlNodeToInternal, internalNodeToNode } from "./convert/node.js";
|
|
17
|
+
import { gqlFileToExternal } from "./convert/file.js";
|
|
18
|
+
import { sodium } from "../sodium.js";
|
|
19
|
+
import { decryptCryptoBox, encryptCryptoBox } from "../crypto/index.js";
|
|
20
|
+
import { generate } from "shortid";
|
|
21
|
+
import { md5 } from "../worker/md5.js";
|
|
22
|
+
import { encrypt, decrypt } from "../worker/sodium.js";
|
|
23
|
+
import { chunks, concatenate, enumerate, promiseAllLimit } from "../utils/utils.js";
|
|
24
|
+
import { fileSelector, mailSelector, nodeFullSelector, nodeSelector } from "./types/selectors.js";
|
|
25
|
+
import { convertInternalMailToExternal } from "./convert/mail.js";
|
|
26
|
+
import { decode } from "jsonwebtoken"; // import { md5 } from "../worker/index.js";
|
|
27
|
+
// import { firstValueFrom, of } from "rxjs";
|
|
28
|
+
|
|
29
|
+
import { serialize } from "bson";
|
|
30
|
+
|
|
31
|
+
const encryptName = async (name, nameKey) => {
|
|
32
|
+
const {
|
|
33
|
+
data
|
|
34
|
+
} = await encryptSecretstream(sodium.from_hex(nameKey), sodium.from_string(name));
|
|
35
|
+
const nameEncrypted = sodium.to_hex(data);
|
|
36
|
+
return nameEncrypted;
|
|
37
|
+
};
|
|
38
|
+
|
|
39
|
+
var _keys = /*#__PURE__*/_classPrivateFieldLooseKey("keys");
|
|
40
|
+
|
|
41
|
+
export class SecrecyClient extends BaseClient {
|
|
42
|
+
constructor(uaSession, uaKeys, uaJwt, env) {
|
|
43
|
+
super(uaSession, env);
|
|
44
|
+
Object.defineProperty(this, _keys, {
|
|
45
|
+
writable: true,
|
|
46
|
+
value: void 0
|
|
47
|
+
});
|
|
48
|
+
|
|
49
|
+
this.perNode = async (nodeId, publicKey) => {
|
|
50
|
+
var _node$access;
|
|
51
|
+
|
|
52
|
+
let node = nodesCache.get(nodeId);
|
|
53
|
+
|
|
54
|
+
if (!node) {
|
|
55
|
+
await this.node({
|
|
56
|
+
id: nodeId
|
|
57
|
+
});
|
|
58
|
+
node = nodesCache.get(nodeId);
|
|
59
|
+
|
|
60
|
+
if (!node) {
|
|
61
|
+
return null;
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
const nameKey = (_node$access = node.access) == null ? void 0 : _node$access.nameKey;
|
|
66
|
+
|
|
67
|
+
if (!nameKey) {
|
|
68
|
+
return null;
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
return {
|
|
72
|
+
id: node.id,
|
|
73
|
+
nameKey: sodium.to_hex(encryptCryptoBox(sodium.from_hex(nameKey), publicKey, _classPrivateFieldLooseBase(this, _keys)[_keys].privateKey)),
|
|
74
|
+
files: node.history.map(f => ({
|
|
75
|
+
id: f.id,
|
|
76
|
+
key: sodium.to_hex(encryptCryptoBox(sodium.from_hex(f.key), publicKey, _classPrivateFieldLooseBase(this, _keys)[_keys].privateKey))
|
|
77
|
+
}))
|
|
78
|
+
};
|
|
79
|
+
};
|
|
80
|
+
|
|
81
|
+
this._eachUser = async (files, subject, body, idOrMail) => {
|
|
82
|
+
let u = usersCache.get(idOrMail);
|
|
83
|
+
|
|
84
|
+
if (!u || !("publicKey" in u)) {
|
|
85
|
+
try {
|
|
86
|
+
const req = await this.user({
|
|
87
|
+
userId: idOrMail,
|
|
88
|
+
withPublicKey: true
|
|
89
|
+
});
|
|
90
|
+
|
|
91
|
+
if (!req) {
|
|
92
|
+
return null;
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
if (req.__typename !== "SuccessResponse") {
|
|
96
|
+
return null;
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
u = req.data;
|
|
100
|
+
} catch {
|
|
101
|
+
return null;
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
if (!u) {
|
|
105
|
+
return null;
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
if (!("publicKey" in u)) {
|
|
110
|
+
throw new Error("User " + idOrMail + " have no public key");
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
const recipientsFiles = new Array();
|
|
114
|
+
|
|
115
|
+
for (const f of files) {
|
|
116
|
+
let fileInHistory = filesCache.get(f.id);
|
|
117
|
+
|
|
118
|
+
if (!fileInHistory) {
|
|
119
|
+
await this.file({
|
|
120
|
+
id: f.id
|
|
121
|
+
});
|
|
122
|
+
fileInHistory = filesCache.get(f.id);
|
|
123
|
+
|
|
124
|
+
if (!fileInHistory) {
|
|
125
|
+
throw new Error("File " + f.name + " (" + f.id + ") does not exists");
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
const key = fileInHistory.key;
|
|
130
|
+
recipientsFiles.push({
|
|
131
|
+
id: f.id,
|
|
132
|
+
name: sodium.to_hex(encryptCryptoBox(sodium.from_string(f.name), u.publicKey, _classPrivateFieldLooseBase(this, _keys)[_keys].privateKey)),
|
|
133
|
+
fileKey: sodium.to_hex(encryptCryptoBox(sodium.from_hex(key), u.publicKey, _classPrivateFieldLooseBase(this, _keys)[_keys].privateKey))
|
|
134
|
+
});
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
return {
|
|
138
|
+
recipientId: u.id,
|
|
139
|
+
body: sodium.to_hex(encryptCryptoBox(sodium.from_string(body), u.publicKey, _classPrivateFieldLooseBase(this, _keys)[_keys].privateKey)),
|
|
140
|
+
subject: sodium.to_hex(encryptCryptoBox(sodium.from_string(subject), u.publicKey, _classPrivateFieldLooseBase(this, _keys)[_keys].privateKey)),
|
|
141
|
+
files: recipientsFiles
|
|
142
|
+
};
|
|
143
|
+
};
|
|
144
|
+
|
|
145
|
+
this.jwt = uaJwt;
|
|
146
|
+
this.jwtDecoded = decode(uaJwt);
|
|
147
|
+
_classPrivateFieldLooseBase(this, _keys)[_keys] = uaKeys;
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
get publicKey() {
|
|
151
|
+
return _classPrivateFieldLooseBase(this, _keys)[_keys].publicKey;
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
get appUserId() {
|
|
155
|
+
var _this$jwtDecoded$sub;
|
|
156
|
+
|
|
157
|
+
return (_this$jwtDecoded$sub = this.jwtDecoded.sub) != null ? _this$jwtDecoded$sub : "";
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
async addFileToHistory(_ref) {
|
|
161
|
+
let {
|
|
162
|
+
fileId,
|
|
163
|
+
nodeId
|
|
164
|
+
} = _ref;
|
|
165
|
+
const {
|
|
166
|
+
addFileToHistory
|
|
167
|
+
} = await this.client("mutation")({
|
|
168
|
+
addFileToHistory: [{
|
|
169
|
+
fileId,
|
|
170
|
+
nodeId
|
|
171
|
+
}, {
|
|
172
|
+
"...on ErrorAccessDenied": {
|
|
173
|
+
__typename: true,
|
|
174
|
+
message: true
|
|
175
|
+
},
|
|
176
|
+
"...on ErrorNotExist": {
|
|
177
|
+
__typename: true,
|
|
178
|
+
message: true
|
|
179
|
+
},
|
|
180
|
+
"...on AddFileToHistoryResponse": {
|
|
181
|
+
__typename: true,
|
|
182
|
+
addFileToHistory: nodeSelector
|
|
183
|
+
}
|
|
184
|
+
}]
|
|
185
|
+
});
|
|
186
|
+
|
|
187
|
+
if (!addFileToHistory) {
|
|
188
|
+
return null;
|
|
189
|
+
}
|
|
190
|
+
|
|
191
|
+
if (addFileToHistory.__typename === "ErrorAccessDenied") {
|
|
192
|
+
return addFileToHistory;
|
|
193
|
+
}
|
|
194
|
+
|
|
195
|
+
if (addFileToHistory.__typename === "ErrorNotExist") {
|
|
196
|
+
return addFileToHistory;
|
|
197
|
+
}
|
|
198
|
+
|
|
199
|
+
const node = await gqlNodeToInternal(addFileToHistory.addFileToHistory, _classPrivateFieldLooseBase(this, _keys)[_keys]);
|
|
200
|
+
const file = node.history.find(f => f.id === fileId);
|
|
201
|
+
|
|
202
|
+
if (file) {
|
|
203
|
+
const users = node.users.filter(_ref2 => {
|
|
204
|
+
let [u] = _ref2;
|
|
205
|
+
return u.publicKey !== _classPrivateFieldLooseBase(this, _keys)[_keys].publicKey;
|
|
206
|
+
});
|
|
207
|
+
const input = {
|
|
208
|
+
fileId: file.id,
|
|
209
|
+
users: users.map(_ref3 => {
|
|
210
|
+
let [u] = _ref3;
|
|
211
|
+
return {
|
|
212
|
+
id: u.id,
|
|
213
|
+
key: sodium.to_hex(encryptCryptoBox(sodium.from_hex(file.key), _classPrivateFieldLooseBase(this, _keys)[_keys].publicKey, _classPrivateFieldLooseBase(this, _keys)[_keys].privateKey))
|
|
214
|
+
};
|
|
215
|
+
})
|
|
216
|
+
};
|
|
217
|
+
await this.client("mutation")({
|
|
218
|
+
shareFileInHistory: [{
|
|
219
|
+
input,
|
|
220
|
+
nodeId
|
|
221
|
+
}, {
|
|
222
|
+
"...on ErrorAccessDenied": {
|
|
223
|
+
__typename: true,
|
|
224
|
+
message: true
|
|
225
|
+
},
|
|
226
|
+
"...on ErrorNotFound": {
|
|
227
|
+
__typename: true,
|
|
228
|
+
message: true
|
|
229
|
+
},
|
|
230
|
+
"...on ShareFileInHistoryResponse": {
|
|
231
|
+
__typename: true,
|
|
232
|
+
shareFileInHistory: true
|
|
233
|
+
}
|
|
234
|
+
}]
|
|
235
|
+
});
|
|
236
|
+
}
|
|
237
|
+
|
|
238
|
+
const result = internalNodeToNode(node);
|
|
239
|
+
return {
|
|
240
|
+
__typename: "SuccessResponse",
|
|
241
|
+
data: result
|
|
242
|
+
};
|
|
243
|
+
}
|
|
244
|
+
|
|
245
|
+
async uploadFile(_ref4) {
|
|
246
|
+
let {
|
|
247
|
+
file,
|
|
248
|
+
encryptProgress,
|
|
249
|
+
uploadProgress,
|
|
250
|
+
signal
|
|
251
|
+
} = _ref4;
|
|
252
|
+
const fileKey = secretstreamKeygen();
|
|
253
|
+
const fileBuffer = file instanceof File ? new Uint8Array(await file.arrayBuffer()) : file;
|
|
254
|
+
const compressed = compress(fileBuffer);
|
|
255
|
+
const {
|
|
256
|
+
data: encryptedFile,
|
|
257
|
+
md5: md5File,
|
|
258
|
+
md5Encrypted
|
|
259
|
+
} = await encrypt(fileKey, compressed, encryptProgress, signal);
|
|
260
|
+
const encryptedFileKey = encryptCryptoBox(fileKey, _classPrivateFieldLooseBase(this, _keys)[_keys].publicKey, _classPrivateFieldLooseBase(this, _keys)[_keys].privateKey);
|
|
261
|
+
const {
|
|
262
|
+
uploadFile
|
|
263
|
+
} = await this.client("mutation", {
|
|
264
|
+
scalars: {
|
|
265
|
+
Json: {
|
|
266
|
+
encode: e => JSON.stringify(e),
|
|
267
|
+
decode: e => JSON.parse(e)
|
|
268
|
+
},
|
|
269
|
+
DateTime: {
|
|
270
|
+
decode: e => new Date(e),
|
|
271
|
+
encode: e => e.toISOString()
|
|
272
|
+
},
|
|
273
|
+
BigInt: {
|
|
274
|
+
decode: e => BigInt(e),
|
|
275
|
+
encode: e => e.toString()
|
|
276
|
+
}
|
|
277
|
+
}
|
|
278
|
+
})({
|
|
279
|
+
uploadFile: [{
|
|
280
|
+
fileSize: encryptedFile.byteLength,
|
|
281
|
+
fileSizeBefore: fileBuffer.byteLength,
|
|
282
|
+
fileKey: sodium.to_hex(encryptedFileKey),
|
|
283
|
+
md5Encrypted,
|
|
284
|
+
md5: md5File
|
|
285
|
+
}, {
|
|
286
|
+
"...on ErrorAccessDenied": {
|
|
287
|
+
__typename: true,
|
|
288
|
+
message: true
|
|
289
|
+
},
|
|
290
|
+
"...on ErrorLimit": {
|
|
291
|
+
__typename: true,
|
|
292
|
+
message: true
|
|
293
|
+
},
|
|
294
|
+
"...on ErrorNotFound": {
|
|
295
|
+
__typename: true,
|
|
296
|
+
message: true
|
|
297
|
+
},
|
|
298
|
+
"...on UploadFileResponse": {
|
|
299
|
+
__typename: true,
|
|
300
|
+
uploadFile: {
|
|
301
|
+
fileId: true,
|
|
302
|
+
filePartSize: true,
|
|
303
|
+
parts: {
|
|
304
|
+
fields: true,
|
|
305
|
+
order: true,
|
|
306
|
+
url: true
|
|
307
|
+
}
|
|
308
|
+
}
|
|
309
|
+
}
|
|
310
|
+
}]
|
|
311
|
+
});
|
|
312
|
+
|
|
313
|
+
if (!uploadFile) {
|
|
314
|
+
return null;
|
|
315
|
+
}
|
|
316
|
+
|
|
317
|
+
if (uploadFile.__typename === "ErrorAccessDenied") {
|
|
318
|
+
return uploadFile;
|
|
319
|
+
}
|
|
320
|
+
|
|
321
|
+
if (uploadFile.__typename === "ErrorLimit") {
|
|
322
|
+
return uploadFile;
|
|
323
|
+
}
|
|
324
|
+
|
|
325
|
+
if (uploadFile.__typename === "ErrorNotFound") {
|
|
326
|
+
return uploadFile;
|
|
327
|
+
}
|
|
328
|
+
|
|
329
|
+
if (!uploadFile.uploadFile) {
|
|
330
|
+
return null;
|
|
331
|
+
}
|
|
332
|
+
|
|
333
|
+
uploadProgress == null ? void 0 : uploadProgress({
|
|
334
|
+
total: encryptedFile.byteLength,
|
|
335
|
+
current: 0,
|
|
336
|
+
percent: 0
|
|
337
|
+
});
|
|
338
|
+
|
|
339
|
+
if (uploadFile.uploadFile.parts.length === 0) {
|
|
340
|
+
uploadProgress == null ? void 0 : uploadProgress({
|
|
341
|
+
total: encryptedFile.byteLength,
|
|
342
|
+
current: encryptedFile.byteLength,
|
|
343
|
+
percent: 1
|
|
344
|
+
});
|
|
345
|
+
return {
|
|
346
|
+
__typename: "SuccessResponse",
|
|
347
|
+
data: uploadFile.uploadFile.fileId
|
|
348
|
+
};
|
|
349
|
+
}
|
|
350
|
+
|
|
351
|
+
const uploadPartEnded = async (md5, order) => {
|
|
352
|
+
if (!uploadFile.uploadFile) {
|
|
353
|
+
return null;
|
|
354
|
+
}
|
|
355
|
+
|
|
356
|
+
const {
|
|
357
|
+
uploadFilePartEnd
|
|
358
|
+
} = await this.client("mutation")({
|
|
359
|
+
uploadFilePartEnd: [{
|
|
360
|
+
fileId: uploadFile.uploadFile.fileId,
|
|
361
|
+
md5,
|
|
362
|
+
order
|
|
363
|
+
}, {
|
|
364
|
+
"...on ErrorAccessDenied": {
|
|
365
|
+
__typename: true,
|
|
366
|
+
message: true
|
|
367
|
+
},
|
|
368
|
+
"...on UploadFilePartEndResponse": {
|
|
369
|
+
__typename: true,
|
|
370
|
+
uploadFilePartEnd: true
|
|
371
|
+
}
|
|
372
|
+
}]
|
|
373
|
+
});
|
|
374
|
+
|
|
375
|
+
if (!uploadFilePartEnd) {
|
|
376
|
+
return null;
|
|
377
|
+
}
|
|
378
|
+
|
|
379
|
+
if (uploadFilePartEnd.__typename === "ErrorAccessDenied") {
|
|
380
|
+
return uploadFilePartEnd;
|
|
381
|
+
}
|
|
382
|
+
|
|
383
|
+
return {
|
|
384
|
+
__typename: "SuccessResponse",
|
|
385
|
+
data: uploadFilePartEnd.uploadFilePartEnd
|
|
386
|
+
};
|
|
387
|
+
};
|
|
388
|
+
|
|
389
|
+
const uploadEnded = async () => {
|
|
390
|
+
if (!uploadFile.uploadFile) {
|
|
391
|
+
return null;
|
|
392
|
+
}
|
|
393
|
+
|
|
394
|
+
const {
|
|
395
|
+
uploadFileEnd
|
|
396
|
+
} = await this.client("mutation")({
|
|
397
|
+
uploadFileEnd: [{
|
|
398
|
+
fileId: uploadFile.uploadFile.fileId
|
|
399
|
+
}, {
|
|
400
|
+
"...on ErrorAccessDenied": {
|
|
401
|
+
__typename: true,
|
|
402
|
+
message: true
|
|
403
|
+
},
|
|
404
|
+
"...on ErrorNotFound": {
|
|
405
|
+
__typename: true,
|
|
406
|
+
message: true
|
|
407
|
+
},
|
|
408
|
+
"...on UploadFileEndResponse": {
|
|
409
|
+
__typename: true,
|
|
410
|
+
uploadFileEnd: true
|
|
411
|
+
}
|
|
412
|
+
}]
|
|
413
|
+
});
|
|
414
|
+
|
|
415
|
+
if (!uploadFileEnd) {
|
|
416
|
+
return null;
|
|
417
|
+
}
|
|
418
|
+
|
|
419
|
+
if (uploadFileEnd.__typename === "ErrorAccessDenied") {
|
|
420
|
+
return uploadFileEnd;
|
|
421
|
+
}
|
|
422
|
+
|
|
423
|
+
if (uploadFileEnd.__typename === "ErrorNotFound") {
|
|
424
|
+
return uploadFileEnd;
|
|
425
|
+
}
|
|
426
|
+
|
|
427
|
+
if (!uploadFileEnd.uploadFileEnd) {
|
|
428
|
+
return null;
|
|
429
|
+
}
|
|
430
|
+
|
|
431
|
+
return {
|
|
432
|
+
__typename: "SuccessResponse",
|
|
433
|
+
data: uploadFileEnd.uploadFileEnd
|
|
434
|
+
};
|
|
435
|
+
};
|
|
436
|
+
|
|
437
|
+
const chunkParts = new Array();
|
|
438
|
+
|
|
439
|
+
for (const [index, chunk] of enumerate(chunks(encryptedFile, Number(uploadFile.uploadFile.filePartSize)))) {
|
|
440
|
+
chunkParts.push({
|
|
441
|
+
order: index + 1,
|
|
442
|
+
data: chunk,
|
|
443
|
+
md5: await md5(chunk)
|
|
444
|
+
});
|
|
445
|
+
}
|
|
446
|
+
|
|
447
|
+
const progressParts = {};
|
|
448
|
+
|
|
449
|
+
const onProgress = (part, progressEvent) => {
|
|
450
|
+
progressParts[part] = progressEvent;
|
|
451
|
+
const current = Object.values(progressParts).reduce((prv, cur) => prv + cur.loaded, 0);
|
|
452
|
+
uploadProgress == null ? void 0 : uploadProgress({
|
|
453
|
+
percent: current / encryptedFile.byteLength,
|
|
454
|
+
total: encryptedFile.byteLength,
|
|
455
|
+
current
|
|
456
|
+
});
|
|
457
|
+
};
|
|
458
|
+
|
|
459
|
+
const byPart = async part => {
|
|
460
|
+
if (!uploadFile.uploadFile) {
|
|
461
|
+
return;
|
|
462
|
+
}
|
|
463
|
+
|
|
464
|
+
const formData = new FormData();
|
|
465
|
+
const chunk = chunkParts.find(p => p.order === part.order);
|
|
466
|
+
|
|
467
|
+
if (!chunk) {
|
|
468
|
+
return;
|
|
469
|
+
}
|
|
470
|
+
|
|
471
|
+
for (const [key, value] of Object.entries(part.fields)) {
|
|
472
|
+
formData.append(key, value);
|
|
473
|
+
}
|
|
474
|
+
|
|
475
|
+
formData.append("file", new Blob([chunk.data]), uploadFile.uploadFile.fileId + "-" + chunk.order);
|
|
476
|
+
await axios.post(part.url, formData, {
|
|
477
|
+
onUploadProgress: progressEvent => onProgress(part.order, progressEvent),
|
|
478
|
+
signal
|
|
479
|
+
});
|
|
480
|
+
await uploadPartEnded(chunk.md5, chunk.order); // if ((e as any).response.status === 0) {
|
|
481
|
+
// // TODO https://github.com/sindresorhus/ky/issues/305
|
|
482
|
+
// } else {
|
|
483
|
+
// throw e;
|
|
484
|
+
// }
|
|
485
|
+
};
|
|
486
|
+
|
|
487
|
+
if (!uploadFile.uploadFile) {
|
|
488
|
+
return null;
|
|
489
|
+
}
|
|
490
|
+
|
|
491
|
+
await promiseAllLimit(3, uploadFile.uploadFile.parts.map(p => () => byPart(p)));
|
|
492
|
+
const result = await uploadEnded();
|
|
493
|
+
|
|
494
|
+
if (!result) {
|
|
495
|
+
return null;
|
|
496
|
+
}
|
|
497
|
+
|
|
498
|
+
if (result.__typename === "ErrorAccessDenied") {
|
|
499
|
+
return result;
|
|
500
|
+
}
|
|
501
|
+
|
|
502
|
+
if (result.__typename === "ErrorNotFound") {
|
|
503
|
+
return result;
|
|
504
|
+
}
|
|
505
|
+
|
|
506
|
+
return {
|
|
507
|
+
__typename: "SuccessResponse",
|
|
508
|
+
data: result.data
|
|
509
|
+
};
|
|
510
|
+
}
|
|
511
|
+
|
|
512
|
+
async uploadFileInCloud(_ref5) {
|
|
513
|
+
let {
|
|
514
|
+
file,
|
|
515
|
+
name,
|
|
516
|
+
nodeId,
|
|
517
|
+
encryptProgress,
|
|
518
|
+
uploadProgress,
|
|
519
|
+
signal
|
|
520
|
+
} = _ref5;
|
|
521
|
+
const fileId = await this.uploadFile({
|
|
522
|
+
file,
|
|
523
|
+
encryptProgress,
|
|
524
|
+
uploadProgress,
|
|
525
|
+
signal
|
|
526
|
+
});
|
|
527
|
+
|
|
528
|
+
if (!fileId) {
|
|
529
|
+
return null;
|
|
530
|
+
}
|
|
531
|
+
|
|
532
|
+
if (fileId.__typename === "ErrorAccessDenied") {
|
|
533
|
+
return fileId;
|
|
534
|
+
}
|
|
535
|
+
|
|
536
|
+
if (fileId.__typename === "ErrorLimit") {
|
|
537
|
+
return fileId;
|
|
538
|
+
}
|
|
539
|
+
|
|
540
|
+
if (fileId.__typename === "ErrorNotFound") {
|
|
541
|
+
return fileId;
|
|
542
|
+
}
|
|
543
|
+
|
|
544
|
+
const result = await this.saveInCloud({
|
|
545
|
+
fileId: fileId.data,
|
|
546
|
+
name,
|
|
547
|
+
nodeId
|
|
548
|
+
});
|
|
549
|
+
|
|
550
|
+
if (!result) {
|
|
551
|
+
return null;
|
|
552
|
+
}
|
|
553
|
+
|
|
554
|
+
if (result.__typename === "ErrorAccessDenied") {
|
|
555
|
+
return result;
|
|
556
|
+
}
|
|
557
|
+
|
|
558
|
+
if (result.__typename === "ErrorBasic") {
|
|
559
|
+
return result;
|
|
560
|
+
}
|
|
561
|
+
|
|
562
|
+
if (result.__typename === "ErrorLimit") {
|
|
563
|
+
return result;
|
|
564
|
+
}
|
|
565
|
+
|
|
566
|
+
if (result.__typename === "ErrorNotFound") {
|
|
567
|
+
return result;
|
|
568
|
+
}
|
|
569
|
+
|
|
570
|
+
if (result.__typename === "ErrorNotExist") {
|
|
571
|
+
return result;
|
|
572
|
+
}
|
|
573
|
+
|
|
574
|
+
return {
|
|
575
|
+
__typename: "SuccessResponse",
|
|
576
|
+
data: result.data
|
|
577
|
+
};
|
|
578
|
+
}
|
|
579
|
+
|
|
580
|
+
async logout(sessionId) {
|
|
581
|
+
nodesCache.clear();
|
|
582
|
+
filesCache.clear();
|
|
583
|
+
await super.logout(sessionId);
|
|
584
|
+
}
|
|
585
|
+
|
|
586
|
+
async createFolder(_ref6) {
|
|
587
|
+
var _folder$parent$users$, _folder$parent, _folder$parent$users;
|
|
588
|
+
|
|
589
|
+
let {
|
|
590
|
+
name,
|
|
591
|
+
parentFolderId
|
|
592
|
+
} = _ref6;
|
|
593
|
+
const key = secretstreamKeygen();
|
|
594
|
+
const encryptedName = await encryptName(name, sodium.to_hex(key));
|
|
595
|
+
const encryptedKey = encryptCryptoBox(key, _classPrivateFieldLooseBase(this, _keys)[_keys].publicKey, _classPrivateFieldLooseBase(this, _keys)[_keys].privateKey);
|
|
596
|
+
const {
|
|
597
|
+
createFolder
|
|
598
|
+
} = await this.client("mutation")({
|
|
599
|
+
createFolder: [{
|
|
600
|
+
name: encryptedName,
|
|
601
|
+
parentFolderId,
|
|
602
|
+
key: sodium.to_hex(encryptedKey)
|
|
603
|
+
}, {
|
|
604
|
+
"...on ErrorAccessDenied": {
|
|
605
|
+
__typename: true,
|
|
606
|
+
message: true
|
|
607
|
+
},
|
|
608
|
+
"...on ErrorNotExist": {
|
|
609
|
+
__typename: true,
|
|
610
|
+
message: true
|
|
611
|
+
},
|
|
612
|
+
"...on CreateFolderResponse": {
|
|
613
|
+
__typename: true,
|
|
614
|
+
createFolder: nodeFullSelector
|
|
615
|
+
}
|
|
616
|
+
}]
|
|
617
|
+
});
|
|
618
|
+
|
|
619
|
+
if (!createFolder) {
|
|
620
|
+
//throw new Error(`Can't create folder`);
|
|
621
|
+
return null;
|
|
622
|
+
}
|
|
623
|
+
|
|
624
|
+
if (createFolder.__typename === "ErrorAccessDenied") {
|
|
625
|
+
return createFolder;
|
|
626
|
+
}
|
|
627
|
+
|
|
628
|
+
if (createFolder.__typename === "ErrorNotExist") {
|
|
629
|
+
return createFolder;
|
|
630
|
+
}
|
|
631
|
+
|
|
632
|
+
if (!createFolder.createFolder) {
|
|
633
|
+
return null;
|
|
634
|
+
}
|
|
635
|
+
|
|
636
|
+
const folder = await gqlNodeToExternalNodeFull(createFolder.createFolder, _classPrivateFieldLooseBase(this, _keys)[_keys]);
|
|
637
|
+
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(_ref7 => {
|
|
638
|
+
let [u] = _ref7;
|
|
639
|
+
return u.publicKey !== _classPrivateFieldLooseBase(this, _keys)[_keys].publicKey;
|
|
640
|
+
})) != null ? _folder$parent$users$ : [];
|
|
641
|
+
|
|
642
|
+
if (users.length) {
|
|
643
|
+
await Promise.all(users.map(_ref8 => {
|
|
644
|
+
let [u, rights] = _ref8;
|
|
645
|
+
return this.shareNode({
|
|
646
|
+
nodeId: folder.id,
|
|
647
|
+
rights,
|
|
648
|
+
userId: u.id
|
|
649
|
+
});
|
|
650
|
+
}));
|
|
651
|
+
}
|
|
652
|
+
|
|
653
|
+
return {
|
|
654
|
+
__typename: "SuccessResponse",
|
|
655
|
+
data: folder
|
|
656
|
+
};
|
|
657
|
+
}
|
|
658
|
+
|
|
659
|
+
async node(_temp) {
|
|
660
|
+
let {
|
|
661
|
+
id,
|
|
662
|
+
deleted
|
|
663
|
+
} = _temp === void 0 ? {} : _temp;
|
|
664
|
+
const {
|
|
665
|
+
node
|
|
666
|
+
} = await this.client("query")({
|
|
667
|
+
node: [{
|
|
668
|
+
deleted,
|
|
669
|
+
id
|
|
670
|
+
}, {
|
|
671
|
+
"...on ErrorAccessDenied": {
|
|
672
|
+
__typename: true,
|
|
673
|
+
message: true
|
|
674
|
+
},
|
|
675
|
+
"...on NodeResponse": {
|
|
676
|
+
__typename: true,
|
|
677
|
+
node: nodeFullSelector
|
|
678
|
+
}
|
|
679
|
+
}]
|
|
680
|
+
});
|
|
681
|
+
|
|
682
|
+
if (!node) {
|
|
683
|
+
return null;
|
|
684
|
+
}
|
|
685
|
+
|
|
686
|
+
if (node.__typename === "ErrorAccessDenied") {
|
|
687
|
+
return node;
|
|
688
|
+
}
|
|
689
|
+
|
|
690
|
+
if (!node.node) {
|
|
691
|
+
return null;
|
|
692
|
+
}
|
|
693
|
+
|
|
694
|
+
const result = await gqlNodeToExternalNodeFull(node.node, _classPrivateFieldLooseBase(this, _keys)[_keys]);
|
|
695
|
+
return {
|
|
696
|
+
__typename: "SuccessResponse",
|
|
697
|
+
data: result
|
|
698
|
+
};
|
|
699
|
+
}
|
|
700
|
+
|
|
701
|
+
async file(_ref9) {
|
|
702
|
+
let {
|
|
703
|
+
id
|
|
704
|
+
} = _ref9;
|
|
705
|
+
const {
|
|
706
|
+
file
|
|
707
|
+
} = await this.client("query")({
|
|
708
|
+
file: [{
|
|
709
|
+
id
|
|
710
|
+
}, {
|
|
711
|
+
"...on ErrorAccessDenied": {
|
|
712
|
+
__typename: true,
|
|
713
|
+
message: true
|
|
714
|
+
},
|
|
715
|
+
"...on FileQueryResponse": {
|
|
716
|
+
__typename: true,
|
|
717
|
+
file: fileSelector
|
|
718
|
+
}
|
|
719
|
+
}]
|
|
720
|
+
});
|
|
721
|
+
|
|
722
|
+
if (!file) {
|
|
723
|
+
return null;
|
|
724
|
+
}
|
|
725
|
+
|
|
726
|
+
if (file.__typename === "ErrorAccessDenied") {
|
|
727
|
+
return file;
|
|
728
|
+
}
|
|
729
|
+
|
|
730
|
+
if (!file.file) {
|
|
731
|
+
return null;
|
|
732
|
+
}
|
|
733
|
+
|
|
734
|
+
const result = gqlFileToExternal(file.file, _classPrivateFieldLooseBase(this, _keys)[_keys]);
|
|
735
|
+
return {
|
|
736
|
+
__typename: "SuccessResponse",
|
|
737
|
+
data: result
|
|
738
|
+
};
|
|
739
|
+
}
|
|
740
|
+
|
|
741
|
+
async mail(_ref10) {
|
|
742
|
+
let {
|
|
743
|
+
id
|
|
744
|
+
} = _ref10;
|
|
745
|
+
const {
|
|
746
|
+
mail
|
|
747
|
+
} = await this.client("query", {
|
|
748
|
+
scalars: {
|
|
749
|
+
DateTime: {
|
|
750
|
+
decode: e => new Date(e),
|
|
751
|
+
encode: e => e.toISOString()
|
|
752
|
+
}
|
|
753
|
+
}
|
|
754
|
+
})({
|
|
755
|
+
mail: [{
|
|
756
|
+
id
|
|
757
|
+
}, {
|
|
758
|
+
"...on ErrorAccessDenied": {
|
|
759
|
+
__typename: true,
|
|
760
|
+
message: true
|
|
761
|
+
},
|
|
762
|
+
"...on QueryMailResponse": {
|
|
763
|
+
__typename: true,
|
|
764
|
+
mail: mailSelector
|
|
765
|
+
}
|
|
766
|
+
}]
|
|
767
|
+
});
|
|
768
|
+
|
|
769
|
+
if (!mail) {
|
|
770
|
+
return null;
|
|
771
|
+
}
|
|
772
|
+
|
|
773
|
+
if (mail.__typename === "ErrorAccessDenied") {
|
|
774
|
+
return mail;
|
|
775
|
+
}
|
|
776
|
+
|
|
777
|
+
if (!mail.mail) {
|
|
778
|
+
return null;
|
|
779
|
+
}
|
|
780
|
+
|
|
781
|
+
const result = convertInternalMailToExternal(mail.mail, _classPrivateFieldLooseBase(this, _keys)[_keys]);
|
|
782
|
+
|
|
783
|
+
if (!result) {
|
|
784
|
+
return null;
|
|
785
|
+
}
|
|
786
|
+
|
|
787
|
+
return {
|
|
788
|
+
__typename: "SuccessResponse",
|
|
789
|
+
data: result
|
|
790
|
+
};
|
|
791
|
+
}
|
|
792
|
+
|
|
793
|
+
async deletedNodes() {
|
|
794
|
+
const {
|
|
795
|
+
deletedNodes
|
|
796
|
+
} = await this.client("query")({
|
|
797
|
+
deletedNodes: {
|
|
798
|
+
"...on ErrorAccessDenied": {
|
|
799
|
+
__typename: true,
|
|
800
|
+
message: true
|
|
801
|
+
},
|
|
802
|
+
"...on DeletedNodesResponse": {
|
|
803
|
+
__typename: true,
|
|
804
|
+
deletedNodes: nodeSelector
|
|
805
|
+
}
|
|
806
|
+
}
|
|
807
|
+
});
|
|
808
|
+
|
|
809
|
+
if (!deletedNodes) {
|
|
810
|
+
return null;
|
|
811
|
+
}
|
|
812
|
+
|
|
813
|
+
if (deletedNodes.__typename === "ErrorAccessDenied") {
|
|
814
|
+
return deletedNodes;
|
|
815
|
+
}
|
|
816
|
+
|
|
817
|
+
const nodes = new Array();
|
|
818
|
+
|
|
819
|
+
for (const node of deletedNodes.deletedNodes) {
|
|
820
|
+
nodes.push(await gqlNodeToExternal(node, _classPrivateFieldLooseBase(this, _keys)[_keys]));
|
|
821
|
+
}
|
|
822
|
+
|
|
823
|
+
return {
|
|
824
|
+
__typename: "SuccessResponse",
|
|
825
|
+
data: nodes
|
|
826
|
+
};
|
|
827
|
+
}
|
|
828
|
+
|
|
829
|
+
async sharedNodes() {
|
|
830
|
+
const {
|
|
831
|
+
sharedNodes
|
|
832
|
+
} = await this.client("query")({
|
|
833
|
+
sharedNodes: {
|
|
834
|
+
"...on ErrorAccessDenied": {
|
|
835
|
+
__typename: true,
|
|
836
|
+
message: true
|
|
837
|
+
},
|
|
838
|
+
"...on SharedNodesResponse": {
|
|
839
|
+
__typename: true,
|
|
840
|
+
sharedNodes: nodeSelector
|
|
841
|
+
}
|
|
842
|
+
}
|
|
843
|
+
});
|
|
844
|
+
|
|
845
|
+
if (!sharedNodes) {
|
|
846
|
+
return null;
|
|
847
|
+
}
|
|
848
|
+
|
|
849
|
+
if (sharedNodes.__typename === "ErrorAccessDenied") {
|
|
850
|
+
return sharedNodes;
|
|
851
|
+
}
|
|
852
|
+
|
|
853
|
+
const nodes = new Array();
|
|
854
|
+
|
|
855
|
+
for (const folder of sharedNodes.sharedNodes) {
|
|
856
|
+
nodes.push(await gqlNodeToExternal(folder, _classPrivateFieldLooseBase(this, _keys)[_keys]));
|
|
857
|
+
}
|
|
858
|
+
|
|
859
|
+
return {
|
|
860
|
+
__typename: "SuccessResponse",
|
|
861
|
+
data: nodes
|
|
862
|
+
};
|
|
863
|
+
}
|
|
864
|
+
|
|
865
|
+
async nodesSharedWithMe(type) {
|
|
866
|
+
const {
|
|
867
|
+
nodesSharedWithMe
|
|
868
|
+
} = await this.client("query")({
|
|
869
|
+
nodesSharedWithMe: [{
|
|
870
|
+
type
|
|
871
|
+
}, {
|
|
872
|
+
"...on ErrorAccessDenied": {
|
|
873
|
+
__typename: true,
|
|
874
|
+
message: true
|
|
875
|
+
},
|
|
876
|
+
"...on NodesSharedWithMeResponse": {
|
|
877
|
+
__typename: true,
|
|
878
|
+
nodesSharedWithMe: nodeSelector
|
|
879
|
+
}
|
|
880
|
+
}]
|
|
881
|
+
});
|
|
882
|
+
|
|
883
|
+
if (!nodesSharedWithMe) {
|
|
884
|
+
return null;
|
|
885
|
+
}
|
|
886
|
+
|
|
887
|
+
if (nodesSharedWithMe.__typename === "ErrorAccessDenied") {
|
|
888
|
+
return nodesSharedWithMe;
|
|
889
|
+
}
|
|
890
|
+
|
|
891
|
+
const nodes = new Array();
|
|
892
|
+
|
|
893
|
+
for (const folder of nodesSharedWithMe.nodesSharedWithMe) {
|
|
894
|
+
nodes.push(await gqlNodeToExternal(folder, _classPrivateFieldLooseBase(this, _keys)[_keys]));
|
|
895
|
+
}
|
|
896
|
+
|
|
897
|
+
return {
|
|
898
|
+
__typename: "SuccessResponse",
|
|
899
|
+
data: nodes
|
|
900
|
+
};
|
|
901
|
+
}
|
|
902
|
+
|
|
903
|
+
async deleteNodeSharing(_ref11) {
|
|
904
|
+
let {
|
|
905
|
+
nodeId,
|
|
906
|
+
userId
|
|
907
|
+
} = _ref11;
|
|
908
|
+
const {
|
|
909
|
+
deleteNodeSharing
|
|
910
|
+
} = await this.client("mutation")({
|
|
911
|
+
deleteNodeSharing: [{
|
|
912
|
+
nodeId,
|
|
913
|
+
userId
|
|
914
|
+
}, {
|
|
915
|
+
"...on ErrorAccessDenied": {
|
|
916
|
+
__typename: true,
|
|
917
|
+
message: true
|
|
918
|
+
},
|
|
919
|
+
"...on DeleteNodeSharingResponse": {
|
|
920
|
+
__typename: true,
|
|
921
|
+
deleteNodeSharing: true
|
|
922
|
+
}
|
|
923
|
+
}]
|
|
924
|
+
});
|
|
925
|
+
|
|
926
|
+
if (!deleteNodeSharing) {
|
|
927
|
+
return null;
|
|
928
|
+
}
|
|
929
|
+
|
|
930
|
+
if (deleteNodeSharing.__typename === "ErrorAccessDenied") {
|
|
931
|
+
return deleteNodeSharing;
|
|
932
|
+
}
|
|
933
|
+
|
|
934
|
+
return {
|
|
935
|
+
__typename: "SuccessResponse",
|
|
936
|
+
data: deleteNodeSharing.deleteNodeSharing
|
|
937
|
+
};
|
|
938
|
+
}
|
|
939
|
+
|
|
940
|
+
async duplicateNode(_ref12) {
|
|
941
|
+
var _node$access2;
|
|
942
|
+
|
|
943
|
+
let {
|
|
944
|
+
nodeId,
|
|
945
|
+
folderId,
|
|
946
|
+
customName
|
|
947
|
+
} = _ref12;
|
|
948
|
+
let node = nodesCache.get(nodeId);
|
|
949
|
+
|
|
950
|
+
if (!node) {
|
|
951
|
+
await this.node({
|
|
952
|
+
id: nodeId
|
|
953
|
+
});
|
|
954
|
+
node = nodesCache.get(nodeId);
|
|
955
|
+
|
|
956
|
+
if (!node) {
|
|
957
|
+
throw new Error("Node (" + nodeId + ") does not exists");
|
|
958
|
+
}
|
|
959
|
+
}
|
|
960
|
+
|
|
961
|
+
if (!((_node$access2 = node.access) != null && _node$access2.nameKey)) {
|
|
962
|
+
throw new Error("Can't have access to node " + nodeId);
|
|
963
|
+
}
|
|
964
|
+
|
|
965
|
+
customName = customName ? await encryptName(customName, node.access.nameKey) : null;
|
|
966
|
+
const {
|
|
967
|
+
duplicateNode
|
|
968
|
+
} = await this.client("mutation")({
|
|
969
|
+
duplicateNode: [{
|
|
970
|
+
nodeId,
|
|
971
|
+
folderId,
|
|
972
|
+
customName
|
|
973
|
+
}, {
|
|
974
|
+
"...on ErrorAccessDenied": {
|
|
975
|
+
__typename: true,
|
|
976
|
+
message: true
|
|
977
|
+
},
|
|
978
|
+
"...on ErrorNotFound": {
|
|
979
|
+
__typename: true,
|
|
980
|
+
message: true
|
|
981
|
+
},
|
|
982
|
+
"...on DuplicateNodeResponse": {
|
|
983
|
+
__typename: true,
|
|
984
|
+
duplicateNode: true
|
|
985
|
+
}
|
|
986
|
+
}]
|
|
987
|
+
});
|
|
988
|
+
|
|
989
|
+
if (!duplicateNode) {
|
|
990
|
+
return null;
|
|
991
|
+
}
|
|
992
|
+
|
|
993
|
+
if (duplicateNode.__typename === "ErrorAccessDenied") {
|
|
994
|
+
return duplicateNode;
|
|
995
|
+
}
|
|
996
|
+
|
|
997
|
+
if (duplicateNode.__typename === "ErrorNotFound") {
|
|
998
|
+
return duplicateNode;
|
|
999
|
+
}
|
|
1000
|
+
|
|
1001
|
+
return {
|
|
1002
|
+
__typename: "SuccessResponse",
|
|
1003
|
+
data: duplicateNode.duplicateNode
|
|
1004
|
+
};
|
|
1005
|
+
}
|
|
1006
|
+
|
|
1007
|
+
async deleteNodeCloudTrash(_ref13) {
|
|
1008
|
+
var _deleteNodeCloudTrash;
|
|
1009
|
+
|
|
1010
|
+
let {
|
|
1011
|
+
ids
|
|
1012
|
+
} = _ref13;
|
|
1013
|
+
const {
|
|
1014
|
+
deleteNodeCloudTrash
|
|
1015
|
+
} = await this.client("mutation")({
|
|
1016
|
+
deleteNodeCloudTrash: [{
|
|
1017
|
+
ids
|
|
1018
|
+
}, {
|
|
1019
|
+
"...on ErrorAccessDenied": {
|
|
1020
|
+
__typename: true,
|
|
1021
|
+
message: true
|
|
1022
|
+
},
|
|
1023
|
+
"...on DeleteNodeCloudTrashResponse": {
|
|
1024
|
+
__typename: true,
|
|
1025
|
+
deleteNodeCloudTrash: true
|
|
1026
|
+
}
|
|
1027
|
+
}]
|
|
1028
|
+
});
|
|
1029
|
+
|
|
1030
|
+
if (!deleteNodeCloudTrash) {
|
|
1031
|
+
return null;
|
|
1032
|
+
}
|
|
1033
|
+
|
|
1034
|
+
if (deleteNodeCloudTrash.__typename === "ErrorAccessDenied") {
|
|
1035
|
+
return deleteNodeCloudTrash;
|
|
1036
|
+
}
|
|
1037
|
+
|
|
1038
|
+
return {
|
|
1039
|
+
__typename: "SuccessResponse",
|
|
1040
|
+
data: (_deleteNodeCloudTrash = deleteNodeCloudTrash.deleteNodeCloudTrash) != null ? _deleteNodeCloudTrash : false
|
|
1041
|
+
};
|
|
1042
|
+
}
|
|
1043
|
+
|
|
1044
|
+
async shareNode(_ref14) {
|
|
1045
|
+
var _shareNodeFinish$shar;
|
|
1046
|
+
|
|
1047
|
+
let {
|
|
1048
|
+
nodeId,
|
|
1049
|
+
userId,
|
|
1050
|
+
rights
|
|
1051
|
+
} = _ref14;
|
|
1052
|
+
const user = await this.user({
|
|
1053
|
+
userId,
|
|
1054
|
+
withPublicKey: true
|
|
1055
|
+
});
|
|
1056
|
+
|
|
1057
|
+
if (!user) {
|
|
1058
|
+
return user;
|
|
1059
|
+
}
|
|
1060
|
+
|
|
1061
|
+
if (user.__typename === "ErrorNotFound") {
|
|
1062
|
+
return user;
|
|
1063
|
+
}
|
|
1064
|
+
|
|
1065
|
+
const {
|
|
1066
|
+
shareNode
|
|
1067
|
+
} = await this.client("mutation")({
|
|
1068
|
+
shareNode: [{
|
|
1069
|
+
nodeId,
|
|
1070
|
+
userId
|
|
1071
|
+
}, {
|
|
1072
|
+
"...on ErrorAccessDenied": {
|
|
1073
|
+
__typename: true,
|
|
1074
|
+
message: true
|
|
1075
|
+
},
|
|
1076
|
+
"...on ErrorNotFound": {
|
|
1077
|
+
__typename: true,
|
|
1078
|
+
message: true
|
|
1079
|
+
},
|
|
1080
|
+
"...on ShareNodeResponse": {
|
|
1081
|
+
__typename: true,
|
|
1082
|
+
nodes: true
|
|
1083
|
+
}
|
|
1084
|
+
}]
|
|
1085
|
+
});
|
|
1086
|
+
|
|
1087
|
+
if (!shareNode) {
|
|
1088
|
+
return null;
|
|
1089
|
+
}
|
|
1090
|
+
|
|
1091
|
+
if (shareNode.__typename === "ErrorAccessDenied") {
|
|
1092
|
+
return shareNode;
|
|
1093
|
+
}
|
|
1094
|
+
|
|
1095
|
+
if (shareNode.__typename === "ErrorNotFound") {
|
|
1096
|
+
return shareNode;
|
|
1097
|
+
}
|
|
1098
|
+
|
|
1099
|
+
const shareNodes = {
|
|
1100
|
+
nodes: []
|
|
1101
|
+
};
|
|
1102
|
+
|
|
1103
|
+
for (const id of shareNode.nodes) {
|
|
1104
|
+
const nameKey = await this.perNode(id, user.data.publicKey);
|
|
1105
|
+
|
|
1106
|
+
if (nameKey) {
|
|
1107
|
+
shareNodes.nodes.push(nameKey);
|
|
1108
|
+
}
|
|
1109
|
+
}
|
|
1110
|
+
|
|
1111
|
+
const {
|
|
1112
|
+
shareNodeFinish
|
|
1113
|
+
} = await this.client("mutation")({
|
|
1114
|
+
shareNodeFinish: [{
|
|
1115
|
+
rights,
|
|
1116
|
+
userId,
|
|
1117
|
+
shareNodes
|
|
1118
|
+
}, {
|
|
1119
|
+
"...on ErrorAccessDenied": {
|
|
1120
|
+
__typename: true,
|
|
1121
|
+
message: true
|
|
1122
|
+
},
|
|
1123
|
+
"...on ErrorNotFound": {
|
|
1124
|
+
__typename: true,
|
|
1125
|
+
message: true
|
|
1126
|
+
},
|
|
1127
|
+
"...on ShareNodeFinishResponse": {
|
|
1128
|
+
__typename: true,
|
|
1129
|
+
shareNodeFinish: true
|
|
1130
|
+
}
|
|
1131
|
+
}]
|
|
1132
|
+
});
|
|
1133
|
+
|
|
1134
|
+
if (!shareNodeFinish) {
|
|
1135
|
+
return null;
|
|
1136
|
+
}
|
|
1137
|
+
|
|
1138
|
+
if (shareNodeFinish.__typename === "ErrorAccessDenied") {
|
|
1139
|
+
return shareNodeFinish;
|
|
1140
|
+
}
|
|
1141
|
+
|
|
1142
|
+
if (shareNodeFinish.__typename === "ErrorNotFound") {
|
|
1143
|
+
return shareNodeFinish;
|
|
1144
|
+
}
|
|
1145
|
+
|
|
1146
|
+
return {
|
|
1147
|
+
__typename: "SuccessResponse",
|
|
1148
|
+
data: (_shareNodeFinish$shar = shareNodeFinish.shareNodeFinish) != null ? _shareNodeFinish$shar : false
|
|
1149
|
+
};
|
|
1150
|
+
}
|
|
1151
|
+
|
|
1152
|
+
async updateNode(_ref15) {
|
|
1153
|
+
var _node$access3, _node$access4;
|
|
1154
|
+
|
|
1155
|
+
let {
|
|
1156
|
+
nodeId,
|
|
1157
|
+
name,
|
|
1158
|
+
isFavorite,
|
|
1159
|
+
deletedAt
|
|
1160
|
+
} = _ref15;
|
|
1161
|
+
let node = nodesCache.get(nodeId);
|
|
1162
|
+
|
|
1163
|
+
if (!node) {
|
|
1164
|
+
await this.node({
|
|
1165
|
+
id: nodeId
|
|
1166
|
+
});
|
|
1167
|
+
node = nodesCache.get(nodeId);
|
|
1168
|
+
|
|
1169
|
+
if (!node) {
|
|
1170
|
+
throw "Can't find Node " + nodeId;
|
|
1171
|
+
}
|
|
1172
|
+
}
|
|
1173
|
+
|
|
1174
|
+
if (!((_node$access3 = node.access) != null && _node$access3.nameKey)) {
|
|
1175
|
+
throw new Error("Can't have access to node " + nodeId);
|
|
1176
|
+
}
|
|
1177
|
+
|
|
1178
|
+
name = name ? (_node$access4 = node.access) != null && _node$access4.nameKey ? await encryptName(name, node.access.nameKey) : name : null;
|
|
1179
|
+
const {
|
|
1180
|
+
updateNode
|
|
1181
|
+
} = await this.client("mutation")({
|
|
1182
|
+
updateNode: [{
|
|
1183
|
+
nodeId,
|
|
1184
|
+
name,
|
|
1185
|
+
isFavorite,
|
|
1186
|
+
deletedAt
|
|
1187
|
+
}, {
|
|
1188
|
+
"...on ErrorAccessDenied": {
|
|
1189
|
+
__typename: true,
|
|
1190
|
+
message: true
|
|
1191
|
+
},
|
|
1192
|
+
"...on ErrorNotExist": {
|
|
1193
|
+
__typename: true,
|
|
1194
|
+
message: true
|
|
1195
|
+
},
|
|
1196
|
+
"...on UpdateNodeResponse": {
|
|
1197
|
+
__typename: true,
|
|
1198
|
+
updateNode: nodeFullSelector
|
|
1199
|
+
}
|
|
1200
|
+
}]
|
|
1201
|
+
});
|
|
1202
|
+
|
|
1203
|
+
if (!updateNode) {
|
|
1204
|
+
return null;
|
|
1205
|
+
}
|
|
1206
|
+
|
|
1207
|
+
if (updateNode.__typename === "ErrorAccessDenied") {
|
|
1208
|
+
return updateNode;
|
|
1209
|
+
}
|
|
1210
|
+
|
|
1211
|
+
if (updateNode.__typename === "ErrorNotExist") {
|
|
1212
|
+
return updateNode;
|
|
1213
|
+
}
|
|
1214
|
+
|
|
1215
|
+
if (!updateNode.updateNode) {
|
|
1216
|
+
return null;
|
|
1217
|
+
}
|
|
1218
|
+
|
|
1219
|
+
const result = await gqlNodeToExternalNodeFull(updateNode.updateNode, _classPrivateFieldLooseBase(this, _keys)[_keys]);
|
|
1220
|
+
return {
|
|
1221
|
+
__typename: "SuccessResponse",
|
|
1222
|
+
data: result
|
|
1223
|
+
};
|
|
1224
|
+
}
|
|
1225
|
+
|
|
1226
|
+
async deletedMails(_ref16) {
|
|
1227
|
+
let {
|
|
1228
|
+
mailType
|
|
1229
|
+
} = _ref16;
|
|
1230
|
+
const {
|
|
1231
|
+
deletedMails
|
|
1232
|
+
} = await this.client("query", {
|
|
1233
|
+
scalars: {
|
|
1234
|
+
DateTime: {
|
|
1235
|
+
decode: e => new Date(e),
|
|
1236
|
+
encode: e => e.toISOString()
|
|
1237
|
+
}
|
|
1238
|
+
}
|
|
1239
|
+
})({
|
|
1240
|
+
deletedMails: [{
|
|
1241
|
+
mailType
|
|
1242
|
+
}, {
|
|
1243
|
+
"...on ErrorAccessDenied": {
|
|
1244
|
+
__typename: true,
|
|
1245
|
+
message: true
|
|
1246
|
+
},
|
|
1247
|
+
"...on DeletedMailsResponse": {
|
|
1248
|
+
__typename: true,
|
|
1249
|
+
deletedMails: mailSelector
|
|
1250
|
+
}
|
|
1251
|
+
}]
|
|
1252
|
+
});
|
|
1253
|
+
|
|
1254
|
+
if (!deletedMails) {
|
|
1255
|
+
return null;
|
|
1256
|
+
}
|
|
1257
|
+
|
|
1258
|
+
if (deletedMails.__typename === "ErrorAccessDenied") {
|
|
1259
|
+
return deletedMails;
|
|
1260
|
+
}
|
|
1261
|
+
|
|
1262
|
+
const mails = new Array();
|
|
1263
|
+
|
|
1264
|
+
for (const m of deletedMails.deletedMails) {
|
|
1265
|
+
const mail = convertInternalMailToExternal(m, _classPrivateFieldLooseBase(this, _keys)[_keys]);
|
|
1266
|
+
|
|
1267
|
+
if (mail) {
|
|
1268
|
+
mails.push(mail);
|
|
1269
|
+
}
|
|
1270
|
+
}
|
|
1271
|
+
|
|
1272
|
+
return {
|
|
1273
|
+
__typename: "SuccessResponse",
|
|
1274
|
+
data: mails
|
|
1275
|
+
};
|
|
1276
|
+
}
|
|
1277
|
+
|
|
1278
|
+
async updateAppNotifications(notifications) {
|
|
1279
|
+
const {
|
|
1280
|
+
updateAppNotifications
|
|
1281
|
+
} = await this.client("mutation", {
|
|
1282
|
+
scalars: {
|
|
1283
|
+
DateTime: {
|
|
1284
|
+
decode: e => new Date(e),
|
|
1285
|
+
encode: e => e.toISOString()
|
|
1286
|
+
}
|
|
1287
|
+
}
|
|
1288
|
+
})({
|
|
1289
|
+
updateAppNotifications: [notifications, {
|
|
1290
|
+
"...on ErrorAccessDenied": {
|
|
1291
|
+
__typename: true,
|
|
1292
|
+
message: true
|
|
1293
|
+
},
|
|
1294
|
+
"...on UpdateAppNotificationsResponse": {
|
|
1295
|
+
__typename: true,
|
|
1296
|
+
updateAppNotifications: {
|
|
1297
|
+
enableAll: true,
|
|
1298
|
+
mail: true,
|
|
1299
|
+
cloud: true,
|
|
1300
|
+
disableAllUntil: true
|
|
1301
|
+
}
|
|
1302
|
+
}
|
|
1303
|
+
}]
|
|
1304
|
+
});
|
|
1305
|
+
|
|
1306
|
+
if (!updateAppNotifications) {
|
|
1307
|
+
return null;
|
|
1308
|
+
}
|
|
1309
|
+
|
|
1310
|
+
if (updateAppNotifications.__typename === "ErrorAccessDenied") {
|
|
1311
|
+
return updateAppNotifications;
|
|
1312
|
+
}
|
|
1313
|
+
|
|
1314
|
+
if (!updateAppNotifications.updateAppNotifications) {
|
|
1315
|
+
return null;
|
|
1316
|
+
}
|
|
1317
|
+
|
|
1318
|
+
return {
|
|
1319
|
+
__typename: "SuccessResponse",
|
|
1320
|
+
data: updateAppNotifications.updateAppNotifications
|
|
1321
|
+
};
|
|
1322
|
+
}
|
|
1323
|
+
|
|
1324
|
+
async appNotifications() {
|
|
1325
|
+
const {
|
|
1326
|
+
appNotifications
|
|
1327
|
+
} = await this.client("query", {
|
|
1328
|
+
scalars: {
|
|
1329
|
+
DateTime: {
|
|
1330
|
+
decode: e => new Date(e),
|
|
1331
|
+
encode: e => e.toISOString()
|
|
1332
|
+
}
|
|
1333
|
+
}
|
|
1334
|
+
})({
|
|
1335
|
+
appNotifications: {
|
|
1336
|
+
"...on ErrorAccessDenied": {
|
|
1337
|
+
__typename: true,
|
|
1338
|
+
message: true
|
|
1339
|
+
},
|
|
1340
|
+
"...on ErrorNotFound": {
|
|
1341
|
+
__typename: true,
|
|
1342
|
+
message: true
|
|
1343
|
+
},
|
|
1344
|
+
"...on UserAppNotifications": {
|
|
1345
|
+
__typename: true,
|
|
1346
|
+
enableAll: true,
|
|
1347
|
+
mail: true,
|
|
1348
|
+
cloud: true,
|
|
1349
|
+
disableAllUntil: true
|
|
1350
|
+
}
|
|
1351
|
+
}
|
|
1352
|
+
});
|
|
1353
|
+
|
|
1354
|
+
if (!appNotifications) {
|
|
1355
|
+
return null;
|
|
1356
|
+
}
|
|
1357
|
+
|
|
1358
|
+
if (appNotifications.__typename === "ErrorAccessDenied") {
|
|
1359
|
+
return appNotifications;
|
|
1360
|
+
}
|
|
1361
|
+
|
|
1362
|
+
if (appNotifications.__typename === "ErrorNotFound") {
|
|
1363
|
+
return appNotifications;
|
|
1364
|
+
}
|
|
1365
|
+
|
|
1366
|
+
return {
|
|
1367
|
+
__typename: "SuccessResponse",
|
|
1368
|
+
data: appNotifications
|
|
1369
|
+
};
|
|
1370
|
+
}
|
|
1371
|
+
|
|
1372
|
+
async createMail(data, customMessage) {
|
|
1373
|
+
const mail = await this.createDraftMail(data);
|
|
1374
|
+
|
|
1375
|
+
if (!mail) {
|
|
1376
|
+
return {
|
|
1377
|
+
__typename: "SuccessResponse",
|
|
1378
|
+
data: false
|
|
1379
|
+
};
|
|
1380
|
+
}
|
|
1381
|
+
|
|
1382
|
+
if (mail.__typename === "ErrorBasic") {
|
|
1383
|
+
return mail;
|
|
1384
|
+
}
|
|
1385
|
+
|
|
1386
|
+
if (mail.__typename === "ErrorAccessDenied") {
|
|
1387
|
+
return mail;
|
|
1388
|
+
}
|
|
1389
|
+
|
|
1390
|
+
const result = await this.sendDraftMail(mail.data.mailIntegrityId, customMessage);
|
|
1391
|
+
|
|
1392
|
+
if (!result) {
|
|
1393
|
+
return null;
|
|
1394
|
+
}
|
|
1395
|
+
|
|
1396
|
+
if (result.__typename === "ErrorAccessDenied") {
|
|
1397
|
+
return result;
|
|
1398
|
+
}
|
|
1399
|
+
|
|
1400
|
+
if (result.__typename === "ErrorBasic") {
|
|
1401
|
+
return result;
|
|
1402
|
+
}
|
|
1403
|
+
|
|
1404
|
+
return {
|
|
1405
|
+
__typename: "SuccessResponse",
|
|
1406
|
+
data: result.data
|
|
1407
|
+
};
|
|
1408
|
+
}
|
|
1409
|
+
|
|
1410
|
+
async waitingReceivedMails() {
|
|
1411
|
+
const {
|
|
1412
|
+
user
|
|
1413
|
+
} = await this.client("query")({
|
|
1414
|
+
user: [{}, {
|
|
1415
|
+
"...on ErrorNotFound": {
|
|
1416
|
+
__typename: true,
|
|
1417
|
+
message: true
|
|
1418
|
+
},
|
|
1419
|
+
"...on UserResponse": {
|
|
1420
|
+
__typename: true,
|
|
1421
|
+
user: {
|
|
1422
|
+
waitingReceivedMails: {
|
|
1423
|
+
date: true,
|
|
1424
|
+
attachmentsCount: true,
|
|
1425
|
+
sender: {
|
|
1426
|
+
id: true,
|
|
1427
|
+
firstname: true,
|
|
1428
|
+
lastname: true,
|
|
1429
|
+
email: true,
|
|
1430
|
+
publicKey: true
|
|
1431
|
+
},
|
|
1432
|
+
recipients: {
|
|
1433
|
+
id: true,
|
|
1434
|
+
firstname: true,
|
|
1435
|
+
lastname: true,
|
|
1436
|
+
email: true,
|
|
1437
|
+
publicKey: true
|
|
1438
|
+
},
|
|
1439
|
+
temporaryRecipients: {
|
|
1440
|
+
email: true
|
|
1441
|
+
}
|
|
1442
|
+
}
|
|
1443
|
+
}
|
|
1444
|
+
}
|
|
1445
|
+
}]
|
|
1446
|
+
});
|
|
1447
|
+
|
|
1448
|
+
if (!user) {
|
|
1449
|
+
return null;
|
|
1450
|
+
}
|
|
1451
|
+
|
|
1452
|
+
if (user.__typename === "ErrorNotFound") {
|
|
1453
|
+
return user;
|
|
1454
|
+
}
|
|
1455
|
+
|
|
1456
|
+
if (!user.user) {
|
|
1457
|
+
return null;
|
|
1458
|
+
}
|
|
1459
|
+
|
|
1460
|
+
const result = user.user.waitingReceivedMails.map(m => _objectSpread(_objectSpread({
|
|
1461
|
+
id: generate()
|
|
1462
|
+
}, m), {}, {
|
|
1463
|
+
date: new Date(m.date)
|
|
1464
|
+
}));
|
|
1465
|
+
return {
|
|
1466
|
+
__typename: "SuccessResponse",
|
|
1467
|
+
data: result
|
|
1468
|
+
};
|
|
1469
|
+
}
|
|
1470
|
+
|
|
1471
|
+
async updateDraftMail(draftId, _ref17) {
|
|
1472
|
+
let {
|
|
1473
|
+
body,
|
|
1474
|
+
subject,
|
|
1475
|
+
files,
|
|
1476
|
+
recipientsIds,
|
|
1477
|
+
replyTo
|
|
1478
|
+
} = _ref17;
|
|
1479
|
+
const drafts = await this.draftMails();
|
|
1480
|
+
|
|
1481
|
+
if (!drafts) {
|
|
1482
|
+
return null;
|
|
1483
|
+
}
|
|
1484
|
+
|
|
1485
|
+
if (drafts.__typename !== "SuccessResponse") {
|
|
1486
|
+
return drafts;
|
|
1487
|
+
}
|
|
1488
|
+
|
|
1489
|
+
const draft = drafts.data.find(d => d.mailIntegrityId === draftId);
|
|
1490
|
+
|
|
1491
|
+
if (!draft) {
|
|
1492
|
+
throw new Error("Invalid draft " + draftId);
|
|
1493
|
+
}
|
|
1494
|
+
|
|
1495
|
+
let hashKey = null;
|
|
1496
|
+
let hash = null;
|
|
1497
|
+
|
|
1498
|
+
if (body || subject) {
|
|
1499
|
+
hashKey = sodium.randombytes_buf(sodium.crypto_generichash_KEYBYTES, "hex");
|
|
1500
|
+
hash = sodium.crypto_generichash(sodium.crypto_generichash_BYTES, JSON.stringify({
|
|
1501
|
+
body,
|
|
1502
|
+
subject
|
|
1503
|
+
}), hashKey, "hex");
|
|
1504
|
+
}
|
|
1505
|
+
|
|
1506
|
+
const senderFiles = new Array();
|
|
1507
|
+
|
|
1508
|
+
if (files) {
|
|
1509
|
+
for (const f of files) {
|
|
1510
|
+
let file = filesCache.get(f.id);
|
|
1511
|
+
|
|
1512
|
+
if (!file) {
|
|
1513
|
+
await this.file({
|
|
1514
|
+
id: f.id
|
|
1515
|
+
});
|
|
1516
|
+
file = filesCache.get(f.id);
|
|
1517
|
+
|
|
1518
|
+
if (!file) {
|
|
1519
|
+
throw new Error("File " + f.name + " (" + f.id + ") does not exists");
|
|
1520
|
+
}
|
|
1521
|
+
}
|
|
1522
|
+
|
|
1523
|
+
senderFiles.push({
|
|
1524
|
+
id: file.id,
|
|
1525
|
+
fileKey: sodium.to_hex(encryptCryptoBox(sodium.from_string(file.key), _classPrivateFieldLooseBase(this, _keys)[_keys].publicKey, _classPrivateFieldLooseBase(this, _keys)[_keys].privateKey)),
|
|
1526
|
+
name: sodium.to_hex(encryptCryptoBox(sodium.from_string(f.name), _classPrivateFieldLooseBase(this, _keys)[_keys].publicKey, _classPrivateFieldLooseBase(this, _keys)[_keys].privateKey))
|
|
1527
|
+
});
|
|
1528
|
+
}
|
|
1529
|
+
}
|
|
1530
|
+
|
|
1531
|
+
const {
|
|
1532
|
+
updateDraftMail
|
|
1533
|
+
} = await this.client("mutation", {
|
|
1534
|
+
scalars: {
|
|
1535
|
+
Json: {
|
|
1536
|
+
encode: e => JSON.stringify(e),
|
|
1537
|
+
decode: e => JSON.parse(e)
|
|
1538
|
+
},
|
|
1539
|
+
DateTime: {
|
|
1540
|
+
decode: e => new Date(e),
|
|
1541
|
+
encode: e => e.toISOString()
|
|
1542
|
+
},
|
|
1543
|
+
BigInt: {
|
|
1544
|
+
decode: e => BigInt(e),
|
|
1545
|
+
encode: e => e.toString()
|
|
1546
|
+
}
|
|
1547
|
+
}
|
|
1548
|
+
})({
|
|
1549
|
+
updateDraftMail: [{
|
|
1550
|
+
draftId,
|
|
1551
|
+
recipients: recipientsIds,
|
|
1552
|
+
replyTo,
|
|
1553
|
+
body: body ? sodium.to_hex(encryptCryptoBox(sodium.from_string(body), _classPrivateFieldLooseBase(this, _keys)[_keys].publicKey, _classPrivateFieldLooseBase(this, _keys)[_keys].privateKey)) : null,
|
|
1554
|
+
subject: subject ? sodium.to_hex(encryptCryptoBox(sodium.from_string(subject), _classPrivateFieldLooseBase(this, _keys)[_keys].publicKey, _classPrivateFieldLooseBase(this, _keys)[_keys].privateKey)) : null,
|
|
1555
|
+
senderFiles,
|
|
1556
|
+
hash,
|
|
1557
|
+
hashKey
|
|
1558
|
+
}, {
|
|
1559
|
+
"...on ErrorAccessDenied": {
|
|
1560
|
+
__typename: true,
|
|
1561
|
+
message: true
|
|
1562
|
+
},
|
|
1563
|
+
"...on ErrorBasic": {
|
|
1564
|
+
__typename: true,
|
|
1565
|
+
message: true
|
|
1566
|
+
},
|
|
1567
|
+
"...on UpdateDraftMailResponse": {
|
|
1568
|
+
__typename: true,
|
|
1569
|
+
updateDraftMail: mailSelector
|
|
1570
|
+
}
|
|
1571
|
+
}]
|
|
1572
|
+
});
|
|
1573
|
+
|
|
1574
|
+
if (!updateDraftMail) {
|
|
1575
|
+
return null;
|
|
1576
|
+
}
|
|
1577
|
+
|
|
1578
|
+
if (updateDraftMail.__typename === "ErrorAccessDenied") {
|
|
1579
|
+
return updateDraftMail;
|
|
1580
|
+
}
|
|
1581
|
+
|
|
1582
|
+
if (updateDraftMail.__typename === "ErrorBasic") {
|
|
1583
|
+
return updateDraftMail;
|
|
1584
|
+
}
|
|
1585
|
+
|
|
1586
|
+
if (!updateDraftMail.updateDraftMail) {
|
|
1587
|
+
return null;
|
|
1588
|
+
}
|
|
1589
|
+
|
|
1590
|
+
const result = convertInternalMailToExternal(updateDraftMail.updateDraftMail, _classPrivateFieldLooseBase(this, _keys)[_keys]);
|
|
1591
|
+
return {
|
|
1592
|
+
__typename: "SuccessResponse",
|
|
1593
|
+
data: result
|
|
1594
|
+
};
|
|
1595
|
+
}
|
|
1596
|
+
|
|
1597
|
+
async deleteDraftMail(draftId) {
|
|
1598
|
+
var _deleteDraftMail$dele;
|
|
1599
|
+
|
|
1600
|
+
const {
|
|
1601
|
+
deleteDraftMail
|
|
1602
|
+
} = await this.client("mutation")({
|
|
1603
|
+
deleteDraftMail: [{
|
|
1604
|
+
draftId
|
|
1605
|
+
}, {
|
|
1606
|
+
"...on ErrorAccessDenied": {
|
|
1607
|
+
__typename: true,
|
|
1608
|
+
message: true
|
|
1609
|
+
},
|
|
1610
|
+
"...on DeleteDraftMailResponse": {
|
|
1611
|
+
__typename: true,
|
|
1612
|
+
deleteDraftMail: true
|
|
1613
|
+
}
|
|
1614
|
+
}]
|
|
1615
|
+
});
|
|
1616
|
+
|
|
1617
|
+
if (!deleteDraftMail) {
|
|
1618
|
+
return null;
|
|
1619
|
+
}
|
|
1620
|
+
|
|
1621
|
+
if (deleteDraftMail.__typename === "ErrorAccessDenied") {
|
|
1622
|
+
return deleteDraftMail;
|
|
1623
|
+
}
|
|
1624
|
+
|
|
1625
|
+
if (!deleteDraftMail.deleteDraftMail) {
|
|
1626
|
+
return null;
|
|
1627
|
+
}
|
|
1628
|
+
|
|
1629
|
+
return {
|
|
1630
|
+
__typename: "SuccessResponse",
|
|
1631
|
+
data: (_deleteDraftMail$dele = deleteDraftMail.deleteDraftMail) != null ? _deleteDraftMail$dele : false
|
|
1632
|
+
};
|
|
1633
|
+
}
|
|
1634
|
+
|
|
1635
|
+
async deleteMailTrash(_ref18) {
|
|
1636
|
+
let {
|
|
1637
|
+
ids
|
|
1638
|
+
} = _ref18;
|
|
1639
|
+
const {
|
|
1640
|
+
deleteMailTrash
|
|
1641
|
+
} = await this.client("mutation")({
|
|
1642
|
+
deleteMailTrash: [{
|
|
1643
|
+
ids
|
|
1644
|
+
}, {
|
|
1645
|
+
"...on ErrorAccessDenied": {
|
|
1646
|
+
__typename: true,
|
|
1647
|
+
message: true
|
|
1648
|
+
},
|
|
1649
|
+
"...on DeleteMailTrashResponse": {
|
|
1650
|
+
__typename: true,
|
|
1651
|
+
deleteMailTrash: true
|
|
1652
|
+
}
|
|
1653
|
+
}]
|
|
1654
|
+
});
|
|
1655
|
+
|
|
1656
|
+
if (!deleteMailTrash) {
|
|
1657
|
+
return null;
|
|
1658
|
+
}
|
|
1659
|
+
|
|
1660
|
+
if (deleteMailTrash.__typename === "ErrorAccessDenied") {
|
|
1661
|
+
return deleteMailTrash;
|
|
1662
|
+
}
|
|
1663
|
+
|
|
1664
|
+
if (!deleteMailTrash.deleteMailTrash) {
|
|
1665
|
+
return null;
|
|
1666
|
+
}
|
|
1667
|
+
|
|
1668
|
+
return {
|
|
1669
|
+
__typename: "SuccessResponse",
|
|
1670
|
+
data: deleteMailTrash.deleteMailTrash
|
|
1671
|
+
};
|
|
1672
|
+
}
|
|
1673
|
+
|
|
1674
|
+
async emptyMailTrash() {
|
|
1675
|
+
const {
|
|
1676
|
+
emptyMailTrash
|
|
1677
|
+
} = await this.client("mutation")({
|
|
1678
|
+
emptyMailTrash: {
|
|
1679
|
+
"...on ErrorAccessDenied": {
|
|
1680
|
+
__typename: true,
|
|
1681
|
+
message: true
|
|
1682
|
+
},
|
|
1683
|
+
"...on EmptyMailTrashResponse": {
|
|
1684
|
+
__typename: true,
|
|
1685
|
+
emptyMailTrash: true
|
|
1686
|
+
}
|
|
1687
|
+
}
|
|
1688
|
+
});
|
|
1689
|
+
|
|
1690
|
+
if (!emptyMailTrash) {
|
|
1691
|
+
return null;
|
|
1692
|
+
}
|
|
1693
|
+
|
|
1694
|
+
if (emptyMailTrash.__typename === "ErrorAccessDenied") {
|
|
1695
|
+
return emptyMailTrash;
|
|
1696
|
+
}
|
|
1697
|
+
|
|
1698
|
+
if (!emptyMailTrash.emptyMailTrash) {
|
|
1699
|
+
return null;
|
|
1700
|
+
}
|
|
1701
|
+
|
|
1702
|
+
return {
|
|
1703
|
+
__typename: "SuccessResponse",
|
|
1704
|
+
data: emptyMailTrash.emptyMailTrash
|
|
1705
|
+
};
|
|
1706
|
+
}
|
|
1707
|
+
|
|
1708
|
+
async emptyCloudTrash() {
|
|
1709
|
+
const {
|
|
1710
|
+
emptyCloudTrash
|
|
1711
|
+
} = await this.client("mutation")({
|
|
1712
|
+
emptyCloudTrash: {
|
|
1713
|
+
"...on ErrorAccessDenied": {
|
|
1714
|
+
__typename: true,
|
|
1715
|
+
message: true
|
|
1716
|
+
},
|
|
1717
|
+
"...on EmptyCloudTrashResponse": {
|
|
1718
|
+
__typename: true,
|
|
1719
|
+
emptyCloudTrash: true
|
|
1720
|
+
}
|
|
1721
|
+
}
|
|
1722
|
+
});
|
|
1723
|
+
|
|
1724
|
+
if (!emptyCloudTrash) {
|
|
1725
|
+
return null;
|
|
1726
|
+
}
|
|
1727
|
+
|
|
1728
|
+
if (emptyCloudTrash.__typename === "ErrorAccessDenied") {
|
|
1729
|
+
return emptyCloudTrash;
|
|
1730
|
+
}
|
|
1731
|
+
|
|
1732
|
+
if (!emptyCloudTrash.emptyCloudTrash) {
|
|
1733
|
+
return null;
|
|
1734
|
+
}
|
|
1735
|
+
|
|
1736
|
+
return {
|
|
1737
|
+
__typename: "SuccessResponse",
|
|
1738
|
+
data: emptyCloudTrash.emptyCloudTrash
|
|
1739
|
+
};
|
|
1740
|
+
}
|
|
1741
|
+
|
|
1742
|
+
async recoverNode(id) {
|
|
1743
|
+
var _recoverNode$recoverN;
|
|
1744
|
+
|
|
1745
|
+
const {
|
|
1746
|
+
recoverNode
|
|
1747
|
+
} = await this.client("mutation")({
|
|
1748
|
+
recoverNode: [{
|
|
1749
|
+
id
|
|
1750
|
+
}, {
|
|
1751
|
+
"...on ErrorAccessDenied": {
|
|
1752
|
+
__typename: true,
|
|
1753
|
+
message: true
|
|
1754
|
+
},
|
|
1755
|
+
"...on ErrorNotExist": {
|
|
1756
|
+
__typename: true,
|
|
1757
|
+
message: true
|
|
1758
|
+
},
|
|
1759
|
+
"...on RecoverNodeResponse": {
|
|
1760
|
+
__typename: true,
|
|
1761
|
+
recoverNode: true
|
|
1762
|
+
}
|
|
1763
|
+
}]
|
|
1764
|
+
});
|
|
1765
|
+
|
|
1766
|
+
if (!recoverNode) {
|
|
1767
|
+
return null;
|
|
1768
|
+
}
|
|
1769
|
+
|
|
1770
|
+
if (recoverNode.__typename === "ErrorAccessDenied") {
|
|
1771
|
+
return recoverNode;
|
|
1772
|
+
}
|
|
1773
|
+
|
|
1774
|
+
if (recoverNode.__typename === "ErrorNotExist") {
|
|
1775
|
+
return recoverNode;
|
|
1776
|
+
}
|
|
1777
|
+
|
|
1778
|
+
return {
|
|
1779
|
+
__typename: "SuccessResponse",
|
|
1780
|
+
data: (_recoverNode$recoverN = recoverNode.recoverNode) != null ? _recoverNode$recoverN : false
|
|
1781
|
+
};
|
|
1782
|
+
}
|
|
1783
|
+
|
|
1784
|
+
async recoverMail(_ref19) {
|
|
1785
|
+
let {
|
|
1786
|
+
mailId
|
|
1787
|
+
} = _ref19;
|
|
1788
|
+
const {
|
|
1789
|
+
recoverMail
|
|
1790
|
+
} = await this.client("mutation")({
|
|
1791
|
+
recoverMail: [{
|
|
1792
|
+
mailId
|
|
1793
|
+
}, {
|
|
1794
|
+
"...on ErrorAccessDenied": {
|
|
1795
|
+
__typename: true,
|
|
1796
|
+
message: true
|
|
1797
|
+
},
|
|
1798
|
+
"...on ErrorBasic": {
|
|
1799
|
+
__typename: true,
|
|
1800
|
+
message: true
|
|
1801
|
+
},
|
|
1802
|
+
"...on RecoverMailResponse": {
|
|
1803
|
+
__typename: true,
|
|
1804
|
+
recoverMail: true
|
|
1805
|
+
}
|
|
1806
|
+
}]
|
|
1807
|
+
});
|
|
1808
|
+
|
|
1809
|
+
if (!recoverMail) {
|
|
1810
|
+
return null;
|
|
1811
|
+
}
|
|
1812
|
+
|
|
1813
|
+
if (recoverMail.__typename === "ErrorAccessDenied") {
|
|
1814
|
+
return recoverMail;
|
|
1815
|
+
}
|
|
1816
|
+
|
|
1817
|
+
if (recoverMail.__typename === "ErrorBasic") {
|
|
1818
|
+
return recoverMail;
|
|
1819
|
+
}
|
|
1820
|
+
|
|
1821
|
+
return {
|
|
1822
|
+
__typename: "SuccessResponse",
|
|
1823
|
+
data: recoverMail.recoverMail
|
|
1824
|
+
};
|
|
1825
|
+
}
|
|
1826
|
+
|
|
1827
|
+
async deleteFile(_ref20) {
|
|
1828
|
+
let {
|
|
1829
|
+
fileId,
|
|
1830
|
+
nodeId
|
|
1831
|
+
} = _ref20;
|
|
1832
|
+
const {
|
|
1833
|
+
deleteFile
|
|
1834
|
+
} = await this.client("mutation")({
|
|
1835
|
+
deleteFile: [{
|
|
1836
|
+
fileId,
|
|
1837
|
+
nodeId
|
|
1838
|
+
}, {
|
|
1839
|
+
"...on ErrorAccessDenied": {
|
|
1840
|
+
__typename: true,
|
|
1841
|
+
message: true
|
|
1842
|
+
},
|
|
1843
|
+
"...on ErrorNotExist": {
|
|
1844
|
+
__typename: true,
|
|
1845
|
+
message: true
|
|
1846
|
+
},
|
|
1847
|
+
"...on DeleteFileResponse": {
|
|
1848
|
+
__typename: true,
|
|
1849
|
+
deleteFile: true
|
|
1850
|
+
}
|
|
1851
|
+
}]
|
|
1852
|
+
});
|
|
1853
|
+
|
|
1854
|
+
if (!deleteFile) {
|
|
1855
|
+
return null;
|
|
1856
|
+
}
|
|
1857
|
+
|
|
1858
|
+
if (deleteFile.__typename === "ErrorAccessDenied") {
|
|
1859
|
+
return deleteFile;
|
|
1860
|
+
}
|
|
1861
|
+
|
|
1862
|
+
if (deleteFile.__typename === "ErrorNotExist") {
|
|
1863
|
+
return deleteFile;
|
|
1864
|
+
}
|
|
1865
|
+
|
|
1866
|
+
return {
|
|
1867
|
+
__typename: "SuccessResponse",
|
|
1868
|
+
data: deleteFile.deleteFile
|
|
1869
|
+
};
|
|
1870
|
+
}
|
|
1871
|
+
|
|
1872
|
+
async deleteNode(_ref21) {
|
|
1873
|
+
let {
|
|
1874
|
+
nodeId
|
|
1875
|
+
} = _ref21;
|
|
1876
|
+
const {
|
|
1877
|
+
deleteNode
|
|
1878
|
+
} = await this.client("mutation")({
|
|
1879
|
+
deleteNode: [{
|
|
1880
|
+
id: nodeId
|
|
1881
|
+
}, {
|
|
1882
|
+
"...on ErrorAccessDenied": {
|
|
1883
|
+
__typename: true,
|
|
1884
|
+
message: true
|
|
1885
|
+
},
|
|
1886
|
+
"...on DeleteNodeResponse": {
|
|
1887
|
+
__typename: true,
|
|
1888
|
+
deleteNode: true
|
|
1889
|
+
}
|
|
1890
|
+
}]
|
|
1891
|
+
});
|
|
1892
|
+
|
|
1893
|
+
if (!deleteNode) {
|
|
1894
|
+
return null;
|
|
1895
|
+
}
|
|
1896
|
+
|
|
1897
|
+
if (deleteNode.__typename === "ErrorAccessDenied") {
|
|
1898
|
+
return deleteNode;
|
|
1899
|
+
}
|
|
1900
|
+
|
|
1901
|
+
return {
|
|
1902
|
+
__typename: "SuccessResponse",
|
|
1903
|
+
data: deleteNode.deleteNode
|
|
1904
|
+
};
|
|
1905
|
+
}
|
|
1906
|
+
|
|
1907
|
+
async moveNodes(_ref22) {
|
|
1908
|
+
var _moveNodes$moveNodes;
|
|
1909
|
+
|
|
1910
|
+
let {
|
|
1911
|
+
nodeIds,
|
|
1912
|
+
parentNodeId
|
|
1913
|
+
} = _ref22;
|
|
1914
|
+
const {
|
|
1915
|
+
moveNodes
|
|
1916
|
+
} = await this.client("mutation")({
|
|
1917
|
+
moveNodes: [{
|
|
1918
|
+
nodeIds,
|
|
1919
|
+
parentNodeId
|
|
1920
|
+
}, {
|
|
1921
|
+
"...on ErrorAccessDenied": {
|
|
1922
|
+
__typename: true,
|
|
1923
|
+
message: true
|
|
1924
|
+
},
|
|
1925
|
+
"...on MoveNodesResponse": {
|
|
1926
|
+
__typename: true,
|
|
1927
|
+
moveNodes: true
|
|
1928
|
+
}
|
|
1929
|
+
}]
|
|
1930
|
+
});
|
|
1931
|
+
|
|
1932
|
+
if (!moveNodes) {
|
|
1933
|
+
return null;
|
|
1934
|
+
}
|
|
1935
|
+
|
|
1936
|
+
if (moveNodes.__typename === "ErrorAccessDenied") {
|
|
1937
|
+
return moveNodes;
|
|
1938
|
+
}
|
|
1939
|
+
|
|
1940
|
+
return {
|
|
1941
|
+
__typename: "SuccessResponse",
|
|
1942
|
+
data: (_moveNodes$moveNodes = moveNodes.moveNodes) != null ? _moveNodes$moveNodes : false
|
|
1943
|
+
};
|
|
1944
|
+
}
|
|
1945
|
+
|
|
1946
|
+
async folderSize(_ref23) {
|
|
1947
|
+
let {
|
|
1948
|
+
folderId
|
|
1949
|
+
} = _ref23;
|
|
1950
|
+
const {
|
|
1951
|
+
folderSize
|
|
1952
|
+
} = await this.client("query", {
|
|
1953
|
+
scalars: {
|
|
1954
|
+
BigInt: {
|
|
1955
|
+
decode: e => BigInt(e),
|
|
1956
|
+
encode: e => e.toString()
|
|
1957
|
+
}
|
|
1958
|
+
}
|
|
1959
|
+
})({
|
|
1960
|
+
folderSize: [{
|
|
1961
|
+
folderId
|
|
1962
|
+
}, {
|
|
1963
|
+
"...on ErrorAccessDenied": {
|
|
1964
|
+
__typename: true,
|
|
1965
|
+
message: true
|
|
1966
|
+
},
|
|
1967
|
+
"...on FolderSizeResponse": {
|
|
1968
|
+
__typename: true,
|
|
1969
|
+
size: true,
|
|
1970
|
+
sizeBefore: true
|
|
1971
|
+
}
|
|
1972
|
+
}]
|
|
1973
|
+
});
|
|
1974
|
+
|
|
1975
|
+
if (!folderSize) {
|
|
1976
|
+
return null;
|
|
1977
|
+
}
|
|
1978
|
+
|
|
1979
|
+
if (folderSize.__typename === "ErrorAccessDenied") {
|
|
1980
|
+
return folderSize;
|
|
1981
|
+
}
|
|
1982
|
+
|
|
1983
|
+
return {
|
|
1984
|
+
__typename: "SuccessResponse",
|
|
1985
|
+
data: {
|
|
1986
|
+
size: BigInt(folderSize.size),
|
|
1987
|
+
sizeBefore: BigInt(folderSize.sizeBefore)
|
|
1988
|
+
}
|
|
1989
|
+
};
|
|
1990
|
+
}
|
|
1991
|
+
|
|
1992
|
+
async deleteMail(_ref24) {
|
|
1993
|
+
let {
|
|
1994
|
+
mailId
|
|
1995
|
+
} = _ref24;
|
|
1996
|
+
const {
|
|
1997
|
+
deleteMail
|
|
1998
|
+
} = await this.client("mutation")({
|
|
1999
|
+
deleteMail: [{
|
|
2000
|
+
mailId
|
|
2001
|
+
}, {
|
|
2002
|
+
"...on ErrorAccessDenied": {
|
|
2003
|
+
__typename: true,
|
|
2004
|
+
message: true
|
|
2005
|
+
},
|
|
2006
|
+
"...on DeleteMailResponse": {
|
|
2007
|
+
__typename: true,
|
|
2008
|
+
deleteMail: true
|
|
2009
|
+
}
|
|
2010
|
+
}]
|
|
2011
|
+
});
|
|
2012
|
+
|
|
2013
|
+
if (!deleteMail) {
|
|
2014
|
+
return null;
|
|
2015
|
+
}
|
|
2016
|
+
|
|
2017
|
+
if (deleteMail.__typename === "ErrorAccessDenied") {
|
|
2018
|
+
return deleteMail;
|
|
2019
|
+
}
|
|
2020
|
+
|
|
2021
|
+
return {
|
|
2022
|
+
__typename: "SuccessResponse",
|
|
2023
|
+
data: deleteMail.deleteMail
|
|
2024
|
+
};
|
|
2025
|
+
}
|
|
2026
|
+
|
|
2027
|
+
async saveInCloud(_ref25) {
|
|
2028
|
+
var _node$parent;
|
|
2029
|
+
|
|
2030
|
+
let {
|
|
2031
|
+
fileId,
|
|
2032
|
+
name,
|
|
2033
|
+
nodeId
|
|
2034
|
+
} = _ref25;
|
|
2035
|
+
|
|
2036
|
+
if (nodeId && !nodesCache.has(nodeId)) {
|
|
2037
|
+
await this.node({
|
|
2038
|
+
id: nodeId
|
|
2039
|
+
});
|
|
2040
|
+
|
|
2041
|
+
if (!nodesCache.has(nodeId)) {
|
|
2042
|
+
return {
|
|
2043
|
+
__typename: "ErrorBasic",
|
|
2044
|
+
message: "The node " + nodeId + " does not exists"
|
|
2045
|
+
};
|
|
2046
|
+
}
|
|
2047
|
+
}
|
|
2048
|
+
|
|
2049
|
+
let key = "";
|
|
2050
|
+
const file = filesCache.get(fileId);
|
|
2051
|
+
|
|
2052
|
+
if (!file) {
|
|
2053
|
+
var _filesCache$get;
|
|
2054
|
+
|
|
2055
|
+
await this.file({
|
|
2056
|
+
id: fileId
|
|
2057
|
+
});
|
|
2058
|
+
const file = (_filesCache$get = filesCache.get(fileId)) != null ? _filesCache$get : null;
|
|
2059
|
+
|
|
2060
|
+
if (!file) {
|
|
2061
|
+
const receivedMails = await this.receivedMails();
|
|
2062
|
+
|
|
2063
|
+
if (!receivedMails) {
|
|
2064
|
+
return null;
|
|
2065
|
+
}
|
|
2066
|
+
|
|
2067
|
+
if (receivedMails.__typename !== "SuccessResponse") {
|
|
2068
|
+
return null;
|
|
2069
|
+
}
|
|
2070
|
+
|
|
2071
|
+
const mail = receivedMails.data.find(m => m.files.some(f => f.id === fileId));
|
|
2072
|
+
|
|
2073
|
+
if (!mail) {
|
|
2074
|
+
return {
|
|
2075
|
+
__typename: "ErrorBasic",
|
|
2076
|
+
message: "Can't find mail with the file " + fileId
|
|
2077
|
+
};
|
|
2078
|
+
}
|
|
2079
|
+
|
|
2080
|
+
const fileMail = mail.files.find(f => f.id === fileId);
|
|
2081
|
+
|
|
2082
|
+
if (!fileMail) {
|
|
2083
|
+
return {
|
|
2084
|
+
__typename: "ErrorBasic",
|
|
2085
|
+
message: "Can't find mail with the file " + fileId
|
|
2086
|
+
};
|
|
2087
|
+
}
|
|
2088
|
+
|
|
2089
|
+
const fileKey = decryptCryptoBox(sodium.from_hex(fileMail.key), mail.sender.publicKey, _classPrivateFieldLooseBase(this, _keys)[_keys].privateKey);
|
|
2090
|
+
key = sodium.to_hex(fileKey);
|
|
2091
|
+
} else {
|
|
2092
|
+
key = file.key;
|
|
2093
|
+
}
|
|
2094
|
+
} else {
|
|
2095
|
+
key = file.key;
|
|
2096
|
+
}
|
|
2097
|
+
|
|
2098
|
+
if (key === "") {
|
|
2099
|
+
return {
|
|
2100
|
+
__typename: "ErrorBasic",
|
|
2101
|
+
message: "Unexpected error 3"
|
|
2102
|
+
};
|
|
2103
|
+
}
|
|
2104
|
+
|
|
2105
|
+
key = sodium.to_hex(encryptCryptoBox(sodium.from_hex(key), _classPrivateFieldLooseBase(this, _keys)[_keys].publicKey, _classPrivateFieldLooseBase(this, _keys)[_keys].privateKey));
|
|
2106
|
+
const nameKey = secretstreamKeygen();
|
|
2107
|
+
const encryptedName = await encryptName(name, sodium.to_hex(nameKey));
|
|
2108
|
+
const encryptedNameKey = sodium.to_hex(encryptCryptoBox(nameKey, _classPrivateFieldLooseBase(this, _keys)[_keys].publicKey, _classPrivateFieldLooseBase(this, _keys)[_keys].privateKey));
|
|
2109
|
+
const {
|
|
2110
|
+
saveInCloud
|
|
2111
|
+
} = await this.client("mutation")({
|
|
2112
|
+
saveInCloud: [{
|
|
2113
|
+
fileId,
|
|
2114
|
+
key,
|
|
2115
|
+
nodeId,
|
|
2116
|
+
filename: encryptedName,
|
|
2117
|
+
nameKey: encryptedNameKey
|
|
2118
|
+
}, {
|
|
2119
|
+
"...on ErrorAccessDenied": {
|
|
2120
|
+
__typename: true,
|
|
2121
|
+
message: true
|
|
2122
|
+
},
|
|
2123
|
+
"...on ErrorNotFound": {
|
|
2124
|
+
__typename: true,
|
|
2125
|
+
message: true
|
|
2126
|
+
},
|
|
2127
|
+
"...on ErrorLimit": {
|
|
2128
|
+
__typename: true,
|
|
2129
|
+
message: true
|
|
2130
|
+
},
|
|
2131
|
+
"...on ErrorBasic": {
|
|
2132
|
+
__typename: true,
|
|
2133
|
+
message: true
|
|
2134
|
+
},
|
|
2135
|
+
"...on ErrorNotExist": {
|
|
2136
|
+
__typename: true,
|
|
2137
|
+
message: true
|
|
2138
|
+
},
|
|
2139
|
+
"...on SaveInCloudResponse": {
|
|
2140
|
+
__typename: true,
|
|
2141
|
+
saveInCloud: nodeFullSelector
|
|
2142
|
+
}
|
|
2143
|
+
}]
|
|
2144
|
+
});
|
|
2145
|
+
|
|
2146
|
+
if (!saveInCloud) {
|
|
2147
|
+
return null;
|
|
2148
|
+
}
|
|
2149
|
+
|
|
2150
|
+
if (saveInCloud.__typename === "ErrorAccessDenied") {
|
|
2151
|
+
return saveInCloud;
|
|
2152
|
+
}
|
|
2153
|
+
|
|
2154
|
+
if (saveInCloud.__typename === "ErrorNotFound") {
|
|
2155
|
+
return saveInCloud;
|
|
2156
|
+
}
|
|
2157
|
+
|
|
2158
|
+
if (saveInCloud.__typename === "ErrorLimit") {
|
|
2159
|
+
return saveInCloud;
|
|
2160
|
+
}
|
|
2161
|
+
|
|
2162
|
+
if (saveInCloud.__typename === "ErrorBasic") {
|
|
2163
|
+
return saveInCloud;
|
|
2164
|
+
}
|
|
2165
|
+
|
|
2166
|
+
if (saveInCloud.__typename === "ErrorNotExist") {
|
|
2167
|
+
return saveInCloud;
|
|
2168
|
+
}
|
|
2169
|
+
|
|
2170
|
+
if (!saveInCloud.saveInCloud) {
|
|
2171
|
+
return null;
|
|
2172
|
+
}
|
|
2173
|
+
|
|
2174
|
+
const node = await gqlNodeToExternalNodeFull(saveInCloud.saveInCloud, _classPrivateFieldLooseBase(this, _keys)[_keys]);
|
|
2175
|
+
const me = (_node$parent = node.parent) == null ? void 0 : _node$parent.users.find(_ref26 => {
|
|
2176
|
+
let [u] = _ref26;
|
|
2177
|
+
return u.publicKey === _classPrivateFieldLooseBase(this, _keys)[_keys].publicKey;
|
|
2178
|
+
});
|
|
2179
|
+
|
|
2180
|
+
if (me && ["admin", "write"].includes(me[1])) {
|
|
2181
|
+
var _node$parent$users$fi, _node$parent2;
|
|
2182
|
+
|
|
2183
|
+
const others = (_node$parent$users$fi = (_node$parent2 = node.parent) == null ? void 0 : _node$parent2.users.filter(_ref27 => {
|
|
2184
|
+
let [u] = _ref27;
|
|
2185
|
+
return u.publicKey !== _classPrivateFieldLooseBase(this, _keys)[_keys].publicKey;
|
|
2186
|
+
})) != null ? _node$parent$users$fi : [];
|
|
2187
|
+
await Promise.all(others.map(_ref28 => {
|
|
2188
|
+
let [u, rights] = _ref28;
|
|
2189
|
+
return this.shareNode({
|
|
2190
|
+
nodeId: node.id,
|
|
2191
|
+
rights,
|
|
2192
|
+
userId: u.id
|
|
2193
|
+
});
|
|
2194
|
+
}));
|
|
2195
|
+
}
|
|
2196
|
+
|
|
2197
|
+
return {
|
|
2198
|
+
__typename: "SuccessResponse",
|
|
2199
|
+
data: node
|
|
2200
|
+
};
|
|
2201
|
+
}
|
|
2202
|
+
|
|
2203
|
+
async dbGet(_ref29) {
|
|
2204
|
+
let {
|
|
2205
|
+
field,
|
|
2206
|
+
userId
|
|
2207
|
+
} = _ref29;
|
|
2208
|
+
const {
|
|
2209
|
+
dbGet
|
|
2210
|
+
} = await this.client("query")({
|
|
2211
|
+
dbGet: [{
|
|
2212
|
+
field,
|
|
2213
|
+
userId
|
|
2214
|
+
}, {
|
|
2215
|
+
"...on ErrorAccessDenied": {
|
|
2216
|
+
__typename: true,
|
|
2217
|
+
message: true
|
|
2218
|
+
},
|
|
2219
|
+
"...on ErrorNotExist": {
|
|
2220
|
+
__typename: true,
|
|
2221
|
+
message: true
|
|
2222
|
+
},
|
|
2223
|
+
"...on ErrorNotFound": {
|
|
2224
|
+
__typename: true,
|
|
2225
|
+
message: true
|
|
2226
|
+
},
|
|
2227
|
+
"...on DbGetResponse": {
|
|
2228
|
+
__typename: true,
|
|
2229
|
+
json: true
|
|
2230
|
+
}
|
|
2231
|
+
}]
|
|
2232
|
+
});
|
|
2233
|
+
|
|
2234
|
+
if (!dbGet) {
|
|
2235
|
+
return null;
|
|
2236
|
+
}
|
|
2237
|
+
|
|
2238
|
+
if (dbGet.__typename === "ErrorAccessDenied") {
|
|
2239
|
+
return dbGet;
|
|
2240
|
+
}
|
|
2241
|
+
|
|
2242
|
+
if (dbGet.__typename === "ErrorNotExist") {
|
|
2243
|
+
return dbGet;
|
|
2244
|
+
}
|
|
2245
|
+
|
|
2246
|
+
if (dbGet.__typename === "ErrorNotFound") {
|
|
2247
|
+
return dbGet;
|
|
2248
|
+
}
|
|
2249
|
+
|
|
2250
|
+
return {
|
|
2251
|
+
__typename: "SuccessResponse",
|
|
2252
|
+
data: dbGet.json.res
|
|
2253
|
+
};
|
|
2254
|
+
}
|
|
2255
|
+
|
|
2256
|
+
async dbSet(_ref30) {
|
|
2257
|
+
let {
|
|
2258
|
+
value,
|
|
2259
|
+
userId
|
|
2260
|
+
} = _ref30;
|
|
2261
|
+
|
|
2262
|
+
if (typeof value !== "object") {
|
|
2263
|
+
throw new Error("value should be an object including fields you want to update.");
|
|
2264
|
+
}
|
|
2265
|
+
|
|
2266
|
+
const {
|
|
2267
|
+
dbSet
|
|
2268
|
+
} = await this.client("mutation")({
|
|
2269
|
+
dbSet: [{
|
|
2270
|
+
userId,
|
|
2271
|
+
value: serialize(value).toString("base64")
|
|
2272
|
+
}, {
|
|
2273
|
+
"...on ErrorAccessDenied": {
|
|
2274
|
+
__typename: true,
|
|
2275
|
+
message: true
|
|
2276
|
+
},
|
|
2277
|
+
"...on ErrorNotFound": {
|
|
2278
|
+
__typename: true,
|
|
2279
|
+
message: true
|
|
2280
|
+
},
|
|
2281
|
+
"...on DbSetResponse": {
|
|
2282
|
+
__typename: true,
|
|
2283
|
+
dbSet: true
|
|
2284
|
+
}
|
|
2285
|
+
}]
|
|
2286
|
+
});
|
|
2287
|
+
|
|
2288
|
+
if (!dbSet) {
|
|
2289
|
+
return null;
|
|
2290
|
+
}
|
|
2291
|
+
|
|
2292
|
+
if (dbSet.__typename === "ErrorAccessDenied") {
|
|
2293
|
+
return dbSet;
|
|
2294
|
+
}
|
|
2295
|
+
|
|
2296
|
+
if (dbSet.__typename === "ErrorNotFound") {
|
|
2297
|
+
return dbSet;
|
|
2298
|
+
}
|
|
2299
|
+
|
|
2300
|
+
if (!dbSet.dbSet) {
|
|
2301
|
+
return null;
|
|
2302
|
+
}
|
|
2303
|
+
|
|
2304
|
+
return {
|
|
2305
|
+
__typename: "SuccessResponse",
|
|
2306
|
+
data: dbSet.dbSet.res
|
|
2307
|
+
};
|
|
2308
|
+
}
|
|
2309
|
+
|
|
2310
|
+
async dbSearch(_ref31) {
|
|
2311
|
+
let {
|
|
2312
|
+
search,
|
|
2313
|
+
field
|
|
2314
|
+
} = _ref31;
|
|
2315
|
+
const {
|
|
2316
|
+
dbSearch
|
|
2317
|
+
} = await this.client("query", {
|
|
2318
|
+
scalars: {
|
|
2319
|
+
Json: {
|
|
2320
|
+
encode: e => JSON.stringify(e),
|
|
2321
|
+
decode: e => JSON.parse(e)
|
|
2322
|
+
},
|
|
2323
|
+
DateTime: {
|
|
2324
|
+
decode: e => new Date(e),
|
|
2325
|
+
encode: e => e.toISOString()
|
|
2326
|
+
},
|
|
2327
|
+
BigInt: {
|
|
2328
|
+
decode: e => BigInt(e),
|
|
2329
|
+
encode: e => e.toString()
|
|
2330
|
+
}
|
|
2331
|
+
}
|
|
2332
|
+
})({
|
|
2333
|
+
dbSearch: [{
|
|
2334
|
+
search,
|
|
2335
|
+
field
|
|
2336
|
+
}, {
|
|
2337
|
+
"...on ErrorAccessDenied": {
|
|
2338
|
+
__typename: true,
|
|
2339
|
+
message: true
|
|
2340
|
+
},
|
|
2341
|
+
"...on ErrorBasic": {
|
|
2342
|
+
__typename: true,
|
|
2343
|
+
message: true
|
|
2344
|
+
},
|
|
2345
|
+
"...on ErrorNotExist": {
|
|
2346
|
+
__typename: true,
|
|
2347
|
+
message: true
|
|
2348
|
+
},
|
|
2349
|
+
"...on DbSearchResponse": {
|
|
2350
|
+
__typename: true,
|
|
2351
|
+
json: true
|
|
2352
|
+
}
|
|
2353
|
+
}]
|
|
2354
|
+
});
|
|
2355
|
+
|
|
2356
|
+
if (!dbSearch) {
|
|
2357
|
+
return null;
|
|
2358
|
+
}
|
|
2359
|
+
|
|
2360
|
+
if (dbSearch.__typename === "ErrorAccessDenied") {
|
|
2361
|
+
return dbSearch;
|
|
2362
|
+
}
|
|
2363
|
+
|
|
2364
|
+
if (dbSearch.__typename === "ErrorBasic") {
|
|
2365
|
+
return dbSearch;
|
|
2366
|
+
}
|
|
2367
|
+
|
|
2368
|
+
if (dbSearch.__typename === "ErrorNotExist") {
|
|
2369
|
+
return dbSearch;
|
|
2370
|
+
}
|
|
2371
|
+
|
|
2372
|
+
return {
|
|
2373
|
+
__typename: "SuccessResponse",
|
|
2374
|
+
data: dbSearch.json
|
|
2375
|
+
};
|
|
2376
|
+
}
|
|
2377
|
+
|
|
2378
|
+
async sendDraftMail(draftId, customMessage) {
|
|
2379
|
+
const drafts = await this.draftMails();
|
|
2380
|
+
|
|
2381
|
+
if (!drafts) {
|
|
2382
|
+
return {
|
|
2383
|
+
__typename: "SuccessResponse",
|
|
2384
|
+
data: false
|
|
2385
|
+
};
|
|
2386
|
+
}
|
|
2387
|
+
|
|
2388
|
+
if (drafts.__typename !== "SuccessResponse") {
|
|
2389
|
+
return {
|
|
2390
|
+
__typename: "SuccessResponse",
|
|
2391
|
+
data: false
|
|
2392
|
+
};
|
|
2393
|
+
}
|
|
2394
|
+
|
|
2395
|
+
const draft = drafts.data.find(d => d.mailIntegrityId === draftId);
|
|
2396
|
+
|
|
2397
|
+
if (!draft) {
|
|
2398
|
+
return {
|
|
2399
|
+
__typename: "SuccessResponse",
|
|
2400
|
+
data: false
|
|
2401
|
+
};
|
|
2402
|
+
}
|
|
2403
|
+
|
|
2404
|
+
const recipients = new Array();
|
|
2405
|
+
const temporaryRecipients = new Array();
|
|
2406
|
+
|
|
2407
|
+
for (const {
|
|
2408
|
+
email
|
|
2409
|
+
} of draft.temporaryRecipients) {
|
|
2410
|
+
if (!email) {
|
|
2411
|
+
continue;
|
|
2412
|
+
}
|
|
2413
|
+
|
|
2414
|
+
const input = await this._eachUser(draft.files, draft.subject, draft.body, email);
|
|
2415
|
+
|
|
2416
|
+
if (!input) {
|
|
2417
|
+
temporaryRecipients.push(email);
|
|
2418
|
+
} else {
|
|
2419
|
+
recipients.push(input);
|
|
2420
|
+
}
|
|
2421
|
+
}
|
|
2422
|
+
|
|
2423
|
+
for (const {
|
|
2424
|
+
id
|
|
2425
|
+
} of draft.recipients) {
|
|
2426
|
+
const input = await this._eachUser(draft.files, draft.subject, draft.body, id);
|
|
2427
|
+
|
|
2428
|
+
if (!input) {
|
|
2429
|
+
temporaryRecipients.push(id);
|
|
2430
|
+
} else {
|
|
2431
|
+
recipients.push(input);
|
|
2432
|
+
}
|
|
2433
|
+
}
|
|
2434
|
+
|
|
2435
|
+
const {
|
|
2436
|
+
sendDraftMail
|
|
2437
|
+
} = await this.client("mutation")({
|
|
2438
|
+
sendDraftMail: [{
|
|
2439
|
+
temporaryRecipients,
|
|
2440
|
+
recipients,
|
|
2441
|
+
draftMailId: draft.mailIntegrityId,
|
|
2442
|
+
customMessage
|
|
2443
|
+
}, {
|
|
2444
|
+
"...on ErrorAccessDenied": {
|
|
2445
|
+
__typename: true,
|
|
2446
|
+
message: true
|
|
2447
|
+
},
|
|
2448
|
+
"...on ErrorBasic": {
|
|
2449
|
+
__typename: true,
|
|
2450
|
+
message: true
|
|
2451
|
+
},
|
|
2452
|
+
"...on SendDraftMailResponse": {
|
|
2453
|
+
__typename: true,
|
|
2454
|
+
sendDraftMail: true
|
|
2455
|
+
}
|
|
2456
|
+
}]
|
|
2457
|
+
});
|
|
2458
|
+
|
|
2459
|
+
if (!sendDraftMail) {
|
|
2460
|
+
return null;
|
|
2461
|
+
}
|
|
2462
|
+
|
|
2463
|
+
if (sendDraftMail.__typename === "ErrorAccessDenied") {
|
|
2464
|
+
return sendDraftMail;
|
|
2465
|
+
}
|
|
2466
|
+
|
|
2467
|
+
if (sendDraftMail.__typename === "ErrorBasic") {
|
|
2468
|
+
return sendDraftMail;
|
|
2469
|
+
}
|
|
2470
|
+
|
|
2471
|
+
if (!sendDraftMail.sendDraftMail) {
|
|
2472
|
+
return null;
|
|
2473
|
+
}
|
|
2474
|
+
|
|
2475
|
+
return {
|
|
2476
|
+
__typename: "SuccessResponse",
|
|
2477
|
+
data: sendDraftMail.sendDraftMail
|
|
2478
|
+
};
|
|
2479
|
+
}
|
|
2480
|
+
|
|
2481
|
+
async sendWaitingEmails() {
|
|
2482
|
+
// TODO opti this
|
|
2483
|
+
const mails = await this.sentMails();
|
|
2484
|
+
|
|
2485
|
+
if (!mails) {
|
|
2486
|
+
return {
|
|
2487
|
+
__typename: "SuccessResponse",
|
|
2488
|
+
data: false
|
|
2489
|
+
};
|
|
2490
|
+
}
|
|
2491
|
+
|
|
2492
|
+
if (mails.__typename !== "SuccessResponse") {
|
|
2493
|
+
return {
|
|
2494
|
+
__typename: "SuccessResponse",
|
|
2495
|
+
data: false
|
|
2496
|
+
};
|
|
2497
|
+
}
|
|
2498
|
+
|
|
2499
|
+
const filtered = mails.data.filter(m => m.temporaryRecipients.length > 0);
|
|
2500
|
+
|
|
2501
|
+
for (const mail of filtered) {
|
|
2502
|
+
for (const {
|
|
2503
|
+
email
|
|
2504
|
+
} of mail.temporaryRecipients) {
|
|
2505
|
+
if (!email) {
|
|
2506
|
+
continue;
|
|
2507
|
+
}
|
|
2508
|
+
|
|
2509
|
+
try {
|
|
2510
|
+
const input = await this._eachUser(mail.files, mail.subject, mail.body, email);
|
|
2511
|
+
|
|
2512
|
+
if (!input) {
|
|
2513
|
+
continue;
|
|
2514
|
+
}
|
|
2515
|
+
|
|
2516
|
+
await this.client("mutation")({
|
|
2517
|
+
sendOneMail: [{
|
|
2518
|
+
mailIntegrityId: mail.mailIntegrityId,
|
|
2519
|
+
recipient: input
|
|
2520
|
+
}, {
|
|
2521
|
+
"...on ErrorAccessDenied": {
|
|
2522
|
+
__typename: true,
|
|
2523
|
+
message: true
|
|
2524
|
+
},
|
|
2525
|
+
"...on ErrorBasic": {
|
|
2526
|
+
__typename: true,
|
|
2527
|
+
message: true
|
|
2528
|
+
},
|
|
2529
|
+
"...on RecoverNodeResponse": {
|
|
2530
|
+
__typename: true,
|
|
2531
|
+
recoverNode: true
|
|
2532
|
+
}
|
|
2533
|
+
}]
|
|
2534
|
+
});
|
|
2535
|
+
} catch {
|
|
2536
|
+
continue;
|
|
2537
|
+
}
|
|
2538
|
+
}
|
|
2539
|
+
}
|
|
2540
|
+
|
|
2541
|
+
return {
|
|
2542
|
+
__typename: "SuccessResponse",
|
|
2543
|
+
data: false
|
|
2544
|
+
};
|
|
2545
|
+
}
|
|
2546
|
+
|
|
2547
|
+
async createDraftMail(_ref32) {
|
|
2548
|
+
let {
|
|
2549
|
+
body,
|
|
2550
|
+
subject,
|
|
2551
|
+
files,
|
|
2552
|
+
recipientsIds,
|
|
2553
|
+
replyTo
|
|
2554
|
+
} = _ref32;
|
|
2555
|
+
const hashKey = sodium.randombytes_buf(sodium.crypto_generichash_KEYBYTES, "hex");
|
|
2556
|
+
const hash = sodium.crypto_generichash(sodium.crypto_generichash_BYTES, JSON.stringify({
|
|
2557
|
+
body,
|
|
2558
|
+
subject
|
|
2559
|
+
}), hashKey, "hex");
|
|
2560
|
+
const senderFiles = new Array();
|
|
2561
|
+
|
|
2562
|
+
for (const f of files) {
|
|
2563
|
+
let file = filesCache.get(f.id);
|
|
2564
|
+
|
|
2565
|
+
if (!file) {
|
|
2566
|
+
await this.file({
|
|
2567
|
+
id: f.id
|
|
2568
|
+
});
|
|
2569
|
+
file = filesCache.get(f.id);
|
|
2570
|
+
|
|
2571
|
+
if (!file) {
|
|
2572
|
+
throw new Error("File " + f.name + " (" + f.id + ") does not exists");
|
|
2573
|
+
}
|
|
2574
|
+
}
|
|
2575
|
+
|
|
2576
|
+
senderFiles.push({
|
|
2577
|
+
id: file.id,
|
|
2578
|
+
fileKey: sodium.to_hex(encryptCryptoBox(sodium.from_string(file.key), _classPrivateFieldLooseBase(this, _keys)[_keys].publicKey, _classPrivateFieldLooseBase(this, _keys)[_keys].privateKey)),
|
|
2579
|
+
name: sodium.to_hex(encryptCryptoBox(sodium.from_string(f.name), _classPrivateFieldLooseBase(this, _keys)[_keys].publicKey, _classPrivateFieldLooseBase(this, _keys)[_keys].privateKey))
|
|
2580
|
+
});
|
|
2581
|
+
}
|
|
2582
|
+
|
|
2583
|
+
const {
|
|
2584
|
+
createDraftMail
|
|
2585
|
+
} = await this.client("mutation", {
|
|
2586
|
+
scalars: {
|
|
2587
|
+
DateTime: {
|
|
2588
|
+
decode: e => new Date(e),
|
|
2589
|
+
encode: e => e.toISOString()
|
|
2590
|
+
}
|
|
2591
|
+
}
|
|
2592
|
+
})({
|
|
2593
|
+
createDraftMail: [{
|
|
2594
|
+
recipients: recipientsIds,
|
|
2595
|
+
replyTo,
|
|
2596
|
+
body: sodium.to_hex(encryptCryptoBox(sodium.from_string(body), _classPrivateFieldLooseBase(this, _keys)[_keys].publicKey, _classPrivateFieldLooseBase(this, _keys)[_keys].privateKey)),
|
|
2597
|
+
subject: sodium.to_hex(encryptCryptoBox(sodium.from_string(subject), _classPrivateFieldLooseBase(this, _keys)[_keys].publicKey, _classPrivateFieldLooseBase(this, _keys)[_keys].privateKey)),
|
|
2598
|
+
senderFiles,
|
|
2599
|
+
hash,
|
|
2600
|
+
hashKey
|
|
2601
|
+
}, {
|
|
2602
|
+
"...on ErrorAccessDenied": {
|
|
2603
|
+
__typename: true,
|
|
2604
|
+
message: true
|
|
2605
|
+
},
|
|
2606
|
+
"...on ErrorBasic": {
|
|
2607
|
+
__typename: true,
|
|
2608
|
+
message: true
|
|
2609
|
+
},
|
|
2610
|
+
"...on CreateDraftMailResponse": {
|
|
2611
|
+
__typename: true,
|
|
2612
|
+
createDraftMail: mailSelector
|
|
2613
|
+
}
|
|
2614
|
+
}]
|
|
2615
|
+
});
|
|
2616
|
+
|
|
2617
|
+
if (!createDraftMail) {
|
|
2618
|
+
return null;
|
|
2619
|
+
}
|
|
2620
|
+
|
|
2621
|
+
if (createDraftMail.__typename === "ErrorAccessDenied") {
|
|
2622
|
+
return createDraftMail;
|
|
2623
|
+
}
|
|
2624
|
+
|
|
2625
|
+
if (createDraftMail.__typename === "ErrorBasic") {
|
|
2626
|
+
return createDraftMail;
|
|
2627
|
+
}
|
|
2628
|
+
|
|
2629
|
+
if (!createDraftMail.createDraftMail) {
|
|
2630
|
+
return null;
|
|
2631
|
+
}
|
|
2632
|
+
|
|
2633
|
+
const result = convertInternalMailToExternal(createDraftMail.createDraftMail, _classPrivateFieldLooseBase(this, _keys)[_keys]);
|
|
2634
|
+
return {
|
|
2635
|
+
__typename: "SuccessResponse",
|
|
2636
|
+
data: result
|
|
2637
|
+
};
|
|
2638
|
+
}
|
|
2639
|
+
|
|
2640
|
+
async fileContent(_ref33) {
|
|
2641
|
+
let {
|
|
2642
|
+
fileId,
|
|
2643
|
+
onDownloadProgress,
|
|
2644
|
+
progressDecrypt,
|
|
2645
|
+
signal
|
|
2646
|
+
} = _ref33;
|
|
2647
|
+
const {
|
|
2648
|
+
fileContent
|
|
2649
|
+
} = await this.client("query", {
|
|
2650
|
+
scalars: {
|
|
2651
|
+
Bytes: {
|
|
2652
|
+
decode: e => {
|
|
2653
|
+
console.log(e, typeof e);
|
|
2654
|
+
return Uint8Array.from([1, 2, 3]);
|
|
2655
|
+
},
|
|
2656
|
+
encode: e => e.toString()
|
|
2657
|
+
}
|
|
2658
|
+
}
|
|
2659
|
+
})({
|
|
2660
|
+
fileContent: [{
|
|
2661
|
+
fileId
|
|
2662
|
+
}, {
|
|
2663
|
+
"...on ErrorAccessDenied": {
|
|
2664
|
+
__typename: true,
|
|
2665
|
+
message: true
|
|
2666
|
+
},
|
|
2667
|
+
"...on ErrorBasic": {
|
|
2668
|
+
__typename: true,
|
|
2669
|
+
message: true
|
|
2670
|
+
},
|
|
2671
|
+
"...on FileContentResponse": {
|
|
2672
|
+
__typename: true,
|
|
2673
|
+
file: {
|
|
2674
|
+
"...on FileContentCloud": {
|
|
2675
|
+
__typename: true,
|
|
2676
|
+
parts: {
|
|
2677
|
+
contentUrl: true,
|
|
2678
|
+
order: true,
|
|
2679
|
+
md5: true
|
|
2680
|
+
},
|
|
2681
|
+
key: true,
|
|
2682
|
+
publicKey: true,
|
|
2683
|
+
totalSize: true,
|
|
2684
|
+
md5: true,
|
|
2685
|
+
md5Encrypted: true
|
|
2686
|
+
},
|
|
2687
|
+
"...on FileContentLite": {
|
|
2688
|
+
__typename: true,
|
|
2689
|
+
content: true,
|
|
2690
|
+
id: true,
|
|
2691
|
+
key: true,
|
|
2692
|
+
publicKey: true,
|
|
2693
|
+
md5: true,
|
|
2694
|
+
md5Encrypted: true,
|
|
2695
|
+
totalSize: true
|
|
2696
|
+
},
|
|
2697
|
+
"...on FileContentReceivedMail": {
|
|
2698
|
+
__typename: true,
|
|
2699
|
+
maybeParts: {
|
|
2700
|
+
contentUrl: true,
|
|
2701
|
+
order: true,
|
|
2702
|
+
md5: true
|
|
2703
|
+
},
|
|
2704
|
+
maybeContent: true,
|
|
2705
|
+
key: true,
|
|
2706
|
+
senderPublicKey: true,
|
|
2707
|
+
totalSize: true,
|
|
2708
|
+
md5: true,
|
|
2709
|
+
md5Encrypted: true
|
|
2710
|
+
},
|
|
2711
|
+
"...on FileContentSentMail": {
|
|
2712
|
+
__typename: true,
|
|
2713
|
+
maybeParts: {
|
|
2714
|
+
contentUrl: true,
|
|
2715
|
+
order: true,
|
|
2716
|
+
md5: true
|
|
2717
|
+
},
|
|
2718
|
+
maybeContent: true,
|
|
2719
|
+
key: true,
|
|
2720
|
+
totalSize: true,
|
|
2721
|
+
md5: true,
|
|
2722
|
+
md5Encrypted: true
|
|
2723
|
+
}
|
|
2724
|
+
}
|
|
2725
|
+
}
|
|
2726
|
+
}]
|
|
2727
|
+
});
|
|
2728
|
+
|
|
2729
|
+
if (!fileContent) {
|
|
2730
|
+
return null;
|
|
2731
|
+
}
|
|
2732
|
+
|
|
2733
|
+
if (fileContent.__typename === "ErrorAccessDenied") {
|
|
2734
|
+
return fileContent;
|
|
2735
|
+
}
|
|
2736
|
+
|
|
2737
|
+
if (fileContent.__typename === "ErrorBasic") {
|
|
2738
|
+
return fileContent;
|
|
2739
|
+
}
|
|
2740
|
+
|
|
2741
|
+
const file = fileContent.file;
|
|
2742
|
+
|
|
2743
|
+
if (!file) {
|
|
2744
|
+
return null;
|
|
2745
|
+
}
|
|
2746
|
+
|
|
2747
|
+
const progressParts = {};
|
|
2748
|
+
|
|
2749
|
+
const onProgress = (part, progressEvent) => {
|
|
2750
|
+
progressParts[part] = progressEvent;
|
|
2751
|
+
const transferredBytes = Object.values(progressParts).reduce((prv, cur) => prv + cur.transferredBytes, 0);
|
|
2752
|
+
const totalBytes = Number(file.totalSize);
|
|
2753
|
+
onDownloadProgress == null ? void 0 : onDownloadProgress({
|
|
2754
|
+
percent: transferredBytes / totalBytes,
|
|
2755
|
+
totalBytes,
|
|
2756
|
+
transferredBytes
|
|
2757
|
+
});
|
|
2758
|
+
};
|
|
2759
|
+
|
|
2760
|
+
const encryptedContentFromParts = async fileParts => {
|
|
2761
|
+
const parts = new Array();
|
|
2762
|
+
|
|
2763
|
+
const byPart = async part => {
|
|
2764
|
+
const buf = new Uint8Array(await ky.get(part.contentUrl, {
|
|
2765
|
+
timeout: false,
|
|
2766
|
+
onDownloadProgress: pr => onProgress(part.order, pr),
|
|
2767
|
+
signal: signal
|
|
2768
|
+
}).arrayBuffer());
|
|
2769
|
+
const md5Part = await md5(buf);
|
|
2770
|
+
|
|
2771
|
+
if (md5Part !== part.md5) {
|
|
2772
|
+
throw new Error("Invalid md5 for part " + part.order + " of file " + fileId);
|
|
2773
|
+
}
|
|
2774
|
+
|
|
2775
|
+
parts.push({
|
|
2776
|
+
data: buf,
|
|
2777
|
+
order: part.order
|
|
2778
|
+
});
|
|
2779
|
+
};
|
|
2780
|
+
|
|
2781
|
+
await promiseAllLimit(3, fileParts.map(p => () => byPart(p)));
|
|
2782
|
+
return concatenate(...parts.sort((a, b) => a.order - b.order).map(p => p.data));
|
|
2783
|
+
};
|
|
2784
|
+
|
|
2785
|
+
const finalize = async encryptedContent => {
|
|
2786
|
+
// const md5Encrypted = await firstValueFrom(md5(of(encryptedContent)));
|
|
2787
|
+
const md5Encrypted = await md5(encryptedContent);
|
|
2788
|
+
|
|
2789
|
+
if (md5Encrypted !== file.md5Encrypted) {
|
|
2790
|
+
throw new Error("Encrypted content does not match");
|
|
2791
|
+
}
|
|
2792
|
+
|
|
2793
|
+
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);
|
|
2794
|
+
const src = await decrypt(key, encryptedContent, progressDecrypt, signal); // const md5Content = await firstValueFrom(md5(of(src)));
|
|
2795
|
+
|
|
2796
|
+
const md5Content = await md5(src);
|
|
2797
|
+
|
|
2798
|
+
if (md5Content !== file.md5) {
|
|
2799
|
+
throw new Error("Content does not match");
|
|
2800
|
+
}
|
|
2801
|
+
|
|
2802
|
+
return uncompress(src);
|
|
2803
|
+
};
|
|
2804
|
+
|
|
2805
|
+
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;
|
|
2806
|
+
|
|
2807
|
+
if (!encryptedContent) {
|
|
2808
|
+
return null;
|
|
2809
|
+
}
|
|
2810
|
+
|
|
2811
|
+
return {
|
|
2812
|
+
__typename: "SuccessResponse",
|
|
2813
|
+
data: await finalize(encryptedContent)
|
|
2814
|
+
};
|
|
2815
|
+
}
|
|
2816
|
+
|
|
2817
|
+
async readMail(_ref34) {
|
|
2818
|
+
let {
|
|
2819
|
+
mailId
|
|
2820
|
+
} = _ref34;
|
|
2821
|
+
const {
|
|
2822
|
+
readMail
|
|
2823
|
+
} = await this.client("mutation")({
|
|
2824
|
+
readMail: [{
|
|
2825
|
+
mailId
|
|
2826
|
+
}, {
|
|
2827
|
+
"...on ErrorAccessDenied": {
|
|
2828
|
+
__typename: true,
|
|
2829
|
+
message: true
|
|
2830
|
+
},
|
|
2831
|
+
"...on ErrorBasic": {
|
|
2832
|
+
__typename: true,
|
|
2833
|
+
message: true
|
|
2834
|
+
},
|
|
2835
|
+
"...on ErrorNotFound": {
|
|
2836
|
+
__typename: true,
|
|
2837
|
+
message: true
|
|
2838
|
+
},
|
|
2839
|
+
"...on ReadMailResponse": {
|
|
2840
|
+
__typename: true,
|
|
2841
|
+
readMail: true
|
|
2842
|
+
}
|
|
2843
|
+
}]
|
|
2844
|
+
});
|
|
2845
|
+
|
|
2846
|
+
if (!readMail) {
|
|
2847
|
+
return null;
|
|
2848
|
+
}
|
|
2849
|
+
|
|
2850
|
+
if (readMail.__typename === "ErrorAccessDenied") {
|
|
2851
|
+
return readMail;
|
|
2852
|
+
}
|
|
2853
|
+
|
|
2854
|
+
if (readMail.__typename === "ErrorBasic") {
|
|
2855
|
+
return readMail;
|
|
2856
|
+
}
|
|
2857
|
+
|
|
2858
|
+
if (readMail.__typename === "ErrorNotFound") {
|
|
2859
|
+
return readMail;
|
|
2860
|
+
}
|
|
2861
|
+
|
|
2862
|
+
if (!readMail.readMail) {
|
|
2863
|
+
return null;
|
|
2864
|
+
}
|
|
2865
|
+
|
|
2866
|
+
return {
|
|
2867
|
+
__typename: "SuccessResponse",
|
|
2868
|
+
data: readMail.readMail
|
|
2869
|
+
};
|
|
2870
|
+
}
|
|
2871
|
+
|
|
2872
|
+
async unreadMail(_ref35) {
|
|
2873
|
+
let {
|
|
2874
|
+
mailId
|
|
2875
|
+
} = _ref35;
|
|
2876
|
+
const {
|
|
2877
|
+
unreadMail
|
|
2878
|
+
} = await this.client("mutation")({
|
|
2879
|
+
unreadMail: [{
|
|
2880
|
+
mailId
|
|
2881
|
+
}, {
|
|
2882
|
+
"...on ErrorAccessDenied": {
|
|
2883
|
+
__typename: true,
|
|
2884
|
+
message: true
|
|
2885
|
+
},
|
|
2886
|
+
"...on ErrorBasic": {
|
|
2887
|
+
__typename: true,
|
|
2888
|
+
message: true
|
|
2889
|
+
},
|
|
2890
|
+
"...on ErrorNotFound": {
|
|
2891
|
+
__typename: true,
|
|
2892
|
+
message: true
|
|
2893
|
+
},
|
|
2894
|
+
"...on UnreadMailResponse": {
|
|
2895
|
+
__typename: true,
|
|
2896
|
+
unreadMail: true
|
|
2897
|
+
}
|
|
2898
|
+
}]
|
|
2899
|
+
});
|
|
2900
|
+
|
|
2901
|
+
if (!unreadMail) {
|
|
2902
|
+
return null;
|
|
2903
|
+
}
|
|
2904
|
+
|
|
2905
|
+
if (unreadMail.__typename === "ErrorAccessDenied") {
|
|
2906
|
+
return unreadMail;
|
|
2907
|
+
}
|
|
2908
|
+
|
|
2909
|
+
if (unreadMail.__typename === "ErrorBasic") {
|
|
2910
|
+
return unreadMail;
|
|
2911
|
+
}
|
|
2912
|
+
|
|
2913
|
+
if (unreadMail.__typename === "ErrorNotFound") {
|
|
2914
|
+
return unreadMail;
|
|
2915
|
+
}
|
|
2916
|
+
|
|
2917
|
+
if (!unreadMail.unreadMail) {
|
|
2918
|
+
return null;
|
|
2919
|
+
}
|
|
2920
|
+
|
|
2921
|
+
return {
|
|
2922
|
+
__typename: "SuccessResponse",
|
|
2923
|
+
data: unreadMail.unreadMail
|
|
2924
|
+
};
|
|
2925
|
+
}
|
|
2926
|
+
|
|
2927
|
+
async appSettings() {
|
|
2928
|
+
var _user$user;
|
|
2929
|
+
|
|
2930
|
+
const {
|
|
2931
|
+
user
|
|
2932
|
+
} = await this.client("query")({
|
|
2933
|
+
user: [{}, {
|
|
2934
|
+
"...on ErrorNotFound": {
|
|
2935
|
+
__typename: true,
|
|
2936
|
+
message: true
|
|
2937
|
+
},
|
|
2938
|
+
"...on UserResponse": {
|
|
2939
|
+
__typename: true,
|
|
2940
|
+
user: {
|
|
2941
|
+
appSettings: {
|
|
2942
|
+
cloudFileDaysForDelete: true,
|
|
2943
|
+
cloudFolderDaysForDelete: true,
|
|
2944
|
+
historyFileDaysForDelete: true,
|
|
2945
|
+
historyMaxFileCount: true
|
|
2946
|
+
}
|
|
2947
|
+
}
|
|
2948
|
+
}
|
|
2949
|
+
}]
|
|
2950
|
+
});
|
|
2951
|
+
|
|
2952
|
+
if (!user) {
|
|
2953
|
+
return null;
|
|
2954
|
+
}
|
|
2955
|
+
|
|
2956
|
+
if (user.__typename === "ErrorNotFound") {
|
|
2957
|
+
return user;
|
|
2958
|
+
}
|
|
2959
|
+
|
|
2960
|
+
if (!((_user$user = user.user) != null && _user$user.appSettings)) {
|
|
2961
|
+
return null;
|
|
2962
|
+
}
|
|
2963
|
+
|
|
2964
|
+
return {
|
|
2965
|
+
__typename: "SuccessResponse",
|
|
2966
|
+
data: user.user.appSettings
|
|
2967
|
+
};
|
|
2968
|
+
}
|
|
2969
|
+
|
|
2970
|
+
async updateAppSettings(settings) {
|
|
2971
|
+
const {
|
|
2972
|
+
updateAppSettings
|
|
2973
|
+
} = await this.client("mutation")({
|
|
2974
|
+
updateAppSettings: [settings, {
|
|
2975
|
+
"...on ErrorAccessDenied": {
|
|
2976
|
+
__typename: true,
|
|
2977
|
+
message: true
|
|
2978
|
+
},
|
|
2979
|
+
"...on ErrorBasic": {
|
|
2980
|
+
__typename: true,
|
|
2981
|
+
message: true
|
|
2982
|
+
},
|
|
2983
|
+
"...on UpdateAppSettingsResponse": {
|
|
2984
|
+
__typename: true,
|
|
2985
|
+
updateAppSettings: {
|
|
2986
|
+
cloudFileDaysForDelete: true,
|
|
2987
|
+
cloudFolderDaysForDelete: true,
|
|
2988
|
+
historyFileDaysForDelete: true,
|
|
2989
|
+
historyMaxFileCount: true
|
|
2990
|
+
}
|
|
2991
|
+
}
|
|
2992
|
+
}]
|
|
2993
|
+
});
|
|
2994
|
+
|
|
2995
|
+
if (!updateAppSettings) {
|
|
2996
|
+
return null;
|
|
2997
|
+
}
|
|
2998
|
+
|
|
2999
|
+
if (updateAppSettings.__typename === "ErrorAccessDenied") {
|
|
3000
|
+
return updateAppSettings;
|
|
3001
|
+
}
|
|
3002
|
+
|
|
3003
|
+
if (updateAppSettings.__typename === "ErrorBasic") {
|
|
3004
|
+
return updateAppSettings;
|
|
3005
|
+
}
|
|
3006
|
+
|
|
3007
|
+
return {
|
|
3008
|
+
__typename: "SuccessResponse",
|
|
3009
|
+
data: updateAppSettings.updateAppSettings
|
|
3010
|
+
};
|
|
3011
|
+
}
|
|
3012
|
+
|
|
3013
|
+
async receivedMails() {
|
|
3014
|
+
const {
|
|
3015
|
+
user
|
|
3016
|
+
} = await this.client("query", {
|
|
3017
|
+
scalars: {
|
|
3018
|
+
DateTime: {
|
|
3019
|
+
decode: e => new Date(e),
|
|
3020
|
+
encode: e => e.toISOString()
|
|
3021
|
+
}
|
|
3022
|
+
}
|
|
3023
|
+
})({
|
|
3024
|
+
user: [{}, {
|
|
3025
|
+
"...on ErrorNotFound": {
|
|
3026
|
+
__typename: true,
|
|
3027
|
+
message: true
|
|
3028
|
+
},
|
|
3029
|
+
"...on UserResponse": {
|
|
3030
|
+
__typename: true,
|
|
3031
|
+
user: {
|
|
3032
|
+
receivedMails: mailSelector
|
|
3033
|
+
}
|
|
3034
|
+
}
|
|
3035
|
+
}]
|
|
3036
|
+
});
|
|
3037
|
+
|
|
3038
|
+
if (!user) {
|
|
3039
|
+
return null;
|
|
3040
|
+
}
|
|
3041
|
+
|
|
3042
|
+
if (user.__typename === "ErrorNotFound") {
|
|
3043
|
+
return user;
|
|
3044
|
+
}
|
|
3045
|
+
|
|
3046
|
+
if (!user.user) {
|
|
3047
|
+
return null;
|
|
3048
|
+
} // TODO get actual mails on this app only
|
|
3049
|
+
|
|
3050
|
+
|
|
3051
|
+
const receivedMails = new Array();
|
|
3052
|
+
|
|
3053
|
+
for (const m of user.user.receivedMails) {
|
|
3054
|
+
const mail = convertInternalMailToExternal(m, _classPrivateFieldLooseBase(this, _keys)[_keys]);
|
|
3055
|
+
|
|
3056
|
+
if (mail) {
|
|
3057
|
+
receivedMails.push(mail);
|
|
3058
|
+
}
|
|
3059
|
+
}
|
|
3060
|
+
|
|
3061
|
+
return {
|
|
3062
|
+
__typename: "SuccessResponse",
|
|
3063
|
+
data: receivedMails
|
|
3064
|
+
};
|
|
3065
|
+
}
|
|
3066
|
+
|
|
3067
|
+
async sentMails() {
|
|
3068
|
+
const {
|
|
3069
|
+
user
|
|
3070
|
+
} = await this.client("query", {
|
|
3071
|
+
scalars: {
|
|
3072
|
+
DateTime: {
|
|
3073
|
+
decode: e => new Date(e),
|
|
3074
|
+
encode: e => e.toISOString()
|
|
3075
|
+
}
|
|
3076
|
+
}
|
|
3077
|
+
})({
|
|
3078
|
+
user: [{}, {
|
|
3079
|
+
"...on ErrorNotFound": {
|
|
3080
|
+
__typename: true,
|
|
3081
|
+
message: true
|
|
3082
|
+
},
|
|
3083
|
+
"...on UserResponse": {
|
|
3084
|
+
__typename: true,
|
|
3085
|
+
user: {
|
|
3086
|
+
sentMails: mailSelector
|
|
3087
|
+
}
|
|
3088
|
+
}
|
|
3089
|
+
}]
|
|
3090
|
+
});
|
|
3091
|
+
|
|
3092
|
+
if (!user) {
|
|
3093
|
+
return null;
|
|
3094
|
+
}
|
|
3095
|
+
|
|
3096
|
+
if (user.__typename === "ErrorNotFound") {
|
|
3097
|
+
return user;
|
|
3098
|
+
}
|
|
3099
|
+
|
|
3100
|
+
if (!user.user) {
|
|
3101
|
+
return null;
|
|
3102
|
+
} // TODO get actual mails on this app only
|
|
3103
|
+
|
|
3104
|
+
|
|
3105
|
+
const sentMails = new Array();
|
|
3106
|
+
|
|
3107
|
+
for (const m of user.user.sentMails) {
|
|
3108
|
+
const mail = convertInternalMailToExternal(m, _classPrivateFieldLooseBase(this, _keys)[_keys]);
|
|
3109
|
+
|
|
3110
|
+
if (mail) {
|
|
3111
|
+
sentMails.push(mail);
|
|
3112
|
+
}
|
|
3113
|
+
}
|
|
3114
|
+
|
|
3115
|
+
return {
|
|
3116
|
+
__typename: "SuccessResponse",
|
|
3117
|
+
data: sentMails
|
|
3118
|
+
};
|
|
3119
|
+
}
|
|
3120
|
+
|
|
3121
|
+
async draftMails() {
|
|
3122
|
+
const {
|
|
3123
|
+
user
|
|
3124
|
+
} = await this.client("query", {
|
|
3125
|
+
scalars: {
|
|
3126
|
+
DateTime: {
|
|
3127
|
+
decode: e => new Date(e),
|
|
3128
|
+
encode: e => e.toISOString()
|
|
3129
|
+
}
|
|
3130
|
+
}
|
|
3131
|
+
})({
|
|
3132
|
+
user: [{}, {
|
|
3133
|
+
"...on ErrorNotFound": {
|
|
3134
|
+
__typename: true,
|
|
3135
|
+
message: true
|
|
3136
|
+
},
|
|
3137
|
+
"...on UserResponse": {
|
|
3138
|
+
__typename: true,
|
|
3139
|
+
user: {
|
|
3140
|
+
draftMails: mailSelector
|
|
3141
|
+
}
|
|
3142
|
+
}
|
|
3143
|
+
}]
|
|
3144
|
+
});
|
|
3145
|
+
|
|
3146
|
+
if (!user) {
|
|
3147
|
+
return null;
|
|
3148
|
+
}
|
|
3149
|
+
|
|
3150
|
+
if (user.__typename === "ErrorNotFound") {
|
|
3151
|
+
return user;
|
|
3152
|
+
}
|
|
3153
|
+
|
|
3154
|
+
if (!user.user) {
|
|
3155
|
+
return null;
|
|
3156
|
+
} // TODO get actual mails on this app only
|
|
3157
|
+
|
|
3158
|
+
|
|
3159
|
+
const draftMails = new Array();
|
|
3160
|
+
|
|
3161
|
+
for (const m of user.user.draftMails) {
|
|
3162
|
+
const draft = convertInternalMailToExternal(m, _classPrivateFieldLooseBase(this, _keys)[_keys]);
|
|
3163
|
+
|
|
3164
|
+
if (draft) {
|
|
3165
|
+
draftMails.push(draft);
|
|
3166
|
+
}
|
|
3167
|
+
}
|
|
3168
|
+
|
|
3169
|
+
return {
|
|
3170
|
+
__typename: "SuccessResponse",
|
|
3171
|
+
data: draftMails
|
|
3172
|
+
};
|
|
3173
|
+
}
|
|
3174
|
+
|
|
3175
|
+
async unreadReceivedMailsCount() {
|
|
3176
|
+
const {
|
|
3177
|
+
unreadReceivedMailsCount
|
|
3178
|
+
} = await this.client("query")({
|
|
3179
|
+
unreadReceivedMailsCount: {
|
|
3180
|
+
"...on ErrorAccessDenied": {
|
|
3181
|
+
__typename: true,
|
|
3182
|
+
message: true
|
|
3183
|
+
},
|
|
3184
|
+
"...on UnreadReceivedMailsCountResponse": {
|
|
3185
|
+
__typename: true,
|
|
3186
|
+
count: true
|
|
3187
|
+
}
|
|
3188
|
+
}
|
|
3189
|
+
});
|
|
3190
|
+
|
|
3191
|
+
if (!unreadReceivedMailsCount) {
|
|
3192
|
+
return null;
|
|
3193
|
+
}
|
|
3194
|
+
|
|
3195
|
+
if (unreadReceivedMailsCount.__typename === "ErrorAccessDenied") {
|
|
3196
|
+
return unreadReceivedMailsCount;
|
|
3197
|
+
}
|
|
3198
|
+
|
|
3199
|
+
return {
|
|
3200
|
+
__typename: "SuccessResponse",
|
|
3201
|
+
data: unreadReceivedMailsCount.count
|
|
3202
|
+
};
|
|
3203
|
+
}
|
|
3204
|
+
|
|
3205
|
+
}
|
|
3206
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["ky","axios","BaseClient","nodesCache","filesCache","usersCache","secretstreamKeygen","encryptSecretstream","compress","uncompress","gqlNodeToExternal","gqlNodeToExternalNodeFull","gqlNodeToInternal","internalNodeToNode","gqlFileToExternal","sodium","decryptCryptoBox","encryptCryptoBox","generate","md5","encrypt","decrypt","chunks","concatenate","enumerate","promiseAllLimit","fileSelector","mailSelector","nodeFullSelector","nodeSelector","convertInternalMailToExternal","decode","serialize","encryptName","name","nameKey","data","from_hex","from_string","nameEncrypted","to_hex","SecrecyClient","constructor","uaSession","uaKeys","uaJwt","env","perNode","nodeId","publicKey","node","get","id","access","privateKey","files","history","map","f","key","_eachUser","subject","body","idOrMail","u","req","user","userId","withPublicKey","__typename","Error","recipientsFiles","Array","fileInHistory","file","push","fileKey","recipientId","jwt","jwtDecoded","appUserId","sub","addFileToHistory","fileId","client","message","find","users","filter","input","shareFileInHistory","result","uploadFile","encryptProgress","uploadProgress","signal","fileBuffer","File","Uint8Array","arrayBuffer","compressed","encryptedFile","md5File","md5Encrypted","encryptedFileKey","scalars","Json","encode","e","JSON","stringify","parse","DateTime","Date","toISOString","BigInt","toString","fileSize","byteLength","fileSizeBefore","filePartSize","parts","fields","order","url","total","current","percent","length","uploadPartEnded","uploadFilePartEnd","uploadEnded","uploadFileEnd","chunkParts","index","chunk","Number","progressParts","onProgress","part","progressEvent","Object","values","reduce","prv","cur","loaded","byPart","formData","FormData","p","value","entries","append","Blob","post","onUploadProgress","uploadFileInCloud","saveInCloud","logout","sessionId","clear","createFolder","parentFolderId","encryptedName","encryptedKey","folder","parent","Promise","all","rights","shareNode","deleted","mail","deletedNodes","nodes","sharedNodes","nodesSharedWithMe","type","deleteNodeSharing","duplicateNode","folderId","customName","deleteNodeCloudTrash","ids","shareNodes","shareNodeFinish","updateNode","isFavorite","deletedAt","deletedMails","mailType","mails","m","updateAppNotifications","notifications","enableAll","cloud","disableAllUntil","appNotifications","createMail","customMessage","createDraftMail","sendDraftMail","mailIntegrityId","waitingReceivedMails","date","attachmentsCount","sender","firstname","lastname","email","recipients","temporaryRecipients","updateDraftMail","draftId","recipientsIds","replyTo","drafts","draftMails","draft","d","hashKey","hash","randombytes_buf","crypto_generichash_KEYBYTES","crypto_generichash","crypto_generichash_BYTES","senderFiles","deleteDraftMail","deleteMailTrash","emptyMailTrash","emptyCloudTrash","recoverNode","recoverMail","mailId","deleteFile","deleteNode","moveNodes","nodeIds","parentNodeId","folderSize","size","sizeBefore","deleteMail","has","receivedMails","some","fileMail","encryptedNameKey","filename","me","includes","others","dbGet","field","json","res","dbSet","dbSearch","search","draftMailId","sendWaitingEmails","sentMails","filtered","sendOneMail","recipient","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","readMail","unreadMail","appSettings","cloudFileDaysForDelete","cloudFolderDaysForDelete","historyFileDaysForDelete","historyMaxFileCount","updateAppSettings","settings","unreadReceivedMailsCount","count"],"sources":["../../src/client/index.ts"],"sourcesContent":["/* eslint-disable @typescript-eslint/naming-convention */\nimport type { MailType, NodeType } from \"./../zeus/index\";\nimport type { DownloadProgress } from \"ky\";\nimport ky from \"ky\";\nimport axios from \"axios\";\nimport type { FolderSize, SuccessResponse, UserData } from \"../BaseClient.js\";\nimport { BaseClient } from \"../BaseClient.js\";\nimport { nodesCache, filesCache, usersCache } from \"../cache.js\";\nimport type { Progress } from \"../crypto/file.js\";\nimport { secretstreamKeygen, encryptSecretstream } from \"../crypto/file.js\";\nimport { compress, uncompress } from \"../minify/index.js\";\nimport {\n  gqlNodeToExternal,\n  gqlNodeToExternalNodeFull,\n  gqlNodeToInternal,\n  internalNodeToNode\n} from \"./convert/node.js\";\nimport type {\n  MailFileInput,\n  ReceivedMail,\n  SentMail,\n  Node,\n  NodeFull,\n  UserAppSettings,\n  UserAppNotifications,\n  FileMetadata as SecrecyFile,\n  WaitingReceivedMail,\n  DraftMail,\n  MailRecipientInput,\n  ShareNodesInput,\n  NameKeyInput,\n  Mail,\n  ShareFileInHistoryInput,\n  FilePartResponse,\n  FileContentPart\n} from \"./types/index.js\";\nimport { gqlFileToExternal } from \"./convert/file.js\";\nimport { sodium } from \"../sodium.js\";\nimport type { KeyPair } from \"../crypto/index.js\";\nimport { decryptCryptoBox, encryptCryptoBox } from \"../crypto/index.js\";\nimport type { SecrecyEnv } from \"./helpers.js\";\nimport { generate } from \"shortid\";\nimport { md5 } from \"../worker/md5.js\";\nimport { encrypt, decrypt } from \"../worker/sodium.js\";\nimport {\n  chunks,\n  concatenate,\n  enumerate,\n  promiseAllLimit\n} from \"../utils/utils.js\";\nimport {\n  fileSelector,\n  mailSelector,\n  nodeFullSelector,\n  nodeSelector\n} from \"./types/selectors.js\";\nimport type { Rights } from \"../zeus/index.js\";\nimport { convertInternalMailToExternal } from \"./convert/mail.js\";\nimport type { JwtPayload } from \"jsonwebtoken\";\nimport { decode } from \"jsonwebtoken\";\n// import { md5 } from \"../worker/index.js\";\n// import { firstValueFrom, of } from \"rxjs\";\n\nimport type {\n  ErrorAccessDenied,\n  ErrorNotFound,\n  ErrorBasic,\n  ErrorNotExist,\n  ErrorLimit\n} from \"../error.js\";\nimport { serialize } from \"bson\";\n\nexport type NewMail = {\n  body: string;\n  subject: string;\n  files: { id: string; name: string }[];\n  recipientsIds: string[];\n  replyTo?: string | null | undefined;\n};\nexport type ProgressCallback = (progress: Progress) => Promise<void>;\n\nconst encryptName = async (name: string, nameKey: string): Promise<string> => {\n  const { data } = await encryptSecretstream(\n    sodium.from_hex(nameKey),\n    sodium.from_string(name)\n  );\n  const nameEncrypted = sodium.to_hex(data);\n  return nameEncrypted;\n};\n\nexport class SecrecyClient extends BaseClient {\n  jwt: string;\n\n  jwtDecoded: JwtPayload;\n\n  #keys: KeyPair;\n\n  constructor(\n    uaSession: string,\n    uaKeys: KeyPair,\n    uaJwt: string,\n    env: SecrecyEnv\n  ) {\n    super(uaSession, env);\n    this.jwt = uaJwt;\n    this.jwtDecoded = decode(uaJwt) as JwtPayload;\n    this.#keys = uaKeys;\n  }\n\n  get publicKey(): string {\n    return this.#keys.publicKey;\n  }\n\n  get appUserId(): string {\n    return this.jwtDecoded.sub ?? \"\";\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.client(\"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: nodeSelector\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 gqlNodeToInternal(\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.client(\"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.client(\"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.client(\"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.client(\"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 logout(sessionId?: string | null | undefined): Promise<void> {\n    nodesCache.clear();\n    filesCache.clear();\n    await super.logout(sessionId);\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.client(\"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.client(\"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<SecrecyFile> | ErrorAccessDenied | null> {\n    const { file } = await this.client(\"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 mail({\n    id\n  }: {\n    id: string;\n  }): Promise<SuccessResponse<Mail> | ErrorAccessDenied | null> {\n    const { mail } = await this.client(\"query\", {\n      scalars: {\n        DateTime: {\n          decode: (e: unknown) => new Date(e as string),\n          encode: (e: unknown) => (e as Date).toISOString()\n        }\n      }\n    })({\n      mail: [\n        {\n          id\n        },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on QueryMailResponse\": {\n            __typename: true,\n            mail: mailSelector\n          }\n        }\n      ]\n    });\n\n    if (!mail) {\n      return null;\n    }\n\n    if (mail.__typename === \"ErrorAccessDenied\") {\n      return mail;\n    }\n\n    if (!mail.mail) {\n      return null;\n    }\n\n    const result = convertInternalMailToExternal(mail.mail, this.#keys);\n\n    if (!result) {\n      return null;\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: result\n    };\n  }\n\n  async deletedNodes(): Promise<\n    SuccessResponse<Node[]> | ErrorAccessDenied | null\n  > {\n    const { deletedNodes } = await this.client(\"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.client(\"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.client(\"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.client(\"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.client(\"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.client(\"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 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.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.client(\"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.client(\"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.client(\"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 deletedMails({\n    mailType\n  }: {\n    mailType: MailType;\n  }): Promise<SuccessResponse<Mail[]> | ErrorAccessDenied | null> {\n    const { deletedMails } = await this.client(\"query\", {\n      scalars: {\n        DateTime: {\n          decode: (e: unknown) => new Date(e as string),\n          encode: (e: unknown) => (e as Date).toISOString()\n        }\n      }\n    })({\n      deletedMails: [\n        { mailType },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on DeletedMailsResponse\": {\n            __typename: true,\n            deletedMails: mailSelector\n          }\n        }\n      ]\n    });\n\n    if (!deletedMails) {\n      return null;\n    }\n\n    if (deletedMails.__typename === \"ErrorAccessDenied\") {\n      return deletedMails;\n    }\n\n    const mails = new Array<Mail>();\n\n    for (const m of deletedMails.deletedMails) {\n      const mail = convertInternalMailToExternal(m, this.#keys);\n      if (mail) {\n        mails.push(mail);\n      }\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: mails\n    };\n  }\n\n  async updateAppNotifications(\n    notifications: Partial<UserAppNotifications>\n  ): Promise<SuccessResponse<UserAppNotifications> | ErrorAccessDenied | null> {\n    const { updateAppNotifications } = await this.client(\"mutation\", {\n      scalars: {\n        DateTime: {\n          decode: (e: unknown) => new Date(e as string),\n          encode: (e: unknown) => (e as Date).toISOString()\n        }\n      }\n    })({\n      updateAppNotifications: [\n        notifications,\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on UpdateAppNotificationsResponse\": {\n            __typename: true,\n            updateAppNotifications: {\n              enableAll: true,\n              mail: true,\n              cloud: true,\n              disableAllUntil: true\n            }\n          }\n        }\n      ]\n    });\n\n    if (!updateAppNotifications) {\n      return null;\n    }\n\n    if (updateAppNotifications.__typename === \"ErrorAccessDenied\") {\n      return updateAppNotifications;\n    }\n\n    if (!updateAppNotifications.updateAppNotifications) {\n      return null;\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: updateAppNotifications.updateAppNotifications\n    };\n  }\n\n  async appNotifications(): Promise<\n    | SuccessResponse<UserAppNotifications>\n    | ErrorAccessDenied\n    | ErrorNotFound\n    | null\n  > {\n    const { appNotifications } = await this.client(\"query\", {\n      scalars: {\n        DateTime: {\n          decode: (e: unknown) => new Date(e as string),\n          encode: (e: unknown) => (e as Date).toISOString()\n        }\n      }\n    })({\n      appNotifications: {\n        \"...on ErrorAccessDenied\": {\n          __typename: true,\n          message: true\n        },\n        \"...on ErrorNotFound\": {\n          __typename: true,\n          message: true\n        },\n        \"...on UserAppNotifications\": {\n          __typename: true,\n          enableAll: true,\n          mail: true,\n          cloud: true,\n          disableAllUntil: true\n        }\n      }\n    });\n\n    if (!appNotifications) {\n      return null;\n    }\n\n    if (appNotifications.__typename === \"ErrorAccessDenied\") {\n      return appNotifications;\n    }\n\n    if (appNotifications.__typename === \"ErrorNotFound\") {\n      return appNotifications;\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: appNotifications\n    };\n  }\n\n  async createMail(\n    data: NewMail,\n    customMessage?: string | null | undefined\n  ): Promise<SuccessResponse<boolean> | ErrorBasic | ErrorAccessDenied | null> {\n    const mail = await this.createDraftMail(data);\n    if (!mail) {\n      return {\n        __typename: \"SuccessResponse\",\n        data: false\n      };\n    }\n\n    if (mail.__typename === \"ErrorBasic\") {\n      return mail;\n    }\n\n    if (mail.__typename === \"ErrorAccessDenied\") {\n      return mail;\n    }\n\n    const result = await this.sendDraftMail(\n      mail.data.mailIntegrityId,\n      customMessage\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    return {\n      __typename: \"SuccessResponse\",\n      data: result.data\n    };\n  }\n\n  async waitingReceivedMails(): Promise<\n    SuccessResponse<WaitingReceivedMail[]> | ErrorNotFound | null\n  > {\n    const { user } = await this.client(\"query\")({\n      user: [\n        {},\n        {\n          \"...on ErrorNotFound\": {\n            __typename: true,\n            message: true\n          },\n          \"...on UserResponse\": {\n            __typename: true,\n            user: {\n              waitingReceivedMails: {\n                date: true,\n                attachmentsCount: true,\n                sender: {\n                  id: true,\n                  firstname: true,\n                  lastname: true,\n                  email: true,\n                  publicKey: true\n                },\n                recipients: {\n                  id: true,\n                  firstname: true,\n                  lastname: true,\n                  email: true,\n                  publicKey: true\n                },\n                temporaryRecipients: {\n                  email: true\n                }\n              }\n            }\n          }\n        }\n      ]\n    });\n\n    if (!user) {\n      return null;\n    }\n\n    if (user.__typename === \"ErrorNotFound\") {\n      return user;\n    }\n\n    if (!user.user) {\n      return null;\n    }\n\n    const result = user.user.waitingReceivedMails.map(m => ({\n      id: generate(),\n      ...m,\n      date: new Date(m.date as string)\n    }));\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: result\n    };\n  }\n\n  async updateDraftMail(\n    draftId: string,\n    { body, subject, files, recipientsIds, replyTo }: Partial<NewMail>\n  ): Promise<\n    | SuccessResponse<DraftMail>\n    | ErrorNotFound\n    | ErrorAccessDenied\n    | ErrorBasic\n    | null\n  > {\n    const drafts = await this.draftMails();\n    if (!drafts) {\n      return null;\n    }\n\n    if (drafts.__typename !== \"SuccessResponse\") {\n      return drafts;\n    }\n    const draft = drafts.data.find(d => d.mailIntegrityId === draftId);\n    if (!draft) {\n      throw new Error(`Invalid draft ${draftId}`);\n    }\n    let hashKey: string | null = null;\n    let hash: string | null = null;\n    if (body || subject) {\n      hashKey = sodium.randombytes_buf(\n        sodium.crypto_generichash_KEYBYTES,\n        \"hex\"\n      );\n      hash = sodium.crypto_generichash(\n        sodium.crypto_generichash_BYTES,\n        JSON.stringify({ body, subject }),\n        hashKey,\n        \"hex\"\n      );\n    }\n    const senderFiles = new Array<MailFileInput>();\n    if (files) {\n      for (const f of files) {\n        let file = filesCache.get(f.id);\n        if (!file) {\n          await this.file({ id: f.id });\n          file = filesCache.get(f.id);\n          if (!file) {\n            throw new Error(`File ${f.name} (${f.id}) does not exists`);\n          }\n        }\n        senderFiles.push({\n          id: file.id,\n          fileKey: sodium.to_hex(\n            encryptCryptoBox(\n              sodium.from_string(file.key),\n              this.#keys.publicKey,\n              this.#keys.privateKey\n            )\n          ),\n          name: sodium.to_hex(\n            encryptCryptoBox(\n              sodium.from_string(f.name),\n              this.#keys.publicKey,\n              this.#keys.privateKey\n            )\n          )\n        });\n      }\n    }\n\n    const { updateDraftMail } = await this.client(\"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      updateDraftMail: [\n        {\n          draftId,\n          recipients: recipientsIds,\n          replyTo,\n          body: body\n            ? sodium.to_hex(\n                encryptCryptoBox(\n                  sodium.from_string(body),\n                  this.#keys.publicKey,\n                  this.#keys.privateKey\n                )\n              )\n            : null,\n          subject: subject\n            ? sodium.to_hex(\n                encryptCryptoBox(\n                  sodium.from_string(subject),\n                  this.#keys.publicKey,\n                  this.#keys.privateKey\n                )\n              )\n            : null,\n          senderFiles,\n          hash,\n          hashKey\n        },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on ErrorBasic\": {\n            __typename: true,\n            message: true\n          },\n          \"...on UpdateDraftMailResponse\": {\n            __typename: true,\n            updateDraftMail: mailSelector\n          }\n        }\n      ]\n    });\n\n    if (!updateDraftMail) {\n      return null;\n    }\n\n    if (updateDraftMail.__typename === \"ErrorAccessDenied\") {\n      return updateDraftMail;\n    }\n\n    if (updateDraftMail.__typename === \"ErrorBasic\") {\n      return updateDraftMail;\n    }\n\n    if (!updateDraftMail.updateDraftMail) {\n      return null;\n    }\n\n    const result = convertInternalMailToExternal(\n      updateDraftMail.updateDraftMail,\n      this.#keys\n    ) as DraftMail;\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: result\n    };\n  }\n\n  async deleteDraftMail(\n    draftId: string\n  ): Promise<SuccessResponse<boolean> | ErrorAccessDenied | null> {\n    const { deleteDraftMail } = await this.client(\"mutation\")({\n      deleteDraftMail: [\n        {\n          draftId\n        },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on DeleteDraftMailResponse\": {\n            __typename: true,\n            deleteDraftMail: true\n          }\n        }\n      ]\n    });\n\n    if (!deleteDraftMail) {\n      return null;\n    }\n\n    if (deleteDraftMail.__typename === \"ErrorAccessDenied\") {\n      return deleteDraftMail;\n    }\n\n    if (!deleteDraftMail.deleteDraftMail) {\n      return null;\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: deleteDraftMail.deleteDraftMail ?? false\n    };\n  }\n\n  async deleteMailTrash({\n    ids\n  }: {\n    ids: Array<string>;\n  }): Promise<SuccessResponse<boolean> | ErrorAccessDenied | null> {\n    const { deleteMailTrash } = await this.client(\"mutation\")({\n      deleteMailTrash: [\n        {\n          ids\n        },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on DeleteMailTrashResponse\": {\n            __typename: true,\n            deleteMailTrash: true\n          }\n        }\n      ]\n    });\n\n    if (!deleteMailTrash) {\n      return null;\n    }\n\n    if (deleteMailTrash.__typename === \"ErrorAccessDenied\") {\n      return deleteMailTrash;\n    }\n\n    if (!deleteMailTrash.deleteMailTrash) {\n      return null;\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: deleteMailTrash.deleteMailTrash\n    };\n  }\n\n  async emptyMailTrash(): Promise<\n    SuccessResponse<boolean> | ErrorAccessDenied | null\n  > {\n    const { emptyMailTrash } = await this.client(\"mutation\")({\n      emptyMailTrash: {\n        \"...on ErrorAccessDenied\": {\n          __typename: true,\n          message: true\n        },\n        \"...on EmptyMailTrashResponse\": {\n          __typename: true,\n          emptyMailTrash: true\n        }\n      }\n    });\n\n    if (!emptyMailTrash) {\n      return null;\n    }\n\n    if (emptyMailTrash.__typename === \"ErrorAccessDenied\") {\n      return emptyMailTrash;\n    }\n\n    if (!emptyMailTrash.emptyMailTrash) {\n      return null;\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: emptyMailTrash.emptyMailTrash\n    };\n  }\n\n  async emptyCloudTrash(): Promise<\n    SuccessResponse<boolean> | ErrorAccessDenied | null\n  > {\n    const { emptyCloudTrash } = await this.client(\"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.client(\"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 recoverMail({\n    mailId\n  }: {\n    mailId: string;\n  }): Promise<\n    SuccessResponse<boolean> | ErrorAccessDenied | ErrorBasic | null\n  > {\n    const { recoverMail } = await this.client(\"mutation\")({\n      recoverMail: [\n        { mailId },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on ErrorBasic\": {\n            __typename: true,\n            message: true\n          },\n          \"...on RecoverMailResponse\": {\n            __typename: true,\n            recoverMail: true\n          }\n        }\n      ]\n    });\n\n    if (!recoverMail) {\n      return null;\n    }\n\n    if (recoverMail.__typename === \"ErrorAccessDenied\") {\n      return recoverMail;\n    }\n\n    if (recoverMail.__typename === \"ErrorBasic\") {\n      return recoverMail;\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: recoverMail.recoverMail\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.client(\"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.client(\"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 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.client(\"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 folderSize({\n    folderId\n  }: {\n    folderId?: string | null | undefined;\n  }): Promise<SuccessResponse<FolderSize> | ErrorAccessDenied | null> {\n    const { folderSize } = await this.client(\"query\", {\n      scalars: {\n        BigInt: {\n          decode: (e: unknown) => BigInt(e as string),\n          encode: (e: unknown) => (e as bigint).toString()\n        }\n      }\n    })({\n      folderSize: [\n        {\n          folderId\n        },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on FolderSizeResponse\": {\n            __typename: true,\n            size: true,\n            sizeBefore: true\n          }\n        }\n      ]\n    });\n\n    if (!folderSize) {\n      return null;\n    }\n\n    if (folderSize.__typename === \"ErrorAccessDenied\") {\n      return folderSize;\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: {\n        size: BigInt(folderSize.size),\n        sizeBefore: BigInt(folderSize.sizeBefore)\n      }\n    };\n  }\n\n  async deleteMail({\n    mailId\n  }: {\n    mailId: string;\n  }): Promise<SuccessResponse<boolean> | ErrorAccessDenied | null> {\n    const { deleteMail } = await this.client(\"mutation\")({\n      deleteMail: [\n        {\n          mailId\n        },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on DeleteMailResponse\": {\n            __typename: true,\n            deleteMail: true\n          }\n        }\n      ]\n    });\n\n    if (!deleteMail) {\n      return null;\n    }\n\n    if (deleteMail.__typename === \"ErrorAccessDenied\") {\n      return deleteMail;\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: deleteMail.deleteMail\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.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.client(\"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  async dbGet<U>({\n    field,\n    userId\n  }: {\n    field: string;\n    userId?: string | null | undefined;\n  }): Promise<\n    | SuccessResponse<U>\n    | ErrorAccessDenied\n    | ErrorNotExist\n    | ErrorNotFound\n    | null\n  > {\n    const { dbGet } = await this.client(\"query\")({\n      dbGet: [\n        {\n          field,\n          userId\n        },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on ErrorNotExist\": {\n            __typename: true,\n            message: true\n          },\n          \"...on ErrorNotFound\": {\n            __typename: true,\n            message: true\n          },\n          \"...on DbGetResponse\": {\n            __typename: true,\n            json: true\n          }\n        }\n      ]\n    });\n\n    if (!dbGet) {\n      return null;\n    }\n\n    if (dbGet.__typename === \"ErrorAccessDenied\") {\n      return dbGet;\n    }\n\n    if (dbGet.__typename === \"ErrorNotExist\") {\n      return dbGet;\n    }\n\n    if (dbGet.__typename === \"ErrorNotFound\") {\n      return dbGet;\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: (dbGet.json as { res: U }).res\n    };\n  }\n\n  async dbSet<T extends UserData, U extends Document>({\n    value,\n    userId\n  }: {\n    value: U;\n    userId?: string | null | undefined;\n  }): Promise<SuccessResponse<T> | ErrorAccessDenied | ErrorNotFound | null> {\n    if (typeof value !== \"object\") {\n      throw new Error(\n        `value should be an object including fields you want to update.`\n      );\n    }\n\n    const { dbSet } = await this.client(\"mutation\")({\n      dbSet: [\n        {\n          userId,\n          value: serialize(value).toString(\"base64\")\n        },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on ErrorNotFound\": {\n            __typename: true,\n            message: true\n          },\n          \"...on DbSetResponse\": {\n            __typename: true,\n            dbSet: true\n          }\n        }\n      ]\n    });\n\n    if (!dbSet) {\n      return null;\n    }\n\n    if (dbSet.__typename === \"ErrorAccessDenied\") {\n      return dbSet;\n    }\n\n    if (dbSet.__typename === \"ErrorNotFound\") {\n      return dbSet;\n    }\n\n    if (!dbSet.dbSet) {\n      return null;\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: (dbSet.dbSet as { res: T }).res\n    };\n  }\n\n  async dbSearch<T>({\n    search,\n    field\n  }: {\n    field: string;\n    search: string;\n  }): Promise<\n    SuccessResponse<T[]> | ErrorAccessDenied | ErrorBasic | ErrorNotExist | null\n  > {\n    const { dbSearch } = await this.client(\"query\", {\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      dbSearch: [\n        {\n          search,\n          field\n        },\n        {\n          \"...on ErrorAccessDenied\": {\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 DbSearchResponse\": {\n            __typename: true,\n            json: true\n          }\n        }\n      ]\n    });\n\n    if (!dbSearch) {\n      return null;\n    }\n\n    if (dbSearch.__typename === \"ErrorAccessDenied\") {\n      return dbSearch;\n    }\n\n    if (dbSearch.__typename === \"ErrorBasic\") {\n      return dbSearch;\n    }\n\n    if (dbSearch.__typename === \"ErrorNotExist\") {\n      return dbSearch;\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: dbSearch.json as unknown as T[]\n    };\n  }\n\n  async sendDraftMail(\n    draftId: string,\n    customMessage?: string | null | undefined\n  ): Promise<SuccessResponse<boolean> | ErrorAccessDenied | ErrorBasic | null> {\n    const drafts = await this.draftMails();\n\n    if (!drafts) {\n      return {\n        __typename: \"SuccessResponse\",\n        data: false\n      };\n    }\n    if (drafts.__typename !== \"SuccessResponse\") {\n      return {\n        __typename: \"SuccessResponse\",\n        data: false\n      };\n    }\n\n    const draft = drafts.data.find(d => d.mailIntegrityId === draftId);\n    if (!draft) {\n      return {\n        __typename: \"SuccessResponse\",\n        data: false\n      };\n    }\n    const recipients = new Array<MailRecipientInput>();\n\n    const temporaryRecipients = new Array<string>();\n\n    for (const { email } of draft.temporaryRecipients) {\n      if (!email) {\n        continue;\n      }\n      const input = await this._eachUser(\n        draft.files,\n        draft.subject,\n        draft.body,\n        email\n      );\n\n      if (!input) {\n        temporaryRecipients.push(email);\n      } else {\n        recipients.push(input);\n      }\n    }\n\n    for (const { id } of draft.recipients) {\n      const input = await this._eachUser(\n        draft.files,\n        draft.subject,\n        draft.body,\n        id\n      );\n\n      if (!input) {\n        temporaryRecipients.push(id);\n      } else {\n        recipients.push(input);\n      }\n    }\n\n    const { sendDraftMail } = await this.client(\"mutation\")({\n      sendDraftMail: [\n        {\n          temporaryRecipients,\n          recipients,\n          draftMailId: draft.mailIntegrityId,\n          customMessage\n        },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on ErrorBasic\": {\n            __typename: true,\n            message: true\n          },\n          \"...on SendDraftMailResponse\": {\n            __typename: true,\n            sendDraftMail: true\n          }\n        }\n      ]\n    });\n\n    if (!sendDraftMail) {\n      return null;\n    }\n\n    if (sendDraftMail.__typename === \"ErrorAccessDenied\") {\n      return sendDraftMail;\n    }\n\n    if (sendDraftMail.__typename === \"ErrorBasic\") {\n      return sendDraftMail;\n    }\n\n    if (!sendDraftMail.sendDraftMail) {\n      return null;\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: sendDraftMail.sendDraftMail\n    };\n  }\n\n  async sendWaitingEmails(): Promise<\n    SuccessResponse<boolean> | ErrorAccessDenied | ErrorBasic | null\n  > {\n    // TODO opti this\n    const mails = await this.sentMails();\n\n    if (!mails) {\n      return {\n        __typename: \"SuccessResponse\",\n        data: false\n      };\n    }\n    if (mails.__typename !== \"SuccessResponse\") {\n      return {\n        __typename: \"SuccessResponse\",\n        data: false\n      };\n    }\n    const filtered = mails.data.filter(m => m.temporaryRecipients.length > 0);\n\n    for (const mail of filtered) {\n      for (const { email } of mail.temporaryRecipients) {\n        if (!email) {\n          continue;\n        }\n        try {\n          const input = await this._eachUser(\n            mail.files,\n            mail.subject,\n            mail.body,\n            email\n          );\n\n          if (!input) {\n            continue;\n          }\n          await this.client(\"mutation\")({\n            sendOneMail: [\n              {\n                mailIntegrityId: mail.mailIntegrityId,\n                recipient: input\n              },\n              {\n                \"...on ErrorAccessDenied\": {\n                  __typename: true,\n                  message: true\n                },\n                \"...on ErrorBasic\": {\n                  __typename: true,\n                  message: true\n                },\n                \"...on RecoverNodeResponse\": {\n                  __typename: true,\n                  recoverNode: true\n                }\n              }\n            ]\n          });\n        } catch {\n          continue;\n        }\n      }\n    }\n    return {\n      __typename: \"SuccessResponse\",\n      data: false\n    };\n  }\n\n  async createDraftMail({\n    body,\n    subject,\n    files,\n    recipientsIds,\n    replyTo\n  }: NewMail): Promise<\n    SuccessResponse<DraftMail> | ErrorAccessDenied | ErrorBasic | null\n  > {\n    const hashKey = sodium.randombytes_buf(\n      sodium.crypto_generichash_KEYBYTES,\n      \"hex\"\n    );\n    const hash = sodium.crypto_generichash(\n      sodium.crypto_generichash_BYTES,\n      JSON.stringify({ body, subject }),\n      hashKey,\n      \"hex\"\n    );\n    const senderFiles = new Array<MailFileInput>();\n    for (const f of files) {\n      let file = filesCache.get(f.id);\n      if (!file) {\n        await this.file({ id: f.id });\n        file = filesCache.get(f.id);\n        if (!file) {\n          throw new Error(`File ${f.name} (${f.id}) does not exists`);\n        }\n      }\n      senderFiles.push({\n        id: file.id,\n        fileKey: sodium.to_hex(\n          encryptCryptoBox(\n            sodium.from_string(file.key),\n            this.#keys.publicKey,\n            this.#keys.privateKey\n          )\n        ),\n        name: sodium.to_hex(\n          encryptCryptoBox(\n            sodium.from_string(f.name),\n            this.#keys.publicKey,\n            this.#keys.privateKey\n          )\n        )\n      });\n    }\n\n    const { createDraftMail } = await this.client(\"mutation\", {\n      scalars: {\n        DateTime: {\n          decode: (e: unknown) => new Date(e as string),\n          encode: (e: unknown) => (e as Date).toISOString()\n        }\n      }\n    })({\n      createDraftMail: [\n        {\n          recipients: recipientsIds,\n          replyTo,\n          body: sodium.to_hex(\n            encryptCryptoBox(\n              sodium.from_string(body),\n              this.#keys.publicKey,\n              this.#keys.privateKey\n            )\n          ),\n          subject: sodium.to_hex(\n            encryptCryptoBox(\n              sodium.from_string(subject),\n              this.#keys.publicKey,\n              this.#keys.privateKey\n            )\n          ),\n          senderFiles,\n          hash,\n          hashKey\n        },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on ErrorBasic\": {\n            __typename: true,\n            message: true\n          },\n          \"...on CreateDraftMailResponse\": {\n            __typename: true,\n            createDraftMail: mailSelector\n          }\n        }\n      ]\n    });\n\n    if (!createDraftMail) {\n      return null;\n    }\n\n    if (createDraftMail.__typename === \"ErrorAccessDenied\") {\n      return createDraftMail;\n    }\n\n    if (createDraftMail.__typename === \"ErrorBasic\") {\n      return createDraftMail;\n    }\n\n    if (!createDraftMail.createDraftMail) {\n      return null;\n    }\n\n    const result = convertInternalMailToExternal(\n      createDraftMail.createDraftMail,\n      this.#keys\n    ) as DraftMail;\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.client(\"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 readMail({\n    mailId\n  }: {\n    mailId: string;\n  }): Promise<\n    | SuccessResponse<boolean>\n    | ErrorAccessDenied\n    | ErrorBasic\n    | ErrorNotFound\n    | null\n  > {\n    const { readMail } = await this.client(\"mutation\")({\n      readMail: [\n        { mailId },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on ErrorBasic\": {\n            __typename: true,\n            message: true\n          },\n          \"...on ErrorNotFound\": {\n            __typename: true,\n            message: true\n          },\n          \"...on ReadMailResponse\": {\n            __typename: true,\n            readMail: true\n          }\n        }\n      ]\n    });\n\n    if (!readMail) {\n      return null;\n    }\n\n    if (readMail.__typename === \"ErrorAccessDenied\") {\n      return readMail;\n    }\n\n    if (readMail.__typename === \"ErrorBasic\") {\n      return readMail;\n    }\n\n    if (readMail.__typename === \"ErrorNotFound\") {\n      return readMail;\n    }\n\n    if (!readMail.readMail) {\n      return null;\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: readMail.readMail\n    };\n  }\n\n  async unreadMail({\n    mailId\n  }: {\n    mailId: string;\n  }): Promise<\n    | SuccessResponse<boolean>\n    | ErrorAccessDenied\n    | ErrorBasic\n    | ErrorNotFound\n    | null\n  > {\n    const { unreadMail } = await this.client(\"mutation\")({\n      unreadMail: [\n        { mailId },\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on ErrorBasic\": {\n            __typename: true,\n            message: true\n          },\n          \"...on ErrorNotFound\": {\n            __typename: true,\n            message: true\n          },\n          \"...on UnreadMailResponse\": {\n            __typename: true,\n            unreadMail: true\n          }\n        }\n      ]\n    });\n\n    if (!unreadMail) {\n      return null;\n    }\n\n    if (unreadMail.__typename === \"ErrorAccessDenied\") {\n      return unreadMail;\n    }\n\n    if (unreadMail.__typename === \"ErrorBasic\") {\n      return unreadMail;\n    }\n\n    if (unreadMail.__typename === \"ErrorNotFound\") {\n      return unreadMail;\n    }\n\n    if (!unreadMail.unreadMail) {\n      return null;\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: unreadMail.unreadMail\n    };\n  }\n\n  async appSettings(): Promise<\n    SuccessResponse<UserAppSettings> | ErrorNotFound | null\n  > {\n    const { user } = await this.client(\"query\")({\n      user: [\n        {},\n        {\n          \"...on ErrorNotFound\": {\n            __typename: true,\n            message: true\n          },\n          \"...on UserResponse\": {\n            __typename: true,\n            user: {\n              appSettings: {\n                cloudFileDaysForDelete: true,\n                cloudFolderDaysForDelete: true,\n                historyFileDaysForDelete: true,\n                historyMaxFileCount: true\n              }\n            }\n          }\n        }\n      ]\n    });\n\n    if (!user) {\n      return null;\n    }\n\n    if (user.__typename === \"ErrorNotFound\") {\n      return user;\n    }\n\n    if (!user.user?.appSettings) {\n      return null;\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: user.user.appSettings\n    };\n  }\n\n  async updateAppSettings(\n    settings: Partial<UserAppSettings>\n  ): Promise<\n    SuccessResponse<UserAppSettings> | ErrorAccessDenied | ErrorBasic | null\n  > {\n    const { updateAppSettings } = await this.client(\"mutation\")({\n      updateAppSettings: [\n        settings,\n        {\n          \"...on ErrorAccessDenied\": {\n            __typename: true,\n            message: true\n          },\n          \"...on ErrorBasic\": {\n            __typename: true,\n            message: true\n          },\n          \"...on UpdateAppSettingsResponse\": {\n            __typename: true,\n            updateAppSettings: {\n              cloudFileDaysForDelete: true,\n              cloudFolderDaysForDelete: true,\n              historyFileDaysForDelete: true,\n              historyMaxFileCount: true\n            }\n          }\n        }\n      ]\n    });\n\n    if (!updateAppSettings) {\n      return null;\n    }\n\n    if (updateAppSettings.__typename === \"ErrorAccessDenied\") {\n      return updateAppSettings;\n    }\n\n    if (updateAppSettings.__typename === \"ErrorBasic\") {\n      return updateAppSettings;\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: updateAppSettings.updateAppSettings\n    };\n  }\n\n  async receivedMails(): Promise<\n    SuccessResponse<ReceivedMail[]> | ErrorNotFound | null\n  > {\n    const { user } = await this.client(\"query\", {\n      scalars: {\n        DateTime: {\n          decode: (e: unknown) => new Date(e as string),\n          encode: (e: unknown) => (e as Date).toISOString()\n        }\n      }\n    })({\n      user: [\n        {},\n        {\n          \"...on ErrorNotFound\": {\n            __typename: true,\n            message: true\n          },\n          \"...on UserResponse\": {\n            __typename: true,\n            user: {\n              receivedMails: mailSelector\n            }\n          }\n        }\n      ]\n    });\n\n    if (!user) {\n      return null;\n    }\n\n    if (user.__typename === \"ErrorNotFound\") {\n      return user;\n    }\n\n    if (!user.user) {\n      return null;\n    }\n\n    // TODO get actual mails on this app only\n\n    const receivedMails = new Array<ReceivedMail>();\n\n    for (const m of user.user.receivedMails) {\n      const mail = convertInternalMailToExternal(m, this.#keys);\n      if (mail) {\n        receivedMails.push(mail as ReceivedMail);\n      }\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: receivedMails\n    };\n  }\n\n  async sentMails(): Promise<\n    SuccessResponse<SentMail[]> | ErrorNotFound | null\n  > {\n    const { user } = await this.client(\"query\", {\n      scalars: {\n        DateTime: {\n          decode: (e: unknown) => new Date(e as string),\n          encode: (e: unknown) => (e as Date).toISOString()\n        }\n      }\n    })({\n      user: [\n        {},\n        {\n          \"...on ErrorNotFound\": {\n            __typename: true,\n            message: true\n          },\n          \"...on UserResponse\": {\n            __typename: true,\n            user: {\n              sentMails: mailSelector\n            }\n          }\n        }\n      ]\n    });\n\n    if (!user) {\n      return null;\n    }\n\n    if (user.__typename === \"ErrorNotFound\") {\n      return user;\n    }\n\n    if (!user.user) {\n      return null;\n    }\n\n    // TODO get actual mails on this app only\n\n    const sentMails = new Array<SentMail>();\n\n    for (const m of user.user.sentMails) {\n      const mail = convertInternalMailToExternal(m, this.#keys);\n      if (mail) {\n        sentMails.push(mail as SentMail);\n      }\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: sentMails\n    };\n  }\n\n  async draftMails(): Promise<\n    SuccessResponse<DraftMail[]> | ErrorNotFound | null\n  > {\n    const { user } = await this.client(\"query\", {\n      scalars: {\n        DateTime: {\n          decode: (e: unknown) => new Date(e as string),\n          encode: (e: unknown) => (e as Date).toISOString()\n        }\n      }\n    })({\n      user: [\n        {},\n        {\n          \"...on ErrorNotFound\": {\n            __typename: true,\n            message: true\n          },\n          \"...on UserResponse\": {\n            __typename: true,\n            user: {\n              draftMails: mailSelector\n            }\n          }\n        }\n      ]\n    });\n\n    if (!user) {\n      return null;\n    }\n\n    if (user.__typename === \"ErrorNotFound\") {\n      return user;\n    }\n\n    if (!user.user) {\n      return null;\n    }\n\n    // TODO get actual mails on this app only\n\n    const draftMails = new Array<DraftMail>();\n\n    for (const m of user.user.draftMails) {\n      const draft = convertInternalMailToExternal(m, this.#keys) as DraftMail;\n      if (draft) {\n        draftMails.push(draft);\n      }\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: draftMails\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: 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  async unreadReceivedMailsCount(): Promise<\n    SuccessResponse<number> | ErrorAccessDenied | null\n  > {\n    const { unreadReceivedMailsCount } = await this.client(\"query\")({\n      unreadReceivedMailsCount: {\n        \"...on ErrorAccessDenied\": {\n          __typename: true,\n          message: true\n        },\n        \"...on UnreadReceivedMailsCountResponse\": {\n          __typename: true,\n          count: true\n        }\n      }\n    });\n\n    if (!unreadReceivedMailsCount) {\n      return null;\n    }\n\n    if (unreadReceivedMailsCount.__typename === \"ErrorAccessDenied\") {\n      return unreadReceivedMailsCount;\n    }\n\n    return {\n      __typename: \"SuccessResponse\",\n      data: unreadReceivedMailsCount.count\n    };\n  }\n\n  private _eachUser = async (\n    files: { id: string; name: string }[],\n    subject: string,\n    body: string,\n    idOrMail: string\n  ): Promise<MailRecipientInput | null> => {\n    let u = usersCache.get(idOrMail);\n\n    if (!u || !(\"publicKey\" in u)) {\n      try {\n        const req = await this.user({\n          userId: idOrMail,\n          withPublicKey: true\n        });\n        if (!req) {\n          return null;\n        }\n        if (req.__typename !== \"SuccessResponse\") {\n          return null;\n        }\n        u = req.data;\n      } catch {\n        return null;\n      }\n\n      if (!u) {\n        return null;\n      }\n    }\n\n    if (!(\"publicKey\" in u)) {\n      throw new Error(`User ${idOrMail} have no public key`);\n    }\n\n    const recipientsFiles = new Array<MailFileInput>();\n\n    for (const f of files) {\n      let fileInHistory = filesCache.get(f.id);\n      if (!fileInHistory) {\n        await this.file({ id: f.id });\n        fileInHistory = filesCache.get(f.id);\n        if (!fileInHistory) {\n          throw new Error(`File ${f.name} (${f.id}) does not exists`);\n        }\n      }\n      const key = fileInHistory.key;\n      recipientsFiles.push({\n        id: f.id,\n        name: sodium.to_hex(\n          encryptCryptoBox(\n            sodium.from_string(f.name),\n            u.publicKey,\n            this.#keys.privateKey\n          )\n        ),\n        fileKey: sodium.to_hex(\n          encryptCryptoBox(\n            sodium.from_hex(key),\n            u.publicKey,\n            this.#keys.privateKey\n          )\n        )\n      });\n    }\n\n    return {\n      recipientId: u.id,\n      body: sodium.to_hex(\n        encryptCryptoBox(\n          sodium.from_string(body),\n          u.publicKey,\n          this.#keys.privateKey\n        )\n      ),\n      subject: sodium.to_hex(\n        encryptCryptoBox(\n          sodium.from_string(subject),\n          u.publicKey,\n          this.#keys.privateKey\n        )\n      ),\n      files: recipientsFiles\n    };\n  };\n}\n"],"mappings":"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"}
|