|
|
|
|
@ -21,12 +21,8 @@
|
|
|
|
|
|
|
|
|
|
bool CAPRSParser::parseFrame(const std::string& frameStr, CAPRSFrame& frame)
|
|
|
|
|
{
|
|
|
|
|
return parseFrame(frameStr, frame, false);
|
|
|
|
|
}
|
|
|
|
|
bool CAPRSParser::parseFrame(const std::string& frameStr, CAPRSFrame& frame, bool doNotEnforceFrameType)
|
|
|
|
|
{
|
|
|
|
|
frame.clear();
|
|
|
|
|
bool ret = doNotEnforceFrameType;
|
|
|
|
|
bool ret = false;
|
|
|
|
|
|
|
|
|
|
if(!frameStr.empty()) {
|
|
|
|
|
auto pos = frameStr.find_first_of(':');
|
|
|
|
|
@ -48,14 +44,9 @@ bool CAPRSParser::parseFrame(const std::string& frameStr, CAPRSFrame& frame, boo
|
|
|
|
|
|
|
|
|
|
frame.getBody().assign(body);
|
|
|
|
|
|
|
|
|
|
if(!doNotEnforceFrameType) {
|
|
|
|
|
setFrameType(frame);
|
|
|
|
|
if(frame.getType() == APFT_UNKNOWN) {
|
|
|
|
|
CLog::logInfo("Invalid or unsupported APRS frame : %s", frameStr);
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
ret = true;
|
|
|
|
|
}
|
|
|
|
|
ret = parseInt(frame);
|
|
|
|
|
if(!ret) {
|
|
|
|
|
frame.clear();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
@ -64,37 +55,89 @@ bool CAPRSParser::parseFrame(const std::string& frameStr, CAPRSFrame& frame, boo
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CAPRSParser::setFrameType(CAPRSFrame& frame)
|
|
|
|
|
bool CAPRSParser::parseInt(CAPRSFrame& frame)
|
|
|
|
|
{
|
|
|
|
|
APRS_FRAME_TYPE type = APFT_UNKNOWN;
|
|
|
|
|
std::string body(frame.getBody());
|
|
|
|
|
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;
|
|
|
|
|
|
|
|
|
|
if(!body.empty()) {
|
|
|
|
|
switch (body[0])
|
|
|
|
|
{
|
|
|
|
|
case ':':
|
|
|
|
|
if(body[10] == ':' && std::all_of(body.begin() + 1, body.begin() + 10,
|
|
|
|
|
[](char c){ return c == ' ' || c == '-' || (c >= 'A' && c <= 'Z') || (c >= '0' && c <= '9'); }))
|
|
|
|
|
type = APFT_MESSAGE;
|
|
|
|
|
break;
|
|
|
|
|
// case '!':
|
|
|
|
|
// if(body.length() >= 2U && body[1] == '!') {
|
|
|
|
|
// // This is ultimeter 200 weather station
|
|
|
|
|
// type = APFT_UNKNOWN;
|
|
|
|
|
// break;
|
|
|
|
|
// } else {
|
|
|
|
|
// type = APFT_POSITION;
|
|
|
|
|
// }
|
|
|
|
|
// case '=':
|
|
|
|
|
// case '/':
|
|
|
|
|
// case '@':
|
|
|
|
|
// break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
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 ':':
|
|
|
|
|
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;
|
|
|
|
|
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;
|
|
|
|
|
default:
|
|
|
|
|
type = APFT_UNKNOWN;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
frame.getType() = type;
|
|
|
|
|
if(type == APFT_UNKNOWN)
|
|
|
|
|
frame.clear();
|
|
|
|
|
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] */
|
|
|
|
|
}
|