1. Vonage Learn
  2. Blog
  3. 2021
  4. 03
  5. 30
  6. Building a Python Error Alerting Tool
Building a Python Error Alerting Tool

< Tutorial />

Building a Python Error Alerting Tool

No matter how much we care about quality and testing, software is almost guaranteed to go wrong at some point. As a result, monitoring logs to keep track of application health are essential.

There certainly are multiple services and open source projects taking care of monitoring application logs. In my experience, though, they’re usually either expensive, time-consuming to integrate or bloated with features I will hardly use. When I’m deploying small projects that do not require fancy monitoring, sometimes I wish I could have a native Python solution to get simple alerts when something goes wrong in my code.

The purpose of this tutorial is precisely to fulfill this need. We will build a simple and flexible Python error alerting tool that can be plugged into any project. A logging HTTP handler object will asynchronously send alerts through the Vonage SMS API into our phones when new errors or warnings, for example, come in.

Requirements

We will use Python 3.9.1 (the latest stable version) in the tutorial, but the code should also work on Python 3.6+. Python is available on Linux, macOS and Windows. To download and install, follow the instructions at the official website.

You will also need a Vonage account to receive error alerts through SMS. Create an account if you’re not registered yet. Vonage provides new subscribers with € 2.00 in credits to test the APIs for free.

The Vonage API key and secret will also be necessary; make sure to grab them in the Dashboard settings:

Vonage Dashboard

PyPI http-logging library will be used for log caching and asynchronous communication with the Vonage API. It prevents our main Python application from being disrupted by the alerting mechanism.

Preparing the Local Environment

Virtualenv and dependencies

Create a directory for the project:

~$ mkdir vonage-alerts
~$ cd vonage-alerts

Creating a virtual environment is often good practice, so let’s get this done first:

~/vonage-alerts$ python3.9 -m venv .env
~/vonage-alerts$ source .env/bin/activate

On a Windows computer, replace the source command in the last line above with:

~/vonage-alerts$ .venv\Scripts\activate

Make sure the environment is working as expected:

(.env) ~/vonage-alerts$ python --version
Python 3.9.1

Now let’s create our Python dependencies file:

(.env) ~/vonage-alerts$ touch requirements.txt

Open it with your preferred text editor and add the following lines:

http-logging
vonage

Close the file and install dependencies with the pip install command:

(.env) ~/vonage-alerts$ pip install -r requirements.txt

Environment Variables

Our custom logging logic will require some information that will be supplied through environment variables.

The Vonage API Key is required for authentication with the SMS service. A phone number will also be necessary to send SMS messages.

(.env) ~/vonage-alerts$ export VONAGE_API_KEY="abc123"
(.env) ~/vonage-alerts$ export VONAGE_API_SECRET="xyz123"
(.env) ~/vonage-alerts$ export ALERT_PHONE_NUMBER="+1234567890"

The export command should work on Linux and macOS. On Windows, use set instead. If you are using a PowerShell console, then this command should do the work:

$Env: VONAGE_API_KEY = "abc123"
$Env: VONAGE_API_SECRET = "xyz123"
$Env: export ALERT_PHONE_NUMBER = "+1234567890"

HTTP Logging Handler

As mentioned before, we will rely on the http-logging library to connect our logs to the Vonage APIs.

A native logging HTTP handler from the Python standard library would also do the job. However, we are not going to use it because it generates blocking HTTP requests, which can negatively affect the execution of our main Python application.

The http-logging library runs silently in a background thread and is also capable of caching logs in a local SQLite database to reduce the number of network requests. For those reasons, it will be a lot less intrusive than a native HTTP handler.

The library is based on the Python Logstash Async, but generalized to work with any backend apart from Logstash (in our tutorial, we will use Vonage). Read more about it in the project documentation wiki.

Vonage HTTP Transport

The first thing we need is to create a custom HTTP Transport class. This is the one carrying instructions for how to send logs to the Vonage API.

Before we dive into it, let’s create a new Python file to contain our custom logging code:

(.env) ~/vonage-alerts$ touch logging_vonage.py

Now open this file—it’s time for some Python fun!

Our own HTTP Transport class will inherit from the [http_logging.AsyncHttpTransport](https://github.com/hacktlib/py-async-http-logging/wiki/3.-HTTP-Transport-Class) one. First, import the necessary libraries at the top of the file and then declare a new class like demonstrated below:

import logging
import os

from vonage import Sms

from http_logging import HttpHost, SupportClass
from http_logging.handler import AsyncHttpHandler
from http_logging.transport import AsyncHttpTransport

class VonageHttpTransport(http_logging.transport.AsyncHttpTransport):
    pass

Right now, this class will behave exactly like the original. Let’s add some custom functionality to it. The AsyncHttpTransport implements a send method, which is responsible for sending logs to a remote host. Initially, it uses the requests library for that. In our case, we have the Vonage SDK, which makes our lives a lot easier and removes HTTP protocol boredom.

Ok, enough talking. Let’s start coding with the Vonage SDK by declaring a new send method:

class VonageHttpTransport(AsyncHttpTransport):

    def send(self, events: dict, **kwargs) -> None:
        batches = self._HttpTransport__batches(events)

        sms_logs = ', '.join([
            f"{log['level']['name']}: {log['message']}"
            for batch in batches
            for log in batch
        ])

        sms_message = f'[Python Logger {self.logger_name}] {sms_logs}'

        sms_client = Sms(
            key=self.vonage_api_key,
            secret=self.vonage_api_secret,
        )

        response = sms_client.send_message({
            'from': f'Python Logger {self.logger_name}',
            'to': self.alert_phone_number,
            'text': sms_message,
        })

        if not response['messages'][0]['status'] == 0:
            raise ConnectionError(response["messages"][0].get("error-text"))

The send method takes an events argument; a list that is converted in a batch of logs using the HttpTransport.__batches method. Batches are then processed to extract basic data points into a log string.

Each log string contains only the log level name (e.g. “Warning” or “Error) and a log message. SMS stands for Short Message Service, so we want to keep our alert message short. Our primary goal is to alert, not support, the full debugging through SMS. Minimal information is sent to provide context and help the developer start the debugging process.

The logs are then concatenated using the string.join method and prefixed with the logger name for providing information regarding the application context (this should be helpful in case multiple projects are using this alerting tool).

Lastly, we instantiate a vonage.Sms client from the Vonage SDK and use it to send the SMS message to our phone. The response status is checked and, if not “OK”, we raise a ConnectionError. This error raised makes sure the log alerting mechanism is retried later and will not disrupt our main Python application, since the VonageHttpTransport class will be running in a background thread.

Notice that we are using some class attributes in the new send method: logger_name, vonage_api_key, vonage_api_secret, alert_phone_number. Let’s override the __init__ method to make sure those are properly set on class instantiation:

class VonageHttpTransport(AsyncHttpTransport):

    def __init__(
        self,
        logger_name: str,
        vonage_api_key: str,
        vonage_api_secret: str,
        alert_phone_number: str,
        *args,
        **kwargs,
    ) -> None:
        self.logger_name = logger_name
        self.vonage_api_key = vonage_api_key
        self.vonage_api_secret = vonage_api_secret
        self.alert_phone_number = alert_phone_number
        super().__init__(*args, **kwargs)

Our new HTTP Transport class is now ready. But before we move into some real logging action, we first need to create the logic that will instantiate an actual Logger object using the new VonageHttpTransport class.

Vonage Log Handler

The VonageHttpTransport class looks good, but it can’t go to battle on its own. We aren’t actually capable of using it to log anything in our applications, so let’s move one step further and make it combat-ready.

The missing piece in our puzzle is an actual HTTP Handler class. This should be an http_logging.AsyncHttpHandler, but, surely, instantiated with the custom VonageHttpTransport.

Let’s create a getLogger function inside logging_vonage.py, to mimic the Python’s native logging.getLogger behavior:

def getLogger(name: str) -> logging.Logger:
    pass

As the native Python’s getLogger function, ours takes a name string as an argument and returns an instance of the logging.Logger class. Next, we will build the functionality for this function step by step.

We start by instantiating an HttpHost. This isn’t really needed by the VonageHttpTransport, since we’re delegating HTTP requests to the Vonage SDK, but is a required part of the http-logging library API signature:

def getLogger(name: str) -> logging.Logger:
    host = HttpHost(name='vonage.com')

Next we need a SupportClass holding our HTTP Transport object:

    support_class = SupportClass(
        http_host=host,
        _transport=VonageHttpTransport(
            http_host=host,
            logger_name=name,
            vonage_api_key=os.environ.get('VONAGE_API_KEY'),
            vonage_api_secret=os.environ.get('VONAGE_API_SECRET'),
            alert_phone_number=os.environ.get('ALERT_PHONE_NUMBER'),
        ),
    )

This SupportClass object is then used to instantiate our AsyncHttpHandler:

    vonage_handler = AsyncHttpHandler(
        http_host=host,
        support_class=support_class,
    )

Finally, we instantiate a logging.Logger object, add the vonage_handler as its handler and return it:

    logger = logging.getLogger(name)
    logger.addHandler(vonage_handler)

    return logger

In the end, our getLogger function should look like the following:

def getLogger(name: str) -> logging.Logger:
    host = HttpHost(name='vonage.com')

    support_class = SupportClass(
        http_host=host,
        _transport=VonageHttpTransport(
            http_host=host,
            logger_name=name,
            vonage_api_key=os.environ.get('VONAGE_API_KEY'),
            vonage_api_secret=os.environ.get('VONAGE_API_SECRET'),
            alert_phone_number=os.environ.get('ALERT_PHONE_NUMBER'),
        ),
    )

    vonage_handler = AsyncHttpHandler(
        http_host=host,
        support_class=support_class,
    )

    logger = logging.getLogger(name)
    logger.addHandler(vonage_handler)

    return logger

Notice that the API key and secret and phone number are being retrieved from the environment variables we set at the beginning of this tutorial. This provides flexibility in case we want to use this code in multiple projects, and also avoids hardcoding API secrets, which usually is not a good idea. ;)

Multiple Handlers

The Python’s logging machinery is very powerful, and the logging.Logger object is flexible enough to extend it with multiple handlers.

As explained above, the VonageHttpTransport class will send minimal information about logs due to the inherent text length limitations of the SMS system. Nonetheless, in the event of an error that requires further debugging, we certainly will want to grab the entire stack trace, information about which line of code failed, exact timestamps, etc.

We can meet that detailed log demand by using the Logger.addHandler and adding one or more additional handlers to the Vonage Logger object.

For example, to send logs not only to our phone but also to the console, we may use the logging.StreamHandler, like demonstrated below:

import logging
import logging_vonage

logger = logging_vonage.getLogger('')
logger.addHandler(logging.StreamHandler())

Anything logged with the above logger object will be printed to the console and sent to our phone through the Vonage SMS API.

A logging.FileHandler may be used to store logs in the local filesystem if that makes sense in an implementation. You could also use the same http_logging.AsyncHttpHandler again, but in this case, sending logs to a different backend host apart from the Vonage API. Testing With a Sample Application All right, time to see some real-world action with bells and whistles. Kidding, we’re just about to make our phones beep with the Vonage SMS API. :D

Create a new file in the project directory called sample_app:

(.env) ~/vonage-alerts$ touch sample_app.py

Open it and add the following content:

import logging
import logging_vonage


logger = logging_vonage.getLogger('sampleapp')

logger.addHandler(logging.StreamHandler())

logger.debug('Debugging...')
logger.warning('You\'ve been warned!')
logger.error('This is a test error')

try:
    1/0
except ArithmeticError as exc:
    logger.exception(exc)

Notice we’re instantiating a logger object from the logging_vonage module we built earlier. The logging.StreamHandler() is also being used so that full traces are logged to our console, not only sent to our phone.

In the console, run this script with:

(.env) ~/vonage-alerts$ python sample_app.py

The following output should be printed to the console:

You've been warned!
This is a test error
division by zero
Traceback (most recent call last):
  File "/home/vonage-alerts/sample_app.py", line 14
    1/0
ZeroDivisionError: division by zero

Hopefully, if you have everything set up correctly (a Vonage account and API key/secret), you should receive an SMS message shortly with the following text:

[Python Logger sampleapp] WARNING: You've been warned!, ERROR: This is a test error, ERROR: division by zero

Debug message

Notice that the debug message 'Debugging...' was not printed to the console nor concatenated into the SMS message. That is because the default log level in the Python logging library is WARNING. The DEBUG level is considered lower than WARNING and, thus, discarded.

If you want the DEBUG message to be captured, set the level accordingly as shown below:

logger = logging_vonage.getLogger('sampleapp')
logger.addHandler(logging.StreamHandler())
logger.setLevel(logging.DEBUG)

Run the sample_app.py script again and you should see the debug message printed to the console and also concatenated into the SMS message.

Observe that, despite our logger reliance on a custom Handler (http_logging.AsyncHttpHandler) and a custom Transport (logging_vonage.VonageHttpTransport) class, it behaves just like any other Python Logger object. This makes it fully compatible as a drop-in replacement for any Python project you currently have, in case you’d like to integrate the SMS-alerting mechanism we’ve just developed throughout your stack and in any future project.

Wrapping Up

There it goes! We now have a simple and non-intrusive Python alerting tool to keep on top of what is going on with the applications we have deployed. It extends the basic Python logging native features to use the same API we’re used to, and runs anywhere our Python apps are executed. Financially, it has no fixed costs and is relatively cheap to maintain (SMS message fees only).

The http-logging library keeps a local cache of logs, so in case the Vonage API or the cell phone carrier experience any downtime or network instability, our logger can retry sending the SMS alerts some time later.