ente/lib/utils/trash_diff_fetcher.dart

132 lines
4.7 KiB
Dart
Raw Normal View History

2021-10-12 14:48:35 +00:00
import 'dart:convert';
2021-10-12 19:27:11 +00:00
import 'dart:math';
2021-10-12 14:48:35 +00:00
import 'package:flutter_sodium/flutter_sodium.dart';
import 'package:logging/logging.dart';
import 'package:photos/core/network.dart';
import 'package:photos/models/magic_metadata.dart';
2021-10-12 14:48:35 +00:00
import 'package:photos/models/trash_file.dart';
import 'package:photos/utils/crypto_util.dart';
import 'package:photos/utils/file_download_util.dart';
class TrashDiffFetcher {
final _logger = Logger("TrashDiffFetcher");
2022-10-14 09:45:00 +00:00
final _enteDio = Network.instance.enteDio;
2021-10-12 14:48:35 +00:00
2021-10-24 02:49:34 +00:00
Future<Diff> getTrashFilesDiff(int sinceTime) async {
2021-10-12 14:48:35 +00:00
try {
2022-10-14 09:45:00 +00:00
final response = await _enteDio.get(
"/trash/v2/diff",
2021-10-12 14:48:35 +00:00
queryParameters: {
"sinceTime": sinceTime,
},
);
2021-10-12 19:27:11 +00:00
int latestUpdatedAtTime = 0;
2021-10-12 21:29:05 +00:00
final trashedFiles = <TrashFile>[];
2022-05-08 06:13:43 +00:00
final deletedUploadIDs = <int>[];
2021-10-12 21:29:05 +00:00
final restoredFiles = <TrashFile>[];
2021-10-12 14:48:35 +00:00
if (response != null) {
final diff = response.data["diff"] as List;
2021-10-24 02:49:34 +00:00
final bool hasMore = response.data["hasMore"] as bool;
2021-10-12 14:48:35 +00:00
final startTime = DateTime.now();
for (final item in diff) {
2022-05-29 09:51:25 +00:00
final trash = TrashFile();
2022-05-29 09:49:00 +00:00
trash.createdAt = item['createdAt'];
trash.updateAt = item['updatedAt'];
latestUpdatedAtTime = max(latestUpdatedAtTime, trash.updateAt);
2022-05-05 18:38:58 +00:00
if (item["isDeleted"]) {
2022-05-08 06:13:43 +00:00
deletedUploadIDs.add(item["file"]["id"]);
2022-05-05 18:38:58 +00:00
continue;
}
2022-05-29 09:49:00 +00:00
2021-10-12 14:48:35 +00:00
trash.deleteBy = item['deleteBy'];
2021-10-12 21:29:05 +00:00
trash.uploadedFileID = item["file"]["id"];
trash.collectionID = item["file"]["collectionID"];
trash.updationTime = item["file"]["updationTime"];
trash.ownerID = item["file"]["ownerID"];
trash.encryptedKey = item["file"]["encryptedKey"];
trash.keyDecryptionNonce = item["file"]["keyDecryptionNonce"];
trash.fileDecryptionHeader = item["file"]["file"]["decryptionHeader"];
trash.thumbnailDecryptionHeader =
2021-10-12 14:48:35 +00:00
item["file"]["thumbnail"]["decryptionHeader"];
2021-10-12 21:29:05 +00:00
trash.metadataDecryptionHeader =
2021-10-12 14:48:35 +00:00
item["file"]["metadata"]["decryptionHeader"];
2021-10-12 21:29:05 +00:00
final fileDecryptionKey = decryptFileKey(trash);
2021-10-12 14:48:35 +00:00
final encodedMetadata = await CryptoUtil.decryptChaCha(
Sodium.base642bin(item["file"]["metadata"]["encryptedData"]),
fileDecryptionKey,
2022-12-29 13:40:36 +00:00
Sodium.base642bin(trash.metadataDecryptionHeader!),
2021-10-12 14:48:35 +00:00
);
2022-08-29 14:43:31 +00:00
final Map<String, dynamic> metadata =
2021-10-12 14:48:35 +00:00
jsonDecode(utf8.decode(encodedMetadata));
2021-10-12 21:29:05 +00:00
trash.applyMetadata(metadata);
2021-10-12 14:48:35 +00:00
if (item["file"]['magicMetadata'] != null) {
final utfEncodedMmd = await CryptoUtil.decryptChaCha(
2022-06-11 08:23:52 +00:00
Sodium.base642bin(item["file"]['magicMetadata']['data']),
fileDecryptionKey,
Sodium.base642bin(item["file"]['magicMetadata']['header']),
);
2021-10-12 21:29:05 +00:00
trash.mMdEncodedJson = utf8.decode(utfEncodedMmd);
trash.mMdVersion = item["file"]['magicMetadata']['version'];
2021-10-12 14:48:35 +00:00
}
if (item["file"]['pubMagicMetadata'] != null) {
final utfEncodedMmd = await CryptoUtil.decryptChaCha(
2022-06-11 08:23:52 +00:00
Sodium.base642bin(item["file"]['pubMagicMetadata']['data']),
fileDecryptionKey,
Sodium.base642bin(item["file"]['pubMagicMetadata']['header']),
);
trash.pubMmdEncodedJson = utf8.decode(utfEncodedMmd);
trash.pubMmdVersion = item["file"]['pubMagicMetadata']['version'];
trash.pubMagicMetadata =
2022-12-29 13:40:36 +00:00
PubMagicMetadata.fromEncodedJson(trash.pubMmdEncodedJson!);
}
2021-10-12 14:48:35 +00:00
if (item['isRestored']) {
restoredFiles.add(trash);
continue;
}
trashedFiles.add(trash);
}
final endTime = DateTime.now();
2022-06-11 08:23:52 +00:00
_logger.info(
"time for parsing " +
diff.length.toString() +
": " +
Duration(
microseconds: (endTime.microsecondsSinceEpoch -
startTime.microsecondsSinceEpoch),
).inMilliseconds.toString(),
);
return Diff(
trashedFiles,
restoredFiles,
deletedUploadIDs,
hasMore,
latestUpdatedAtTime,
);
2021-10-12 14:48:35 +00:00
} else {
2022-05-05 18:38:58 +00:00
return Diff(<TrashFile>[], <TrashFile>[], <int>[], false, 0);
2021-10-12 14:48:35 +00:00
}
} catch (e, s) {
_logger.severe(e, s);
rethrow;
}
}
}
class Diff {
2021-10-12 21:29:05 +00:00
final List<TrashFile> trashedFiles;
final List<TrashFile> restoredFiles;
2022-05-08 06:13:43 +00:00
final List<int> deletedUploadIDs;
2021-10-24 02:49:34 +00:00
final bool hasMore;
2021-10-12 19:27:11 +00:00
final int lastSyncedTimeStamp;
2021-10-12 14:48:35 +00:00
2022-06-11 08:23:52 +00:00
Diff(
this.trashedFiles,
this.restoredFiles,
this.deletedUploadIDs,
this.hasMore,
this.lastSyncedTimeStamp,
);
2021-10-12 14:48:35 +00:00
}