This project came to me when I needed a way to view messages throughout the execution of my code and I was unable to rely on printing to the console while running a Windowed Application. I had used the debugger, until began to run into loops with hundreds of iterations. It had been suggested to me to use a logging tool.
I have grown accustomed to having a logging tool build into my IDE from using Android Studio’s LogCat. For my C# projects I use Visual Studio Code. While there is no logger integrated by default, there are dozens of extensions that can be downloaded an installed built by other developers.
I decided to develop my own as well. I approached the program in the most minimalist of fashions. The only thing I would consider a feature is the ability to the type and destination of the stream. The class is less than 100 lines of code, and that’s partially due to the excessive formatting of the Log string.
SimpleLogger
/**
The MIT License (MIT)
Copyright © 2019 <John L. Mooney>
Permission is hereby granted, free of charge, to any person obtaining a copy of this software
and associated documentation files (the “Software”), to deal in the Software without restriction,
including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense,
and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so,
subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial
portions of the Software.
THE SOFTWARE IS PROVIDED “AS IS”, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF
CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
DEALINGS IN THE SOFTWARE.
*/
//
// SimpleLogger.cs
//
using System;
using System.IO;
using System.Text;
namespace SimpleLogger
{
public enum LogLevel
{
info,
debug,
warning,
error
}
public class SimpleLogger
{
private static SimpleLogger logger = null;
private static bool initialized = false;
private Stream logStream;
private SimpleLogger(Stream stream)
{
this.logStream = stream;
}
public static SimpleLogger GetLogger()
{
if(!initialized)
{
throw new UninitializedException("Initialize the Logger first.");
}
return logger;
}
public static void Initialize(Stream stream)
{
if(!initialized)
{
logger = new SimpleLogger(stream);
initialized = true;
}
}
public void Log(LogLevel level, string tag, string msg)
{
string date = DateTime.Now.ToString("MM/dd/yyyy hh:mm:ss");
string strA = string.Format("[{0}]", date);
string strB = string.Format("<{0}>", level.ToString());
string strC = string.Format("::{0}::", tag);
string strD = string.Format("\"{0}\"", msg);
string str0 = string.Format("{0, -20}", strA);
string str1 = string.Format("{0, -10}", strB);
string str2 = string.Format("{0, -16}", strC);
string str3 = string.Format("{0, -20}", strD);
string str4 = Environment.NewLine;
string fullmsg = string.Format("{0} {1} {2} {3}{4}",
str0, str1, str2, str3, str4);
ToStream(fullmsg);
}
private void ToStream(string msg)
{
byte[] text = new UTF8Encoding(true).GetBytes(msg);
logStream.Write(text, 0, text.Length);
logStream.Flush();
}
}
}
Category Levels of Logging
I created four unique labels to categorize different levels of log statements. I chose the the terms based on what I have used in alternative logging systems. I enumerated them in an enum structure entitled “LogLevel.”public enum LogLevel
{
info,
debug,
warning,
error
}
Singleton
I followed the “Singleton” pattern by ensuring that there is only one static instance of the SimpleLogger to be shared among all who want to use the logger. The static instance must be initialized to null for this pattern to work. The rest is done through the implement of the Initialize() and GetLogger() methods.
The Initialize() method accepts a Stream derived object and passes it to the private constructor to set the member variable, logStream, thereby instantiating an static object of SimpleLogger. It also sets the boolean flag, “initialized,” to true so that It cannot be initialized more than once.
GetLogger() is the mechanism for obtaining the SimpleLogger instance, since there are no public constructors. when the method is called there is a check to see if the SimpleLogger has been initialized. If it has, then an object exists then it is returned to the calling function. In the alternative case, it throws an exception informing the user that the Logger needs to be initialized
UninitializedException.cs
I implemented a custom Exception class to represent the exception that should be thrown when a developer attempts to retrieve an uninitialized SimpleLogger instance. This is a skeleton exception class that provides no custom service, but allows for a specifically appropriate name and expansion in the future.
/**
The MIT License (MIT)
Copyright © 2019 <John L. Mooney>
Permission is hereby granted, free of charge, to any person obtaining a copy of this software
and associated documentation files (the “Software”), to deal in the Software without restriction,
including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense,
and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so,
subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial
portions of the Software.
THE SOFTWARE IS PROVIDED “AS IS”, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF
CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
DEALINGS IN THE SOFTWARE.
*/
//
// UninitializedException.cs
//
using System;
namespace SimpleLogger
{
public class UninitializedException : Exception
{
public UninitializedException()
{
}
public UninitializedException(string message)
: base(message)
{
}
public UninitializedException(string message, Exception inner)
: base(message, inner)
{
}
}
}
Tag
Before I discuss the Log() method and it’s format, I want to explain one of elements of the message, the tag.
The Log’s tag is meant to give more context to the message than the log level alone. Supplying the log statement with a custom tag allows for easier message filtering. A personal example of this is when I code on a project with others, the others and myself will often pass in our respective first name into the tag parameter. This provides us with the ability to identify the logged message by owner through use of the tag. There are better and more practical uses. Another is example could be logging in multi-threaded application. If each thread has a identifier (string or numeric ID), then the messages could be tagged with the thread’s identifier, allowing you to trace the behavior and events of each thread.
Format and Log
SimpleLogger’s Log() method is what allows the user to output the log messages to the specified stream. As stated earlier in the post, I was a bit excessive in the formatting of the log message. I designed the message to provide the date and time of the message, then the log level, followed by a “tag” and finally the message intended to be logged.
public void Log(LogLevel level, string tag, string msg)
{
string date = DateTime.Now.ToString("MM/dd/yyyy hh:mm:ss");
string strA = string.Format("[{0}]", date);
string strB = string.Format("<{0}>", level.ToString());
string strC = string.Format("::{0}::", tag);
string strD = string.Format("\"{0}\"", msg);
string str0 = string.Format("{0, -20}", strA);
string str1 = string.Format("{0, -10}", strB);
string str2 = string.Format("{0, -16}", strC);
string str3 = string.Format("{0, -20}", strD);
string str4 = Environment.NewLine;
string fullmsg = string.Format("{0} {1} {2} {3}{4}",
str0, str1, str2, str3, str4);
ToStream(fullmsg);
}
To explain my verbosity. I wanted to create a string that would allow me to be able to easily identify which element of the log message is which. My purpose was to able to perform searches and filters using unique formatting surrounding each element (e.g. “[]” around the date/time or “::” on both sides of the tag). These distinguishing factors assist in parsing.
The component string variables all start with the substring “str” and are followed by an alpha or a numeric. The alpha are used to tack on the distinguishing characteristics of the log message element. The numeric are used to format the alpha strings within right-aligned fixed width space providing the log file with the illusion of columns. “str4” is used to shorten the constant for the platform independent new line. “fullmsg” is a concatenation of all the other strings, which are then passed to the ToStream() method to be written to the stream.
ToStream( )
Simple method- Take the string argument and convert it to bytes
- Write the bytes out to the stream
- Flush the steam to prevent the bytes from hanging around in memory
private void ToStream(string msg)
{
byte[] text = new UTF8Encoding(true).GetBytes(msg);
logStream.Write(text, 0, text.Length);
logStream.Flush();
}
Example Usages
Simple Example
/** The MIT License (MIT) Copyright © 2019 <John L. Mooney> Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the “Software”), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED “AS IS”, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ // // Program.cs // using System; using System.IO; namespace SimpleLogger { class Program { static void Main(string[] args) { FileStream stream = new FileStream(@"logs\simple.log", FileMode.Append, FileAccess.Write, FileShare.ReadWrite, 4096); SimpleLogger.Initialize(stream); SimpleLogger logger = SimpleLogger.GetLogger(); logger.Log(LogLevel.warning, "Soup Nazi", "No soup for you"); logger.Log(LogLevel.info, "TRON", "Greetings Program"); logger.Log(LogLevel.warning, "Arithmatic", "Possible divide by zero"); logger.Log(LogLevel.error, "Coverage", "Some paths may never be executed"); logger.Log(LogLevel.info, "TRON", "I fight for the users"); logger.Log(LogLevel.debug, "Edge case", "This line of code should never be reached"); } } }
The output should look something like this:
[11/07/2019 07:15:07] <warning> ::Soup Nazi:: "No soup for you" [11/07/2019 07:15:07] <info> ::TRON:: "Greetings Program" [11/07/2019 07:15:07] <warning> ::Arithmatic:: "Possible divide by zero" [11/07/2019 07:15:07] <error> ::Coverage:: "Some paths may never be executed" [11/07/2019 07:15:07] <info> ::TRON:: "I fight for the users" [11/07/2019 07:15:07] <debug> ::Edge case:: "This line of code should never be reached"
A More Thorough Example
This is a GUI application so the main mostly consists of of boiler-plate. The important thing to take from this file is that a FileStream is created here followed by the initialization of the SimplerLogger. Here the the Logger will be initialized once and before anyone attempts to use it.
Program.cs
/** The MIT License (MIT) Copyright © 2019 <John L. Mooney> Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the “Software”), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED “AS IS”, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ // // Program.cs // using System; using System.IO; using System.Windows.Forms; namespace SimpleLogger { static class Program { [STAThread] static void Main() { FileStream logStream = new FileStream(@"logs/squares.log", FileMode.Append, FileAccess.Write, FileShare.ReadWrite, 4096); SimpleLogger.Initialize(logStream); Application.SetHighDpiMode(HighDpiMode.SystemAware); Application.EnableVisualStyles(); Application.SetCompatibleTextRenderingDefault(false); Application.Run(new Squares(128)); } } }
The Square class is an extension of the PictureBox class. I used it to make square boxes and painted them solid colors and named them by their respective color. The name, color and side length are passed in by construct by the Squares object that generates the boxes for its window.
Square has static instance of SimpleLogger called “logger.” To prevent every instance of Square from calling GetLogger() upon construction, I put the assignment logger = GetLogger() in a static constructor.
The reason this is a better example of the uses of SimpleLogger is because there are more strategic placements for logging. I created events for MouseClick, MouseEnter, MouseLeave to demonstrate logging based on an event. As mentioned earlier in the post, I opted to tag the logs by the name of the object. When I click a red colored square in the form window, I can verify that the event triggered is coming from the correct object using the log file (or the console).
/** The MIT License (MIT) Copyright © 2019 <John L. Mooney> Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the “Software”), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED “AS IS”, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ using System; using System.Drawing; using System.Windows.Forms; namespace SimpleLogger { public class Square : PictureBox { private static SimpleLogger logger; private new string Name{get; set;} private Brush brush; static Square() { try { logger = SimpleLogger.GetLogger(); } catch(Exception e) { MessageBox.Show(e.Message); Environment.Exit(1); } } public Square(string name, Color color, int side) { Name = name; brush = new SolidBrush(color); Size = new Size(side, side); logger.Log(LogLevel.warning, "Square", "Initializing Square to name of " + Name); Init(); } private void Init() { Paint += new PaintEventHandler(OnPaint); MouseClick += new MouseEventHandler(OnClickedMe); MouseEnter += new EventHandler(OnEnter); MouseLeave += new EventHandler(OnLeave); } private void OnPaint(object sender, PaintEventArgs e) { Point origin = new Point(0, 0); Rectangle rect = new Rectangle(origin, Size); e.Graphics.FillRectangle(brush, rect); logger.Log(LogLevel.debug, "Programmer", "Rectange: " + rect); } public override string ToString() { return Name; } public void OnClickedMe(object sender, MouseEventArgs e) { logger.Log(LogLevel.debug, sender.ToString(), "Hey, that tickles."); } public void OnEnter(object sender, EventArgs e) { logger.Log(LogLevel.info, sender.ToString(), "I see you"); } public void OnLeave(object sender, EventArgs e) { logger.Log(LogLevel.info, sender.ToString(), "Where did you go?"); } protected override void Dispose(bool disposing) { base.Dispose(disposing); } } }
The Squares class is a form that displays a grid of Square objects. It logs the creation of each Square object tagging it “Squares.”
/**
The MIT License (MIT)
Copyright © 2019 <John L. Mooney>
Permission is hereby granted, free of charge, to any person obtaining a copy of this software
and associated documentation files (the “Software”), to deal in the Software without restriction,
including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense,
and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so,
subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial
portions of the Software.
THE SOFTWARE IS PROVIDED “AS IS”, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF
CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
DEALINGS IN THE SOFTWARE.
*/
using System;
using System.Collections.Generic;
using System.Drawing;
using System.Windows.Forms;
namespace SimpleLogger
{
public class Squares : Form
{
private int SquareSide;
private List<Square> squares;
private List<Point> Coords;
private static SimpleLogger logger;
private static Color[] colors = {Color.Red, Color.Aqua, Color.Yellow, Color.Green,
Color.Blue, Color.Brown, Color.Crimson, Color.Cyan};
private static string[] names = {"Red", "Aqua", "Yellow", "Green", "Blue", "Brown",
"Crimson", "Cyan"};
static Squares()
{
try
{
logger = SimpleLogger.GetLogger();
}
catch(Exception e)
{
MessageBox.Show(e.Message);
Environment.Exit(1);
}
}
public Squares(int side)
{
SquareSide = side;
squares = new List<Square>();
Coords = new List<Point>();
Init();
}
private void Init()
{
int index = 0;
this.ClientSize = new Size(512, 512);
this.Text = "LogTesting";
int col = 4;
int row = 4;
Point start = new Point(0, 0);
for(int i = 0; i < col; i++)
{
for(int j = 0; j < row; j++)
{
Square sqr = new Square(names[index], colors[index], SquareSide);
Controls.Add(sqr);
squares.Add(sqr);
logger.Log(LogLevel.info, "Squares", "Just added a square to the Form. This Square's name is " + names[index]);
index = (index + 1) % colors.Length;
Point coord = new Point((i * SquareSide), (j * SquareSide));
Coords.Add(coord);
}
}
InitPlacement();
}
public void InitPlacement()
{
int index = 0;
foreach(Square square in squares)
{
square.Location = Coords[index++];
}
}
protected override void Dispose(bool disposing)
{
base.Dispose(disposing);
}
}
}