ente/lib/utils/diff_fetcher.dart

109 lines
4.2 KiB
Dart
Raw Normal View History

import 'dart:convert';
import 'package:dio/dio.dart';
import 'package:flutter_sodium/flutter_sodium.dart';
import 'package:logging/logging.dart';
import 'package:photos/core/configuration.dart';
import 'package:photos/core/event_bus.dart';
2020-11-19 18:22:30 +00:00
import 'package:photos/core/network.dart';
import 'package:photos/db/files_db.dart';
import 'package:photos/events/collection_updated_event.dart';
2021-04-21 13:09:18 +00:00
import 'package:photos/events/local_photos_updated_event.dart';
import 'package:photos/events/remote_sync_event.dart';
import 'package:photos/models/file.dart';
import 'package:photos/utils/crypto_util.dart';
import 'package:photos/utils/file_util.dart';
class DiffFetcher {
final _logger = Logger("DiffFetcher");
2020-11-19 18:22:30 +00:00
final _dio = Network.instance.getDio();
Future<Diff> getEncryptedFilesDiff(
2020-10-28 15:45:05 +00:00
int collectionID, int sinceTime, int limit) async {
return _dio
.get(
2020-10-28 15:45:05 +00:00
Configuration.instance.getHttpEndpoint() + "/collections/diff",
2020-10-12 18:25:33 +00:00
options: Options(
headers: {"X-Auth-Token": Configuration.instance.getToken()}),
queryParameters: {
2020-10-28 15:45:05 +00:00
"collectionID": collectionID,
"sinceTime": sinceTime,
"limit": limit,
},
)
.catchError((e) => _logger.severe(e))
.then((response) async {
final files = List<File>();
if (response != null) {
Bus.instance.fire(RemoteSyncEvent(true));
final diff = response.data["diff"] as List;
final startTime = DateTime.now();
final existingFiles =
await FilesDB.instance.getUploadedFileIDs(collectionID);
for (final item in diff) {
final file = File();
file.uploadedFileID = item["id"];
file.collectionID = item["collectionID"];
if (item["isDeleted"]) {
if (existingFiles.contains(file.uploadedFileID)) {
await FilesDB.instance.deleteFromCollection(
file.uploadedFileID, file.collectionID);
Bus.instance.fire(
CollectionUpdatedEvent(file.collectionID, [file]));
Bus.instance.fire(LocalPhotosUpdatedEvent([file]));
}
continue;
}
file.updationTime = item["updationTime"];
if (existingFiles.contains(file.uploadedFileID)) {
final existingFile = await FilesDB.instance
.getUploadedFile(file.uploadedFileID, file.collectionID);
if (existingFile != null &&
existingFile.updationTime == file.updationTime) {
continue;
}
}
file.ownerID = item["ownerID"];
file.encryptedKey = item["encryptedKey"];
file.keyDecryptionNonce = item["keyDecryptionNonce"];
2020-10-06 23:56:49 +00:00
file.fileDecryptionHeader = item["file"]["decryptionHeader"];
file.thumbnailDecryptionHeader =
2020-10-06 23:56:49 +00:00
item["thumbnail"]["decryptionHeader"];
file.metadataDecryptionHeader =
item["metadata"]["decryptionHeader"];
final encodedMetadata = await CryptoUtil.decryptChaCha(
Sodium.base642bin(item["metadata"]["encryptedData"]),
2020-10-10 23:43:47 +00:00
decryptFileKey(file),
Sodium.base642bin(file.metadataDecryptionHeader),
);
Map<String, dynamic> metadata =
jsonDecode(utf8.decode(encodedMetadata));
file.applyMetadata(metadata);
files.add(file);
}
final endTime = DateTime.now();
_logger.info("time for parsing " +
files.length.toString() +
": " +
Duration(
microseconds: (endTime.microsecondsSinceEpoch -
startTime.microsecondsSinceEpoch))
.inMilliseconds
.toString());
return Diff(files, diff.length);
} else {
Bus.instance.fire(RemoteSyncEvent(false));
return Diff(List<File>(), 0);
}
});
}
}
class Diff {
final List<File> updatedFiles;
final int fetchCount;
Diff(this.updatedFiles, this.fetchCount);
}