This is a personal project that was heavily inspired by Colin Waddell's project, but supplements flight information of nearby planes with real-time ADS-B and UAT data from dump1090 and dump978. Uses the FlightAware API instead of FlightRadar24 to get a plane's departure and destination airports.
Designed primarily to run on a Raspberry Pi and Raspberry Pi OS, but can be run on other setups (your mileage may vary).
As usual, this project was developed before being tracked by git
.
Show/Hide images
Neat ๐ |
- Clock additions: โ- Sunrise & sunset times (shown) โ- dump1090 signal statistics - Plane display: โ- Enhanced Readout mode (shown)โ- Blinking callsign upon switch to active plane display (shown) or plane switch - Brightness changes based on sunrise/sunset or select time-of-day โ- Brightness change when switching to active plane (shown) |
I like this, how do I build my own?
Coming Soonโข.
If you want one, I can also build one for you. (also Coming Soonโข)
- Visualize and figure out what planes are flying nearby your location, in a cool-looking way!
- Shows a plane's callsign (or registration as fallback), distance and direction from your location, the plane's country of registration, current altitude, and speed, all provided from
dump1090
- With API access you also can see the origin and destination airport, as well as how long the plane has been flying
- If you don't want to use the API, there's an available "Enhanced Readout" mode that shows even more plane info from
dump1090
, such as latitude, longitude, ground track, vertical speed, and RSSI
- Shows a plane's callsign (or registration as fallback), distance and direction from your location, the plane's country of registration, current altitude, and speed, all provided from
- It's a neat looking clock when there aren't any planes flying overhead
- When
dump1090
is running, shows overall stats like how many planes you're tracking at the moment, how many planes flew by today, and the furthest plane you can detect
- When
- Automatically switches to other plane(s) if more than one is within the area
- Fully Python based
- The python script has been verified to run in both Linux (Debian) and Windows
- Does not need to run on the same hardware that
dump1090
is running from - Reads
dump978
data if it's present as well - Customizable features such as:
- Range of which planes need to be in for detailed tracking
- Height filtering
- Units (aeronautical, metric, or imperial)
- Clock style (12 hour or 24 hour)
- ๐ Brightness based on time of day or when there's an active plane shown
- ๐ Display sunrise and sunset times or detailed signal stats for your dump1090 receiver
- Writing to a stats file that keeps count of number of planes flying by per day (and API usage as well)
- API limiting per day (those API calls can get expensive)
- ๐ Colors ๐
- Can emulate an RGB Matrix display in a browser if you don't have the actual hardware
- Detailed console output when run interactively
- Small memory footprint
- Runs from a initialization script that handles everything such as initial setup and running the python script (Linux only)
- Set up to automatically start on boot via
systemd
- Set up to automatically start on boot via
- Can be configured to run automatically inside
tmux
- ๐ Tested to work with Ultrafeeder and ADSB.im setups
- ๐ Easily update to latest builds here on Github
- Automagically migrate settings, even if new options appear or are removed in the future
Show/Hide
Using this project assumes you have the following:
- A working
dump1090
instance or similar where/data/aircraft.json
can be read- Ex:
tar1090
,piaware
/skyaware
,dump1090-fa
,dump1090-mutability
, andreadsb
- Note: the script will automatically look at these locations and choose which one works
- This script does not need to be on the same device that
dump1090
is running from (see Configuration section)
- Ex:
- Python 3.8 or newer
- A working internet connection for setup
- for Linux distros:
ssh
access if running headlessapt
as the package manager- Root access (necessary for accessing the RGBMatrix hardware)
systemd
based system
- The rgbmatrix library installed and present on the system
- Refer to adafruit's guide on how to get this working if it's not installed already
rgbmatrix
does not need to be strictly installed to run this script (see Usage section)
- The physical RGB matrix hardware (again, not strictly necessary)
- Using the adafruit matrix bonnet
- Using
32x64
sized matrix display (this is the only layout this script was designed for)
- Your location set in
dump1090
- A FlightAware API key (optional) for getting additional plane information such as origin/destination airports
- RGBMatrixEmulator (optional, installed by default when using the initalization script) for emulating the display output if you don't have the physical hardware or just want to see the output in a web browser
- a running
dump978
instance if you're in the US and live near airports that handle general aviation more than commercial flights
Make sure you meet the above prerequisites. To begin:
git clone --depth 1 https://github.com/WeegeeNumbuh1/FlightGazer
Important
Once the above command is completed, it is recommended to configure your setup now before running the initalization file. See the Configuration section below, then return to this step.
if running Linux (Debian) / Raspberry Pi
then run the following:
sudo bash FlightGazer/FlightGazer-init.sh
which will set up everything needed to run FlightGazer and then will start FlightGazer afterwards. (Click here to view what the init.sh file does)
if running Windows
You will need to put in some elbow grease here.
[!IMPORTANT] You're likely not going to be running
rgbmatrix
on Windows. Instead, useRGBMatrixEmulator
.
pip install psutil
python3 -m venv --system-site-packages "\path\to\new-python-venv"
cd "\path\to\new-python-venv\Scripts"
pip install requests
pip install pydispatcher
pip install schedule
pip install RGBMatrixEmulator
pip install suntime
pip install ruamel.yaml
If you don't care for running in a virtual environment, skip the python3 -m venv
and cd "path\to..."
lines and install the packages globally.
The config.yaml
file is where settings are configured. It has descriptions/explanations for all configurable options. It is found (and needs to be) in the same directory as the main script itself. Edit it as needed before running FlightGazer. If you changed any setting, FlightGazer must be restarted for the change to take effect.
Note
If the configuration file is missing or has invalid values, the main script has built-in fallbacks. It will alert you as necessary.
(Just don't try to purposely break the script!)
There is colors.py
in the setup
folder of FlightGazer that controls the color for each element shown on the display. You can configure the colors in that file.
Configuration details for a remote dump1090 installation
Set CUSTOM_DUMP1090_LOCATION
to the IP address of the device running dump1090.
Example: http://192.168.xxx.xxx:8080
Turning off the screen at night
ENABLE_TWO_BRIGHTNESS: true
BRIGHTNESS_2: 0
If you don't want it to turn off at sunset,
USE_SUNRISE_SETSET: false
then set BRIGHTNESS_SWITCH_TIME
to whatever time you want.
Note that FlightGazer will still be running and driving the screen even with a brightness of 0
so CPU usage will remain the same.
Only turn on the screen when there's a plane nearby (no clock)
BRIGHTNESS: 0
ENABLE_TWO_BRIGHTNESS: false
ACTIVE_PLANE_DISPLAY_BRIGHTNESS: <your value here>
Reduce flickering on a physical RGB matrix display
- Do the PWM mod
- Reserve a CPU core solely for the display
- Lower the value for
LED_PWM_BITS
(though8
seems good enough)
The main python script (FlightGazer.py
) is designed to be started by the FlightGazer-init.sh
file.
Important
By default, the script is designed to run at boot (via systemd with flightgazer.service
). You can check its status with:
systemctl status flightgazer # press 'q' to exit
# or
sudo tmux attach -d -t FlightGazer # press 'Ctrl+B' then 'd' to close
# or
journalctl -u flightgazer # use arrow keys to navigate, press 'q' to exit
However, the script and python file are also designed to run interactively in a console. If you run the following command manually:
sudo path/to/FlightGazer/FlightGazer-init.sh
The script automatically detects that you're running interactively and will display realtime output, like so:
Example output
===== FlightGazer Console Output ===== Time now: 2025-01-15 00:00:00 | Runtime: 98 days, 23:48:05
Filters enabled: <60nmi, <15000ft
(* indicates in focus, - indicates focused previously)
[Inside focus loop 64, next switch on loop 75, watching: 'aa3ae5']
Plane scratchpad: {'aa3ae5', 'a10d75'}
* [001] UAL343 (US, aa3ae5) | SPD: 263.1kt @ 288.8ยฐ | ALT: 8225.0ft, 3520.0ft/min | DIST: NW 22.4nmi (4x.005, -8x.192) | RSSI: -8.2dBFS
- [002] N167UD (US, a10d75) | SPD: 58.7kt @ 283.8ยฐ | ALT: 1100.0ft, 0.0ft/min | DIST: E 53.2nmi (4x.985, -8x.078) | RSSI: -23.2dBFS
API results for UAL343: ORD -> SFO, 0:24 flight time
> dump1090 response 28.107 ms | Processing 3.223 ms | Display formatting 0.279 ms | Last API response 349.265 ms
> Detected 154 plane(s), 2 plane(s) in range, max range: 177.2nmi | Gain: 40.2dB, Noise: -34.6dB, Strong signals: 3.4%
> API stats for today: 13 success, 0 fail, 0 no data, 0 cache hits
> Total flybys today: 13
> CPU & memory usage: 16.925% overall CPU | 7.734MiB
FlightGazer-init.sh
supports optional arguments that adjust the behavior of the main python script. Expand the table below to see all possible operating modes. Multiple flags can be passed as well.
Table of operating modes
Flag | Is interactive? |
What it does |
---|---|---|
(no flag) | โ | Default operating mode when not run as a service. Minimizes console output. Will use rgbmatrix . Uses RGBMatrixEmulator as a fallback. |
-d |
โ | Do not load any display driver. Only print console output. Overrides -e . |
-e |
โ | Use RGBMatrixEmulator as the display driver instead of actual hardware.Display by default can be seen in an internet browser. (see the Tip below) |
-f |
โ | No Filter mode. Ignores set RANGE and HEIGHT_LIMIT settings and shows all planes detected.Display will never show plane details and remain as a clock. Useful for low traffic areas. |
-t |
โ | Run in tmux . Useful for long-running interactive sessions. Default operating mode when started as a service. |
-h |
โ | Print the help message. |
Tip
An important one is -e
, which switches the display renderer from rgbmatrix
to RGBMatrixEmulator
. This is useful in case you are not able to run the display output on physical hardware and is the fallback when actual hardware is not available.
By default, RGBMatrixEmulator
can be viewed through a web browser: http://ip-address-of-device-running-FlightGazer:8888
Advanced use
There's nothing stopping you from calling the python file directly. However FlightGazer-init.sh
was designed to make running it smoother by handling the initial setup, making sure all the dependencies are there before running the actual python script, and automatically using the full paths for both the virtual python environment binaries and for the script itself, along with handling any arguments/flags that need to be passed.
You can run it like so:
sudo /etc/FlightGazer-pyvenv/bin/python3 /path/to/FlightGazer/FlightGazer.py
The main python file accepts the same arguments as the initialization script, but you can always pass -h
to see all possible operating modes.
To shutdown FlightGazer, do any one of the following:
Show/Hide
sudo systemctl stop flightgazer
sudo tmux attach -d -t FlightGazer
# then, press 'Ctrl+C' to stop
# if you started FlightGazer interactively and manually
Ctrl+C
To restart, simply do the following:
What the initialization script does
- Checks if there is an internet connection
- Checks if
first_run_complete
exists- Checks last modified date: if greater than a month, runs updates for installed dependencies
- If file exists and is new-ish, then this isn't an initial installation and we just run the main python script
- Updates package list
- Installs:
- python3-dev
- python3-venv
- tmux
- Create a new systemd service
flightgazer.service
if not present - Write out
RGBMatrixEmulator
config file - Makes virtual python environment at
etc/FlightGazer-pyvenv
- Updates
pip
as necessary and installs the following python packages in the virtual environment:- requests
- pydispatcher
- schedule
- psutil (usually provided in Raspberry Pi OS)
- suntime
- ruamel.yaml
- RGBMatrixEmulator
- Writes
first_run_complete
blank file toetc/FlightGazer-pyvenv
to show initial setup is done
Running on Windows
No fancy initialization script here. Run FlightGazer as so:
\path\to\new-python-venv\Scripts\python \path\to\FlightGazer\FlightGazer.py -i -e
or, if you didn't set up the virtual environment:
python \path\to\FlightGazer\FlightGazer.py -i -e
Pass -h
to see all operating modes.
Useful commands
Terminating all FlightGazer-related processes (Linux):
kill -15 $(ps aux | grep '[F]lightGazer.py' | awk '{print $2}')
Changing systemd
startup command
sudo nano /etc/systemd/system/flightgazer.service
systemctl daemon-reload
Disabling startup at boot
sudo systemctl disable flightgazer.service
Version v2.x and newer:
sudo bash /path/to/FlightGazer/update.sh
Windows
You can run git clone --depth 1 https://github.com/WeegeeNumbuh1/FlightGazer \a\different\directory
and manually migrate your config.
Upgrading from v.1.x to v.2.x and newer
Download update.sh
and place it inside the FlightGazer directory. Change the permissions to executable with chmod +x update.sh
. Then run the update script.
The migrator cannot migrate v.1.x configuration files to the newer format so you must manually migrate your settings.
sudo bash /path/to/FlightGazer/uninstall.sh
Windows
Simply delete the folder (and the virtual python environment if you set that up as well).
FAQ's (Open these before raising an issue)
Q: My RGB display is blank when running this, what broke?
A: Check the HAT_PWM_ENABLED
value in config.yaml
and make sure it matches your hardware setup.
Additionally, this project assumes the use of the adafruit rgbmatrix bonnet and only 1 rgb panel. Other setups are not supported.
Q: I restarted my system but it took longer for my display to start. What's going on?
A: The initialization script that starts FlightGazer checks if there are any updates to the dependencies it uses.
If it has been over a month since it last checked, then the next time it restarts, it will run these checks. It usually only adds another 30 seconds to the startup time, but if your internet connection is slow or the system is loaded with other processes, then it could take longer.
Q: I see a dot on the right of the plane readout display. What is it?
A: That is an indicator of how many planes are within your defined area. The number of dots lit up indicate how many are present. There will always be at least one lit up, all the way to 6. If the number is greater than 1, FlightGazer will start switching between planes to show you what else is flying in your area.
Q: Can I customize the colors?
A: Click here
Q: Can I customize the layout beyond what can be done in config.yaml
(clock, plane info, etc)?
A: Sure, just change some things in the script.
Q: What about showing other plane info like what airline it is or what kind of plane it is?
A: That requires additional API calls or another API entirely. Plus, to put all possible text would require scrolling which would complicate things further (I did not feel like I needed this info).
You can also use Planefence for this functionality.
Q: Why use the FlightAware API? Why not something more "free" like adsbdb?
A: adsbdb only has reported route information for aircraft that have predetermined origins and destinations. In my experience, it cannot handle position-only flights (i.e. general aviation, military, etc) and is lacking information for some flights. The commercial APIs handle these situations much more elegantly (which is the price to pay, I guess). Moreover, FlightAware's API is the only commercial one that has a usable free tier. I do wish FlightAware had a much lighter API call for pulling very basic information like what this project uses.
Q: Some of your code is not pythonic!!!1!!111
A: but it works, does it not?
Show/Hide
-
Flyby stats are not 100% accurate (but can be close, depending on your
FLYBY_STALENESS
setting inconfig.py
)- This stat relies on the number of unique planes seen, not each occurence of an actual flyby
- This is somewhat by design, covering the case of living near a general aviation airport and having the same plane do numerous touch-and-go landings
For example, if plane with hex IDabcdef
passes by at 06:00, then passes by again at 18:00, it won't count as a flyby
This has been fixed in v.1.3.0 with the addition of a new parameter,FLYBY_STALENESS
- This stat relies on the number of unique planes seen, not each occurence of an actual flyby
-
If using
No Filter
mode and restarting FlightGazer often, we can artifically inflate the flyby count- FlightGazer has a feature where it will write out stats before shutting down so that it can reload those stats upon restart (if it's the same day). The flyby count is simply a number and has no additional information such as the IDs of planes
- Upon reload, FlightGazer fills in dummy IDs equal to the value of the last written flyby count in its internal list of plane IDs it keeps track of for flybys
- The flyby count runs under the assumption that the flyby area itself is small, but since
No Filter
removes that restriction, it's a free-for-all - This is not usually a problem, as long as we don't restart often in the same day
- May not ever get fixed
-
On rare occasions are times when there will be two entries of the same plane
- This is an edge case that's been noted since the v.0.x days due to dump1090 listing the same plane twice
- This is further complicated if dump978 is also active and the plane uses a dual mode transponder
- Having MLAT enabled also increases the chances of this bug occurring
- So far there is no fix planned for this as this does not break core functionality
- At worst it confuses the plane selector algorithm but even then it still selects normally
- May be mitigated in the far future
-
If FlightGazer crashes when run in tmux via systemctl, it will always report an exit status of 0
- Complicated to fix due to all the signal handling between the main script and the initialization script
- May not actually get fixed (simply just code better, smh)
Found a bug? Want to suggest a new feature? Open an issue here on Github.
Read: Changelog.txt
.
Faraway ideas:
- Brightness changing
- Docker image?
- Settings management from webpage?
- Another
dump1090
xrgbmatrix
project, but renders out a minimap instead and uses larger RGB matrix panels- Fun fact: I used the same font from the above project for FlightGazer
- This All About Circuits Article from 2017
- Uses all the same core components that this project relies on at a surface-level: FlightAware's API (the older
Firehose
one),dump1090
,rgbmatrix
- Uses all the same core components that this project relies on at a surface-level: FlightAware's API (the older
- Planefence, a logger for all the planes that flyby your location
- Inspired the functionality of the stats file FlightGazer writes out
* (dust) *
* (to be chosen eventually) *
Huge shout out to RGBMatrixEmulator. This tool was invaluable for getting the layout dialed in and figuring out the logic needed to update the display correctly, all while avoiding having to program directly on the Raspberry Pi itself (VSCode Remote on a Zero 2W is literally impossible, I've tried).
^ Thanks to the fellow tech nerds in here for all the suggestions over the evolution of this project