config.dart 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188
  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:yaml/yaml.dart';
  8. import 'package:path/path.dart' as path;
  9. import 'package:flutter_emoji/flutter_emoji.dart';
  10. final Map<String, dynamic> _defaultConfig = {
  11. 'selected': 'example.yaml',
  12. 'updateInterval': 86400,
  13. 'updateSubsAtStart': false,
  14. 'setSystemProxy': false,
  15. 'startAtLogin': false,
  16. 'breakConnections': false,
  17. 'language': 'zh_CN',
  18. 'port': 9899,
  19. 'subs': [],
  20. };
  21. class ConfigController extends GetxController {
  22. late final dio;
  23. var config = Config.fromJson(_defaultConfig).obs;
  24. var clashCoreApiAddress = '127.0.0.1:9090'.obs;
  25. var clashCoreApiSecret = ''.obs;
  26. var clashCoreDns = ''.obs;
  27. var clashCoreTunEnable = false.obs;
  28. var servicePort = 0.obs;
  29. Future<void> initConfig() async {
  30. // var port = await getFreePort();
  31. //dio.addSentry();
  32. dio = Dio(BaseOptions(baseUrl: clashCoreApiAddress.value));
  33. if (!await Paths.config.exists()) await Paths.config.create(recursive: true);
  34. if (!await Files.configCountryMmdb.exists()) await Files.assetsCountryMmdb.copy(Files.configCountryMmdb.path);
  35. if (Platform.isWindows && !await Files.configWintun.exists()) await Files.assetsWintun.copy(Files.configWintun.path);
  36. final locale = Get.deviceLocale!;
  37. _defaultConfig['language'] = '${locale.languageCode}_${locale.countryCode}';
  38. if (await Files.configConfig.exists()) {
  39. final local = json.decode(await Files.configConfig.readAsString());
  40. config.value = Config.fromJson({..._defaultConfig, ...local});
  41. } else {
  42. config.value = Config.fromJson(_defaultConfig);
  43. }
  44. //config.value.port = port;
  45. if (config.value.subs.isEmpty) {
  46. if (!await Files.configExample.exists())
  47. await Files.assetsExample.copy(Files.configExample.path);
  48. config.value.subs.add(ConfigSub(name: 'example.yaml', url: '', updateTime: 0));
  49. config.value.selected = 'example.yaml';
  50. }
  51. await save();
  52. await readClashCoreApi();
  53. }
  54. Future<void> save() async {
  55. await Files.configConfig.writeAsString(json.encode(config.toJson()));
  56. }
  57. Future<void> readClashCoreApi() async {
  58. final configStr = await File(path.join(Paths.config.path, config.value.selected)).readAsString();
  59. // final emoji = EmojiParser();
  60. // final b = emoji.unemojify(_config);
  61. final configJson = loadYaml(configStr.replaceAll(EmojiParser.REGEX_EMOJI, 'emoji'));
  62. // print(_json["external-controller"]);
  63. // https://github.com/dart-lang/yaml/issues/53
  64. // final _extControl = RegExp(r'''(?<!#\s*)external-controller:\s+['"]?([^'"]+?)['"]?\s''').firstMatch(_config)?.group(1);
  65. // final _secret = RegExp(r'''(?<!#\s*)secret:\s+['"]?([^'"]+?)['"]?\s''').firstMatch(_config)?.group(1);
  66. clashCoreApiAddress.value = (configJson["external-controller"] ?? '127.0.0.1:9090').replaceAll('0.0.0.0', '127.0.0.1');
  67. clashCoreApiSecret.value = (configJson["secret"] ?? '');
  68. clashCoreTunEnable.value = configJson["tun"]?["enable"] == true;
  69. clashCoreDns.value = '';
  70. if (configJson["dns"]?["enable"] == true && (configJson["dns"]["listen"] ?? '').isNotEmpty) {
  71. final dns = (configJson["dns"]["listen"] as String).split(":");
  72. final ip = dns[0];
  73. final port = dns[1];
  74. if (port == '53') {
  75. clashCoreDns.value = ip == '0.0.0.0' ? '127.0.0.1' : ip;
  76. }
  77. }
  78. }
  79. Future<void> setLanguage(String language) async {
  80. config.value.language = language;
  81. await save();
  82. config.refresh();
  83. }
  84. Future<void> setSystemProxy(bool open) async {
  85. config.value.setSystemProxy = open;
  86. await save();
  87. config.refresh();
  88. }
  89. Future<void> setUpdateInterval(int value) async {
  90. config.value.updateInterval = value;
  91. await save();
  92. config.refresh();
  93. }
  94. Future<void> setUpdateSubsAtStart(bool value) async {
  95. config.value.updateSubsAtStart = value;
  96. await save();
  97. config.refresh();
  98. }
  99. Future<void> setSelectd(String selected) async {
  100. config.value.selected = selected;
  101. await save();
  102. config.refresh();
  103. }
  104. Future<bool> updateSub(ConfigSub sub) async {
  105. if ((sub.url ?? '').isEmpty) return false;
  106. final res = await dio.get(sub.url!);
  107. final subInfo = res.headers['subscription-userinfo'];
  108. final file = File(path.join(Paths.config.path, sub.name));
  109. final oldConfig = await file.exists() ? await file.readAsString() : '';
  110. final changed = oldConfig != res.data;
  111. if (changed) await file.writeAsString(res.data);
  112. sub.updateTime = DateTime.now().millisecondsSinceEpoch ~/ 1000;
  113. sub.info = null;
  114. if (subInfo != null) {
  115. // final info = Map.fromEntries(
  116. // subInfo.first.split(RegExp(r';\s*')).where((s) => s.isNotEmpty).map((e) => e.split('=')).map((e) => MapEntry(e[0], int.parse(e[1]))));
  117. final entries = subInfo.first.split(RegExp(r';\s*'))
  118. .where((s) => s.isNotEmpty)
  119. .map((e) => e.split('='))
  120. .toList();
  121. final info = <String, dynamic>{};
  122. for (var entry in entries) {
  123. info[entry[0]] = int.parse(entry[1]);
  124. }
  125. sub.info = ConfigSubInfo.fromJson(info);
  126. }
  127. await setSub(sub.name, sub);
  128. return changed;
  129. }
  130. Future<void> setSub(String subName, ConfigSub sub) async {
  131. final idx = config.value.subs.indexWhere((it) => it.name == subName);
  132. config.value.subs[idx] = sub;
  133. if (subName != sub.name) {
  134. final file = File(path.join(Paths.config.path, subName));
  135. if (await file.exists()) await file.rename(path.join(Paths.config.path, sub.name));
  136. }
  137. await save();
  138. config.refresh();
  139. }
  140. Future<void> addSub(ConfigSub sub) async {
  141. config.value.subs.add(sub);
  142. final file = File(path.join(Paths.config.path, sub.name));
  143. if (!await file.exists()) await file.create();
  144. await save();
  145. config.refresh();
  146. }
  147. Future<void> deleteSub(String subName) async {
  148. final file = File(path.join(Paths.config.path, subName));
  149. if (await file.exists()) await file.delete();
  150. config.value.subs.removeWhere((it) => it.name == subName);
  151. await save();
  152. config.refresh();
  153. }
  154. Future<void> setBreakConnections(bool value) async {
  155. config.value.breakConnections = value;
  156. await save();
  157. config.refresh();
  158. }
  159. Future<int> getFreePort() async {
  160. var server = await HttpServer.bind(InternetAddress.loopbackIPv4, 0);
  161. int port = server.port;
  162. await server.close();
  163. return port;
  164. }
  165. }