Asynchronous Server/Client with Python [0x02]: Logging

David Mentgen
TestingOnProd
Published in
5 min readOct 17, 2021

In part 0x02 of this series, we’re going to add a logging system to our server! As I mentioned in the last post, this server is going to have quite a lot of moving parts; it’s going to be useful to have a way to monitor what it’s doing. If we ever end up in a situation where our code crashes or misbehaves for some unknown reason, our logs should at least give us enough information to form some educated guesses about where the problem is coming from. In addition, logs can also provide a history of our chat room if you want one.

Part 0x03 Client Data & Server Broadcast

Links to the code on Github and useful documentation/references can be found below under the heading “Sauce, References, & Documentation”.

Logging Basics

Before we jump right into the actual coding, we need to decide a few things about what we want from our logging system.

The first thing to consider is what you will need messages for, as well as how urgent messages for some things should be compared to others. In Python logging, there are a few different levels of logging that we can use… ( https://docs.python.org/3/library/logging.html#logging-levels):

  • Critical — Something REALLY bad happened
  • Error — Something bad happened
  • Warning — Something unexpected happened, but is probably not going to break our code
  • Info — Useful information (Server start/stop, client connect/disconnect, user messages, etc)
  • Debug — Information that might be helpful to us, the developer, when troubleshooting problems

In this project, it is up to you to determine exactly how you need to use these throughout the code. The levels above are listed in descending order of importance, and the names themselves should give some idea as to what kinds of events might be appropriate for each label.

Next, we should consider how we’re going to format our logs. Luckily, there are some built-in formatters we can use to help us out here… ( https://docs.python.org/3/library/logging.html#formatter-objects).

For the sake of simplicity, in this example we will be using the following format:

[Datetime] — [Logging Level] — Message

Finally, we should have some idea of where we’re going to store our logs. For this project, let’s just create a “logs” folder in our current directory and store each new log there. We can use basic python file IO to make this happen; I’ll go further into detail about that later. For now, we’re set to move ahead!

Getting Started

For part 0x02, we are going to modify our Server.py file that we wrote in part one of this series. To get started, let’s go ahead and add our new imports to the top of our file (logging and datetime):

In our Server.py, we’ll want to monitor all of our Server’s activities. To begin, let’s modify our server’s constructor to include the logger we plan on using:

Now we’ll need to create our initialize_logger() function within our class so that we can setup the logger. Our self.initialize_logger() function will be responsible for setting up our log file, log format, and for returning a logger object accessible in self.logger. We’ll need to use that logger object to write logs into a file.

Let’s implement that function right below the new logger property that we just defined…

This function might look like a lot, but it’s not actually that complicated. Let’s break down what’s happening line by line…

  • Lines 10–11: We are using os and pathlib to check if our logs folder exists within our current working directory. If this folder does not exist, we will create it. If the logs folder already exists, then it will proceed without raising any errors, as indicated by us setting the “exists_ok” flag to “true” in line 11.
  • Line 13: In python’s Logging library, we are able to work with more than one logging object at a time. This might be useful in larger applications with many distinct moving parts executing under the same program. In our case, we are only dealing with the server so we can just reference our “server” logging object.
  • Line 14: Setting the logging level here determines the lowest priority of logs that our logging object will concern itself with. In our case, we care about ALL logs, so we will set our level to the lowest: debug. This ensures that our logs contain anything and everything that we might send it. If we were to set our logging level to something higher, then our logging object will ignore any attempts to log messages of a lower priority level.
  • Lines 16–21: Here, we are setting up our stream and file handling so that we are able to save and track our logs. We’re also setting up our file, which will be named after the current datetime.
  • Lines 23–25: This establishes the format of our entries.
  • Lines 27–30 This applies our settings to our logging object.

At this point, the hard part of logging is now behind us. All we have to do now is start using our logger! To keep things simple, I will paste the entirety of our Server.py so far below. Note that I have replaced all of our print statements with the logger:

Now if we want to see our logger in action, all we have to do is run our server! Nothing has really changed in terms of functionality from our code in the previous blog post, so testing this code should be exactly the same!

Conclusion

The point of this installment was to add a more robust logging system to our code that’s capable of persisting in a file after our program exits for future reference. This can be useful in a number of different cases, such as reviewing server logs for a chat history or for checking the logs while troubleshooting or improving the server. This feature will seriously come in handy as our server code increases in size and complexity.

In the next part of this project series, we will continue adding features to our server that will enable it to better track and manage its connections. We’ll also modify the server so that it can broadcast messages to all clients! This means one client will be able to see messages from all of the other clients; basically, we’ll be able to do group chats instead of just having a bunch of direct messages.

I hope you are happy with how your progress here is coming along!

Part 0x03 Client Data & Server Broadcast

Sauce, References, & Documentation

Github: https://github.com/D3ISM3/Python-Asynchronous-Server-Client
Python Logging Documentation: https://docs.python.org/3/library/logging.html

Originally published at http://testingonprod.com on October 17, 2021.

--

--