From 1cc5bf3fd7965b785902a6c34ae4a9262365eb51 Mon Sep 17 00:00:00 2001 From: lltcggie Date: Sun, 30 Apr 2023 00:39:24 +0900 Subject: [PATCH] v1.0.0 --- GPUTemperatureMonitoring.sln | 25 ++ GPUTemperatureMonitoring/App.config | 14 + .../GPUTemperatureMonitoring.csproj | 124 +++++++ GPUTemperatureMonitoring/Icon.ico | Bin 0 -> 45451 bytes GPUTemperatureMonitoring/Program.cs | 317 ++++++++++++++++++ .../Properties/AssemblyInfo.cs | 36 ++ .../Properties/Resources.Designer.cs | 71 ++++ .../Properties/Resources.resx | 117 +++++++ .../Properties/Settings.Designer.cs | 30 ++ .../Properties/Settings.settings | 7 + GPUTemperatureMonitoring/app.manifest | 64 ++++ Setting.json | 7 + 12 files changed, 812 insertions(+) create mode 100644 GPUTemperatureMonitoring.sln create mode 100644 GPUTemperatureMonitoring/App.config create mode 100644 GPUTemperatureMonitoring/GPUTemperatureMonitoring.csproj create mode 100644 GPUTemperatureMonitoring/Icon.ico create mode 100644 GPUTemperatureMonitoring/Program.cs create mode 100644 GPUTemperatureMonitoring/Properties/AssemblyInfo.cs create mode 100644 GPUTemperatureMonitoring/Properties/Resources.Designer.cs create mode 100644 GPUTemperatureMonitoring/Properties/Resources.resx create mode 100644 GPUTemperatureMonitoring/Properties/Settings.Designer.cs create mode 100644 GPUTemperatureMonitoring/Properties/Settings.settings create mode 100644 GPUTemperatureMonitoring/app.manifest create mode 100644 Setting.json diff --git a/GPUTemperatureMonitoring.sln b/GPUTemperatureMonitoring.sln new file mode 100644 index 0000000..3a19118 --- /dev/null +++ b/GPUTemperatureMonitoring.sln @@ -0,0 +1,25 @@ + +Microsoft Visual Studio Solution File, Format Version 12.00 +# Visual Studio Version 17 +VisualStudioVersion = 17.5.33530.505 +MinimumVisualStudioVersion = 10.0.40219.1 +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "GPUTemperatureMonitoring", "GPUTemperatureMonitoring\GPUTemperatureMonitoring.csproj", "{E1D50143-93C5-4585-8516-A3C2B465A444}" +EndProject +Global + GlobalSection(SolutionConfigurationPlatforms) = preSolution + Debug|x64 = Debug|x64 + Release|x64 = Release|x64 + EndGlobalSection + GlobalSection(ProjectConfigurationPlatforms) = postSolution + {E1D50143-93C5-4585-8516-A3C2B465A444}.Debug|x64.ActiveCfg = Debug|x64 + {E1D50143-93C5-4585-8516-A3C2B465A444}.Debug|x64.Build.0 = Debug|x64 + {E1D50143-93C5-4585-8516-A3C2B465A444}.Release|x64.ActiveCfg = Release|x64 + {E1D50143-93C5-4585-8516-A3C2B465A444}.Release|x64.Build.0 = Release|x64 + EndGlobalSection + GlobalSection(SolutionProperties) = preSolution + HideSolutionNode = FALSE + EndGlobalSection + GlobalSection(ExtensibilityGlobals) = postSolution + SolutionGuid = {3B80A9E7-0F7F-4B03-9113-38A67896EEBB} + EndGlobalSection +EndGlobal diff --git a/GPUTemperatureMonitoring/App.config b/GPUTemperatureMonitoring/App.config new file mode 100644 index 0000000..1afacce --- /dev/null +++ b/GPUTemperatureMonitoring/App.config @@ -0,0 +1,14 @@ + + + + + + + + + + + + + + \ No newline at end of file diff --git a/GPUTemperatureMonitoring/GPUTemperatureMonitoring.csproj b/GPUTemperatureMonitoring/GPUTemperatureMonitoring.csproj new file mode 100644 index 0000000..4e9eca6 --- /dev/null +++ b/GPUTemperatureMonitoring/GPUTemperatureMonitoring.csproj @@ -0,0 +1,124 @@ + + + + + Debug + AnyCPU + {E1D50143-93C5-4585-8516-A3C2B465A444} + WinExe + GPUTemperatureMonitoring + GPUTemperatureMonitoring + v4.7.2 + 512 + true + true + + + + + app.manifest + + + true + bin\x64\Debug\ + DEBUG;TRACE + full + x64 + 7.3 + prompt + true + + + bin\x64\Release\ + TRACE + true + pdbonly + x64 + 7.3 + prompt + true + + + Icon.ico + + + false + + + false + + + LocalIntranet + + + + + + + + + + + + + + + + + + + + + + ResXFileCodeGenerator + Resources.Designer.cs + Designer + + + True + Resources.resx + + + + SettingsSingleFileGenerator + Settings.Designer.cs + + + True + Settings.settings + True + + + + + + + + PreserveNewest + + + + + 0.9.2 + + + 7.1.3 + + + 7.0.2 + + + + + False + Microsoft .NET Framework 4.7.2 %28x86 および x64%29 + true + + + False + .NET Framework 3.5 SP1 + false + + + + \ No newline at end of file diff --git a/GPUTemperatureMonitoring/Icon.ico b/GPUTemperatureMonitoring/Icon.ico new file mode 100644 index 0000000000000000000000000000000000000000..5d06b9f2857b39f0b5d3395e3e7999ba6bcc3a38 GIT binary patch literal 45451 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 literal 0 HcmV?d00001 diff --git a/GPUTemperatureMonitoring/Program.cs b/GPUTemperatureMonitoring/Program.cs new file mode 100644 index 0000000..57c15f1 --- /dev/null +++ b/GPUTemperatureMonitoring/Program.cs @@ -0,0 +1,317 @@ +using LibreHardwareMonitor.Hardware; +using LibreHardwareMonitor.Hardware.Cpu; +using Microsoft.Toolkit.Uwp.Notifications; +using System; +using System.Collections.Generic; +using System.Diagnostics; +using System.Drawing; +using System.IO; +using System.Linq; +using System.Net; +using System.Text; +using System.Text.Json; +using System.Threading.Tasks; +using System.Timers; +using System.Web; +using System.Windows.Forms; +using static System.Windows.Forms.VisualStyles.VisualStyleElement; + + +namespace GPUTemperatureMonitoring +{ + public class MonitoringSetting + { + public string SensorPath { get; set; } = string.Empty; + public string LINENotifyToken { get; set; } = string.Empty; + public int MonitoringIntervalMS { get; set; } = 0; + public int TemperatureThreshold { get; set; } = 0; + public int FailedNotifyIntervalS { get; set; } = 0; + + public MonitoringSetting(string SensorPath, string LINENotifyToken, int MonitoringIntervalMS, int TemperatureThreshold, int FailedNotifyIntervalS) + { + this.SensorPath = SensorPath; + this.LINENotifyToken = LINENotifyToken; + this.MonitoringIntervalMS = MonitoringIntervalMS; + this.TemperatureThreshold = TemperatureThreshold; + this.FailedNotifyIntervalS = FailedNotifyIntervalS; + } + } + + internal class TemperatureSensor : IDisposable + { + class UpdateVisitor : IVisitor + { + public void VisitComputer(IComputer computer) + { + computer.Traverse(this); + } + public void VisitHardware(IHardware hardware) + { + hardware.Update(); + foreach (IHardware subHardware in hardware.SubHardware) subHardware.Accept(this); + } + public void VisitSensor(ISensor sensor) { } + public void VisitParameter(IParameter parameter) { } + } + + private Computer mComputer; + private string mSensorPath; + + public TemperatureSensor(string sensorPath) + { + this.mSensorPath = sensorPath; + + mComputer = new Computer + { + IsGpuEnabled = true, + }; + mComputer.Open(); + mComputer.Accept(new UpdateVisitor()); + } + + public void Dispose() + { + mComputer.Close(); + } + + public float GetTemperature() + { + float temperature = -1.0f; + foreach (IHardware hardware in mComputer.Hardware) + { + hardware.Update(); + foreach (ISensor sensor in hardware.Sensors) + { + //Debug.Print("\tSensor: {0}", sensor.Identifier.ToString()); + + if (sensor.Identifier.ToString() == mSensorPath) + { + temperature = sensor.Value.GetValueOrDefault(); + goto LOOPEND; + } + } + } + LOOPEND: + return temperature; + } + }; + + internal static class Program + { + private static MonitoringSetting mSetting; + private static TemperatureSensor mSensor; + private static System.Timers.Timer mTimer; + private static NotifyIcon mIcon; + + /// + /// アプリケーションのメイン エントリ ポイントです。 + /// + [STAThread] + static void Main() + { + mSetting = ReadSetting(); + + using (mSensor = new TemperatureSensor(mSetting.SensorPath)) + { + Application.EnableVisualStyles(); + Application.SetCompatibleTextRenderingDefault(false); + + CreateNotifyIcon(); + SetTimer(); + + OnInit(); + + Application.Run(); + + mIcon.Dispose(); + mTimer.Dispose(); + } + } + + static void OnInit() + { + new ToastContentBuilder() + .AddText("GPU温度監視開始", AdaptiveTextStyle.Default) + .Show(); + } + + static MonitoringSetting ReadSetting() + { + MonitoringSetting ret; + + string settingPath = Path.Combine(GetExeDir(), "Setting.json"); + using (FileStream fileStream = File.OpenRead(settingPath)) + { + using (StreamReader reader = new StreamReader(fileStream, System.Text.Encoding.UTF8)) + { + ret = JsonSerializer.Deserialize(reader.ReadToEnd()); + } + } + + return ret; + } + + static string GetExeDir() + { + return Path.GetDirectoryName(Application.ExecutablePath); + } + + private static void LINENotify(string message) + { + string LINE_url = "https://notify-api.line.me/api/notify"; + Encoding enc = Encoding.UTF8; + string payload = "message=" + HttpUtility.UrlEncode(message, enc); + + using (WebClient client = new WebClient()) + { + client.Encoding = enc; + client.Headers.Add("Content-Type", "application/x-www-form-urlencoded"); + client.Headers.Add("Authorization", "Bearer " + mSetting.LINENotifyToken); + client.UploadString(LINE_url, payload); + } + } + + private static void SetTimer() + { + mTimer = new System.Timers.Timer(mSetting.MonitoringIntervalMS); + mTimer.Elapsed += OnTimedEvent; + mTimer.AutoReset = true; + mTimer.Enabled = true; + } + + private static void CreateNotifyIcon() + { + // 常駐アプリ(タスクトレイのアイコン)を作成 + mIcon = new NotifyIcon(); + mIcon.Icon = new Icon("Icon.ico"); + mIcon.ContextMenuStrip = ContextMenu(); + mIcon.Text = "GPU温度監視"; + mIcon.Visible = true; + } + + private static ContextMenuStrip ContextMenu() + { + // アイコンを右クリックしたときのメニューを返却 + var menu = new ContextMenuStrip(); + menu.Items.Add("終了", null, (s, e) => + { + Application.Exit(); + }); + return menu; + } + + private static void OnTimedEvent(Object source, ElapsedEventArgs e) + { + var temperature = mSensor.GetTemperature(); + Debug.Print("{0}", temperature); + mIcon.Text = string.Format("GPU温度監視: {0}℃", temperature); + ChangeState(temperature); + } + + static bool mIsBeforeOverThreshold = false; + static TimeSpan mBeforeNotifyOverThresholdTime; + static object mLockObj = new object(); + + static void ChangeState(float temperature) + { + lock (mLockObj) + { + bool isOverThreshold = temperature > mSetting.TemperatureThreshold; + try + { + if (isOverThreshold) + { + if (!mIsBeforeOverThreshold) // 前回OKだったけど今回オーバーした + { + mBeforeNotifyOverThresholdTime = DateTime.Now.TimeOfDay; + OnStartOverThreshold(temperature); + } + else // 引き続きオーバーした + { + var diff = DateTime.Now.TimeOfDay - mBeforeNotifyOverThresholdTime; + if (diff.TotalSeconds >= mSetting.FailedNotifyIntervalS) + { + mBeforeNotifyOverThresholdTime = DateTime.Now.TimeOfDay; + OnContinueOverThreshold(temperature); + } + } + } + else + { + if (mIsBeforeOverThreshold) // 前回オーバーしたしたけど今回OKだった + { + OnFixedOverThreshold(temperature); + } + } + } + finally + { + mIsBeforeOverThreshold = isOverThreshold; + } + } + } + + static void OnStartOverThreshold(float temperature) + { + mIcon.Icon = SystemIcons.Warning; + + new ToastContentBuilder() + .AddText("GPU温度警告!", AdaptiveTextStyle.Caption) + .AddText(string.Format("GPU温度が閾値を超えています!")) + .AddText(string.Format("{0}℃", temperature)) + .Show(); + + string lineMessage = string.Format("警告\nGPU温度が閾値を超えています!\n{0}℃", temperature); + if (!EventLog.SourceExists("GPUTemperatureMonitoring")) + { + EventLog.CreateEventSource("GPUTemperatureMonitoring", "Application"); + } + EventLog.WriteEntry("GPUTemperatureMonitoring", lineMessage, + System.Diagnostics.EventLogEntryType.Warning, 0, 0); + + LINENotify(lineMessage); + } + + static void OnContinueOverThreshold(float temperature) + { + mIcon.Icon = SystemIcons.Warning; + + new ToastContentBuilder() + .AddText("GPU温度警告!", AdaptiveTextStyle.Caption) + .AddText(string.Format("GPU温度が閾値を超えています!")) + .AddText(string.Format("{0}℃", temperature)) + .Show(); + + string lineMessage = string.Format("警告\nGPU温度が閾値を超えています!\n{0}℃", temperature); + if (!EventLog.SourceExists("GPUTemperatureMonitoring")) + { + EventLog.CreateEventSource("GPUTemperatureMonitoring", "Application"); + } + EventLog.WriteEntry("GPUTemperatureMonitoring", lineMessage, + System.Diagnostics.EventLogEntryType.Warning, 0, 1); + + LINENotify(lineMessage); + } + + static void OnFixedOverThreshold(float temperature) + { + mIcon.Icon = new Icon("Icon.ico"); + + new ToastContentBuilder() + .AddText("GPU温度正常化", AdaptiveTextStyle.Default) + .AddText(string.Format("GPU温度が閾値以下に下がりました。")) + .AddText(string.Format("{0}℃", temperature)) + .Show(); + + string lineMessage = string.Format("GPU温度正常化\nGPU温度が閾値以下に下がりました。\n{0}℃", temperature); + if (!EventLog.SourceExists("GPUTemperatureMonitoring")) + { + EventLog.CreateEventSource("GPUTemperatureMonitoring", "Application"); + } + EventLog.WriteEntry("GPUTemperatureMonitoring", lineMessage, + System.Diagnostics.EventLogEntryType.Information, 1, 0); + + LINENotify(lineMessage); + } + } +} diff --git a/GPUTemperatureMonitoring/Properties/AssemblyInfo.cs b/GPUTemperatureMonitoring/Properties/AssemblyInfo.cs new file mode 100644 index 0000000..a8d0be0 --- /dev/null +++ b/GPUTemperatureMonitoring/Properties/AssemblyInfo.cs @@ -0,0 +1,36 @@ +using System.Reflection; +using System.Runtime.CompilerServices; +using System.Runtime.InteropServices; + +// アセンブリに関する一般的な情報は、次の方法で制御されます +// 制御されます。アセンブリに関連付けられている情報を変更するには、 +// これらの属性値を変更します。 +[assembly: AssemblyTitle("GPUTemperatureMonitoring")] +[assembly: AssemblyDescription("")] +[assembly: AssemblyConfiguration("")] +[assembly: AssemblyCompany("")] +[assembly: AssemblyProduct("GPUTemperatureMonitoring")] +[assembly: AssemblyCopyright("Copyright © 2023")] +[assembly: AssemblyTrademark("")] +[assembly: AssemblyCulture("")] + +// ComVisible を false に設定すると、このアセンブリ内の型は COM コンポーネントから +// 参照できなくなります。COM からこのアセンブリ内の型にアクセスする必要がある場合は、 +// その型の ComVisible 属性を true に設定してください。 +[assembly: ComVisible(false)] + +// このプロジェクトが COM に公開される場合、次の GUID が typelib の ID になります +[assembly: Guid("e1d50143-93c5-4585-8516-a3c2b465a444")] + +// アセンブリのバージョン情報は、以下の 4 つの値で構成されています: +// +// メジャー バージョン +// マイナー バージョン +// ビルド番号 +// リビジョン +// +// すべての値を指定するか、次を使用してビルド番号とリビジョン番号を既定に設定できます +// 既定値にすることができます: +// [assembly: AssemblyVersion("1.0.*")] +[assembly: AssemblyVersion("1.0.0.0")] +[assembly: AssemblyFileVersion("1.0.0.0")] diff --git a/GPUTemperatureMonitoring/Properties/Resources.Designer.cs b/GPUTemperatureMonitoring/Properties/Resources.Designer.cs new file mode 100644 index 0000000..c69d1e5 --- /dev/null +++ b/GPUTemperatureMonitoring/Properties/Resources.Designer.cs @@ -0,0 +1,71 @@ +//------------------------------------------------------------------------------ +// +// このコードはツールによって生成されました。 +// ランタイム バージョン:4.0.30319.42000 +// +// このファイルへの変更は、以下の状況下で不正な動作の原因になったり、 +// コードが再生成されるときに損失したりします +// +//------------------------------------------------------------------------------ + +namespace GPUTemperatureMonitoring.Properties +{ + + + /// + /// ローカライズされた文字列などを検索するための、厳密に型指定されたリソース クラスです。 + /// + // このクラスは StronglyTypedResourceBuilder クラスが ResGen + // または Visual Studio のようなツールを使用して自動生成されました。 + // メンバーを追加または削除するには、.ResX ファイルを編集して、/str オプションと共に + // ResGen を実行し直すか、または VS プロジェクトをリビルドします。 + [global::System.CodeDom.Compiler.GeneratedCodeAttribute("System.Resources.Tools.StronglyTypedResourceBuilder", "4.0.0.0")] + [global::System.Diagnostics.DebuggerNonUserCodeAttribute()] + [global::System.Runtime.CompilerServices.CompilerGeneratedAttribute()] + internal class Resources + { + + private static global::System.Resources.ResourceManager resourceMan; + + private static global::System.Globalization.CultureInfo resourceCulture; + + [global::System.Diagnostics.CodeAnalysis.SuppressMessageAttribute("Microsoft.Performance", "CA1811:AvoidUncalledPrivateCode")] + internal Resources() + { + } + + /// + /// このクラスで使用されるキャッシュされた ResourceManager インスタンスを返します。 + /// + [global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)] + internal static global::System.Resources.ResourceManager ResourceManager + { + get + { + if ((resourceMan == null)) + { + global::System.Resources.ResourceManager temp = new global::System.Resources.ResourceManager("GPUTemperatureMonitoring.Properties.Resources", typeof(Resources).Assembly); + resourceMan = temp; + } + return resourceMan; + } + } + + /// + /// すべてについて、現在のスレッドの CurrentUICulture プロパティをオーバーライドします + /// 現在のスレッドの CurrentUICulture プロパティをオーバーライドします。 + /// + [global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)] + internal static global::System.Globalization.CultureInfo Culture + { + get + { + return resourceCulture; + } + set + { + resourceCulture = value; + } + } + } +} diff --git a/GPUTemperatureMonitoring/Properties/Resources.resx b/GPUTemperatureMonitoring/Properties/Resources.resx new file mode 100644 index 0000000..ffecec8 --- /dev/null +++ b/GPUTemperatureMonitoring/Properties/Resources.resx @@ -0,0 +1,117 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + text/microsoft-resx + + + 2.0 + + + System.Resources.ResXResourceReader, System.Windows.Forms, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089 + + + System.Resources.ResXResourceWriter, System.Windows.Forms, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089 + + \ No newline at end of file diff --git a/GPUTemperatureMonitoring/Properties/Settings.Designer.cs b/GPUTemperatureMonitoring/Properties/Settings.Designer.cs new file mode 100644 index 0000000..fb33394 --- /dev/null +++ b/GPUTemperatureMonitoring/Properties/Settings.Designer.cs @@ -0,0 +1,30 @@ +//------------------------------------------------------------------------------ +// +// This code was generated by a tool. +// Runtime Version:4.0.30319.42000 +// +// Changes to this file may cause incorrect behavior and will be lost if +// the code is regenerated. +// +//------------------------------------------------------------------------------ + +namespace GPUTemperatureMonitoring.Properties +{ + + + [global::System.Runtime.CompilerServices.CompilerGeneratedAttribute()] + [global::System.CodeDom.Compiler.GeneratedCodeAttribute("Microsoft.VisualStudio.Editors.SettingsDesigner.SettingsSingleFileGenerator", "11.0.0.0")] + internal sealed partial class Settings : global::System.Configuration.ApplicationSettingsBase + { + + private static Settings defaultInstance = ((Settings)(global::System.Configuration.ApplicationSettingsBase.Synchronized(new Settings()))); + + public static Settings Default + { + get + { + return defaultInstance; + } + } + } +} diff --git a/GPUTemperatureMonitoring/Properties/Settings.settings b/GPUTemperatureMonitoring/Properties/Settings.settings new file mode 100644 index 0000000..abf36c5 --- /dev/null +++ b/GPUTemperatureMonitoring/Properties/Settings.settings @@ -0,0 +1,7 @@ + + + + + + + diff --git a/GPUTemperatureMonitoring/app.manifest b/GPUTemperatureMonitoring/app.manifest new file mode 100644 index 0000000..f2c47a2 --- /dev/null +++ b/GPUTemperatureMonitoring/app.manifest @@ -0,0 +1,64 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/Setting.json b/Setting.json new file mode 100644 index 0000000..dec11de --- /dev/null +++ b/Setting.json @@ -0,0 +1,7 @@ +{ + "SensorPath": "/gpu-nvidia/0/temperature/0", + "LINENotifyToken": "", + "MonitoringIntervalMS": 5000, + "TemperatureThreshold": 70, + "FailedNotifyIntervalS": 300 +}