2023-10-09 09:32:07 +08:00
|
|
|
import 'dart:io';
|
|
|
|
|
2023-11-10 21:58:29 +08:00
|
|
|
import 'package:starcitizen_doctor/common/conf.dart';
|
2023-10-09 09:32:07 +08:00
|
|
|
import 'package:starcitizen_doctor/common/utils/base_utils.dart';
|
|
|
|
|
|
|
|
class SystemHelper {
|
2023-11-07 22:35:25 +08:00
|
|
|
static String powershellPath = "powershell.exe";
|
|
|
|
|
2023-11-10 21:58:29 +08:00
|
|
|
static initPowershellPath() async {
|
2023-11-07 22:35:25 +08:00
|
|
|
var result = await Process.run(powershellPath, ["echo", "ping"]);
|
|
|
|
if (!result.stdout.toString().startsWith("ping") &&
|
|
|
|
powershellPath == "powershell.exe") {
|
|
|
|
Map<String, String> envVars = Platform.environment;
|
|
|
|
final systemRoot = envVars["SYSTEMROOT"];
|
|
|
|
if (systemRoot != null) {
|
|
|
|
final autoSearchPath =
|
|
|
|
"$systemRoot\\System32\\WindowsPowerShell\\v1.0\\powershell.exe";
|
|
|
|
dPrint("auto search powershell path === $autoSearchPath");
|
|
|
|
powershellPath = autoSearchPath;
|
2023-11-10 21:58:29 +08:00
|
|
|
initPowershellPath();
|
2023-11-07 22:35:25 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-10-09 09:32:07 +08:00
|
|
|
static Future<bool> checkNvmePatchStatus() async {
|
|
|
|
try {
|
2023-11-07 22:35:25 +08:00
|
|
|
var result = await Process.run(SystemHelper.powershellPath, [
|
2023-10-09 09:32:07 +08:00
|
|
|
"Get-ItemProperty",
|
|
|
|
"-Path",
|
|
|
|
"\"HKLM:\\SYSTEM\\CurrentControlSet\\Services\\stornvme\\Parameters\\Device\"",
|
|
|
|
"-Name",
|
|
|
|
"\"ForcedPhysicalSectorSizeInBytes\""
|
|
|
|
]);
|
|
|
|
dPrint("checkNvmePatchStatus result ==== ${result.stdout}");
|
|
|
|
if (result.stderr == "" &&
|
|
|
|
result.stdout.toString().contains("{* 4095}")) {
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static Future<String> addNvmePatch() async {
|
2023-11-10 21:58:29 +08:00
|
|
|
var result = await powershellAdminRun([
|
2023-10-09 09:32:07 +08:00
|
|
|
'New-ItemProperty',
|
|
|
|
"-Path",
|
|
|
|
"\"HKLM:\\SYSTEM\\CurrentControlSet\\Services\\stornvme\\Parameters\\Device\"",
|
|
|
|
"-Name",
|
|
|
|
"ForcedPhysicalSectorSizeInBytes",
|
|
|
|
"-PropertyType MultiString",
|
|
|
|
"-Force -Value",
|
|
|
|
"\"* 4095\""
|
|
|
|
]);
|
|
|
|
dPrint("nvme_PhysicalBytes result == ${result.stdout}");
|
|
|
|
return result.stderr;
|
|
|
|
}
|
|
|
|
|
|
|
|
static doRemoveNvmePath() async {
|
|
|
|
try {
|
2023-11-10 21:58:29 +08:00
|
|
|
var result = await powershellAdminRun([
|
2023-10-09 09:32:07 +08:00
|
|
|
"Clear-ItemProperty",
|
|
|
|
"-Path",
|
|
|
|
"\"HKLM:\\SYSTEM\\CurrentControlSet\\Services\\stornvme\\Parameters\\Device\"",
|
|
|
|
"-Name",
|
|
|
|
"\"ForcedPhysicalSectorSizeInBytes\""
|
|
|
|
]);
|
|
|
|
dPrint("doRemoveNvmePath result ==== ${result.stdout}");
|
|
|
|
if (result.stderr == "") {
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// 获取 RSI 启动器 目录
|
|
|
|
static Future<String> getRSILauncherPath() async {
|
|
|
|
Map<String, String> envVars = Platform.environment;
|
|
|
|
final programDataPath = envVars["programdata"];
|
|
|
|
final rsiFilePath =
|
|
|
|
"$programDataPath\\Microsoft\\Windows\\Start Menu\\Programs\\Roberts Space Industries\\RSI Launcher.lnk";
|
|
|
|
final rsiLinkFile = File(rsiFilePath);
|
|
|
|
if (await rsiLinkFile.exists()) {
|
2023-11-07 22:35:25 +08:00
|
|
|
final r = await Process.run(SystemHelper.powershellPath, [
|
2023-10-09 09:32:07 +08:00
|
|
|
"(New-Object -ComObject WScript.Shell).CreateShortcut(\"$rsiFilePath\").targetpath"
|
|
|
|
]);
|
|
|
|
if (r.stdout.toString().contains("RSI Launcher.exe")) {
|
|
|
|
final start = r.stdout.toString().split("RSI Launcher.exe");
|
|
|
|
return "${start[0]}RSI Launcher.exe";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
|
|
|
static killRSILauncher() async {
|
|
|
|
var psr = await Process.run(
|
2023-11-10 21:58:29 +08:00
|
|
|
powershellPath, ["ps", "\"RSI Launcher\"", "|select -expand id"]);
|
2023-10-09 09:32:07 +08:00
|
|
|
if (psr.stderr == "") {
|
|
|
|
for (var value in (psr.stdout ?? "").toString().split("\n")) {
|
|
|
|
dPrint(value);
|
|
|
|
if (value != "") {
|
|
|
|
Process.killPid(int.parse(value));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static Future<List<String>> getPID(String name) async {
|
2023-11-10 21:58:29 +08:00
|
|
|
final r = await Process.run(powershellPath, ["(ps $name).Id"]);
|
2023-10-13 21:58:25 +08:00
|
|
|
final str = r.stdout.toString().trim();
|
|
|
|
dPrint(str);
|
|
|
|
if (str.isEmpty) return [];
|
|
|
|
return str.split("\n");
|
2023-10-09 09:32:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static checkAndLaunchRSILauncher(String path) async {
|
|
|
|
// check running and kill
|
|
|
|
await killRSILauncher();
|
|
|
|
// launch
|
2023-11-10 21:58:29 +08:00
|
|
|
final r = await Process.run(powershellPath, [
|
|
|
|
'Start-Process',
|
|
|
|
"'$path'",
|
|
|
|
'-Verb RunAs',
|
|
|
|
]);
|
2023-10-09 09:32:07 +08:00
|
|
|
dPrint(path);
|
|
|
|
dPrint(r.stdout);
|
|
|
|
dPrint(r.stderr);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Future<int> getSystemMemorySizeGB() async {
|
2023-11-10 21:58:29 +08:00
|
|
|
final r = await Process.run(powershellPath, [
|
2023-10-09 09:32:07 +08:00
|
|
|
"(Get-CimInstance Win32_PhysicalMemory | Measure-Object -Property capacity -Sum).sum /1gb"
|
|
|
|
]);
|
|
|
|
return int.tryParse(r.stdout.toString().trim()) ?? 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static Future<String> getSystemCimInstance(String win32InstanceName,
|
|
|
|
{pathName = "Name"}) async {
|
|
|
|
final r = await Process.run(
|
2023-11-10 21:58:29 +08:00
|
|
|
powershellPath, ["(Get-CimInstance $win32InstanceName).$pathName"]);
|
2023-10-09 09:32:07 +08:00
|
|
|
return r.stdout.toString().trim();
|
|
|
|
}
|
|
|
|
|
|
|
|
static Future<String> getSystemName() async {
|
|
|
|
final r = await Process.run(
|
2023-11-10 21:58:29 +08:00
|
|
|
powershellPath, ["(Get-ComputerInfo | Select-Object -expand OsName)"]);
|
2023-10-09 09:32:07 +08:00
|
|
|
return r.stdout.toString().trim();
|
|
|
|
}
|
|
|
|
|
2023-10-29 17:10:39 +08:00
|
|
|
static Future<String> getCpuName() async {
|
|
|
|
final r = await Process.run(
|
2023-11-10 21:58:29 +08:00
|
|
|
powershellPath, ["(Get-WmiObject -Class Win32_Processor).Name"]);
|
2023-10-29 17:10:39 +08:00
|
|
|
return r.stdout.toString().trim();
|
|
|
|
}
|
|
|
|
|
2023-10-09 09:32:07 +08:00
|
|
|
static Future<String> getGpuInfo() async {
|
|
|
|
const cmd = r"""
|
|
|
|
$adapterMemory = (Get-ItemProperty -Path "HKLM:\SYSTEM\ControlSet001\Control\Class\{4d36e968-e325-11ce-bfc1-08002be10318}\0*" -Name "HardwareInformation.AdapterString", "HardwareInformation.qwMemorySize" -Exclude PSPath -ErrorAction SilentlyContinue)
|
|
|
|
foreach ($adapter in $adapterMemory) {
|
|
|
|
[PSCustomObject] @{
|
|
|
|
Model=$adapter."HardwareInformation.AdapterString"
|
|
|
|
"VRAM (GB)"=[math]::round($adapter."HardwareInformation.qwMemorySize"/1GB)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
""";
|
2023-11-10 21:58:29 +08:00
|
|
|
final r = await Process.run(powershellPath, [cmd]);
|
2023-10-09 09:32:07 +08:00
|
|
|
return r.stdout.toString().trim();
|
|
|
|
}
|
|
|
|
|
|
|
|
static Future<String> getDiskInfo() async {
|
2023-11-10 21:58:29 +08:00
|
|
|
return (await Process.run(powershellPath,
|
2023-10-09 09:32:07 +08:00
|
|
|
["Get-PhysicalDisk | format-table BusType,FriendlyName,Size"]))
|
|
|
|
.stdout
|
|
|
|
.toString()
|
|
|
|
.trim();
|
|
|
|
}
|
|
|
|
|
|
|
|
static Future<int> getDirLen(String path, {List<String>? skipPath}) async {
|
|
|
|
if (path == "") return 0;
|
|
|
|
int totalSize = 0;
|
|
|
|
try {
|
|
|
|
final l = await Directory(path).list(recursive: true).toList();
|
|
|
|
for (var element in l) {
|
|
|
|
if (element is File) {
|
|
|
|
bool skip = false;
|
|
|
|
if (skipPath != null) {
|
|
|
|
for (var value in skipPath) {
|
|
|
|
if (element.absolute.path.startsWith(value)) {
|
|
|
|
skip = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!skip) totalSize += await element.length();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} catch (_) {}
|
|
|
|
return totalSize;
|
|
|
|
}
|
2023-11-10 21:58:29 +08:00
|
|
|
|
|
|
|
static Future<ProcessResult> powershellAdminRun(List<String> args) async {
|
|
|
|
// 创建 PowerShell 脚本文件
|
|
|
|
final scriptContent = """
|
|
|
|
${args.join(' ')}
|
|
|
|
""";
|
|
|
|
|
|
|
|
// 将脚本内容写入临时文件
|
|
|
|
final scriptFile =
|
|
|
|
File('${AppConf.applicationSupportDir}\\temp\\psh_script.ps1');
|
|
|
|
if (await scriptFile.exists()) {
|
|
|
|
await scriptFile.delete();
|
|
|
|
}
|
|
|
|
if (!await scriptFile.exists()) {
|
|
|
|
await scriptFile.create(recursive: true);
|
|
|
|
}
|
|
|
|
await scriptFile.writeAsString(scriptContent);
|
|
|
|
|
|
|
|
List<String> command = [
|
|
|
|
'Start-Process',
|
|
|
|
'powershell.exe',
|
|
|
|
'-Verb RunAs',
|
|
|
|
'-ArgumentList',
|
|
|
|
"'-NoProfile','-NonInteractive','-ExecutionPolicy','Bypass','-File','${scriptFile.absolute.path}'",
|
|
|
|
'-Wait',
|
|
|
|
'-PassThru '
|
|
|
|
];
|
|
|
|
|
|
|
|
final r = await Process.run(
|
|
|
|
'powershell.exe',
|
|
|
|
command,
|
|
|
|
runInShell: true,
|
|
|
|
);
|
|
|
|
await scriptFile.delete();
|
|
|
|
return r;
|
|
|
|
}
|
2023-10-09 09:32:07 +08:00
|
|
|
}
|