-
Notifications
You must be signed in to change notification settings - Fork 1
Expand file tree
/
Copy pathLog.cs
More file actions
133 lines (112 loc) · 3.87 KB
/
Log.cs
File metadata and controls
133 lines (112 loc) · 3.87 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
using System;
using System.Diagnostics;
using System.IO;
using System.Text;
public class Logger
{
private readonly string _logFilePath;
private static readonly object _lock = new();
// Singleton instance for global access
public static Logger Instance { get; private set; }
static Logger()
{
Instance = new Logger();
}
private Logger()
{
// Ensure the logs directory exists
string logDirectory = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "logs");
Directory.CreateDirectory(logDirectory);
// Create a new log file with the startup time and date in its name
string fileName = $"log_{DateTime.Now:yyyy-MM-dd_HH-mm-ss}.txt";
_logFilePath = Path.Combine(logDirectory, fileName);
// Write initial message to the log file
WriteLine("=== Application Started ===");
// Add a custom trace listener to capture Debug/Trace output
Trace.Listeners.Add(new LogTraceListener(this));
}
/// <summary>
/// Logs a message to the console and the log file.
/// </summary>
/// <param name="message">The message to log.</param>
public void Log(string message)
{
string timestampedMessage = $"[{DateTime.Now:yyyy-MM-dd HH:mm:ss}] {message}";
Console.WriteLine(timestampedMessage);
AppendToFile(timestampedMessage);
}
/// <summary>
/// Logs a message to the console and the log file, formatted with additional arguments.
/// </summary>
/// <param name="message">The message template.</param>
/// <param name="args">Arguments to format into the message.</param>
public void Log(string message, params object[] args)
{
Log(string.Format(message, args));
}
/// <summary>
/// Writes a line to the log file only.
/// </summary>
/// <param name="message">The message to write.</param>
public void WriteLine(string message)
{
string timestampedMessage = $"[{DateTime.Now:yyyy-MM-dd HH:mm:ss}] {message}";
AppendToFile(timestampedMessage);
}
/// <summary>
/// Logs an exception in a human-readable format.
/// </summary>
/// <param name="ex">The exception to log.</param>
public void LogException(Exception ex)
{
var sb = new StringBuilder();
sb.AppendLine("=== Exception Caught ===");
sb.AppendLine($"Timestamp: {DateTime.Now:yyyy-MM-dd HH:mm:ss}");
sb.AppendLine($"Message: {ex.Message}");
sb.AppendLine("Stack Trace:");
sb.AppendLine(ex.StackTrace);
if (ex.InnerException != null)
{
sb.AppendLine("Inner Exception:");
AppendInnerExceptionDetails(sb, ex.InnerException);
}
sb.AppendLine("========================");
// Log the exception details
Log(sb.ToString());
}
private void AppendInnerExceptionDetails(StringBuilder sb, Exception ex)
{
sb.AppendLine($"Message: {ex.Message}");
sb.AppendLine("Stack Trace:");
sb.AppendLine(ex.StackTrace);
if (ex.InnerException != null)
{
sb.AppendLine("Further Inner Exception:");
AppendInnerExceptionDetails(sb, ex.InnerException);
}
}
private void AppendToFile(string message)
{
lock (_lock)
{
File.AppendAllText(_logFilePath, message + Environment.NewLine);
}
}
// Custom TraceListener to redirect Debug and Trace calls to the LogSystem
private class LogTraceListener : TraceListener
{
private readonly Logger _logSystem;
public LogTraceListener(Logger logSystem)
{
_logSystem = logSystem;
}
public override void Write(string message)
{
_logSystem.Log(message);
}
public override void WriteLine(string message)
{
_logSystem.Log(message);
}
}
}