import 'dart:async'; import 'dart:io' as io; import 'dart:typed_data'; import 'package:dio/dio.dart'; import 'package:flutter_cache_manager/flutter_cache_manager.dart'; import 'package:flutter_image_compress/flutter_image_compress.dart'; import 'package:flutter_sodium/flutter_sodium.dart'; import 'package:logging/logging.dart'; import 'package:path/path.dart'; import 'package:photos/core/cache/image_cache.dart'; import 'package:photos/core/cache/video_cache_manager.dart'; import 'package:photos/core/configuration.dart'; import 'package:photos/core/constants.dart'; import 'package:photos/core/network.dart'; import 'package:photos/models/file.dart' as ente; import 'package:photos/models/file_type.dart'; import 'package:photos/services/collections_service.dart'; import 'package:photos/utils/thumbnail_util.dart'; import 'crypto_util.dart'; import 'file_uploader_util.dart'; final _logger = Logger("FileUtil"); void preloadFile(ente.File file) { if (file.fileType == FileType.video) { return; } getFile(file); } Future getFile(ente.File file) async { if (file.isRemoteFile()) { return getFileFromServer(file); } else { final cachedFile = FileLruCache.get(file); if (cachedFile == null) { final diskFile = await _getLocalDiskFile(file); FileLruCache.put(file, diskFile); return diskFile; } return cachedFile; } } Future _getLocalDiskFile(ente.File file) async { if (file.isSharedMediaToAppSandbox()) { var localFile = io.File(getSharedMediaFilePath(file)); return localFile.exists().then((exist) { return exist ? localFile : null; }); } else { return file.getAsset().then((asset) async { if (asset == null || !(await asset.exists)) { return null; } return asset.file; }); } } String getSharedMediaFilePath(ente.File file) { return Configuration.instance.getSharedMediaCacheDirectory() + "/" + file.localID.replaceAll(kSharedMediaIdentifier, ''); } void preloadThumbnail(ente.File file) { if (file.isRemoteFile()) { getThumbnailFromServer(file); } else { getThumbnailFromLocal(file); } } final Map> fileDownloadsInProgress = Map>(); Future getFileFromServer(ente.File file, {ProgressCallback progressCallback}) async { final cacheManager = file.fileType == FileType.video ? VideoCacheManager.instance : DefaultCacheManager(); return cacheManager.getFileFromCache(file.getDownloadUrl()).then((info) { if (info == null) { if (!fileDownloadsInProgress.containsKey(file.uploadedFileID)) { fileDownloadsInProgress[file.uploadedFileID] = _downloadAndDecrypt( file, cacheManager, progressCallback: progressCallback, ); } return fileDownloadsInProgress[file.uploadedFileID]; } else { return info.file; } }); } Future _downloadAndDecrypt( ente.File file, BaseCacheManager cacheManager, {ProgressCallback progressCallback}) async { _logger.info("Downloading file " + file.uploadedFileID.toString()); final encryptedFilePath = Configuration.instance.getTempDirectory() + file.generatedID.toString() + ".encrypted"; final decryptedFilePath = Configuration.instance.getTempDirectory() + file.generatedID.toString() + ".decrypted"; final encryptedFile = io.File(encryptedFilePath); final decryptedFile = io.File(decryptedFilePath); final startTime = DateTime.now().millisecondsSinceEpoch; return Network.instance .getDio() .download( file.getDownloadUrl(), encryptedFilePath, options: Options( headers: {"X-Auth-Token": Configuration.instance.getToken()}, ), onReceiveProgress: progressCallback, ) .then((response) async { if (response.statusCode != 200) { _logger.warning("Could not download file: ", response.toString()); return null; } else if (!encryptedFile.existsSync()) { _logger.warning("File was not downloaded correctly."); return null; } _logger.info("File downloaded: " + file.uploadedFileID.toString()); _logger.info("Download speed: " + (io.File(encryptedFilePath).lengthSync() / (DateTime.now().millisecondsSinceEpoch - startTime)) .toString() + "kBps"); await CryptoUtil.decryptFile(encryptedFilePath, decryptedFilePath, Sodium.base642bin(file.fileDecryptionHeader), decryptFileKey(file)); _logger.info("File decrypted: " + file.uploadedFileID.toString()); encryptedFile.deleteSync(); var fileExtension = "unknown"; try { fileExtension = extension(file.title).substring(1).toLowerCase(); } catch (e) { _logger.severe("Could not capture file extension"); } var outputFile = decryptedFile; if ((fileExtension == "unknown" && file.fileType == FileType.image) || (io.Platform.isAndroid && fileExtension == "heic")) { outputFile = await FlutterImageCompress.compressAndGetFile( decryptedFilePath, decryptedFilePath + ".jpg", keepExif: true, ); decryptedFile.deleteSync(); } final cachedFile = await cacheManager.putFile( file.getDownloadUrl(), outputFile.readAsBytesSync(), eTag: file.getDownloadUrl(), maxAge: Duration(days: 365), fileExtension: fileExtension, ); outputFile.deleteSync(); fileDownloadsInProgress.remove(file.uploadedFileID); return cachedFile; }).catchError((e) { fileDownloadsInProgress.remove(file.uploadedFileID); }); } Uint8List decryptFileKey(ente.File file) { final encryptedKey = Sodium.base642bin(file.encryptedKey); final nonce = Sodium.base642bin(file.keyDecryptionNonce); final collectionKey = CollectionsService.instance.getCollectionKey(file.collectionID); return CryptoUtil.decryptSync(encryptedKey, collectionKey, nonce); } Future compressThumbnail(Uint8List thumbnail) { return FlutterImageCompress.compressWithList( thumbnail, minHeight: kCompressedThumbnailResolution, minWidth: kCompressedThumbnailResolution, quality: 25, ); } void clearCache(ente.File file) { if (file.fileType == FileType.video) { VideoCacheManager.instance.removeFile(file.getDownloadUrl()); } else { DefaultCacheManager().removeFile(file.getDownloadUrl()); } final cachedThumbnail = io.File( Configuration.instance.getThumbnailCacheDirectory() + "/" + file.uploadedFileID.toString()); if (cachedThumbnail.existsSync()) { cachedThumbnail.deleteSync(); } }