Files
CarlMonitor/LibreHardwareMonitor-0.9.4/LibreHardwareMonitor/Utilities/Logger.cs
2025-04-07 07:44:27 -07:00

227 lines
7.0 KiB
C#

// This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
// If a copy of the MPL was not distributed with this file, You can obtain one at http://mozilla.org/MPL/2.0/.
// Copyright (C) LibreHardwareMonitor and Contributors.
// Partial Copyright (C) Michael Möller <mmoeller@openhardwaremonitor.org> and Contributors.
// All Rights Reserved.
using System;
using System.Collections.Generic;
using System.Globalization;
using System.IO;
using System.Linq;
using LibreHardwareMonitor.Hardware;
namespace LibreHardwareMonitor.Utilities;
public class Logger
{
private const string FileNameFormat = "LibreHardwareMonitorLog-{0:yyyy-MM-dd}{1}.csv";
private readonly IComputer _computer;
private DateTime _day = DateTime.MinValue;
private string _fileName;
private string[] _identifiers;
private ISensor[] _sensors;
private DateTime _lastLoggedTime = DateTime.MinValue;
public LoggerFileRotation FileRotationMethod = LoggerFileRotation.PerSession;
public Logger(IComputer computer)
{
_computer = computer;
_computer.HardwareAdded += HardwareAdded;
_computer.HardwareRemoved += HardwareRemoved;
}
private void HardwareRemoved(IHardware hardware)
{
hardware.SensorAdded -= SensorAdded;
hardware.SensorRemoved -= SensorRemoved;
foreach (ISensor sensor in hardware.Sensors)
SensorRemoved(sensor);
foreach (IHardware subHardware in hardware.SubHardware)
HardwareRemoved(subHardware);
}
private void HardwareAdded(IHardware hardware)
{
foreach (ISensor sensor in hardware.Sensors)
SensorAdded(sensor);
hardware.SensorAdded += SensorAdded;
hardware.SensorRemoved += SensorRemoved;
foreach (IHardware subHardware in hardware.SubHardware)
HardwareAdded(subHardware);
}
private void SensorAdded(ISensor sensor)
{
if (_sensors == null)
return;
for (int i = 0; i < _sensors.Length; i++)
{
if (sensor.Identifier.ToString() == _identifiers[i])
_sensors[i] = sensor;
}
}
private void SensorRemoved(ISensor sensor)
{
if (_sensors == null)
return;
for (int i = 0; i < _sensors.Length; i++)
{
if (sensor == _sensors[i])
_sensors[i] = null;
}
}
private static string GetFileName(DateTime date, uint sessionNumber = 0)
{
return AppDomain.CurrentDomain.BaseDirectory + Path.DirectorySeparatorChar
+ string.Format(FileNameFormat, date, sessionNumber == 0 ? "" : "-" + sessionNumber);
}
private bool OpenExistingLogFile()
{
if (!File.Exists(_fileName))
return false;
try
{
string line;
using (StreamReader reader = new StreamReader(_fileName))
line = reader.ReadLine();
if (string.IsNullOrEmpty(line))
return false;
_identifiers = line.Split(',').Skip(1).ToArray();
}
catch
{
_identifiers = null;
return false;
}
if (_identifiers.Length == 0)
{
_identifiers = null;
return false;
}
_sensors = new ISensor[_identifiers.Length];
SensorVisitor visitor = new SensorVisitor(sensor =>
{
for (int i = 0; i < _identifiers.Length; i++)
if (sensor.Identifier.ToString() == _identifiers[i])
_sensors[i] = sensor;
});
visitor.VisitComputer(_computer);
return true;
}
private void CreateNewLogFile()
{
IList<ISensor> list = new List<ISensor>();
SensorVisitor visitor = new SensorVisitor(sensor =>
{
list.Add(sensor);
});
visitor.VisitComputer(_computer);
_sensors = list.ToArray();
_identifiers = _sensors.Select(s => s.Identifier.ToString()).ToArray();
using (StreamWriter writer = new StreamWriter(_fileName, false))
{
writer.Write(",");
for (int i = 0; i < _sensors.Length; i++)
{
writer.Write(_sensors[i].Identifier);
if (i < _sensors.Length - 1)
writer.Write(",");
else
writer.WriteLine();
}
writer.Write("Time,");
for (int i = 0; i < _sensors.Length; i++)
{
writer.Write('"');
writer.Write(_sensors[i].Name);
writer.Write('"');
if (i < _sensors.Length - 1)
writer.Write(",");
else
writer.WriteLine();
}
}
}
public TimeSpan LoggingInterval { get; set; }
public void Log()
{
DateTime now = DateTime.Now;
if (_lastLoggedTime + LoggingInterval - new TimeSpan(5000000) > now)
return;
switch (FileRotationMethod)
{
case LoggerFileRotation.PerSession:
// Create file if it does not exist or the logging interval has passed (+ some margin)
if (!File.Exists(_fileName) || now - _lastLoggedTime > (LoggingInterval + TimeSpan.FromMilliseconds(100)))
{
uint sessionNumber = 1;
do {
_fileName = GetFileName(DateTime.Now, sessionNumber);
sessionNumber++;
} while (File.Exists(_fileName));
CreateNewLogFile();
}
break;
case LoggerFileRotation.Daily:
// Create a new file if the day has changed or the file does not exist
if (_day != now.Date || !File.Exists(_fileName))
{
_day = now.Date;
_fileName = GetFileName(_day);
if (!OpenExistingLogFile())
CreateNewLogFile();
}
break;
}
try
{
using (StreamWriter writer = new StreamWriter(new FileStream(_fileName, FileMode.Append, FileAccess.Write, FileShare.ReadWrite)))
{
writer.Write(now.ToString("G", CultureInfo.InvariantCulture));
writer.Write(",");
for (int i = 0; i < _sensors.Length; i++)
{
if (_sensors[i] != null)
{
float? value = _sensors[i].Value;
if (value.HasValue)
writer.Write(value.Value.ToString("R", CultureInfo.InvariantCulture));
}
if (i < _sensors.Length - 1)
writer.Write(",");
else
writer.WriteLine();
}
}
}
catch (IOException) { }
_lastLoggedTime = now;
}
}