aria2c 按需加载

This commit is contained in:
2024-02-24 17:41:44 +08:00
parent 27b5d9259f
commit ca2f584f20
6 changed files with 77 additions and 43 deletions

View File

@ -10,9 +10,10 @@ class BinaryModuleConf {
"aria2c": "0", "aria2c": "0",
}; };
static Future extractModel() async { static Future extractModule(List<String> modules) async {
final workingDir = "${AppConf.applicationSupportDir}\\modules"; final workingDir = "${AppConf.applicationSupportDir}\\modules";
for (var m in _modules.entries) { for (var m in _modules.entries) {
if (!modules.contains(m.key)) continue;
final name = m.key; final name = m.key;
final version = m.value; final version = m.value;
final dir = "$workingDir\\$name"; final dir = "$workingDir\\$name";

View File

@ -5,36 +5,46 @@ import 'package:aria2/aria2.dart';
import 'package:flutter/foundation.dart'; import 'package:flutter/foundation.dart';
import 'package:starcitizen_doctor/base/ui.dart'; import 'package:starcitizen_doctor/base/ui.dart';
import 'package:starcitizen_doctor/common/conf/app_conf.dart'; import 'package:starcitizen_doctor/common/conf/app_conf.dart';
import 'package:starcitizen_doctor/common/conf/binary_conf.dart';
import 'package:starcitizen_doctor/common/helper/system_helper.dart'; import 'package:starcitizen_doctor/common/helper/system_helper.dart';
class Aria2cManager { class Aria2cManager {
static bool _isDaemonRunning = false; static int? _daemonPID;
static final String _aria2cDir = static final String _aria2cDir =
"${AppConf.applicationSupportDir}\\modules\\aria2c"; "${AppConf.applicationSupportDir}\\modules\\aria2c";
static Aria2c? _aria2c; static Aria2c? _aria2c;
static Aria2c get aria2c { static Aria2c getClient() {
if (!_isDaemonRunning) throw Exception("Aria2c Daemon not running!"); if (_aria2c != null) return _aria2c!;
if (_aria2c == null) { throw "not connect!";
_aria2c = Aria2c( }
"ws://127.0.0.1:64664/jsonrpc", "websocket", "ScToolbox_64664");
_aria2c!.getVersion().then((value) { static bool get isAvailable => _daemonPID != null && _aria2c != null;
dPrint("Aria2cManager.connected! version == ${value.version}");
}); static Future checkLazyLoad() async {
try {
final sessionFile = File("$_aria2cDir\\aria2.session");
// 有下载任务则第一时间初始化
if (await sessionFile.exists() &&
(await sessionFile.readAsString()).trim().isNotEmpty) {
await launchDaemon();
}
} catch (e) {
dPrint("Aria2cManager.checkLazyLoad Error:$e");
} }
return _aria2c!;
} }
static Future launchDaemon() async { static Future launchDaemon() async {
if (_isDaemonRunning) return; if (_daemonPID != null) return;
await BinaryModuleConf.extractModule(["aria2c"]);
/// skip for debug hot reload /// skip for debug hot reload
if (kDebugMode) { if (kDebugMode) {
if ((await SystemHelper.getPID("aria2c")).isNotEmpty) { if ((await SystemHelper.getPID("aria2c")).isNotEmpty) {
dPrint("[Aria2cManager] debug skip for hot reload"); dPrint("[Aria2cManager] debug skip for hot reload");
_isDaemonRunning = true; _daemonPID = 0;
return; return;
} }
} }
@ -69,21 +79,26 @@ class Aria2cManager {
if (event.trim().isEmpty) return; if (event.trim().isEmpty) return;
dPrint("[aria2c]: ${event.trim()}"); dPrint("[aria2c]: ${event.trim()}");
if (event.contains("IPv4 RPC: listening on TCP port")) { if (event.contains("IPv4 RPC: listening on TCP port")) {
_isDaemonRunning = true; _daemonPID = p.pid;
aria2c; _aria2c = Aria2c(
"ws://127.0.0.1:64664/jsonrpc", "websocket", "ScToolbox_64664");
_aria2c!.getVersion().then((value) {
dPrint("Aria2cManager.connected! version == ${value.version}");
});
} }
}, onDone: () { }, onDone: () {
dPrint("[aria2c] onDone: "); dPrint("[aria2c] onDone: ");
_isDaemonRunning = false; _daemonPID = null;
}, onError: (e) { }, onError: (e) {
dPrint("[aria2c] stdout ERROR: $e"); dPrint("[aria2c] stdout ERROR: $e");
_isDaemonRunning = false; _daemonPID = null;
}); });
p.pid;
p.stderr.transform(utf8.decoder).listen((event) { p.stderr.transform(utf8.decoder).listen((event) {
dPrint("[aria2c] stderr ERROR : $event"); dPrint("[aria2c] stderr ERROR : $event");
}); });
while (true) { while (true) {
if (_isDaemonRunning) return; if (_daemonPID != null) return;
await Future.delayed(const Duration(milliseconds: 100)); await Future.delayed(const Duration(milliseconds: 100));
} }
} }

View File

@ -37,18 +37,22 @@ class DownloadsUIModel extends BaseUIModel {
onTapButton(String key) async { onTapButton(String key) async {
switch (key) { switch (key) {
case "pause_all": case "pause_all":
await Aria2cManager.aria2c.pauseAll(); if (!Aria2cManager.isAvailable) return;
await Aria2cManager.getClient().pauseAll();
return; return;
case "resume_all": case "resume_all":
await Aria2cManager.aria2c.unpauseAll(); if (!Aria2cManager.isAvailable) return;
await Aria2cManager.getClient().unpauseAll();
return; return;
case "cancel_all": case "cancel_all":
final userOK = await showConfirmDialogs( final userOK = await showConfirmDialogs(
context!, "确认取消全部任务?", const Text("如果文件不再需要,你可能需要手动删除下载文件。")); context!, "确认取消全部任务?", const Text("如果文件不再需要,你可能需要手动删除下载文件。"));
if (userOK == true) { if (userOK == true) {
if (!Aria2cManager.isAvailable) return;
try { try {
for (var value in [...tasks, ...waitingTasks]) { for (var value in [...tasks, ...waitingTasks]) {
await Aria2cManager.aria2c.remove(value.gid!); await Aria2cManager.getClient().remove(value.gid!);
} }
} catch (e) { } catch (e) {
dPrint("DownloadsUIModel cancel_all Error: $e"); dPrint("DownloadsUIModel cancel_all Error: $e");
@ -65,12 +69,15 @@ class DownloadsUIModel extends BaseUIModel {
try { try {
while (true) { while (true) {
if (!mounted) return; if (!mounted) return;
tasks.clear(); if (Aria2cManager.isAvailable) {
tasks = await Aria2cManager.aria2c.tellActive(); final aria2c = Aria2cManager.getClient();
waitingTasks = await Aria2cManager.aria2c.tellWaiting(0, 1000000); tasks.clear();
stoppedTasks = await Aria2cManager.aria2c.tellStopped(0, 1000000); tasks = await aria2c.tellActive();
globalStat = await Aria2cManager.aria2c.getGlobalStat(); waitingTasks = await aria2c.tellWaiting(0, 1000000);
notifyListeners(); stoppedTasks = await aria2c.tellStopped(0, 1000000);
globalStat = await aria2c.getGlobalStat();
notifyListeners();
}
await Future.delayed(const Duration(seconds: 1)); await Future.delayed(const Duration(seconds: 1));
} }
} catch (e) { } catch (e) {
@ -136,14 +143,17 @@ class DownloadsUIModel extends BaseUIModel {
} }
Future<void> resumeTask(String? gid) async { Future<void> resumeTask(String? gid) async {
final aria2c = Aria2cManager.getClient();
if (gid != null) { if (gid != null) {
await Aria2cManager.aria2c.unpause(gid); await aria2c.unpause(gid);
} }
} }
Future<void> pauseTask(String? gid) async { Future<void> pauseTask(String? gid) async {
final aria2c = Aria2cManager.getClient();
if (gid != null) { if (gid != null) {
await Aria2cManager.aria2c.pause(gid); await aria2c.pause(gid);
} }
} }
@ -153,7 +163,8 @@ class DownloadsUIModel extends BaseUIModel {
final ok = await showConfirmDialogs( final ok = await showConfirmDialogs(
context!, "确认取消下载?", const Text("如果文件不再需要,你可能需要手动删除下载文件。")); context!, "确认取消下载?", const Text("如果文件不再需要,你可能需要手动删除下载文件。"));
if (ok == true) { if (ok == true) {
await Aria2cManager.aria2c.remove(gid); final aria2c = Aria2cManager.getClient();
await aria2c.remove(gid);
} }
} }
} }
@ -220,12 +231,13 @@ class DownloadsUIModel extends BaseUIModel {
], ],
)); ));
if (ok == true) { if (ok == true) {
await handleError(() => Aria2cManager.launchDaemon());
final aria2c = Aria2cManager.getClient();
final upByte = Aria2cManager.textToByte(upCtrl.text.trim()); final upByte = Aria2cManager.textToByte(upCtrl.text.trim());
final downByte = Aria2cManager.textToByte(downCtrl.text.trim()); final downByte = Aria2cManager.textToByte(downCtrl.text.trim());
final r = await handleError( final r = await handleError(() => aria2c.changeGlobalOption(Aria2Option()
() => Aria2cManager.aria2c.changeGlobalOption(Aria2Option() ..maxOverallUploadLimit = upByte
..maxOverallUploadLimit = upByte ..maxOverallDownloadLimit = downByte));
..maxOverallDownloadLimit = downByte));
if (r != null) { if (r != null) {
await box.put('downloader_up_limit', upCtrl.text.trim()); await box.put('downloader_up_limit', upCtrl.text.trim());
await box.put('downloader_down_limit', downCtrl.text.trim()); await box.put('downloader_down_limit', downCtrl.text.trim());

View File

@ -108,13 +108,17 @@ class IndexUIModel extends BaseUIModel {
void _listenAria2c() async { void _listenAria2c() async {
while (true) { while (true) {
if (!mounted) return;
try { try {
aria2globalStat = await Aria2cManager.aria2c.getGlobalStat(); if (Aria2cManager.isAvailable) {
notifyListeners(); final aria2c = Aria2cManager.getClient();
aria2globalStat = await aria2c.getGlobalStat();
notifyListeners();
}
} catch (e) { } catch (e) {
dPrint("aria2globalStat update error:$e"); dPrint("aria2globalStat update error:$e");
} }
await Future.delayed(const Duration(seconds: 10)); await Future.delayed(const Duration(seconds: 5));
} }
} }
} }

View File

@ -1,6 +1,5 @@
import 'package:starcitizen_doctor/api/analytics.dart'; import 'package:starcitizen_doctor/api/analytics.dart';
import 'package:starcitizen_doctor/base/ui_model.dart'; import 'package:starcitizen_doctor/base/ui_model.dart';
import 'package:starcitizen_doctor/common/conf/binary_conf.dart';
import 'package:starcitizen_doctor/common/conf/url_conf.dart'; import 'package:starcitizen_doctor/common/conf/url_conf.dart';
import 'package:starcitizen_doctor/common/io/aria2c.dart'; import 'package:starcitizen_doctor/common/io/aria2c.dart';
import 'package:starcitizen_doctor/ui/index_ui.dart'; import 'package:starcitizen_doctor/ui/index_ui.dart';
@ -29,8 +28,7 @@ class SplashUIModel extends BaseUIModel {
await AppConf.checkUpdate(); await AppConf.checkUpdate();
step = 2; step = 2;
notifyListeners(); notifyListeners();
await handleError(() => BinaryModuleConf.extractModel()); await Aria2cManager.checkLazyLoad();
await handleError(() => Aria2cManager.launchDaemon());
Navigator.pushAndRemoveUntil( Navigator.pushAndRemoveUntil(
context!, context!,
BaseUIContainer( BaseUIContainer(

View File

@ -381,13 +381,17 @@ class ToolsUIModel extends BaseUIModel {
return; return;
} }
/// 启动模块
await handleError(() => Aria2cManager.launchDaemon());
final aria2c = Aria2cManager.getClient();
try { try {
final b64Str = base64Encode(btData.data!); final b64Str = base64Encode(btData.data!);
final gid = await Aria2cManager.aria2c final gid =
.addTorrent(b64Str, extraParams: {"dir": savePath}); await aria2c.addTorrent(b64Str, extraParams: {"dir": savePath});
_working = false; _working = false;
dPrint("Aria2cManager.aria2c.addUri resp === $gid"); dPrint("Aria2cManager.aria2c.addUri resp === $gid");
await Aria2cManager.aria2c.saveSession(); await aria2c.saveSession();
BaseUIContainer( BaseUIContainer(
uiCreate: () => DownloadsUI(), uiCreate: () => DownloadsUI(),
modelCreate: () => DownloadsUIModel()).push(context!); modelCreate: () => DownloadsUIModel()).push(context!);