@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.
- package/dist/cache/index.d.ts +1 -0
- package/dist/cache/index.js +3 -1
- package/dist/cache/index.js.map +1 -1
- package/dist/cache/memoryCache.d.ts +1 -1
- package/dist/cache/nullCache.d.ts +14 -0
- package/dist/cache/nullCache.js +37 -0
- package/dist/cache/nullCache.js.map +1 -0
- package/dist/config.d.ts +16 -1
- package/dist/config.js +1 -1
- package/dist/config.js.map +1 -1
- package/dist/crypto/openPGPCrypto.js +2 -0
- package/dist/crypto/openPGPCrypto.js.map +1 -1
- package/dist/diagnostic/eventsGenerator.d.ts +14 -0
- package/dist/diagnostic/eventsGenerator.js +49 -0
- package/dist/diagnostic/eventsGenerator.js.map +1 -0
- package/dist/diagnostic/httpClient.d.ts +16 -0
- package/dist/diagnostic/httpClient.js +81 -0
- package/dist/diagnostic/httpClient.js.map +1 -0
- package/dist/diagnostic/index.d.ts +10 -0
- package/dist/diagnostic/index.js +35 -0
- package/dist/diagnostic/index.js.map +1 -0
- package/dist/diagnostic/integrityVerificationStream.d.ts +21 -0
- package/dist/diagnostic/integrityVerificationStream.js +56 -0
- package/dist/diagnostic/integrityVerificationStream.js.map +1 -0
- package/dist/diagnostic/interface.d.ts +102 -0
- package/dist/diagnostic/interface.js +3 -0
- package/dist/diagnostic/interface.js.map +1 -0
- package/dist/diagnostic/sdkDiagnostic.d.ts +22 -0
- package/dist/diagnostic/sdkDiagnostic.js +216 -0
- package/dist/diagnostic/sdkDiagnostic.js.map +1 -0
- package/dist/diagnostic/sdkDiagnosticFull.d.ts +18 -0
- package/dist/diagnostic/sdkDiagnosticFull.js +35 -0
- package/dist/diagnostic/sdkDiagnosticFull.js.map +1 -0
- package/dist/diagnostic/telemetry.d.ts +25 -0
- package/dist/diagnostic/telemetry.js +70 -0
- package/dist/diagnostic/telemetry.js.map +1 -0
- package/dist/diagnostic/zipGenerators.d.ts +9 -0
- package/dist/diagnostic/zipGenerators.js +64 -0
- package/dist/diagnostic/zipGenerators.js.map +1 -0
- package/dist/diagnostic/zipGenerators.test.js +144 -0
- package/dist/diagnostic/zipGenerators.test.js.map +1 -0
- package/dist/errors.d.ts +8 -3
- package/dist/errors.js +11 -4
- package/dist/errors.js.map +1 -1
- package/dist/interface/config.d.ts +26 -0
- package/dist/interface/config.js +3 -0
- package/dist/interface/config.js.map +1 -0
- package/dist/interface/download.d.ts +2 -2
- package/dist/interface/events.d.ts +60 -20
- package/dist/interface/events.js +11 -1
- package/dist/interface/events.js.map +1 -1
- package/dist/interface/httpClient.d.ts +0 -14
- package/dist/interface/index.d.ts +9 -5
- package/dist/interface/index.js +2 -1
- package/dist/interface/index.js.map +1 -1
- package/dist/interface/nodes.d.ts +21 -1
- package/dist/interface/nodes.js +11 -0
- package/dist/interface/nodes.js.map +1 -1
- package/dist/interface/sharing.d.ts +1 -0
- package/dist/interface/upload.d.ts +57 -3
- package/dist/internal/apiService/driveTypes.d.ts +1341 -465
- package/dist/internal/apiService/errors.js +2 -2
- package/dist/internal/apiService/errors.js.map +1 -1
- package/dist/internal/apiService/transformers.js +2 -0
- package/dist/internal/apiService/transformers.js.map +1 -1
- package/dist/internal/asyncIteratorMap.d.ts +15 -0
- package/dist/internal/asyncIteratorMap.js +59 -0
- package/dist/internal/asyncIteratorMap.js.map +1 -0
- package/dist/internal/asyncIteratorMap.test.js +120 -0
- package/dist/internal/asyncIteratorMap.test.js.map +1 -0
- package/dist/internal/download/apiService.js +32 -31
- package/dist/internal/download/apiService.js.map +1 -1
- package/dist/internal/download/fileDownloader.d.ts +2 -2
- package/dist/internal/download/fileDownloader.js.map +1 -1
- package/dist/internal/events/apiService.d.ts +4 -6
- package/dist/internal/events/apiService.js +15 -22
- package/dist/internal/events/apiService.js.map +1 -1
- package/dist/internal/events/coreEventManager.d.ts +7 -10
- package/dist/internal/events/coreEventManager.js +19 -36
- package/dist/internal/events/coreEventManager.js.map +1 -1
- package/dist/internal/events/coreEventManager.test.d.ts +1 -0
- package/dist/internal/events/coreEventManager.test.js +87 -0
- package/dist/internal/events/coreEventManager.test.js.map +1 -0
- package/dist/internal/events/eventManager.d.ts +11 -36
- package/dist/internal/events/eventManager.js +59 -105
- package/dist/internal/events/eventManager.js.map +1 -1
- package/dist/internal/events/eventManager.test.js +167 -82
- package/dist/internal/events/eventManager.test.js.map +1 -1
- package/dist/internal/events/index.d.ts +13 -33
- package/dist/internal/events/index.js +56 -72
- package/dist/internal/events/index.js.map +1 -1
- package/dist/internal/events/interface.d.ts +59 -14
- package/dist/internal/events/interface.js +13 -3
- package/dist/internal/events/interface.js.map +1 -1
- package/dist/internal/events/volumeEventManager.d.ts +7 -17
- package/dist/internal/events/volumeEventManager.js +58 -45
- package/dist/internal/events/volumeEventManager.js.map +1 -1
- package/dist/internal/events/volumeEventManager.test.d.ts +1 -0
- package/dist/internal/events/volumeEventManager.test.js +203 -0
- package/dist/internal/events/volumeEventManager.test.js.map +1 -0
- package/dist/internal/nodes/apiService.d.ts +2 -2
- package/dist/internal/nodes/apiService.js +16 -6
- package/dist/internal/nodes/apiService.js.map +1 -1
- package/dist/internal/nodes/apiService.test.js +30 -8
- package/dist/internal/nodes/apiService.test.js.map +1 -1
- package/dist/internal/nodes/cache.d.ts +10 -1
- package/dist/internal/nodes/cache.js +18 -0
- package/dist/internal/nodes/cache.js.map +1 -1
- package/dist/internal/nodes/cache.test.js +1 -0
- package/dist/internal/nodes/cache.test.js.map +1 -1
- package/dist/internal/nodes/cryptoService.d.ts +1 -1
- package/dist/internal/nodes/cryptoService.js.map +1 -1
- package/dist/internal/nodes/cryptoService.test.js +34 -0
- package/dist/internal/nodes/cryptoService.test.js.map +1 -1
- package/dist/internal/nodes/events.d.ts +7 -83
- package/dist/internal/nodes/events.js +43 -217
- package/dist/internal/nodes/events.js.map +1 -1
- package/dist/internal/nodes/events.test.js +27 -277
- package/dist/internal/nodes/events.test.js.map +1 -1
- package/dist/internal/nodes/index.d.ts +3 -4
- package/dist/internal/nodes/index.js +5 -5
- package/dist/internal/nodes/index.js.map +1 -1
- package/dist/internal/nodes/interface.d.ts +3 -1
- package/dist/internal/nodes/nodesAccess.d.ts +15 -0
- package/dist/internal/nodes/nodesAccess.js +65 -7
- package/dist/internal/nodes/nodesAccess.js.map +1 -1
- package/dist/internal/nodes/nodesAccess.test.js +132 -93
- package/dist/internal/nodes/nodesAccess.test.js.map +1 -1
- package/dist/internal/nodes/nodesManagement.d.ts +1 -3
- package/dist/internal/nodes/nodesManagement.js +12 -26
- package/dist/internal/nodes/nodesManagement.js.map +1 -1
- package/dist/internal/nodes/nodesManagement.test.js +35 -14
- package/dist/internal/nodes/nodesManagement.test.js.map +1 -1
- package/dist/internal/shares/cache.d.ts +2 -0
- package/dist/internal/shares/cache.js +2 -0
- package/dist/internal/shares/cache.js.map +1 -1
- package/dist/internal/shares/manager.d.ts +1 -0
- package/dist/internal/shares/manager.js +3 -0
- package/dist/internal/shares/manager.js.map +1 -1
- package/dist/internal/sharing/apiService.js +20 -2
- package/dist/internal/sharing/apiService.js.map +1 -1
- package/dist/internal/sharing/cryptoService.js +1 -0
- package/dist/internal/sharing/cryptoService.js.map +1 -1
- package/dist/internal/sharing/events.d.ts +23 -55
- package/dist/internal/sharing/events.js +46 -138
- package/dist/internal/sharing/events.js.map +1 -1
- package/dist/internal/sharing/events.test.js +77 -180
- package/dist/internal/sharing/events.test.js.map +1 -1
- package/dist/internal/sharing/index.d.ts +4 -5
- package/dist/internal/sharing/index.js +5 -5
- package/dist/internal/sharing/index.js.map +1 -1
- package/dist/internal/sharing/interface.d.ts +3 -0
- package/dist/internal/sharing/sharingManagement.d.ts +2 -3
- package/dist/internal/sharing/sharingManagement.js +7 -9
- package/dist/internal/sharing/sharingManagement.js.map +1 -1
- package/dist/internal/sharing/sharingManagement.test.js +9 -39
- package/dist/internal/sharing/sharingManagement.test.js.map +1 -1
- package/dist/internal/upload/apiService.d.ts +2 -3
- package/dist/internal/upload/apiService.js +7 -4
- package/dist/internal/upload/apiService.js.map +1 -1
- package/dist/internal/upload/fileUploader.d.ts +49 -53
- package/dist/internal/upload/fileUploader.js +91 -395
- package/dist/internal/upload/fileUploader.js.map +1 -1
- package/dist/internal/upload/fileUploader.test.js +38 -292
- package/dist/internal/upload/fileUploader.test.js.map +1 -1
- package/dist/internal/upload/index.d.ts +5 -5
- package/dist/internal/upload/index.js +23 -44
- package/dist/internal/upload/index.js.map +1 -1
- package/dist/internal/upload/interface.d.ts +2 -0
- package/dist/internal/upload/manager.d.ts +6 -6
- package/dist/internal/upload/manager.js +32 -66
- package/dist/internal/upload/manager.js.map +1 -1
- package/dist/internal/upload/manager.test.js +100 -117
- package/dist/internal/upload/manager.test.js.map +1 -1
- package/dist/internal/upload/streamUploader.d.ts +62 -0
- package/dist/internal/upload/streamUploader.js +440 -0
- package/dist/internal/upload/streamUploader.js.map +1 -0
- package/dist/internal/upload/streamUploader.test.d.ts +1 -0
- package/dist/internal/upload/streamUploader.test.js +358 -0
- package/dist/internal/upload/streamUploader.test.js.map +1 -0
- package/dist/protonDriveClient.d.ts +22 -165
- package/dist/protonDriveClient.js +27 -191
- package/dist/protonDriveClient.js.map +1 -1
- package/dist/protonDrivePhotosClient.js +3 -2
- package/dist/protonDrivePhotosClient.js.map +1 -1
- package/package.json +4 -4
- package/src/cache/index.ts +1 -0
- package/src/cache/memoryCache.ts +1 -1
- package/src/cache/nullCache.ts +38 -0
- package/src/config.ts +17 -2
- package/src/crypto/openPGPCrypto.ts +2 -0
- package/src/diagnostic/eventsGenerator.ts +48 -0
- package/src/diagnostic/httpClient.ts +80 -0
- package/src/diagnostic/index.ts +38 -0
- package/src/diagnostic/integrityVerificationStream.ts +56 -0
- package/src/diagnostic/interface.ts +158 -0
- package/src/diagnostic/sdkDiagnostic.ts +238 -0
- package/src/diagnostic/sdkDiagnosticFull.ts +40 -0
- package/src/diagnostic/telemetry.ts +71 -0
- package/src/diagnostic/zipGenerators.test.ts +177 -0
- package/src/diagnostic/zipGenerators.ts +70 -0
- package/src/errors.ts +13 -4
- package/src/interface/config.ts +28 -0
- package/src/interface/download.ts +2 -2
- package/src/interface/events.ts +66 -21
- package/src/interface/httpClient.ts +0 -16
- package/src/interface/index.ts +9 -5
- package/src/interface/nodes.ts +32 -12
- package/src/interface/sharing.ts +1 -0
- package/src/interface/upload.ts +59 -3
- package/src/internal/apiService/driveTypes.ts +1341 -465
- package/src/internal/apiService/errors.ts +3 -2
- package/src/internal/apiService/transformers.ts +2 -0
- package/src/internal/asyncIteratorMap.test.ts +150 -0
- package/src/internal/asyncIteratorMap.ts +64 -0
- package/src/internal/download/apiService.ts +11 -8
- package/src/internal/download/fileDownloader.ts +2 -2
- package/src/internal/events/apiService.ts +25 -28
- package/src/internal/events/coreEventManager.test.ts +101 -0
- package/src/internal/events/coreEventManager.ts +20 -45
- package/src/internal/events/eventManager.test.ts +201 -88
- package/src/internal/events/eventManager.ts +69 -115
- package/src/internal/events/index.ts +54 -84
- package/src/internal/events/interface.ts +70 -15
- package/src/internal/events/volumeEventManager.test.ts +243 -0
- package/src/internal/events/volumeEventManager.ts +55 -53
- package/src/internal/nodes/apiService.test.ts +36 -7
- package/src/internal/nodes/apiService.ts +19 -7
- package/src/internal/nodes/cache.test.ts +1 -0
- package/src/internal/nodes/cache.ts +21 -2
- package/src/internal/nodes/cryptoService.test.ts +38 -0
- package/src/internal/nodes/cryptoService.ts +1 -1
- package/src/internal/nodes/events.test.ts +29 -335
- package/src/internal/nodes/events.ts +45 -253
- package/src/internal/nodes/index.ts +6 -8
- package/src/internal/nodes/interface.ts +6 -3
- package/src/internal/nodes/nodesAccess.test.ts +133 -91
- package/src/internal/nodes/nodesAccess.ts +70 -8
- package/src/internal/nodes/nodesManagement.test.ts +39 -15
- package/src/internal/nodes/nodesManagement.ts +12 -30
- package/src/internal/shares/cache.ts +4 -2
- package/src/internal/shares/manager.ts +9 -5
- package/src/internal/sharing/apiService.ts +25 -2
- package/src/internal/sharing/cache.ts +1 -1
- package/src/internal/sharing/cryptoService.ts +1 -0
- package/src/internal/sharing/events.test.ts +89 -195
- package/src/internal/sharing/events.ts +42 -156
- package/src/internal/sharing/index.ts +6 -9
- package/src/internal/sharing/interface.ts +6 -2
- package/src/internal/sharing/sharingManagement.test.ts +10 -40
- package/src/internal/sharing/sharingManagement.ts +7 -11
- package/src/internal/upload/apiService.ts +5 -6
- package/src/internal/upload/fileUploader.test.ts +46 -376
- package/src/internal/upload/fileUploader.ts +114 -494
- package/src/internal/upload/index.ts +30 -54
- package/src/internal/upload/interface.ts +2 -0
- package/src/internal/upload/manager.test.ts +107 -124
- package/src/internal/upload/manager.ts +48 -80
- package/src/internal/upload/streamUploader.test.ts +468 -0
- package/src/internal/upload/streamUploader.ts +550 -0
- package/src/protonDriveClient.ts +80 -248
- package/src/protonDrivePhotosClient.ts +4 -3
- package/dist/internal/events/cache.d.ts +0 -28
- package/dist/internal/events/cache.js +0 -67
- package/dist/internal/events/cache.js.map +0 -1
- package/dist/internal/events/cache.test.js +0 -43
- package/dist/internal/events/cache.test.js.map +0 -1
- package/dist/internal/nodes/index.test.js +0 -112
- package/dist/internal/nodes/index.test.js.map +0 -1
- package/src/internal/events/cache.test.ts +0 -47
- package/src/internal/events/cache.ts +0 -80
- package/src/internal/nodes/index.test.ts +0 -135
- /package/dist/{internal/events/cache.test.d.ts → diagnostic/zipGenerators.test.d.ts} +0 -0
- /package/dist/internal/{nodes/index.test.d.ts → asyncIteratorMap.test.d.ts} +0 -0
|
@@ -1,271 +1,63 @@
|
|
|
1
|
-
import { Logger
|
|
2
|
-
import {
|
|
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
|
|
34
|
-
*
|
|
6
|
+
* Provides internal event handling.
|
|
7
|
+
*
|
|
35
8
|
* The service is responsible for handling events regarding node metadata
|
|
36
|
-
* from the DriveEventsService
|
|
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
|
|
41
|
-
private
|
|
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
|
-
|
|
49
|
-
|
|
50
|
-
|
|
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
|
-
|
|
55
|
-
|
|
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
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
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
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
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
|
-
|
|
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
|
-
|
|
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
|
|
43
|
-
const nodesManagement = new NodesManagement(api, cryptoCache, cryptoService, nodesAccess
|
|
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
|
-
|
|
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.
|