file_downloader.dart 3.5 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091
  1. import 'dart:convert';
  2. import 'package:dio/dio.dart';
  3. import 'package:flutter_sodium/flutter_sodium.dart';
  4. import 'package:logging/logging.dart';
  5. import 'package:photos/core/configuration.dart';
  6. import 'package:photos/core/event_bus.dart';
  7. import 'package:photos/core/network.dart';
  8. import 'package:photos/db/files_db.dart';
  9. import 'package:photos/events/collection_updated_event.dart';
  10. import 'package:photos/events/remote_sync_event.dart';
  11. import 'package:photos/models/file.dart';
  12. import 'package:photos/repositories/file_repository.dart';
  13. import 'package:photos/utils/crypto_util.dart';
  14. import 'package:photos/utils/file_util.dart';
  15. class DiffFetcher {
  16. final _logger = Logger("FileDownloader");
  17. final _dio = Network.instance.getDio();
  18. Future<Diff> getEncryptedFilesDiff(
  19. int collectionID, int sinceTime, int limit) async {
  20. return _dio
  21. .get(
  22. Configuration.instance.getHttpEndpoint() + "/collections/diff",
  23. options: Options(
  24. headers: {"X-Auth-Token": Configuration.instance.getToken()}),
  25. queryParameters: {
  26. "collectionID": collectionID,
  27. "sinceTime": sinceTime,
  28. "limit": limit,
  29. },
  30. )
  31. .catchError((e) => _logger.severe(e))
  32. .then((response) async {
  33. final files = List<File>();
  34. if (response != null) {
  35. Bus.instance.fire(RemoteSyncEvent(true));
  36. final diff = response.data["diff"] as List;
  37. for (final item in diff) {
  38. final file = File();
  39. file.uploadedFileID = item["id"];
  40. file.collectionID = item["collectionID"];
  41. if (item["isDeleted"]) {
  42. await FilesDB.instance.deleteFromCollection(
  43. file.uploadedFileID, file.collectionID);
  44. Bus.instance.fire(
  45. CollectionUpdatedEvent(collectionID: file.collectionID));
  46. FileRepository.instance.reloadFiles();
  47. continue;
  48. }
  49. file.updationTime = item["updationTime"];
  50. final existingFile = await FilesDB.instance
  51. .getUploadedFile(file.uploadedFileID, file.collectionID);
  52. if (existingFile != null &&
  53. existingFile.updationTime == file.updationTime) {
  54. continue;
  55. }
  56. file.ownerID = item["ownerID"];
  57. file.encryptedKey = item["encryptedKey"];
  58. file.keyDecryptionNonce = item["keyDecryptionNonce"];
  59. file.fileDecryptionHeader = item["file"]["decryptionHeader"];
  60. file.thumbnailDecryptionHeader =
  61. item["thumbnail"]["decryptionHeader"];
  62. file.metadataDecryptionHeader =
  63. item["metadata"]["decryptionHeader"];
  64. final encodedMetadata = CryptoUtil.decryptChaCha(
  65. Sodium.base642bin(item["metadata"]["encryptedData"]),
  66. decryptFileKey(file),
  67. Sodium.base642bin(file.metadataDecryptionHeader),
  68. );
  69. Map<String, dynamic> metadata =
  70. jsonDecode(utf8.decode(encodedMetadata));
  71. file.applyMetadata(metadata);
  72. files.add(file);
  73. }
  74. return Diff(files, diff.length);
  75. } else {
  76. Bus.instance.fire(RemoteSyncEvent(false));
  77. return Diff(List<File>(), 0);
  78. }
  79. });
  80. }
  81. }
  82. class Diff {
  83. final List<File> updatedFiles;
  84. final int fetchCount;
  85. Diff(this.updatedFiles, this.fetchCount);
  86. }