diff_fetcher.dart 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134
  1. import 'dart:convert';
  2. import 'dart:math';
  3. import 'package:logging/logging.dart';
  4. import 'package:photos/core/network/network.dart';
  5. import 'package:photos/db/files_db.dart';
  6. import 'package:photos/models/file.dart';
  7. import "package:photos/models/metadata/file_magic.dart";
  8. import 'package:photos/utils/crypto_util.dart';
  9. import 'package:photos/utils/file_download_util.dart';
  10. class DiffFetcher {
  11. final _logger = Logger("DiffFetcher");
  12. final _enteDio = NetworkClient.instance.enteDio;
  13. Future<Diff> getEncryptedFilesDiff(int collectionID, int sinceTime) async {
  14. _logger.info(
  15. "Fetching diff in collection " +
  16. collectionID.toString() +
  17. " since " +
  18. sinceTime.toString(),
  19. );
  20. try {
  21. final response = await _enteDio.get(
  22. "/collections/v2/diff",
  23. queryParameters: {
  24. "collectionID": collectionID,
  25. "sinceTime": sinceTime,
  26. },
  27. );
  28. final files = <File>[];
  29. int latestUpdatedAtTime = 0;
  30. final diff = response.data["diff"] as List;
  31. final bool hasMore = response.data["hasMore"] as bool;
  32. final startTime = DateTime.now();
  33. final existingFiles =
  34. await FilesDB.instance.getUploadedFileIDs(collectionID);
  35. final deletedFiles = <File>[];
  36. for (final item in diff) {
  37. final file = File();
  38. file.uploadedFileID = item["id"];
  39. file.collectionID = item["collectionID"];
  40. file.updationTime = item["updationTime"];
  41. latestUpdatedAtTime = max(latestUpdatedAtTime, file.updationTime!);
  42. if (item["isDeleted"]) {
  43. if (existingFiles.contains(file.uploadedFileID)) {
  44. deletedFiles.add(file);
  45. }
  46. continue;
  47. }
  48. if (existingFiles.contains(file.uploadedFileID)) {
  49. final existingFile = await FilesDB.instance
  50. .getUploadedFile(file.uploadedFileID!, file.collectionID!);
  51. if (existingFile != null) {
  52. file.generatedID = existingFile.generatedID;
  53. }
  54. }
  55. file.ownerID = item["ownerID"];
  56. file.encryptedKey = item["encryptedKey"];
  57. file.keyDecryptionNonce = item["keyDecryptionNonce"];
  58. file.fileDecryptionHeader = item["file"]["decryptionHeader"];
  59. file.thumbnailDecryptionHeader = item["thumbnail"]["decryptionHeader"];
  60. file.metadataDecryptionHeader = item["metadata"]["decryptionHeader"];
  61. if (item["info"] != null) {
  62. file.fileSize = item["info"]["fileSize"];
  63. }
  64. final fileKey = getFileKey(file);
  65. final encodedMetadata = await CryptoUtil.decryptChaCha(
  66. CryptoUtil.base642bin(item["metadata"]["encryptedData"]),
  67. fileKey,
  68. CryptoUtil.base642bin(file.metadataDecryptionHeader!),
  69. );
  70. final Map<String, dynamic> metadata =
  71. jsonDecode(utf8.decode(encodedMetadata));
  72. file.applyMetadata(metadata);
  73. if (item['magicMetadata'] != null) {
  74. final utfEncodedMmd = await CryptoUtil.decryptChaCha(
  75. CryptoUtil.base642bin(item['magicMetadata']['data']),
  76. fileKey,
  77. CryptoUtil.base642bin(item['magicMetadata']['header']),
  78. );
  79. file.mMdEncodedJson = utf8.decode(utfEncodedMmd);
  80. file.mMdVersion = item['magicMetadata']['version'];
  81. file.magicMetadata =
  82. MagicMetadata.fromEncodedJson(file.mMdEncodedJson!);
  83. }
  84. if (item['pubMagicMetadata'] != null) {
  85. final utfEncodedMmd = await CryptoUtil.decryptChaCha(
  86. CryptoUtil.base642bin(item['pubMagicMetadata']['data']),
  87. fileKey,
  88. CryptoUtil.base642bin(item['pubMagicMetadata']['header']),
  89. );
  90. file.pubMmdEncodedJson = utf8.decode(utfEncodedMmd);
  91. file.pubMmdVersion = item['pubMagicMetadata']['version'];
  92. file.pubMagicMetadata =
  93. PubMagicMetadata.fromEncodedJson(file.pubMmdEncodedJson!);
  94. }
  95. files.add(file);
  96. }
  97. final endTime = DateTime.now();
  98. _logger.info(
  99. "time for parsing " +
  100. files.length.toString() +
  101. " items within collection " +
  102. collectionID.toString() +
  103. ": " +
  104. Duration(
  105. microseconds: (endTime.microsecondsSinceEpoch -
  106. startTime.microsecondsSinceEpoch),
  107. ).inMilliseconds.toString(),
  108. );
  109. return Diff(files, deletedFiles, hasMore, latestUpdatedAtTime);
  110. } catch (e, s) {
  111. _logger.severe(e, s);
  112. rethrow;
  113. }
  114. }
  115. }
  116. class Diff {
  117. final List<File> updatedFiles;
  118. final List<File> deletedFiles;
  119. final bool hasMore;
  120. final int latestUpdatedAtTime;
  121. Diff(
  122. this.updatedFiles,
  123. this.deletedFiles,
  124. this.hasMore,
  125. this.latestUpdatedAtTime,
  126. );
  127. }