ente/lib/utils/file_util.dart

349 lines
11 KiB
Dart
Raw Normal View History

import 'dart:async';
import 'dart:io' as io;
2021-09-28 09:24:41 +00:00
import 'dart:io';
import 'dart:typed_data';
import 'package:archive/archive.dart';
import 'package:dio/dio.dart';
2022-12-29 13:37:08 +00:00
import 'package:flutter/foundation.dart';
import 'package:flutter_cache_manager/flutter_cache_manager.dart';
import 'package:flutter_image_compress/flutter_image_compress.dart';
2021-07-21 20:47:43 +00:00
import 'package:logging/logging.dart';
import 'package:motionphoto/motionphoto.dart';
2021-08-03 11:58:17 +00:00
import 'package:path/path.dart';
2021-08-09 06:21:09 +00:00
import 'package:photos/core/cache/image_cache.dart';
import 'package:photos/core/cache/thumbnail_cache.dart';
2020-08-13 20:03:29 +00:00
import 'package:photos/core/cache/video_cache_manager.dart';
import 'package:photos/core/configuration.dart';
2020-07-29 19:07:23 +00:00
import 'package:photos/core/constants.dart';
2021-05-29 21:58:42 +00:00
import 'package:photos/models/file.dart' as ente;
2020-08-13 01:24:48 +00:00
import 'package:photos/models/file_type.dart';
import 'package:photos/utils/file_download_util.dart';
2021-08-09 06:21:09 +00:00
import 'package:photos/utils/thumbnail_util.dart';
final _logger = Logger("FileUtil");
2020-09-28 16:16:04 +00:00
2021-05-29 21:58:42 +00:00
void preloadFile(ente.File file) {
2020-08-13 01:24:48 +00:00
if (file.fileType == FileType.video) {
return;
}
2021-06-11 07:19:19 +00:00
getFile(file);
2021-06-02 13:54:31 +00:00
}
2021-08-24 07:14:33 +00:00
// IMPORTANT: Delete the returned file if `isOrigin` is set to true
// https://github.com/CaiJingLong/flutter_photo_manager#cache-problem-of-ios
2022-12-29 13:37:08 +00:00
Future<io.File?> getFile(
ente.File file, {
bool liveVideo = false,
bool isOrigin = false,
} // only relevant for live photos
2021-08-04 10:11:38 +00:00
) async {
2022-09-23 01:48:25 +00:00
if (file.isRemoteFile) {
2021-08-04 10:11:38 +00:00
return getFileFromServer(file, liveVideo: liveVideo);
} else {
2022-09-23 01:48:25 +00:00
final String key = file.tag + liveVideo.toString() + isOrigin.toString();
2021-08-04 10:11:38 +00:00
final cachedFile = FileLruCache.get(key);
2021-06-02 13:54:31 +00:00
if (cachedFile == null) {
final diskFile = await _getLocalDiskFile(
file,
liveVideo: liveVideo,
isOrigin: isOrigin,
);
2021-09-28 09:24:41 +00:00
// do not cache origin file for IOS as they are immediately deleted
// after usage
2023-01-06 04:52:19 +00:00
if (!(isOrigin && Platform.isIOS) && diskFile != null) {
FileLruCache.put(key, diskFile);
2021-09-28 09:24:41 +00:00
}
2021-06-02 13:54:31 +00:00
return diskFile;
}
2021-06-02 13:54:31 +00:00
return cachedFile;
}
}
Future<bool> doesLocalFileExist(ente.File file) async {
return await _getLocalDiskFile(file) != null;
}
2022-12-29 13:37:08 +00:00
Future<io.File?> _getLocalDiskFile(
ente.File file, {
bool liveVideo = false,
bool isOrigin = false,
}) async {
2022-09-23 01:48:25 +00:00
if (file.isSharedMediaToAppSandbox) {
2022-08-29 14:43:31 +00:00
final localFile = io.File(getSharedMediaFilePath(file));
2021-07-24 13:38:47 +00:00
return localFile.exists().then((exist) {
return exist ? localFile : null;
});
2021-08-04 10:11:38 +00:00
} else if (file.fileType == FileType.livePhoto && liveVideo) {
2022-12-29 13:37:08 +00:00
return Motionphoto.getLivePhotoFile(file.localID!);
2021-08-09 06:21:09 +00:00
} else {
2022-09-23 01:48:25 +00:00
return file.getAsset.then((asset) async {
if (asset == null || !(await asset.exists)) {
return null;
}
return isOrigin ? asset.originFile : asset.file;
});
}
}
String getSharedMediaFilePath(ente.File file) {
2022-12-29 13:37:08 +00:00
return getSharedMediaPathFromLocalID(file.localID!);
2022-07-05 20:05:51 +00:00
}
String getSharedMediaPathFromLocalID(String localID) {
if (localID.startsWith(oldSharedMediaIdentifier)) {
2022-07-05 20:05:51 +00:00
return Configuration.instance.getOldSharedMediaCacheDirectory() +
"/" +
localID.replaceAll(oldSharedMediaIdentifier, '');
2022-07-05 20:05:51 +00:00
} else {
return Configuration.instance.getSharedMediaDirectory() +
"/" +
localID.replaceAll(sharedMediaIdentifier, '');
2022-07-05 20:05:51 +00:00
}
}
2021-05-29 21:58:42 +00:00
void preloadThumbnail(ente.File file) {
2022-09-23 01:48:25 +00:00
if (file.isRemoteFile) {
2021-04-27 16:29:58 +00:00
getThumbnailFromServer(file);
} else {
getThumbnailFromLocal(file);
2021-04-27 16:29:58 +00:00
}
2020-07-29 19:07:23 +00:00
}
2020-08-13 01:24:48 +00:00
2022-12-29 13:37:08 +00:00
final Map<String, Future<io.File?>> fileDownloadsInProgress =
2022-12-31 16:24:32 +00:00
<String, Future<io.File?>>{};
2022-12-29 13:37:08 +00:00
Future<io.File?> getFileFromServer(
ente.File file, {
2022-12-29 13:37:08 +00:00
ProgressCallback? progressCallback,
bool liveVideo = false, // only needed in case of live photos
}) async {
final cacheManager = (file.fileType == FileType.video || liveVideo)
2021-05-02 16:05:36 +00:00
? VideoCacheManager.instance
2020-08-13 20:03:29 +00:00
: DefaultCacheManager();
2022-09-23 01:48:25 +00:00
final fileFromCache = await cacheManager.getFileFromCache(file.downloadUrl);
if (fileFromCache != null) {
return fileFromCache.file;
}
final downloadID = file.uploadedFileID.toString() + liveVideo.toString();
if (!fileDownloadsInProgress.containsKey(downloadID)) {
if (file.fileType == FileType.livePhoto) {
2022-06-11 08:23:52 +00:00
fileDownloadsInProgress[downloadID] = _getLivePhotoFromServer(
file,
progressCallback: progressCallback,
needLiveVideo: liveVideo,
);
fileDownloadsInProgress[downloadID]!.whenComplete(() {
fileDownloadsInProgress.remove(downloadID);
});
} else {
fileDownloadsInProgress[downloadID] = _downloadAndCache(
2022-06-11 08:23:52 +00:00
file,
cacheManager,
progressCallback: progressCallback,
);
fileDownloadsInProgress[downloadID]!.whenComplete(() {
fileDownloadsInProgress.remove(downloadID);
});
}
}
return fileDownloadsInProgress[downloadID];
}
2022-05-11 04:51:25 +00:00
Future<bool> isFileCached(ente.File file, {bool liveVideo = false}) async {
final cacheManager = (file.fileType == FileType.video || liveVideo)
? VideoCacheManager.instance
: DefaultCacheManager();
2022-09-23 01:48:25 +00:00
final fileInfo = await cacheManager.getFileFromCache(file.downloadUrl);
return fileInfo != null;
}
2022-12-29 13:37:08 +00:00
final Map<int, Future<_LivePhoto?>> _livePhotoDownloadsTracker =
<int, Future<_LivePhoto?>>{};
2022-12-29 13:37:08 +00:00
Future<io.File?> _getLivePhotoFromServer(
2022-06-11 08:23:52 +00:00
ente.File file, {
2022-12-29 13:37:08 +00:00
ProgressCallback? progressCallback,
required bool needLiveVideo,
2022-06-11 08:23:52 +00:00
}) async {
2022-12-29 13:37:08 +00:00
final downloadID = file.uploadedFileID!;
try {
2022-07-03 09:45:00 +00:00
if (!_livePhotoDownloadsTracker.containsKey(downloadID)) {
_livePhotoDownloadsTracker[downloadID] =
_downloadLivePhoto(file, progressCallback: progressCallback);
}
2022-07-03 09:45:00 +00:00
final livePhoto = await _livePhotoDownloadsTracker[file.uploadedFileID];
_livePhotoDownloadsTracker.remove(downloadID);
2021-10-05 04:57:46 +00:00
if (livePhoto == null) {
return null;
}
2021-10-05 04:57:46 +00:00
return needLiveVideo ? livePhoto.video : livePhoto.image;
2022-05-11 04:51:25 +00:00
} catch (e, s) {
_logger.warning("live photo get failed", e, s);
2022-07-03 09:45:00 +00:00
_livePhotoDownloadsTracker.remove(downloadID);
return null;
}
2020-08-13 01:24:48 +00:00
}
2022-12-29 13:37:08 +00:00
Future<_LivePhoto?> _downloadLivePhoto(
2022-06-11 08:23:52 +00:00
ente.File file, {
2022-12-29 13:37:08 +00:00
ProgressCallback? progressCallback,
2022-06-11 08:23:52 +00:00
}) async {
return downloadAndDecrypt(file, progressCallback: progressCallback)
.then((decryptedFile) async {
if (decryptedFile == null) {
return null;
}
_logger.fine("Decoded zipped live photo from " + decryptedFile.path);
2022-12-29 13:37:08 +00:00
io.File? imageFileCache, videoFileCache;
2022-08-29 14:43:31 +00:00
final List<int> bytes = await decryptedFile.readAsBytes();
final Archive archive = ZipDecoder().decodeBytes(bytes);
final tempPath = Configuration.instance.getTempDirectory();
// Extract the contents of Zip compressed archive to disk
for (ArchiveFile archiveFile in archive) {
if (archiveFile.isFile) {
2022-08-29 14:43:31 +00:00
final String filename = archiveFile.name;
final String fileExtension = getExtension(archiveFile.name);
final String decodePath =
tempPath + file.uploadedFileID.toString() + filename;
2022-08-29 14:43:31 +00:00
final List<int> data = archiveFile.content;
if (filename.startsWith("image")) {
2021-08-09 06:26:11 +00:00
final imageFile = io.File(decodePath);
await imageFile.create(recursive: true);
await imageFile.writeAsBytes(data);
io.File imageConvertedFile = imageFile;
if ((fileExtension == "unknown") ||
(io.Platform.isAndroid && fileExtension == "heic")) {
2022-12-29 13:37:08 +00:00
final compressResult =
await FlutterImageCompress.compressAndGetFile(
decodePath,
decodePath + ".jpg",
keepExif: true,
);
2021-08-09 06:21:09 +00:00
await imageFile.delete();
2022-12-29 13:37:08 +00:00
if (compressResult == null) {
throw Exception("Failed to compress file");
} else {
imageConvertedFile = compressResult;
}
}
imageFileCache = await DefaultCacheManager().putFile(
2022-09-23 01:48:25 +00:00
file.downloadUrl,
2021-08-09 06:26:11 +00:00
await imageConvertedFile.readAsBytes(),
2022-09-23 01:48:25 +00:00
eTag: file.downloadUrl,
2022-07-04 06:02:17 +00:00
maxAge: const Duration(days: 365),
fileExtension: fileExtension,
);
2021-08-09 06:21:09 +00:00
await imageConvertedFile.delete();
} else if (filename.startsWith("video")) {
2021-08-09 06:26:11 +00:00
final videoFile = io.File(decodePath);
await videoFile.create(recursive: true);
await videoFile.writeAsBytes(data);
videoFileCache = await VideoCacheManager.instance.putFile(
2022-09-23 01:48:25 +00:00
file.downloadUrl,
2021-08-09 06:30:30 +00:00
await videoFile.readAsBytes(),
2022-09-23 01:48:25 +00:00
eTag: file.downloadUrl,
2022-07-04 06:02:17 +00:00
maxAge: const Duration(days: 365),
fileExtension: fileExtension,
);
2021-08-09 06:21:09 +00:00
await videoFile.delete();
}
}
}
2022-12-29 13:37:08 +00:00
if (imageFileCache != null && videoFileCache != null) {
return _LivePhoto(imageFileCache, videoFileCache);
} else {
debugPrint("Warning: Either image or video is missing from remoteLive");
return null;
}
}).catchError((e) {
2022-09-23 01:48:25 +00:00
_logger.warning("failed to download live photos : ${file.tag}", e);
throw e;
});
}
2022-12-29 13:37:08 +00:00
Future<io.File?> _downloadAndCache(
2022-06-11 08:23:52 +00:00
ente.File file,
BaseCacheManager cacheManager, {
2022-12-29 13:37:08 +00:00
ProgressCallback? progressCallback,
2022-06-11 08:23:52 +00:00
}) async {
return downloadAndDecrypt(file, progressCallback: progressCallback)
.then((decryptedFile) async {
if (decryptedFile == null) {
return null;
2020-09-28 16:15:21 +00:00
}
2022-08-29 14:43:31 +00:00
final decryptedFilePath = decryptedFile.path;
2022-12-29 13:37:08 +00:00
final String fileExtension = getExtension(file.title ?? '');
io.File outputFile = decryptedFile;
if ((fileExtension == "unknown" && file.fileType == FileType.image) ||
2021-06-29 04:10:24 +00:00
(io.Platform.isAndroid && fileExtension == "heic")) {
2022-12-29 13:37:08 +00:00
final compressResult = await FlutterImageCompress.compressAndGetFile(
2020-11-20 11:05:17 +00:00
decryptedFilePath,
decryptedFilePath + ".jpg",
keepExif: true,
);
2022-12-29 13:37:08 +00:00
if (compressResult == null) {
throw Exception("Failed to convert heic to jpg");
} else {
outputFile = compressResult;
}
2021-08-09 06:21:09 +00:00
await decryptedFile.delete();
2020-11-20 11:05:17 +00:00
}
2020-09-25 20:25:03 +00:00
final cachedFile = await cacheManager.putFile(
2022-09-23 01:48:25 +00:00
file.downloadUrl,
2021-08-09 06:30:30 +00:00
await outputFile.readAsBytes(),
2022-09-23 01:48:25 +00:00
eTag: file.downloadUrl,
2022-07-04 06:02:17 +00:00
maxAge: const Duration(days: 365),
fileExtension: fileExtension,
);
2021-08-09 06:21:09 +00:00
await outputFile.delete();
2020-09-25 20:25:03 +00:00
return cachedFile;
2020-10-06 23:56:37 +00:00
}).catchError((e) {
2022-09-23 01:48:25 +00:00
_logger.warning("failed to download file : ${file.tag}", e);
throw e;
2020-08-13 01:24:48 +00:00
});
}
2020-08-13 21:33:31 +00:00
String getExtension(String nameOrPath) {
2021-08-09 06:21:09 +00:00
var fileExtension = "unknown";
try {
fileExtension = extension(nameOrPath).substring(1).toLowerCase();
} catch (e) {
_logger.severe("Could not capture file extension");
}
return fileExtension;
}
Future<Uint8List> compressThumbnail(Uint8List thumbnail) {
return FlutterImageCompress.compressWithList(
thumbnail,
minHeight: compressedThumbnailResolution,
minWidth: compressedThumbnailResolution,
quality: 25,
);
}
2021-08-09 06:21:09 +00:00
Future<void> clearCache(ente.File file) async {
if (file.fileType == FileType.video) {
2022-09-23 01:48:25 +00:00
VideoCacheManager.instance.removeFile(file.downloadUrl);
} else {
2022-09-23 01:48:25 +00:00
DefaultCacheManager().removeFile(file.downloadUrl);
}
final cachedThumbnail = io.File(
2022-06-11 08:23:52 +00:00
Configuration.instance.getThumbnailCacheDirectory() +
"/" +
file.uploadedFileID.toString(),
);
if (cachedThumbnail.existsSync()) {
2021-08-09 06:21:09 +00:00
await cachedThumbnail.delete();
}
ThumbnailLruCache.clearCache(file);
}
class _LivePhoto {
final io.File image;
final io.File video;
_LivePhoto(this.image, this.video);
}