ente/mobile/lib/utils/diff_fetcher.dart

122 lines
4.5 KiB
Dart
Raw Permalink Normal View History

import 'dart:convert';
import 'dart:math';
import 'package:logging/logging.dart';
2023-02-03 07:39:04 +00:00
import 'package:photos/core/network/network.dart';
import 'package:photos/db/files_db.dart';
2023-08-25 04:39:30 +00:00
import 'package:photos/models/file/file.dart';
import "package:photos/models/metadata/file_magic.dart";
import 'package:photos/utils/crypto_util.dart';
import 'package:photos/utils/file_download_util.dart';
class DiffFetcher {
final _logger = Logger("DiffFetcher");
2023-02-03 07:39:04 +00:00
final _enteDio = NetworkClient.instance.enteDio;
2021-10-29 12:00:56 +00:00
Future<Diff> getEncryptedFilesDiff(int collectionID, int sinceTime) async {
2021-10-01 13:28:18 +00:00
try {
2022-10-14 09:45:00 +00:00
final response = await _enteDio.get(
"/collections/v2/diff",
2021-10-01 13:28:18 +00:00
queryParameters: {
"collectionID": collectionID,
"sinceTime": sinceTime,
},
);
int latestUpdatedAtTime = 0;
2022-12-30 15:42:03 +00:00
final diff = response.data["diff"] as List;
final bool hasMore = response.data["hasMore"] as bool;
final startTime = DateTime.now();
2023-08-08 15:41:58 +00:00
late Set<int> existingUploadIDs;
2024-04-20 06:01:09 +00:00
if (diff.isNotEmpty) {
existingUploadIDs =
await FilesDB.instance.getUploadedFileIDs(collectionID);
2023-08-08 15:41:58 +00:00
}
2023-08-24 16:56:24 +00:00
final deletedFiles = <EnteFile>[];
final updatedFiles = <EnteFile>[];
2022-12-30 15:42:03 +00:00
for (final item in diff) {
2023-08-24 16:56:24 +00:00
final file = EnteFile();
2022-12-30 15:42:03 +00:00
file.uploadedFileID = item["id"];
file.collectionID = item["collectionID"];
file.updationTime = item["updationTime"];
latestUpdatedAtTime = max(latestUpdatedAtTime, file.updationTime!);
if (item["isDeleted"]) {
if (existingUploadIDs.contains(file.uploadedFileID)) {
2022-12-30 15:42:03 +00:00
deletedFiles.add(file);
}
2022-12-30 15:42:03 +00:00
continue;
}
if (existingUploadIDs.contains(file.uploadedFileID)) {
2022-12-30 15:42:03 +00:00
final existingFile = await FilesDB.instance
.getUploadedFile(file.uploadedFileID!, file.collectionID!);
if (existingFile != null) {
file.generatedID = existingFile.generatedID;
file.addedTime = existingFile.addedTime;
2022-09-24 10:36:05 +00:00
}
2022-12-30 15:42:03 +00:00
}
file.ownerID = item["ownerID"];
file.encryptedKey = item["encryptedKey"];
file.keyDecryptionNonce = item["keyDecryptionNonce"];
file.fileDecryptionHeader = item["file"]["decryptionHeader"];
file.thumbnailDecryptionHeader = item["thumbnail"]["decryptionHeader"];
file.metadataDecryptionHeader = item["metadata"]["decryptionHeader"];
if (item["info"] != null) {
file.fileSize = item["info"]["fileSize"];
}
2023-02-27 10:44:27 +00:00
final fileKey = getFileKey(file);
2022-12-30 15:42:03 +00:00
final encodedMetadata = await CryptoUtil.decryptChaCha(
2023-02-03 04:41:45 +00:00
CryptoUtil.base642bin(item["metadata"]["encryptedData"]),
2023-02-27 10:44:27 +00:00
fileKey,
2023-02-03 04:41:45 +00:00
CryptoUtil.base642bin(file.metadataDecryptionHeader!),
2022-12-30 15:42:03 +00:00
);
final Map<String, dynamic> metadata =
jsonDecode(utf8.decode(encodedMetadata));
file.applyMetadata(metadata);
if (item['magicMetadata'] != null) {
final utfEncodedMmd = await CryptoUtil.decryptChaCha(
2023-02-03 04:41:45 +00:00
CryptoUtil.base642bin(item['magicMetadata']['data']),
2023-02-27 10:44:27 +00:00
fileKey,
2023-02-03 04:41:45 +00:00
CryptoUtil.base642bin(item['magicMetadata']['header']),
);
2022-12-30 15:42:03 +00:00
file.mMdEncodedJson = utf8.decode(utfEncodedMmd);
file.mMdVersion = item['magicMetadata']['version'];
file.magicMetadata =
MagicMetadata.fromEncodedJson(file.mMdEncodedJson!);
}
2022-12-30 15:42:03 +00:00
if (item['pubMagicMetadata'] != null) {
final utfEncodedMmd = await CryptoUtil.decryptChaCha(
2023-02-03 04:41:45 +00:00
CryptoUtil.base642bin(item['pubMagicMetadata']['data']),
2023-02-27 10:44:27 +00:00
fileKey,
2023-02-03 04:41:45 +00:00
CryptoUtil.base642bin(item['pubMagicMetadata']['header']),
2022-12-30 15:42:03 +00:00
);
file.pubMmdEncodedJson = utf8.decode(utfEncodedMmd);
file.pubMmdVersion = item['pubMagicMetadata']['version'];
file.pubMagicMetadata =
PubMagicMetadata.fromEncodedJson(file.pubMmdEncodedJson!);
}
updatedFiles.add(file);
}
2023-08-08 15:35:23 +00:00
_logger.info('[Collection-$collectionID] parsed ${diff.length} '
2024-04-20 06:01:09 +00:00
'diff items ( ${updatedFiles.length} updated) in ${DateTime.now().difference(startTime).inMilliseconds}ms');
return Diff(updatedFiles, deletedFiles, hasMore, latestUpdatedAtTime);
2021-10-01 13:28:18 +00:00
} catch (e, s) {
_logger.severe(e, s);
rethrow;
}
}
}
class Diff {
2023-08-24 16:56:24 +00:00
final List<EnteFile> updatedFiles;
final List<EnteFile> deletedFiles;
2021-10-29 12:00:56 +00:00
final bool hasMore;
final int latestUpdatedAtTime;
2022-06-11 08:23:52 +00:00
Diff(
this.updatedFiles,
this.deletedFiles,
this.hasMore,
this.latestUpdatedAtTime,
);
}