@protontech/drive-sdk 0.9.9 → 0.10.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 (165) hide show
  1. package/dist/crypto/driveCrypto.d.ts +15 -15
  2. package/dist/crypto/driveCrypto.js.map +1 -1
  3. package/dist/crypto/hmac.d.ts +3 -3
  4. package/dist/crypto/hmac.js.map +1 -1
  5. package/dist/crypto/interface.d.ts +45 -25
  6. package/dist/crypto/interface.js.map +1 -1
  7. package/dist/crypto/openPGPCrypto.d.ts +37 -37
  8. package/dist/crypto/openPGPCrypto.js.map +1 -1
  9. package/dist/crypto/utils.d.ts +1 -1
  10. package/dist/interface/index.d.ts +3 -3
  11. package/dist/interface/index.js.map +1 -1
  12. package/dist/interface/nodes.d.ts +8 -0
  13. package/dist/interface/photos.d.ts +18 -1
  14. package/dist/interface/sharing.d.ts +2 -0
  15. package/dist/interface/telemetry.d.ts +1 -0
  16. package/dist/interface/telemetry.js.map +1 -1
  17. package/dist/interface/thumbnail.d.ts +2 -2
  18. package/dist/internal/apiService/apiService.js +25 -12
  19. package/dist/internal/apiService/apiService.js.map +1 -1
  20. package/dist/internal/apiService/apiService.test.js +33 -5
  21. package/dist/internal/apiService/apiService.test.js.map +1 -1
  22. package/dist/internal/apiService/driveTypes.d.ts +2942 -3187
  23. package/dist/internal/apiService/errors.test.js +17 -7
  24. package/dist/internal/apiService/errors.test.js.map +1 -1
  25. package/dist/internal/devices/manager.d.ts +1 -0
  26. package/dist/internal/devices/manager.js +11 -0
  27. package/dist/internal/devices/manager.js.map +1 -1
  28. package/dist/internal/download/apiService.d.ts +1 -1
  29. package/dist/internal/download/cryptoService.d.ts +4 -4
  30. package/dist/internal/download/cryptoService.js.map +1 -1
  31. package/dist/internal/download/fileDownloader.js.map +1 -1
  32. package/dist/internal/download/fileDownloader.test.js.map +1 -1
  33. package/dist/internal/download/thumbnailDownloader.js.map +1 -1
  34. package/dist/internal/nodes/cryptoService.d.ts +4 -4
  35. package/dist/internal/nodes/cryptoService.js +5 -3
  36. package/dist/internal/nodes/cryptoService.js.map +1 -1
  37. package/dist/internal/nodes/cryptoService.test.js.map +1 -1
  38. package/dist/internal/nodes/interface.d.ts +1 -1
  39. package/dist/internal/nodes/nodesManagement.js +0 -1
  40. package/dist/internal/nodes/nodesManagement.js.map +1 -1
  41. package/dist/internal/photos/addToAlbum.d.ts +46 -0
  42. package/dist/internal/photos/addToAlbum.js +257 -0
  43. package/dist/internal/photos/addToAlbum.js.map +1 -0
  44. package/dist/internal/photos/addToAlbum.test.d.ts +1 -0
  45. package/dist/internal/photos/addToAlbum.test.js +409 -0
  46. package/dist/internal/photos/addToAlbum.test.js.map +1 -0
  47. package/dist/internal/photos/albums.d.ts +5 -3
  48. package/dist/internal/photos/albums.js +13 -1
  49. package/dist/internal/photos/albums.js.map +1 -1
  50. package/dist/internal/photos/albums.test.js +2 -1
  51. package/dist/internal/photos/albums.test.js.map +1 -1
  52. package/dist/internal/photos/albumsCrypto.d.ts +20 -3
  53. package/dist/internal/photos/albumsCrypto.js +27 -0
  54. package/dist/internal/photos/albumsCrypto.js.map +1 -1
  55. package/dist/internal/photos/apiService.d.ts +19 -3
  56. package/dist/internal/photos/apiService.js +104 -5
  57. package/dist/internal/photos/apiService.js.map +1 -1
  58. package/dist/internal/photos/apiService.test.d.ts +1 -0
  59. package/dist/internal/photos/apiService.test.js +199 -0
  60. package/dist/internal/photos/apiService.test.js.map +1 -0
  61. package/dist/internal/photos/errors.d.ts +4 -0
  62. package/dist/internal/photos/errors.js +17 -0
  63. package/dist/internal/photos/errors.js.map +1 -0
  64. package/dist/internal/photos/index.js +1 -1
  65. package/dist/internal/photos/index.js.map +1 -1
  66. package/dist/internal/photos/interface.d.ts +15 -1
  67. package/dist/internal/photos/interface.js.map +1 -1
  68. package/dist/internal/photos/nodes.js +32 -2
  69. package/dist/internal/photos/nodes.js.map +1 -1
  70. package/dist/internal/photos/nodes.test.js +25 -5
  71. package/dist/internal/photos/nodes.test.js.map +1 -1
  72. package/dist/internal/photos/upload.d.ts +2 -2
  73. package/dist/internal/photos/upload.js.map +1 -1
  74. package/dist/internal/shares/apiService.js +1 -0
  75. package/dist/internal/shares/apiService.js.map +1 -1
  76. package/dist/internal/shares/interface.d.ts +1 -0
  77. package/dist/internal/sharing/apiService.d.ts +8 -1
  78. package/dist/internal/sharing/apiService.js +23 -1
  79. package/dist/internal/sharing/apiService.js.map +1 -1
  80. package/dist/internal/sharing/cryptoService.js +8 -4
  81. package/dist/internal/sharing/cryptoService.js.map +1 -1
  82. package/dist/internal/sharing/sharingManagement.d.ts +1 -0
  83. package/dist/internal/sharing/sharingManagement.js +15 -2
  84. package/dist/internal/sharing/sharingManagement.js.map +1 -1
  85. package/dist/internal/sharing/sharingManagement.test.js +30 -5
  86. package/dist/internal/sharing/sharingManagement.test.js.map +1 -1
  87. package/dist/internal/sharingPublic/nodes.d.ts +2 -2
  88. package/dist/internal/sharingPublic/nodes.js.map +1 -1
  89. package/dist/internal/upload/apiService.d.ts +5 -5
  90. package/dist/internal/upload/apiService.js.map +1 -1
  91. package/dist/internal/upload/blockVerifier.d.ts +2 -2
  92. package/dist/internal/upload/blockVerifier.js.map +1 -1
  93. package/dist/internal/upload/chunkStreamReader.d.ts +2 -2
  94. package/dist/internal/upload/chunkStreamReader.js.map +1 -1
  95. package/dist/internal/upload/chunkStreamReader.test.js.map +1 -1
  96. package/dist/internal/upload/cryptoService.d.ts +7 -7
  97. package/dist/internal/upload/cryptoService.js.map +1 -1
  98. package/dist/internal/upload/interface.d.ts +6 -6
  99. package/dist/internal/upload/manager.d.ts +1 -1
  100. package/dist/internal/upload/manager.js.map +1 -1
  101. package/dist/internal/upload/streamUploader.d.ts +1 -1
  102. package/dist/internal/utils.d.ts +1 -1
  103. package/dist/protonDriveClient.d.ts +8 -0
  104. package/dist/protonDriveClient.js +11 -0
  105. package/dist/protonDriveClient.js.map +1 -1
  106. package/dist/protonDrivePhotosClient.d.ts +17 -2
  107. package/dist/protonDrivePhotosClient.js +21 -1
  108. package/dist/protonDrivePhotosClient.js.map +1 -1
  109. package/dist/transformers.js +2 -0
  110. package/dist/transformers.js.map +1 -1
  111. package/package.json +4 -4
  112. package/src/crypto/driveCrypto.ts +15 -15
  113. package/src/crypto/hmac.ts +4 -4
  114. package/src/crypto/interface.ts +58 -27
  115. package/src/crypto/openPGPCrypto.ts +26 -26
  116. package/src/interface/index.ts +10 -2
  117. package/src/interface/nodes.ts +1 -0
  118. package/src/interface/photos.ts +19 -1
  119. package/src/interface/sharing.ts +2 -0
  120. package/src/interface/telemetry.ts +1 -0
  121. package/src/interface/thumbnail.ts +2 -2
  122. package/src/internal/apiService/apiService.test.ts +38 -6
  123. package/src/internal/apiService/apiService.ts +33 -12
  124. package/src/internal/apiService/driveTypes.ts +2942 -3187
  125. package/src/internal/devices/manager.ts +14 -0
  126. package/src/internal/download/apiService.ts +1 -1
  127. package/src/internal/download/cryptoService.ts +4 -4
  128. package/src/internal/download/fileDownloader.test.ts +4 -4
  129. package/src/internal/download/fileDownloader.ts +6 -6
  130. package/src/internal/download/thumbnailDownloader.ts +4 -4
  131. package/src/internal/nodes/cryptoService.test.ts +2 -2
  132. package/src/internal/nodes/cryptoService.ts +11 -8
  133. package/src/internal/nodes/interface.ts +1 -1
  134. package/src/internal/nodes/nodesManagement.ts +0 -1
  135. package/src/internal/photos/addToAlbum.test.ts +515 -0
  136. package/src/internal/photos/addToAlbum.ts +341 -0
  137. package/src/internal/photos/albums.test.ts +20 -23
  138. package/src/internal/photos/albums.ts +31 -2
  139. package/src/internal/photos/albumsCrypto.ts +54 -3
  140. package/src/internal/photos/apiService.test.ts +233 -0
  141. package/src/internal/photos/apiService.ts +172 -27
  142. package/src/internal/photos/errors.ts +11 -0
  143. package/src/internal/photos/index.ts +1 -1
  144. package/src/internal/photos/interface.ts +18 -3
  145. package/src/internal/photos/nodes.test.ts +27 -6
  146. package/src/internal/photos/nodes.ts +34 -2
  147. package/src/internal/photos/upload.ts +2 -2
  148. package/src/internal/shares/apiService.ts +1 -0
  149. package/src/internal/shares/interface.ts +1 -0
  150. package/src/internal/sharing/apiService.ts +49 -5
  151. package/src/internal/sharing/cryptoService.ts +10 -4
  152. package/src/internal/sharing/sharingManagement.test.ts +33 -5
  153. package/src/internal/sharing/sharingManagement.ts +28 -6
  154. package/src/internal/sharingPublic/nodes.ts +1 -1
  155. package/src/internal/upload/apiService.ts +5 -5
  156. package/src/internal/upload/blockVerifier.ts +3 -3
  157. package/src/internal/upload/chunkStreamReader.test.ts +7 -7
  158. package/src/internal/upload/chunkStreamReader.ts +3 -3
  159. package/src/internal/upload/cryptoService.ts +9 -9
  160. package/src/internal/upload/interface.ts +6 -6
  161. package/src/internal/upload/manager.ts +2 -2
  162. package/src/internal/upload/streamUploader.ts +1 -1
  163. package/src/protonDriveClient.ts +15 -3
  164. package/src/protonDrivePhotosClient.ts +39 -15
  165. package/src/transformers.ts +2 -0
@@ -8,7 +8,7 @@ export type NodeRevisionDraft = {
8
8
  nodeRevisionUid: string;
9
9
  nodeKeys: NodeRevisionDraftKeys;
10
10
  parentNodeKeys?: {
11
- hashKey: Uint8Array;
11
+ hashKey: Uint8Array<ArrayBuffer>;
12
12
  };
13
13
  // newNodeInfo is set only when revision is created with the new node.
14
14
  newNodeInfo?: {
@@ -64,19 +64,19 @@ export type NodeCryptoSigningKeys = {
64
64
  export type EncryptedBlockMetadata = {
65
65
  encryptedSize: number;
66
66
  originalSize: number;
67
- hash: Uint8Array;
67
+ hash: Uint8Array<ArrayBuffer>;
68
68
  };
69
69
 
70
70
  export type EncryptedBlock = EncryptedBlockMetadata & {
71
71
  index: number;
72
- encryptedData: Uint8Array;
72
+ encryptedData: Uint8Array<ArrayBuffer>;
73
73
  armoredSignature: string;
74
- verificationToken: Uint8Array;
74
+ verificationToken: Uint8Array<ArrayBuffer>;
75
75
  };
76
76
 
77
77
  export type EncryptedThumbnail = EncryptedBlockMetadata & {
78
78
  type: ThumbnailType;
79
- encryptedData: Uint8Array;
79
+ encryptedData: Uint8Array<ArrayBuffer>;
80
80
  };
81
81
 
82
82
  export type UploadTokens = {
@@ -101,7 +101,7 @@ export interface NodesService {
101
101
  key: PrivateKey;
102
102
  passphraseSessionKey: SessionKey;
103
103
  contentKeyPacketSessionKey?: SessionKey;
104
- hashKey?: Uint8Array;
104
+ hashKey?: Uint8Array<ArrayBuffer>;
105
105
  }>;
106
106
  getNodeSigningKeys(
107
107
  uids: { nodeUid: string; parentNodeUid?: string } | { nodeUid?: string; parentNodeUid: string },
@@ -73,7 +73,7 @@ export class UploadManager {
73
73
 
74
74
  private async createDraftOnAPI(
75
75
  parentFolderUid: string,
76
- parentHashKey: Uint8Array,
76
+ parentHashKey: Uint8Array<ArrayBuffer>,
77
77
  name: string,
78
78
  metadata: UploadMetadata,
79
79
  generatedNodeCrypto: NodeCrypto,
@@ -226,7 +226,7 @@ export class UploadManager {
226
226
 
227
227
  async commitDraft(
228
228
  nodeRevisionDraft: NodeRevisionDraft,
229
- manifest: Uint8Array,
229
+ manifest: Uint8Array<ArrayBuffer>,
230
230
  extendedAttributes: {
231
231
  modificationTime?: Date;
232
232
  size: number;
@@ -655,7 +655,7 @@ export class StreamUploader {
655
655
  return uploadedBlocks.map((block) => block.originalSize);
656
656
  }
657
657
 
658
- protected get manifest(): Uint8Array {
658
+ protected get manifest(): Uint8Array<ArrayBuffer> {
659
659
  this.uploadedThumbnails.sort((a, b) => a.type - b.type);
660
660
  this.uploadedBlocks.sort((a, b) => a.index - b.index);
661
661
  const hashes = [
@@ -223,12 +223,12 @@ export class ProtonDriveClient {
223
223
  return keys.contentKeyPacketSessionKey;
224
224
  },
225
225
  getPublicLinkInfo: async (url: string) => {
226
- const { token } = getTokenAndPasswordFromUrl(url)
226
+ const { token } = getTokenAndPasswordFromUrl(url);
227
227
  this.logger.info(`Getting info for public link token ${token}`);
228
228
  return this.publicSessionManager.getInfo(token);
229
229
  },
230
230
  authPublicLink: async (url: string, customPassword?: string, isAnonymousContext: boolean = false) => {
231
- const { token, password: urlPassword } = getTokenAndPasswordFromUrl(url)
231
+ const { token, password: urlPassword } = getTokenAndPasswordFromUrl(url);
232
232
  this.logger.info(`Authenticating public link token ${token}`);
233
233
 
234
234
  const { httpClient, shareKey, rootUid, publicRole } = await this.publicSessionManager.auth(
@@ -681,7 +681,7 @@ export class ProtonDriveClient {
681
681
  * @param customPassword - The optional custom password.
682
682
  */
683
683
  async createBookmark(url: string, customPassword?: string): Promise<void> {
684
- const { token, password: urlPassword } = getTokenAndPasswordFromUrl(url)
684
+ const { token, password: urlPassword } = getTokenAndPasswordFromUrl(url);
685
685
  this.logger.info(`Creating bookmark for token ${token}`);
686
686
  await this.sharing.access.createBookmark(token, urlPassword, customPassword);
687
687
  }
@@ -912,6 +912,18 @@ export class ProtonDriveClient {
912
912
  yield* this.devices.iterateDevices(signal);
913
913
  }
914
914
 
915
+ /**
916
+ * Get the device entity by its UID.
917
+ *
918
+ * @param deviceOrUid - Device entity or its UID string.
919
+ * @returns The device entity.
920
+ * @throws {@link ValidationError} If the device is not found.
921
+ */
922
+ async getDevice(deviceOrUid: DeviceOrUid): Promise<Device> {
923
+ this.logger.info(`Getting device ${getUid(deviceOrUid)}`);
924
+ return this.devices.getDevice(getUid(deviceOrUid));
925
+ }
926
+
915
927
  /**
916
928
  * Creates a new device.
917
929
  *
@@ -17,6 +17,7 @@ import {
17
17
  NonProtonInvitationOrUid,
18
18
  ProtonInvitationWithNode,
19
19
  NodeResult,
20
+ NodeResultWithError,
20
21
  } from './interface';
21
22
  import { getConfig } from './config';
22
23
  import { DriveCrypto } from './crypto';
@@ -120,13 +121,7 @@ export class ProtonDrivePhotosClient {
120
121
  this.photoShares,
121
122
  fullConfig.clientUid,
122
123
  );
123
- this.photos = initPhotosModule(
124
- telemetry,
125
- apiService,
126
- cryptoModule,
127
- this.photoShares,
128
- this.nodes.access,
129
- );
124
+ this.photos = initPhotosModule(telemetry, apiService, cryptoModule, this.photoShares, this.nodes.access);
130
125
  this.sharing = initSharingModule(
131
126
  telemetry,
132
127
  apiService,
@@ -238,7 +233,7 @@ export class ProtonDrivePhotosClient {
238
233
  */
239
234
  async *iterateTrashedNodes(signal?: AbortSignal): AsyncGenerator<MaybePhotoNode> {
240
235
  this.logger.info('Iterating trashed nodes');
241
- yield * convertInternalPhotoNodeIterator(this.nodes.access.iterateTrashedNodes(signal));
236
+ yield* convertInternalPhotoNodeIterator(this.nodes.access.iterateTrashedNodes(signal));
242
237
  }
243
238
 
244
239
  /**
@@ -249,7 +244,7 @@ export class ProtonDrivePhotosClient {
249
244
  async *iterateNodes(nodeUids: NodeOrUid[], signal?: AbortSignal): AsyncGenerator<MaybeMissingPhotoNode> {
250
245
  this.logger.info(`Iterating ${nodeUids.length} nodes`);
251
246
  // TODO: expose photo type
252
- yield * convertInternalMissingPhotoNodeIterator(this.nodes.access.iterateNodes(getUids(nodeUids), signal));
247
+ yield* convertInternalMissingPhotoNodeIterator(this.nodes.access.iterateNodes(getUids(nodeUids), signal));
253
248
  }
254
249
 
255
250
  /**
@@ -299,7 +294,7 @@ export class ProtonDrivePhotosClient {
299
294
  */
300
295
  async *deleteNodes(nodeUids: NodeOrUid[], signal?: AbortSignal): AsyncGenerator<NodeResult> {
301
296
  this.logger.info(`Deleting ${nodeUids.length} nodes`);
302
- yield * this.nodes.management.deleteTrashedNodes(getUids(nodeUids), signal);
297
+ yield* this.nodes.management.deleteTrashedNodes(getUids(nodeUids), signal);
303
298
  }
304
299
 
305
300
  /**
@@ -317,7 +312,7 @@ export class ProtonDrivePhotosClient {
317
312
  */
318
313
  async *iterateSharedNodes(signal?: AbortSignal): AsyncGenerator<MaybePhotoNode> {
319
314
  this.logger.info('Iterating shared nodes by me');
320
- yield * convertInternalPhotoNodeIterator(this.sharing.access.iterateSharedNodes(signal));
315
+ yield* convertInternalPhotoNodeIterator(this.sharing.access.iterateSharedNodes(signal));
321
316
  }
322
317
 
323
318
  /**
@@ -484,7 +479,9 @@ export class ProtonDrivePhotosClient {
484
479
  */
485
480
  async isDuplicatePhoto(name: string, generateSha1: () => Promise<string>, signal?: AbortSignal): Promise<boolean> {
486
481
  this.logger.info(`Checking if photo is a duplicate`);
487
- return this.photos.timeline.findPhotoDuplicates(name, generateSha1, signal).then(nodeUids => nodeUids.length !== 0);
482
+ return this.photos.timeline
483
+ .findPhotoDuplicates(name, generateSha1, signal)
484
+ .then((nodeUids) => nodeUids.length !== 0);
488
485
  }
489
486
 
490
487
  /**
@@ -505,7 +502,11 @@ export class ProtonDrivePhotosClient {
505
502
  * @param signal - An optional abort signal to cancel the operation.
506
503
  * @returns An array of node UIDs of duplicate photos. Empty array if no duplicates found.
507
504
  */
508
- async findPhotoDuplicates(name: string, generateSha1: () => Promise<string>, signal?: AbortSignal): Promise<string[]> {
505
+ async findPhotoDuplicates(
506
+ name: string,
507
+ generateSha1: () => Promise<string>,
508
+ signal?: AbortSignal,
509
+ ): Promise<string[]> {
509
510
  this.logger.info(`Checking if photo have duplicates`);
510
511
  return this.photos.timeline.findPhotoDuplicates(name, generateSha1, signal);
511
512
  }
@@ -575,7 +576,7 @@ export class ProtonDrivePhotosClient {
575
576
  async *iterateAlbums(signal?: AbortSignal): AsyncGenerator<MaybePhotoNode> {
576
577
  this.logger.info('Iterating albums');
577
578
  // TODO: expose album type
578
- yield * convertInternalPhotoNodeIterator(this.photos.albums.iterateAlbums(signal));
579
+ yield* convertInternalPhotoNodeIterator(this.photos.albums.iterateAlbums(signal));
579
580
  }
580
581
 
581
582
  /**
@@ -592,6 +593,29 @@ export class ProtonDrivePhotosClient {
592
593
  yield* this.photos.albums.iterateAlbum(getUid(albumNodeUid), signal);
593
594
  }
594
595
 
596
+ /**
597
+ * Adds photos to an album.
598
+ *
599
+ * Photos are added in batches. Each photo's related photos (e.g., live
600
+ * photo components) are always included with the main photo.
601
+ *
602
+ * The album has a limit of 10,000 photos. If the limit is reached,
603
+ * a `ValidationError` is thrown.
604
+ *
605
+ * @param albumNodeUid - The UID of the album to add photos to.
606
+ * @param photoNodeUids - The UIDs of the photos to add to the album.
607
+ * @param signal - An optional abort signal to cancel the operation.
608
+ * @returns An async generator of the added photo results.
609
+ */
610
+ async *addPhotosToAlbum(
611
+ albumNodeUid: NodeOrUid,
612
+ photoNodeUids: NodeOrUid[],
613
+ signal?: AbortSignal,
614
+ ): AsyncGenerator<NodeResultWithError> {
615
+ this.logger.info(`Adding ${photoNodeUids.length} photos to album ${getUid(albumNodeUid)}`);
616
+ yield* this.photos.albums.addPhotos(getUid(albumNodeUid), getUids(photoNodeUids), signal);
617
+ }
618
+
595
619
  /**
596
620
  * Removes photos from an album.
597
621
  *
@@ -608,7 +632,7 @@ export class ProtonDrivePhotosClient {
608
632
  albumNodeUid: NodeOrUid,
609
633
  photoNodeUids: NodeOrUid[],
610
634
  signal?: AbortSignal,
611
- ): AsyncGenerator<NodeResult> {
635
+ ): AsyncGenerator<NodeResultWithError> {
612
636
  this.logger.info(`Removing ${photoNodeUids.length} photos from album ${getUid(albumNodeUid)}`);
613
637
  yield* this.photos.albums.removePhotos(getUid(albumNodeUid), getUids(photoNodeUids), signal);
614
638
  }
@@ -160,11 +160,13 @@ export function convertInternalPhotoNode(photoNode: InternalPartialPhotoNode): P
160
160
  return resultOk({
161
161
  ...node.value,
162
162
  photo: photoNode.photo,
163
+ album: photoNode.album,
163
164
  } as PublicPhotoNode);
164
165
  }
165
166
  return resultError({
166
167
  ...node.error,
167
168
  photo: photoNode.photo,
169
+ album: photoNode.album,
168
170
  } as PublicDegradedPhotoNode);
169
171
  }
170
172