import 'package:logging/logging.dart'; import 'package:nc_photos/account.dart'; import 'package:nc_photos/entity/album.dart'; import 'package:nc_photos/entity/file.dart'; import 'package:nc_photos/exception.dart'; import 'package:nc_photos/remote_storage_util.dart' as remote_storage_util; import 'package:nc_photos/use_case/compat/v15.dart'; import 'package:nc_photos/use_case/compat/v25.dart'; import 'package:nc_photos/use_case/ls.dart'; import 'package:tuple/tuple.dart'; class ListAlbum { ListAlbum(this.fileRepo, this.albumRepo); /// List all albums associated with [account] /// /// The returned stream would emit either Album data or a tuple of exception /// and stacktrace Stream call(Account account) async* { bool hasAlbum = false; await for (final result in _call(account)) { hasAlbum = true; yield result; } if (!hasAlbum) { if (await CompatV15.migrateAlbumFiles(account, fileRepo)) { // migrated, try again yield* _call(account); } } } Stream _call(Account account) async* { List ls; try { ls = await Ls(fileRepo)( account, File( path: remote_storage_util.getRemoteAlbumsDir(account), )); } catch (e, stacktrace) { if (e is ApiException && e.response.statusCode == 404) { // no albums return; } yield Tuple2(e, stacktrace); return; } final albumFiles = ls.where((element) => element.isCollection != true).toList(); for (var i = 0; i < albumFiles.length; ++i) { var f = albumFiles[i]; try { if (CompatV25.isAlbumFileNeedMigration(f)) { f = await CompatV25.migrateAlbumFile(fileRepo, account, f); } albumFiles[i] = f; yield await albumRepo.get(account, f); } catch (e, stacktrace) { yield Tuple2(e, stacktrace); } } try { albumRepo.cleanUp( account, remote_storage_util.getRemoteAlbumsDir(account), albumFiles); } catch (e, stacktrace) { // not important, log and ignore _log.shout("[_call] Failed while cleanUp", e, stacktrace); } } final FileRepo fileRepo; final AlbumRepo albumRepo; static final _log = Logger("use_case.list_album.ListAlbum"); }