You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
F4KXL_DStarGateway/APRS/APRSParser.cpp

165 lines
6.0 KiB

/*
* Copyright (C) 2021-2022 by Geoffrey Merck F4FXL / KC3FRA
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*/
#include "APRSParser.h"
#include "Log.h"
bool CAPRSParser::parseFrame(const std::string& frameStr, CAPRSFrame& frame)
{
frame.clear();
bool ret = false;
if(!frameStr.empty()) {
auto pos = frameStr.find_first_of(':');
if(pos != std::string::npos && pos != frameStr.length() - 1) {
auto header = frameStr.substr(0, pos); // contains source, dest and path
auto body = frameStr.substr(pos +1);
std::vector<std::string> headerSplits;
boost::split(headerSplits, header, [](char c) { return c == ',' || c == '>';});
//we need at least source and dest to form a valid frame, also headers shall not contain empty strings
if(headerSplits.size() >= 2 && std::none_of(headerSplits.begin(), headerSplits.end(), [](std::string s){ return s.empty(); })) {
frame.getSource().assign(headerSplits[0]);
frame.getDestination().assign(headerSplits[1]);
for(unsigned int i = 2; i < headerSplits.size(); i++) {
frame.getPath().push_back(headerSplits[i]);
}
frame.getBody().assign(body);
ret = parseInt(frame);
if(!ret) {
frame.clear();
}
}
}
}
return ret;
}
bool CAPRSParser::parseInt(CAPRSFrame& frame)
{
APRS_FRAME_TYPE type = APFT_UNKNOWN;
unsigned char typeChar = frame.getBody()[0];
std::string body(frame.getBody().substr(1));//strip the type char for processing purposes
if(body.empty())
return false;
switch (typeChar)
{
case '!':
if(body[0] == '!') {
// This is ultimeter 200 weather station
return false;
}
[[fallthrough]];
case '=':
case '/':
case '@':
{
if(body.length() < 10) return false;//enough chars to have a chance to parse it ?
/* Normal or compressed location packet, with or without
* timestamp, with or without messaging capability
*
* ! and / have messaging, / and @ have a prepended timestamp
*/
type = APFT_POSITION;
if(typeChar == '/' || typeChar== '@')//With a prepended timestamp, jump over it.
body = body.substr(7U);
auto posChar = body[0];
if(valid_sym_table_compressed(posChar)//Compressed format
&& body.length() >= 13){//we need at least 13 char
//icom unsupported, ignore for now
return false;//parse_aprs_compressed(pb, body, body_end);
}
else if(posChar >= '0' && posChar <= '9' //Normal uncompressed format
&& body.length() >=19){//we need at least 19 chars for it to be valid
// if(ensureIsIcomCompatible(packet))
// return Parse(packet.Raw(), packet);
}
}
break;
case '$' :
if(body.length() > 10) {
type = APFT_NMEA;
}
break;
case ':':
// we have either message or telemetry labels or telemetry EQNS
if(body[9] == ':'
&& std::all_of(body.begin(), body.begin() + 9, [](char c){ return c == ' ' || c == '-' || (c >= 'A' && c <= 'Z') || (c >= '0' && c <= '9'); })) {
type = APFT_MESSAGE;
//If reciepient is same as source and we donot have a sequence number at the end of message, Then it is telemetry
if(body.find(frame.getSource()) == 0U) {
auto eqnsPos = body.find("EQNS.");
auto parmPos = body.find("PARM.");
auto seqNumPos = body.find_last_of('{');
if((eqnsPos == 10U || parmPos == 10U) && seqNumPos == std::string::npos) {
type = APFT_TELEMETRY;
}
}
}
break;
case '>':
type = APFT_STATUS;
break;
case '#': /* Peet Bros U-II Weather Station */
case '*': /* Peet Bros U-I Weather Station */
case '_': /* Weather report without position */
type = APFT_WX;
break;
case '{':
type = APFT_UNKNOWN; //
break;
case 'T':
if(body[0] == '#') {
type = APFT_TELEMETRY;
}
break;
case ';' :
if( body.length() >= 10 && (body[9] == '*' || body[9] == '_')) {
type = APFT_OBJECT;
}
break;
default:
type = APFT_UNKNOWN;
break;
}
frame.getType() = type;
return type != APFT_UNKNOWN;
}
bool CAPRSParser::valid_sym_table_compressed(unsigned char c)
{
return (c == '/' || c == '\\' || (c >= 0x41 && c <= 0x5A)
|| (c >= 0x61 && c <= 0x6A)); /* [\/\\A-Za-j] */
}
bool CAPRSParser::valid_sym_table_uncompressed(unsigned char c)
{
return (c == '/' || c == '\\' || (c >= 0x41 && c <= 0x5A)
|| (c >= 0x30 && c <= 0x39)); /* [\/\\A-Z0-9] */
}

Powered by TurnKey Linux.