Skip to content

Commit 3464944

Browse files
authored
Merge pull request #86 from nutdotnet/79-winnut-crash
Error Reporting & Exception Catching Refactor; Synology DSM 7 quirk compatibility
2 parents f0991d9 + 708cfe7 commit 3464944

File tree

6 files changed

+158
-200
lines changed

6 files changed

+158
-200
lines changed

WinNUT_V2/WinNUT-Client/ApplicationEvents.vb

+73-79
Original file line numberDiff line numberDiff line change
@@ -7,8 +7,11 @@
77
'
88
' This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY
99

10+
Imports System.Globalization
1011
Imports System.IO
12+
Imports System.Text.RegularExpressions
1113
Imports Microsoft.VisualBasic.ApplicationServices
14+
Imports Newtonsoft.Json
1215
Imports WinNUT_Client_Common
1316

1417
Namespace My
@@ -19,23 +22,31 @@ Namespace My
1922
' StartupNextInstance : Déclenché lors du lancement d'une application à instance unique et si cette application est déjà active.
2023
' NetworkAvailabilityChanged : Déclenché quand la connexion réseau est connectée ou déconnectée.
2124
Partial Friend Class MyApplication
25+
' Default culture for output so logs can be shared with the project.
26+
Private Shared DEF_CULTURE_INFO As CultureInfo = CultureInfo.InvariantCulture
27+
28+
2229
Private CrashBug_Form As New Form
2330
Private BtnClose As New Button
2431
Private BtnGenerate As New Button
2532
Private Msg_Crash As New Label
2633
Private Msg_Error As New TextBox
2734

35+
Private crashReportData As String
36+
2837
Private Sub MyApplication_Startup(sender As Object, e As StartupEventArgs) Handles Me.Startup
29-
'Init WinNUT Variables
38+
' Uncomment below and comment out Handles line for _UnhandledException sub when debugging unhandled exceptions.
39+
' AddHandler AppDomain.CurrentDomain.UnhandledException, AddressOf AppDomainUnhandledException
3040
Init_Globals()
31-
LogFile.LogTracing("Init Globals Variables Complete", LogLvl.LOG_DEBUG, Me)
41+
LogFile.LogTracing("MyApplication_Startup complete.", LogLvl.LOG_DEBUG, Me)
3242
End Sub
3343

34-
Private Sub MyApplication_UnhandledException(ByVal sender As Object, ByVal e As UnhandledExceptionEventArgs) Handles Me.UnhandledException
35-
e.ExitApplication = False
44+
Private Sub AppDomainUnhandledException(sender As Object, e As System.UnhandledExceptionEventArgs)
45+
MyApplication_UnhandledException(sender, New UnhandledExceptionEventArgs(False, e.ExceptionObject))
46+
End Sub
3647

37-
Dim Frms As New FormCollection
38-
Frms = Application.OpenForms()
48+
Private Sub MyApplication_UnhandledException(sender As Object, e As UnhandledExceptionEventArgs) Handles Me.UnhandledException
49+
e.ExitApplication = False
3950

4051
With Msg_Crash
4152
.Location = New Point(6, 6)
@@ -49,19 +60,12 @@ Namespace My
4960
.Size = New Point(470, 100)
5061
End With
5162

52-
Dim Exception_data = BuildExceptionString(e.Exception)
53-
54-
If e.Exception.InnerException IsNot Nothing Then
55-
Exception_data &= vbNewLine & "InnerException present:" & vbNewLine
56-
Exception_data &= BuildExceptionString(e.Exception.InnerException)
57-
End If
58-
5963
With Msg_Error
6064
.Location = New Point(6, 110)
6165
.Multiline = True
6266
.ScrollBars = ScrollBars.Vertical
6367
.ReadOnly = True
64-
.Text = Exception_data.ToString()
68+
.Text = e.Exception.ToString()
6569
.Size = New Point(470, 300)
6670
End With
6771

@@ -93,99 +97,89 @@ Namespace My
9397
.Controls.Add(BtnGenerate)
9498
End With
9599

100+
crashReportData = GenerateCrashReport(e.Exception)
101+
96102
AddHandler BtnClose.Click, AddressOf Application.Close_Button_Click
97103
AddHandler BtnGenerate.Click, AddressOf Application.Generate_Button_Click
98-
AddHandler CrashBug_Form.FormClosing, AddressOf Application.CrashBug_FormClosing
99104

100105
CrashBug_Form.Show()
101106
CrashBug_Form.BringToFront()
102107
WinNUT.HasCrashed = True
103108
End Sub
104109

105-
''' <summary>
106-
''' Generate a friendly message describing an exception.
107-
''' </summary>
108-
''' <param name="ex">The exception that will be read for the message.</param>
109-
''' <returns>The final string representation of the exception.</returns>
110-
Private Function BuildExceptionString(ex As Exception) As String
111-
Dim retStr = String.Empty
112-
113-
retStr &= String.Format("Exception type: {0}" & vbNewLine, ex.GetType.ToString)
114-
retStr &= String.Format("Exception message: {0}" & vbNewLine, ex.Message)
115-
retStr &= "Exception stack trace:" & vbNewLine
116-
retStr &= ex.StackTrace & vbNewLine
117-
118-
Return retStr
119-
End Function
120-
121-
Private Sub CrashBug_FormClosing(sender As Object, e As FormClosingEventArgs)
122-
End
123-
End Sub
124-
Private Sub Close_Button_Click(sender As Object, e As EventArgs)
125-
End
126-
End Sub
127-
128-
Private Sub Generate_Button_Click(sender As Object, e As EventArgs)
129-
'Generate a bug report with all essential datas
130-
Dim Crash_Report As String = "WinNUT Bug Report" & vbNewLine
131-
Dim WinNUT_Config As New Dictionary(Of String, Object)
132-
Try
133-
WinNUT_Config = Arr_Reg_Key
134-
Catch ex As Exception
135-
Crash_Report &= "ALERT: Encountered exception while trying to access Arr_Reg_Key:" & vbNewLine
136-
Crash_Report &= BuildExceptionString(ex)
137-
End Try
138-
139-
' Initialize directory for data
140-
Dim CrashLog_Dir = ApplicationData & "\CrashLog"
141-
If Not Computer.FileSystem.DirectoryExists(CrashLog_Dir) Then
142-
Computer.FileSystem.CreateDirectory(CrashLog_Dir)
143-
End If
110+
Private Shared Function GenerateCrashReport(ex As Exception) As String
111+
Dim jsonSerializerSettings As New JsonSerializerSettings()
112+
jsonSerializerSettings.Culture = DEF_CULTURE_INFO
113+
jsonSerializerSettings.Formatting = Formatting.Indented
144114

145-
Dim CrashLog_Filename As String = "Crash_Report_" & Format(Now, "dd-MM-yyyy") & "_" &
146-
String.Format("{0}-{1}-{2}.txt", Now.Hour.ToString("00"), Now.Minute.ToString("00"), Now.Second.ToString("00"))
115+
Dim reportStream As New StringWriter(DEF_CULTURE_INFO)
116+
reportStream.WriteLine("WinNUT Bug Report")
117+
reportStream.WriteLine("Generated at " + Date.UtcNow.ToString("F", DEF_CULTURE_INFO))
118+
reportStream.WriteLine()
119+
reportStream.WriteLine("OS Version: " & Computer.Info.OSVersion)
120+
reportStream.WriteLine("WinNUT Version: " & ProgramVersion)
147121

122+
#Region "Config output"
123+
Dim confCopy = New Dictionary(Of String, Object)
148124

125+
reportStream.WriteLine()
126+
reportStream.WriteLine("==== Parameters ====")
127+
reportStream.WriteLine()
149128

150-
Crash_Report &= "Os Version : " & Computer.Info.OSVersion & vbNewLine
151-
Crash_Report &= "WinNUT Version : " & Reflection.Assembly.GetEntryAssembly().GetName().Version.ToString & vbNewLine
152-
153-
Crash_Report &= vbNewLine & "WinNUT Parameters : " & vbNewLine
154-
If WinNUT_Config.Count > 0 Then
155-
' Prepare config values by removing sensitive information.
129+
' Censor any identifying information
130+
If Arr_Reg_Key IsNot Nothing AndAlso Arr_Reg_Key.Count > 0 Then
156131
For Each kvp As KeyValuePair(Of String, Object) In Arr_Reg_Key
132+
Dim newVal As String
157133
Select Case kvp.Key
158134
Case "ServerAddress", "Port", "UPSName", "NutLogin", "NutPassword"
159-
WinNUT_Config.Remove(kvp.Key)
135+
newVal = "*****"
136+
Case Else
137+
newVal = kvp.Value
160138
End Select
139+
140+
confCopy.Add(kvp.Key, newVal)
161141
Next
162-
Crash_Report &= Newtonsoft.Json.JsonConvert.SerializeObject(WinNUT_Config, Newtonsoft.Json.Formatting.Indented) & vbNewLine
163142

143+
reportStream.WriteLine(JsonConvert.SerializeObject(confCopy, jsonSerializerSettings))
144+
reportStream.WriteLine()
164145
Else
165-
Crash_Report &= "[EMPTY]" & vbNewLine
146+
reportStream.WriteLine("[EMPTY]")
166147
End If
148+
#End Region
167149

168-
Crash_Report &= vbNewLine & "Error Message : " & vbNewLine
169-
Crash_Report &= Msg_Error.Text & vbNewLine & vbNewLine
170-
Crash_Report &= "Last Events :" & vbNewLine
150+
#Region "Exceptions"
151+
reportStream.WriteLine("==== Exception ====")
152+
reportStream.WriteLine()
153+
reportStream.WriteLine(Regex.Unescape(JsonConvert.SerializeObject(ex, jsonSerializerSettings)))
154+
reportStream.WriteLine()
155+
#End Region
171156

172-
For Each WinNUT_Event In LogFile.LastEvents
173-
Crash_Report &= WinNUT_Event & vbNewLine
174-
Next
157+
reportStream.WriteLine("==== Last Events ====")
175158

176-
Computer.Clipboard.SetText(Crash_Report)
159+
LogFile.LastEvents.Reverse()
160+
reportStream.WriteLine()
161+
reportStream.WriteLine(Regex.Unescape(JsonConvert.SerializeObject(LogFile.LastEvents, jsonSerializerSettings)))
177162

178-
Dim CrashLog_Report As StreamWriter
179-
CrashLog_Report = Computer.FileSystem.OpenTextFileWriter(CrashLog_Dir & "\" & CrashLog_Filename, True)
180-
CrashLog_Report.WriteLine(Crash_Report)
163+
Return reportStream.ToString()
164+
End Function
165+
166+
Private Sub Generate_Button_Click(sender As Object, e As EventArgs)
167+
Dim logFileName = "CrashReport_" + Date.Now.ToString("s").Replace(":", ".") + ".txt"
168+
169+
Computer.Clipboard.SetText(crashReportData)
170+
171+
Directory.CreateDirectory(TEMP_DATA_PATH)
172+
Dim CrashLog_Report = New StreamWriter(Path.Combine(TEMP_DATA_PATH, logFileName))
173+
CrashLog_Report.WriteLine(crashReportData)
181174
CrashLog_Report.Close()
182175

183176
' Open an Explorer window to the crash log.
184-
' Dim fullFilepath As String = CrashLog_Dir & "\" & CrashLog_Filename
185-
' If WinNUT IsNot Nothing Then
186-
Process.Start(CrashLog_Dir)
187-
' End If
177+
Process.Start(TEMP_DATA_PATH)
188178
End
189179
End Sub
180+
181+
Private Sub Close_Button_Click(sender As Object, e As EventArgs)
182+
CrashBug_Form.Close()
183+
End Sub
190184
End Class
191185
End Namespace

WinNUT_V2/WinNUT-Client/WinNUT-client.vbproj

+2-2
Original file line numberDiff line numberDiff line change
@@ -396,10 +396,10 @@
396396
</ItemGroup>
397397
<ItemGroup>
398398
<PackageReference Include="Microsoft.Toolkit.Uwp.Notifications">
399-
<Version>7.1.2</Version>
399+
<Version>7.1.3</Version>
400400
</PackageReference>
401401
<PackageReference Include="Newtonsoft.Json">
402-
<Version>13.0.1</Version>
402+
<Version>13.0.3</Version>
403403
</PackageReference>
404404
</ItemGroup>
405405
<Import Project="$(MSBuildToolsPath)\Microsoft.VisualBasic.targets" />

WinNUT_V2/WinNUT-Client/WinNUT.vb

+14-18
Original file line numberDiff line numberDiff line change
@@ -86,12 +86,6 @@ Public Class WinNUT
8686
Private Event RequestConnect()
8787

8888
Private Sub WinNUT_Load(sender As Object, e As EventArgs) Handles MyBase.Load
89-
' Make sure we have an app directory to write to.
90-
' SetupAppDirectory()
91-
92-
AddHandler Microsoft.Win32.SystemEvents.PowerModeChanged, AddressOf SystemEvents_PowerModeChanged
93-
AddHandler RequestConnect, AddressOf UPS_Connect
94-
9589
'Add Main Gui's Strings
9690
StrLog.Insert(AppResxStr.STR_MAIN_OLDINI_RENAMED, My.Resources.Frm_Main_Str_01)
9791
StrLog.Insert(AppResxStr.STR_MAIN_OLDINI, My.Resources.Frm_Main_Str_02)
@@ -273,6 +267,9 @@ Public Class WinNUT
273267
HasFocus = False
274268
End If
275269

270+
AddHandler Microsoft.Win32.SystemEvents.PowerModeChanged, AddressOf SystemEvents_PowerModeChanged
271+
AddHandler RequestConnect, AddressOf UPS_Connect
272+
276273
LogFile.LogTracing(String.Format("{0} v{1} completed initialization.", My.Application.Info.ProductName, My.Application.Info.Version),
277274
LogLvl.LOG_NOTICE, Me)
278275
End Sub
@@ -300,6 +297,7 @@ Public Class WinNUT
300297
Arr_Reg_Key.Item("AutoReconnect"))
301298

302299
UPS_Device = New UPS_Device(Nut_Config, LogFile, Arr_Reg_Key.Item("Delay"))
300+
AddHandler UPS_Device.EncounteredNUTException, AddressOf HandleNUTException
303301
UPS_Device.Connect_UPS(retryOnConnFailure)
304302
End Sub
305303

@@ -334,15 +332,16 @@ Public Class WinNUT
334332
End Sub
335333

336334
''' <summary>
337-
''' Prepare application for and handle disconnecting from the UPS.
335+
''' Prepare application for and initiate disconnecting from the UPS.
338336
''' </summary>
339-
Private Sub UPSDisconnect() ' Handles UPS_Device.Disconnected
340-
' LogFile.LogTracing("Running Client disconnect subroutine.", LogLvl.LOG_DEBUG, Me)
337+
Private Sub UPSDisconnect()
338+
LogFile.LogTracing("Running Client disconnect subroutine.", LogLvl.LOG_DEBUG, Me)
341339

342340
If UPS_Device IsNot Nothing Then
343341
UPS_Device.Disconnect(True)
342+
RemoveHandler UPS_Device.EncounteredNUTException, AddressOf HandleNUTException
344343
Else
345-
LogFile.LogTracing("Attempted to disconnect when UPS_Device is Nothing.", LogLvl.LOG_DEBUG, Me)
344+
LogFile.LogTracing("Attempted to disconnect when UPS_Device is Nothing.", LogLvl.LOG_ERROR, Me)
346345
End If
347346
End Sub
348347

@@ -418,18 +417,14 @@ Public Class WinNUT
418417
NotifyIcon.Visible = True
419418
e.Cancel = True
420419
Else
421-
' TODO: Common shutdown subroutine? --v
422420
LogFile.LogTracing("Init Disconnecting Before Close WinNut", LogLvl.LOG_DEBUG, Me)
423-
RemoveHandler Microsoft.Win32.SystemEvents.PowerModeChanged, AddressOf SystemEvents_PowerModeChanged
424421
UPSDisconnect()
425-
LogFile.LogTracing("WinNut Is now Closed", LogLvl.LOG_DEBUG, Me)
426-
End
427422
End If
428423
End Sub
429424

430425
Private Sub Menu_Quit_Click_1(sender As Object, e As EventArgs) Handles Menu_Quit.Click
431426
LogFile.LogTracing("Close WinNut From Menu Quit", LogLvl.LOG_DEBUG, Me)
432-
End
427+
Application.Exit()
433428
End Sub
434429

435430
Private Sub Menu_Settings_Click(sender As Object, e As EventArgs) Handles Menu_Settings.Click
@@ -442,7 +437,7 @@ Public Class WinNUT
442437

443438
Private Sub Menu_Sys_Exit_Click(sender As Object, e As EventArgs) Handles Menu_Sys_Exit.Click
444439
LogFile.LogTracing("Close WinNut From Systray", LogLvl.LOG_DEBUG, Me)
445-
End
440+
Application.Exit()
446441
End Sub
447442

448443
Private Sub Menu_Sys_Settings_Click(sender As Object, e As EventArgs) Handles Menu_Sys_Settings.Click
@@ -575,11 +570,12 @@ Public Class WinNUT
575570
LogFile.LogTracing("Battery Status => " & Status, LogLvl.LOG_DEBUG, Me)
576571
End Sub
577572

578-
Sub HandleNUTException(ex As NutException, sender As Object) Handles UPS_Device.EncounteredNUTException
573+
Private Sub HandleNUTException(sender As UPS_Device, ex As NutException)
579574
If ex.LastTransaction.ResponseType = NUTResponse.UNKNOWNUPS Then
580575
Event_Unknown_UPS()
581576
End If
582577

578+
LogFile.LogTracing("NUT protocol error encoutnered:" + vbNewLine + ex.ToString(), LogLvl.LOG_NOTICE, sender)
583579
End Sub
584580

585581
Public Sub Event_Unknown_UPS() ' Handles UPS_Device.Unknown_UPS
@@ -837,7 +833,7 @@ Public Class WinNUT
837833
' Setup logging preferences
838834
If Arr_Reg_Key.Item("UseLogFile") Then
839835
LogFile.LogLevelValue = Arr_Reg_Key.Item("Log Level")
840-
LogFile.InitializeLogFile(ApplicationData)
836+
LogFile.InitializeLogFile(ApplicationDataPath)
841837
ElseIf LogFile.IsWritingToFile Then
842838
LogFile.DeleteLogFile()
843839
End If

0 commit comments

Comments
 (0)