import 'dart:async'; import 'dart:io'; import 'package:http/http.dart' as http; import 'package:logging/logging.dart'; import 'package:nc_photos/exception.dart'; import 'package:nc_photos/platform/download.dart' as itf; import 'package:nc_photos/platform/k.dart' as platform_k; import 'package:nc_photos_plugin/nc_photos_plugin.dart'; import 'package:path_provider/path_provider.dart'; import 'package:uuid/uuid.dart'; class DownloadBuilder extends itf.DownloadBuilder { @override build({ required String url, Map? headers, String? mimeType, required String filename, String? parentDir, bool? shouldNotify, }) { if (platform_k.isAndroid) { return _AndroidDownload( url: url, headers: headers, mimeType: mimeType, filename: filename, parentDir: parentDir, shouldNotify: shouldNotify, ); } else { throw UnimplementedError(); } } } class _AndroidDownload extends itf.Download { _AndroidDownload({ required this.url, this.headers, this.mimeType, required this.filename, this.parentDir, this.shouldNotify, }); @override call() async { if (_isInitialDownload) { await _cleanUp(); _isInitialDownload = false; } final file = await _createTempFile(); try { // download file to a temp dir final fileWrite = file.openWrite(); try { final uri = Uri.parse(url); final req = http.Request("GET", uri)..headers.addAll(headers ?? {}); final response = await http.Client().send(req); bool isEnd = false; Object? error; final subscription = response.stream.listen( fileWrite.add, onDone: () { isEnd = true; }, onError: (e, stackTrace) { _log.severe("Failed while request", e, stackTrace); isEnd = true; error = e; }, cancelOnError: true, ); // wait until download finished while (!isEnd) { if (shouldInterrupt) { await subscription.cancel(); break; } await Future.delayed(const Duration(seconds: 1)); } if (error != null) { throw error!; } } finally { fileWrite.close(); } if (shouldInterrupt) { throw JobCanceledException(); } // copy the file to the actual dir return await MediaStore.copyFileToDownload( file.path, filename: filename, subDir: parentDir, ); } finally { file.delete(); } } @override cancel() { shouldInterrupt = true; return true; } Future _openDownloadDir() async { final tempDir = await getTemporaryDirectory(); final downloadDir = Directory("${tempDir.path}/downloads"); if (!await downloadDir.exists()) { return downloadDir.create(); } else { return downloadDir; } } Future _createTempFile() async { final downloadDir = await _openDownloadDir(); while (true) { final fileName = const Uuid().v4(); final file = File("${downloadDir.path}/$fileName"); if (await file.exists()) { continue; } return file; } } /// Clean up remaining cache files from previous runs /// /// Normally the files will be deleted automatically Future _cleanUp() async { final downloadDir = await _openDownloadDir(); await for (final f in downloadDir.list(followLinks: false)) { _log.warning("[_cleanUp] Deleting file: ${f.path}"); await f.delete(); } } final String url; final Map? headers; final String? mimeType; final String filename; final String? parentDir; final bool? shouldNotify; bool shouldInterrupt = false; static bool _isInitialDownload = true; static final _log = Logger("mobile.download._AndroidDownload"); }