nc-photos/app/lib/bloc/ls_dir.dart

218 lines
5.4 KiB
Dart
Raw Normal View History

2021-04-10 06:28:12 +02:00
import 'package:bloc/bloc.dart';
2021-08-16 21:05:00 +02:00
import 'package:equatable/equatable.dart';
2021-04-10 06:28:12 +02:00
import 'package:logging/logging.dart';
import 'package:nc_photos/account.dart';
2022-01-10 20:15:57 +01:00
import 'package:nc_photos/debug_util.dart';
2021-04-10 06:28:12 +02:00
import 'package:nc_photos/entity/file.dart';
2022-01-10 13:02:52 +01:00
import 'package:nc_photos/exception.dart';
2021-04-10 06:28:12 +02:00
import 'package:nc_photos/use_case/ls.dart';
2021-08-16 21:05:00 +02:00
class LsDirBlocItem with EquatableMixin {
LsDirBlocItem(this.file, this.isE2ee, this.children);
2021-04-10 06:28:12 +02:00
@override
toString({bool isDeep = false}) {
if (isDeep) {
return "$runtimeType:${_toDeepString(0)}";
} else {
2021-07-23 22:05:57 +02:00
final childrenStr =
children == null ? "null" : "List {length: ${children!.length}}";
return "$runtimeType {"
"file: '${file.path}', "
2021-07-23 22:05:57 +02:00
"children: $childrenStr, "
"}";
}
}
String _toDeepString(int level) {
String product = "\n" + " " * (level * 2) + "-${file.path}";
if (children != null) {
2021-07-23 22:05:57 +02:00
for (final c in children!) {
product += c._toDeepString(level + 1);
}
}
return product;
}
2021-08-16 21:05:00 +02:00
@override
get props => [
file,
children,
];
2021-07-23 22:05:57 +02:00
final File file;
final bool isE2ee;
2021-04-10 06:28:12 +02:00
/// Child directories under this directory
///
/// Null if this dir is not listed, due to things like depth limitation
2021-07-23 22:05:57 +02:00
List<LsDirBlocItem>? children;
2021-04-10 06:28:12 +02:00
}
abstract class LsDirBlocEvent {
const LsDirBlocEvent();
}
class LsDirBlocQuery extends LsDirBlocEvent {
const LsDirBlocQuery(
this.account,
this.root, {
this.depth = 1,
});
2021-04-10 06:28:12 +02:00
@override
toString() {
return "$runtimeType {"
"account: $account, "
"root: '${root.path}', "
"depth: $depth, "
2021-04-10 06:28:12 +02:00
"}";
}
LsDirBlocQuery copyWith({
2021-07-23 22:05:57 +02:00
Account? account,
File? root,
int? depth,
}) {
return LsDirBlocQuery(
account ?? this.account,
root ?? this.root,
depth: depth ?? this.depth,
);
}
2021-04-10 06:28:12 +02:00
final Account account;
final File root;
final int depth;
2021-04-10 06:28:12 +02:00
}
2021-08-16 21:05:00 +02:00
abstract class LsDirBlocState with EquatableMixin {
2021-07-23 22:05:57 +02:00
const LsDirBlocState(this.account, this.root, this.items);
2021-04-10 06:28:12 +02:00
@override
toString() {
return "$runtimeType {"
"account: $account, "
2021-08-16 21:05:00 +02:00
"root: ${root.path}, "
2021-04-10 06:28:12 +02:00
"items: List {length: ${items.length}}, "
"}";
}
2021-08-16 21:05:00 +02:00
@override
get props => [
account,
root,
items,
];
2021-07-23 22:05:57 +02:00
final Account? account;
final File root;
final List<LsDirBlocItem> items;
2021-04-10 06:28:12 +02:00
}
class LsDirBlocInit extends LsDirBlocState {
LsDirBlocInit() : super(null, File(path: ""), const []);
2021-04-10 06:28:12 +02:00
}
class LsDirBlocLoading extends LsDirBlocState {
2021-07-23 22:05:57 +02:00
const LsDirBlocLoading(Account? account, File root, List<LsDirBlocItem> items)
: super(account, root, items);
2021-04-10 06:28:12 +02:00
}
class LsDirBlocSuccess extends LsDirBlocState {
2021-07-23 22:05:57 +02:00
const LsDirBlocSuccess(Account? account, File root, List<LsDirBlocItem> items)
: super(account, root, items);
2021-04-10 06:28:12 +02:00
}
class LsDirBlocFailure extends LsDirBlocState {
const LsDirBlocFailure(
2021-07-23 22:05:57 +02:00
Account? account, File root, List<LsDirBlocItem> items, this.exception)
: super(account, root, items);
2021-04-10 06:28:12 +02:00
@override
toString() {
return "$runtimeType {"
"super: ${super.toString()}, "
"exception: $exception, "
"}";
}
2021-08-16 21:05:00 +02:00
@override
get props => [
...super.props,
exception,
];
2021-04-10 06:28:12 +02:00
final dynamic exception;
}
/// A bloc that return all directories under a dir recursively
class LsDirBloc extends Bloc<LsDirBlocEvent, LsDirBlocState> {
2022-07-09 07:59:09 +02:00
LsDirBloc(this.fileRepo) : super(LsDirBlocInit()) {
on<LsDirBlocEvent>(_onEvent);
}
2021-04-10 06:28:12 +02:00
2022-07-09 07:59:09 +02:00
Future<void> _onEvent(
LsDirBlocEvent event, Emitter<LsDirBlocState> emit) async {
_log.info("[_onEvent] $event");
2021-04-10 06:28:12 +02:00
if (event is LsDirBlocQuery) {
2022-07-09 07:59:09 +02:00
await _onEventQuery(event, emit);
2021-04-10 06:28:12 +02:00
}
}
2022-07-09 07:59:09 +02:00
Future<void> _onEventQuery(
LsDirBlocQuery ev, Emitter<LsDirBlocState> emit) async {
2021-04-10 06:28:12 +02:00
try {
2022-07-09 07:59:09 +02:00
emit(LsDirBlocLoading(ev.account, ev.root, state.items));
emit(LsDirBlocSuccess(ev.account, ev.root, await _query(ev)));
2021-04-10 06:28:12 +02:00
} catch (e) {
_log.severe("[_onEventQuery] Exception while request", e);
2022-07-09 07:59:09 +02:00
emit(LsDirBlocFailure(ev.account, ev.root, state.items, e));
2021-04-10 06:28:12 +02:00
}
}
Future<List<LsDirBlocItem>> _query(LsDirBlocQuery ev) async {
final product = <LsDirBlocItem>[];
var files = _cache[ev.root.path];
if (files == null) {
2021-08-16 21:05:00 +02:00
files = (await Ls(fileRepo)(ev.account, ev.root))
2021-07-23 22:05:57 +02:00
.where((f) => f.isCollection ?? false)
.toList();
_cache[ev.root.path] = files;
}
2022-01-10 13:02:52 +01:00
final removes = <File>[];
2021-04-10 06:28:12 +02:00
for (final f in files) {
2022-01-10 13:02:52 +01:00
try {
List<LsDirBlocItem>? children;
if (ev.depth > 1) {
children = await _query(ev.copyWith(root: f, depth: ev.depth - 1));
}
product.add(LsDirBlocItem(f, false, children));
2022-01-10 13:02:52 +01:00
} on ApiException catch (e) {
if (e.response.statusCode == 404) {
// this could happen when the server db contains dangling entries
2022-01-10 20:15:57 +01:00
_log.warning(
"[call] HTTP404 error while listing dir: ${logFilename(f.path)}");
2022-01-10 13:02:52 +01:00
removes.add(f);
} else if (f.isCollection == true && e.response.statusCode == 403) {
// e2ee dir
_log.warning("[call] HTTP403 error, likely E2EE dir: ${f.path}");
product.add(LsDirBlocItem(f, true, []));
2022-01-10 13:02:52 +01:00
} else {
rethrow;
}
2021-04-10 06:28:12 +02:00
}
2022-01-10 13:02:52 +01:00
}
if (removes.isNotEmpty) {
files.removeWhere((f) => removes.contains(f));
2021-04-10 06:28:12 +02:00
}
return product;
2021-04-10 06:28:12 +02:00
}
2021-08-16 21:05:00 +02:00
final FileRepo fileRepo;
final _cache = <String, List<File>>{};
2021-04-10 06:28:12 +02:00
static final _log = Logger("bloc.ls_dir.LsDirBloc");
}