@protontech/drive-sdk 0.3.2 → 0.4.1

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 (171) hide show
  1. package/dist/interface/nodes.d.ts +4 -0
  2. package/dist/interface/nodes.js.map +1 -1
  3. package/dist/internal/apiService/errorCodes.d.ts +1 -0
  4. package/dist/internal/apiService/errors.d.ts +3 -0
  5. package/dist/internal/apiService/errors.js +7 -1
  6. package/dist/internal/apiService/errors.js.map +1 -1
  7. package/dist/internal/devices/interface.d.ts +1 -1
  8. package/dist/internal/devices/manager.js +1 -1
  9. package/dist/internal/devices/manager.js.map +1 -1
  10. package/dist/internal/devices/manager.test.js +3 -3
  11. package/dist/internal/devices/manager.test.js.map +1 -1
  12. package/dist/internal/events/apiService.js +1 -1
  13. package/dist/internal/events/apiService.js.map +1 -1
  14. package/dist/internal/events/coreEventManager.js +1 -1
  15. package/dist/internal/events/coreEventManager.js.map +1 -1
  16. package/dist/internal/events/coreEventManager.test.js +18 -24
  17. package/dist/internal/events/coreEventManager.test.js.map +1 -1
  18. package/dist/internal/events/index.d.ts +3 -4
  19. package/dist/internal/events/index.js +4 -4
  20. package/dist/internal/events/index.js.map +1 -1
  21. package/dist/internal/events/interface.d.ts +3 -0
  22. package/dist/internal/nodes/apiService.d.ts +12 -3
  23. package/dist/internal/nodes/apiService.js +54 -13
  24. package/dist/internal/nodes/apiService.js.map +1 -1
  25. package/dist/internal/nodes/apiService.test.js +35 -2
  26. package/dist/internal/nodes/apiService.test.js.map +1 -1
  27. package/dist/internal/nodes/cache.test.js +1 -0
  28. package/dist/internal/nodes/cache.test.js.map +1 -1
  29. package/dist/internal/nodes/cryptoService.d.ts +1 -1
  30. package/dist/internal/nodes/cryptoService.js +1 -1
  31. package/dist/internal/nodes/cryptoService.js.map +1 -1
  32. package/dist/internal/nodes/cryptoService.test.js +4 -4
  33. package/dist/internal/nodes/cryptoService.test.js.map +1 -1
  34. package/dist/internal/nodes/errors.d.ts +4 -0
  35. package/dist/internal/nodes/errors.js +9 -0
  36. package/dist/internal/nodes/errors.js.map +1 -0
  37. package/dist/internal/nodes/extendedAttributes.d.ts +2 -2
  38. package/dist/internal/nodes/extendedAttributes.js +15 -11
  39. package/dist/internal/nodes/extendedAttributes.js.map +1 -1
  40. package/dist/internal/nodes/extendedAttributes.test.js +19 -1
  41. package/dist/internal/nodes/extendedAttributes.test.js.map +1 -1
  42. package/dist/internal/nodes/index.test.js +2 -1
  43. package/dist/internal/nodes/index.test.js.map +1 -1
  44. package/dist/internal/nodes/interface.d.ts +5 -1
  45. package/dist/internal/nodes/nodesAccess.d.ts +3 -3
  46. package/dist/internal/nodes/nodesAccess.js +25 -15
  47. package/dist/internal/nodes/nodesAccess.js.map +1 -1
  48. package/dist/internal/nodes/nodesAccess.test.js +48 -8
  49. package/dist/internal/nodes/nodesAccess.test.js.map +1 -1
  50. package/dist/internal/nodes/nodesManagement.d.ts +2 -0
  51. package/dist/internal/nodes/nodesManagement.js +87 -9
  52. package/dist/internal/nodes/nodesManagement.js.map +1 -1
  53. package/dist/internal/nodes/nodesManagement.test.js +81 -5
  54. package/dist/internal/nodes/nodesManagement.test.js.map +1 -1
  55. package/dist/internal/photos/albums.d.ts +9 -7
  56. package/dist/internal/photos/albums.js +26 -13
  57. package/dist/internal/photos/albums.js.map +1 -1
  58. package/dist/internal/photos/apiService.d.ts +34 -3
  59. package/dist/internal/photos/apiService.js +96 -3
  60. package/dist/internal/photos/apiService.js.map +1 -1
  61. package/dist/internal/photos/index.d.ts +31 -4
  62. package/dist/internal/photos/index.js +57 -7
  63. package/dist/internal/photos/index.js.map +1 -1
  64. package/dist/internal/photos/interface.d.ts +25 -1
  65. package/dist/internal/photos/shares.d.ts +43 -0
  66. package/dist/internal/photos/shares.js +112 -0
  67. package/dist/internal/photos/shares.js.map +1 -0
  68. package/dist/internal/photos/timeline.d.ts +15 -0
  69. package/dist/internal/photos/timeline.js +22 -0
  70. package/dist/internal/photos/timeline.js.map +1 -0
  71. package/dist/internal/photos/upload.d.ts +59 -0
  72. package/dist/internal/photos/upload.js +104 -0
  73. package/dist/internal/photos/upload.js.map +1 -0
  74. package/dist/internal/shares/manager.d.ts +1 -1
  75. package/dist/internal/shares/manager.js +4 -4
  76. package/dist/internal/shares/manager.js.map +1 -1
  77. package/dist/internal/shares/manager.test.js +7 -7
  78. package/dist/internal/shares/manager.test.js.map +1 -1
  79. package/dist/internal/sharing/interface.d.ts +1 -1
  80. package/dist/internal/sharing/sharingAccess.js +1 -1
  81. package/dist/internal/sharing/sharingAccess.js.map +1 -1
  82. package/dist/internal/sharing/sharingAccess.test.js +1 -1
  83. package/dist/internal/sharing/sharingAccess.test.js.map +1 -1
  84. package/dist/internal/sharingPublic/apiService.js +2 -0
  85. package/dist/internal/sharingPublic/apiService.js.map +1 -1
  86. package/dist/internal/upload/apiService.d.ts +2 -2
  87. package/dist/internal/upload/apiService.js +1 -1
  88. package/dist/internal/upload/apiService.js.map +1 -1
  89. package/dist/internal/upload/cryptoService.d.ts +2 -2
  90. package/dist/internal/upload/cryptoService.js.map +1 -1
  91. package/dist/internal/upload/fileUploader.d.ts +1 -0
  92. package/dist/internal/upload/fileUploader.js +3 -0
  93. package/dist/internal/upload/fileUploader.js.map +1 -1
  94. package/dist/internal/upload/interface.d.ts +3 -0
  95. package/dist/internal/upload/manager.d.ts +12 -11
  96. package/dist/internal/upload/manager.js +8 -2
  97. package/dist/internal/upload/manager.js.map +1 -1
  98. package/dist/internal/upload/manager.test.js +8 -0
  99. package/dist/internal/upload/manager.test.js.map +1 -1
  100. package/dist/internal/upload/streamUploader.d.ts +34 -24
  101. package/dist/internal/upload/streamUploader.js +7 -4
  102. package/dist/internal/upload/streamUploader.js.map +1 -1
  103. package/dist/internal/upload/streamUploader.test.js +1 -1
  104. package/dist/internal/upload/streamUploader.test.js.map +1 -1
  105. package/dist/protonDriveClient.d.ts +20 -3
  106. package/dist/protonDriveClient.js +23 -4
  107. package/dist/protonDriveClient.js.map +1 -1
  108. package/dist/protonDrivePhotosClient.d.ts +102 -12
  109. package/dist/protonDrivePhotosClient.js +149 -29
  110. package/dist/protonDrivePhotosClient.js.map +1 -1
  111. package/dist/transformers.d.ts +1 -1
  112. package/dist/transformers.js +1 -0
  113. package/dist/transformers.js.map +1 -1
  114. package/package.json +1 -1
  115. package/src/interface/nodes.ts +4 -0
  116. package/src/internal/apiService/errorCodes.ts +1 -0
  117. package/src/internal/apiService/errors.ts +6 -0
  118. package/src/internal/devices/interface.ts +1 -1
  119. package/src/internal/devices/manager.test.ts +3 -3
  120. package/src/internal/devices/manager.ts +1 -1
  121. package/src/internal/events/apiService.ts +1 -1
  122. package/src/internal/events/coreEventManager.test.ts +21 -27
  123. package/src/internal/events/coreEventManager.ts +1 -1
  124. package/src/internal/events/index.ts +3 -4
  125. package/src/internal/events/interface.ts +4 -0
  126. package/src/internal/nodes/apiService.test.ts +58 -1
  127. package/src/internal/nodes/apiService.ts +104 -17
  128. package/src/internal/nodes/cache.test.ts +1 -0
  129. package/src/internal/nodes/cryptoService.test.ts +8 -8
  130. package/src/internal/nodes/cryptoService.ts +1 -1
  131. package/src/internal/nodes/errors.ts +5 -0
  132. package/src/internal/nodes/extendedAttributes.test.ts +23 -1
  133. package/src/internal/nodes/extendedAttributes.ts +26 -18
  134. package/src/internal/nodes/index.test.ts +2 -1
  135. package/src/internal/nodes/interface.ts +6 -1
  136. package/src/internal/nodes/nodesAccess.test.ts +68 -8
  137. package/src/internal/nodes/nodesAccess.ts +42 -15
  138. package/src/internal/nodes/nodesManagement.test.ts +100 -5
  139. package/src/internal/nodes/nodesManagement.ts +101 -13
  140. package/src/internal/photos/albums.ts +31 -12
  141. package/src/internal/photos/apiService.ts +159 -4
  142. package/src/internal/photos/index.ts +116 -9
  143. package/src/internal/photos/interface.ts +23 -1
  144. package/src/internal/photos/shares.ts +134 -0
  145. package/src/internal/photos/timeline.ts +24 -0
  146. package/src/internal/photos/upload.ts +209 -0
  147. package/src/internal/shares/manager.test.ts +7 -7
  148. package/src/internal/shares/manager.ts +4 -4
  149. package/src/internal/sharing/interface.ts +1 -1
  150. package/src/internal/sharing/sharingAccess.test.ts +1 -1
  151. package/src/internal/sharing/sharingAccess.ts +1 -1
  152. package/src/internal/sharingPublic/apiService.ts +2 -0
  153. package/src/internal/upload/apiService.ts +3 -3
  154. package/src/internal/upload/cryptoService.ts +2 -2
  155. package/src/internal/upload/fileUploader.ts +12 -0
  156. package/src/internal/upload/interface.ts +3 -0
  157. package/src/internal/upload/manager.test.ts +8 -0
  158. package/src/internal/upload/manager.ts +20 -10
  159. package/src/internal/upload/streamUploader.test.ts +17 -12
  160. package/src/internal/upload/streamUploader.ts +35 -27
  161. package/src/protonDriveClient.ts +33 -4
  162. package/src/protonDrivePhotosClient.ts +251 -32
  163. package/src/transformers.ts +2 -0
  164. package/dist/internal/photos/cache.d.ts +0 -6
  165. package/dist/internal/photos/cache.js +0 -15
  166. package/dist/internal/photos/cache.js.map +0 -1
  167. package/dist/internal/photos/photosTimeline.d.ts +0 -10
  168. package/dist/internal/photos/photosTimeline.js +0 -19
  169. package/dist/internal/photos/photosTimeline.js.map +0 -1
  170. package/src/internal/photos/cache.ts +0 -11
  171. package/src/internal/photos/photosTimeline.ts +0 -17
@@ -1,13 +1,168 @@
1
- import { DriveAPIService } from '../apiService';
1
+ import { c } from 'ttag';
2
2
 
3
+ import { DriveAPIService, drivePaths, NotFoundAPIError } from '../apiService';
4
+ import { EncryptedRootShare, EncryptedShareCrypto, ShareType } from '../shares/interface';
5
+ import { makeNodeUid } from '../uids';
6
+
7
+ type GetVolumesResponse = drivePaths['/drive/volumes']['get']['responses']['200']['content']['application/json'];
8
+
9
+ type GetShareResponse = drivePaths['/drive/shares/{shareID}']['get']['responses']['200']['content']['application/json'];
10
+
11
+ type PostCreateVolumeRequest = Extract<
12
+ drivePaths['/drive/photos/volumes']['post']['requestBody'],
13
+ { content: object }
14
+ >['content']['application/json'];
15
+ type PostCreateVolumeResponse =
16
+ drivePaths['/drive/photos/volumes']['post']['responses']['200']['content']['application/json'];
17
+
18
+ type GetTimelineResponse =
19
+ drivePaths['/drive/volumes/{volumeID}/photos']['get']['responses']['200']['content']['application/json'];
20
+
21
+ type GetAlbumsResponse =
22
+ drivePaths['/drive/photos/volumes/{volumeID}/albums']['get']['responses']['200']['content']['application/json'];
23
+
24
+ /**
25
+ * Provides API communication for fetching and manipulating photos and albums
26
+ * metadata.
27
+ *
28
+ * The service is responsible for transforming local objects to API payloads
29
+ * and vice versa. It should not contain any business logic.
30
+ */
3
31
  export class PhotosAPIService {
4
32
  constructor(private apiService: DriveAPIService) {
5
33
  this.apiService = apiService;
6
34
  }
7
35
 
8
- async *iterateTimeline(): AsyncGenerator<any> {}
36
+ async getPhotoShare(): Promise<EncryptedRootShare> {
37
+ // TODO: Switch to drive/v2/shares/photos once available.
38
+
39
+ const volumesResponse = await this.apiService.get<GetVolumesResponse>('drive/volumes');
40
+
41
+ const photoVolume = volumesResponse.Volumes.find((volume) => volume.Type === 2);
42
+
43
+ if (!photoVolume) {
44
+ throw new NotFoundAPIError(c('Error').t`Photo volume not found`);
45
+ }
46
+
47
+ const response = await this.apiService.get<GetShareResponse>(`drive/shares/${photoVolume.Share.ShareID}`);
9
48
 
10
- async *iterateAlbums(): AsyncGenerator<any> {}
49
+ if (!response.AddressID) {
50
+ throw new Error('Photo root share has not address ID set');
51
+ }
11
52
 
12
- async createAlbum(object: any): Promise<any> {}
53
+ return {
54
+ volumeId: response.VolumeID,
55
+ shareId: response.ShareID,
56
+ rootNodeId: response.LinkID,
57
+ creatorEmail: response.Creator,
58
+ encryptedCrypto: {
59
+ armoredKey: response.Key,
60
+ armoredPassphrase: response.Passphrase,
61
+ armoredPassphraseSignature: response.PassphraseSignature,
62
+ },
63
+ addressId: response.AddressID,
64
+ type: ShareType.Photo,
65
+ };
66
+ }
67
+
68
+ async createPhotoVolume(
69
+ share: {
70
+ addressId: string;
71
+ addressKeyId: string;
72
+ } & EncryptedShareCrypto,
73
+ node: {
74
+ encryptedName: string;
75
+ armoredKey: string;
76
+ armoredPassphrase: string;
77
+ armoredPassphraseSignature: string;
78
+ armoredHashKey: string;
79
+ },
80
+ ): Promise<{ volumeId: string; shareId: string; rootNodeId: string }> {
81
+ const response = await this.apiService.post<PostCreateVolumeRequest, PostCreateVolumeResponse>(
82
+ 'drive/photos/volumes',
83
+ {
84
+ Share: {
85
+ AddressID: share.addressId,
86
+ AddressKeyID: share.addressKeyId,
87
+ Key: share.armoredKey,
88
+ Passphrase: share.armoredPassphrase,
89
+ PassphraseSignature: share.armoredPassphraseSignature,
90
+ },
91
+ Link: {
92
+ Name: node.encryptedName,
93
+ NodeKey: node.armoredKey,
94
+ NodePassphrase: node.armoredPassphrase,
95
+ NodePassphraseSignature: node.armoredPassphraseSignature,
96
+ NodeHashKey: node.armoredHashKey,
97
+ },
98
+ },
99
+ );
100
+ return {
101
+ volumeId: response.Volume.VolumeID,
102
+ shareId: response.Volume.Share.ShareID,
103
+ rootNodeId: response.Volume.Share.LinkID,
104
+ };
105
+ }
106
+
107
+ async *iterateTimeline(
108
+ volumeId: string,
109
+ signal?: AbortSignal,
110
+ ): AsyncGenerator<{
111
+ nodeUid: string;
112
+ captureTime: Date;
113
+ tags: number[];
114
+ }> {
115
+ let anchor = '';
116
+ while (true) {
117
+ const response = await this.apiService.get<GetTimelineResponse>(
118
+ `drive/volumes/${volumeId}/photos?${anchor ? `PreviousPageLastLinkID=${anchor}` : ''}`,
119
+ signal,
120
+ );
121
+ for (const photo of response.Photos) {
122
+ const nodeUid = makeNodeUid(volumeId, photo.LinkID);
123
+ yield {
124
+ nodeUid,
125
+ captureTime: new Date(photo.CaptureTime * 1000),
126
+ tags: photo.Tags,
127
+ };
128
+ }
129
+
130
+ if (!response.Photos.length) {
131
+ break;
132
+ }
133
+ anchor = response.Photos[response.Photos.length - 1].LinkID;
134
+ }
135
+ }
136
+
137
+ async *iterateAlbums(
138
+ volumeId: string,
139
+ signal?: AbortSignal,
140
+ ): AsyncGenerator<{
141
+ albumUid: string;
142
+ coverNodeUid?: string;
143
+ photoCount: number;
144
+ lastActivityTime: Date;
145
+ }> {
146
+ let anchor = '';
147
+ while (true) {
148
+ const response = await this.apiService.get<GetAlbumsResponse>(
149
+ `drive/photos/volumes/${volumeId}/albums?${anchor ? `AnchorID=${anchor}` : ''}`,
150
+ signal,
151
+ );
152
+ for (const album of response.Albums) {
153
+ const albumUid = makeNodeUid(volumeId, album.LinkID);
154
+ yield {
155
+ albumUid,
156
+ coverNodeUid: album.CoverLinkID ? makeNodeUid(volumeId, album.CoverLinkID) : undefined,
157
+ photoCount: album.PhotoCount,
158
+ lastActivityTime: new Date(album.LastActivityTime * 1000),
159
+ };
160
+ }
161
+
162
+ if (!response.More || !response.AnchorID) {
163
+ break;
164
+ }
165
+ anchor = response.AnchorID;
166
+ }
167
+ }
13
168
  }
@@ -1,23 +1,130 @@
1
1
  import { DriveAPIService } from '../apiService';
2
- import { ProtonDriveEntitiesCache } from '../../interface';
3
- import { PhotosAPIService } from './apiService';
4
- import { PhotosCache } from './cache';
5
- import { PhotosTimeline } from './photosTimeline';
2
+ import { DriveCrypto } from '../../crypto';
3
+ import {
4
+ ProtonDriveAccount,
5
+ ProtonDriveCryptoCache,
6
+ ProtonDriveEntitiesCache,
7
+ ProtonDriveTelemetry,
8
+ } from '../../interface';
9
+ import { SharesCache } from '../shares/cache';
10
+ import { SharesCryptoCache } from '../shares/cryptoCache';
11
+ import { SharesCryptoService } from '../shares/cryptoService';
12
+ import { NodesService as UploadNodesService } from '../upload/interface';
13
+ import { UploadTelemetry } from '../upload/telemetry';
14
+ import { UploadQueue } from '../upload/queue';
6
15
  import { Albums } from './albums';
7
- import { NodesService } from './interface';
16
+ import { PhotosAPIService } from './apiService';
17
+ import { NodesService, SharesService } from './interface';
18
+ import { PhotoSharesManager } from './shares';
19
+ import { PhotosTimeline } from './timeline';
20
+ import {
21
+ PhotoFileUploader,
22
+ PhotoUploadAPIService,
23
+ PhotoUploadCryptoService,
24
+ PhotoUploadManager,
25
+ PhotoUploadMetadata,
26
+ } from './upload';
8
27
 
28
+ /**
29
+ * Provides facade for the whole photos module.
30
+ *
31
+ * The photos module is responsible for handling photos and albums metadata,
32
+ * including API communication, crypto, caching, and event handling.
33
+ */
9
34
  export function initPhotosModule(
10
35
  apiService: DriveAPIService,
11
- driveEntitiesCache: ProtonDriveEntitiesCache,
36
+ photoShares: PhotoSharesManager,
12
37
  nodesService: NodesService,
13
38
  ) {
14
39
  const api = new PhotosAPIService(apiService);
15
- const cache = new PhotosCache(driveEntitiesCache);
16
- const timeline = new PhotosTimeline(api, cache, nodesService);
17
- const albums = new Albums(api, cache, nodesService);
40
+ const timeline = new PhotosTimeline(api, photoShares);
41
+ const albums = new Albums(api, photoShares, nodesService);
18
42
 
19
43
  return {
20
44
  timeline,
21
45
  albums,
22
46
  };
23
47
  }
48
+
49
+ /**
50
+ * Provides facade for the photo share module.
51
+ *
52
+ * The photo share wraps the core share module, but uses photos volume instead
53
+ * of main volume. It provides the same interface so it can be used in the same
54
+ * way in various modules that use shares.
55
+ */
56
+ export function initPhotoSharesModule(
57
+ telemetry: ProtonDriveTelemetry,
58
+ apiService: DriveAPIService,
59
+ driveEntitiesCache: ProtonDriveEntitiesCache,
60
+ driveCryptoCache: ProtonDriveCryptoCache,
61
+ account: ProtonDriveAccount,
62
+ crypto: DriveCrypto,
63
+ sharesService: SharesService,
64
+ ) {
65
+ const api = new PhotosAPIService(apiService);
66
+ const cache = new SharesCache(telemetry.getLogger('shares-cache'), driveEntitiesCache);
67
+ const cryptoCache = new SharesCryptoCache(telemetry.getLogger('shares-cache'), driveCryptoCache);
68
+ const cryptoService = new SharesCryptoService(telemetry, crypto, account);
69
+
70
+ return new PhotoSharesManager(
71
+ telemetry.getLogger('photos-shares'),
72
+ api,
73
+ cache,
74
+ cryptoCache,
75
+ cryptoService,
76
+ sharesService,
77
+ );
78
+ }
79
+
80
+ /**
81
+ * Provides facade for the photo upload module.
82
+ *
83
+ * The photo upload wraps the core upload module and adds photo specific metadata.
84
+ * It provides the same interface so it can be used in the same way.
85
+ */
86
+ export function initPhotoUploadModule(
87
+ telemetry: ProtonDriveTelemetry,
88
+ apiService: DriveAPIService,
89
+ driveCrypto: DriveCrypto,
90
+ sharesService: SharesService,
91
+ nodesService: UploadNodesService,
92
+ clientUid?: string,
93
+ ) {
94
+ const api = new PhotoUploadAPIService(apiService, clientUid);
95
+ const cryptoService = new PhotoUploadCryptoService(driveCrypto, nodesService);
96
+
97
+ const uploadTelemetry = new UploadTelemetry(telemetry, sharesService);
98
+ const manager = new PhotoUploadManager(telemetry, api, cryptoService, nodesService, clientUid);
99
+
100
+ const queue = new UploadQueue();
101
+
102
+ async function getFileUploader(
103
+ parentFolderUid: string,
104
+ name: string,
105
+ metadata: PhotoUploadMetadata,
106
+ signal?: AbortSignal,
107
+ ): Promise<PhotoFileUploader> {
108
+ await queue.waitForCapacity(signal);
109
+
110
+ const onFinish = () => {
111
+ queue.releaseCapacity();
112
+ };
113
+
114
+ return new PhotoFileUploader(
115
+ uploadTelemetry,
116
+ api,
117
+ cryptoService,
118
+ manager,
119
+ parentFolderUid,
120
+ name,
121
+ metadata,
122
+ onFinish,
123
+ signal,
124
+ );
125
+ }
126
+
127
+ return {
128
+ getFileUploader,
129
+ };
130
+ }
@@ -1,5 +1,27 @@
1
- import { MissingNode } from '../../interface';
1
+ import { PrivateKey } from '../../crypto';
2
+ import { MissingNode, MetricVolumeType } from '../../interface';
2
3
  import { DecryptedNode } from '../nodes';
4
+ import { EncryptedShare } from '../shares';
5
+
6
+ export interface SharesService {
7
+ getOwnVolumeIDs(): Promise<{ volumeId: string; rootNodeId: string }>;
8
+ loadEncryptedShare(shareId: string): Promise<EncryptedShare>;
9
+ getSharePrivateKey(shareId: string): Promise<PrivateKey>;
10
+ getMyFilesShareMemberEmailKey(): Promise<{
11
+ email: string;
12
+ addressId: string;
13
+ addressKey: PrivateKey;
14
+ addressKeyId: string;
15
+ }>;
16
+ getContextShareMemberEmailKey(shareId: string): Promise<{
17
+ email: string;
18
+ addressId: string;
19
+ addressKey: PrivateKey;
20
+ addressKeyId: string;
21
+ }>;
22
+ isOwnVolume(volumeId: string): Promise<boolean>;
23
+ getVolumeMetricContext(volumeId: string): Promise<MetricVolumeType>;
24
+ }
3
25
 
4
26
  export interface NodesService {
5
27
  getNode(nodeUid: string): Promise<DecryptedNode>;
@@ -0,0 +1,134 @@
1
+ import { PrivateKey } from '../../crypto';
2
+ import { Logger, MetricVolumeType } from '../../interface';
3
+ import { NotFoundAPIError } from '../apiService';
4
+ import { SharesCache } from '../shares/cache';
5
+ import { SharesCryptoCache } from '../shares/cryptoCache';
6
+ import { SharesCryptoService } from '../shares/cryptoService';
7
+ import { EncryptedShare, VolumeShareNodeIDs } from '../shares/interface';
8
+ import { PhotosAPIService } from './apiService';
9
+ import { SharesService } from './interface';
10
+
11
+ /**
12
+ * Provides high-level actions for managing photo share.
13
+ *
14
+ * The photo share manager wraps the core share service, but uses photos volume
15
+ * instead of main volume. It provides the same interface so it can be used in
16
+ * the same way in various modules that use shares.
17
+ */
18
+ export class PhotoSharesManager {
19
+ private photoRootIds?: VolumeShareNodeIDs;
20
+
21
+ constructor(
22
+ private logger: Logger,
23
+ private apiService: PhotosAPIService,
24
+ private cache: SharesCache,
25
+ private cryptoCache: SharesCryptoCache,
26
+ private cryptoService: SharesCryptoService,
27
+ private sharesService: SharesService,
28
+ ) {
29
+ this.logger = logger;
30
+ this.apiService = apiService;
31
+ this.cache = cache;
32
+ this.cryptoCache = cryptoCache;
33
+ this.cryptoService = cryptoService;
34
+ this.sharesService = sharesService;
35
+ }
36
+
37
+ async getOwnVolumeIDs(): Promise<VolumeShareNodeIDs> {
38
+ if (this.photoRootIds) {
39
+ return this.photoRootIds;
40
+ }
41
+
42
+ try {
43
+ const encryptedShare = await this.apiService.getPhotoShare();
44
+
45
+ // Once any place needs IDs for My files, it will most likely
46
+ // need also the keys for decrypting the tree. It is better to
47
+ // decrypt the share here right away.
48
+ const { share: myFilesShare, key } = await this.cryptoService.decryptRootShare(encryptedShare);
49
+ await this.cryptoCache.setShareKey(myFilesShare.shareId, key);
50
+ await this.cache.setVolume({
51
+ volumeId: myFilesShare.volumeId,
52
+ shareId: myFilesShare.shareId,
53
+ rootNodeId: myFilesShare.rootNodeId,
54
+ creatorEmail: encryptedShare.creatorEmail,
55
+ addressId: encryptedShare.addressId,
56
+ });
57
+
58
+ this.photoRootIds = {
59
+ volumeId: myFilesShare.volumeId,
60
+ shareId: myFilesShare.shareId,
61
+ rootNodeId: myFilesShare.rootNodeId,
62
+ };
63
+ return this.photoRootIds;
64
+ } catch (error: unknown) {
65
+ if (error instanceof NotFoundAPIError) {
66
+ this.logger.warn('Active photo volume not found, creating a new one');
67
+ return this.createVolume();
68
+ }
69
+ this.logger.error('Failed to get active photo volume', error);
70
+ throw error;
71
+ }
72
+ }
73
+
74
+ private async createVolume(): Promise<VolumeShareNodeIDs> {
75
+ const address = await this.sharesService.getMyFilesShareMemberEmailKey();
76
+ const bootstrap = await this.cryptoService.generateVolumeBootstrap(address.addressKey);
77
+ const photoRootIds = await this.apiService.createPhotoVolume(
78
+ {
79
+ addressId: address.addressId,
80
+ addressKeyId: address.addressKeyId,
81
+ ...bootstrap.shareKey.encrypted,
82
+ },
83
+ {
84
+ ...bootstrap.rootNode.key.encrypted,
85
+ encryptedName: bootstrap.rootNode.encryptedName,
86
+ armoredHashKey: bootstrap.rootNode.armoredHashKey,
87
+ },
88
+ );
89
+ await this.cryptoCache.setShareKey(photoRootIds.shareId, bootstrap.shareKey.decrypted);
90
+ return photoRootIds;
91
+ }
92
+
93
+ async getSharePrivateKey(shareId: string): Promise<PrivateKey> {
94
+ return this.sharesService.getSharePrivateKey(shareId);
95
+ }
96
+
97
+ async getMyFilesShareMemberEmailKey(): Promise<{
98
+ email: string;
99
+ addressId: string;
100
+ addressKey: PrivateKey;
101
+ addressKeyId: string;
102
+ }> {
103
+ return this.sharesService.getMyFilesShareMemberEmailKey();
104
+ }
105
+
106
+ async getContextShareMemberEmailKey(shareId: string): Promise<{
107
+ email: string;
108
+ addressId: string;
109
+ addressKey: PrivateKey;
110
+ addressKeyId: string;
111
+ }> {
112
+ return this.sharesService.getContextShareMemberEmailKey(shareId);
113
+ }
114
+
115
+ async isOwnVolume(volumeId: string): Promise<boolean> {
116
+ const { volumeId: myVolumeId } = await this.getOwnVolumeIDs();
117
+ if (volumeId === myVolumeId) {
118
+ return true;
119
+ }
120
+ return this.sharesService.isOwnVolume(volumeId);
121
+ }
122
+
123
+ async getVolumeMetricContext(volumeId: string): Promise<MetricVolumeType> {
124
+ const { volumeId: myVolumeId } = await this.getOwnVolumeIDs();
125
+ if (volumeId === myVolumeId) {
126
+ return MetricVolumeType.OwnVolume;
127
+ }
128
+ return this.sharesService.getVolumeMetricContext(volumeId);
129
+ }
130
+
131
+ async loadEncryptedShare(shareId: string): Promise<EncryptedShare> {
132
+ return this.sharesService.loadEncryptedShare(shareId);
133
+ }
134
+ }
@@ -0,0 +1,24 @@
1
+ import { PhotosAPIService } from './apiService';
2
+ import { PhotoSharesManager } from './shares';
3
+
4
+ /**
5
+ * Provides access to the photo timeline.
6
+ */
7
+ export class PhotosTimeline {
8
+ constructor(
9
+ private apiService: PhotosAPIService,
10
+ private photoShares: PhotoSharesManager,
11
+ ) {
12
+ this.apiService = apiService;
13
+ this.photoShares = photoShares;
14
+ }
15
+
16
+ async* iterateTimeline(signal?: AbortSignal): AsyncGenerator<{
17
+ nodeUid: string;
18
+ captureTime: Date;
19
+ tags: number[];
20
+ }> {
21
+ const { volumeId } = await this.photoShares.getOwnVolumeIDs();
22
+ yield* this.apiService.iterateTimeline(volumeId, signal);
23
+ }
24
+ }