@protontech/drive-sdk 0.0.10

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (594) hide show
  1. package/README.md +17 -0
  2. package/dist/cache/index.d.ts +2 -0
  3. package/dist/cache/index.js +6 -0
  4. package/dist/cache/index.js.map +1 -0
  5. package/dist/cache/interface.d.ts +105 -0
  6. package/dist/cache/interface.js +3 -0
  7. package/dist/cache/interface.js.map +1 -0
  8. package/dist/cache/memoryCache.d.ts +18 -0
  9. package/dist/cache/memoryCache.js +78 -0
  10. package/dist/cache/memoryCache.js.map +1 -0
  11. package/dist/cache/memoryCache.test.d.ts +1 -0
  12. package/dist/cache/memoryCache.test.js +121 -0
  13. package/dist/cache/memoryCache.test.js.map +1 -0
  14. package/dist/config.d.ts +2 -0
  15. package/dist/config.js +11 -0
  16. package/dist/config.js.map +1 -0
  17. package/dist/crypto/driveCrypto.d.ts +209 -0
  18. package/dist/crypto/driveCrypto.js +364 -0
  19. package/dist/crypto/driveCrypto.js.map +1 -0
  20. package/dist/crypto/driveCrypto.test.d.ts +1 -0
  21. package/dist/crypto/driveCrypto.test.js +42 -0
  22. package/dist/crypto/driveCrypto.test.js.map +1 -0
  23. package/dist/crypto/hmac.d.ts +22 -0
  24. package/dist/crypto/hmac.js +44 -0
  25. package/dist/crypto/hmac.js.map +1 -0
  26. package/dist/crypto/index.d.ts +6 -0
  27. package/dist/crypto/index.js +13 -0
  28. package/dist/crypto/index.js.map +1 -0
  29. package/dist/crypto/interface.d.ts +125 -0
  30. package/dist/crypto/interface.js +10 -0
  31. package/dist/crypto/interface.js.map +1 -0
  32. package/dist/crypto/openPGPCrypto.d.ts +145 -0
  33. package/dist/crypto/openPGPCrypto.js +250 -0
  34. package/dist/crypto/openPGPCrypto.js.map +1 -0
  35. package/dist/crypto/utils.d.ts +2 -0
  36. package/dist/crypto/utils.js +35 -0
  37. package/dist/crypto/utils.js.map +1 -0
  38. package/dist/errors.d.ts +138 -0
  39. package/dist/errors.js +163 -0
  40. package/dist/errors.js.map +1 -0
  41. package/dist/index.d.ts +24 -0
  42. package/dist/index.js +48 -0
  43. package/dist/index.js.map +1 -0
  44. package/dist/interface/account.d.ts +36 -0
  45. package/dist/interface/account.js +3 -0
  46. package/dist/interface/account.js.map +1 -0
  47. package/dist/interface/author.d.ts +26 -0
  48. package/dist/interface/author.js +3 -0
  49. package/dist/interface/author.js.map +1 -0
  50. package/dist/interface/devices.d.ts +18 -0
  51. package/dist/interface/devices.js +10 -0
  52. package/dist/interface/devices.js.map +1 -0
  53. package/dist/interface/download.d.ts +29 -0
  54. package/dist/interface/download.js +3 -0
  55. package/dist/interface/download.js.map +1 -0
  56. package/dist/interface/events.d.ts +26 -0
  57. package/dist/interface/events.js +11 -0
  58. package/dist/interface/events.js.map +1 -0
  59. package/dist/interface/httpClient.d.ts +38 -0
  60. package/dist/interface/httpClient.js +3 -0
  61. package/dist/interface/httpClient.js.map +1 -0
  62. package/dist/interface/index.d.ts +43 -0
  63. package/dist/interface/index.js +22 -0
  64. package/dist/interface/index.js.map +1 -0
  65. package/dist/interface/nodes.d.ts +161 -0
  66. package/dist/interface/nodes.js +21 -0
  67. package/dist/interface/nodes.js.map +1 -0
  68. package/dist/interface/result.d.ts +9 -0
  69. package/dist/interface/result.js +11 -0
  70. package/dist/interface/result.js.map +1 -0
  71. package/dist/interface/sharing.d.ts +73 -0
  72. package/dist/interface/sharing.js +9 -0
  73. package/dist/interface/sharing.js.map +1 -0
  74. package/dist/interface/telemetry.d.ts +61 -0
  75. package/dist/interface/telemetry.js +18 -0
  76. package/dist/interface/telemetry.js.map +1 -0
  77. package/dist/interface/thumbnail.d.ts +17 -0
  78. package/dist/interface/thumbnail.js +9 -0
  79. package/dist/interface/thumbnail.js.map +1 -0
  80. package/dist/interface/upload.d.ts +16 -0
  81. package/dist/interface/upload.js +3 -0
  82. package/dist/interface/upload.js.map +1 -0
  83. package/dist/internal/apiService/apiService.d.ts +51 -0
  84. package/dist/internal/apiService/apiService.js +305 -0
  85. package/dist/internal/apiService/apiService.js.map +1 -0
  86. package/dist/internal/apiService/apiService.test.d.ts +1 -0
  87. package/dist/internal/apiService/apiService.test.js +237 -0
  88. package/dist/internal/apiService/apiService.test.js.map +1 -0
  89. package/dist/internal/apiService/coreTypes.d.ts +24438 -0
  90. package/dist/internal/apiService/coreTypes.js +7 -0
  91. package/dist/internal/apiService/coreTypes.js.map +1 -0
  92. package/dist/internal/apiService/driveTypes.d.ts +11840 -0
  93. package/dist/internal/apiService/driveTypes.js +7 -0
  94. package/dist/internal/apiService/driveTypes.js.map +1 -0
  95. package/dist/internal/apiService/errorCodes.d.ts +30 -0
  96. package/dist/internal/apiService/errorCodes.js +11 -0
  97. package/dist/internal/apiService/errorCodes.js.map +1 -0
  98. package/dist/internal/apiService/errors.d.ts +19 -0
  99. package/dist/internal/apiService/errors.js +76 -0
  100. package/dist/internal/apiService/errors.js.map +1 -0
  101. package/dist/internal/apiService/errors.test.d.ts +1 -0
  102. package/dist/internal/apiService/errors.test.js +69 -0
  103. package/dist/internal/apiService/errors.test.js.map +1 -0
  104. package/dist/internal/apiService/index.d.ts +7 -0
  105. package/dist/internal/apiService/index.js +30 -0
  106. package/dist/internal/apiService/index.js.map +1 -0
  107. package/dist/internal/apiService/observerStream.d.ts +3 -0
  108. package/dist/internal/apiService/observerStream.js +15 -0
  109. package/dist/internal/apiService/observerStream.js.map +1 -0
  110. package/dist/internal/apiService/transformers.d.ts +4 -0
  111. package/dist/internal/apiService/transformers.js +48 -0
  112. package/dist/internal/apiService/transformers.js.map +1 -0
  113. package/dist/internal/batchLoading.d.ts +34 -0
  114. package/dist/internal/batchLoading.js +68 -0
  115. package/dist/internal/batchLoading.js.map +1 -0
  116. package/dist/internal/batchLoading.test.d.ts +1 -0
  117. package/dist/internal/batchLoading.test.js +50 -0
  118. package/dist/internal/batchLoading.test.js.map +1 -0
  119. package/dist/internal/devices/apiService.d.ts +38 -0
  120. package/dist/internal/devices/apiService.js +105 -0
  121. package/dist/internal/devices/apiService.js.map +1 -0
  122. package/dist/internal/devices/cryptoService.d.ts +30 -0
  123. package/dist/internal/devices/cryptoService.js +47 -0
  124. package/dist/internal/devices/cryptoService.js.map +1 -0
  125. package/dist/internal/devices/index.d.ts +15 -0
  126. package/dist/internal/devices/index.js +22 -0
  127. package/dist/internal/devices/index.js.map +1 -0
  128. package/dist/internal/devices/interface.d.ts +31 -0
  129. package/dist/internal/devices/interface.js +3 -0
  130. package/dist/internal/devices/interface.js.map +1 -0
  131. package/dist/internal/devices/manager.d.ts +18 -0
  132. package/dist/internal/devices/manager.js +103 -0
  133. package/dist/internal/devices/manager.js.map +1 -0
  134. package/dist/internal/devices/manager.test.d.ts +1 -0
  135. package/dist/internal/devices/manager.test.js +100 -0
  136. package/dist/internal/devices/manager.test.js.map +1 -0
  137. package/dist/internal/download/apiService.d.ts +27 -0
  138. package/dist/internal/download/apiService.js +110 -0
  139. package/dist/internal/download/apiService.js.map +1 -0
  140. package/dist/internal/download/controller.d.ts +8 -0
  141. package/dist/internal/download/controller.js +22 -0
  142. package/dist/internal/download/controller.js.map +1 -0
  143. package/dist/internal/download/cryptoService.d.ts +17 -0
  144. package/dist/internal/download/cryptoService.js +82 -0
  145. package/dist/internal/download/cryptoService.js.map +1 -0
  146. package/dist/internal/download/fileDownloader.d.ts +33 -0
  147. package/dist/internal/download/fileDownloader.js +226 -0
  148. package/dist/internal/download/fileDownloader.js.map +1 -0
  149. package/dist/internal/download/fileDownloader.test.d.ts +1 -0
  150. package/dist/internal/download/fileDownloader.test.js +324 -0
  151. package/dist/internal/download/fileDownloader.test.js.map +1 -0
  152. package/dist/internal/download/index.d.ts +10 -0
  153. package/dist/internal/download/index.js +82 -0
  154. package/dist/internal/download/index.js.map +1 -0
  155. package/dist/internal/download/interface.d.ts +35 -0
  156. package/dist/internal/download/interface.js +3 -0
  157. package/dist/internal/download/interface.js.map +1 -0
  158. package/dist/internal/download/queue.d.ts +5 -0
  159. package/dist/internal/download/queue.js +31 -0
  160. package/dist/internal/download/queue.js.map +1 -0
  161. package/dist/internal/download/telemetry.d.ts +14 -0
  162. package/dist/internal/download/telemetry.js +106 -0
  163. package/dist/internal/download/telemetry.js.map +1 -0
  164. package/dist/internal/download/telemetry.test.d.ts +1 -0
  165. package/dist/internal/download/telemetry.test.js +111 -0
  166. package/dist/internal/download/telemetry.test.js.map +1 -0
  167. package/dist/internal/download/thumbnailDownloader.d.ts +19 -0
  168. package/dist/internal/download/thumbnailDownloader.js +198 -0
  169. package/dist/internal/download/thumbnailDownloader.js.map +1 -0
  170. package/dist/internal/download/thumbnailDownloader.test.d.ts +1 -0
  171. package/dist/internal/download/thumbnailDownloader.test.js +179 -0
  172. package/dist/internal/download/thumbnailDownloader.test.js.map +1 -0
  173. package/dist/internal/errors.d.ts +6 -0
  174. package/dist/internal/errors.js +28 -0
  175. package/dist/internal/errors.js.map +1 -0
  176. package/dist/internal/errors.test.d.ts +1 -0
  177. package/dist/internal/errors.test.js +22 -0
  178. package/dist/internal/errors.test.js.map +1 -0
  179. package/dist/internal/events/apiService.d.ts +18 -0
  180. package/dist/internal/events/apiService.js +81 -0
  181. package/dist/internal/events/apiService.js.map +1 -0
  182. package/dist/internal/events/cache.d.ts +28 -0
  183. package/dist/internal/events/cache.js +67 -0
  184. package/dist/internal/events/cache.js.map +1 -0
  185. package/dist/internal/events/cache.test.d.ts +1 -0
  186. package/dist/internal/events/cache.test.js +43 -0
  187. package/dist/internal/events/cache.test.js.map +1 -0
  188. package/dist/internal/events/coreEventManager.d.ts +25 -0
  189. package/dist/internal/events/coreEventManager.js +63 -0
  190. package/dist/internal/events/coreEventManager.js.map +1 -0
  191. package/dist/internal/events/eventManager.d.ts +62 -0
  192. package/dist/internal/events/eventManager.js +159 -0
  193. package/dist/internal/events/eventManager.js.map +1 -0
  194. package/dist/internal/events/eventManager.test.d.ts +1 -0
  195. package/dist/internal/events/eventManager.test.js +120 -0
  196. package/dist/internal/events/eventManager.test.js.map +1 -0
  197. package/dist/internal/events/index.d.ts +50 -0
  198. package/dist/internal/events/index.js +117 -0
  199. package/dist/internal/events/index.js.map +1 -0
  200. package/dist/internal/events/interface.d.ts +47 -0
  201. package/dist/internal/events/interface.js +12 -0
  202. package/dist/internal/events/interface.js.map +1 -0
  203. package/dist/internal/events/volumeEventManager.d.ts +27 -0
  204. package/dist/internal/events/volumeEventManager.js +69 -0
  205. package/dist/internal/events/volumeEventManager.js.map +1 -0
  206. package/dist/internal/nodes/apiService.d.ts +54 -0
  207. package/dist/internal/nodes/apiService.js +308 -0
  208. package/dist/internal/nodes/apiService.js.map +1 -0
  209. package/dist/internal/nodes/apiService.test.d.ts +1 -0
  210. package/dist/internal/nodes/apiService.test.js +344 -0
  211. package/dist/internal/nodes/apiService.test.js.map +1 -0
  212. package/dist/internal/nodes/cache.d.ts +57 -0
  213. package/dist/internal/nodes/cache.js +226 -0
  214. package/dist/internal/nodes/cache.js.map +1 -0
  215. package/dist/internal/nodes/cache.test.d.ts +1 -0
  216. package/dist/internal/nodes/cache.test.js +160 -0
  217. package/dist/internal/nodes/cache.test.js.map +1 -0
  218. package/dist/internal/nodes/cryptoCache.d.ts +16 -0
  219. package/dist/internal/nodes/cryptoCache.js +50 -0
  220. package/dist/internal/nodes/cryptoCache.js.map +1 -0
  221. package/dist/internal/nodes/cryptoCache.test.d.ts +1 -0
  222. package/dist/internal/nodes/cryptoCache.test.js +79 -0
  223. package/dist/internal/nodes/cryptoCache.test.js.map +1 -0
  224. package/dist/internal/nodes/cryptoService.d.ts +75 -0
  225. package/dist/internal/nodes/cryptoService.js +409 -0
  226. package/dist/internal/nodes/cryptoService.js.map +1 -0
  227. package/dist/internal/nodes/cryptoService.test.d.ts +1 -0
  228. package/dist/internal/nodes/cryptoService.test.js +698 -0
  229. package/dist/internal/nodes/cryptoService.test.js.map +1 -0
  230. package/dist/internal/nodes/events.d.ts +91 -0
  231. package/dist/internal/nodes/events.js +243 -0
  232. package/dist/internal/nodes/events.js.map +1 -0
  233. package/dist/internal/nodes/events.test.d.ts +1 -0
  234. package/dist/internal/nodes/events.test.js +318 -0
  235. package/dist/internal/nodes/events.test.js.map +1 -0
  236. package/dist/internal/nodes/extendedAttributes.d.ts +23 -0
  237. package/dist/internal/nodes/extendedAttributes.js +117 -0
  238. package/dist/internal/nodes/extendedAttributes.js.map +1 -0
  239. package/dist/internal/nodes/extendedAttributes.test.d.ts +1 -0
  240. package/dist/internal/nodes/extendedAttributes.test.js +176 -0
  241. package/dist/internal/nodes/extendedAttributes.test.js.map +1 -0
  242. package/dist/internal/nodes/index.d.ts +26 -0
  243. package/dist/internal/nodes/index.js +40 -0
  244. package/dist/internal/nodes/index.js.map +1 -0
  245. package/dist/internal/nodes/index.test.d.ts +1 -0
  246. package/dist/internal/nodes/index.test.js +112 -0
  247. package/dist/internal/nodes/index.test.js.map +1 -0
  248. package/dist/internal/nodes/interface.d.ts +135 -0
  249. package/dist/internal/nodes/interface.js +3 -0
  250. package/dist/internal/nodes/interface.js.map +1 -0
  251. package/dist/internal/nodes/mediaTypes.d.ts +2 -0
  252. package/dist/internal/nodes/mediaTypes.js +13 -0
  253. package/dist/internal/nodes/mediaTypes.js.map +1 -0
  254. package/dist/internal/nodes/nodesAccess.d.ts +49 -0
  255. package/dist/internal/nodes/nodesAccess.js +332 -0
  256. package/dist/internal/nodes/nodesAccess.js.map +1 -0
  257. package/dist/internal/nodes/nodesAccess.test.d.ts +1 -0
  258. package/dist/internal/nodes/nodesAccess.test.js +487 -0
  259. package/dist/internal/nodes/nodesAccess.test.js.map +1 -0
  260. package/dist/internal/nodes/nodesManagement.d.ts +33 -0
  261. package/dist/internal/nodes/nodesManagement.js +222 -0
  262. package/dist/internal/nodes/nodesManagement.js.map +1 -0
  263. package/dist/internal/nodes/nodesManagement.test.d.ts +1 -0
  264. package/dist/internal/nodes/nodesManagement.test.js +178 -0
  265. package/dist/internal/nodes/nodesManagement.test.js.map +1 -0
  266. package/dist/internal/nodes/nodesRevisions.d.ts +18 -0
  267. package/dist/internal/nodes/nodesRevisions.js +55 -0
  268. package/dist/internal/nodes/nodesRevisions.js.map +1 -0
  269. package/dist/internal/nodes/validations.d.ts +4 -0
  270. package/dist/internal/nodes/validations.js +21 -0
  271. package/dist/internal/nodes/validations.js.map +1 -0
  272. package/dist/internal/photos/albums.d.ts +13 -0
  273. package/dist/internal/photos/albums.js +30 -0
  274. package/dist/internal/photos/albums.js.map +1 -0
  275. package/dist/internal/photos/apiService.d.ts +8 -0
  276. package/dist/internal/photos/apiService.js +18 -0
  277. package/dist/internal/photos/apiService.js.map +1 -0
  278. package/dist/internal/photos/cache.d.ts +6 -0
  279. package/dist/internal/photos/cache.js +15 -0
  280. package/dist/internal/photos/cache.js.map +1 -0
  281. package/dist/internal/photos/index.d.ts +9 -0
  282. package/dist/internal/photos/index.js +18 -0
  283. package/dist/internal/photos/index.js.map +1 -0
  284. package/dist/internal/photos/interface.d.ts +6 -0
  285. package/dist/internal/photos/interface.js +3 -0
  286. package/dist/internal/photos/interface.js.map +1 -0
  287. package/dist/internal/photos/photosTimeline.d.ts +10 -0
  288. package/dist/internal/photos/photosTimeline.js +20 -0
  289. package/dist/internal/photos/photosTimeline.js.map +1 -0
  290. package/dist/internal/sdkEvents.d.ts +12 -0
  291. package/dist/internal/sdkEvents.js +44 -0
  292. package/dist/internal/sdkEvents.js.map +1 -0
  293. package/dist/internal/sdkEvents.test.d.ts +1 -0
  294. package/dist/internal/sdkEvents.test.js +45 -0
  295. package/dist/internal/sdkEvents.test.js.map +1 -0
  296. package/dist/internal/shares/apiService.d.ts +51 -0
  297. package/dist/internal/shares/apiService.js +128 -0
  298. package/dist/internal/shares/apiService.js.map +1 -0
  299. package/dist/internal/shares/cache.d.ts +15 -0
  300. package/dist/internal/shares/cache.js +63 -0
  301. package/dist/internal/shares/cache.js.map +1 -0
  302. package/dist/internal/shares/cache.test.d.ts +1 -0
  303. package/dist/internal/shares/cache.test.js +54 -0
  304. package/dist/internal/shares/cache.test.js.map +1 -0
  305. package/dist/internal/shares/cryptoCache.d.ts +20 -0
  306. package/dist/internal/shares/cryptoCache.js +35 -0
  307. package/dist/internal/shares/cryptoCache.js.map +1 -0
  308. package/dist/internal/shares/cryptoCache.test.d.ts +1 -0
  309. package/dist/internal/shares/cryptoCache.test.js +58 -0
  310. package/dist/internal/shares/cryptoCache.test.js.map +1 -0
  311. package/dist/internal/shares/cryptoService.d.ts +42 -0
  312. package/dist/internal/shares/cryptoService.js +126 -0
  313. package/dist/internal/shares/cryptoService.js.map +1 -0
  314. package/dist/internal/shares/cryptoService.test.d.ts +1 -0
  315. package/dist/internal/shares/cryptoService.test.js +120 -0
  316. package/dist/internal/shares/cryptoService.test.js.map +1 -0
  317. package/dist/internal/shares/index.d.ts +15 -0
  318. package/dist/internal/shares/index.js +26 -0
  319. package/dist/internal/shares/index.js.map +1 -0
  320. package/dist/internal/shares/interface.d.ts +94 -0
  321. package/dist/internal/shares/interface.js +11 -0
  322. package/dist/internal/shares/interface.js.map +1 -0
  323. package/dist/internal/shares/manager.d.ts +68 -0
  324. package/dist/internal/shares/manager.js +186 -0
  325. package/dist/internal/shares/manager.js.map +1 -0
  326. package/dist/internal/shares/manager.test.d.ts +1 -0
  327. package/dist/internal/shares/manager.test.js +176 -0
  328. package/dist/internal/shares/manager.test.js.map +1 -0
  329. package/dist/internal/sharing/apiService.d.ts +80 -0
  330. package/dist/internal/sharing/apiService.js +325 -0
  331. package/dist/internal/sharing/apiService.js.map +1 -0
  332. package/dist/internal/sharing/cache.d.ts +37 -0
  333. package/dist/internal/sharing/cache.js +105 -0
  334. package/dist/internal/sharing/cache.js.map +1 -0
  335. package/dist/internal/sharing/cache.test.d.ts +1 -0
  336. package/dist/internal/sharing/cache.test.js +82 -0
  337. package/dist/internal/sharing/cache.test.js.map +1 -0
  338. package/dist/internal/sharing/cryptoService.d.ts +112 -0
  339. package/dist/internal/sharing/cryptoService.js +269 -0
  340. package/dist/internal/sharing/cryptoService.js.map +1 -0
  341. package/dist/internal/sharing/events.d.ts +57 -0
  342. package/dist/internal/sharing/events.js +153 -0
  343. package/dist/internal/sharing/events.js.map +1 -0
  344. package/dist/internal/sharing/events.test.d.ts +1 -0
  345. package/dist/internal/sharing/events.test.js +246 -0
  346. package/dist/internal/sharing/events.test.js.map +1 -0
  347. package/dist/internal/sharing/index.d.ts +20 -0
  348. package/dist/internal/sharing/index.js +30 -0
  349. package/dist/internal/sharing/index.js.map +1 -0
  350. package/dist/internal/sharing/interface.d.ts +169 -0
  351. package/dist/internal/sharing/interface.js +9 -0
  352. package/dist/internal/sharing/interface.js.map +1 -0
  353. package/dist/internal/sharing/sharingAccess.d.ts +32 -0
  354. package/dist/internal/sharing/sharingAccess.js +119 -0
  355. package/dist/internal/sharing/sharingAccess.js.map +1 -0
  356. package/dist/internal/sharing/sharingAccess.test.d.ts +1 -0
  357. package/dist/internal/sharing/sharingAccess.test.js +84 -0
  358. package/dist/internal/sharing/sharingAccess.test.js.map +1 -0
  359. package/dist/internal/sharing/sharingManagement.d.ts +43 -0
  360. package/dist/internal/sharing/sharingManagement.js +439 -0
  361. package/dist/internal/sharing/sharingManagement.js.map +1 -0
  362. package/dist/internal/sharing/sharingManagement.test.d.ts +1 -0
  363. package/dist/internal/sharing/sharingManagement.test.js +788 -0
  364. package/dist/internal/sharing/sharingManagement.test.js.map +1 -0
  365. package/dist/internal/uids.d.ts +38 -0
  366. package/dist/internal/uids.js +85 -0
  367. package/dist/internal/uids.js.map +1 -0
  368. package/dist/internal/upload/apiService.d.ts +65 -0
  369. package/dist/internal/upload/apiService.js +137 -0
  370. package/dist/internal/upload/apiService.js.map +1 -0
  371. package/dist/internal/upload/blockVerifier.d.ts +16 -0
  372. package/dist/internal/upload/blockVerifier.js +33 -0
  373. package/dist/internal/upload/blockVerifier.js.map +1 -0
  374. package/dist/internal/upload/chunkStreamReader.d.ts +13 -0
  375. package/dist/internal/upload/chunkStreamReader.js +46 -0
  376. package/dist/internal/upload/chunkStreamReader.js.map +1 -0
  377. package/dist/internal/upload/chunkStreamReader.test.d.ts +1 -0
  378. package/dist/internal/upload/chunkStreamReader.test.js +75 -0
  379. package/dist/internal/upload/chunkStreamReader.test.js.map +1 -0
  380. package/dist/internal/upload/controller.d.ts +8 -0
  381. package/dist/internal/upload/controller.js +25 -0
  382. package/dist/internal/upload/controller.js.map +1 -0
  383. package/dist/internal/upload/cryptoService.d.ts +29 -0
  384. package/dist/internal/upload/cryptoService.js +104 -0
  385. package/dist/internal/upload/cryptoService.js.map +1 -0
  386. package/dist/internal/upload/digests.d.ts +8 -0
  387. package/dist/internal/upload/digests.js +22 -0
  388. package/dist/internal/upload/digests.js.map +1 -0
  389. package/dist/internal/upload/fileUploader.d.ts +65 -0
  390. package/dist/internal/upload/fileUploader.js +463 -0
  391. package/dist/internal/upload/fileUploader.js.map +1 -0
  392. package/dist/internal/upload/fileUploader.test.d.ts +1 -0
  393. package/dist/internal/upload/fileUploader.test.js +393 -0
  394. package/dist/internal/upload/fileUploader.test.js.map +1 -0
  395. package/dist/internal/upload/index.d.ts +16 -0
  396. package/dist/internal/upload/index.js +77 -0
  397. package/dist/internal/upload/index.js.map +1 -0
  398. package/dist/internal/upload/interface.d.ts +117 -0
  399. package/dist/internal/upload/interface.js +3 -0
  400. package/dist/internal/upload/interface.js.map +1 -0
  401. package/dist/internal/upload/manager.d.ts +31 -0
  402. package/dist/internal/upload/manager.js +250 -0
  403. package/dist/internal/upload/manager.js.map +1 -0
  404. package/dist/internal/upload/manager.test.d.ts +1 -0
  405. package/dist/internal/upload/manager.test.js +349 -0
  406. package/dist/internal/upload/manager.test.js.map +1 -0
  407. package/dist/internal/upload/queue.d.ts +5 -0
  408. package/dist/internal/upload/queue.js +32 -0
  409. package/dist/internal/upload/queue.js.map +1 -0
  410. package/dist/internal/upload/telemetry.d.ts +15 -0
  411. package/dist/internal/upload/telemetry.js +111 -0
  412. package/dist/internal/upload/telemetry.js.map +1 -0
  413. package/dist/internal/upload/telemetry.test.d.ts +1 -0
  414. package/dist/internal/upload/telemetry.test.js +107 -0
  415. package/dist/internal/upload/telemetry.test.js.map +1 -0
  416. package/dist/internal/utils.d.ts +1 -0
  417. package/dist/internal/utils.js +13 -0
  418. package/dist/internal/utils.js.map +1 -0
  419. package/dist/internal/wait.d.ts +3 -0
  420. package/dist/internal/wait.js +28 -0
  421. package/dist/internal/wait.js.map +1 -0
  422. package/dist/internal/wait.test.d.ts +1 -0
  423. package/dist/internal/wait.test.js +21 -0
  424. package/dist/internal/wait.test.js.map +1 -0
  425. package/dist/protonDriveClient.d.ts +593 -0
  426. package/dist/protonDriveClient.js +768 -0
  427. package/dist/protonDriveClient.js.map +1 -0
  428. package/dist/protonDrivePhotosClient.d.ts +16 -0
  429. package/dist/protonDrivePhotosClient.js +46 -0
  430. package/dist/protonDrivePhotosClient.js.map +1 -0
  431. package/dist/telemetry.d.ts +187 -0
  432. package/dist/telemetry.js +297 -0
  433. package/dist/telemetry.js.map +1 -0
  434. package/dist/tests/logger.d.ts +2 -0
  435. package/dist/tests/logger.js +12 -0
  436. package/dist/tests/logger.js.map +1 -0
  437. package/dist/tests/telemetry.d.ts +2 -0
  438. package/dist/tests/telemetry.js +11 -0
  439. package/dist/tests/telemetry.js.map +1 -0
  440. package/dist/transformers.d.ts +17 -0
  441. package/dist/transformers.js +77 -0
  442. package/dist/transformers.js.map +1 -0
  443. package/dist/version.d.ts +1 -0
  444. package/dist/version.js +6 -0
  445. package/dist/version.js.map +1 -0
  446. package/package.json +49 -0
  447. package/src/cache/index.ts +2 -0
  448. package/src/cache/interface.ts +104 -0
  449. package/src/cache/memoryCache.test.ts +150 -0
  450. package/src/cache/memoryCache.ts +85 -0
  451. package/src/config.ts +9 -0
  452. package/src/crypto/driveCrypto.test.ts +45 -0
  453. package/src/crypto/driveCrypto.ts +696 -0
  454. package/src/crypto/hmac.ts +46 -0
  455. package/src/crypto/index.ts +6 -0
  456. package/src/crypto/interface.ts +230 -0
  457. package/src/crypto/openPGPCrypto.ts +398 -0
  458. package/src/crypto/utils.ts +40 -0
  459. package/src/errors.ts +168 -0
  460. package/src/index.ts +30 -0
  461. package/src/interface/account.ts +38 -0
  462. package/src/interface/author.ts +29 -0
  463. package/src/interface/devices.ts +21 -0
  464. package/src/interface/download.ts +32 -0
  465. package/src/interface/events.ts +30 -0
  466. package/src/interface/httpClient.ts +42 -0
  467. package/src/interface/index.ts +46 -0
  468. package/src/interface/nodes.ts +168 -0
  469. package/src/interface/result.ts +11 -0
  470. package/src/interface/sharing.ts +87 -0
  471. package/src/interface/telemetry.ts +107 -0
  472. package/src/interface/thumbnail.ts +14 -0
  473. package/src/interface/upload.ts +19 -0
  474. package/src/internal/apiService/apiService.test.ts +285 -0
  475. package/src/internal/apiService/apiService.ts +353 -0
  476. package/src/internal/apiService/coreTypes.ts +24439 -0
  477. package/src/internal/apiService/driveTypes.ts +11841 -0
  478. package/src/internal/apiService/errorCodes.ts +38 -0
  479. package/src/internal/apiService/errors.test.ts +63 -0
  480. package/src/internal/apiService/errors.ts +91 -0
  481. package/src/internal/apiService/index.ts +7 -0
  482. package/src/internal/apiService/observerStream.ts +10 -0
  483. package/src/internal/apiService/transformers.ts +45 -0
  484. package/src/internal/batchLoading.test.ts +58 -0
  485. package/src/internal/batchLoading.ts +74 -0
  486. package/src/internal/devices/apiService.ts +142 -0
  487. package/src/internal/devices/cryptoService.ts +65 -0
  488. package/src/internal/devices/index.ts +31 -0
  489. package/src/internal/devices/interface.ts +28 -0
  490. package/src/internal/devices/manager.test.ts +129 -0
  491. package/src/internal/devices/manager.ts +113 -0
  492. package/src/internal/download/apiService.ts +145 -0
  493. package/src/internal/download/controller.ts +22 -0
  494. package/src/internal/download/cryptoService.ts +96 -0
  495. package/src/internal/download/fileDownloader.test.ts +381 -0
  496. package/src/internal/download/fileDownloader.ts +266 -0
  497. package/src/internal/download/index.ts +122 -0
  498. package/src/internal/download/interface.ts +38 -0
  499. package/src/internal/download/queue.ts +30 -0
  500. package/src/internal/download/telemetry.test.ts +134 -0
  501. package/src/internal/download/telemetry.ts +115 -0
  502. package/src/internal/download/thumbnailDownloader.test.ts +226 -0
  503. package/src/internal/download/thumbnailDownloader.ts +250 -0
  504. package/src/internal/errors.test.ts +21 -0
  505. package/src/internal/errors.ts +28 -0
  506. package/src/internal/events/apiService.ts +87 -0
  507. package/src/internal/events/cache.test.ts +47 -0
  508. package/src/internal/events/cache.ts +80 -0
  509. package/src/internal/events/coreEventManager.ts +69 -0
  510. package/src/internal/events/eventManager.test.ts +139 -0
  511. package/src/internal/events/eventManager.ts +168 -0
  512. package/src/internal/events/index.ts +131 -0
  513. package/src/internal/events/interface.ts +51 -0
  514. package/src/internal/events/volumeEventManager.ts +74 -0
  515. package/src/internal/nodes/apiService.test.ts +394 -0
  516. package/src/internal/nodes/apiService.ts +455 -0
  517. package/src/internal/nodes/cache.test.ts +197 -0
  518. package/src/internal/nodes/cache.ts +240 -0
  519. package/src/internal/nodes/cryptoCache.test.ts +92 -0
  520. package/src/internal/nodes/cryptoCache.ts +47 -0
  521. package/src/internal/nodes/cryptoService.test.ts +814 -0
  522. package/src/internal/nodes/cryptoService.ts +550 -0
  523. package/src/internal/nodes/events.test.ts +383 -0
  524. package/src/internal/nodes/events.ts +271 -0
  525. package/src/internal/nodes/extendedAttributes.test.ts +177 -0
  526. package/src/internal/nodes/extendedAttributes.ts +180 -0
  527. package/src/internal/nodes/index.test.ts +135 -0
  528. package/src/internal/nodes/index.ts +52 -0
  529. package/src/internal/nodes/interface.ts +152 -0
  530. package/src/internal/nodes/mediaTypes.ts +10 -0
  531. package/src/internal/nodes/nodesAccess.test.ts +555 -0
  532. package/src/internal/nodes/nodesAccess.ts +360 -0
  533. package/src/internal/nodes/nodesManagement.test.ts +221 -0
  534. package/src/internal/nodes/nodesManagement.ts +272 -0
  535. package/src/internal/nodes/nodesRevisions.ts +58 -0
  536. package/src/internal/nodes/validations.ts +26 -0
  537. package/src/internal/photos/albums.ts +29 -0
  538. package/src/internal/photos/apiService.ts +16 -0
  539. package/src/internal/photos/cache.ts +11 -0
  540. package/src/internal/photos/index.ts +23 -0
  541. package/src/internal/photos/interface.ts +7 -0
  542. package/src/internal/photos/photosTimeline.ts +18 -0
  543. package/src/internal/sdkEvents.test.ts +55 -0
  544. package/src/internal/sdkEvents.ts +52 -0
  545. package/src/internal/shares/apiService.ts +176 -0
  546. package/src/internal/shares/cache.test.ts +58 -0
  547. package/src/internal/shares/cache.ts +64 -0
  548. package/src/internal/shares/cryptoCache.test.ts +70 -0
  549. package/src/internal/shares/cryptoCache.ts +35 -0
  550. package/src/internal/shares/cryptoService.test.ts +137 -0
  551. package/src/internal/shares/cryptoService.ts +143 -0
  552. package/src/internal/shares/index.ts +35 -0
  553. package/src/internal/shares/interface.ts +104 -0
  554. package/src/internal/shares/manager.test.ts +210 -0
  555. package/src/internal/shares/manager.ts +213 -0
  556. package/src/internal/sharing/apiService.ts +483 -0
  557. package/src/internal/sharing/cache.test.ts +99 -0
  558. package/src/internal/sharing/cache.ts +109 -0
  559. package/src/internal/sharing/cryptoService.ts +378 -0
  560. package/src/internal/sharing/events.test.ts +268 -0
  561. package/src/internal/sharing/events.ts +166 -0
  562. package/src/internal/sharing/index.ts +43 -0
  563. package/src/internal/sharing/interface.ts +177 -0
  564. package/src/internal/sharing/sharingAccess.test.ts +102 -0
  565. package/src/internal/sharing/sharingAccess.ts +133 -0
  566. package/src/internal/sharing/sharingManagement.test.ts +895 -0
  567. package/src/internal/sharing/sharingManagement.ts +521 -0
  568. package/src/internal/uids.ts +83 -0
  569. package/src/internal/upload/apiService.ts +243 -0
  570. package/src/internal/upload/blockVerifier.ts +40 -0
  571. package/src/internal/upload/chunkStreamReader.test.ts +89 -0
  572. package/src/internal/upload/chunkStreamReader.ts +49 -0
  573. package/src/internal/upload/controller.ts +25 -0
  574. package/src/internal/upload/cryptoService.ts +162 -0
  575. package/src/internal/upload/digests.ts +18 -0
  576. package/src/internal/upload/fileUploader.test.ts +513 -0
  577. package/src/internal/upload/fileUploader.ts +576 -0
  578. package/src/internal/upload/index.ts +126 -0
  579. package/src/internal/upload/interface.ts +128 -0
  580. package/src/internal/upload/manager.test.ts +390 -0
  581. package/src/internal/upload/manager.ts +320 -0
  582. package/src/internal/upload/queue.ts +31 -0
  583. package/src/internal/upload/telemetry.test.ts +129 -0
  584. package/src/internal/upload/telemetry.ts +121 -0
  585. package/src/internal/utils.ts +9 -0
  586. package/src/internal/wait.test.ts +21 -0
  587. package/src/internal/wait.ts +26 -0
  588. package/src/protonDriveClient.ts +861 -0
  589. package/src/protonDrivePhotosClient.ts +56 -0
  590. package/src/telemetry.ts +346 -0
  591. package/src/tests/logger.ts +10 -0
  592. package/src/tests/telemetry.ts +9 -0
  593. package/src/transformers.ts +98 -0
  594. package/src/version.ts +4 -0
@@ -0,0 +1,895 @@
1
+ import { getMockLogger } from "../../tests/logger";
2
+ import { Member, MemberRole, NonProtonInvitation, NonProtonInvitationState, ProtonDriveAccount, ProtonInvitation, PublicLink, resultOk } from "../../interface";
3
+ import { SharingAPIService } from "./apiService";
4
+ import { SharingCryptoService } from "./cryptoService";
5
+ import { SharesService, NodesService, NodesEvents } from "./interface";
6
+ import { SharingManagement } from "./sharingManagement";
7
+
8
+ describe("SharingManagement", () => {
9
+ let apiService: SharingAPIService;
10
+ let cryptoService: SharingCryptoService;
11
+ let accountService: ProtonDriveAccount;
12
+ let sharesService: SharesService;
13
+ let nodesService: NodesService;
14
+ let nodesEvents: NodesEvents;
15
+
16
+ let sharingManagement: SharingManagement;
17
+
18
+ beforeEach(() => {
19
+ // @ts-expect-error No need to implement all methods for mocking
20
+ apiService = {
21
+ createStandardShare: jest.fn().mockReturnValue("newShareId"),
22
+ getShareInvitations: jest.fn().mockResolvedValue([]),
23
+ getShareExternalInvitations: jest.fn().mockResolvedValue([]),
24
+ getShareMembers: jest.fn().mockResolvedValue([]),
25
+ inviteProtonUser: jest.fn().mockImplementation((_, invitation) => ({
26
+ ...invitation,
27
+ uid: "created-invitation",
28
+ })),
29
+ updateInvitation: jest.fn(),
30
+ deleteInvitation: jest.fn(),
31
+ inviteExternalUser: jest.fn().mockImplementation((_, invitation) => ({
32
+ ...invitation,
33
+ uid: "created-external-invitation",
34
+ state: NonProtonInvitationState.Pending,
35
+ })),
36
+ updateExternalInvitation: jest.fn(),
37
+ deleteExternalInvitation: jest.fn(),
38
+ updateMember: jest.fn(),
39
+ removeMember: jest.fn(),
40
+ getPublicLink: jest.fn().mockResolvedValue(undefined),
41
+ removePublicLink: jest.fn(),
42
+ deleteShare: jest.fn(),
43
+ resendInvitationEmail: jest.fn(),
44
+ resendExternalInvitationEmail: jest.fn(),
45
+ createPublicLink: jest.fn().mockResolvedValue({
46
+ uid: "publicLinkUid",
47
+ publicUrl: "publicLinkUrl",
48
+ }),
49
+ updatePublicLink: jest.fn(),
50
+ }
51
+ // @ts-expect-error No need to implement all methods for mocking
52
+ cryptoService = {
53
+ generateShareKeys: jest.fn().mockResolvedValue({ shareKey: { encrypted: "encrypted-key", decrypted: { passphraseSessionKey: "pass-session-key", } } }),
54
+ decryptShare: jest.fn().mockImplementation((share) => share),
55
+ decryptInvitation: jest.fn().mockImplementation((invitation) => invitation),
56
+ decryptExternalInvitation: jest.fn().mockImplementation((invitation) => invitation),
57
+ decryptMember: jest.fn().mockImplementation((member) => member),
58
+ encryptInvitation: jest.fn().mockImplementation(() => { }),
59
+ encryptExternalInvitation: jest.fn().mockImplementation((invitation) => ({
60
+ ...invitation,
61
+ base64ExternalInvitationSignature: "extenral-signature",
62
+ })),
63
+ decryptPublicLink: jest.fn().mockImplementation((publicLink) => publicLink),
64
+ generatePublicLinkPassword: jest.fn().mockResolvedValue("generatedPassword"),
65
+ encryptPublicLink: jest.fn().mockImplementation(() => ({
66
+ crypto: "publicLinkCrypto",
67
+ srp: "publicLinkSrp",
68
+ })),
69
+ }
70
+ // @ts-expect-error No need to implement all methods for mocking
71
+ accountService = {
72
+ hasProtonAccount: jest.fn().mockResolvedValue(true),
73
+ }
74
+ // @ts-expect-error No need to implement all methods for mocking
75
+ sharesService = {
76
+ loadEncryptedShare: jest.fn().mockResolvedValue({ id: "shareId", addressId: "addressId", creatorEmail: "address@example.com", passphraseSessionKey: "sharePassphraseSessionKey" }),
77
+ getContextShareMemberEmailKey: jest.fn().mockResolvedValue({ email: "volume-email", addressId: "addressId", addressKey: "volume-key" }),
78
+ }
79
+ // @ts-expect-error No need to implement all methods for mocking
80
+ nodesService = {
81
+ getNode: jest.fn().mockImplementation((nodeUid) => ({ nodeUid, shareId: "shareId", name: { ok: true, value: "name" } })),
82
+ getNodeKeys: jest.fn().mockImplementation((nodeUid) => ({ key: "node-key" })),
83
+ getNodePrivateAndSessionKeys: jest.fn().mockImplementation((nodeUid) => ({})),
84
+ getRootNodeEmailKey: jest.fn().mockResolvedValue({ email: "volume-email", addressKey: "volume-key" }),
85
+ }
86
+ nodesEvents = {
87
+ nodeUpdated: jest.fn(),
88
+ }
89
+
90
+ sharingManagement = new SharingManagement(getMockLogger(), apiService, cryptoService, accountService, sharesService, nodesService, nodesEvents);
91
+ });
92
+
93
+ describe("getSharingInfo", () => {
94
+ it("should return empty sharing info for unshared node", async () => {
95
+ nodesService.getNode = jest.fn().mockResolvedValue({ nodeUid: "nodeUid", shareId: undefined });
96
+ const sharingInfo = await sharingManagement.getSharingInfo("nodeUid");
97
+
98
+ expect(sharingInfo).toEqual(undefined);
99
+ expect(apiService.getShareInvitations).not.toHaveBeenCalled();
100
+ expect(apiService.getShareExternalInvitations).not.toHaveBeenCalled();
101
+ expect(apiService.getShareMembers).not.toHaveBeenCalled();
102
+ });
103
+
104
+ it("should return invitations", async () => {
105
+ const invitation = { uid: "invitaiton", addedByEmail: "email" };
106
+ apiService.getShareInvitations = jest.fn().mockResolvedValue([
107
+ invitation,
108
+ ]);
109
+
110
+ const sharingInfo = await sharingManagement.getSharingInfo("nodeUid");
111
+
112
+ expect(sharingInfo).toEqual({
113
+ protonInvitations: [invitation],
114
+ nonProtonInvitations: [],
115
+ members: [],
116
+ publicLink: undefined,
117
+ });
118
+ expect(cryptoService.decryptInvitation).toHaveBeenCalledWith(invitation);
119
+ });
120
+
121
+ it("should return external invitations", async () => {
122
+ const externalInvitation = { uid: "external-invitation", addedByEmail: "email" };
123
+ apiService.getShareExternalInvitations = jest.fn().mockResolvedValue([
124
+ externalInvitation,
125
+ ]);
126
+
127
+ const sharingInfo = await sharingManagement.getSharingInfo("nodeUid");
128
+
129
+ expect(sharingInfo).toEqual({
130
+ protonInvitations: [],
131
+ nonProtonInvitations: [externalInvitation],
132
+ members: [],
133
+ publicLink: undefined,
134
+ });
135
+ expect(cryptoService.decryptExternalInvitation).toHaveBeenCalledWith(externalInvitation);
136
+ });
137
+
138
+ it("should return members", async () => {
139
+ const member = { uid: "member", addedByEmail: "email" };
140
+ apiService.getShareMembers = jest.fn().mockResolvedValue([
141
+ member,
142
+ ]);
143
+
144
+ const sharingInfo = await sharingManagement.getSharingInfo("nodeUid");
145
+
146
+ expect(sharingInfo).toEqual({
147
+ protonInvitations: [],
148
+ nonProtonInvitations: [],
149
+ members: [member],
150
+ publicLink: undefined,
151
+ });
152
+ expect(cryptoService.decryptMember).toHaveBeenCalledWith(member);
153
+ });
154
+
155
+ it("should return public link", async () => {
156
+ const publicLink = {
157
+ uid: 'shared~publicLink',
158
+ }
159
+ apiService.getPublicLink = jest.fn().mockResolvedValue(publicLink);
160
+
161
+ const sharingInfo = await sharingManagement.getSharingInfo("nodeUid");
162
+
163
+ expect(sharingInfo).toEqual({
164
+ protonInvitations: [],
165
+ nonProtonInvitations: [],
166
+ members: [],
167
+ publicLink: publicLink,
168
+ });
169
+ expect(cryptoService.decryptPublicLink).toHaveBeenCalledWith(publicLink);
170
+ });
171
+ });
172
+
173
+ describe("shareNode with share creation", () => {
174
+ const nodeUid = "volumeId~nodeUid";
175
+
176
+ it("should create share if no exists", async () => {
177
+ nodesService.getNode = jest.fn().mockImplementation((nodeUid) => ({ nodeUid, parentUid: 'parentUid', name: { ok: true, value: "name" } }));
178
+
179
+ const sharingInfo = await sharingManagement.shareNode(nodeUid, { users: ["email"] });
180
+
181
+ expect(sharingInfo).toEqual({
182
+ protonInvitations: [{
183
+ uid: "created-invitation",
184
+ addedByEmail: { ok: true, value: "volume-email" },
185
+ inviteeEmail: "email",
186
+ role: "viewer",
187
+ }],
188
+ nonProtonInvitations: [],
189
+ members: [],
190
+ publicLink: undefined,
191
+ });
192
+ expect(apiService.updateInvitation).not.toHaveBeenCalled();
193
+ expect(apiService.inviteProtonUser).toHaveBeenCalled();
194
+ expect(nodesEvents.nodeUpdated).toHaveBeenCalledWith({
195
+ uid: nodeUid,
196
+ shareId: "newShareId",
197
+ isShared: true,
198
+ });
199
+ });
200
+ })
201
+
202
+ describe("shareNode with share re-use", () => {
203
+ const nodeUid = "volumeId~nodeUid";
204
+
205
+ let invitation: ProtonInvitation;
206
+ let externalInvitation: NonProtonInvitation;
207
+ let member: Member;
208
+
209
+ beforeEach(async () => {
210
+ invitation = {
211
+ uid: "invitation",
212
+ addedByEmail: resultOk("added-email"),
213
+ inviteeEmail: "internal-email",
214
+ role: MemberRole.Viewer,
215
+ invitationTime: new Date(),
216
+ };
217
+ externalInvitation = {
218
+ uid: "external-invitation",
219
+ addedByEmail: resultOk("added-email"),
220
+ inviteeEmail: "external-email",
221
+ role: MemberRole.Viewer,
222
+ invitationTime: new Date(),
223
+ state: NonProtonInvitationState.Pending,
224
+ };
225
+ member = {
226
+ uid: "member",
227
+ addedByEmail: resultOk("added-email"),
228
+ inviteeEmail: "member-email",
229
+ role: MemberRole.Viewer,
230
+ invitationTime: new Date(),
231
+ };
232
+
233
+ apiService.getShareInvitations = jest.fn().mockResolvedValue([
234
+ invitation,
235
+ ]);
236
+
237
+ apiService.getShareExternalInvitations = jest.fn().mockResolvedValue([
238
+ externalInvitation,
239
+ ]);
240
+
241
+ apiService.getShareMembers = jest.fn().mockResolvedValue([
242
+ member,
243
+ ]);
244
+ });
245
+
246
+ describe("invitations", () => {
247
+ beforeEach(() => {
248
+ accountService.hasProtonAccount = jest.fn().mockResolvedValue(true);
249
+ });
250
+
251
+ it("should share node with proton email with default role", async () => {
252
+ const sharingInfo = await sharingManagement.shareNode(nodeUid, { users: ["email"] });
253
+
254
+ expect(sharingInfo).toEqual({
255
+ protonInvitations: [invitation, {
256
+ uid: "created-invitation",
257
+ addedByEmail: { ok: true, value: "volume-email" },
258
+ inviteeEmail: "email",
259
+ role: "viewer",
260
+ }],
261
+ nonProtonInvitations: [externalInvitation],
262
+ members: [member],
263
+ publicLink: undefined,
264
+ });
265
+ expect(apiService.updateInvitation).not.toHaveBeenCalled();
266
+ expect(apiService.inviteProtonUser).toHaveBeenCalled();
267
+ expect(nodesEvents.nodeUpdated).not.toHaveBeenCalled();
268
+ });
269
+
270
+ it("should share node with proton email with specific role", async () => {
271
+ const sharingInfo = await sharingManagement.shareNode(nodeUid, { users: [{ email: "email", role: MemberRole.Editor }] });
272
+
273
+ expect(sharingInfo).toEqual({
274
+ protonInvitations: [invitation, {
275
+ uid: "created-invitation",
276
+ addedByEmail: { ok: true, value: "volume-email" },
277
+ inviteeEmail: "email",
278
+ role: "editor",
279
+ }],
280
+ nonProtonInvitations: [externalInvitation],
281
+ members: [member],
282
+ publicLink: undefined,
283
+ });
284
+ expect(apiService.updateInvitation).not.toHaveBeenCalled();
285
+ expect(apiService.inviteProtonUser).toHaveBeenCalled();
286
+ expect(nodesEvents.nodeUpdated).not.toHaveBeenCalled();
287
+ });
288
+
289
+ it("should update existing role", async () => {
290
+ const sharingInfo = await sharingManagement.shareNode(nodeUid, { users: [{ email: "internal-email", role: MemberRole.Editor }] });
291
+
292
+ expect(sharingInfo).toEqual({
293
+ protonInvitations: [{
294
+ ...invitation,
295
+ role: "editor",
296
+ }],
297
+ nonProtonInvitations: [externalInvitation],
298
+ members: [member],
299
+ publicLink: undefined,
300
+ });
301
+ expect(apiService.updateInvitation).toHaveBeenCalled();
302
+ expect(apiService.inviteProtonUser).not.toHaveBeenCalled();
303
+ expect(nodesEvents.nodeUpdated).not.toHaveBeenCalled();
304
+ });
305
+
306
+ it("should be no-op if no change", async () => {
307
+ const sharingInfo = await sharingManagement.shareNode(nodeUid, { users: [{ email: "internal-email", role: MemberRole.Viewer }] });
308
+
309
+ expect(sharingInfo).toEqual({
310
+ protonInvitations: [invitation],
311
+ nonProtonInvitations: [externalInvitation],
312
+ members: [member],
313
+ publicLink: undefined,
314
+ });
315
+ expect(apiService.updateInvitation).not.toHaveBeenCalled();
316
+ expect(apiService.inviteProtonUser).not.toHaveBeenCalled();
317
+ expect(nodesEvents.nodeUpdated).not.toHaveBeenCalled();
318
+ });
319
+ });
320
+
321
+ describe("external invitations", () => {
322
+ beforeEach(() => {
323
+ accountService.hasProtonAccount = jest.fn().mockResolvedValue(false);
324
+ });
325
+
326
+ it("should share node with external email with default role", async () => {
327
+ const sharingInfo = await sharingManagement.shareNode(nodeUid, { users: ["email"] });
328
+
329
+ expect(sharingInfo).toEqual({
330
+ protonInvitations: [invitation],
331
+ nonProtonInvitations: [externalInvitation, {
332
+ uid: "created-external-invitation",
333
+ addedByEmail: { ok: true, value: "volume-email" },
334
+ inviteeEmail: "email",
335
+ role: "viewer",
336
+ state: "pending",
337
+ }],
338
+ members: [member],
339
+ publicLink: undefined,
340
+ });
341
+ expect(apiService.updateExternalInvitation).not.toHaveBeenCalled();
342
+ expect(apiService.inviteExternalUser).toHaveBeenCalled();
343
+ expect(nodesEvents.nodeUpdated).not.toHaveBeenCalled();
344
+ });
345
+
346
+ it("should share node with external email with specific role", async () => {
347
+ const sharingInfo = await sharingManagement.shareNode(nodeUid, { users: [{ email: "email", role: MemberRole.Editor }] });
348
+
349
+ expect(sharingInfo).toEqual({
350
+ protonInvitations: [invitation],
351
+ nonProtonInvitations: [externalInvitation, {
352
+ uid: "created-external-invitation",
353
+ addedByEmail: { ok: true, value: "volume-email" },
354
+ inviteeEmail: "email",
355
+ role: "editor",
356
+ state: "pending",
357
+ }],
358
+ members: [member],
359
+ publicLink: undefined,
360
+ });
361
+ expect(apiService.updateExternalInvitation).not.toHaveBeenCalled();
362
+ expect(apiService.inviteExternalUser).toHaveBeenCalled();
363
+ expect(nodesEvents.nodeUpdated).not.toHaveBeenCalled();
364
+ });
365
+
366
+ it("should update existing role", async () => {
367
+ const sharingInfo = await sharingManagement.shareNode(nodeUid, { users: [{ email: "external-email", role: MemberRole.Editor }] });
368
+
369
+ expect(sharingInfo).toEqual({
370
+ protonInvitations: [invitation],
371
+ nonProtonInvitations: [{
372
+ ...externalInvitation,
373
+ role: "editor",
374
+ }],
375
+ members: [member],
376
+ publicLink: undefined,
377
+ });
378
+ expect(apiService.updateExternalInvitation).toHaveBeenCalled();
379
+ expect(apiService.inviteExternalUser).not.toHaveBeenCalled();
380
+ expect(nodesEvents.nodeUpdated).not.toHaveBeenCalled();
381
+ });
382
+
383
+ it("should be no-op if no change", async () => {
384
+ const sharingInfo = await sharingManagement.shareNode(nodeUid, { users: [{ email: "external-email", role: MemberRole.Viewer }] });
385
+
386
+ expect(sharingInfo).toEqual({
387
+ protonInvitations: [invitation],
388
+ nonProtonInvitations: [externalInvitation],
389
+ members: [member],
390
+ publicLink: undefined,
391
+ });
392
+ expect(apiService.updateExternalInvitation).not.toHaveBeenCalled();
393
+ expect(apiService.inviteExternalUser).not.toHaveBeenCalled();
394
+ expect(nodesEvents.nodeUpdated).not.toHaveBeenCalled();
395
+ });
396
+ });
397
+
398
+ describe("mix of internal and external invitations", () => {
399
+ beforeEach(() => {
400
+ accountService.hasProtonAccount = jest.fn()
401
+ .mockResolvedValueOnce(true)
402
+ .mockResolvedValueOnce(false);
403
+ });
404
+
405
+ it("should share node with proton and external email with default role", async () => {
406
+ const sharingInfo = await sharingManagement.shareNode(nodeUid, { users: ["email", "email2"] });
407
+
408
+ expect(sharingInfo).toEqual({
409
+ protonInvitations: [invitation, {
410
+ uid: "created-invitation",
411
+ addedByEmail: { ok: true, value: "volume-email" },
412
+ inviteeEmail: "email",
413
+ role: "viewer",
414
+ }],
415
+ nonProtonInvitations: [externalInvitation, {
416
+ uid: "created-external-invitation",
417
+ addedByEmail: { ok: true, value: "volume-email" },
418
+ inviteeEmail: "email2",
419
+ role: "viewer",
420
+ state: "pending",
421
+ }],
422
+ members: [member],
423
+ publicLink: undefined,
424
+ });
425
+ expect(apiService.updateInvitation).not.toHaveBeenCalled();
426
+ expect(apiService.inviteProtonUser).toHaveBeenCalledWith("shareId", expect.objectContaining({
427
+ inviteeEmail: "email",
428
+ }), expect.anything());
429
+ expect(apiService.inviteExternalUser).toHaveBeenCalledWith("shareId", expect.objectContaining({
430
+ inviteeEmail: "email2",
431
+ }), expect.anything());
432
+ expect(nodesEvents.nodeUpdated).not.toHaveBeenCalled();
433
+ });
434
+ });
435
+
436
+ describe("members", () => {
437
+ it("should update member via proton user", async () => {
438
+ const sharingInfo = await sharingManagement.shareNode(nodeUid, { users: [{ email: "member-email", role: MemberRole.Editor }] });
439
+
440
+ expect(sharingInfo).toEqual({
441
+ protonInvitations: [invitation],
442
+ nonProtonInvitations: [externalInvitation],
443
+ members: [{
444
+ ...member,
445
+ role: "editor",
446
+ }],
447
+ publicLink: undefined,
448
+ });
449
+ expect(apiService.updateMember).toHaveBeenCalled();
450
+ expect(apiService.updateInvitation).not.toHaveBeenCalled();
451
+ expect(apiService.inviteProtonUser).not.toHaveBeenCalled();
452
+ expect(nodesEvents.nodeUpdated).not.toHaveBeenCalled();
453
+ });
454
+
455
+ it("should be no-op if no change via proton user", async () => {
456
+ const sharingInfo = await sharingManagement.shareNode(nodeUid, { users: [{ email: "member-email", role: MemberRole.Viewer }] });
457
+
458
+ expect(sharingInfo).toEqual({
459
+ protonInvitations: [invitation],
460
+ nonProtonInvitations: [externalInvitation],
461
+ members: [member],
462
+ publicLink: undefined,
463
+ });
464
+ expect(apiService.updateMember).not.toHaveBeenCalled();
465
+ expect(apiService.updateInvitation).not.toHaveBeenCalled();
466
+ expect(apiService.inviteProtonUser).not.toHaveBeenCalled();
467
+ expect(nodesEvents.nodeUpdated).not.toHaveBeenCalled();
468
+ });
469
+
470
+ it("should update member via non-proton user", async () => {
471
+ const sharingInfo = await sharingManagement.shareNode(nodeUid, { users: [{ email: "member-email", role: MemberRole.Editor }] });
472
+
473
+ expect(sharingInfo).toEqual({
474
+ protonInvitations: [invitation],
475
+ nonProtonInvitations: [externalInvitation],
476
+ members: [{
477
+ ...member,
478
+ role: "editor",
479
+ }],
480
+ publicLink: undefined,
481
+ });
482
+ expect(apiService.updateMember).toHaveBeenCalled();
483
+ expect(apiService.updateInvitation).not.toHaveBeenCalled();
484
+ expect(apiService.inviteProtonUser).not.toHaveBeenCalled();
485
+ expect(nodesEvents.nodeUpdated).not.toHaveBeenCalled();
486
+ });
487
+
488
+ it("should be no-op if no change via non-proton user", async () => {
489
+ const sharingInfo = await sharingManagement.shareNode(nodeUid, { users: [{ email: "member-email", role: MemberRole.Viewer }] });
490
+
491
+ expect(sharingInfo).toEqual({
492
+ protonInvitations: [invitation],
493
+ nonProtonInvitations: [externalInvitation],
494
+ members: [member],
495
+ publicLink: undefined,
496
+ });
497
+ expect(apiService.updateMember).not.toHaveBeenCalled();
498
+ expect(apiService.updateInvitation).not.toHaveBeenCalled();
499
+ expect(apiService.inviteProtonUser).not.toHaveBeenCalled();
500
+ expect(nodesEvents.nodeUpdated).not.toHaveBeenCalled();
501
+ });
502
+ });
503
+
504
+ describe("public link", () => {
505
+ it("should share node with public link", async () => {
506
+ jest.useFakeTimers();
507
+ jest.setSystemTime(new Date('2025-01-01'));
508
+
509
+ const sharingInfo = await sharingManagement.shareNode(nodeUid, {
510
+ publicLink: {
511
+ role: MemberRole.Viewer,
512
+ customPassword: undefined,
513
+ expiration: undefined,
514
+ },
515
+ });
516
+
517
+ expect(sharingInfo).toEqual({
518
+ protonInvitations: [invitation],
519
+ nonProtonInvitations: [externalInvitation],
520
+ members: [member],
521
+ publicLink: {
522
+ uid: "publicLinkUid",
523
+ role: MemberRole.Viewer,
524
+ url: "publicLinkUrl#generatedPassword",
525
+ creationTime: new Date(),
526
+ expirationTime: undefined,
527
+ customPassword: undefined,
528
+ creatorEmail: "volume-email",
529
+ },
530
+ });
531
+ expect(cryptoService.generatePublicLinkPassword).toHaveBeenCalled();
532
+ expect(cryptoService.encryptPublicLink).toHaveBeenCalledWith("volume-email", "sharePassphraseSessionKey", "generatedPassword");
533
+ expect(apiService.createPublicLink).toHaveBeenCalledWith("shareId", expect.objectContaining({
534
+ role: MemberRole.Viewer,
535
+ includesCustomPassword: false,
536
+ expirationDuration: undefined,
537
+ crypto: "publicLinkCrypto",
538
+ srp: "publicLinkSrp",
539
+ }));
540
+ });
541
+
542
+ it("should share node with custom password and expiration", async () => {
543
+ jest.useFakeTimers();
544
+ jest.setSystemTime(new Date('2025-01-01'));
545
+
546
+ const sharingInfo = await sharingManagement.shareNode(nodeUid, {
547
+ publicLink: {
548
+ role: MemberRole.Viewer,
549
+ customPassword: "customPassword",
550
+ expiration: new Date('2025-01-02'),
551
+ },
552
+ });
553
+
554
+ expect(sharingInfo).toEqual({
555
+ protonInvitations: [invitation],
556
+ nonProtonInvitations: [externalInvitation],
557
+ members: [member],
558
+ publicLink: {
559
+ uid: "publicLinkUid",
560
+ role: MemberRole.Viewer,
561
+ url: "publicLinkUrl#generatedPassword",
562
+ creationTime: new Date(),
563
+ expirationTime: new Date('2025-01-02'),
564
+ customPassword: "customPassword",
565
+ creatorEmail: "volume-email",
566
+ },
567
+ });
568
+ expect(cryptoService.generatePublicLinkPassword).toHaveBeenCalled();
569
+ expect(cryptoService.encryptPublicLink).toHaveBeenCalledWith("volume-email", "sharePassphraseSessionKey", "generatedPasswordcustomPassword");
570
+ expect(apiService.createPublicLink).toHaveBeenCalledWith("shareId", expect.objectContaining({
571
+ role: MemberRole.Viewer,
572
+ includesCustomPassword: true,
573
+ expirationDuration: 86400,
574
+ crypto: "publicLinkCrypto",
575
+ srp: "publicLinkSrp",
576
+ }));
577
+ });
578
+
579
+ it("should update public link with custom password and expiration", async () => {
580
+ jest.useFakeTimers();
581
+ jest.setSystemTime(new Date('2025-01-01'));
582
+
583
+ const publicLink = {
584
+ uid: 'publicLinkUid',
585
+ url: "publicLinkUrl#generatedpas", // Generated password must be 12 chararacters long.
586
+ creationTime: new Date('2025-01-01'),
587
+ role: MemberRole.Viewer,
588
+ customPassword: undefined,
589
+ expirationTime: undefined,
590
+ creatorEmail: "publicLinkCreatorEmail",
591
+ }
592
+ apiService.getPublicLink = jest.fn().mockResolvedValue(publicLink);
593
+
594
+ const sharingInfo = await sharingManagement.shareNode(nodeUid, {
595
+ publicLink: {
596
+ role: MemberRole.Editor,
597
+ customPassword: "customPassword",
598
+ expiration: new Date('2025-01-02'),
599
+ },
600
+ });
601
+
602
+ expect(sharingInfo).toEqual({
603
+ protonInvitations: [invitation],
604
+ nonProtonInvitations: [externalInvitation],
605
+ members: [member],
606
+ publicLink: {
607
+ uid: "publicLinkUid",
608
+ role: MemberRole.Editor,
609
+ url: "publicLinkUrl#generatedpas",
610
+ creationTime: new Date('2025-01-01'),
611
+ expirationTime: new Date('2025-01-02'),
612
+ customPassword: "customPassword",
613
+ creatorEmail: "publicLinkCreatorEmail",
614
+ },
615
+ });
616
+ expect(cryptoService.encryptPublicLink).toHaveBeenCalledWith("publicLinkCreatorEmail", "sharePassphraseSessionKey", "generatedpascustomPassword");
617
+ expect(apiService.updatePublicLink).toHaveBeenCalledWith("publicLinkUid", expect.objectContaining({
618
+ role: MemberRole.Editor,
619
+ includesCustomPassword: true,
620
+ expirationDuration: 86400,
621
+ crypto: "publicLinkCrypto",
622
+ srp: "publicLinkSrp",
623
+ }));
624
+ });
625
+
626
+ it("should not allow updating legacy public link", async () => {
627
+ apiService.getPublicLink = jest.fn().mockResolvedValue({
628
+ uid: 'publicLinkUid',
629
+ url: "publicLinkUrl#aaa", // Legacy public links doesn't have 12 chars.
630
+ });
631
+
632
+ await expect(sharingManagement.shareNode(nodeUid, {
633
+ publicLink: true,
634
+ })).rejects.toThrow("Legacy public link cannot be updated. Please re-create a new public link.");
635
+ });
636
+
637
+ it("should not allow updating legacy public link without generated password", async () => {
638
+ apiService.getPublicLink = jest.fn().mockResolvedValue({
639
+ uid: 'publicLinkUid',
640
+ url: "publicLinkUrl",
641
+ });
642
+
643
+ await expect(sharingManagement.shareNode(nodeUid, {
644
+ publicLink: true,
645
+ })).rejects.toThrow("Legacy public link cannot be updated. Please re-create a new public link.");
646
+ });
647
+
648
+ it("should not allow creating public link with expiration in the past", async () => {
649
+ jest.useFakeTimers();
650
+ jest.setSystemTime(new Date('2025-01-01'));
651
+
652
+ await expect(sharingManagement.shareNode(nodeUid, {
653
+ publicLink: {
654
+ role: MemberRole.Viewer,
655
+ expiration: new Date('2024-01-01'),
656
+ },
657
+ })).rejects.toThrow("Expiration date cannot be in the past");
658
+ expect(apiService.createStandardShare).not.toHaveBeenCalled();
659
+ expect(apiService.createPublicLink).not.toHaveBeenCalled();
660
+ });
661
+ });
662
+ });
663
+
664
+ describe("unsahreNode", () => {
665
+ const nodeUid = "volumeId~nodeUid";
666
+
667
+ let invitation: ProtonInvitation;
668
+ let externalInvitation: NonProtonInvitation;
669
+ let member: Member;
670
+ let publicLink: PublicLink;
671
+
672
+ beforeEach(async () => {
673
+ invitation = {
674
+ uid: "invitation",
675
+ addedByEmail: resultOk("added-email"),
676
+ inviteeEmail: "internal-email",
677
+ role: MemberRole.Viewer,
678
+ invitationTime: new Date(),
679
+ };
680
+ externalInvitation = {
681
+ uid: "external-invitation",
682
+ addedByEmail: resultOk("added-email"),
683
+ inviteeEmail: "external-email",
684
+ role: MemberRole.Viewer,
685
+ invitationTime: new Date(),
686
+ state: NonProtonInvitationState.Pending,
687
+ };
688
+ member = {
689
+ uid: "member",
690
+ addedByEmail: resultOk("added-email"),
691
+ inviteeEmail: "member-email",
692
+ role: MemberRole.Viewer,
693
+ invitationTime: new Date(),
694
+ };
695
+ publicLink = {
696
+ uid: "publicLink",
697
+ creationTime: new Date(),
698
+ role: MemberRole.Viewer,
699
+ url: "url",
700
+ }
701
+
702
+ apiService.getShareInvitations = jest.fn().mockResolvedValue([
703
+ invitation,
704
+ ]);
705
+ apiService.getShareExternalInvitations = jest.fn().mockResolvedValue([
706
+ externalInvitation,
707
+ ]);
708
+ apiService.getShareMembers = jest.fn().mockResolvedValue([
709
+ member,
710
+ ]);
711
+ apiService.getPublicLink = jest.fn().mockResolvedValue(publicLink);
712
+ });
713
+
714
+ it("should delete invitation", async () => {
715
+ const sharingInfo = await sharingManagement.unshareNode(nodeUid, { users: ["internal-email"] });
716
+
717
+ expect(sharingInfo).toEqual({
718
+ protonInvitations: [],
719
+ nonProtonInvitations: [externalInvitation],
720
+ members: [member],
721
+ publicLink,
722
+ });
723
+ expect(apiService.deleteShare).not.toHaveBeenCalled();
724
+ expect(apiService.deleteInvitation).toHaveBeenCalled();
725
+ expect(apiService.deleteExternalInvitation).not.toHaveBeenCalled();
726
+ expect(apiService.removeMember).not.toHaveBeenCalled();
727
+ expect(apiService.removePublicLink).not.toHaveBeenCalled();
728
+ expect(nodesEvents.nodeUpdated).not.toHaveBeenCalled();
729
+ });
730
+
731
+ it("should delete external invitation", async () => {
732
+ const sharingInfo = await sharingManagement.unshareNode(nodeUid, { users: ["external-email"] });
733
+
734
+ expect(sharingInfo).toEqual({
735
+ protonInvitations: [invitation],
736
+ nonProtonInvitations: [],
737
+ members: [member],
738
+ publicLink,
739
+ });
740
+ expect(apiService.deleteShare).not.toHaveBeenCalled();
741
+ expect(apiService.deleteInvitation).not.toHaveBeenCalled();
742
+ expect(apiService.deleteExternalInvitation).toHaveBeenCalled();
743
+ expect(apiService.removeMember).not.toHaveBeenCalled();
744
+ expect(apiService.removePublicLink).not.toHaveBeenCalled();
745
+ expect(nodesEvents.nodeUpdated).not.toHaveBeenCalled();
746
+ });
747
+
748
+ it("should remove member", async () => {
749
+ const sharingInfo = await sharingManagement.unshareNode(nodeUid, { users: ["member-email"] });
750
+
751
+ expect(sharingInfo).toEqual({
752
+ protonInvitations: [invitation],
753
+ nonProtonInvitations: [externalInvitation],
754
+ members: [],
755
+ publicLink,
756
+ });
757
+ expect(apiService.deleteShare).not.toHaveBeenCalled();
758
+ expect(apiService.deleteInvitation).not.toHaveBeenCalled();
759
+ expect(apiService.deleteExternalInvitation).not.toHaveBeenCalled();
760
+ expect(apiService.removeMember).toHaveBeenCalled();
761
+ expect(apiService.removePublicLink).not.toHaveBeenCalled();
762
+ expect(nodesEvents.nodeUpdated).not.toHaveBeenCalled();
763
+ });
764
+
765
+ it("should be no-op if not shared with email", async () => {
766
+ const sharingInfo = await sharingManagement.unshareNode(nodeUid, { users: ["non-existing-email"] });
767
+
768
+ expect(sharingInfo).toEqual({
769
+ protonInvitations: [invitation],
770
+ nonProtonInvitations: [externalInvitation],
771
+ members: [member],
772
+ publicLink,
773
+ });
774
+ expect(apiService.deleteShare).not.toHaveBeenCalled();
775
+ expect(apiService.deleteInvitation).not.toHaveBeenCalled();
776
+ expect(apiService.deleteExternalInvitation).not.toHaveBeenCalled();
777
+ expect(apiService.removeMember).not.toHaveBeenCalled();
778
+ expect(apiService.removePublicLink).not.toHaveBeenCalled();
779
+ expect(nodesEvents.nodeUpdated).not.toHaveBeenCalled();
780
+ });
781
+
782
+ it("should remove public link", async () => {
783
+ const sharingInfo = await sharingManagement.unshareNode(nodeUid, { publicLink: "remove" });
784
+
785
+ expect(sharingInfo).toEqual({
786
+ protonInvitations: [invitation],
787
+ nonProtonInvitations: [externalInvitation],
788
+ members: [member],
789
+ publicLink: undefined,
790
+ });
791
+ expect(apiService.deleteShare).not.toHaveBeenCalled();
792
+ expect(apiService.deleteInvitation).not.toHaveBeenCalled();
793
+ expect(apiService.deleteExternalInvitation).not.toHaveBeenCalled();
794
+ expect(apiService.removeMember).not.toHaveBeenCalled();
795
+ expect(apiService.removePublicLink).toHaveBeenCalled();
796
+ expect(nodesEvents.nodeUpdated).not.toHaveBeenCalled();
797
+ });
798
+
799
+ it("should remove share if all is removed", async () => {
800
+ const sharingInfo = await sharingManagement.unshareNode(nodeUid);
801
+
802
+ expect(sharingInfo).toEqual(undefined);
803
+ expect(apiService.deleteShare).toHaveBeenCalled();
804
+ expect(apiService.deleteInvitation).not.toHaveBeenCalled();
805
+ expect(apiService.deleteExternalInvitation).not.toHaveBeenCalled();
806
+ expect(apiService.removeMember).not.toHaveBeenCalled();
807
+ expect(apiService.removePublicLink).not.toHaveBeenCalled();
808
+ expect(nodesEvents.nodeUpdated).toHaveBeenCalledWith({
809
+ uid: nodeUid,
810
+ shareId: undefined,
811
+ isShared: false,
812
+ });
813
+ });
814
+
815
+ it("should remove share if everything is manually removed", async () => {
816
+ const sharingInfo = await sharingManagement.unshareNode(nodeUid, {
817
+ users: ["internal-email", "external-email", "member-email"],
818
+ publicLink: "remove",
819
+ });
820
+
821
+ expect(sharingInfo).toEqual(undefined);
822
+ expect(apiService.deleteShare).toHaveBeenCalled();
823
+ expect(apiService.deleteInvitation).toHaveBeenCalled();
824
+ expect(apiService.deleteExternalInvitation).toHaveBeenCalled();
825
+ expect(apiService.removeMember).toHaveBeenCalled();
826
+ expect(apiService.removePublicLink).toHaveBeenCalled();
827
+ expect(nodesEvents.nodeUpdated).toHaveBeenCalledWith({
828
+ uid: nodeUid,
829
+ shareId: undefined,
830
+ isShared: false,
831
+ });
832
+ });
833
+ });
834
+
835
+ describe("resendInvitationEmail", () => {
836
+ const nodeUid = "volumeId~nodeUid";
837
+
838
+ const invitation: ProtonInvitation = {
839
+ uid: "invitation",
840
+ addedByEmail: resultOk("added-email"),
841
+ inviteeEmail: "internal-email",
842
+ role: MemberRole.Viewer,
843
+ invitationTime: new Date(),
844
+ };
845
+ const externalInvitation: NonProtonInvitation = {
846
+ uid: "external-invitation",
847
+ addedByEmail: resultOk("added-email"),
848
+ inviteeEmail: "external-email",
849
+ role: MemberRole.Viewer,
850
+ invitationTime: new Date(),
851
+ state: NonProtonInvitationState.Pending,
852
+ };
853
+
854
+ beforeEach(() => {
855
+ apiService.getShareInvitations = jest.fn().mockResolvedValue([invitation]);
856
+ apiService.getShareExternalInvitations = jest.fn().mockResolvedValue([externalInvitation]);
857
+ apiService.getShareMembers = jest.fn().mockResolvedValue([]);
858
+ apiService.getPublicLink = jest.fn().mockResolvedValue(undefined);
859
+ });
860
+
861
+ it("should resend email for proton invitation", async () => {
862
+ await sharingManagement.resendInvitationEmail(nodeUid, invitation.uid);
863
+
864
+ expect(apiService.resendInvitationEmail).toHaveBeenCalledWith(invitation.uid);
865
+ expect(apiService.resendExternalInvitationEmail).not.toHaveBeenCalled();
866
+ });
867
+
868
+ it("should resend email for external invitation", async () => {
869
+ await sharingManagement.resendInvitationEmail(nodeUid, externalInvitation.uid);
870
+
871
+ expect(apiService.resendExternalInvitationEmail).toHaveBeenCalledWith(externalInvitation.uid);
872
+ expect(apiService.resendInvitationEmail).not.toHaveBeenCalled();
873
+ });
874
+
875
+ it("should throw error when no sharing found for node", async () => {
876
+ nodesService.getNode = jest.fn().mockResolvedValue({ nodeUid, shareId: undefined });
877
+
878
+ await expect(
879
+ sharingManagement.resendInvitationEmail(nodeUid, invitation.uid)
880
+ ).rejects.toThrow("Node is not shared");
881
+
882
+ expect(apiService.resendInvitationEmail).not.toHaveBeenCalled();
883
+ expect(apiService.resendExternalInvitationEmail).not.toHaveBeenCalled();
884
+ });
885
+
886
+ it("should log when no invitation found", async () => {
887
+ await expect(
888
+ sharingManagement.resendInvitationEmail(nodeUid, "non-existent-uid")
889
+ ).rejects.toThrow("Invitation not found");
890
+
891
+ expect(apiService.resendInvitationEmail).not.toHaveBeenCalled();
892
+ expect(apiService.resendExternalInvitationEmail).not.toHaveBeenCalled();
893
+ });
894
+ });
895
+ });