DeviceManager.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289
  1. using FISLib.Hardware;
  2. using System;
  3. using System.Collections.Generic;
  4. using System.Linq;
  5. using System.Management;
  6. using System.Net.NetworkInformation;
  7. using System.Threading;
  8. using System.Threading.Tasks;
  9. using Vinno.FIS.Sonopost.Managers.Interfaces;
  10. using Vinno.FIS.Sonopost.Settings;
  11. using Vinno.IUS.Common.Log;
  12. namespace Vinno.FIS.Sonopost.Managers
  13. {
  14. internal class DeviceManager : SonopostManager, IDeviceManager
  15. {
  16. private const string Scope = "root\\CIMV2";
  17. private const string Win32USBHubQuery = "SELECT * FROM {0} WITHIN 2 WHERE TargetInstance ISA 'Win32_USBHub'";
  18. private const string Win32_USBControllerDeviceQuery = "SELECT * FROM {0} WITHIN 2 WHERE TargetInstance ISA 'Win32_USBControllerDevice'";
  19. private const string CreationEvent = "__InstanceCreationEvent";
  20. private const string DeletionEvent = "__InstanceDeletionEvent";
  21. private readonly IHardwareService _fisHardwareService;
  22. private readonly List<FISCameraInfo> _cameraInfos;
  23. private readonly List<FISMicInfo> _micInfos;
  24. private readonly object _deviceLocker = new object();
  25. private readonly object _locker = new object();
  26. private bool _networkChanged;
  27. private ManagementEventWatcher _wifiModuleInsertWatcher;
  28. private ManagementEventWatcher _wifiModuleRemoveWatcher;
  29. private ManagementEventWatcher _usbInsertWatcher;
  30. private ManagementEventWatcher _usbRemoveWatcher;
  31. /// <summary>
  32. /// 当WIFI 模块插入时触发
  33. /// </summary>
  34. public event EventHandler<EventArrivedEventArgs> WifiModuleInserted;
  35. /// <summary>
  36. /// 当U盘设备插入时触发
  37. /// </summary>
  38. public event EventHandler<string> UDiskInserted;
  39. /// <summary>
  40. /// 当网络地址变化时触发
  41. /// </summary>
  42. public event EventHandler NetworkAddressChanged;
  43. public DeviceManager()
  44. {
  45. _cameraInfos = new List<FISCameraInfo>();
  46. _micInfos = new List<FISMicInfo>();
  47. _fisHardwareService = AppManager.Instance.GetManager<IFISManager>().FISHardwareService;
  48. RefreshCameras();
  49. }
  50. public void Init()
  51. {
  52. var scope = new ManagementScope(Scope);
  53. scope.Options.EnablePrivileges = true;
  54. _wifiModuleInsertWatcher = new ManagementEventWatcher(scope, new WqlEventQuery(string.Format(Win32_USBControllerDeviceQuery, CreationEvent)));
  55. _wifiModuleInsertWatcher.EventArrived += OnWifiModuleInserted;
  56. _wifiModuleInsertWatcher.Start();
  57. _wifiModuleRemoveWatcher = new ManagementEventWatcher(scope, new WqlEventQuery(string.Format(Win32_USBControllerDeviceQuery, DeletionEvent)));
  58. _wifiModuleRemoveWatcher.EventArrived += OnWifiModuleRemoved;
  59. _wifiModuleRemoveWatcher.Start();
  60. _usbInsertWatcher = new ManagementEventWatcher(scope, new WqlEventQuery(string.Format(Win32USBHubQuery, CreationEvent)));
  61. _usbInsertWatcher.EventArrived += OnUSBInserted;
  62. _usbInsertWatcher.Start();
  63. _usbRemoveWatcher = new ManagementEventWatcher(scope, new WqlEventQuery(string.Format(Win32USBHubQuery, DeletionEvent)));
  64. _usbRemoveWatcher.EventArrived += OnUSBRemoved;
  65. _usbRemoveWatcher.Start();
  66. NetworkChange.NetworkAddressChanged += OnNetworkAddressChanged;
  67. Logger.WriteLineInfo($"Device Manager Init Invoke");
  68. }
  69. private void OnNetworkAddressChanged(object sender, EventArgs e)
  70. {
  71. lock (_deviceLocker)
  72. {
  73. if (_networkChanged)
  74. {
  75. return;
  76. }
  77. _networkChanged = true;
  78. Task.Run(() =>
  79. {
  80. Thread.Sleep(1000);
  81. Logger.WriteLineInfo($"Device Manager OnNetworkAddressChanged Invoke");
  82. NetworkAddressChanged?.Invoke(this, e);
  83. _networkChanged = false;
  84. });
  85. }
  86. }
  87. private void OnWifiModuleInserted(object sender, EventArrivedEventArgs e)
  88. {
  89. lock (_deviceLocker)
  90. {
  91. try
  92. {
  93. var instance = (ManagementBaseObject)e.NewEvent["TargetInstance"];
  94. var dependent = (instance["Dependent"] as String).Replace("\"", String.Empty).Split(new Char[] { '=' })[1].Replace("\\\\", "\\");
  95. if (dependent.ToLower().Contains("vwifimp"))
  96. {
  97. Logger.WriteLineInfo($"Device Manager:Wifi Module Inserted , Dependent:{dependent}");
  98. WifiModuleInserted?.Invoke(this, e);
  99. }
  100. }
  101. catch (Exception ex)
  102. {
  103. Logger.WriteLineError($"Device Manager OnWifiModuleInserted Error:{ex}");
  104. }
  105. }
  106. }
  107. private void OnWifiModuleRemoved(object sender, EventArrivedEventArgs e)
  108. {
  109. lock (_deviceLocker)
  110. {
  111. try
  112. {
  113. var instance = (ManagementBaseObject)e.NewEvent["TargetInstance"];
  114. var dependent = (instance["Dependent"] as String).Replace("\"", String.Empty).Split(new Char[] { '=' })[1].Replace("\\\\", "\\");
  115. if (dependent.ToLower().Contains("vwifimp"))
  116. {
  117. Logger.WriteLineInfo($"Device Manager : Wifi Module Removed , Dependent:{dependent}");
  118. }
  119. }
  120. catch (Exception ex)
  121. {
  122. Logger.WriteLineError($"Device Manager OnWifiModuleRemoved Error:{ex}");
  123. }
  124. }
  125. }
  126. /// <summary>
  127. /// 刷新摄像头
  128. /// </summary>
  129. public void RefreshCameras()
  130. {
  131. var camearaInfos = _fisHardwareService.GetCameraList();
  132. lock (_locker)
  133. {
  134. _cameraInfos.Clear();
  135. _cameraInfos.AddRange(camearaInfos);
  136. Logger.WriteLineInfo($"RefreshCameras:{_cameraInfos.Count}");
  137. }
  138. }
  139. public List<FISCameraInfo> GetCameras()
  140. {
  141. lock (_locker)
  142. {
  143. return _cameraInfos;
  144. }
  145. }
  146. /// <summary>
  147. /// 刷新摄像头
  148. /// </summary>
  149. public void RefreshMics()
  150. {
  151. var micInfos = _fisHardwareService.GetMicList().Where(x => !SonopostSystemSettings.Instance.SkippedMicDeviceNames.Contains(x.Name));
  152. lock (_locker)
  153. {
  154. _micInfos.Clear();
  155. _micInfos.AddRange(micInfos);
  156. Logger.WriteLineInfo($"RefreshMics:{_micInfos.Count}");
  157. }
  158. }
  159. public List<FISMicInfo> GetMics()
  160. {
  161. lock (_locker)
  162. {
  163. return _micInfos;
  164. }
  165. }
  166. private void OnUSBInserted(object sender, EventArrivedEventArgs e)
  167. {
  168. try
  169. {
  170. var instance = (ManagementBaseObject)e.NewEvent["TargetInstance"];
  171. var description = (string)instance.GetPropertyValue("Description");
  172. var deviceId = (string)instance.GetPropertyValue("DeviceID");
  173. Logger.WriteLineInfo($"Device Manager:UDisk Insert:{description},{deviceId}");
  174. var serialNum = deviceId.Substring(deviceId.LastIndexOf('\\')).Replace("\\", "");
  175. var letter = GetDriveLetter(serialNum);
  176. if (!string.IsNullOrEmpty(letter))
  177. {
  178. UDiskInserted?.Invoke(this, letter);
  179. }
  180. }
  181. catch (Exception ex)
  182. {
  183. Logger.WriteLineError($"Device Manager UDisk insert Error:{ex}");
  184. }
  185. }
  186. private void OnUSBRemoved(object sender, EventArrivedEventArgs e)
  187. {
  188. try
  189. {
  190. var instance = (ManagementBaseObject)e.NewEvent["TargetInstance"];
  191. var description = (string)instance.GetPropertyValue("Description");
  192. var deviceId = (string)instance.GetPropertyValue("DeviceID");
  193. Logger.WriteLineInfo($"Device Manager:USB Remove:{description},{deviceId}");
  194. }
  195. catch (Exception ex)
  196. {
  197. Logger.WriteLineError($"Device Manager USB Remove Error:{ex}");
  198. }
  199. }
  200. public override void DoDispose()
  201. {
  202. try
  203. {
  204. _cameraInfos.Clear();
  205. NetworkChange.NetworkAddressChanged -= OnNetworkAddressChanged;
  206. if (_wifiModuleInsertWatcher != null)
  207. {
  208. _wifiModuleInsertWatcher.EventArrived -= OnWifiModuleInserted;
  209. _wifiModuleInsertWatcher.Stop();
  210. _wifiModuleInsertWatcher = null;
  211. }
  212. if (_wifiModuleRemoveWatcher != null)
  213. {
  214. _wifiModuleRemoveWatcher.EventArrived -= OnWifiModuleInserted;
  215. _wifiModuleRemoveWatcher.Stop();
  216. _wifiModuleRemoveWatcher = null;
  217. }
  218. if (_usbInsertWatcher != null)
  219. {
  220. _usbInsertWatcher.EventArrived -= OnUSBInserted;
  221. _usbInsertWatcher.Stop();
  222. _usbInsertWatcher = null;
  223. }
  224. if (_usbRemoveWatcher != null)
  225. {
  226. _usbRemoveWatcher.EventArrived -= OnUSBRemoved;
  227. _usbRemoveWatcher.Stop();
  228. _usbRemoveWatcher = null;
  229. }
  230. }
  231. catch (Exception ex)
  232. {
  233. Logger.WriteLineError($"DeviceManager DoDispose Error:{ex}");
  234. }
  235. base.DoDispose();
  236. }
  237. private string GetDriveLetter(string serialNum)
  238. {
  239. var devices = SelectDevices();
  240. foreach (var device in devices)
  241. {
  242. var deviceId = (string)device.GetPropertyValue("DeviceID");
  243. var pnpDeviceId = (string)device.GetPropertyValue("PNPDeviceID");
  244. var serialNumber = pnpDeviceId.Remove(pnpDeviceId.Length - 2).Substring(pnpDeviceId.LastIndexOf('\\')).Replace("\\", "");
  245. if (serialNumber.Contains(serialNumber))
  246. {
  247. return (string)SelectPartitions(device).SelectMany(SelectDisks).Select(disk => disk["Name"]).Single();
  248. }
  249. }
  250. return string.Empty;
  251. }
  252. private IEnumerable<ManagementObject> SelectDevices()
  253. {
  254. return new ManagementObjectSearcher(@"SELECT * FROM Win32_DiskDrive WHERE InterfaceType LIKE 'USB%'").Get().Cast<ManagementObject>();
  255. }
  256. public IEnumerable<ManagementObject> SelectPartitions(ManagementObject device)
  257. {
  258. var deviceId = device.Properties["DeviceID"].Value;
  259. return new ManagementObjectSearcher("ASSOCIATORS OF {Win32_DiskDrive.DeviceID='" + deviceId + "'} WHERE AssocClass = Win32_DiskDriveToDiskPartition").Get().Cast<ManagementObject>();
  260. }
  261. public IEnumerable<ManagementObject> SelectDisks(ManagementObject partition)
  262. {
  263. var deviceId = partition["DeviceID"];
  264. return new ManagementObjectSearcher("ASSOCIATORS OF {Win32_DiskPartition.DeviceID='" + deviceId + "'} WHERE AssocClass = Win32_LogicalDiskToPartition").Get().Cast<ManagementObject>();
  265. }
  266. }
  267. }