diff_fetcher.dart 5.2 KB

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