1
0

SensorCollector.cs 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185
  1. using System.Collections.Concurrent;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using NLog;
  5. using LibreHardwareMonitor.Hardware;
  6. using LibreHardwareMonitor.Hardware.Storage;
  7. namespace OhmGraphite
  8. {
  9. public class SensorCollector : IGiveSensors
  10. {
  11. private static readonly Logger Logger = LogManager.GetCurrentClassLogger();
  12. private readonly Computer _computer;
  13. private readonly MetricConfig _config;
  14. private readonly IVisitor _updateVisitor = new UpdateVisitor();
  15. private readonly ConcurrentDictionary<Identifier, object> _ids =
  16. new ConcurrentDictionary<Identifier, object>();
  17. private readonly ConcurrentDictionary<Identifier, OhmNvme> _nvmes =
  18. new ConcurrentDictionary<Identifier, OhmNvme>();
  19. public SensorCollector(Computer computer, MetricConfig config)
  20. {
  21. _computer = computer;
  22. _config = config;
  23. }
  24. public void Open()
  25. {
  26. foreach (var hardware in _computer.Hardware)
  27. {
  28. HardwareAdded(hardware);
  29. }
  30. _computer.HardwareAdded += HardwareAdded;
  31. _computer.HardwareRemoved += HardwareRemoved;
  32. _computer.Open();
  33. }
  34. public void Close()
  35. {
  36. _computer.HardwareRemoved -= HardwareRemoved;
  37. _computer.HardwareAdded -= HardwareAdded;
  38. foreach (var hardware in _computer.Hardware)
  39. {
  40. HardwareRemoved(hardware);
  41. }
  42. _computer.Close();
  43. }
  44. public void Start() => Open();
  45. public void Dispose() => Close();
  46. private void HardwareAdded(IHardware hardware)
  47. {
  48. if (!_ids.TryAdd(hardware.Identifier, hardware))
  49. {
  50. Logger.Debug("Hardware previously added: {0}", hardware.Identifier);
  51. return;
  52. }
  53. Logger.Debug("Hardware added: {0}", hardware.Identifier);
  54. hardware.SensorAdded += SensorAdded;
  55. hardware.SensorRemoved += SensorRemoved;
  56. foreach (var sensor in hardware.Sensors)
  57. {
  58. SensorAdded(sensor);
  59. }
  60. if (hardware is NVMeGeneric nvme)
  61. {
  62. var ohmNvme = new OhmNvme(nvme);
  63. _nvmes.TryAdd(hardware.Identifier, ohmNvme);
  64. SensorAdded(ohmNvme.MediaErrors);
  65. SensorAdded(ohmNvme.PowerCycles);
  66. SensorAdded(ohmNvme.ErrorInfoLogEntryCount);
  67. SensorAdded(ohmNvme.UnsafeShutdowns);
  68. }
  69. foreach (var sub in hardware.SubHardware)
  70. {
  71. HardwareAdded(sub);
  72. }
  73. }
  74. private void HardwareRemoved(IHardware hardware)
  75. {
  76. _ids.TryRemove(hardware.Identifier, out _);
  77. Logger.Debug("Hardware removed: {0}", hardware.Identifier);
  78. hardware.SensorAdded -= SensorAdded;
  79. hardware.SensorRemoved -= SensorRemoved;
  80. foreach (var sensor in hardware.Sensors)
  81. {
  82. SensorRemoved(sensor);
  83. }
  84. if (_nvmes.TryRemove(hardware.Identifier, out OhmNvme ohmNvme))
  85. {
  86. SensorRemoved(ohmNvme.MediaErrors);
  87. SensorRemoved(ohmNvme.PowerCycles);
  88. SensorRemoved(ohmNvme.ErrorInfoLogEntryCount);
  89. SensorRemoved(ohmNvme.UnsafeShutdowns);
  90. }
  91. foreach (var sub in hardware.SubHardware)
  92. {
  93. HardwareRemoved(sub);
  94. }
  95. }
  96. private void SensorAdded(ISensor sensor)
  97. {
  98. var added = _ids.TryAdd(sensor.Identifier, sensor);
  99. var msg = added ? "Sensor added: {0} \"{1}\"" : "Sensor previously added: {0} \"{1}\"";
  100. Logger.Info(msg, sensor.Identifier, SensorName(sensor));
  101. }
  102. private void SensorRemoved(ISensor sensor)
  103. {
  104. Logger.Debug("Sensor removed: {0}", sensor.Identifier);
  105. _ids.TryRemove(sensor.Identifier, out _);
  106. }
  107. public IEnumerable<ReportedValue> ReadAllSensors()
  108. {
  109. _computer.Accept(_updateVisitor);
  110. foreach (var nvme in _nvmes.Values)
  111. {
  112. nvme.Update();
  113. }
  114. return _ids.Values.OfType<ISensor>().SelectMany(ReportedValues);
  115. }
  116. private string SensorName(ISensor sensor)
  117. {
  118. // Remove hardware index indentifier for motherboards to preserve
  119. // librehardwaremonitor pre-0.9.3 behavior
  120. // https://github.com/nickbabcock/OhmGraphite/pull/433
  121. return sensor.Hardware.HardwareType == LibreHardwareMonitor.Hardware.HardwareType.Motherboard
  122. ? string.Join("/", sensor.Name.Split('/').Where((x, i) => i != 2))
  123. : sensor.Name;
  124. }
  125. private IEnumerable<ReportedValue> ReportedValues(ISensor sensor)
  126. {
  127. string sensorName = SensorName(sensor);
  128. string id = sensor.Identifier.ToString();
  129. // Only report a value if the sensor was able to get a value
  130. // as 0 is different than "didn't read". For example, are the
  131. // fans really spinning at 0 RPM or was the value not read.
  132. if (!sensor.Value.HasValue)
  133. {
  134. Logger.Debug($"{id} did not have a value");
  135. }
  136. else if (float.IsNaN(sensor.Value.Value))
  137. {
  138. Logger.Debug($"{id} had a NaN value");
  139. }
  140. else if (float.IsInfinity(sensor.Value.Value))
  141. {
  142. Logger.Debug($"{id} had an infinite value");
  143. }
  144. else if (!_config.IsHidden(sensor.Identifier.ToString()) && !_config.IsHidden(sensorName))
  145. {
  146. var hwInstance = sensor.Hardware.Identifier.ToString();
  147. var ind = hwInstance.LastIndexOf('/');
  148. hwInstance = hwInstance.Substring(ind + 1);
  149. var name = _config.TryGetAlias(sensor.Identifier.ToString(), out string alias) ? alias : sensorName;
  150. yield return new ReportedValue(id,
  151. name,
  152. sensor.Value.Value,
  153. sensor.SensorType.ToOwnSensor(),
  154. sensor.Hardware.Name,
  155. sensor.Hardware.HardwareType.ToOwnHardware(),
  156. hwInstance,
  157. sensor.Index);
  158. }
  159. }
  160. }
  161. }