AirNow API Key

We'll be using the US government's site airnow.gov to retrieve the air quality index through its API. In order to do so, you'll need to register for an account and get your API key.

Go to this link and register. Once registered, you'll get an email containing your API key, also known as the "airnow token".

Copy and paste this key into your secrets.py file that is on the root level of your CIRCUITPY drive, so it looks something like this:

secrets = {
    'ssid' : 'your_wifi_ssid',
    'password' : 'your_wifi_password',
    'airnow_token' : 'XXX0XXXX-0000-0000-0000-XXXXXXXX0000'
}
Airnow.Gov only provides data for the United States and only for specific postal Zip codes. Please check to ensure data for your area is available. While this guide can be used as a template for using other data sources, changes to do so are not in the scope of this guide and Adafruit cannot assist in changes for other data sources.

Adafruit IO Time Server

In order to get the precise time, our project will query the Adafruit IO Internet of Things service for the time. Adafruit IO is absolutely free to use, but you'll need to log in with your Adafruit account to use it. If you don't already have an Adafruit login, create one here.

If you haven't used Adafruit IO before, check out this guide for more info.

Once you have logged into your account, there are two pieces of information you'll need to place in your secrets.py file: Adafruit IO username, and Adafruit IO key. Head to io.adafruit.com and simply click the View AIO Key link on the left hand side of the Adafruit IO page to get this information.

Then, add them to the secrets.py file like this:

secrets = {
    'ssid' : 'your_wifi_ssid',
    'password' : 'your_wifi_password',
    'airnow_token' : 'XXX0XXXX-0000-0000-0000-XXXXXXXX0000',
    'aio_username' : '_your_aio_username_',
    'aio_key' : '_your_big_huge_super_long_aio_key_'
}

CircuitPython Code

In the embedded code element below, click on the Download: Project Zip link, and save the .zip archive file to your computer.

Then, uncompress the .zip file, it will unpack to a folder named PyPortal_AirQuality.

Copy the contents of the PyPortal_AirQuality directory to your PyPortal's CIRCUITPY drive.

This is what the final contents of the CIRCUITPY drive will look like:

# SPDX-FileCopyrightText: 2019 Limor Fried for Adafruit Industries
#
# SPDX-License-Identifier: MIT

"""
This example will access the twitter follow button API, grab a number like
the number of followers... and display it on a screen!
if you can find something that spits out JSON data, we can display it
"""
import time
import board
from adafruit_pyportal import PyPortal

# Get wifi details and more from a secrets.py file
try:
    from secrets import secrets
except ImportError:
    print("WiFi secrets are kept in secrets.py, please add them there!")
    raise

# Change this to your zip code, not all locations have AQI data. Check
# https://airnow.gov/ to see if there's data available!
LOCATION = "90210"

# Set up where we'll be fetching data from
DATA_SOURCE = "http://www.airnowapi.org/aq/forecast/zipCode/?format=application/json"
# You'll need to get a token from airnow.gov, looks like '4d36e978-e325-11cec1-08002be10318'
DATA_SOURCE += "&zipCode="+LOCATION+"&API_KEY="+secrets['airnow_token']
DATA_LOCATION = [1, "AQI"]

# the current working directory (where this file is)
cwd = ("/"+__file__).rsplit('/', 1)[0]
# Initialize the pyportal object and let us know what data to fetch and where
# to display it
pyportal = PyPortal(url=DATA_SOURCE,
                    json_path=DATA_LOCATION,
                    status_neopixel=board.NEOPIXEL,
                    default_bg=0x000000,
                    text_font=cwd+"/fonts/Helvetica-Bold-100.bdf",
                    text_position=(90, 100),
                    text_color=0x000000,
                    caption_text="Air Quality Index for "+LOCATION,
                    caption_font=cwd+"/fonts/HelveticaNeue-24.bdf",
                    caption_position=(15, 220),
                    caption_color=0x000000,)

while True:
    try:
        value = pyportal.fetch()
        print("Response is", value)
        if 0 <= value <= 50:
            pyportal.set_background(0x66bb6a)  # good
        if 51 <= value <= 100:
            pyportal.set_background(0xffeb3b)  # moderate
        if 101 <= value <= 150:
            pyportal.set_background(0xf39c12)  # sensitive
        if 151 <= value <= 200:
            pyportal.set_background(0xff5722)  # unhealthy
        if 201 <= value <= 300:
            pyportal.set_background(0x8e24aa)  # very unhealthy
        if 301 <= value <= 500:
            pyportal.set_background(0xb71c1c ) # hazardous

    except RuntimeError as e:
        print("Some error occured, retrying! -", e)

    time.sleep(10*60)  # wait 10 minutes before getting again
If you run into any errors, such as "ImportError: no module named `adafruit_display_text.label`" be sure to update your libraries to the latest release bundle!

How It Works

The PyPortal Air Quality Display has a few steps it takes to provide you with the information you desire!

We'll be displaying the Air Quality Index for Particle Pollution (PM) for fine particles, 2.5 micrometers in diameter or smaller. PM2.5 particles come from all kinds of combustion.

WiFi

First, the program will connect to the WiFi network specified in the secrets.py file.

API Query and JSON

Then, it will send a request to the AirNow URL using your LOCATION United States Zip code specified in the code, as well as your AirNow token from the secrets.py file.

The query looks something like this:

http://www.airnowapi.org/aq/forecast/zipCode/?format=application/json&zipCode=90210&API_KEY=XXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX

(where all of those 'x's are your token).

When this query is complete, it returns a JSON file that looks like this:

[
  {
    "DateIssue": "2019-03-09 ",
    "DateForecast": "2019-03-10 ",
    "ReportingArea": "NW Coastal LA",
    "StateCode": "CA",
    "Latitude": 34.0505,
    "Longitude": -118.4566,
    "ParameterName": "O3",
    "AQI": 32,
    "Category": {
      "Number": 1,
      "Name": "Good"
    },
    "ActionDay": false
  },
  {
    "DateIssue": "2019-03-09 ",
    "DateForecast": "2019-03-10 ",
    "ReportingArea": "NW Coastal LA",
    "StateCode": "CA",
    "Latitude": 34.0505,
    "Longitude": -118.4566,
    "ParameterName": "PM2.5",
    "AQI": 33,
    "Category": {
      "Number": 1,
      "Name": "Good"
    },
    "ActionDay": false
  },
  {
    "DateIssue": "2019-03-09 ",
    "DateForecast": "2019-03-10 ",
    "ReportingArea": "NW Coastal LA",
    "StateCode": "CA",
    "Latitude": 34.0505,
    "Longitude": -118.4566,
    "ParameterName": "PM10",
    "AQI": 7,
    "Category": {
      "Number": 1,
      "Name": "Good"
    },
    "ActionDay": false
  },
  {
    "DateIssue": "2019-03-09 ",
    "DateForecast": "2019-03-10 ",
    "ReportingArea": "NW Coastal LA",
    "StateCode": "CA",
    "Latitude": 34.0505,
    "Longitude": -118.4566,
    "ParameterName": "NO2",
    "AQI": 22,
    "Category": {
      "Number": 1,
      "Name": "Good"
    },
    "ActionDay": false
  },
  {
    "DateIssue": "2019-03-09 ",
    "DateForecast": "2019-03-10 ",
    "ReportingArea": "NW Coastal LA",
    "StateCode": "CA",
    "Latitude": 34.0505,
    "Longitude": -118.4566,
    "ParameterName": "CO",
    "AQI": 5,
    "Category": {
      "Number": 1,
      "Name": "Good"
    },
    "ActionDay": false
  },
  {
    "DateIssue": "2019-03-09 ",
    "DateForecast": "2019-03-11 ",
    "ReportingArea": "NW Coastal LA",
    "StateCode": "CA",
    "Latitude": 34.0505,
    "Longitude": -118.4566,
    "ParameterName": "O3",
    "AQI": 29,
    "Category": {
      "Number": 1,
      "Name": "Good"
    },
    "ActionDay": false
  },
  {
    "DateIssue": "2019-03-09 ",
    "DateForecast": "2019-03-11 ",
    "ReportingArea": "NW Coastal LA",
    "StateCode": "CA",
    "Latitude": 34.0505,
    "Longitude": -118.4566,
    "ParameterName": "PM2.5",
    "AQI": 33,
    "Category": {
      "Number": 1,
      "Name": "Good"
    },
    "ActionDay": false
  },
  {
    "DateIssue": "2019-03-09 ",
    "DateForecast": "2019-03-11 ",
    "ReportingArea": "NW Coastal LA",
    "StateCode": "CA",
    "Latitude": 34.0505,
    "Longitude": -118.4566,
    "ParameterName": "PM10",
    "AQI": 9,
    "Category": {
      "Number": 1,
      "Name": "Good"
    },
    "ActionDay": false
  },
  {
    "DateIssue": "2019-03-09 ",
    "DateForecast": "2019-03-11 ",
    "ReportingArea": "NW Coastal LA",
    "StateCode": "CA",
    "Latitude": 34.0505,
    "Longitude": -118.4566,
    "ParameterName": "NO2",
    "AQI": 18,
    "Category": {
      "Number": 1,
      "Name": "Good"
    },
    "ActionDay": false
  },
  {
    "DateIssue": "2019-03-09 ",
    "DateForecast": "2019-03-11 ",
    "ReportingArea": "NW Coastal LA",
    "StateCode": "CA",
    "Latitude": 34.0505,
    "Longitude": -118.4566,
    "ParameterName": "CO",
    "AQI": 6,
    "Category": {
      "Number": 1,
      "Name": "Good"
    },
    "ActionDay": false
  }
]

You can use online code "beautifiers" such as https://codebeautify.org/jsonviewer or http://jsonviewer.stack.hu to get a nice hierarchical view of the data in the JSON file:

JSON Traversal

The JSON file is formatted in a way that makes it easy to traverse the hierarchy and parse the data. In it, you'll see keys, such as DateIssue, ReportingArea, ParameterName, and AQI, and their respective values. The key : value pair we care about for the Air Quality Display is the Air Quality Index of particle pollution (PM2.5), or AQI. This is the pair:

  • "AQI" : "33"

In order to fetch this data from the file, we need to be able to describe their locations in the file hierarchically. This is helpful, for example, in differentiating between the 'AQI' in the 'PM2.5' section vs. the 'AQI' in the 'NO2' section. To avoid name clashing we rely on JSON traversal.

In the airquality.py file, you'll see how this is done. For example, the key we want for PM2.5 is found in this hierarchy of the JSON file: [1, "AQI"]

This means there is a sub-tree to the top level of the file indexed [1], which is the second section, as the first is indexed zero. Then, below that level is the 'AQI' key we want.

The value of that key in this case is '33' which is considered to be in the "good" range of particle pollution.

Background

For the background, it displays a solid color based on the current air quality index value. 0-50 is considered good, for example, and has a green background. Here is the full set:

  • 0-50, "good", green
  • 51-100, "moderate", yellow
  • 101-150, "sensitive", orange
  • 151-200 "unhealthy", red
  • 210-300, "very unhealthy", purple
  • 301-500, "hazardous", maroon

Font

The AQI value is displayed as text created with a bitmapped font to overlay on top of the background. The fonts used here are bitmap fonts made from the Helvetica typeface. You can learn more about converting type in this guide.

At the bottom of the screen, there is a caption text that types out "Air Quality Index for " followed by the United States Zip code you entered.

PyPortal Constructor

When we set up the pyportal constructor, we are providing it with these things:

  • url to query
  • json_path to traverse and find the key:value pair we need
  • default_bg default background color
  • details of text and caption fonts, positions, and colors

Fetch

With the PyPortal set up, we can then use pyportal.fetch() to do the query and parsing of the AirNow data and then display it on screen.

Text Position

Depending on the design of your background bitmap and the length of the text you're displaying, you may want to reposition the text and caption. You can do this with the text_position and caption_position options.

The PyPortal's display is 320 pixels wide and 240 pixels high. In order to refer to those positions on the screen, we use an x/y coordinate system, where x is horizontal and y is vertical.

The origin of this coordinate system is the upper left corner. This means that a pixel placed at the upper left corner would be (0,0) and the lower right corner would be (320, 240).

So, if you wanted to move the subscriber count text to the right and up closer to the top, your code may look like this for that part of the pyportal constructor: text_position=(250, 10)

Text Color

Another way to customize your display is to adjust the color of the text. The line text_color=0xFFFFFF in the constructor shows how. You will need to use the hexadecimal value for any color you want to display.

You can use something like https://htmlcolorcodes.com/ to pick your color and then copy the hex value, in this example it would be 0x0ED9EE

Now, we'll look at mounting the PyPortal into a case for display!

This guide was first published on Mar 13, 2019. It was last updated on Mar 13, 2019.

This page (Code PyPortal with CircuitPython) was last updated on Dec 03, 2023.

Text editor powered by tinymce.