service.dart 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252
  1. import 'dart:io';
  2. import 'dart:async';
  3. import 'dart:convert';
  4. import 'package:get/get.dart';
  5. import 'package:naiyouwl/app/bean/ClashServiceInfo.dart';
  6. import 'package:naiyouwl/app/const/const.dart';
  7. import 'package:naiyouwl/app/controller/controllers.dart';
  8. import 'package:naiyouwl/app/utils/logger.dart';
  9. import 'package:naiyouwl/app/utils/shell.dart';
  10. import 'package:path/path.dart' as path;
  11. import 'package:http/http.dart' as http;
  12. import '../utils/utils.dart';
  13. final headers = {"User-Agent": "ccore-for-flutter/0.0.1"};
  14. class ServiceController extends GetxController {
  15. var url = "";
  16. var client = http.Client();
  17. var serviceMode = false.obs;
  18. var coreStatus = RunningState.stoped.obs;
  19. var serviceStatus = RunningState.stoped.obs;
  20. var installStatus = false.obs; // 新增变量用于记录安装/卸载状态
  21. bool get serviceIsRuning => serviceStatus.value == RunningState.running;
  22. Future<void> initConfig() async{
  23. url = "127.0.0.1:${controllers.config.config.value.servicePort}";
  24. //controllers.config.setSerivcePort(controllers.config.config.value.servicePort);
  25. }
  26. Future<void> startTunService() async {
  27. try {
  28. while (true) {
  29. final data = await fetchInfo() ?? ClashServiceInfo.fromJson({'code': -1, 'mode': '', 'status': '-1', 'version': '-1'});
  30. if (data.mode != 'service-mode') {
  31. if (serviceStatus.value == RunningState.running) {
  32. await stopService();
  33. }
  34. await install();
  35. } else {
  36. break;
  37. }
  38. }
  39. } catch (e) {
  40. serviceStatus.value = RunningState.error;
  41. log.debug(e.toString());
  42. }
  43. }
  44. Future<ClashServiceInfo?> fetchInfo() async {
  45. try {
  46. final res = await client.post(Uri.http(url, 'info'), headers: headers);
  47. var jsonResponse = jsonDecode(res.body) as Map<String, dynamic>;
  48. return ClashServiceInfo.fromJson(jsonResponse);
  49. } catch (e) {
  50. print('Exception: $e');
  51. return null;
  52. }
  53. }
  54. Future<void> fetchStartInit() async {
  55. controllers.config.config.value.selected = Files.makeInitProxyConfig.path;
  56. controllers.global.updateMsg("服务模式初始化...");
  57. await fetchStop();
  58. try {
  59. var ut = Uri.http(url, 'start');
  60. final body = json.encode({
  61. "args": [
  62. '-d',
  63. Paths.config.path,
  64. '-f',
  65. path.join(Paths.config.path, controllers.config.config.value.selected)
  66. ]
  67. });
  68. final res = await client.post(ut, body: body, headers: headers);
  69. var jsonResponse = jsonDecode(res.body) as Map<String, dynamic>;
  70. if (jsonResponse["code"] != 0) {
  71. coreStatus.value = RunningState.error;
  72. throw jsonResponse["msg"];
  73. }
  74. controllers.global.updateMsg("服务模式初始化成功..");
  75. coreStatus.value = RunningState.running;
  76. } catch (e) {
  77. print('Exception: $e');
  78. coreStatus.value = RunningState.error;
  79. controllers.global.updateMsg("服务模式初始化失败..");
  80. }
  81. }
  82. //应该程序退出调用
  83. Future<void> fetchStop() async {
  84. try {
  85. var ut = Uri.http(url, 'stop');
  86. await client.post(ut, headers: headers);
  87. coreStatus.value = RunningState.stoped;
  88. } catch (e) {
  89. print('Exception: $e');
  90. }
  91. }
  92. Future<void> fetchSetProxy() async {
  93. try {
  94. var dns_ip = "";
  95. if (controllers.global.routeModesSelect.value == "tun") {
  96. dns_ip = "198.18.0.2";
  97. }
  98. var ut = Uri.http(url, 'on');
  99. final body = json.encode({
  100. "http_port": controllers.config.mixedPort.value,
  101. "https_port": controllers.config.mixedPort.value,
  102. "socks_port": controllers.config.mixedPort.value,
  103. "bypass": "127.0.0.1, 192.168.0.0/16, 10.0.0.0/8, 172.16.0.0/12, 100.64.0.0/10, 17.0.0.0/8, localhost, *.local, *.crashlytics.com, seed-sequoia.siri.apple.com, sequoia.apple.com,xd.apple.com",
  104. "dns_ip": dns_ip,
  105. "dns_type": "",
  106. "address": "127.0.0.1"
  107. });
  108. await client.post(ut, body: body, headers: headers);
  109. } catch (e) {
  110. print('Exception: $e');
  111. }
  112. }
  113. Future<void> fetchSetProxyStop() async {
  114. try {
  115. var dnsIp = "";
  116. if (controllers.global.routeModesSelect.value == "tun") {
  117. dnsIp = "223.5.5.5";
  118. }
  119. var ut = Uri.http(url, 'off');
  120. final body = json.encode({
  121. "bypass": "",
  122. "dns_ip": dnsIp
  123. });
  124. await client.post(ut, body: body, headers: headers);
  125. } catch (e) {
  126. print('Exception: $e');
  127. }
  128. }
  129. Future<void> waitServiceStart() async {
  130. while (true) {
  131. await Future.delayed(const Duration(milliseconds: 100));
  132. try {
  133. await isCanOperationService();
  134. break;
  135. } catch (_) {}
  136. }
  137. }
  138. Future<void> install() async {
  139. try {
  140. final res = await runAsAdmin(Files.assetsClashService.path, ["-port", "${controllers.config.config.value.servicePort}", "stop", "uninstall", "install", "start"]);
  141. log.debug('install', res.stdout, res.stderr);
  142. if (res.exitCode != 0) {
  143. throw res.stderr;
  144. }
  145. await waitServiceStart();
  146. serviceStatus.value = RunningState.running;
  147. installStatus.value = true; // 安装成功
  148. } catch (e) {
  149. installStatus.value = false; // 安装失败
  150. log.debug('安装失败: $e');
  151. throw e;
  152. }
  153. }
  154. Future<void> uninstall() async {
  155. try {
  156. final res = await runAsAdmin(Files.assetsClashService.path, ["stop", "uninstall"]);
  157. log.debug('uninstall', res.stdout, res.stderr);
  158. if (res.exitCode != 0) throw res.stderr;
  159. installStatus.value = false; // 卸载成功
  160. serviceStatus.value = RunningState.stoped;
  161. } catch (e) {
  162. installStatus.value = true; // 卸载失败
  163. log.debug('卸载失败: $e');
  164. throw e;
  165. }
  166. }
  167. Future<void> stopService() async {
  168. serviceStatus.value = RunningState.stopping;
  169. if (coreStatus.value == RunningState.running) await fetchStop();
  170. serviceStatus.value = RunningState.stoped;
  171. }
  172. Future<void> serviceModeSwitch(bool open) async {
  173. if (serviceStatus.value == RunningState.running) await stopService();
  174. await controllers.service.stopClashCore();
  175. await fetchStop();
  176. try {
  177. controllers.global.updateMsg(open ? "安装服务" : "卸载服务");
  178. open ? await install() : await uninstall();
  179. } catch (e) {
  180. print(e.toString());
  181. }
  182. if(open)
  183. {
  184. await fetchStartInit();
  185. }
  186. }
  187. Future<void> stopClash() async {
  188. controllers.config.config.value.selected = Files.makeInitProxyConfig.path;
  189. if (coreStatus.value == RunningState.running) {
  190. await controllers.config.readClashCoreApi();
  191. controllers.core.setApi(controllers.config.clashCoreApiAddress.value, controllers.config.clashCoreApiSecret.value);
  192. await controllers.core.changeConfig(path.join(Paths.config.path, controllers.config.config.value.selected));
  193. }
  194. }
  195. Future<void> reloadClashCore() async {
  196. try {
  197. controllers.config.config.value.selected = Files.makeProxyConfig.path;
  198. if (coreStatus.value == RunningState.running) {
  199. controllers.global.updateMsg("切换配置...");
  200. await controllers.config.readClashCoreApi();
  201. controllers.core.setApi(controllers.config.clashCoreApiAddress.value, controllers.config.clashCoreApiSecret.value);
  202. await controllers.core.changeConfig(path.join(Paths.config.path, controllers.config.config.value.selected));
  203. controllers.global.updateMsg("fetchReloadConfig${controllers.config.clashCoreApiAddress.value}...");
  204. }
  205. } catch (e) {
  206. controllers.global.updateMsg("重新配置...");
  207. await controllers.config.readClashCoreApi();
  208. controllers.core.setApi(controllers.config.clashCoreApiAddress.value, controllers.config.clashCoreApiSecret.value);
  209. await controllers.core.changeConfig(path.join(Paths.config.path, controllers.config.config.value.selected));
  210. }
  211. }
  212. Future<bool> isCanOperationService() async {
  213. try {
  214. final res = await client.post(Uri.http(url, 'info'), headers: headers);
  215. if(res.statusCode == 200)
  216. {
  217. serviceStatus.value = RunningState.running;
  218. }
  219. return installStatus.value = res.statusCode == 200;
  220. } catch (e) {
  221. print('Exception when checking service: $e');
  222. return false;
  223. }
  224. }
  225. }