|
@@ -0,0 +1,423 @@
|
|
|
+using System;
|
|
|
+using System.ComponentModel;
|
|
|
+using System.Runtime.ConstrainedExecution;
|
|
|
+using System.Runtime.InteropServices;
|
|
|
+using System.Text;
|
|
|
+using System.Threading;
|
|
|
+using Microsoft.Win32;
|
|
|
+
|
|
|
+namespace SvdCli.ServiceUtils
|
|
|
+{
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+ public static class ServiceHelper
|
|
|
+ {
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+ public static void Install(string serviceName, string displayName, string binaryFilePath, string description, ServiceStartType startType, ServiceAccount account = ServiceAccount.LocalSystem, string[] dependencies = null)
|
|
|
+ {
|
|
|
+ IntPtr scm = OpenSCManager();
|
|
|
+
|
|
|
+ IntPtr service = IntPtr.Zero;
|
|
|
+ try
|
|
|
+ {
|
|
|
+ service = Win32Class.CreateService(scm, serviceName, displayName, Win32Class.SERVICE_ALL_ACCESS, Win32Class.SERVICE_WIN32_OWN_PROCESS, startType, Win32Class.SERVICE_ERROR_NORMAL, binaryFilePath, null, IntPtr.Zero, ProcessDependencies(dependencies), GetServiceAccountName(account), null);
|
|
|
+
|
|
|
+ if (service == IntPtr.Zero)
|
|
|
+ {
|
|
|
+ if (Marshal.GetLastWin32Error() == 0x431)
|
|
|
+ { throw new ApplicationException("服务已存在!"); }
|
|
|
+
|
|
|
+ throw new ApplicationException("服务安装失败!");
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ Win32Class.SERVICE_DESCRIPTION sd = new Win32Class.SERVICE_DESCRIPTION();
|
|
|
+ try
|
|
|
+ {
|
|
|
+ sd.description = Marshal.StringToHGlobalUni(description);
|
|
|
+ Win32Class.ChangeServiceConfig2(service, 1, ref sd);
|
|
|
+ }
|
|
|
+ finally
|
|
|
+ {
|
|
|
+ Marshal.FreeHGlobal(sd.description);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ finally
|
|
|
+ {
|
|
|
+ if (service != IntPtr.Zero)
|
|
|
+ {
|
|
|
+ Win32Class.CloseServiceHandle(service);
|
|
|
+ }
|
|
|
+ Win32Class.CloseServiceHandle(scm);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ public static string GetBinPath(string serviceName)
|
|
|
+ {
|
|
|
+ var key = Registry.LocalMachine.OpenSubKey($@"SYSTEM\CurrentControlSet\services\{serviceName}");
|
|
|
+
|
|
|
+ return key?.GetValue("ImagePath") == null
|
|
|
+ ? null
|
|
|
+ : key.GetValue("ImagePath").ToString();
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+ public static void Uninstall(string serviceName)
|
|
|
+ {
|
|
|
+ IntPtr scmHandle = IntPtr.Zero;
|
|
|
+ IntPtr service = IntPtr.Zero;
|
|
|
+
|
|
|
+ try
|
|
|
+ {
|
|
|
+ service = OpenService(serviceName, out scmHandle);
|
|
|
+
|
|
|
+ StopService(service);
|
|
|
+
|
|
|
+ if (!Win32Class.DeleteService(service) && Marshal.GetLastWin32Error() != 0x430)
|
|
|
+ {
|
|
|
+ throw new ApplicationException("删除服务失败!");
|
|
|
+ }
|
|
|
+ }
|
|
|
+ catch (ServiceNotExistException) { }
|
|
|
+ finally
|
|
|
+ {
|
|
|
+ if (service != IntPtr.Zero)
|
|
|
+ {
|
|
|
+ Win32Class.CloseServiceHandle(service);
|
|
|
+ Win32Class.CloseServiceHandle(scmHandle);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ private static string GetServiceAccountName(ServiceAccount account)
|
|
|
+ {
|
|
|
+ if (account == ServiceAccount.LocalService)
|
|
|
+ {
|
|
|
+ return @"NT AUTHORITY\LocalService";
|
|
|
+ }
|
|
|
+ if (account == ServiceAccount.NetworkService)
|
|
|
+ {
|
|
|
+ return @"NT AUTHORITY\NetworkService";
|
|
|
+ }
|
|
|
+ return null;
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ private static string ProcessDependencies(string[] dependencies)
|
|
|
+ {
|
|
|
+ if (dependencies == null || dependencies.Length == 0)
|
|
|
+ {
|
|
|
+ return null;
|
|
|
+ }
|
|
|
+
|
|
|
+ StringBuilder sb = new StringBuilder();
|
|
|
+ foreach (string s in dependencies)
|
|
|
+ {
|
|
|
+ sb.Append(s).Append('\0');
|
|
|
+ }
|
|
|
+ sb.Append('\0');
|
|
|
+
|
|
|
+ return sb.ToString();
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ private static IntPtr OpenSCManager()
|
|
|
+ {
|
|
|
+ IntPtr scm = Win32Class.OpenSCManager(null, null, Win32Class.SC_MANAGER_ALL_ACCESS);
|
|
|
+
|
|
|
+ if (scm == IntPtr.Zero)
|
|
|
+ {
|
|
|
+ throw new ApplicationException("打开服务管理器失败!");
|
|
|
+ }
|
|
|
+
|
|
|
+ return scm;
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+ private static IntPtr OpenService(string serviceName, out IntPtr scmHandle)
|
|
|
+ {
|
|
|
+ scmHandle = OpenSCManager();
|
|
|
+
|
|
|
+ IntPtr service = Win32Class.OpenService(scmHandle, serviceName, Win32Class.SERVICE_ALL_ACCESS);
|
|
|
+
|
|
|
+ if (service == IntPtr.Zero)
|
|
|
+ {
|
|
|
+ int errCode = Marshal.GetLastWin32Error();
|
|
|
+
|
|
|
+ Win32Class.CloseServiceHandle(scmHandle);
|
|
|
+
|
|
|
+ if (errCode == 0x424)
|
|
|
+ {
|
|
|
+ throw new ServiceNotExistException();
|
|
|
+ }
|
|
|
+
|
|
|
+ throw new Win32Exception();
|
|
|
+ }
|
|
|
+
|
|
|
+ return service;
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ private static void StopService(IntPtr service)
|
|
|
+ {
|
|
|
+ ServiceState currState = GetServiceStatus(service);
|
|
|
+
|
|
|
+ if (currState == ServiceState.Stopped)
|
|
|
+ {
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (currState != ServiceState.StopPending)
|
|
|
+ {
|
|
|
+
|
|
|
+ string[] childSvs = EnumDependentServices(service, EnumServiceState.Active);
|
|
|
+ if (childSvs.Length != 0)
|
|
|
+ {
|
|
|
+ IntPtr scm = OpenSCManager();
|
|
|
+ try
|
|
|
+ {
|
|
|
+ foreach (string childSv in childSvs)
|
|
|
+ {
|
|
|
+ StopService(Win32Class.OpenService(scm, childSv, Win32Class.SERVICE_STOP));
|
|
|
+ }
|
|
|
+ }
|
|
|
+ finally
|
|
|
+ {
|
|
|
+ Win32Class.CloseServiceHandle(scm);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ Win32Class.SERVICE_STATUS status = new Win32Class.SERVICE_STATUS();
|
|
|
+ Win32Class.ControlService(service, Win32Class.SERVICE_CONTROL_STOP, ref status);
|
|
|
+ }
|
|
|
+
|
|
|
+ if (!WaitForStatus(service, ServiceState.Stopped, new TimeSpan(0, 0, 30)))
|
|
|
+ {
|
|
|
+ throw new ApplicationException("停止服务失败!");
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+ private static string[] EnumDependentServices(IntPtr serviceHandle, EnumServiceState state)
|
|
|
+ {
|
|
|
+ int bytesNeeded = 0;
|
|
|
+ int numEnumerated = 0;
|
|
|
+
|
|
|
+
|
|
|
+ if (Win32Class.EnumDependentServices(serviceHandle, state, IntPtr.Zero, 0, ref bytesNeeded, ref numEnumerated))
|
|
|
+ {
|
|
|
+ return new string[0];
|
|
|
+ }
|
|
|
+ if (Marshal.GetLastWin32Error() != 0xEA)
|
|
|
+ {
|
|
|
+ throw new Win32Exception();
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ IntPtr structsStart = Marshal.AllocHGlobal(new IntPtr(bytesNeeded));
|
|
|
+ try
|
|
|
+ {
|
|
|
+
|
|
|
+ if (!Win32Class.EnumDependentServices(serviceHandle, state, structsStart, bytesNeeded, ref bytesNeeded, ref numEnumerated))
|
|
|
+ {
|
|
|
+ throw new Win32Exception();
|
|
|
+ }
|
|
|
+
|
|
|
+ string[] dependentServices = new string[numEnumerated];
|
|
|
+ int sizeOfStruct = Marshal.SizeOf(typeof(Win32Class.ENUM_SERVICE_STATUS));
|
|
|
+ long structsStartAsInt64 = structsStart.ToInt64();
|
|
|
+ for (int i = 0; i < numEnumerated; i++)
|
|
|
+ {
|
|
|
+ Win32Class.ENUM_SERVICE_STATUS structure = new Win32Class.ENUM_SERVICE_STATUS();
|
|
|
+ IntPtr ptr = new IntPtr(structsStartAsInt64 + i * sizeOfStruct);
|
|
|
+ Marshal.PtrToStructure(ptr, structure);
|
|
|
+ dependentServices[i] = structure.serviceName;
|
|
|
+ }
|
|
|
+
|
|
|
+ return dependentServices;
|
|
|
+ }
|
|
|
+ finally
|
|
|
+ {
|
|
|
+ Marshal.FreeHGlobal(structsStart);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ private static ServiceState GetServiceStatus(IntPtr service)
|
|
|
+ {
|
|
|
+ Win32Class.SERVICE_STATUS status = new Win32Class.SERVICE_STATUS();
|
|
|
+
|
|
|
+ if (!Win32Class.QueryServiceStatus(service, ref status))
|
|
|
+ {
|
|
|
+ throw new ApplicationException("获取服务状态出错!");
|
|
|
+ }
|
|
|
+
|
|
|
+ return status.currentState;
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ private static bool WaitForStatus(IntPtr serviceHandle, ServiceState desiredStatus, TimeSpan timeout)
|
|
|
+ {
|
|
|
+ DateTime startTime = DateTime.Now;
|
|
|
+
|
|
|
+ while (GetServiceStatus(serviceHandle) != desiredStatus)
|
|
|
+ {
|
|
|
+ if (DateTime.Now - startTime > timeout) { return false; }
|
|
|
+
|
|
|
+ Thread.Sleep(200);
|
|
|
+ }
|
|
|
+
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ private static class Win32Class
|
|
|
+ {
|
|
|
+ #region 常量定义
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+ public const int SC_MANAGER_ALL_ACCESS = 0xF003F;
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+ public const int SERVICE_WIN32_OWN_PROCESS = 0x10;
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+ public const int SERVICE_ALL_ACCESS = 0xF01FF;
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+ public const int SERVICE_STOP = 0x20;
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+ public const int SERVICE_CONTROL_STOP = 0x1;
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+ public const int SERVICE_ERROR_NORMAL = 0x1;
|
|
|
+
|
|
|
+ #endregion 常量定义
|
|
|
+
|
|
|
+ #region API所需类和结构定义
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+ [StructLayout(LayoutKind.Sequential)]
|
|
|
+ public struct SERVICE_STATUS
|
|
|
+ {
|
|
|
+ public int serviceType;
|
|
|
+ public ServiceState currentState;
|
|
|
+ public int controlsAccepted;
|
|
|
+ public int win32ExitCode;
|
|
|
+ public int serviceSpecificExitCode;
|
|
|
+ public int checkPoint;
|
|
|
+ public int waitHint;
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+ [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Unicode)]
|
|
|
+ public struct SERVICE_DESCRIPTION
|
|
|
+ {
|
|
|
+ public IntPtr description;
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+ [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Unicode)]
|
|
|
+ public class ENUM_SERVICE_STATUS
|
|
|
+ {
|
|
|
+ public string serviceName;
|
|
|
+ public string displayName;
|
|
|
+ public int serviceType;
|
|
|
+ public int currentState;
|
|
|
+ public int controlsAccepted;
|
|
|
+ public int win32ExitCode;
|
|
|
+ public int serviceSpecificExitCode;
|
|
|
+ public int checkPoint;
|
|
|
+ public int waitHint;
|
|
|
+ }
|
|
|
+
|
|
|
+ #endregion API所需类和结构定义
|
|
|
+
|
|
|
+ #region API定义
|
|
|
+
|
|
|
+ [DllImport("advapi32.dll", CharSet = CharSet.Unicode, SetLastError = true)]
|
|
|
+ public static extern bool ChangeServiceConfig2(IntPtr serviceHandle, uint infoLevel, ref SERVICE_DESCRIPTION serviceDesc);
|
|
|
+
|
|
|
+ [DllImport("advapi32.dll", CharSet = CharSet.Unicode, SetLastError = true)]
|
|
|
+ public static extern IntPtr OpenSCManager(string machineName, string databaseName, int dwDesiredAccess);
|
|
|
+
|
|
|
+ [DllImport("advapi32.dll", SetLastError = true, CharSet = CharSet.Auto)]
|
|
|
+ public static extern IntPtr OpenService(IntPtr hSCManager, string lpServiceName, int dwDesiredAccess);
|
|
|
+
|
|
|
+ [DllImport("advapi32.dll", CharSet = CharSet.Unicode, SetLastError = true)]
|
|
|
+ public static extern IntPtr CreateService(IntPtr hSCManager, string lpServiceName, string lpDisplayName, int dwDesiredAccess, int dwServiceType, ServiceStartType dwStartType, int dwErrorControl, string lpBinaryPathName, string lpLoadOrderGroup, IntPtr lpdwTagId, string lpDependencies, string lpServiceStartName, string lpPassword);
|
|
|
+
|
|
|
+ [ReliabilityContract(Consistency.WillNotCorruptState, Cer.Success), DllImport("advapi32.dll", CharSet = CharSet.Unicode, SetLastError = true)]
|
|
|
+ public static extern bool CloseServiceHandle(IntPtr handle);
|
|
|
+
|
|
|
+ [DllImport("advapi32.dll", CharSet = CharSet.Unicode, SetLastError = true)]
|
|
|
+ public static extern bool QueryServiceStatus(IntPtr hService, ref SERVICE_STATUS lpServiceStatus);
|
|
|
+
|
|
|
+ [DllImport("advapi32.dll", CharSet = CharSet.Unicode, SetLastError = true)]
|
|
|
+ public static extern bool DeleteService(IntPtr serviceHandle);
|
|
|
+
|
|
|
+ [DllImport("advapi32.dll", CharSet = CharSet.Unicode, SetLastError = true)]
|
|
|
+ public static extern bool ControlService(IntPtr hService, int dwControl, ref SERVICE_STATUS lpServiceStatus);
|
|
|
+
|
|
|
+ [DllImport("advapi32.dll", CharSet = CharSet.Unicode, SetLastError = true)]
|
|
|
+ public static extern bool EnumDependentServices(IntPtr serviceHandle, EnumServiceState serviceState, IntPtr bufferOfENUM_SERVICE_STATUS, int bufSize, ref int bytesNeeded, ref int numEnumerated);
|
|
|
+
|
|
|
+ #endregion API定义
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ private enum EnumServiceState
|
|
|
+ {
|
|
|
+ Active = 1,
|
|
|
+
|
|
|
+
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ private enum ServiceState
|
|
|
+ {
|
|
|
+ Stopped = 1,
|
|
|
+
|
|
|
+
|
|
|
+ StopPending = 3,
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|