2023-10-09 09:32:07 +08:00
|
|
|
import 'dart:convert';
|
|
|
|
import 'dart:io';
|
|
|
|
|
2023-11-21 23:36:26 +08:00
|
|
|
import 'package:hive/hive.dart';
|
2024-04-27 14:40:01 +08:00
|
|
|
import 'package:starcitizen_doctor/common/conf/const_conf.dart';
|
2024-03-01 20:59:43 +08:00
|
|
|
import 'package:starcitizen_doctor/common/utils/log.dart';
|
2023-11-21 23:36:26 +08:00
|
|
|
|
2023-10-09 09:32:07 +08:00
|
|
|
class SCLoggerHelper {
|
|
|
|
static Future<String?> getLogFilePath() async {
|
2024-06-16 11:52:25 +08:00
|
|
|
if (!Platform.isWindows) return null;
|
2023-10-09 09:32:07 +08:00
|
|
|
Map<String, String> envVars = Platform.environment;
|
|
|
|
final appDataPath = envVars["appdata"];
|
|
|
|
if (appDataPath == null) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
final rsiLauncherPath = "$appDataPath\\rsilauncher";
|
|
|
|
dPrint("rsiLauncherPath:$rsiLauncherPath");
|
|
|
|
final jsonLogPath = "$rsiLauncherPath\\logs\\log.log";
|
|
|
|
return jsonLogPath;
|
|
|
|
}
|
|
|
|
|
|
|
|
static Future<String?> getShaderCachePath() async {
|
|
|
|
Map<String, String> envVars = Platform.environment;
|
|
|
|
final appDataPath = envVars["LOCALAPPDATA"];
|
|
|
|
if (appDataPath == null) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
final scCachePath = "$appDataPath\\Star Citizen";
|
|
|
|
dPrint("getShaderCachePath === $scCachePath");
|
|
|
|
return scCachePath;
|
|
|
|
}
|
|
|
|
|
|
|
|
static Future<List?> getLauncherLogList() async {
|
2024-06-16 11:52:25 +08:00
|
|
|
if (!Platform.isWindows) return [];
|
2024-06-16 12:11:19 +08:00
|
|
|
try {
|
|
|
|
final jsonLogPath = await getLogFilePath();
|
|
|
|
if (jsonLogPath == null) throw "no file path";
|
|
|
|
var jsonString = utf8.decode(await File(jsonLogPath).readAsBytes());
|
2024-09-11 20:41:14 +08:00
|
|
|
return jsonString.split("\n");
|
2024-06-16 12:11:19 +08:00
|
|
|
} catch (e) {
|
|
|
|
dPrint(e);
|
|
|
|
return [];
|
2023-10-09 09:32:07 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static Future<List<String>> getGameInstallPath(List listData,
|
|
|
|
{bool checkExists = true,
|
|
|
|
List<String> withVersion = const ["LIVE"]}) async {
|
|
|
|
List<String> scInstallPaths = [];
|
|
|
|
|
2023-11-21 23:36:26 +08:00
|
|
|
checkAndAddPath(String path, bool checkExists) async {
|
2024-09-11 20:41:14 +08:00
|
|
|
// 将所有连续的 \\ 替换为 \
|
|
|
|
path = path.replaceAll(RegExp(r'\\+'), "\\");
|
2023-11-21 23:36:26 +08:00
|
|
|
if (path.isNotEmpty && !scInstallPaths.contains(path)) {
|
|
|
|
if (!checkExists) {
|
|
|
|
dPrint("find installPath == $path");
|
|
|
|
scInstallPaths.add(path);
|
|
|
|
} else if (await File("$path/Bin64/StarCitizen.exe").exists() &&
|
|
|
|
await File("$path/Data.p4k").exists()) {
|
|
|
|
dPrint("find installPath == $path");
|
|
|
|
scInstallPaths.add(path);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
final confBox = await Hive.openBox("app_conf");
|
|
|
|
final path = confBox.get("custom_game_path");
|
|
|
|
if (path != null && path != "") {
|
|
|
|
for (var v in withVersion) {
|
|
|
|
await checkAndAddPath("$path\\$v", checkExists);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
for (var v in withVersion) {
|
2024-09-11 20:41:14 +08:00
|
|
|
String pattern =
|
|
|
|
r'([a-zA-Z]:\\\\[^\\\\]*\\\\[^\\\\]*\\\\StarCitizen\\\\' + v + r')';
|
|
|
|
RegExp regExp = RegExp(pattern, caseSensitive: false);
|
2023-11-21 23:36:26 +08:00
|
|
|
for (var i = listData.length - 1; i > 0; i--) {
|
2024-09-11 20:41:14 +08:00
|
|
|
final line = listData[i];
|
|
|
|
final matches = regExp.allMatches(line);
|
|
|
|
for (var match in matches) {
|
|
|
|
await checkAndAddPath(match.group(0)!, checkExists);
|
2023-10-09 09:32:07 +08:00
|
|
|
}
|
|
|
|
}
|
2024-09-11 20:41:14 +08:00
|
|
|
}
|
2023-11-21 23:36:26 +08:00
|
|
|
|
2024-09-11 20:41:14 +08:00
|
|
|
if (scInstallPaths.isNotEmpty) {
|
|
|
|
// 动态检测更多位置
|
|
|
|
for (var fileName in List.from(scInstallPaths)) {
|
2024-06-16 12:11:19 +08:00
|
|
|
for (var v in withVersion) {
|
2024-09-11 20:41:14 +08:00
|
|
|
if (fileName.toString().endsWith(v)) {
|
|
|
|
for (var nv in withVersion) {
|
|
|
|
final nextName =
|
|
|
|
"${fileName.toString().replaceAll("\\$v", "")}\\$nv";
|
|
|
|
await checkAndAddPath(nextName, true);
|
2024-06-16 12:11:19 +08:00
|
|
|
}
|
2023-11-22 00:28:14 +08:00
|
|
|
}
|
|
|
|
}
|
2023-11-21 23:36:26 +08:00
|
|
|
}
|
|
|
|
}
|
2024-06-16 12:11:19 +08:00
|
|
|
} catch (e) {
|
|
|
|
dPrint(e);
|
|
|
|
if (scInstallPaths.isEmpty) rethrow;
|
2023-10-09 09:32:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return scInstallPaths;
|
|
|
|
}
|
2024-02-16 23:32:16 +08:00
|
|
|
|
2024-04-27 14:40:01 +08:00
|
|
|
static String getGameChannelID(String installPath) {
|
|
|
|
for (var value in ConstConf.gameChannels) {
|
|
|
|
if (installPath.endsWith("\\$value")) {
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return "UNKNOWN";
|
|
|
|
}
|
|
|
|
|
2024-02-17 00:35:21 +08:00
|
|
|
static Future<List<String>?> getGameRunningLogs(String gameDir) async {
|
|
|
|
final logFile = File("$gameDir/Game.log");
|
|
|
|
if (!await logFile.exists()) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return await logFile.readAsLines(
|
|
|
|
encoding: const Utf8Codec(allowMalformed: true));
|
|
|
|
}
|
|
|
|
|
|
|
|
static MapEntry<String, String>? getGameRunningLogInfo(List<String> logs) {
|
|
|
|
for (var i = logs.length - 1; i > 0; i--) {
|
|
|
|
final line = logs[i];
|
|
|
|
final r = _checkRunningLine(line);
|
|
|
|
if (r != null) {
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return null;
|
2024-02-16 23:32:16 +08:00
|
|
|
}
|
|
|
|
|
2024-02-17 00:35:21 +08:00
|
|
|
static MapEntry<String, String>? _checkRunningLine(String line) {
|
|
|
|
if (line.contains("STATUS_CRYENGINE_OUT_OF_SYSMEM")) {
|
2024-03-17 16:54:09 +08:00
|
|
|
return MapEntry(S.current.doctor_game_error_low_memory,
|
|
|
|
S.current.doctor_game_error_low_memory_info);
|
2024-02-17 00:35:21 +08:00
|
|
|
}
|
|
|
|
if (line.contains("EXCEPTION_ACCESS_VIOLATION")) {
|
2024-03-17 16:54:09 +08:00
|
|
|
return MapEntry(S.current.doctor_game_error_generic_info,
|
2024-02-17 00:35:21 +08:00
|
|
|
"https://docs.qq.com/doc/DUURxUVhzTmZoY09Z");
|
|
|
|
}
|
|
|
|
if (line.contains("DXGI_ERROR_DEVICE_REMOVED")) {
|
2024-03-17 16:54:09 +08:00
|
|
|
return MapEntry(S.current.doctor_game_error_gpu_crash,
|
|
|
|
"https://www.bilibili.com/read/cv19335199");
|
2024-02-17 00:35:21 +08:00
|
|
|
}
|
|
|
|
if (line.contains("Wakeup socket sendto error")) {
|
2024-03-17 16:54:09 +08:00
|
|
|
return MapEntry(S.current.doctor_game_error_socket_error,
|
|
|
|
S.current.doctor_game_error_socket_error_info);
|
2024-02-17 00:35:21 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (line.contains("The requested operation requires elevated")) {
|
2024-03-17 16:54:09 +08:00
|
|
|
return MapEntry(S.current.doctor_game_error_permissions_error,
|
|
|
|
S.current.doctor_game_error_permissions_error_info);
|
2024-02-17 00:35:21 +08:00
|
|
|
}
|
|
|
|
if (line.contains(
|
|
|
|
"The process cannot access the file because is is being used by another process")) {
|
2024-03-17 16:54:09 +08:00
|
|
|
return MapEntry(S.current.doctor_game_error_game_process_error,
|
|
|
|
S.current.doctor_game_error_game_process_error_info);
|
2024-02-17 00:35:21 +08:00
|
|
|
}
|
|
|
|
if (line.contains("0xc0000043")) {
|
2024-03-17 16:54:09 +08:00
|
|
|
return MapEntry(S.current.doctor_game_error_game_damaged_file,
|
|
|
|
S.current.doctor_game_error_game_damaged_file_info);
|
2024-02-17 00:35:21 +08:00
|
|
|
}
|
|
|
|
if (line.contains("option to verify the content of the Data.p4k file")) {
|
2024-03-17 16:54:09 +08:00
|
|
|
return MapEntry(S.current.doctor_game_error_game_damaged_p4k_file,
|
|
|
|
S.current.doctor_game_error_game_damaged_p4k_file_info);
|
2024-02-17 00:35:21 +08:00
|
|
|
}
|
|
|
|
if (line.contains("OUTOFMEMORY Direct3D could not allocate")) {
|
2024-03-17 16:54:09 +08:00
|
|
|
return MapEntry(S.current.doctor_game_error_low_gpu_memory,
|
|
|
|
S.current.doctor_game_error_low_gpu_memory_info);
|
2024-02-17 00:35:21 +08:00
|
|
|
}
|
2024-07-21 17:07:19 +08:00
|
|
|
if (line.contains(
|
|
|
|
"try disabling with r_vulkanDisableLayers = 1 in your user.cfg")) {
|
|
|
|
return MapEntry(S.current.doctor_game_error_gpu_vulkan_crash,
|
|
|
|
S.current.doctor_game_error_gpu_vulkan_crash_info);
|
|
|
|
}
|
2024-02-17 00:35:21 +08:00
|
|
|
|
|
|
|
/// Unknown
|
|
|
|
if (line.contains("network.replicatedEntityHandle")) {
|
|
|
|
return const MapEntry("_", "network.replicatedEntityHandle");
|
|
|
|
}
|
|
|
|
if (line.contains("Exception Unknown")) {
|
|
|
|
return const MapEntry("_", "Exception Unknown");
|
|
|
|
}
|
|
|
|
return null;
|
2024-02-16 23:32:16 +08:00
|
|
|
}
|
2023-10-09 09:32:07 +08:00
|
|
|
}
|