129 lines
3.4 KiB
Python
Executable file
129 lines
3.4 KiB
Python
Executable file
#!/usr/bin/env python
|
|
|
|
"""Read events from rtl_433 and gpsd and print out."""
|
|
|
|
# Needs gpsd (and the Python support from gpsd)
|
|
# Start gpsd and rtl_433 (rtl_433 -F syslog::1433), then this script
|
|
|
|
from __future__ import print_function
|
|
|
|
import socket
|
|
import json
|
|
import gps
|
|
import threading
|
|
|
|
# rtl_433 syslog address
|
|
UDP_IP = "127.0.0.1"
|
|
UDP_PORT = 1433
|
|
|
|
|
|
class GpsPoller(threading.Thread):
|
|
def __init__(self):
|
|
threading.Thread.__init__(self)
|
|
self.gps = gps.gps(mode=gps.WATCH_ENABLE)
|
|
self.running = True
|
|
|
|
def run(self):
|
|
while self.running:
|
|
self.gps.next()
|
|
|
|
@property
|
|
def utc(self):
|
|
return self.gps.utc
|
|
|
|
@property
|
|
def fix(self):
|
|
return self.gps.fix
|
|
|
|
@property
|
|
def satellites(self):
|
|
return self.gps.satellites
|
|
|
|
|
|
def parse_syslog(line):
|
|
"""Try to extract the payload from a syslog line."""
|
|
line = line.decode("ascii") # also UTF-8 if BOM
|
|
if line.startswith("<"):
|
|
# fields should be "<PRI>VER", timestamp, hostname, command, pid, mid, sdata, payload
|
|
fields = line.split(None, 7)
|
|
line = fields[-1]
|
|
return line
|
|
|
|
|
|
def prife(label, data, key):
|
|
"""Print if exists."""
|
|
if key in data:
|
|
print(label, data[key])
|
|
|
|
|
|
|
|
def report_event(data, gpsp):
|
|
"""Print out an rtl_433 event with gps data."""
|
|
|
|
# don't process if it isn't sensor data
|
|
if "model" not in data:
|
|
return
|
|
|
|
# don't process if it isn't TPMS data
|
|
if "type" not in data:
|
|
return
|
|
if data["type"] != "TPMS":
|
|
return
|
|
|
|
# now = int(time.time())
|
|
print("----------------------------------------")
|
|
print("Model ", data["model"])
|
|
prife("ID ", data, "id")
|
|
prife("Status ", data, "status")
|
|
prife("State ", data, "state")
|
|
prife("Flags ", data, "flags")
|
|
prife("Code ", data, "code")
|
|
prife("Pressure (kPa) ", data, "pressure_kPa")
|
|
prife("Pressure (PSI) ", data, "pressure_PSI")
|
|
prife("Temperature (C)", data, "temperature_C")
|
|
prife("Temperature (F)", data, "temperature_F")
|
|
print()
|
|
print("latitude ", gpsp.fix.latitude)
|
|
print("longitude ", gpsp.fix.longitude)
|
|
print("time utc ", gpsp.utc, " + ", gpsp.fix.time)
|
|
print("altitude (m) ", gpsp.fix.altitude)
|
|
print("eps ", gpsp.fix.eps)
|
|
print("epx ", gpsp.fix.epx)
|
|
print("epv ", gpsp.fix.epv)
|
|
print("ept ", gpsp.fix.ept)
|
|
print("speed (m/s) ", gpsp.fix.speed)
|
|
print("climb ", gpsp.fix.climb)
|
|
print("track ", gpsp.fix.track)
|
|
print("mode ", gpsp.fix.mode)
|
|
# print("sats ", gpsp.satellites)
|
|
|
|
|
|
if __name__ == '__main__':
|
|
gpsp = GpsPoller()
|
|
|
|
sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM, socket.IPPROTO_UDP)
|
|
sock.bind((UDP_IP, UDP_PORT))
|
|
|
|
try:
|
|
gpsp.start()
|
|
|
|
while True:
|
|
line, addr = sock.recvfrom(1024)
|
|
try:
|
|
line = parse_syslog(line)
|
|
data = json.loads(line)
|
|
report_event(data, gpsp)
|
|
|
|
except KeyError:
|
|
pass
|
|
|
|
except ValueError:
|
|
pass
|
|
|
|
except (KeyboardInterrupt, SystemExit): #when you press ctrl+c
|
|
print("\nAborted. Exiting...")
|
|
sock.close()
|
|
gpsp.running = False
|
|
gpsp.join() # wait for the thread to finish
|
|
|
|
print("Done.\n")
|