@protontech/drive-sdk 0.0.12 → 0.1.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.
Files changed (274) hide show
  1. package/dist/cache/index.d.ts +1 -0
  2. package/dist/cache/index.js +3 -1
  3. package/dist/cache/index.js.map +1 -1
  4. package/dist/cache/memoryCache.d.ts +1 -1
  5. package/dist/cache/nullCache.d.ts +14 -0
  6. package/dist/cache/nullCache.js +37 -0
  7. package/dist/cache/nullCache.js.map +1 -0
  8. package/dist/config.d.ts +16 -1
  9. package/dist/config.js +1 -1
  10. package/dist/config.js.map +1 -1
  11. package/dist/crypto/openPGPCrypto.js +2 -0
  12. package/dist/crypto/openPGPCrypto.js.map +1 -1
  13. package/dist/diagnostic/eventsGenerator.d.ts +14 -0
  14. package/dist/diagnostic/eventsGenerator.js +49 -0
  15. package/dist/diagnostic/eventsGenerator.js.map +1 -0
  16. package/dist/diagnostic/httpClient.d.ts +16 -0
  17. package/dist/diagnostic/httpClient.js +81 -0
  18. package/dist/diagnostic/httpClient.js.map +1 -0
  19. package/dist/diagnostic/index.d.ts +10 -0
  20. package/dist/diagnostic/index.js +35 -0
  21. package/dist/diagnostic/index.js.map +1 -0
  22. package/dist/diagnostic/integrityVerificationStream.d.ts +21 -0
  23. package/dist/diagnostic/integrityVerificationStream.js +56 -0
  24. package/dist/diagnostic/integrityVerificationStream.js.map +1 -0
  25. package/dist/diagnostic/interface.d.ts +102 -0
  26. package/dist/diagnostic/interface.js +3 -0
  27. package/dist/diagnostic/interface.js.map +1 -0
  28. package/dist/diagnostic/sdkDiagnostic.d.ts +22 -0
  29. package/dist/diagnostic/sdkDiagnostic.js +216 -0
  30. package/dist/diagnostic/sdkDiagnostic.js.map +1 -0
  31. package/dist/diagnostic/sdkDiagnosticFull.d.ts +18 -0
  32. package/dist/diagnostic/sdkDiagnosticFull.js +35 -0
  33. package/dist/diagnostic/sdkDiagnosticFull.js.map +1 -0
  34. package/dist/diagnostic/telemetry.d.ts +25 -0
  35. package/dist/diagnostic/telemetry.js +70 -0
  36. package/dist/diagnostic/telemetry.js.map +1 -0
  37. package/dist/diagnostic/zipGenerators.d.ts +9 -0
  38. package/dist/diagnostic/zipGenerators.js +64 -0
  39. package/dist/diagnostic/zipGenerators.js.map +1 -0
  40. package/dist/diagnostic/zipGenerators.test.js +144 -0
  41. package/dist/diagnostic/zipGenerators.test.js.map +1 -0
  42. package/dist/errors.d.ts +8 -3
  43. package/dist/errors.js +11 -4
  44. package/dist/errors.js.map +1 -1
  45. package/dist/interface/config.d.ts +26 -0
  46. package/dist/interface/config.js +3 -0
  47. package/dist/interface/config.js.map +1 -0
  48. package/dist/interface/download.d.ts +2 -2
  49. package/dist/interface/events.d.ts +60 -20
  50. package/dist/interface/events.js +11 -1
  51. package/dist/interface/events.js.map +1 -1
  52. package/dist/interface/httpClient.d.ts +0 -14
  53. package/dist/interface/index.d.ts +9 -5
  54. package/dist/interface/index.js +2 -1
  55. package/dist/interface/index.js.map +1 -1
  56. package/dist/interface/nodes.d.ts +21 -1
  57. package/dist/interface/nodes.js +11 -0
  58. package/dist/interface/nodes.js.map +1 -1
  59. package/dist/interface/sharing.d.ts +1 -0
  60. package/dist/interface/upload.d.ts +57 -3
  61. package/dist/internal/apiService/driveTypes.d.ts +1341 -465
  62. package/dist/internal/apiService/errors.js +2 -2
  63. package/dist/internal/apiService/errors.js.map +1 -1
  64. package/dist/internal/apiService/transformers.js +2 -0
  65. package/dist/internal/apiService/transformers.js.map +1 -1
  66. package/dist/internal/asyncIteratorMap.d.ts +15 -0
  67. package/dist/internal/asyncIteratorMap.js +59 -0
  68. package/dist/internal/asyncIteratorMap.js.map +1 -0
  69. package/dist/internal/asyncIteratorMap.test.js +120 -0
  70. package/dist/internal/asyncIteratorMap.test.js.map +1 -0
  71. package/dist/internal/download/apiService.js +32 -31
  72. package/dist/internal/download/apiService.js.map +1 -1
  73. package/dist/internal/download/fileDownloader.d.ts +2 -2
  74. package/dist/internal/download/fileDownloader.js.map +1 -1
  75. package/dist/internal/events/apiService.d.ts +4 -6
  76. package/dist/internal/events/apiService.js +15 -22
  77. package/dist/internal/events/apiService.js.map +1 -1
  78. package/dist/internal/events/coreEventManager.d.ts +7 -10
  79. package/dist/internal/events/coreEventManager.js +19 -36
  80. package/dist/internal/events/coreEventManager.js.map +1 -1
  81. package/dist/internal/events/coreEventManager.test.d.ts +1 -0
  82. package/dist/internal/events/coreEventManager.test.js +87 -0
  83. package/dist/internal/events/coreEventManager.test.js.map +1 -0
  84. package/dist/internal/events/eventManager.d.ts +11 -36
  85. package/dist/internal/events/eventManager.js +59 -105
  86. package/dist/internal/events/eventManager.js.map +1 -1
  87. package/dist/internal/events/eventManager.test.js +167 -82
  88. package/dist/internal/events/eventManager.test.js.map +1 -1
  89. package/dist/internal/events/index.d.ts +13 -33
  90. package/dist/internal/events/index.js +56 -72
  91. package/dist/internal/events/index.js.map +1 -1
  92. package/dist/internal/events/interface.d.ts +59 -14
  93. package/dist/internal/events/interface.js +13 -3
  94. package/dist/internal/events/interface.js.map +1 -1
  95. package/dist/internal/events/volumeEventManager.d.ts +7 -17
  96. package/dist/internal/events/volumeEventManager.js +58 -45
  97. package/dist/internal/events/volumeEventManager.js.map +1 -1
  98. package/dist/internal/events/volumeEventManager.test.d.ts +1 -0
  99. package/dist/internal/events/volumeEventManager.test.js +203 -0
  100. package/dist/internal/events/volumeEventManager.test.js.map +1 -0
  101. package/dist/internal/nodes/apiService.d.ts +2 -2
  102. package/dist/internal/nodes/apiService.js +16 -6
  103. package/dist/internal/nodes/apiService.js.map +1 -1
  104. package/dist/internal/nodes/apiService.test.js +30 -8
  105. package/dist/internal/nodes/apiService.test.js.map +1 -1
  106. package/dist/internal/nodes/cache.d.ts +10 -1
  107. package/dist/internal/nodes/cache.js +18 -0
  108. package/dist/internal/nodes/cache.js.map +1 -1
  109. package/dist/internal/nodes/cache.test.js +1 -0
  110. package/dist/internal/nodes/cache.test.js.map +1 -1
  111. package/dist/internal/nodes/cryptoService.d.ts +1 -1
  112. package/dist/internal/nodes/cryptoService.js.map +1 -1
  113. package/dist/internal/nodes/cryptoService.test.js +34 -0
  114. package/dist/internal/nodes/cryptoService.test.js.map +1 -1
  115. package/dist/internal/nodes/events.d.ts +7 -83
  116. package/dist/internal/nodes/events.js +43 -217
  117. package/dist/internal/nodes/events.js.map +1 -1
  118. package/dist/internal/nodes/events.test.js +27 -277
  119. package/dist/internal/nodes/events.test.js.map +1 -1
  120. package/dist/internal/nodes/index.d.ts +3 -4
  121. package/dist/internal/nodes/index.js +5 -5
  122. package/dist/internal/nodes/index.js.map +1 -1
  123. package/dist/internal/nodes/interface.d.ts +3 -1
  124. package/dist/internal/nodes/nodesAccess.d.ts +15 -0
  125. package/dist/internal/nodes/nodesAccess.js +65 -7
  126. package/dist/internal/nodes/nodesAccess.js.map +1 -1
  127. package/dist/internal/nodes/nodesAccess.test.js +132 -93
  128. package/dist/internal/nodes/nodesAccess.test.js.map +1 -1
  129. package/dist/internal/nodes/nodesManagement.d.ts +1 -3
  130. package/dist/internal/nodes/nodesManagement.js +12 -26
  131. package/dist/internal/nodes/nodesManagement.js.map +1 -1
  132. package/dist/internal/nodes/nodesManagement.test.js +35 -14
  133. package/dist/internal/nodes/nodesManagement.test.js.map +1 -1
  134. package/dist/internal/shares/cache.d.ts +2 -0
  135. package/dist/internal/shares/cache.js +2 -0
  136. package/dist/internal/shares/cache.js.map +1 -1
  137. package/dist/internal/shares/manager.d.ts +1 -0
  138. package/dist/internal/shares/manager.js +3 -0
  139. package/dist/internal/shares/manager.js.map +1 -1
  140. package/dist/internal/sharing/apiService.js +20 -2
  141. package/dist/internal/sharing/apiService.js.map +1 -1
  142. package/dist/internal/sharing/cryptoService.js +1 -0
  143. package/dist/internal/sharing/cryptoService.js.map +1 -1
  144. package/dist/internal/sharing/events.d.ts +23 -55
  145. package/dist/internal/sharing/events.js +46 -138
  146. package/dist/internal/sharing/events.js.map +1 -1
  147. package/dist/internal/sharing/events.test.js +77 -180
  148. package/dist/internal/sharing/events.test.js.map +1 -1
  149. package/dist/internal/sharing/index.d.ts +4 -5
  150. package/dist/internal/sharing/index.js +5 -5
  151. package/dist/internal/sharing/index.js.map +1 -1
  152. package/dist/internal/sharing/interface.d.ts +3 -0
  153. package/dist/internal/sharing/sharingManagement.d.ts +2 -3
  154. package/dist/internal/sharing/sharingManagement.js +7 -9
  155. package/dist/internal/sharing/sharingManagement.js.map +1 -1
  156. package/dist/internal/sharing/sharingManagement.test.js +9 -39
  157. package/dist/internal/sharing/sharingManagement.test.js.map +1 -1
  158. package/dist/internal/upload/apiService.d.ts +2 -3
  159. package/dist/internal/upload/apiService.js +7 -4
  160. package/dist/internal/upload/apiService.js.map +1 -1
  161. package/dist/internal/upload/fileUploader.d.ts +49 -53
  162. package/dist/internal/upload/fileUploader.js +91 -395
  163. package/dist/internal/upload/fileUploader.js.map +1 -1
  164. package/dist/internal/upload/fileUploader.test.js +38 -292
  165. package/dist/internal/upload/fileUploader.test.js.map +1 -1
  166. package/dist/internal/upload/index.d.ts +5 -5
  167. package/dist/internal/upload/index.js +23 -44
  168. package/dist/internal/upload/index.js.map +1 -1
  169. package/dist/internal/upload/interface.d.ts +2 -0
  170. package/dist/internal/upload/manager.d.ts +6 -6
  171. package/dist/internal/upload/manager.js +32 -66
  172. package/dist/internal/upload/manager.js.map +1 -1
  173. package/dist/internal/upload/manager.test.js +100 -117
  174. package/dist/internal/upload/manager.test.js.map +1 -1
  175. package/dist/internal/upload/streamUploader.d.ts +62 -0
  176. package/dist/internal/upload/streamUploader.js +440 -0
  177. package/dist/internal/upload/streamUploader.js.map +1 -0
  178. package/dist/internal/upload/streamUploader.test.d.ts +1 -0
  179. package/dist/internal/upload/streamUploader.test.js +358 -0
  180. package/dist/internal/upload/streamUploader.test.js.map +1 -0
  181. package/dist/protonDriveClient.d.ts +22 -165
  182. package/dist/protonDriveClient.js +27 -191
  183. package/dist/protonDriveClient.js.map +1 -1
  184. package/dist/protonDrivePhotosClient.js +3 -2
  185. package/dist/protonDrivePhotosClient.js.map +1 -1
  186. package/package.json +4 -4
  187. package/src/cache/index.ts +1 -0
  188. package/src/cache/memoryCache.ts +1 -1
  189. package/src/cache/nullCache.ts +38 -0
  190. package/src/config.ts +17 -2
  191. package/src/crypto/openPGPCrypto.ts +2 -0
  192. package/src/diagnostic/eventsGenerator.ts +48 -0
  193. package/src/diagnostic/httpClient.ts +80 -0
  194. package/src/diagnostic/index.ts +38 -0
  195. package/src/diagnostic/integrityVerificationStream.ts +56 -0
  196. package/src/diagnostic/interface.ts +158 -0
  197. package/src/diagnostic/sdkDiagnostic.ts +238 -0
  198. package/src/diagnostic/sdkDiagnosticFull.ts +40 -0
  199. package/src/diagnostic/telemetry.ts +71 -0
  200. package/src/diagnostic/zipGenerators.test.ts +177 -0
  201. package/src/diagnostic/zipGenerators.ts +70 -0
  202. package/src/errors.ts +13 -4
  203. package/src/interface/config.ts +28 -0
  204. package/src/interface/download.ts +2 -2
  205. package/src/interface/events.ts +66 -21
  206. package/src/interface/httpClient.ts +0 -16
  207. package/src/interface/index.ts +9 -5
  208. package/src/interface/nodes.ts +32 -12
  209. package/src/interface/sharing.ts +1 -0
  210. package/src/interface/upload.ts +59 -3
  211. package/src/internal/apiService/driveTypes.ts +1341 -465
  212. package/src/internal/apiService/errors.ts +3 -2
  213. package/src/internal/apiService/transformers.ts +2 -0
  214. package/src/internal/asyncIteratorMap.test.ts +150 -0
  215. package/src/internal/asyncIteratorMap.ts +64 -0
  216. package/src/internal/download/apiService.ts +11 -8
  217. package/src/internal/download/fileDownloader.ts +2 -2
  218. package/src/internal/events/apiService.ts +25 -28
  219. package/src/internal/events/coreEventManager.test.ts +101 -0
  220. package/src/internal/events/coreEventManager.ts +20 -45
  221. package/src/internal/events/eventManager.test.ts +201 -88
  222. package/src/internal/events/eventManager.ts +69 -115
  223. package/src/internal/events/index.ts +54 -84
  224. package/src/internal/events/interface.ts +70 -15
  225. package/src/internal/events/volumeEventManager.test.ts +243 -0
  226. package/src/internal/events/volumeEventManager.ts +55 -53
  227. package/src/internal/nodes/apiService.test.ts +36 -7
  228. package/src/internal/nodes/apiService.ts +19 -7
  229. package/src/internal/nodes/cache.test.ts +1 -0
  230. package/src/internal/nodes/cache.ts +21 -2
  231. package/src/internal/nodes/cryptoService.test.ts +38 -0
  232. package/src/internal/nodes/cryptoService.ts +1 -1
  233. package/src/internal/nodes/events.test.ts +29 -335
  234. package/src/internal/nodes/events.ts +45 -253
  235. package/src/internal/nodes/index.ts +6 -8
  236. package/src/internal/nodes/interface.ts +6 -3
  237. package/src/internal/nodes/nodesAccess.test.ts +133 -91
  238. package/src/internal/nodes/nodesAccess.ts +70 -8
  239. package/src/internal/nodes/nodesManagement.test.ts +39 -15
  240. package/src/internal/nodes/nodesManagement.ts +12 -30
  241. package/src/internal/shares/cache.ts +4 -2
  242. package/src/internal/shares/manager.ts +9 -5
  243. package/src/internal/sharing/apiService.ts +25 -2
  244. package/src/internal/sharing/cache.ts +1 -1
  245. package/src/internal/sharing/cryptoService.ts +1 -0
  246. package/src/internal/sharing/events.test.ts +89 -195
  247. package/src/internal/sharing/events.ts +42 -156
  248. package/src/internal/sharing/index.ts +6 -9
  249. package/src/internal/sharing/interface.ts +6 -2
  250. package/src/internal/sharing/sharingManagement.test.ts +10 -40
  251. package/src/internal/sharing/sharingManagement.ts +7 -11
  252. package/src/internal/upload/apiService.ts +5 -6
  253. package/src/internal/upload/fileUploader.test.ts +46 -376
  254. package/src/internal/upload/fileUploader.ts +114 -494
  255. package/src/internal/upload/index.ts +30 -54
  256. package/src/internal/upload/interface.ts +2 -0
  257. package/src/internal/upload/manager.test.ts +107 -124
  258. package/src/internal/upload/manager.ts +48 -80
  259. package/src/internal/upload/streamUploader.test.ts +468 -0
  260. package/src/internal/upload/streamUploader.ts +550 -0
  261. package/src/protonDriveClient.ts +80 -248
  262. package/src/protonDrivePhotosClient.ts +4 -3
  263. package/dist/internal/events/cache.d.ts +0 -28
  264. package/dist/internal/events/cache.js +0 -67
  265. package/dist/internal/events/cache.js.map +0 -1
  266. package/dist/internal/events/cache.test.js +0 -43
  267. package/dist/internal/events/cache.test.js.map +0 -1
  268. package/dist/internal/nodes/index.test.js +0 -112
  269. package/dist/internal/nodes/index.test.js.map +0 -1
  270. package/src/internal/events/cache.test.ts +0 -47
  271. package/src/internal/events/cache.ts +0 -80
  272. package/src/internal/nodes/index.test.ts +0 -135
  273. /package/dist/{internal/events/cache.test.d.ts → diagnostic/zipGenerators.test.d.ts} +0 -0
  274. /package/dist/internal/{nodes/index.test.d.ts → asyncIteratorMap.test.d.ts} +0 -0
@@ -1,271 +1,63 @@
1
- import { Logger, NodeEventCallback } from "../../interface";
2
- import { convertInternalNode } from "../../transformers";
3
- import { DriveEventsService, DriveEvent, DriveEventType } from "../events";
4
- import { DecryptedNode } from "./interface";
1
+ import { Logger } from "../../interface";
2
+ import { DriveEvent, DriveEventType } from "../events";
5
3
  import { NodesCache } from "./cache";
6
- import { NodesAccess } from "./nodesAccess";
7
-
8
- type Listeners = {
9
- /**
10
- * Condition for the listener to be notified about the event.
11
- *
12
- * The condition is a function that receives the event information
13
- * and returns true if the listener should be notified about the
14
- * event.
15
- */
16
- condition: (nodeEventInfo: NodeEventInfo) => boolean,
17
- callback: NodeEventCallback,
18
- }[];
19
-
20
- /**
21
- * Minimal information about the event that is used for listener
22
- * condition. The information is used to determine if the listener
23
- * should be notified about the event.
24
- *
25
- * This must come from the API response to volume events.
26
- */
27
- type NodeEventInfo = {
28
- parentNodeUid?: string,
29
- isTrashed?: boolean,
30
- }
31
4
 
32
5
  /**
33
- * Provides both event handling and subscription mechanism for user.
34
- *
6
+ * Provides internal event handling.
7
+ *
35
8
  * The service is responsible for handling events regarding node metadata
36
- * from the DriveEventsService, and for providing a subscription mechanism
37
- * for the user to listen to updates of specific group of nodes, such as
38
- * any update for trashed nodes.
9
+ * from the DriveEventsService.
39
10
  */
40
- export class NodesEvents {
41
- private listeners: Listeners = [];
42
-
43
- constructor(private logger: Logger, events: DriveEventsService, private cache: NodesCache, private nodesAccess: NodesAccess) {
44
- this.logger = logger;
45
- this.cache = cache;
46
- this.nodesAccess = nodesAccess;
11
+ export class NodesEventsHandler {
12
+ constructor(private logger: Logger, private cache: NodesCache) {
13
+ }
47
14
 
48
- events.addListener(async (events, fullRefreshVolumeId) => {
49
- if (fullRefreshVolumeId) {
50
- await cache.setNodesStaleFromVolume(fullRefreshVolumeId);
15
+ async updateNodesCacheOnEvent(event: DriveEvent): Promise<void> {
16
+ try {
17
+ if (event.type === DriveEventType.TreeRefresh) {
18
+ await this.cache.setNodesStaleFromVolume(event.treeEventScopeId);
51
19
  return;
52
20
  }
53
-
54
- for (const event of events) {
55
- try {
56
- await updateCacheByEvent(logger, event, cache);
57
- } catch (error: unknown) {
58
- logger.error(`Failed to update cache`, error);
59
- }
60
- try {
61
- await notifyListenersByEvent(logger, event, this.listeners, cache, nodesAccess);
62
- } catch (error: unknown) {
63
- logger.error(`Failed to notifiy listeners`, error);
64
- }
65
- // Delete must come last as it will remove the node from the cache
66
- // and we need to first know local status of the node to properly
67
- // notify the listeners.
68
- await deleteFromCacheByEvent(logger, event, cache);
21
+ if (event.type === DriveEventType.TreeRemove) {
22
+ await this.cache.removeVolume(event.treeEventScopeId);
23
+ return;
69
24
  }
70
- });
71
- }
72
-
73
- subscribeToTrashedNodes(callback: NodeEventCallback) {
74
- this.listeners.push({ condition: ({ isTrashed }) => isTrashed || false, callback });
75
- return () => {
76
- this.listeners = this.listeners.filter(listener => listener.callback !== callback);
77
- }
78
- }
79
-
80
- subscribeToChildren(parentNodeUid: string, callback: NodeEventCallback) {
81
- this.listeners.push({ condition: ({ parentNodeUid: parent }) => parent === parentNodeUid, callback });
82
- return () => {
83
- this.listeners = this.listeners.filter(listener => listener.callback !== callback);
84
- }
85
- }
86
-
87
- async nodeCreated(node: DecryptedNode): Promise<void> {
88
- await this.cache.setNode(node);
89
- void this.notifyListenersByNode(node, DriveEventType.NodeCreated);
90
- }
91
-
92
- async nodeUpdated(partialNode: { uid: string } & Partial<DecryptedNode>): Promise<void> {
93
- const originalNode = await this.cache.getNode(partialNode.uid);
94
- const updatedNode = {
95
- ...originalNode,
96
- ...partialNode,
97
- }
98
-
99
- await this.cache.setNode(updatedNode);
100
- void this.notifyListenersByNode(updatedNode, DriveEventType.NodeUpdated);
101
- }
102
-
103
- async nodesDeleted(nodeUids: string[]): Promise<void> {
104
- try {
105
- for await (const originalNode of this.cache.iterateNodes(nodeUids)) {
106
- if (originalNode.ok) {
107
- void this.notifyListenersByNode(originalNode.node, DriveEventType.NodeDeleted);
25
+ if (event.type === DriveEventType.NodeDeleted) {
26
+ await this.cache.removeNodes([event.nodeUid]);
27
+ return;
28
+ }
29
+ if (event.type === DriveEventType.NodeCreated) {
30
+ // FIXME Add it to the parent listing even if it's not cached
31
+ // so it doesn't need to refetch all children
32
+
33
+ // We do not have partial nodes in the cache, so we don't
34
+ // add it. If new node is not added, we need to reset the
35
+ // children loaded flag to force refetch when requested.
36
+ if (event.parentNodeUid) {
37
+ await this.cache.resetFolderChildrenLoaded(event.parentNodeUid);
108
38
  }
39
+ return;
109
40
  }
110
- } catch {}
111
-
112
- await this.cache.removeNodes(nodeUids);
113
- }
114
-
115
- private async notifyListenersByNode(node: DecryptedNode, eventType: DriveEventType.NodeCreated | DriveEventType.NodeUpdated | DriveEventType.NodeDeleted) {
116
- const event: DriveEvent = {
117
- type: eventType,
118
- nodeUid: node.uid,
119
- parentNodeUid: node.parentUid,
120
- isOwnVolume: true,
121
- isTrashed: !!node.trashTime,
122
- isShared: node.isShared,
123
- };
124
- await notifyListenersByEvent(this.logger, event, this.listeners, this.cache, this.nodesAccess);
125
-
126
- }
127
- }
128
-
129
- /**
130
- * For given event, update the cache accordingly.
131
- *
132
- * The function is responsible for updating the cache based on the
133
- * event received from the DriveEventsService. The cache metadata
134
- * are not updated, only the nodes are marked as stale to be
135
- * fetched and decrypted again when requested by the client.
136
- *
137
- * If the node is not found in the cache, the event is silently
138
- * skipped as the node will be fetched and decrypted when requested
139
- * by the client.
140
- *
141
- * If the node cannot be updated in the cache, the node is removed
142
- * from the cache to not block the client. If the node is not possible
143
- * to remove, the function throws an error.
144
- *
145
- * @throws Only if the node is not possible to remove from the cache.
146
- */
147
- export async function updateCacheByEvent(logger: Logger, event: DriveEvent, cache: NodesCache) {
148
- // NodeCreated event is ignored as we do not want to fetch and
149
- // decrypt the node immediately. The node will be fetched and
150
- // decrypted when requested by the client.
151
- if (event.type === DriveEventType.NodeCreated) {
152
- // We do not have partial nodes in the cache, so we don't
153
- // add it. If new node is not added, we need to reset the
154
- // children loaded flag to force refetch when requested.
155
- if (event.parentNodeUid) {
156
- await cache.resetFolderChildrenLoaded(event.parentNodeUid);
157
- }
158
- }
159
- if (event.type === DriveEventType.NodeUpdated || event.type === DriveEventType.NodeUpdatedMetadata) {
160
- let node;
161
- // getNode can fail if the node is not found or if it is
162
- // corrupted. In later case, it will be automatically
163
- // removed from cache. In both cases, lets skip the event
164
- // silently as once requested by client, the node will
165
- // be cached again.
166
- try {
167
- node = await cache.getNode(event.nodeUid);
168
- } catch (error: unknown) {
169
- logger.debug(`Skipping node update event (node not in the cache): ${error}`);
170
- }
171
- if (node) {
172
- node.isStale = true;
173
- // We need to update the parentUid as the node might have
174
- // been moved to another parent. This is important for
175
- // children iteration.
176
- node.parentUid = event.parentNodeUid;
177
- try {
178
- await cache.setNode(node);
179
- } catch (setNodeError: unknown) {
180
- logger.error(`Skipping node update event (failed to update)`, setNodeError);
181
- // If updating node in the cache is failing, lets remove it
182
- // to not block the whole client. If the node is not possible
183
- // to remove, lets throw at this point as cache is in very
184
- // bad state by this point and the rest of the code would start
185
- // to break randomly.
186
- try {
187
- await cache.removeNodes([event.nodeUid]);
188
- } catch (removeNodeError: unknown) {
189
- logger.error(`Skipping node update event (failed to remove after failed update)`, removeNodeError);
190
- // removeNodeError is automatic correction algorithm.
191
- // If that fails, lets throw the original error as that
192
- // is the real problem.
193
- throw setNodeError;
41
+ if (event.type === DriveEventType.NodeUpdated) {
42
+ const node = await this.cache.getNode(event.nodeUid);
43
+ node.isStale = true;
44
+ node.parentUid = event.parentNodeUid;
45
+ node.isShared = event.isShared;
46
+ if (event.isTrashed) {
47
+ node.trashTime ??= new Date();
48
+ } else {
49
+ node.trashTime = undefined;
194
50
  }
51
+ await this.cache.setNode(node);
195
52
  }
196
- }
197
- }
198
- }
199
-
200
- /**
201
- * For given event, delete the node from the cache if it is
202
- * deleted.
203
- */
204
- export async function deleteFromCacheByEvent(logger: Logger, event: DriveEvent, cache: NodesCache) {
205
- if (event.type === DriveEventType.NodeDeleted) {
206
- // removeNodes can fail removing children.
207
- // We do not want to stop processing other events in such
208
- // a case. Lets log the error and continue.
209
- try {
210
- await cache.removeNodes([event.nodeUid]);
211
53
  } catch (error: unknown) {
212
- logger.error(`Skipping node delete event:`, error);
213
- }
214
- }
215
- }
216
-
217
- /**
218
- * For given event, notify the listeners accordingly.
219
- *
220
- * The function is responsible for notifying the listeners about the
221
- * event received from the DriveEventsService. The listeners are
222
- * connected with events based on the condition, such as parent node
223
- * uid for listening to children updates.
224
- *
225
- * The function is responsible for fetching and decrypting the latest
226
- * version of the node metadata. If the node is not found, the event
227
- * is silently skipped as the node will be fetched and decrypted when
228
- * requested by the client.
229
- *
230
- * @throws Only if the client's callback throws.
231
- */
232
- export async function notifyListenersByEvent(logger: Logger, event: DriveEvent, listeners: Listeners, cache: NodesCache, nodesAccess: NodesAccess) {
233
- if (event.type === DriveEventType.ShareWithMeUpdated) {
234
- return;
235
- }
236
-
237
- const subscribedListeners = listeners.filter(({ condition }) => condition(event));
238
- const eventMatchingCondition = subscribedListeners.length > 0;
239
-
240
- if ([DriveEventType.NodeCreated, DriveEventType.NodeUpdated, DriveEventType.NodeUpdatedMetadata].includes(event.type) && eventMatchingCondition) {
241
- if (subscribedListeners.length) {
242
- let node;
243
- try {
244
- node = await nodesAccess.getNode(event.nodeUid);
245
- } catch (error: unknown) {
246
- logger.error(`Skipping node update event to listener`, error);
247
- return;
54
+ if (error instanceof Error) {
55
+ // FIXME throw CacheMissException error and catch it
56
+ if (error.message === 'Entity not found') {
57
+ return;
58
+ }
248
59
  }
249
- subscribedListeners.forEach(({ callback }) => callback({ type: 'update', uid: node.uid, node: convertInternalNode(node) }));
250
- }
251
- }
252
-
253
- if (
254
- ((event.type === DriveEventType.NodeUpdated || event.type === DriveEventType.NodeUpdatedMetadata) && !eventMatchingCondition)
255
- || event.type === DriveEventType.NodeDeleted
256
- ) {
257
- let node: DecryptedNode;
258
- try {
259
- node = await cache.getNode(event.nodeUid);
260
- } catch {}
261
-
262
- const subscribedListeners = listeners.filter(({ condition }) => condition({
263
- parentNodeUid: node?.parentUid,
264
- isTrashed: !!node?.trashTime || false,
265
- }));
266
-
267
- if (subscribedListeners.length) {
268
- subscribedListeners.forEach(({ callback }) => callback({ type: 'remove', uid: event.nodeUid }));
60
+ this.logger.error(`Failed to update node cache for event: ${event.eventId}`, error);
269
61
  }
270
62
  }
271
63
  }
@@ -1,26 +1,25 @@
1
1
  import { DriveAPIService } from "../apiService";
2
2
  import { DriveCrypto } from "../../crypto";
3
- import { DriveEventsService } from "../events";
4
3
  import { ProtonDriveEntitiesCache, ProtonDriveCryptoCache, ProtonDriveAccount, ProtonDriveTelemetry } from "../../interface";
5
4
  import { NodeAPIService } from "./apiService";
6
5
  import { NodesCache } from "./cache";
7
- import { NodesEvents } from "./events";
8
6
  import { NodesCryptoCache } from "./cryptoCache";
9
7
  import { NodesCryptoService } from "./cryptoService";
10
8
  import { SharesService } from "./interface";
11
9
  import { NodesAccess } from "./nodesAccess";
12
10
  import { NodesManagement } from "./nodesManagement";
13
11
  import { NodesRevisons } from "./nodesRevisions";
12
+ import { NodesEventsHandler } from "./events";
14
13
 
15
14
  export type { DecryptedNode, DecryptedRevision } from "./interface";
16
15
  export { generateFileExtendedAttributes } from "./extendedAttributes";
17
16
 
18
17
  /**
19
18
  * Provides facade for the whole nodes module.
20
- *
19
+ *
21
20
  * The nodes module is responsible for handling node metadata, including
22
21
  * API communication, encryption, decryption, caching, and event handling.
23
- *
22
+ *
24
23
  * This facade provides internal interface that other modules can use to
25
24
  * interact with the nodes.
26
25
  */
@@ -31,7 +30,6 @@ export function initNodesModule(
31
30
  driveCryptoCache: ProtonDriveCryptoCache,
32
31
  account: ProtonDriveAccount,
33
32
  driveCrypto: DriveCrypto,
34
- driveEvents: DriveEventsService,
35
33
  sharesService: SharesService,
36
34
  ) {
37
35
  const api = new NodeAPIService(telemetry.getLogger('nodes-api'), apiService);
@@ -39,14 +37,14 @@ export function initNodesModule(
39
37
  const cryptoCache = new NodesCryptoCache(telemetry.getLogger('nodes-cache'), driveCryptoCache);
40
38
  const cryptoService = new NodesCryptoService(telemetry, driveCrypto, account, sharesService);
41
39
  const nodesAccess = new NodesAccess(telemetry.getLogger('nodes'), api, cache, cryptoCache, cryptoService, sharesService);
42
- const nodesEvents = new NodesEvents(telemetry.getLogger('nodes-events'), driveEvents, cache, nodesAccess);
43
- const nodesManagement = new NodesManagement(api, cryptoCache, cryptoService, nodesAccess, nodesEvents);
40
+ const nodesEventHandler = new NodesEventsHandler(telemetry.getLogger('nodes-events'), cache);
41
+ const nodesManagement = new NodesManagement(api, cryptoCache, cryptoService, nodesAccess);
44
42
  const nodesRevisions = new NodesRevisons(telemetry.getLogger('nodes'), api, cryptoService, nodesAccess);
45
43
 
46
44
  return {
47
45
  access: nodesAccess,
48
46
  management: nodesManagement,
49
47
  revisions: nodesRevisions,
50
- events: nodesEvents,
48
+ eventHandler: nodesEventHandler,
51
49
  };
52
50
  }
@@ -26,11 +26,11 @@ interface BaseNode {
26
26
 
27
27
  /**
28
28
  * Interface used only internaly in the nodes module.
29
- *
29
+ *
30
30
  * Outside of the module, the decrypted node interface should be used.
31
31
  */
32
32
  export interface EncryptedNode extends BaseNode {
33
- encryptedCrypto: EncryptedNodeFolderCrypto | EncryptedNodeFileCrypto;
33
+ encryptedCrypto: EncryptedNodeFolderCrypto | EncryptedNodeFileCrypto | EncryptedNodeAlbumCrypto;
34
34
  }
35
35
 
36
36
  export interface EncryptedNodeCrypto {
@@ -56,6 +56,9 @@ export interface EncryptedNodeFolderCrypto extends EncryptedNodeCrypto {
56
56
  };
57
57
  }
58
58
 
59
+ // eslint-disable-next-line @typescript-eslint/no-empty-object-type
60
+ export interface EncryptedNodeAlbumCrypto extends EncryptedNodeCrypto {}
61
+
59
62
  /**
60
63
  * Interface used only internally in the nodes module.
61
64
  *
@@ -93,7 +96,7 @@ export interface DecryptedNode extends Omit<DecryptedUnparsedNode, 'name' | 'act
93
96
  * Interface holding decrypted node key, including session key, and hash key.
94
97
  *
95
98
  * These keys are cached as they are needed for various actions on the node.
96
- *
99
+ *
97
100
  * Passphrase, for example, might be removed at some point. It is needed as
98
101
  * at this moment the move requires both node key passphrase and the session
99
102
  * key.