From 8cf13ae3a18086a83acc246ad03b9964936227ad Mon Sep 17 00:00:00 2001 From: liukaiyuan Date: Sat, 10 Dec 2022 17:54:49 +0800 Subject: [PATCH] =?UTF-8?q?=E4=BA=A4=E4=BA=92=E4=BD=93=E9=AA=8C=E5=8D=87?= =?UTF-8?q?=E7=BA=A7=E3=80=82=E4=B9=B1=E7=A0=81=E9=97=AE=E9=A2=98=E5=86=8D?= =?UTF-8?q?=E5=BA=A6=E4=BF=AE=E5=A4=8D=E3=80=82=E7=AE=97=E6=B3=95=E9=80=BB?= =?UTF-8?q?=E8=BE=91=E4=BC=98=E5=8C=96=E3=80=82?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- LKY_OfficeTools/Common/Com_ExeOS.cs | 226 ++--------------------- LKY_OfficeTools/Lib/Lib_AppInfo.cs | 2 +- LKY_OfficeTools/Lib/Lib_Aria2c.cs | 6 +- LKY_OfficeTools/Lib/Lib_OfficeClean.cs | 27 ++- LKY_OfficeTools/Lib/Lib_OfficeInstall.cs | 63 ++++--- LKY_OfficeTools/OfficeTools.cs | 2 +- 6 files changed, 75 insertions(+), 251 deletions(-) diff --git a/LKY_OfficeTools/Common/Com_ExeOS.cs b/LKY_OfficeTools/Common/Com_ExeOS.cs index b325cdb..d603019 100644 --- a/LKY_OfficeTools/Common/Com_ExeOS.cs +++ b/LKY_OfficeTools/Common/Com_ExeOS.cs @@ -8,8 +8,6 @@ using System; using System.Collections.Generic; using System.Diagnostics; -using System.Runtime.InteropServices; -using System.Security; using static LKY_OfficeTools.Lib.Lib_AppLog; namespace LKY_OfficeTools.Common @@ -25,23 +23,30 @@ internal class Com_ExeOS internal class Run { /// - /// 启动一个外部exe + /// 启动一个外部exe,等待完成后,返回Code。 /// /// /// - /// 默认等待结束后,返回 - /// - internal static bool Exe(string file_path, string args, bool WaitForExit = true) + /// 执行失败时,返回:-920921 + internal static int Exe(string file_path, string args) { try { Process p = new Process(); - return Process(file_path, args, out p, WaitForExit); + var result = Process(file_path, args, out p, true); //默认等待完成 + + //是否执行了 + if (!result) + { + throw new Exception($"执行 {file_path} 异常!"); + } + + return p.ExitCode; } catch (Exception Ex) { new Log(Ex.ToString()); - return false; + return -920921; } } @@ -79,7 +84,6 @@ internal static bool Process(string file_path, string args, out Process ProcessI if (WaitForExit) { ProcessInfo.WaitForExit(); - ProcessInfo.Close(); //只在等待情况下,才进行close,否则异步情况下,close后,ProcessInfo 将为空 } return true; @@ -145,210 +149,6 @@ internal static string Cmd(string args) return null; } } - - /// - /// 在服务模式下显式的运行 exe - /// - internal class ServiceMode - { - #region Structures - [StructLayout(LayoutKind.Sequential)] - internal struct SECURITY_ATTRIBUTES - { - internal int Length; - internal IntPtr lpSecurityDescriptor; - internal bool bInheritHandle; - } - [StructLayout(LayoutKind.Sequential)] - internal struct STARTUPINFO - { - internal int cb; - internal string lpReserved; - internal string lpDesktop; - internal string lpTitle; - internal uint dwX; - internal uint dwY; - internal uint dwXSize; - internal uint dwYSize; - internal uint dwXCountChars; - internal uint dwYCountChars; - internal uint dwFillAttribute; - internal uint dwFlags; - internal short wShowWindow; - internal short cbReserved2; - internal IntPtr lpReserved2; - internal IntPtr hStdInput; - internal IntPtr hStdOutput; - internal IntPtr hStdError; - } - [StructLayout(LayoutKind.Sequential)] - internal struct PROCESS_INFORMATION - { - internal IntPtr hProcess; - internal IntPtr hThread; - internal uint dwProcessId; - internal uint dwThreadId; - } - #endregion - #region Enumerations - enum TOKEN_TYPE : int - { - TokenPrimary = 1, - TokenImpersonation = 2 - } - enum SECURITY_IMPERSONATION_LEVEL : int - { - SecurityAnonymous = 0, - SecurityIdentification = 1, - SecurityImpersonation = 2, - SecurityDelegation = 3, - } - enum WTSInfoClass - { - InitialProgram, - ApplicationName, - WorkingDirectory, - OEMId, - SessionId, - UserName, - WinStationName, - DomainName, - ConnectState, - ClientBuildNumber, - ClientName, - ClientDirectory, - ClientProductId, - ClientHardwareId, - ClientAddress, - ClientDisplay, - ClientProtocolType - } - #endregion - - #region Constants - internal const int TOKEN_DUPLICATE = 0x0002; - internal const uint MAXIMUM_ALLOWED = 0x2000000; - internal const int CREATE_NEW_CONSOLE = 0x00000010; - internal const int IDLE_PRIORITY_CLASS = 0x40; - internal const int NORMAL_PRIORITY_CLASS = 0x20; - internal const int HIGH_PRIORITY_CLASS = 0x80; - internal const int REALTIME_PRIORITY_CLASS = 0x100; - #endregion - - #region Win32 API Imports - [DllImport("kernel32.dll", SetLastError = true)] - private static extern bool CloseHandle(IntPtr hSnapshot); - [DllImport("kernel32.dll")] - static extern uint WTSGetActiveConsoleSessionId(); - [DllImport("wtsapi32.dll", CharSet = CharSet.Unicode, SetLastError = true), SuppressUnmanagedCodeSecurityAttribute] - static extern bool WTSQuerySessionInformation(System.IntPtr hServer, int sessionId, WTSInfoClass wtsInfoClass, out System.IntPtr ppBuffer, out uint pBytesReturned); - [DllImport("advapi32.dll", EntryPoint = "CreateProcessAsUser", SetLastError = true, CharSet = CharSet.Ansi, CallingConvention = CallingConvention.StdCall)] - internal extern static bool CreateProcessAsUser(IntPtr hToken, string lpApplicationName, string lpCommandLine, ref SECURITY_ATTRIBUTES lpProcessAttributes, - ref SECURITY_ATTRIBUTES lpThreadAttributes, bool bInheritHandle, int dwCreationFlags, IntPtr lpEnvironment, - String lpCurrentDirectory, ref STARTUPINFO lpStartupInfo, out PROCESS_INFORMATION lpProcessInformation); - [DllImport("kernel32.dll")] - static extern bool ProcessIdToSessionId(uint dwProcessId, ref uint pSessionId); - [DllImport("advapi32.dll", EntryPoint = "DuplicateTokenEx")] - internal extern static bool DuplicateTokenEx(IntPtr ExistingTokenHandle, uint dwDesiredAccess, - ref SECURITY_ATTRIBUTES lpThreadAttributes, int TokenType, - int ImpersonationLevel, ref IntPtr DuplicateTokenHandle); - [DllImport("kernel32.dll")] - static extern IntPtr OpenProcess(uint dwDesiredAccess, bool bInheritHandle, uint dwProcessId); - [DllImport("advapi32", SetLastError = true), SuppressUnmanagedCodeSecurityAttribute] - static extern bool OpenProcessToken(IntPtr ProcessHandle, int DesiredAccess, ref IntPtr TokenHandle); - #endregion - - internal static string GetCurrentActiveUser() - { - IntPtr hServer = IntPtr.Zero, state = IntPtr.Zero; - uint bCount = 0; - // obtain the currently active session id; every logged on user in the system has a unique session id - uint dwSessionId = WTSGetActiveConsoleSessionId(); - string domain = string.Empty, userName = string.Empty; - if (WTSQuerySessionInformation(hServer, (int)dwSessionId, WTSInfoClass.DomainName, out state, out bCount)) - { - domain = Marshal.PtrToStringAuto(state); - } - if (WTSQuerySessionInformation(hServer, (int)dwSessionId, WTSInfoClass.UserName, out state, out bCount)) - { - userName = Marshal.PtrToStringAuto(state); - } - return string.Format("{0}\\{1}", domain, userName); - } - - /// - /// Launches the given application with full admin rights, and in addition bypasses the Vista UAC prompt - /// - /// The name of the application to launch - /// Process information regarding the launched application that gets returned to the caller - /// - internal static bool StartProcessAndByPassUAC(string exe_path, string command, out PROCESS_INFORMATION procInfo) - { - uint winlogonPid = 0; - IntPtr hUserTokenDup = IntPtr.Zero, hPToken = IntPtr.Zero, hProcess = IntPtr.Zero; - procInfo = new PROCESS_INFORMATION(); - // obtain the currently active session id; every logged on user in the system has a unique session id - uint dwSessionId = WTSGetActiveConsoleSessionId(); - // obtain the process id of the winlogon process that is running within the currently active session - Process[] processes = System.Diagnostics.Process.GetProcessesByName("winlogon"); - foreach (Process p in processes) - { - if ((uint)p.SessionId == dwSessionId) - { - winlogonPid = (uint)p.Id; - } - } - // obtain a handle to the winlogon process - hProcess = OpenProcess(MAXIMUM_ALLOWED, false, winlogonPid); - // obtain a handle to the access token of the winlogon process - if (!OpenProcessToken(hProcess, TOKEN_DUPLICATE, ref hPToken)) - { - CloseHandle(hProcess); - return false; - } - // Security attibute structure used in DuplicateTokenEx and CreateProcessAsUser - // I would prefer to not have to use a security attribute variable and to just - // simply pass null and inherit (by default) the security attributes - // of the existing token. However, in C# structures are value types and therefore - // cannot be assigned the null value. - SECURITY_ATTRIBUTES sa = new SECURITY_ATTRIBUTES(); - sa.Length = Marshal.SizeOf(sa); - // copy the access token of the winlogon process; the newly created token will be a primary token - if (!DuplicateTokenEx(hPToken, MAXIMUM_ALLOWED, ref sa, (int)SECURITY_IMPERSONATION_LEVEL.SecurityIdentification, (int)TOKEN_TYPE.TokenPrimary, ref hUserTokenDup)) - { - CloseHandle(hProcess); - CloseHandle(hPToken); - return false; - } - // By default CreateProcessAsUser creates a process on a non-interactive window station, meaning - // the window station has a desktop that is invisible and the process is incapable of receiving - // user input. To remedy this we set the lpDesktop parameter to indicate we want to enable user - // interaction with the new process. - STARTUPINFO si = new STARTUPINFO(); - si.cb = (int)Marshal.SizeOf(si); - si.lpDesktop = @"winsta0\default"; // interactive window station parameter; basically this indicates that the process created can display a GUI on the desktop - // flags that specify the priority and creation method of the process - int dwCreationFlags = NORMAL_PRIORITY_CLASS | CREATE_NEW_CONSOLE; - // create a new process in the current user's logon session - bool result = CreateProcessAsUser(hUserTokenDup, // client's access token - exe_path, // file to execute - command, // command line - ref sa, // pointer to process SECURITY_ATTRIBUTES - ref sa, // pointer to thread SECURITY_ATTRIBUTES - false, // handles are not inheritable - dwCreationFlags, // creation flags - IntPtr.Zero, // pointer to new environment block - null, // name of current directory - ref si, // pointer to STARTUPINFO structure - out procInfo // receives information about new process - ); - // invalidate the handles - CloseHandle(hProcess); - CloseHandle(hPToken); - CloseHandle(hUserTokenDup); - return result; // return the result - } - } } /// diff --git a/LKY_OfficeTools/Lib/Lib_AppInfo.cs b/LKY_OfficeTools/Lib/Lib_AppInfo.cs index 9075ef4..303027c 100644 --- a/LKY_OfficeTools/Lib/Lib_AppInfo.cs +++ b/LKY_OfficeTools/Lib/Lib_AppInfo.cs @@ -51,7 +51,7 @@ internal class AppAttribute /// /// APP 版本号 /// - internal const string AppVersion = "1.1.1.1207"; + internal const string AppVersion = "1.1.1.1209"; /// /// 开发者拼音全拼 diff --git a/LKY_OfficeTools/Lib/Lib_Aria2c.cs b/LKY_OfficeTools/Lib/Lib_Aria2c.cs index f16d220..a47be67 100644 --- a/LKY_OfficeTools/Lib/Lib_Aria2c.cs +++ b/LKY_OfficeTools/Lib/Lib_Aria2c.cs @@ -45,7 +45,11 @@ internal static int DownFile(string uri, string save_to) string aria2c_params = $"{uri} --dir=\"{file_path}\" --out=\"{filename}\" --continue=true --max-connection-per-server=5 --check-integrity=true --file-allocation=none"; //new Log(aria2c_params); - Com_ExeOS.Run.Exe(aria2c_path, aria2c_params); + var down_result = Com_ExeOS.Run.Exe(aria2c_path, aria2c_params); + if (down_result == -920921) + { + throw new Exception(); + } return 1; } diff --git a/LKY_OfficeTools/Lib/Lib_OfficeClean.cs b/LKY_OfficeTools/Lib/Lib_OfficeClean.cs index 52e13f5..9309fc5 100644 --- a/LKY_OfficeTools/Lib/Lib_OfficeClean.cs +++ b/LKY_OfficeTools/Lib/Lib_OfficeClean.cs @@ -305,7 +305,13 @@ internal static bool RemovePreviousVersion() { string product_name = msi_id_name_dic[now_id_cmd.Key]; //完整的产品名 new Log($"\n >> 开始移除 {product_name} 及其组件 ...", ConsoleColor.DarkYellow); - Com_ExeOS.Run.Exe("msiexec.exe", now_id_cmd.Value); + + //运行卸载 + var msi_result = Com_ExeOS.Run.Exe("msiexec.exe", now_id_cmd.Value); + if (msi_result == -920921) + { + throw new Exception(); + } new Log($" 若长时间无 {product_name.Replace("Microsoft Office ", "")} 卸载界面,您可到: 控制面板 -> 程序和功能 列表中手动卸载。", ConsoleColor.Gray); @@ -341,7 +347,7 @@ internal static bool RemovePreviousVersion() catch (Exception Ex) { new Log(Ex.ToString()); - new Log($" × 卸载 Office 早期版本失败!", ConsoleColor.DarkRed); + new Log($" × 卸载 Office 早期版本出现意外!", ConsoleColor.DarkRed); return false; } } @@ -413,6 +419,7 @@ internal static bool BySaRA() catch (Exception Ex) { new Log(Ex.ToString()); + new Log($" × 卸载 Office 冗余版本出现异常!", ConsoleColor.DarkRed); return false; } } @@ -465,14 +472,21 @@ internal static bool ByODT() new Log($" >> 卸载仍在继续,请等待其自动完成 ...", ConsoleColor.DarkYellow); //执行卸载命令 string uninstall_args = $"/configure \"{ODT_path_xml}\""; - bool isUninstall = Com_ExeOS.Run.Exe(ODT_path_exe, uninstall_args); //卸载 + var uninstall_code = Com_ExeOS.Run.Exe(ODT_path_exe, uninstall_args); //卸载 var reg_info = Register.Read.AllValues(RegistryHive.LocalMachine, @"SOFTWARE\Microsoft\Office\ClickToRun\Configuration", "ProductReleaseIds"); - //未正常结束卸载 OR 注册表ODT至少存在1个版本时,视为卸载失败 - if (!isUninstall || (reg_info != null && reg_info.Count > 0)) + //未正常结束卸载,视为卸载失败 + if (uninstall_code == -920921) + { + new Log($" × 无法卸载 Office ODT 版本!", ConsoleColor.DarkRed); + return false; + } + + //注册表ODT至少存在1个版本时,视为卸载失败 + if (reg_info != null && reg_info.Count > 0) { - new Log($" × 卸载 Office ODT 版本失败!", ConsoleColor.DarkRed); + new Log($" × 已尝试卸载 Office ODT 版本,但系统仍存在 {reg_info.Count} 个无法卸载的版本!", ConsoleColor.DarkRed); return false; } @@ -483,6 +497,7 @@ internal static bool ByODT() catch (Exception Ex) { new Log(Ex.ToString()); + new Log($" × 卸载 Office ODT 版本出现异常!", ConsoleColor.DarkRed); return false; } } diff --git a/LKY_OfficeTools/Lib/Lib_OfficeInstall.cs b/LKY_OfficeTools/Lib/Lib_OfficeInstall.cs index 485a9b3..a6ab29c 100644 --- a/LKY_OfficeTools/Lib/Lib_OfficeInstall.cs +++ b/LKY_OfficeTools/Lib/Lib_OfficeInstall.cs @@ -340,10 +340,10 @@ internal static bool StartInstall() ///命令安装 string install_args = $"/configure \"{ODT_path_xml}\""; //配置命令行 - bool isInstallFinish = Run.Exe(ODT_path_exe, install_args); + var install_code = Run.Exe(ODT_path_exe, install_args); //检查是否因配置不正确等导致,意外退出安装 - if (!isInstallFinish) + if (install_code == -920921) { new Log($" × Office v{OfficeNetVersion.latest_version} 安装意外结束!", ConsoleColor.DarkRed); return false; @@ -356,31 +356,6 @@ internal static bool StartInstall() //检查安装是否成功 InstallState install_state = GetOfficeState(); - //未安装 - if (install_state == InstallState.None) - { - new Log($" × 安装失败,未在当前系统检测到任何 Office 版本!", ConsoleColor.DarkRed); - new Log(install_state); //打点失败注册表记录 - return false; - } - - //包含不同版本 - if (install_state == InstallState.Diff) - { - new Log($" × 已安装的 Office 版本与预期的 v{OfficeNetVersion.latest_version} 版本不符!", ConsoleColor.DarkRed); - new Log(install_state); //打点失败注册表记录 - return false; - } - - //包含多个版本 - if (install_state == InstallState.Multi) - { - //系统存在多个版本 - new Log($" × 安装异常,当前系统存在多个 Office 版本!", ConsoleColor.DarkRed); - new Log(install_state); //打点失败注册表记录 - return false; - } - //安装了最新版 if (install_state == InstallState.Correct) { @@ -388,9 +363,39 @@ internal static bool StartInstall() new Log($" √ 已完成 Office v{OfficeNetVersion.latest_version} 安装。", ConsoleColor.DarkGreen); return true; } + else + { + //安装存在问题 + new Log($"Installing Exception, ExitCode: {install_code}"); //回调错误码 + + //未安装 + if (install_state == InstallState.None) + { + new Log($" × 安装失败,未在当前系统检测到任何 Office 版本!", ConsoleColor.DarkRed); + new Log(install_state); //打点失败注册表记录 + return false; + } + + //包含不同版本 + if (install_state == InstallState.Diff) + { + new Log($" × 已安装的 Office 版本与预期的 v{OfficeNetVersion.latest_version} 版本不符!", ConsoleColor.DarkRed); + new Log(install_state); //打点失败注册表记录 + return false; + } + + //包含多个版本 + if (install_state == InstallState.Multi) + { + //系统存在多个版本 + new Log($" × 安装异常,当前系统存在多个 Office 版本!", ConsoleColor.DarkRed); + new Log(install_state); //打点失败注册表记录 + return false; + } - //其它未可知情况,视为失败 - return false; + //其它未可知情况,视为失败 + return false; + } } } } diff --git a/LKY_OfficeTools/OfficeTools.cs b/LKY_OfficeTools/OfficeTools.cs index 3037307..3ed6f39 100644 --- a/LKY_OfficeTools/OfficeTools.cs +++ b/LKY_OfficeTools/OfficeTools.cs @@ -24,7 +24,7 @@ internal class OfficeTools static void Main(string[] args) { //设定编码,解决英文系统乱码问题 - Console.OutputEncoding = Encoding.Unicode; + Console.OutputEncoding = Encoding.UTF8; //命令行检测 new Lib_AppCommand(args);