Skip to content

Commit 7fcee20

Browse files
committed
New station Capabilities and third party Header types for packet filtering.
1 parent 52927ce commit 7fcee20

File tree

4 files changed

+238
-52
lines changed

4 files changed

+238
-52
lines changed
6.74 KB
Binary file not shown.

doc/User-Guide.pdf

5.22 KB
Binary file not shown.

pfilter.c

+223-49
Original file line numberDiff line numberDiff line change
@@ -874,11 +874,15 @@ static int filt_t (pfstate_t *pf)
874874

875875
case 'p': /* Position */
876876
if (*infop == '!') return (1);
877-
if (*infop == '\'') return (1);
878877
if (*infop == '/') return (1);
879878
if (*infop == '=') return (1);
880879
if (*infop == '@') return (1);
881-
if (*infop == '`') return (1);
880+
if (*infop == '\'') return (1); // MIC-E
881+
if (*infop == '`') return (1); // MIC-E
882+
883+
// What if we have "_" symbol code for weather?
884+
// Still consider as position.
885+
// The same packet can match more than one type here.
882886
break;
883887

884888
case 'o': /* Object */
@@ -897,6 +901,11 @@ static int filt_t (pfstate_t *pf)
897901
if (*infop == '?') return (1);
898902
break;
899903

904+
case 'c': /* station Capabilities - my extension */
905+
/* Most often used for IGate statistics. */
906+
if (*infop == '<') return (1);
907+
break;
908+
900909
case 's': /* Status */
901910
if (*infop == '>') return (1);
902911
break;
@@ -910,16 +919,32 @@ static int filt_t (pfstate_t *pf)
910919
if (*infop == '{') return (1);
911920
break;
912921

922+
case 'h': /* third party Header - my extension */
923+
if (*infop == '}') return (1);
924+
break;
925+
913926
case 'w': /* Weather */
914-
if (*infop == '@') return (1);
915-
if (*infop == '*') return (1);
916-
if (*infop == '_') return (1);
927+
928+
if (*infop == '*') return (1); // Peet Bros
929+
if (*infop == '_') return (1); // Weather report, no position.
930+
if (strncmp(infop, "!!", 2) == 0) return(1); // Ultimeter 2000.
917931

918932
/* '$' is normally raw GPS. Check for special case. */
933+
919934
if (strncmp(infop, "$ULTW", 5) == 0) return (1);
920935

921-
/* TODO: Positions !=/@ can be weather. */
922-
/* Need to check for _ symbol. */
936+
/* Positions !=/@ with symbol code _ are weather. */
937+
938+
if (strchr("!=/@", *infop) != NULL &&
939+
pf->decoded.g_symbol_code == '_') return (1);
940+
941+
/* Object with _ symbol is also weather. APRS protocol spec page 66. */
942+
943+
if (*infop == ';' &&
944+
pf->decoded.g_symbol_code == '_') return (1);
945+
946+
// TODO: need more test cases at end for new weather cases.
947+
923948
break;
924949

925950
case 'n': /* NWS format */
@@ -1059,76 +1084,194 @@ static int filt_r (pfstate_t *pf, char *sdist)
10591084
*
10601085
* Description:
10611086
*
1087+
* s/pri
1088+
* s/pri/alt
1089+
* s/pri/alt/
1090+
* s/pri/alt/over
1091+
*
10621092
* "pri" is zero or more symbols from the primary symbol set.
1093+
* Symbol codes are any printable ASCII character other than | or ~.
1094+
* (Zero symbols here would be sensible only if later alt part is specified.)
10631095
* "alt" is one or more symbols from the alternate symbol set.
1064-
* "over" is overlay characters. Overlays apply only to the alternate symbol set.
1096+
* "over" is overlay characters for the alternate symbol set.
1097+
* Only upper case letters, digits, and \ are allowed here.
1098+
* If the last part is not specified, any overlay or lack of overlay, is ignored.
1099+
* If the last part is specified, only the listed overlays will match.
1100+
* An explicit lack of overlay is represented by the \ character.
10651101
*
10661102
* Examples:
1067-
* s/-> Allow house and car from primary symbol table.
1068-
* s//# Allow alternate table digipeater, with or without overlay.
1069-
* s//#/\ Allow alternate table digipeater, only if no overlay.
1070-
* s//#/SL1 Allow alternate table digipeater, with overlay S, L, or 1
1103+
* s/O Balloon.
1104+
* s/-> House or car from primary symbol table.
1105+
*
1106+
* s//# Alternate table digipeater, with or without overlay.
1107+
* s//#/\ Alternate table digipeater, only if no overlay.
1108+
* s//#/SL1 Alternate table digipeater, with overlay S, L, or 1.
1109+
* s//#/SL\ Alternate table digipeater, with S, L, or no overlay.
1110+
*
1111+
* s/s/s Any variation of watercraft. Either symbol table. With or without overlay.
1112+
* s/s/s/ Ship or ship sideview, only if no overlay.
1113+
* s//s/J Jet Ski.
1114+
*
1115+
* What if you want to use the / symbol when / is being used as a delimiter here? Recall that you
1116+
* can use some other special character after the initial lower case letter and this becomes the
1117+
* delimiter for the rest of the specification.
1118+
*
1119+
* Examples:
1120+
*
1121+
* s:/ Red Dot.
1122+
* s::/ Waypoint Destination, with or without overlay.
1123+
* s:/:/ Either Red Dot or Waypoint Destination.
1124+
* s:/:/: Either Red Dot or Waypoint Destination, no overlay.
1125+
*
1126+
* Bad example:
1127+
*
1128+
* Someone tried using this to include ballons: s/'/O/-/#/_
1129+
* probably following the buddy filter pattern of / between each alternative.
1130+
* There should be an error message because it has more than 3 delimiter characters.
1131+
*
10711132
*
10721133
*------------------------------------------------------------------------------*/
10731134

10741135
static int filt_s (pfstate_t *pf)
10751136
{
10761137
char str[MAX_TOKEN_LEN];
10771138
char *cp;
1078-
char sep[2];
1079-
char *pri, *alt, *over;
1139+
char sep[2]; // Delimiter character. Typically / but it could be different.
1140+
char *pri = NULL, *alt = NULL, *over = NULL, *extra = NULL;
1141+
char *x;
10801142

10811143

10821144
strlcpy (str, pf->token_str, sizeof(str));
10831145
sep[0] = str[1];
10841146
sep[1] = '\0';
10851147
cp = str + 2;
10861148

1087-
// TODO: check here.
1149+
1150+
// First, separate the parts and do a strict syntax check.
10881151

10891152
pri = strsep (&cp, sep);
10901153

1091-
if (pri == NULL) {
1154+
if (pri != NULL) {
1155+
1156+
// Zero length is acceptable if alternate symbol(s) specified. Will check that later.
1157+
1158+
for (x = pri; *x != '\0'; x++) {
1159+
if ( ! isprint(*x) || *x == '|' || *x == '~') {
1160+
print_error (pf, "Symbol filter, primary must be printable ASCII character(s) other than | or ~.");
1161+
return (-1);
1162+
}
1163+
}
1164+
1165+
alt = strsep (&cp, sep);
1166+
1167+
if (alt != NULL) {
1168+
1169+
// Zero length after second / would be pointless.
1170+
1171+
if (strlen(alt) == 0) {
1172+
print_error (pf, "Nothing specified for alternate symbol table.");
1173+
return (-1);
1174+
}
1175+
1176+
for (x = alt; *x != '\0'; x++) {
1177+
if ( ! isprint(*x) || *x == '|' || *x == '~') {
1178+
print_error (pf, "Symbol filter, alternate must be printable ASCII character(s) other than | or ~.");
1179+
return (-1);
1180+
}
1181+
}
1182+
1183+
over = strsep (&cp, sep);
1184+
1185+
if (over != NULL) {
1186+
1187+
// Zero length is acceptable and is not the same as missing.
1188+
1189+
for (x = over; *x != '\0'; x++) {
1190+
if ( (! isupper(*x)) && (! isdigit(*x)) && *x != '\\') {
1191+
print_error (pf, "Symbol filter, overlay must be upper case letter, digit, or \\.");
1192+
return (-1);
1193+
}
1194+
}
1195+
1196+
extra = strsep (&cp, sep);
1197+
1198+
if (extra != NULL) {
1199+
print_error (pf, "More than 3 delimiter characters in Symbol filter.");
1200+
return (-1);
1201+
}
1202+
}
1203+
}
1204+
else {
1205+
// No alt part is OK if at least one primary symbol was specified.
1206+
if (strlen(pri) == 0) {
1207+
print_error (pf, "No symbols specified for Symbol filter.");
1208+
return (-1);
1209+
}
1210+
}
1211+
}
1212+
else {
10921213
print_error (pf, "Missing arguments for Symbol filter.");
10931214
return (-1);
10941215
}
10951216

1096-
if (pf->decoded.g_symbol_table == '/' && strchr(pri, pf->decoded.g_symbol_code) != NULL) {
1097-
/* Found in primary symbols. All done. */
1098-
return (1);
1217+
1218+
// This applies only for Position, Object, Item.
1219+
// decode_aprs() should set symbol code to space to mean undefined.
1220+
1221+
if (pf->decoded.g_symbol_code == ' ') {
1222+
return (0);
1223+
}
1224+
1225+
1226+
// Look for Primary symbols.
1227+
1228+
if (pf->decoded.g_symbol_table == '/') {
1229+
if (pri != NULL && strlen(pri) > 0) {
1230+
return (strchr(pri, pf->decoded.g_symbol_code) != NULL);
1231+
}
10991232
}
11001233

1101-
alt = strsep (&cp, sep);
11021234
if (alt == NULL) {
11031235
return (0);
11041236
}
1105-
if (strlen(alt) == 0) {
1106-
/* We have s/.../ */
1107-
print_error (pf, "Missing alternate symbols for Symbol filter.");
1108-
return (-1);
1109-
}
11101237

11111238
//printf ("alt=\"%s\" sym='%c'\n", alt, pf->decoded.g_symbol_code);
11121239

1113-
if (strchr(alt, pf->decoded.g_symbol_code) == NULL) {
1114-
/* Not found in alternate symbols. Reject. */
1115-
return (0);
1116-
}
1240+
// Look for Alternate symbols.
11171241

1118-
over = strsep (&cp, sep);
1119-
if (over == NULL) {
1120-
/* alternate, with or without overlay. */
1121-
return (pf->decoded.g_symbol_table != '/');
1122-
}
1242+
if (strchr(alt, pf->decoded.g_symbol_code) != NULL) {
11231243

1124-
// printf ("over=\"%s\" table='%c'\n", over, pf->decoded.g_symbol_table);
1244+
// We have a match but that might not be enough.
1245+
// We must see if there was an overlay part specified.
11251246

1126-
if (strlen(over) == 0) {
1127-
return (pf->decoded.g_symbol_table == '\\');
1247+
if (over != NULL) {
1248+
1249+
if (strlen(over) > 0) {
1250+
1251+
// Non-zero length overlay part was specified.
1252+
// Need to match one of them.
1253+
1254+
return (strchr(over, pf->decoded.g_symbol_table) != NULL);
1255+
}
1256+
else {
1257+
1258+
// Zero length overlay part was specified.
1259+
// We must have no overlay, i.e. table is \.
1260+
1261+
return (pf->decoded.g_symbol_table == '\\');
1262+
}
1263+
}
1264+
else {
1265+
1266+
// No check of overlay part. Just make sure it is not primary table.
1267+
1268+
return (pf->decoded.g_symbol_table != '/');
1269+
}
11281270
}
11291271

1130-
return (strchr(over, pf->decoded.g_symbol_table) != NULL);
1131-
}
1272+
return (0);
1273+
1274+
} /* end filt_s */
11321275

11331276

11341277
/*------------------------------------------------------------------------------
@@ -1494,10 +1637,15 @@ int main ()
14941637
pftest (126, "t/n", "CWAPID>APRS:;CWAttttz *DDHHMMzLATLONICONADVISETYPE{seq#", 1);
14951638
pftest (127, "t/", "CWAPID>APRS:;CWAttttz *DDHHMMzLATLONICONADVISETYPE{seq#", 0);
14961639

1497-
pftest (130, "r/42.6/-71.3/10", "WB2OSZ-5>APDW12,WIDE1-1,WIDE2-1:!4237.14NS07120.83W#PHG7140Chelmsford MA", 1);
1498-
pftest (131, "r/42.6/-71.3/10", "WA1PLE-5>APWW10,W1MHL,N8VIM,WIDE2*:@022301h4208.75N/07115.16WoAPRS-IS for Win32", 0);
1640+
pftest (128, "t/c", "S0RCE>DEST:<stationcapabilities", 1);
1641+
pftest (129, "t/h", "S0RCE>DEST:<stationcapabilities", 0);
1642+
pftest (130, "t/h", "S0RCE>DEST:}thirdpartyheaderwhatever", 1);
1643+
pftest (131, "t/c", "S0RCE>DEST:}thirdpartyheaderwhatever", 0);
1644+
1645+
pftest (140, "r/42.6/-71.3/10", "WB2OSZ-5>APDW12,WIDE1-1,WIDE2-1:!4237.14NS07120.83W#PHG7140Chelmsford MA", 1);
1646+
pftest (141, "r/42.6/-71.3/10", "WA1PLE-5>APWW10,W1MHL,N8VIM,WIDE2*:@022301h4208.75N/07115.16WoAPRS-IS for Win32", 0);
14991647

1500-
pftest (140, "( t/t & b/WB2OSZ ) | ( t/o & ! r/42.6/-71.3/1 )", "WB2OSZ>APDW12:;home *111111z4237.14N/07120.83W-Chelmsford MA", 1);
1648+
pftest (145, "( t/t & b/WB2OSZ ) | ( t/o & ! r/42.6/-71.3/1 )", "WB2OSZ>APDW12:;home *111111z4237.14N/07120.83W-Chelmsford MA", 1);
15011649

15021650
pftest (150, "s/->", "WB2OSZ-5>APDW12:!4237.14NS07120.83W#PHG7140Chelmsford MA", 0);
15031651
pftest (151, "s/->", "WB2OSZ-5>APDW12:!4237.14N/07120.83W-PHG7140Chelmsford MA", 1);
@@ -1515,13 +1663,28 @@ int main ()
15151663
pftest (160, "s//#/LS1", "WB2OSZ-5>APDW12:!4237.14NS07120.83W#PHG7140Chelmsford MA", 1);
15161664
pftest (161, "s//#/LS1", "WB2OSZ-5>APDW12:!4237.14N\\07120.83W#PHG7140Chelmsford MA", 0);
15171665
pftest (162, "s//#/LS1", "WB2OSZ-5>APDW12:!4237.14N/07120.83W#PHG7140Chelmsford MA", 0);
1666+
pftest (163, "s//#/LS\\", "WB2OSZ-5>APDW12:!4237.14N\\07120.83W#PHG7140Chelmsford MA", 1);
15181667

1519-
pftest (170, "v/DIGI2/DIGI3", "WB2OSZ-5>APDW12,DIGI1,DIGI2,DIGI3,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA", 1);
1520-
pftest (171, "v/DIGI2/DIGI3", "WB2OSZ-5>APDW12,DIGI1*,DIGI2,DIGI3,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA", 1);
1521-
pftest (172, "v/DIGI2/DIGI3", "WB2OSZ-5>APDW12,DIGI1,DIGI2*,DIGI3,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA", 1);
1522-
pftest (173, "v/DIGI2/DIGI3", "WB2OSZ-5>APDW12,DIGI1,DIGI2,DIGI3*,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA", 0);
1523-
pftest (174, "v/DIGI2/DIGI3", "WB2OSZ-5>APDW12,DIGI1,DIGI2,DIGI3,DIGI4*:!4237.14NS07120.83W#PHG7140Chelmsford MA", 0);
1524-
pftest (175, "v/DIGI9/DIGI2", "WB2OSZ-5>APDW12,DIGI1,DIGI2*,DIGI3,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA", 0);
1668+
pftest (170, "s:/", "WB2OSZ-5>APDW12:!4237.14N/07120.83W/PHG7140Chelmsford MA", 1);
1669+
pftest (171, "s:/", "WB2OSZ-5>APDW12:!4237.14N\\07120.83W/PHG7140Chelmsford MA", 0);
1670+
pftest (172, "s::/", "WB2OSZ-5>APDW12:!4237.14N/07120.83W/PHG7140Chelmsford MA", 0);
1671+
pftest (173, "s::/", "WB2OSZ-5>APDW12:!4237.14N\\07120.83W/PHG7140Chelmsford MA", 1);
1672+
pftest (174, "s:/:/", "WB2OSZ-5>APDW12:!4237.14N/07120.83W/PHG7140Chelmsford MA", 1);
1673+
pftest (175, "s:/:/", "WB2OSZ-5>APDW12:!4237.14N\\07120.83W/PHG7140Chelmsford MA", 1);
1674+
pftest (176, "s:/:/", "WB2OSZ-5>APDW12:!4237.14NX07120.83W/PHG7140Chelmsford MA", 1);
1675+
pftest (177, "s:/:/:X", "WB2OSZ-5>APDW12:!4237.14NX07120.83W/PHG7140Chelmsford MA", 1);
1676+
1677+
// FIXME: Different on Windows and 64 bit Linux.
1678+
//pftest (178, "s:/:/:", "WB2OSZ-5>APDW12:!4237.14NX07120.83W/PHG7140Chelmsford MA", 1);
1679+
1680+
pftest (179, "s:/:/:\\", "WB2OSZ-5>APDW12:!4237.14NX07120.83W/PHG7140Chelmsford MA", 0);
1681+
1682+
pftest (180, "v/DIGI2/DIGI3", "WB2OSZ-5>APDW12,DIGI1,DIGI2,DIGI3,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA", 1);
1683+
pftest (181, "v/DIGI2/DIGI3", "WB2OSZ-5>APDW12,DIGI1*,DIGI2,DIGI3,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA", 1);
1684+
pftest (182, "v/DIGI2/DIGI3", "WB2OSZ-5>APDW12,DIGI1,DIGI2*,DIGI3,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA", 1);
1685+
pftest (183, "v/DIGI2/DIGI3", "WB2OSZ-5>APDW12,DIGI1,DIGI2,DIGI3*,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA", 0);
1686+
pftest (184, "v/DIGI2/DIGI3", "WB2OSZ-5>APDW12,DIGI1,DIGI2,DIGI3,DIGI4*:!4237.14NS07120.83W#PHG7140Chelmsford MA", 0);
1687+
pftest (185, "v/DIGI9/DIGI2", "WB2OSZ-5>APDW12,DIGI1,DIGI2*,DIGI3,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA", 0);
15251688

15261689
/* Test error reporting. */
15271690

@@ -1540,12 +1703,23 @@ int main ()
15401703
pftest (226, "i/30/8/", "WB2OSZ-5>APDW14::W2UB :Happy Birthday{001", 1);
15411704
pftest (227, "i/30/8", "WB2OSZ-5>APDW14::W2UB :Happy Birthday{001", 1);
15421705

1543-
// FIXME: behaves differently on Windows and Linux
1706+
// FIXME: behaves differently on Windows and Linux. Why?
1707+
// On Windows we have our own version of strsep because it's not in the MS library.
1708+
// It must behave differently than the Linux version when nothing follows the last separator.
15441709
//pftest (228, "i/30/", "WB2OSZ-5>APDW14::W2UB :Happy Birthday{001", 1);
15451710

15461711
pftest (229, "i/30", "WB2OSZ-5>APDW14::W2UB :Happy Birthday{001", 1);
15471712
pftest (230, "i/", "WB2OSZ-5>APDW14::W2UB :Happy Birthday{001", -1);
15481713

1714+
pftest (240, "s/", "WB2OSZ-5>APDW12:!4237.14N/07120.83WOPHG7140Chelmsford MA", -1);
1715+
pftest (241, "s/'/O/-/#/_", "WB2OSZ-5>APDW12:!4237.14N/07120.83WOPHG7140Chelmsford MA", -1);
1716+
pftest (242, "s/O/O/c", "WB2OSZ-5>APDW12:!4237.14N/07120.83WOPHG7140Chelmsford MA", -1);
1717+
pftest (243, "s/O/O/1/2", "WB2OSZ-5>APDW12:!4237.14N/07120.83WOPHG7140Chelmsford MA", -1);
1718+
pftest (244, "s/O/|/1", "WB2OSZ-5>APDW12:!4237.14N/07120.83WOPHG7140Chelmsford MA", -1);
1719+
pftest (245, "s//", "WB2OSZ-5>APDW12:!4237.14N/07120.83WOPHG7140Chelmsford MA", -1);
1720+
pftest (246, "s///", "WB2OSZ-5>APDW12:!4237.14N/07120.83WOPHG7140Chelmsford MA", -1);
1721+
1722+
15491723
// TODO: to be continued...
15501724

15511725
if (error_count > 0) {

0 commit comments

Comments
 (0)