mirror of
https://ghfast.top/https://github.com/StarCitizenToolBox/app.git
synced 2025-06-28 18:24:45 +08:00
aria2c 按需加载
This commit is contained in:
@ -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";
|
||||||
|
@ -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));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -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());
|
||||||
|
@ -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));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -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(
|
||||||
|
@ -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!);
|
||||||
|
Reference in New Issue
Block a user