Files
umbrix/lib/features/proxy/overview/proxies_overview_notifier.dart

154 lines
4.7 KiB
Dart
Raw Normal View History

2023-07-06 17:18:41 +03:30
import 'dart:async';
2023-09-06 17:14:24 +03:30
import 'package:combine/combine.dart';
import 'package:dartx/dartx.dart';
2023-12-01 12:56:24 +03:30
import 'package:hiddify/core/localization/translations.dart';
import 'package:hiddify/core/preferences/preferences_provider.dart';
import 'package:hiddify/features/connection/notifier/connection_notifier.dart';
import 'package:hiddify/features/proxy/data/proxy_data_providers.dart';
import 'package:hiddify/features/proxy/model/proxy_entity.dart';
import 'package:hiddify/features/proxy/model/proxy_failure.dart';
2023-09-06 17:14:24 +03:30
import 'package:hiddify/utils/pref_notifier.dart';
2023-09-10 20:27:07 +03:30
import 'package:hiddify/utils/riverpod_utils.dart';
2023-07-06 17:18:41 +03:30
import 'package:hiddify/utils/utils.dart';
import 'package:riverpod_annotation/riverpod_annotation.dart';
import 'package:rxdart/rxdart.dart';
2023-07-06 17:18:41 +03:30
2023-12-01 12:56:24 +03:30
part 'proxies_overview_notifier.g.dart';
2023-07-06 17:18:41 +03:30
2023-09-06 17:14:24 +03:30
enum ProxiesSort {
unsorted,
name,
delay;
String present(TranslationsEn t) => switch (this) {
ProxiesSort.unsorted => t.proxies.sortOptions.unsorted,
ProxiesSort.name => t.proxies.sortOptions.name,
ProxiesSort.delay => t.proxies.sortOptions.delay,
};
}
2023-09-17 14:24:25 +03:30
@Riverpod(keepAlive: true)
class ProxiesSortNotifier extends _$ProxiesSortNotifier {
late final _pref = Pref(
2023-12-01 12:56:24 +03:30
ref.watch(sharedPreferencesProvider).requireValue,
2023-09-17 14:24:25 +03:30
"proxies_sort_mode",
ProxiesSort.unsorted,
mapFrom: ProxiesSort.values.byName,
mapTo: (value) => value.name,
);
@override
ProxiesSort build() => _pref.getValue();
Future<void> update(ProxiesSort value) {
state = value;
return _pref.update(value);
}
}
2023-09-06 17:14:24 +03:30
@riverpod
2023-12-01 12:56:24 +03:30
class ProxiesOverviewNotifier extends _$ProxiesOverviewNotifier with AppLogger {
2023-07-06 17:18:41 +03:30
@override
2023-12-01 12:56:24 +03:30
Stream<List<ProxyGroupEntity>> build() async* {
2023-09-10 20:27:07 +03:30
ref.disposeDelay(const Duration(seconds: 15));
2023-08-29 19:32:31 +03:30
final serviceRunning = await ref.watch(serviceRunningProvider.future);
if (!serviceRunning) {
2023-12-01 12:56:24 +03:30
throw const ServiceNotRunning();
2023-08-19 22:27:23 +03:30
}
2023-09-17 14:24:25 +03:30
final sortBy = ref.watch(proxiesSortNotifierProvider);
2023-09-06 17:14:24 +03:30
yield* ref
2023-12-01 12:56:24 +03:30
.watch(proxyRepositoryProvider)
.watchProxies()
.throttleTime(
const Duration(milliseconds: 100),
leading: false,
trailing: true,
)
2023-09-06 17:14:24 +03:30
.map(
2023-08-29 19:32:31 +03:30
(event) => event.getOrElse(
2023-10-03 21:12:14 +03:30
(err) {
loggy.warning("error receiving proxies", err);
throw err;
2023-08-29 19:32:31 +03:30
},
),
2023-09-06 17:14:24 +03:30
)
.asyncMap((proxies) async => _sortOutbounds(proxies, sortBy));
}
2023-12-01 12:56:24 +03:30
Future<List<ProxyGroupEntity>> _sortOutbounds(
List<ProxyGroupEntity> proxies,
2023-09-06 17:14:24 +03:30
ProxiesSort sortBy,
) async {
return CombineWorker().execute(
() {
final groupWithSelected = {
2023-12-01 12:56:24 +03:30
for (final o in proxies) o.tag: o.selected,
};
2023-12-01 12:56:24 +03:30
final sortedProxies = <ProxyGroupEntity>[];
for (final group in proxies) {
final sortedItems = switch (sortBy) {
2023-09-06 17:14:24 +03:30
ProxiesSort.name => group.items.sortedBy((e) => e.tag),
ProxiesSort.delay => group.items.sortedWith((a, b) {
final ai = a.urlTestDelay;
final bi = b.urlTestDelay;
if (ai == 0 && bi == 0) return -1;
if (ai == 0 && bi > 0) return 1;
if (ai > 0 && bi == 0) return -1;
if (ai == bi && a.type.isGroup) return -1;
return ai.compareTo(bi);
}),
ProxiesSort.unsorted => group.items,
};
2023-12-01 12:56:24 +03:30
final items = <ProxyItemEntity>[];
for (final item in sortedItems) {
if (groupWithSelected.keys.contains(item.tag)) {
items
.add(item.copyWith(selectedTag: groupWithSelected[item.tag]));
} else {
items.add(item);
}
}
2023-12-01 12:56:24 +03:30
sortedProxies.add(group.copyWith(items: items));
2023-09-06 17:14:24 +03:30
}
2023-12-01 12:56:24 +03:30
return sortedProxies;
2023-09-06 17:14:24 +03:30
},
);
2023-07-06 17:18:41 +03:30
}
2023-08-29 19:32:31 +03:30
Future<void> changeProxy(String groupTag, String outboundTag) async {
loggy.debug(
"changing proxy, group: [$groupTag] - outbound: [$outboundTag]",
);
if (state case AsyncData(value: final outbounds)) {
await ref
2023-12-01 12:56:24 +03:30
.read(proxyRepositoryProvider)
.selectProxy(groupTag, outboundTag)
2023-10-03 21:12:14 +03:30
.getOrElse((err) {
loggy.warning("error selecting outbound", err);
throw err;
2023-08-29 19:32:31 +03:30
}).run();
state = AsyncData(
[
...outbounds.map(
(e) => e.tag == groupTag ? e.copyWith(selected: outboundTag) : e,
),
],
).copyWithPrevious(state);
}
}
2023-07-06 17:18:41 +03:30
2023-08-29 19:32:31 +03:30
Future<void> urlTest(String groupTag) async {
loggy.debug("testing group: [$groupTag]");
if (state case AsyncData()) {
2023-12-01 12:56:24 +03:30
await ref
.read(proxyRepositoryProvider)
.urlTest(groupTag)
.getOrElse((err) {
2023-10-05 22:47:24 +03:30
loggy.error("error testing group", err);
2023-10-03 21:12:14 +03:30
throw err;
2023-08-29 19:32:31 +03:30
}).run();
}
2023-07-06 17:18:41 +03:30
}
}