Track, store, and alert on aircraft movement from an ADS-B locally.
SkyFollower reads data from your ADS-B output using the raw message output, enriches that data by calling an external web service for aircraft registration and operator information, and sends notifications about tracked aircraft movements using a rules-based engine. All of the output is stored in MongoDB for any other use you might have in a simple, easy-to-query format.
If you're using Home Assistant for home automation, it can also raise events that can be used to drive other actions in your home automation, like announcing an aircraft passing overhead.
As your ADS-B receiver publishes data, SkyFollower reads that data into a local in-memory database. As messages are received from the aircraft, a profile of the flight is built using the in-memory data until a loss of signal occurs with the tracked aircraft for a period of time at which point the flight is considered completed. Completed flights are persisted to MongoDB.
If the service exits, the in-memory data is persisted to MongoDB immediately.
SkyFollower can run on pretty much any operating system that runs Python 3, but this document will be focused exclusively on explaining how to do so under Ubuntu 20.04 (Focal Fossa).
MongoDB is a document-based datastore and is perfect for operations like tracking historical aircraft movement. It is recommended, but not required, for MongoDB to be installed on the same system as SkyFollower.
MongoDB 4.4 Community Edition is recommended. Follow their installation instructions. Installing MongoDB later than 4.4 may not be possible on your computer, depending on the age and processor capabilities, and has not been tested.
Only perform this operation if you intend to query MongoDB from a separate computer. Set the network binding to allow remote connections.
sudo nano /etc/mongod.conf
net:
port: 27017
bindIp: 0.0.0.0
Restart mongo
sudo systemctl restart mongod
Using apt, install the prerequisite packages, if you don't already have them installed:
sudo apt-get install -y python3 python3-pip libgeos-dev
Python 3.10.5 is recommended. Versions of Python prior to 3.10 have known issues with performance.
Python also requires a number of packages that must be installed to use SkyFollower, note version 2.11 is recommended (pip default is 2.9):
sudo pip3 install pyModeS==2.11 pymongo requests paho-mqtt shapely watchdog schedule
This step is optional, but SkyFollower can retrieve aircraft registration and operator information to enrich your data. This can be helpful if you want to trigger alerts about specific operators or registration numbers. To do so, download and configure the Aircraft Registration and Operator Information.
Clone SkyFollower from GitHub:
sudo git clone https://github.com/BrentIO/SkyFollower.git /etc/P5Software/SkyFollower
You must configure these settings in the settings.json file.
sudo nano /etc/P5Software/SkyFollower/settings.json
Parameter | Default | Description |
---|---|---|
latitude |
38.8969137 | Latitude of the receiver |
longitude |
-77.0357096 | Longitude of the receiver |
adsb -> uri |
my.adsb.server.lan | IP address or domain name of your ADS-B receiver. |
The settings.json file contains all of the user-configurable settings for SkyFollower. In addition to those in the Required Configuration section, optional parameters are described below.
Parameter | Default | Description |
---|---|---|
log_level |
info | Amount of logging that should be written to the text log file. Options are debug , error , warning , critical . If using debug , be mindful that this will cause significant writes, and is intended for debugging purposes only. If omitted, info is used. |
flight_ttl_seconds |
300 | The number of seconds after loss of signal until the flight will be concluded and the data persisted to MongoDB. Configuring this option too low will cause fragmentation of data for long overhead flights with short losses of signal; too high will cause erroneous data where two flights are merged into one long flight, such as a quick turn on an aircraft at a nearby airport. Any positive integer is allowed. |
local_database_mode |
memory | Determines if the cached database is stored in memory or disk. Options are disk or memory . If using disk, be mindful that this will cause significant writes, and is intended for debugging purposes only. |
queue_reader_thread_count |
Number of System CPU's | Overrides the number of queue reader threads, which read ADS-B messages from the source and into SkyFollower. Configuring below 1 is not permitted, and a warning will be issued if you exceed the number of system CPU's. The number of required threads is heavily dependent the message arrival rates. Additional factors which can influence the required number of worker threads includes the speed of response from AROI, the number (and overall complexity) of rules, and the number of geometries. A good rule of thumb is that you should have a minimum of 1 thread per 150 messages/second you receive. |
adsb -> port |
30002 | The port of your ADS-B receiver where the data can be found. Use 30002 when setting the adsb -> type to raw . |
adsb -> type |
raw | The data format your ADS-B receiver is sending to SkyFollower. Use raw when setting the adsb -> port to 30002 . |
mongoDb -> uri |
localhost | URL for the MongoDB server. You should only need to change this if you are not running SkyFollower and MongoDB on the same device. |
mongoDb -> port |
27017 | Port for your MongoDB server. You should only need to change this if you did a custom installation of MongoDB and specifically changed the port binding. |
mongoDb -> database |
SkyFollower | Name of the MongoDB database to store the flight data. |
mongoDb -> collection |
flights | Name of the document collection in the database to store the flight data. |
registration |
If this object is ommitted, retrieving registration data will be disabled. Useful for converting the ICAO hex code to a registration object. For example, querying A8AE7F would return to an object for a Boeing 757. |
|
registration -> enabled |
false | Controls if retrieving registration data is enabled or disabled. |
registration -> uri |
http://localhost:8480/registration/icao_hex/$ICAO_HEX$ | The URL of the webserver where the registration microservice runs. Note, |
registration -> x-api-key |
some_secret_key | The value of the x-api-key header that should be sent for authenticating the request to the service. |
operators |
If this object is ommitted, retrieving operator data will be disabled. Useful for converting aircraft callsigns to operators. For example, querying for DAL would return an object for Delta Air Lines. |
|
operators -> enabled |
false | Controls if retrieving operator data is enabled or disabled. |
operators -> uri |
http://localhost:8480/operator/$IDENT$ | The URL of the webserver where the operators microservice runs. Note, DAL2 . |
operators -> x-api-key |
some_secret_key | The value of the x-api-key header that should be sent for authenticating the request to the service. |
flights |
If this object is ommitted, retrieving flight data will be disabled. Useful for identifying aircraft origin and destination information. For example, querying for ident DAL1 would return a a flight from KJFK to EGLL. |
|
flights -> enabled |
false | Controls if retrieving flight data is enabled or disabled. |
flights -> uri |
http://localhost:8480/flight/$IDENT$ | The URL of the webserver where the flights microservice runs. Note, DAL1 .\n An optional parameter ?airport_icao= can be provided to further refine query results based on the focused airport. For example, adding ?airport_icao=KATL would return flights that only originate or depart from KATL , which is useful when flight number might be used for multiple segments of a flight. |
flights -> x-api-key |
some_secret_key | The value of the x-api-key header that should be sent for authenticating the request to the service. |
mqtt |
If this object is omitted, MQTT will be disabled. | |
mqtt -> enabled |
false | Controls if MQTT is enabled or disabled. |
mqtt -> uri |
my.mqtt.server.lan | IP address or domain name of your MQTT broker. |
mqtt -> port |
1883 | Port for your MQTT broker. |
mqtt -> username |
my_username | Username to use when authenticating to your MQTT broker. |
mqtt -> password |
my_clear_text_password | Password to use when authenticating to your MQTT broker. |
mqtt -> topic |
SkyFollower | Root for all topics. |
files |
Locations to additional files used by SkyFollower | |
files -> areas |
./areas.geojson | Path to the defined areas GeoJSON file |
files -> rules |
./rules.json | Path to the notification rules |
home_assistant |
If this object is omitted, Home Assistant will be disabled. | |
home_assistant -> enabled |
false | Controls if Home Assistant integration is enabled or disabled. |
home_assistant -> discovery_prefix |
homeassistant | The prefix that Home Assistant is listening to for auto discovery messages. By default, Home Assistant is listening for auto discovery on the homeassistant MQTT topic and all child topics. |
Copy the service file to the systemctl directory
sudo mv /etc/P5Software/SkyFollower/SkyFollower.service /lib/systemd/system/
Reload the systemctrl daemon to pick up the new SkyFollower service
sudo systemctl daemon-reload
Enable the service to run
sudo systemctl enable SkyFollower.service
Start the service
sudo systemctl start SkyFollower.service
An area is a simple GeoJSON polygon of an interesting area. The feature must include a property called name
to be permitted. A great tool to define your GeoJSON can be found at https://geojson.io. You can include geometries other than polygons, but SkyFollower will ignore anything that is not a polygon.
Below is an example which will be used in the next section, note that a name
property of LI
has been added to the properties
section. You may also find this example in the areas.example.geojson file.
{
"type": "FeatureCollection",
"features": [
{
"type": "Feature",
"properties": {
"name": "LI"
},
"geometry": {
"type": "Polygon",
"coordinates": [
[
[
-73.8006591796875,
40.82835864973048
],
[
-73.97369384765625,
40.734770989672406
],
[
-74.03961181640625,
40.54720023441049
],
[
-73.7677001953125,
40.538851525354666
],
[
-73.245849609375,
40.58684239087908
],
[
-72.70751953125,
40.73685214795608
],
[
-71.8560791015625,
40.97160353279909
],
[
-71.80938720703125,
41.044145364313174
],
[
-71.89727783203125,
41.14970617453726
],
[
-72.11700439453125,
41.21998578493921
],
[
-72.36145019531249,
41.17038447781618
],
[
-72.70477294921874,
41.03585891144301
],
[
-72.83935546875,
41.04621681452063
],
[
-73.1964111328125,
40.994410999439516
],
[
-73.553466796875,
40.94671366508002
],
[
-73.8006591796875,
40.82835864973048
]
]
]
}
}
]
}
SkyFollower will optionally raise an event via MQTT based on a configured rule being met.
Rules are comprised of the following attributes. Note that the field names are case sensitive.
Field | Data Type | Required | Example | Meaning |
---|---|---|---|---|
name |
string | No | Any Boeing 757-200 | An arbitrary name that is human-friendly to help identify the given rule |
description |
string | No | Boeing 757's are my favorite airplanes | An arbitrary description that is human-friendly to help identify the given rule |
identifer |
string | Yes | All_B752 | A unique identifier that will be sent in the MQTT message that may be used by other systems to apply further notification processing. The identifier is appended to the MQTT topic |
enabled |
boolean | Yes | true | Indication if the rule should be processed or ignored |
conditions |
array | Yes | See below | The array of conditions that must be satisifed for this rule to trigger. Conditions are treated as an AND is inserted between them |
Conditions are evaluated individually, but all conditions must evaluate to true
in order for the rule to be triggered, resulting in an MQTT notification being sent.
The structure of a condition is consistent across each condition type. Values are always strings, and are not case sensitive.
type
is the type of condition evaluation which should occur (see below)value
is the value which will be evaluatedoperator
indicates how the value should be evaluted. Permitted values areequals
,minimum
, andmaximum
. Numeric values are inclusive when usingminimum
ormaximum
Type | Value | Permitted Operators | Meaning |
---|---|---|---|
aircraft_icao_hex |
Any | equals |
Aircraft ICAO hex |
aircraft_powerplant_count |
Any positive integer | equals , minimum , maximum |
Number of powerplants attached to the aircraft |
aircraft_registration |
Any | equals |
Aircraft registration mark |
aircraft_type_designator |
Any | equals |
ICAO type designator for the aircraft |
altitude |
Any positive integer | minimum , maximum |
Aircraft altitude MSL |
area |
Any | equals |
Name of the feature collection in the geoJSON file |
ident |
Any | equals |
Flight Identity |
date |
ISO-8601 date format YYYY-mm-dd | equals , minimum , maximum |
Date in GMT |
heading |
Tuple | equals |
Aircraft ground track in degrees, where the first value is the minimum heading and the second value is the maximum heading. Legal vales are 0 to 359 , inclusive |
military |
Boolean | equals |
If the aircraft is marked as known military |
operator_airline_designator |
Any | equals |
ICAO airline designator |
squawk |
Any positive integer | equals |
Transponder squawk code, including leading zeros as necessary |
velocity |
Any positive integer | minimum , maximum |
Forward velocity of the aircraft in knots |
vertical_speed |
Any integer | minimum , maximum |
Vertical speed of the aircraft in feet per minute |
wake_turbulence_category |
One of: Light , Medium , Medium 1 , Medium 2 , High Vortex Aircraft , Heavy , Super , Rotorcraft , High Performance |
equals |
Aircraft wake turbulence category, as reported from the registry or through the ADS-B message |
If any condition in the trigger evalutes to false, the entire rule evaluates to false.
You may also find all of these examples in the rules.example.json file.
- Any aircraft with an altitude at or below 10,000ft
{
"name": "All aircraft below 10,000",
"identifier": "acft_10k_and_below",
"description": "Any aircraft with an altitude at or below 10,000ft",
"enabled" : true,
"conditions": [
{
"type": "altitude",
"value": "10000",
"operator": "maximum"
}
]
}
- United Airlines Boeing 757's between 12,000 and 15,000ft heading north (between 340 and 020 degrees)
{
"name": "UAL B757-200",
"identifier": "Northbound_United_B75s_12k-15k",
"description": "United Airlines Boeing 757-200's between 12,000 and 15,000ft heading north after takeoff",
"enabled" : true,
"conditions": [
{
"type": "altitude",
"value": "15000",
"operator": "maximum"
},
{
"type": "altitude",
"value": "12000",
"operator": "minimum"
},
{
"type": "operator_airline_designator",
"value": "UAL",
"operator": "equals"
},
{
"type": "aircraft_type_designator",
"value": "B752",
"operator": "equals"
},
{
"type": "heading",
"value": "340,020",
"operator": "equals"
},
{
"type": "vertical_speed",
"value": "500",
"operator": "minimum"
}
]
}
- Delta Flight 2 on Christmas Eve Descending at least 100FPM over Long Island
Refer to the "Areas" section of this guide for the definition of Long Island.
{
"name": "Grandma's Flight Home",
"identifier": "grandma",
"description": "Grandma's Flight Home Arriving on Christmas Eve",
"enabled" : true,
"conditions": [
{
"type": "ident",
"value": "DAL2",
"operator": "equals"
},
{
"type": "date",
"value": "2022-12-24",
"operator": "minimum"
},
{
"type": "date",
"value": "2022-12-24",
"operator": "maximum"
},
{
"type": "vertical_speed",
"value": "-100",
"operator": "maximum"
},
{
"type": "area",
"value": "LI",
"operator": "equals"
}
]
}
Home Assistant will store all of the sensor data changes, which is repetitive with MongoDB and not recommended. Add this to your configuration.yaml
file to filter out storage of SkyFollower events:
recorder:
exclude:
entity_globs:
- sensor.skyfollower_*
- Can I run this on Raspberry Pi? Yes, but...
- There will be changes needed to the installation above, but it should work.
- MongoDB will do a lot of file I/O operations, so if you're running this from an SD card, expect the SD card to die pretty quickly. If you run MongoDB on a separate computer with an SSD/NVMe/traditional hard drive, your experience will be much better and it should be fine.
- Data is stored in memory while SkyFollower is actively tracking the aircraft. If you receive a lot of traffic with your receiver, this may also saturate the limited memory in a Raspberry Pi and cause issues for SkyFollower or other applications.
- What ADS-B receivers is this compatible with?
- Most ADS-B receivers will output a
raw
format, including FlightAware's SkyAware and PlaneFinder's PlaneFinderClient. Running this software does not interfere with those services and can live happily beside them.
- Most ADS-B receivers will output a
- Do I need to know about database management to make this work?
- No, but you might want to learn how to query MongoDB to add additional features or get information that interests you.
- How do you differentiate between a loss of signal and a different flight?
- The setting
flight_ttl_seconds
defines the number of seconds since loss-of-signal where the flight is determined done and the data is then persisted to MongoDB.
- The setting
- What happens when my computer shuts down or I stop the SkyFollower service?
- The ADS-B listener is stopped and data stored in memory is immediately written to MongoDB. Restarting SkyFollower immediately will likely cause a duplication of the flights currently being received, since SkyFollower doesn't retroactively merge documents in MongoDB.
- Is this going to break my map or other tools that I currently use with dump1090 or the like?
- SkyFollower listens to the same stream of data that your other tools do, and therefore should not interfere with those functions. Custom maps, real-time statistics, and other neat add-on's that you might be using will continue to work unaffected. Note, however, that if you install the Aircraft Registration and Operator Information, it is possible this could conflict with the port your map uses, so read those docs carefully.
- The service won't start and there's nothing in any of the logs. What do I do?
- Try starting the service manually by using
sudo python3 /etc/P5Software/SkyFollower/main.py
. If there is an error, it will usually be printed on the screen for you to see.
- Try starting the service manually by using