config.dart 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409
  1. import 'dart:io';
  2. import 'dart:convert';
  3. import 'package:dio/dio.dart';
  4. import 'package:get/get.dart';
  5. import 'package:naiyouwl/app/bean/config.dart';
  6. import 'package:naiyouwl/app/const/const.dart';
  7. import 'package:naiyouwl/app/controller/controllers.dart';
  8. import 'package:yaml/yaml.dart';
  9. import 'package:path/path.dart' as path;
  10. import 'package:flutter_emoji/flutter_emoji.dart';
  11. import '../data/model/NodeMode.dart';
  12. final Map<String, dynamic> _defaultConfig = {
  13. 'selected': 'example.yaml',
  14. 'updateInterval': 86400,
  15. 'updateSubsAtStart': false,
  16. 'setSystemProxy': false,
  17. 'startAtLogin': false,
  18. 'breakConnections': false,
  19. 'language': 'zh_CN',
  20. 'port': 9899,
  21. 'subs': [],
  22. };
  23. class ConfigController extends GetxController {
  24. late final dio;
  25. var config = Config.fromJson(_defaultConfig).obs;
  26. var clashCoreApiAddress = '127.0.0.1:9090'.obs;
  27. var clashCoreApiSecret = ''.obs;
  28. var clashCoreDns = ''.obs;
  29. var clashCoreTunEnable = false.obs;
  30. var servicePort = 9899.obs;
  31. var mixedPort = 9788.obs;
  32. var ApiAddressPort = 9799.obs;
  33. Future<void> initConfig() async {
  34. // var port = await getFreePort();
  35. //dio.addSentry();
  36. dio = Dio(BaseOptions(baseUrl: clashCoreApiAddress.value));
  37. if (!await Paths.config.exists()) await Paths.config.create(recursive: true);
  38. if (!await Files.configCountryMmdb.exists()) await Files.assetsCountryMmdb.copy(Files.configCountryMmdb.path);
  39. if (!await Files.configGeoIP.exists()) await Files.assetsGeoIP.copy(Files.configGeoIP.path);
  40. if (!await Files.configGeosite.exists()) await Files.assetsGeosite.copy(Files.configGeosite.path);
  41. if (Platform.isWindows && !await Files.configWintun.exists()) await Files.assetsWintun.copy(Files.configWintun.path);
  42. final locale = Get.deviceLocale!;
  43. _defaultConfig['language'] = '${locale.languageCode}_${locale.countryCode}';
  44. if (await Files.configConfig.exists()) {
  45. final local = json.decode(await Files.configConfig.readAsString());
  46. config.value = Config.fromJson({..._defaultConfig, ...local});
  47. } else {
  48. config.value = Config.fromJson(_defaultConfig);
  49. }
  50. // bool bg = await isPortOccupied(config.value.servicePort);
  51. // if(bg) {
  52. // config.value.servicePort = await getFreePort();
  53. // }
  54. if (config.value.subs.isEmpty) {
  55. if (!await Files.configExample.exists()) {
  56. await Files.assetsExample.copy(Files.configExample.path);
  57. }
  58. config.value.subs.add(ConfigSub(name: 'example.yaml', url: '', updateTime: 0));
  59. config.value.selected = 'example.yaml';
  60. }
  61. await save();
  62. await makeInitConfig();
  63. }
  64. String nodeToYaml(NodeMode node) {
  65. switch (node.type) {
  66. case 'trojan':
  67. return ''' - { name: ${node.name}, type: ${node.type}, server: ${node.host}, port: ${node.port}, password: ${node.passwd}, udp: 1 }''';
  68. case 'shadowsocks':
  69. return ''' - { name: ${node.name}, type: ss, server: ${node.host}, port: ${node.port}, password: ${node.passwd}, cipher: ${node.method}, udp: 1 }''';
  70. case 'v2ray':
  71. final type = (node.vless == 1) ? 'vless' : 'vmess';
  72. if (type == 'vless') {
  73. return ''' - { name: ${node.name}, type: $type, server: ${node.host}, port: ${node.port}, uuid: ${node.uuid}, alterId: ${node.v2AlterId}, udp: 1, flow: xtls-rprx-vision, servername: www.amazon.com, tls: true, reality-opts: { public-key: ${node.vlessPulkey} } }''';
  74. } else {
  75. return ''' - { name: ${node.name}, type: $type, server: ${node.host}, port: ${node.port}, uuid: ${node.uuid}, alterId: ${node.v2AlterId}, cipher: ${node.method}, udp: 1 }''';
  76. }
  77. default:
  78. return '';
  79. }
  80. }
  81. Future<void> makeInitConfig() async{
  82. var mode = controllers.global.modesSelect;
  83. var initconfig = '''
  84. mixed-port: ${mixedPort.value}
  85. allow-lan: true
  86. bind-address: '*'
  87. mode: $mode
  88. log-level: info
  89. external-controller: '127.0.0.1:${ApiAddressPort.value}'
  90. unified-delay: false
  91. geodata-mode: true
  92. tcp-concurrent: false
  93. find-process-mode: strict
  94. global-client-fingerprint: chrome
  95. proxies:
  96. rules:
  97. - GEOSITE,OpenAI,proxy
  98. - GEOSITE,TikTok,proxy
  99. - GEOSITE,github,proxy
  100. - GEOSITE,twitter,proxy
  101. - GEOSITE,youtube,proxy
  102. - GEOSITE,google,proxy
  103. - GEOSITE,telegram,proxy
  104. - GEOSITE,netflix,proxy
  105. - GEOSITE,geolocation-!cn,proxy
  106. - GEOSITE,cn,DIRECT
  107. - GEOIP,google,proxy
  108. - GEOIP,netflix,proxy
  109. - GEOIP,telegram,proxy
  110. - GEOIP,twitter,proxy
  111. - GEOIP,CN,DIRECT
  112. - MATCH,DIRECT
  113. ''';
  114. await Files.makeInitProxyConfig.writeAsString(initconfig);
  115. config.value.selected = 'init_proxy.yaml';
  116. await readClashCoreApi();
  117. }
  118. Future<void> makeClashConfig(List<NodeMode> nodes) async{
  119. // if( Files.makeProxyConfig.existsSync()){
  120. // Files.makeProxyConfig.deleteSync(recursive: true);
  121. // }
  122. var stack = "system";
  123. if( Platform.isWindows){
  124. stack = "gvisor";
  125. }
  126. var dnsPort = 1553;
  127. if(clashCoreTunEnable.value == true)
  128. {
  129. dnsPort = 53;
  130. }
  131. var dnsEnab = false;
  132. if(clashCoreTunEnable.value == true){
  133. dnsEnab = true;
  134. }
  135. var mode = controllers.global.modesSelect;
  136. var proxies = nodes.map(nodeToYaml).toList();
  137. var proxyGroups = '''
  138. proxy-groups:
  139. - name: proxy
  140. type: select
  141. proxies:
  142. - ${nodes.map((node) => node.name).join('\n - ')}
  143. ''';
  144. var rules = '''
  145. rules:
  146. - GEOSITE,OpenAI,proxy
  147. - GEOSITE,TikTok,proxy
  148. - GEOSITE,github,proxy
  149. - GEOSITE,twitter,proxy
  150. - GEOSITE,youtube,proxy
  151. - GEOSITE,google,proxy
  152. - GEOSITE,telegram,proxy
  153. - GEOSITE,netflix,proxy
  154. - GEOSITE,geolocation-!cn,proxy
  155. - GEOSITE,cn,DIRECT
  156. - GEOIP,google,proxy
  157. - GEOIP,netflix,proxy
  158. - GEOIP,telegram,proxy
  159. - GEOIP,twitter,proxy
  160. - GEOIP,CN,DIRECT
  161. - MATCH,proxy
  162. ''';
  163. var initconfig = '''
  164. mixed-port: ${mixedPort.value}
  165. allow-lan: true
  166. bind-address: '*'
  167. mode: $mode
  168. log-level: info
  169. external-controller: '127.0.0.1:${ApiAddressPort.value}'
  170. unified-delay: false
  171. geodata-mode: true
  172. tcp-concurrent: false
  173. find-process-mode: strict
  174. global-client-fingerprint: chrome
  175. dns:
  176. nameserver:
  177. - 114.114.114.114
  178. - 119.29.29.29
  179. - https://doh.pub/dns-query
  180. - https://dns.alidns.com/dns-query
  181. fallback:
  182. - https://dns.cloudflare.com/dns-query
  183. - "[2001:da8::666]:53"
  184. - https://public.dns.iij.jp/dns-query
  185. - https://jp.tiar.app/dns-query
  186. - https://jp.tiarap.org/dns-query
  187. - tls://dot.tiar.app
  188. enable: $dnsEnab
  189. ipv6: false
  190. # enhanced-mode: redir-host
  191. enhanced-mode: fake-ip
  192. fake-ip-range: 198.18.0.1/16
  193. listen: 0.0.0.0:$dnsPort
  194. fake-ip-filter:
  195. - "*.lan"
  196. default-nameserver:
  197. - 114.114.114.114
  198. - 119.29.29.29
  199. - "[2001:da8::666]:53"
  200. tun:
  201. enable: ${clashCoreTunEnable.value}
  202. stack: $stack
  203. # stack: gvisor
  204. dns-hijack:
  205. - 198.18.0.2:53 # when `fake-ip-range` is 198.18.0.1/16, should hijack 198.18.0.2:53
  206. auto-route: true # auto set global route for Windows
  207. # It is recommended to use `interface-name`
  208. auto-detect-interface: true # auto detect interface, conflict with `interface-name`
  209. proxies:
  210. ${proxies.join('\n')}
  211. $proxyGroups
  212. $rules
  213. ''';
  214. await Files.makeProxyConfig.writeAsString(initconfig);
  215. config.value.selected = 'proxy.yaml';
  216. await readClashCoreApi();
  217. }
  218. Future<void> save() async {
  219. await Files.configConfig.writeAsString(json.encode(config.toJson()));
  220. }
  221. Future<void> readClashCoreApi() async {
  222. final configStr = await File(path.join(Paths.config.path, config.value.selected)).readAsString();
  223. // final emoji = EmojiParser();
  224. // final b = emoji.unemojify(_config);
  225. final configJson = loadYaml(configStr.replaceAll(EmojiParser.REGEX_EMOJI, 'emoji'));
  226. // print(_json["external-controller"]);
  227. // https://github.com/dart-lang/yaml/issues/53
  228. // final _extControl = RegExp(r'''(?<!#\s*)external-controller:\s+['"]?([^'"]+?)['"]?\s''').firstMatch(_config)?.group(1);
  229. // final _secret = RegExp(r'''(?<!#\s*)secret:\s+['"]?([^'"]+?)['"]?\s''').firstMatch(_config)?.group(1);
  230. clashCoreApiAddress.value = (configJson["external-controller"] ?? '127.0.0.1:9090').replaceAll('0.0.0.0', '127.0.0.1');
  231. clashCoreApiSecret.value = (configJson["secret"] ?? '');
  232. clashCoreTunEnable.value = configJson["tun"]?["enable"] == true;
  233. clashCoreDns.value = '';
  234. if (configJson["dns"]?["enable"] == true && (configJson["dns"]["listen"] ?? '').isNotEmpty) {
  235. final dns = (configJson["dns"]["listen"] as String).split(":");
  236. final ip = dns[0];
  237. final port = dns[1];
  238. if (port == '53') {
  239. clashCoreDns.value = ip == '0.0.0.0' ? '127.0.0.1' : ip;
  240. }
  241. }
  242. }
  243. Future<void> setSerivcePort(int port) async {
  244. config.value.servicePort = port;
  245. await save();
  246. config.refresh();
  247. }
  248. Future<void> setLanguage(String language) async {
  249. config.value.language = language;
  250. await save();
  251. config.refresh();
  252. }
  253. Future<void> setSystemProxy(bool open) async {
  254. config.value.setSystemProxy = open;
  255. await save();
  256. config.refresh();
  257. }
  258. Future<void> setUpdateInterval(int value) async {
  259. config.value.updateInterval = value;
  260. await save();
  261. config.refresh();
  262. }
  263. Future<void> setUpdateSubsAtStart(bool value) async {
  264. config.value.updateSubsAtStart = value;
  265. await save();
  266. config.refresh();
  267. }
  268. Future<void> setSelectd(String selected) async {
  269. config.value.selected = selected;
  270. await save();
  271. config.refresh();
  272. }
  273. Future<bool> updateSub(ConfigSub sub) async {
  274. if ((sub.url ?? '').isEmpty) return false;
  275. final res = await dio.get(sub.url!);
  276. final subInfo = res.headers['subscription-userinfo'];
  277. final file = File(path.join(Paths.config.path, sub.name));
  278. final oldConfig = await file.exists() ? await file.readAsString() : '';
  279. final changed = oldConfig != res.data;
  280. if (changed) await file.writeAsString(res.data);
  281. sub.updateTime = DateTime.now().millisecondsSinceEpoch ~/ 1000;
  282. sub.info = null;
  283. if (subInfo != null) {
  284. // final info = Map.fromEntries(
  285. // subInfo.first.split(RegExp(r';\s*')).where((s) => s.isNotEmpty).map((e) => e.split('=')).map((e) => MapEntry(e[0], int.parse(e[1]))));
  286. final entries = subInfo.first.split(RegExp(r';\s*'))
  287. .where((s) => s.isNotEmpty)
  288. .map((e) => e.split('='))
  289. .toList();
  290. final info = <String, dynamic>{};
  291. for (var entry in entries) {
  292. info[entry[0]] = int.parse(entry[1]);
  293. }
  294. sub.info = ConfigSubInfo.fromJson(info);
  295. }
  296. await setSub(sub.name, sub);
  297. return changed;
  298. }
  299. Future<void> setSub(String subName, ConfigSub sub) async {
  300. final idx = config.value.subs.indexWhere((it) => it.name == subName);
  301. config.value.subs[idx] = sub;
  302. if (subName != sub.name) {
  303. final file = File(path.join(Paths.config.path, subName));
  304. if (await file.exists()) await file.rename(path.join(Paths.config.path, sub.name));
  305. }
  306. await save();
  307. config.refresh();
  308. }
  309. Future<void> addSub(ConfigSub sub) async {
  310. config.value.subs.add(sub);
  311. final file = File(path.join(Paths.config.path, sub.name));
  312. if (!await file.exists()) await file.create();
  313. await save();
  314. config.refresh();
  315. }
  316. Future<void> deleteSub(String subName) async {
  317. final file = File(path.join(Paths.config.path, subName));
  318. if (await file.exists()) await file.delete();
  319. config.value.subs.removeWhere((it) => it.name == subName);
  320. await save();
  321. config.refresh();
  322. }
  323. Future<void> setBreakConnections(bool value) async {
  324. config.value.breakConnections = value;
  325. await save();
  326. config.refresh();
  327. }
  328. Future<void> portDetection() async {
  329. bool isOk = await isPortOccupied(mixedPort.value);
  330. if(isOk){
  331. mixedPort.value = await getFreePort();
  332. }
  333. //await Future.delayed(const Duration(seconds: 5)); // 等待5秒
  334. isOk = await isPortOccupied(ApiAddressPort.value);
  335. if(isOk){
  336. ApiAddressPort.value = await getFreePort();
  337. }
  338. //await Future.delayed(const Duration(seconds: 5)); // 等待5秒
  339. if(!controllers.service.isRunning){
  340. isOk = await isPortOccupied(servicePort.value);
  341. if(isOk){
  342. servicePort.value = await getFreePort();
  343. }
  344. }
  345. }
  346. Future<int> getFreePort() async {
  347. var server = await HttpServer.bind(InternetAddress.loopbackIPv4, 0);
  348. int port = server.port;
  349. await server.close();
  350. return port;
  351. }
  352. Future<bool> isPortOccupied(int port) async {
  353. bool isOccupied = false;
  354. ServerSocket? server;
  355. try {
  356. server = await ServerSocket.bind(InternetAddress.loopbackIPv4, port);
  357. } catch (e) {
  358. isOccupied = true;
  359. } finally {
  360. await server?.close();
  361. }
  362. return isOccupied;
  363. }
  364. }