diff_fetcher.dart 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130
  1. import 'dart:convert';
  2. import 'dart:math';
  3. import 'package:dio/dio.dart';
  4. import 'package:flutter_sodium/flutter_sodium.dart';
  5. import 'package:logging/logging.dart';
  6. import 'package:photos/core/configuration.dart';
  7. import 'package:photos/core/event_bus.dart';
  8. import 'package:photos/core/network.dart';
  9. import 'package:photos/db/files_db.dart';
  10. import 'package:photos/events/remote_sync_event.dart';
  11. import 'package:photos/models/file.dart';
  12. import 'package:photos/models/magic_metadata.dart';
  13. import 'package:photos/utils/crypto_util.dart';
  14. import 'package:photos/utils/file_download_util.dart';
  15. class DiffFetcher {
  16. final _logger = Logger("DiffFetcher");
  17. final _dio = Network.instance.getDio();
  18. Future<Diff> getEncryptedFilesDiff(int collectionID, int sinceTime) async {
  19. try {
  20. final response = await _dio.get(
  21. Configuration.instance.getHttpEndpoint() + "/collections/v2/diff",
  22. options: Options(
  23. headers: {"X-Auth-Token": Configuration.instance.getToken()}),
  24. queryParameters: {
  25. "collectionID": collectionID,
  26. "sinceTime": sinceTime,
  27. },
  28. );
  29. final files = <File>[];
  30. int latestUpdatedAtTime = 0;
  31. if (response != null) {
  32. Bus.instance.fire(RemoteSyncEvent(true));
  33. final diff = response.data["diff"] as List;
  34. final bool hasMore = response.data["hasMore"] as bool;
  35. final startTime = DateTime.now();
  36. final existingFiles =
  37. await FilesDB.instance.getUploadedFileIDs(collectionID);
  38. final deletedFiles = <File>[];
  39. for (final item in diff) {
  40. final file = File();
  41. file.uploadedFileID = item["id"];
  42. file.collectionID = item["collectionID"];
  43. file.updationTime = item["updationTime"];
  44. latestUpdatedAtTime = max(latestUpdatedAtTime, file.updationTime);
  45. if (item["isDeleted"]) {
  46. if (existingFiles.contains(file.uploadedFileID)) {
  47. deletedFiles.add(file);
  48. }
  49. continue;
  50. }
  51. if (existingFiles.contains(file.uploadedFileID)) {
  52. final existingFile = await FilesDB.instance
  53. .getUploadedFile(file.uploadedFileID, file.collectionID);
  54. if (existingFile != null) {
  55. file.generatedID = existingFile.generatedID;
  56. }
  57. }
  58. file.ownerID = item["ownerID"];
  59. file.encryptedKey = item["encryptedKey"];
  60. file.keyDecryptionNonce = item["keyDecryptionNonce"];
  61. file.fileDecryptionHeader = item["file"]["decryptionHeader"];
  62. file.thumbnailDecryptionHeader =
  63. item["thumbnail"]["decryptionHeader"];
  64. file.metadataDecryptionHeader = item["metadata"]["decryptionHeader"];
  65. final fileDecryptionKey = decryptFileKey(file);
  66. final encodedMetadata = await CryptoUtil.decryptChaCha(
  67. Sodium.base642bin(item["metadata"]["encryptedData"]),
  68. fileDecryptionKey,
  69. Sodium.base642bin(file.metadataDecryptionHeader),
  70. );
  71. Map<String, dynamic> metadata =
  72. jsonDecode(utf8.decode(encodedMetadata));
  73. file.applyMetadata(metadata);
  74. if (item['magicMetadata'] != null) {
  75. final utfEncodedMmd = await CryptoUtil.decryptChaCha(
  76. Sodium.base642bin(item['magicMetadata']['data']),
  77. fileDecryptionKey,
  78. Sodium.base642bin(item['magicMetadata']['header']));
  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. Sodium.base642bin(item['pubMagicMetadata']['data']),
  87. fileDecryptionKey,
  88. Sodium.base642bin(item['pubMagicMetadata']['header']));
  89. file.pubMmdEncodedJson = utf8.decode(utfEncodedMmd);
  90. file.pubMmdVersion = item['pubMagicMetadata']['version'];
  91. file.pubMagicMetadata =
  92. PubMagicMetadata.fromEncodedJson(file.pubMmdEncodedJson);
  93. }
  94. files.add(file);
  95. }
  96. final endTime = DateTime.now();
  97. _logger.info("time for parsing " +
  98. files.length.toString() +
  99. ": " +
  100. Duration(
  101. microseconds: (endTime.microsecondsSinceEpoch -
  102. startTime.microsecondsSinceEpoch))
  103. .inMilliseconds
  104. .toString());
  105. return Diff(files, deletedFiles, hasMore, latestUpdatedAtTime);
  106. } else {
  107. Bus.instance.fire(RemoteSyncEvent(false));
  108. return Diff(<File>[], <File>[], false, 0);
  109. }
  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(this.updatedFiles, this.deletedFiles, this.hasMore,
  122. this.latestUpdatedAtTime);
  123. }