2016-02-28 11:11:16 +00:00
|
|
|
import re
|
|
|
|
from datetime import datetime
|
|
|
|
|
2016-09-01 11:01:44 +00:00
|
|
|
from ogn.parser.utils import createTimestamp, parseAngle, kts2kmh, feet2m, fpm2ms
|
2016-10-03 12:42:50 +00:00
|
|
|
from ogn.parser.pattern import PATTERN_APRS_POSITION, PATTERN_APRS_STATUS, PATTERN_RECEIVER_BEACON, PATTERN_AIRCRAFT_BEACON
|
2017-09-28 17:10:27 +00:00
|
|
|
from ogn.parser.pattern import PATTERN_NAVITER_BEACON
|
2016-02-28 11:11:16 +00:00
|
|
|
from ogn.parser.exceptions import AprsParseError, OgnParseError
|
|
|
|
|
|
|
|
|
2016-09-23 21:49:56 +00:00
|
|
|
def parse_aprs(message, reference_date=None, reference_time=None):
|
2016-02-28 11:11:16 +00:00
|
|
|
if reference_date is None:
|
2016-09-23 21:49:56 +00:00
|
|
|
now = datetime.utcnow()
|
|
|
|
reference_date = now.date()
|
|
|
|
reference_time = now.time()
|
2016-02-28 11:11:16 +00:00
|
|
|
|
2016-10-03 12:42:50 +00:00
|
|
|
match_position = re.search(PATTERN_APRS_POSITION, message)
|
|
|
|
if match_position:
|
|
|
|
return {'name': match_position.group('callsign'),
|
|
|
|
'dstcall': match_position.group('dstcall'),
|
2017-09-27 18:23:43 +00:00
|
|
|
'receiver_name': match_position.group('receiver'),
|
2016-10-03 12:42:50 +00:00
|
|
|
'timestamp': createTimestamp(match_position.group('time'), reference_date, reference_time),
|
|
|
|
'latitude': parseAngle('0' + match_position.group('latitude') + (match_position.group('latitude_enhancement') or '0')) *
|
|
|
|
(-1 if match_position.group('latitude_sign') == 'S' else 1),
|
|
|
|
'symboltable': match_position.group('symbol_table'),
|
|
|
|
'longitude': parseAngle(match_position.group('longitude') + (match_position.group('longitude_enhancement') or '0')) *
|
|
|
|
(-1 if match_position.group('longitude_sign') == 'W' else 1),
|
|
|
|
'symbolcode': match_position.group('symbol'),
|
2017-05-07 09:47:58 +00:00
|
|
|
'track': int(match_position.group('course')) if match_position.group('course_extension') else None,
|
|
|
|
'ground_speed': int(match_position.group('ground_speed')) * kts2kmh if match_position.group('ground_speed') else None,
|
2016-10-03 12:42:50 +00:00
|
|
|
'altitude': int(match_position.group('altitude')) * feet2m,
|
|
|
|
'comment': match_position.group('comment')}
|
|
|
|
|
|
|
|
match_status = re.search(PATTERN_APRS_STATUS, message)
|
|
|
|
if match_status:
|
|
|
|
return {'name': match_status.group('callsign'),
|
|
|
|
'dstcall': match_status.group('dstcall'),
|
2017-09-27 18:23:43 +00:00
|
|
|
'receiver_name': match_status.group('receiver'),
|
2016-10-03 12:42:50 +00:00
|
|
|
'timestamp': createTimestamp(match_status.group('time'), reference_date, reference_time),
|
|
|
|
'comment': match_status.group('comment')}
|
2016-02-28 11:11:16 +00:00
|
|
|
|
|
|
|
raise AprsParseError(message)
|
|
|
|
|
|
|
|
|
2017-09-28 06:43:50 +00:00
|
|
|
def parse_ogn_aircraft_beacon(aprs_comment):
|
|
|
|
ac_match = re.search(PATTERN_AIRCRAFT_BEACON, aprs_comment)
|
|
|
|
if ac_match:
|
|
|
|
return {'address_type': int(ac_match.group('details'), 16) & 0b00000011,
|
|
|
|
'aircraft_type': (int(ac_match.group('details'), 16) & 0b01111100) >> 2,
|
|
|
|
'stealth': (int(ac_match.group('details'), 16) & 0b10000000) >> 7 == 1,
|
|
|
|
'address': ac_match.group('id'),
|
|
|
|
'climb_rate': int(ac_match.group('climb_rate')) * fpm2ms if ac_match.group('climb_rate') else None,
|
|
|
|
'turn_rate': float(ac_match.group('turn_rate')) if ac_match.group('turn_rate') else None,
|
|
|
|
'flightlevel': float(ac_match.group('flight_level')) if ac_match.group('flight_level') else None,
|
|
|
|
'signal_quality': float(ac_match.group('signal_quality')) if ac_match.group('signal_quality') else None,
|
|
|
|
'error_count': int(ac_match.group('errors')) if ac_match.group('errors') else None,
|
|
|
|
'frequency_offset': float(ac_match.group('frequency_offset')) if ac_match.group('frequency_offset') else None,
|
|
|
|
'gps_status': ac_match.group('gps_accuracy') if ac_match.group('gps_accuracy') else None,
|
|
|
|
'software_version': float(ac_match.group('flarm_software_version')) if ac_match.group('flarm_software_version') else None,
|
|
|
|
'hardware_version': int(ac_match.group('flarm_hardware_version'), 16) if ac_match.group('flarm_hardware_version') else None,
|
|
|
|
'real_address': ac_match.group('flarm_id') if ac_match.group('flarm_id') else None,
|
|
|
|
'signal_power': float(ac_match.group('signal_power')) if ac_match.group('signal_power') else None,
|
|
|
|
'proximity': [hear[4:] for hear in ac_match.group('proximity').split(" ")] if ac_match.group('proximity') else None}
|
2016-02-28 11:11:16 +00:00
|
|
|
else:
|
2017-09-28 06:43:50 +00:00
|
|
|
return None
|
2016-02-28 11:11:16 +00:00
|
|
|
|
|
|
|
|
|
|
|
def parse_ogn_receiver_beacon(aprs_comment):
|
|
|
|
rec_match = re.search(PATTERN_RECEIVER_BEACON, aprs_comment)
|
|
|
|
if rec_match:
|
|
|
|
return {'version': rec_match.group('version'),
|
|
|
|
'platform': rec_match.group('platform'),
|
|
|
|
'cpu_load': float(rec_match.group('cpu_load')),
|
|
|
|
'free_ram': float(rec_match.group('ram_free')),
|
|
|
|
'total_ram': float(rec_match.group('ram_total')),
|
|
|
|
'ntp_error': float(rec_match.group('ntp_offset')),
|
|
|
|
'rt_crystal_correction': float(rec_match.group('ntp_correction')),
|
2016-10-10 17:26:46 +00:00
|
|
|
'voltage': float(rec_match.group('voltage')) if rec_match.group('voltage') else None,
|
|
|
|
'amperage': float(rec_match.group('amperage')) if rec_match.group('amperage') else None,
|
2016-02-28 11:11:16 +00:00
|
|
|
'cpu_temp': float(rec_match.group('cpu_temperature')) if rec_match.group('cpu_temperature') else None,
|
2016-10-09 20:01:06 +00:00
|
|
|
'senders_visible': int(rec_match.group('visible_senders')) if rec_match.group('visible_senders') else None,
|
|
|
|
'senders_total': int(rec_match.group('senders')) if rec_match.group('senders') else None,
|
2017-05-07 09:47:58 +00:00
|
|
|
'rec_crystal_correction': int(rec_match.group('rf_correction_manual')) if rec_match.group('rf_correction_manual') else None,
|
|
|
|
'rec_crystal_correction_fine': float(rec_match.group('rf_correction_automatic')) if rec_match.group('rf_correction_automatic') else None,
|
2017-05-07 10:14:32 +00:00
|
|
|
'rec_input_noise': float(rec_match.group('signal_quality')) if rec_match.group('signal_quality') else None,
|
|
|
|
'senders_signal': float(rec_match.group('senders_signal_quality')) if rec_match.group('senders_signal_quality') else None,
|
2016-10-09 20:01:06 +00:00
|
|
|
'senders_messages': float(rec_match.group('senders_messages')) if rec_match.group('senders_messages') else None,
|
2017-05-07 10:14:32 +00:00
|
|
|
'good_senders_signal': float(rec_match.group('good_senders_signal_quality')) if rec_match.group('good_senders_signal_quality') else None,
|
2016-10-09 20:01:06 +00:00
|
|
|
'good_senders': float(rec_match.group('good_senders')) if rec_match.group('good_senders') else None,
|
|
|
|
'good_and_bad_senders': float(rec_match.group('good_and_bad_senders')) if rec_match.group('good_and_bad_senders') else None}
|
2016-02-28 11:11:16 +00:00
|
|
|
else:
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
2017-09-28 06:43:50 +00:00
|
|
|
def parse_lt24_beacon(aprs_comment):
|
|
|
|
raise NotImplementedError("LT24 parser not implemented")
|
|
|
|
|
|
|
|
|
|
|
|
def parse_naviter_beacon(aprs_comment):
|
2017-09-28 17:10:27 +00:00
|
|
|
ac_match = re.search(PATTERN_NAVITER_BEACON, aprs_comment)
|
2017-09-28 17:16:34 +00:00
|
|
|
return {'stealth': (int(ac_match.group('details'), 16) & 0b1000000000000000) >> 15 == 1,
|
|
|
|
'do_not_track': (int(ac_match.group('details'), 16) & 0b0100000000000000) >> 14 == 1,
|
|
|
|
'aircraft_type': (int(ac_match.group('details'), 16) & 0b0011110000000000) >> 10,
|
|
|
|
'address_type': (int(ac_match.group('details'), 16) & 0b0000001111110000) >> 4,
|
|
|
|
'reserved': (int(ac_match.group('details'), 16) & 0b0000000000001111),
|
|
|
|
'address': ac_match.group('id'),
|
|
|
|
'climb_rate': int(ac_match.group('climb_rate')) * fpm2ms if ac_match.group('climb_rate') else None,
|
|
|
|
'turn_rate': float(ac_match.group('turn_rate')) if ac_match.group('turn_rate') else None}
|
2017-09-28 06:43:50 +00:00
|
|
|
|
|
|
|
|
|
|
|
def parse_skylines_beacon(aprs_comment):
|
|
|
|
raise NotImplementedError("Skylines parser not implemented")
|
|
|
|
|
|
|
|
|
|
|
|
def parse_spider_beacon(aprs_comment):
|
|
|
|
raise NotImplementedError("Spider parser not implemented")
|
|
|
|
|
|
|
|
|
|
|
|
def parse_spot_beacon(aprs_comment):
|
|
|
|
raise NotImplementedError("SPOT parser not implemented")
|
|
|
|
|
|
|
|
|
2017-09-28 17:16:34 +00:00
|
|
|
def parse_capture_beacon(aprs_comment):
|
|
|
|
raise NotImplementedError("Capture beacon parser not yet implemented")
|
|
|
|
|
|
|
|
|
2017-09-27 21:04:47 +00:00
|
|
|
def parse_ogn_beacon(aprs_comment, dstcall="APRS"):
|
2017-09-28 06:43:50 +00:00
|
|
|
if dstcall == "APRS":
|
|
|
|
if not aprs_comment:
|
|
|
|
return {'beacon_type': 'receiver_beacon'}
|
2017-05-16 12:36:38 +00:00
|
|
|
|
2017-09-28 06:43:50 +00:00
|
|
|
ac_data = parse_ogn_aircraft_beacon(aprs_comment)
|
|
|
|
if ac_data:
|
|
|
|
ac_data.update({'beacon_type': 'aircraft_beacon'})
|
|
|
|
return ac_data
|
2016-02-28 11:11:16 +00:00
|
|
|
|
2017-09-28 06:43:50 +00:00
|
|
|
rc_data = parse_ogn_receiver_beacon(aprs_comment)
|
|
|
|
if rc_data:
|
|
|
|
rc_data.update({'beacon_type': 'receiver_beacon'})
|
|
|
|
return rc_data
|
2016-02-28 11:11:16 +00:00
|
|
|
|
2017-09-28 06:43:50 +00:00
|
|
|
raise OgnParseError(aprs_comment)
|
|
|
|
elif dstcall == "OGLT24":
|
|
|
|
ac_data = parse_lt24_beacon(aprs_comment)
|
|
|
|
ac_data.update({'beacon_type': 'lt24_beacon'})
|
|
|
|
return ac_data
|
|
|
|
elif dstcall == "OGNAVI":
|
|
|
|
ac_data = parse_naviter_beacon(aprs_comment)
|
|
|
|
ac_data.update({'beacon_type': 'naviter_beacon'})
|
|
|
|
return ac_data
|
|
|
|
elif dstcall == "OGSKYL":
|
|
|
|
ac_data = parse_skylines_beacon(aprs_comment)
|
|
|
|
ac_data.update({'beacon_type': 'skylines_beacon'})
|
|
|
|
return ac_data
|
|
|
|
elif dstcall == "OGSPID":
|
|
|
|
ac_data = parse_spider_beacon(aprs_comment)
|
|
|
|
ac_data.update({'beacon_type': 'spider_beacon'})
|
|
|
|
return ac_data
|
|
|
|
elif dstcall == "OGSPOT":
|
|
|
|
ac_data = parse_spot_beacon(aprs_comment)
|
|
|
|
ac_data.update({'beacon_type': 'spot_beacon'})
|
|
|
|
return ac_data
|
|
|
|
else:
|
|
|
|
raise ValueError("dstcall {} unknown".format(dstcall))
|