init.dart 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163
  1. // ignore_for_file: avoid_print
  2. import 'dart:io';
  3. import 'package:dio/dio.dart';
  4. import 'package:archive/archive.dart';
  5. import 'package:naiyouwl/app/const/const.dart';
  6. import 'package:path/path.dart' as path;
  7. final dio = Dio();
  8. // https://github.com/dart-lang/sdk/issues/31610
  9. final assetsPath = path.normalize(path.join(Platform.script.toFilePath(), '../../assets'));
  10. final binDir = Directory(path.join(assetsPath, 'bin'));
  11. final depDir = Directory(path.join(assetsPath, 'dep'));
  12. Future downloadLatestClashCore() async {
  13. const version = "v1.16.0";
  14. final String clashCoreName = 'clash.meta-${ClashName.platform}-${ClashName.arch}-cgo-$version';
  15. // Fetching release info from the GitHub API
  16. final info = await dio.get('https://api.github.com/repos/MetaCubeX/Clash.Meta/releases/tags/$version');
  17. // Extracting the download URL from the API response based on the asset name
  18. final Map<String, dynamic> latest = (info.data['assets'] as List<dynamic>)
  19. .firstWhere((it) => (it['name'] as String).contains(clashCoreName));
  20. final String downloadUrl = latest['browser_download_url'];
  21. //final String newName = 'clash-${ClashName.platform}-${ClashName.arch}';
  22. final String name = latest['name'];
  23. final tempFile = File(path.join(binDir.path, '$name.temp'));
  24. print('Downloading $name');
  25. await dio.download(downloadUrl, tempFile.path); // Use the obtained download URL
  26. print('Download Success');
  27. print('Unarchiving $name');
  28. final tempBetys = await tempFile.readAsBytes();
  29. if (name.contains('.gz')) {
  30. final bytes = GZipDecoder().decodeBytes(tempBetys);
  31. final String filePath = path.join(binDir.path, ClashName.name);
  32. await File(filePath).writeAsBytes(bytes);
  33. await Process.run('chmod', ['+x', filePath]);
  34. } else {
  35. final file = ZipDecoder().decodeBytes(tempBetys).first;
  36. await File(path.join(binDir.path, ClashName.name)).writeAsBytes(file.content);
  37. }
  38. await tempFile.delete();
  39. print('Unarchiv Success');
  40. }
  41. Future<void> downloadAndUnarchiveService() async {
  42. // const String token = "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"; // 请确保不在公共代码中硬编码这个token
  43. final serviceName = 'ccore-service-${ClashName.platform}-${ClashName.arch}';
  44. // "name" -> "ccore-service-darwin-amd64-v1.0.0.gz"
  45. //final newSericeName = "$serviceName-v1.0.0.gz";
  46. // 获取GitHub release信息
  47. // dio.options.headers["Authorization"] = "token $token";
  48. final response = await dio.get('https://api.github.com/repos/alroyso/core-service/releases/latest');
  49. final Map<String, dynamic> latest = (response.data['assets'] as List<dynamic>)
  50. .firstWhere((it) => (it['name'] as String).contains(serviceName));
  51. final downloadUrl = latest['browser_download_url'];
  52. print('Downloading $downloadUrl');
  53. final tempFile = File(path.join(binDir.path, '${latest['name']}.temp'));
  54. try
  55. {
  56. print(dio.options.headers);
  57. await dio.download(downloadUrl, tempFile.path);
  58. } catch (e){
  59. print(e);
  60. return;
  61. }
  62. print('Download Success');
  63. print('Unarchiving ${latest['name']}');
  64. final tempBytes = await tempFile.readAsBytes();
  65. if (latest['name'].contains('.gz')) {
  66. final bytes = GZipDecoder().decodeBytes(tempBytes);
  67. await File(path.join(binDir.path, serviceName)).writeAsBytes(bytes);
  68. final filePath = path.join(binDir.path, serviceName);
  69. await Process.run('chmod', ['+x', filePath]);
  70. } else {
  71. final file = ZipDecoder().decodeBytes(tempBytes).first;
  72. await File(path.join(binDir.path, file.name)).writeAsBytes(file.content);
  73. }
  74. await tempFile.delete();
  75. print('Unarchive Success');
  76. }
  77. Future downloadLatestClashService() async {
  78. final String serviceName = 'ccore-service-${ClashName.platform}-${ClashName.arch}';
  79. // // final info = await dio.get('https://api.github.com/repos/alroyso/clash-for-flutter-service/releases/latest');
  80. // // final Map<String, dynamic> latest = (info.data['assets'] as List<dynamic>).firstWhere((it) => (it['name'] as String).contains(serviceName));
  81. // //https://github.com/alroyso/clash-for-flutter-service/releases/download/untagged-8273cca760b55d0725ab/naiyou-service-darwin-arm64-v1.0.1.gz
  82. // var verion = 'v1.0.1';
  83. // final String name = 'naiyou-service-${ClashName.platform}-${ClashName.arch}-$verion.gz';
  84. // final tempFile = File(path.join(binDir.path, '$name.temp'));
  85. // var url = "https://github.com/alroyso/clash-for-flutter-service/releases/tag/untagged-8273cca760b55d0725ab";
  86. // print('Downloading $url/$name');
  87. // dio.options.headers["Authorization"] = "token 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";
  88. // await dio.download('$url/$name', tempFile.path);
  89. // print('Download Success');
  90. //
  91. // print('Unarchiving $name');
  92. // final tempBetys = await tempFile.readAsBytes();
  93. // if (name.contains('.gz')) {
  94. // final bytes = GZipDecoder().decodeBytes(tempBetys);
  95. // final String filePath = path.join(binDir.path, serviceName);
  96. // await File(filePath).writeAsBytes(bytes);
  97. // await Process.run('chmod', ['+x', filePath]);
  98. // } else {
  99. // final file = ZipDecoder().decodeBytes(tempBetys).first;
  100. // await File(path.join(binDir.path, file.name)).writeAsBytes(file.content);
  101. // }
  102. final String filePath = path.join(binDir.path, serviceName);
  103. await Process.run('chmod', ['+x', filePath]);
  104. //await tempFile.delete();
  105. print('Unarchiv Success');
  106. }
  107. Future downloadCountryMmdb() async {
  108. print('Downloading Country.mmdb');
  109. final String geoipFilePath = path.join(depDir.path, 'Country.mmdb');
  110. await dio.download('https://github.com/Dreamacro/maxmind-geoip/releases/latest/download/Country.mmdb', geoipFilePath);
  111. print('Download Success');
  112. final String geositedat = path.join(depDir.path, 'geosite.dat');
  113. await dio.download('https://github.com/MetaCubeX/meta-rules-dat/releases/download/latest/geosite.dat', geositedat);
  114. print('Download Success');
  115. final String geoip = path.join(depDir.path, 'geoip.dat');
  116. await dio.download('https://github.com/MetaCubeX/meta-rules-dat/releases/download/latest/geoip.dat', geoip);
  117. print('Download Success');
  118. }
  119. Future downloadWintun() async {
  120. print('Downloading Wintun');
  121. final File wintunFile = File(path.join(depDir.path, 'wintun.zip.temp'));
  122. await dio.download('https://www.wintun.net/builds/wintun-0.14.1.zip', wintunFile.path);
  123. print('Download Success');
  124. print('Unarchiving wintun.zip');
  125. final files = ZipDecoder().decodeBytes(await wintunFile.readAsBytes());
  126. final file = files.firstWhere((it) => it.name == 'wintun/bin/${ClashName.arch}/wintun.dll');
  127. await File(path.join(depDir.path, 'wintun.dll')).writeAsBytes(file.content);
  128. await wintunFile.delete();
  129. print('Unarchiv Success');
  130. }
  131. void main() async {
  132. ClashName.init();
  133. if (!(await binDir.exists())) await binDir.create();
  134. if (!(await depDir.exists())) await depDir.create();
  135. await downloadLatestClashCore();
  136. await downloadAndUnarchiveService();
  137. await downloadCountryMmdb();
  138. if (Platform.isWindows) await downloadWintun();
  139. }