Add an OLED Stats Display to Raspberry Pi OS Bookworm

Raspberry Pi OS Bookworm has been out for a little over a year now. It comes with a few visual changes like a Wayland native taskbar with plugins like a CPU and GPU monitor, and a new default background, but far more changes have been made under the hood. One of these changes is the requirement to use a virtual environment to install third-party packages with pip rather than installing them system-wide. This places packages in a managed space instead of at system level, which lowers the risk of causing conflicts or breaking other software packages or the operating system.

Pi-OS-Bookworm-Changes

Unfortunately, this change along with a few tweaks to the way the GPIO pins are used has made my previous tutorial for installing an OLED stats display on a Pi 4 outdated. If you try to follow this tutorial you’ll end up with a number of errors around virtual environments and broken libraries.

So today I’m going to run through how to connect and program and OLED stats display on a Raspberry Pi 5 running PI OS Bookworm.

Here’s my video tutorial, read on for the written tutorial;

What You Need For This Project

Note: The above parts are affiliate links. By purchasing products through the above links, you’ll be supporting this channel, with no additional cost to you.

Connect the OLED Display to Your Raspberry Pi

To start out, let’s get the OLED display connected to our Raspberry Pi.

OLED-Stats-Display-Hardware

To do this, you’ll need a 4-wire female-to-female jumper cable. The colours don’t matter, they just help you to keep track of which wire goes to which terminal.

4-Wire-Ribbon-Cable

The OLED display’s terminals are labelled on the front, so make sure that you take note of them or label the wires before installing it in a case or you won’t be able to see them.

The pin arrangement on these displays is most commonly GND, VCC, SCL and SDA.

I2C-OLED-Display-Pinout

You can’t just copy this arrangement though as there are versions of this display that have the GND and VCC pins switched around.

I2C-OLED-Pinout-Differences

If you connect power to the display incorrectly, even just a single time, you’ll likely damage it and it will no longer work, even if you correct the wiring afterwards – ask me how I know this.

Plug your jumper cable into these four pins and then take note of which colours you’ve got connected to which pins.

Next, we can plug the other ends into our Raspberry Pi’s GPIO pins. Make sure that your Pi is off and power is disconnected before doing this, you don’t want to short a connection or plug a lead into an incorrect pin and not have a chance to check the connections before powering it up.

GPIO-Pinout-Diagram
Source – https://www.raspberrypi.com/documentation/computers/raspberry-pi.html

There are a few options for the power pins. I usually plug the GND jumper into pin 9 and the VCC jumper into pin 1. You can use any of the pins labelled ground for the ground jumper and these displays will run on 3.3V or 5V so you can use any of these labelled pins for the VCC jumper.

  • GND to Pin 9 (Ground)
  • VCC (3.3V) to Pin 1 (3.3V)
  • SCL to Pin 5 (GPIO 3)
  • SDA to Pin 3 (GPIO 2)

Next, we need to connect the SCL and SDA jumpers, which just get plugged into the corresponding GPIO pins. Plug SCL into pin 5 and SDA into pin 3. Don’t get mixed up with the GPIO numbers and Pin numbers, I’m working off the physical pin numbers.

Display-Connected-To-GPIO-Pins-Both-Sides

Quickly recheck your connections and we can then start programming the display.

Programming The OLED Stats Display

I’m going to be working on a fresh 64-bit install of Raspberry Pi OS Bookworm, which you can flash to your microSD card or SSD using Raspberry Pi Imager.

Plug the microSD card into your Pi’s card reader and then plug in your power adaptor to boot it up.

Plug-MicroSD-Card-Into-Card-Reader

If this is your first boot, you’ll need to go through a few setup steps and you’ll then land on the desktop.

Raspberry-Pi-OS-First-Boot

It’s possible to do this installation on a headless Pi too as we’ll only be working in the terminal.

Updating The Pi & Installing Libraries

Open up a new terminal window or SSH into your headless Pi and we’ll start by making sure that the Pi is up to date by entering the following commands. When the updates are finished installing, reboot the pi.

sudo apt-get update
sudo apt-get -y upgrade
sudo reboot
Terminal-Updates

Once rebooted, open up the terminal again and then enter these commands to install two libraries.

sudo apt-get install python3-pip
sudo apt install --upgrade python3-setuptools
Terminal-Library-Installation

Now we need to create a virtual environment to continue the setup in. We’ll call this stats_env.

sudo apt install python3-venv
python3 -m venv stats_env --system-site-packages
source stats_env/bin/activate

Once activated, you should see stats_env at the start of your current terminal line.

Terminal-Virtual-Environment

Next, let’s install the Adafruit Blinka library. Once complete, confirm yes at the end to reboot your pi.

cd ~
pip3 install --upgrade adafruit-python-shell
wget https://raw.githubusercontent.com/adafruit/Raspberry-Pi-Installer-Scripts/master/raspi-blinka.py
sudo -E env PATH=$PATH python3 raspi-blinka.py
Terminal-Python-Package-Updates

With the Pi rebooted, let’s now check that it can see the connected display. Enter this command and you should then see an address table and importantly you should get the characters 3c in the top right area of the table.

sudo i2cdetect -y 1
        0  1  2  3  4  5  6  7  8  9  a  b  c  d  e  f
    00:                         -- -- -- -- -- -- -- --
    10: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
    20: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
    30: -- -- -- -- -- -- -- -- -- -- -- -- 3c -- -- --
    40: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
    50: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
    60: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
    70: -- -- -- -- -- -- -- --
I2C-Address-Register

If you don’t see any characters in the table then your Pi isn’t recognising the connected display. First, check your display is connected correctly and then make sure that the I2C interface is enabled. If you get a table full of characters then you’ve probably made a wiring mistake as this happens if SDA is shorted to ground.

To check that the I2C interface is enabled, use the below command to open up configuration options, then select “3 Interfacing Options”, then select “I5 I2C”, “Yes” to enable the interface, “Ok” and then “Finish”

sudo raspi-config

Next, we can install the CircuitPython libraries specific to this display. We start by re-entering the virtual environment and then enter these commands to install the libraries.

source stats_env/bin/activate
pip3 install --upgrade adafruit_blinka
pip3 install adafruit-circuitpython-ssd1306
sudo apt-get install python3-pil
Terminal-Python-Packages-Second-Install

Running The Stats Script

We can then exit the virtual environment and download the stats display Python script from my GitHub repository.

deactivate
sudo apt-get install git
git clone https://github.com/mklements/OLED_Stats.git

We then need to re-enter the virtual environment to run the script.

source stats_env/bin/activate
cd OLED_Stats
Terminal-Script-Execution

There are two options for scripts to run. The first is a text-based one called stats.py and the second is one with icons called monitor.py. You’ll need to change the filename in the run command depending on which one you’d like to run.

python3 stats.py

or

python3 monitor.py
OLED-Icons-Stats-Display-Monitor

The script is now running and your display should show your IP address and system stats.

Stats-Display-Script-Running

We can’t leave it like this though, if we close the terminal or reboot the Pi, the script will terminate and the display will stop being updated. So we need to follow some additional steps to get the script to run automatically on startup.

Programming The Pi To Run The Script On Startup

This is quite easy to do, we just need to copy this file from my GitHub repository into the root directory, then make it executable and then use crontab to tell it to run on startup. You’ll need to open a new terminal window for the below steps. Remember to change your username (“pi” below) if you’re not using a default username

curl -OL https://raw.githubusercontent.com/mklements/OLED_Stats/main/OLED_display
sudo chmod +x /home/pi/OLED_display

The OLED_display script runs the stats.py file by default. To change this to the monitor.py file, you’ll need to open it up in a text or code editor and change the target filename from stats.py to monitor.py.

Open up crontab by entering this command. If this is the first time you’re opening crontab then you’ll be prompted to select an editor, select 1 and hit enter.

crontab -e
Crontab-Automation-Running

Then add this line to the end of the file. Make sure to change your name to your username if you’re not using the default username pi.

@reboot /home/pi/OLED_display &
Crontab-Line-Added

You should now have a working OLED stats display that starts up automatically each time your pi boots up.

Text-Based-Stats-Display

You can go ahead and install it into your case if you haven’t done so already. My favourite case at the moment is my Pimoroni NVMe case which accommodates an NVMe drive underneath the Pi and an Ice Tower for cooling.

Let me know what you’re using your OLED stats display for in the comments section below, or if there’s anything you’d like to see added to the display script in the future.

Michael Klements
Michael Klements
Hi, my name is Michael and I started this blog in 2016 to share my DIY journey with you. I love tinkering with electronics, making, fixing, and building - I'm always looking for new projects and exciting DIY ideas. If you do too, grab a cup of coffee and settle in, I'm happy to have you here.

12 COMMENTS

  1. This is really great. Thanks for sharing. Being that I am from the US can you reply with the replacement code for Temp but in fahrenheit?

    • The Pi reports temperature in celsius by default and it is text based, so you’d need to isolate and convert it to a number, then convert it to farenheit and then back to text. The below should work. You also need to change the display line to now use temp_display rather than str(Temp, ‘utf-8’)

      temp_celsius = float(Temp.decode(‘utf-8’).strip().replace(“‘C”, “”))
      temp_fahrenheit = (temp_celsius * 9 / 5) + 32
      temp_display = f”Temp: {temp_fahrenheit:.1f}°F”

  2. Hi Michael, Thanks for sharing the updated OLED STATS for Bookworm. I would like to use them on a 1.3″ SH1106 Oled display. What would I need to do differently?

    • There is a bit of work involved to use a different display. To use a SH1106 OLED display instead of the SSD1306 in the original script, you need to:

      Use a library that supports the SH1106 driver.
      The adafruit_ssd1306 library only supports SSD1306 displays. For SH1106, you have two main approaches:

      Option A: Use Adafruit’s DisplayIO-based SH1106 Library

      Install the adafruit_displayio_sh1106 library.
      Adjust your code to use displayio for rendering graphics and text.

      Option B: Use the Luma.OLED Library

      Install luma.oled (if not already installed).
      Change the import statements and initialization to use luma.oled’s sh1106 device class.

  3. This is great thanks Michael.
    Is there a version of the script that works with the UPS from your previous UPS case example?
    Thanks.

    • That particular UPS is not officially compatabile with the Pi 5 due to the higher power requirements. I’m looking at updating a script to go with the new version of that UPS.

  4. I ran into an issue with numa.oled running on Raspberry Pi5
    RuntimeError: Cannot determine SOC peripheral base address

    These commands fixed the issue
    sudo apt remove python3-rpi.gpio
    pip3 install rpi-lgpio

  5. Hi Michael,
    thank you for this great project.
    Is there an option to show the disk space of my ZFS pool instead of the space of the SD Card of the system (I mean row 4)?
    Greetings,
    Andreas

    • Yes, you can modify your script to display the disk space of a ZFS pool instead of the boot drive. To achieve this, you’ll need to use the zpool command to retrieve the disk space details of your ZFS pool.
      Something like this;
      cmd = “zpool list -H -o name,size,alloc,free | awk ‘{printf \”%s: %s/%s Used\”, $1, $3, $2}'”
      ZFSSpace = subprocess.check_output(cmd, shell=True).decode(‘utf-8’).strip()
      draw.text((0, 48), ZFSSpace, font=font, fill=255)

LEAVE A REPLY

Please enter your comment!
Please enter your name here

Latest posts

N97 vs N100 vs Raspberry Pi 5: Which Is Right For You?

I recently did some testing to compare the performance and features of an N100 mini PC as a replacement for a Raspberry Pi 5....

Raspberry Pi 5 Case With An Integrated Water-Cooling Loop

A while back, I built a water-cooled Raspberry Pi 4 computer using a Pi CM4 module and the official IO board. This computer and...

Gweike G2 Pro 30W Fibre Laser Unboxing & Review

I've been using diode and CO2 lasers in my home workshop for quite some time. Combined with a 3D printer, they've been fantastic for...

Related posts