1
0
mirror of https://gitea.farpn.net/w1cdn/mwtchahrd.git synced 2025-12-10 06:28:02 -06:00

11 Commits

Author SHA1 Message Date
mattbk
ca858bc038 Bump version to keep track. 2025-11-30 18:04:47 -06:00
w1cdn
fd641c1ca1 Merge pull request 'Avoid using 3rd party locations' (#34) from loc_fix into dev
Reviewed-on: https://gitea.farpn.net/w1cdn/mwtchahrd/pulls/34
2025-11-30 23:50:43 +00:00
mattbk
05a6bb18b0 Update .gitignore. 2025-11-30 11:56:28 -06:00
W1CDN
85521543c6 Set up cross build scripts. 2025-11-30 17:52:14 +00:00
mattbk
6bbfa66c4f Check harder for 3rd party locations and drop them. 2025-11-18 21:28:24 -06:00
w1cdn
d95cf3e368 Merge pull request 'Filter out callsign-SSID combinations' (#32) from filter into dev
Reviewed-on: https://gitea.farpn.net/w1cdn/mwtchahrd/pulls/32
2025-11-19 01:01:32 +00:00
mattbk
20fda492a5 Clean up. 2025-11-18 19:01:12 -06:00
mattbk
97e955156d Update README. 2025-11-15 21:21:41 -06:00
mattbk
c60723b0a5 Update README. 2025-11-15 21:18:48 -06:00
mattbk
780c4b5884 Add simple filter for text strings in source call-SSID. 2025-11-15 21:07:23 -06:00
mattbk
07c57e85a8 Snapshot. 2025-11-13 21:23:32 -06:00
8 changed files with 1961 additions and 43 deletions

2
.gitignore vendored
View File

@@ -6,7 +6,7 @@ target/
# Remove Cargo.lock from gitignore if creating an executable, leave it for libraries # Remove Cargo.lock from gitignore if creating an executable, leave it for libraries
# More information here https://doc.rust-lang.org/cargo/guide/cargo-toml-vs-cargo-lock.html # More information here https://doc.rust-lang.org/cargo/guide/cargo-toml-vs-cargo-lock.html
Cargo.lock #Cargo.lock
#Cross config #Cross config
Cross.toml Cross.toml

1882
Cargo.lock generated Normal file

File diff suppressed because it is too large Load Diff

View File

@@ -1,13 +1,15 @@
[package] [package]
name = "mwtchahrd" name = "mwtchahrd"
version = "0.1.0" version = "0.1.1"
edition = "2021" edition = "2021"
authors = ["Chris, N6CTA <mail@n6cta.com>"] authors = ["Chris, N6CTA <mail@n6cta.com>", "Matt, W1CDN <admin@w1cdn.net>"]
[dependencies] [dependencies]
anyhow = "1.0" anyhow = "1.0"
chrono = "0.4" chrono = "0.4"
clap = { version = "4", features = ["derive"] } clap = { version = "4", features = ["derive"] }
openssl-sys = "0.9.111"
openssl = { version = "0.10", features = ["vendored"] }
rand = "0.9.2" rand = "0.9.2"
regex = "1.12.2" regex = "1.12.2"
reqwest = { version = "0.12.24", features = ["json", "blocking"] } reqwest = { version = "0.12.24", features = ["json", "blocking"] }
@@ -16,3 +18,4 @@ socket2 = "0.5"
[profile.release] [profile.release]
lto = "thin" lto = "thin"

View File

@@ -1,11 +0,0 @@
[target.aarch64-unknown-linux-gnu]
image = "ghcr.io/cross-rs/aarch64-unknown-linux-gnu:edge@sha256:168fa652629cedcd9549e85258be8f83fa008625b187c004d6ea439cf16f6a41"
[target.x86_64-unknown-linux-gnu]
image = "ghcr.io/cross-rs/x86_64-unknown-linux-gnu:edge@sha256:3542b427c3f7d0d2976d7add025c5ba997499411f408b3b26803ccc70fc431da"
[target.armv7-unknown-linux-gnueabihf]
image = "ghcr.io/cross-rs/armv7-unknown-linux-gnueabihf:edge@sha256:3e1def581eb9c9f11cfff85745802f2de5cf9cdeeb5a8495048f393a0993b99b"
[target.arm-unknown-linux-gnueabihf]
image = "ghcr.io/cross-rs/arm-unknown-linux-gnueabihf:edge@sha256:28e7aaae8301506f4f1b3394c8bc23f958d1717043e06f816045e7b8df57e173"

View File

@@ -74,6 +74,15 @@ mwtchahrd -i <IP> -p <PORT> [-c <CHANNEL>] [-d] [-u]
- `-f, --freq <FREQ>` - `-f, --freq <FREQ>`
Spotting frequency [default: 14105000]. Spotting frequency [default: 14105000].
- `-y, --my-lat <MY_LAT>`
Spotter latitude, DD.dddd.
- `-x, --my-lon <MY_LON>`
Spotter longitude, DD.dddd.
- `-t, --filter <CALL-SSID,CALL-SSID,...>`
Filter out spots from these callsign-UUID combinations (self spots, for example), comma-separated. This is not regex, just full matches.
- `-h, --help` - `-h, --help`
Print help. Print help.
@@ -109,10 +118,10 @@ mwtchahrd -i 127.0.0.1 -p 8000 -u
### Spotting to [Spothole.app](https://spothole.app) API ### Spotting to [Spothole.app](https://spothole.app) API
Sends only UI frames and mode is hardcoded as `PKT`. Spot from callsign `W1CDN` and report fixed frequency 14.105 MHz: Sends only UI frames and mode is hardcoded as `PKT`. Spot from callsign `W1CDN` and report fixed frequency 14.105 MHz. Report your location as spotter and don't spot your own beacons.
```bash ```bash
mwtchahrd --ip 192.168.0.6 --port 8000 --spotter W1CDN --spothole --freq 14105000 mwtchahrd --ip 192.168.0.6 --port 8000 --spotter W1CDN --spothole --freq 14105000 --my-lat 47.01 --my-lon -97.01 --filter W1CDN-1,W1CDN-7
``` ```
## Code Overview ## Code Overview

7
auto-build-release.sh Executable file
View File

@@ -0,0 +1,7 @@
echo "Building release for multiple targets";
cargo build --release;
cross build --release --target=aarch64-unknown-linux-gnu;
cross build --release --target=x86_64-unknown-linux-gnu;
cross build --release --target=armv7-unknown-linux-gnueabihf;
cross build --release --target=arm-unknown-linux-gnueabihf;
cross build --release --target=x86_64-pc-windows-gnu;

7
auto-build.sh Executable file
View File

@@ -0,0 +1,7 @@
echo "Building dev for multiple targets";
cargo build;
cross build --target=aarch64-unknown-linux-gnu;
cross build --target=x86_64-unknown-linux-gnu;
cross build --target=armv7-unknown-linux-gnueabihf;
cross build --target=arm-unknown-linux-gnueabihf;
cross build --target=x86_64-pc-windows-gnu;

View File

@@ -83,10 +83,14 @@ struct Cli {
#[arg(short = 'y', long, allow_negative_numbers = true)] #[arg(short = 'y', long, allow_negative_numbers = true)]
my_lat: Option<f64>, my_lat: Option<f64>,
/// Spotter longitude DD.dddd; to send negaitve (W or S), use = e.g. --my-lon=-97.1 /// Spotter longitude DD.dddd
#[arg(short = 'x', long, allow_negative_numbers = true)] #[arg(short = 'x', long, allow_negative_numbers = true)]
my_lon: Option<f64>, my_lon: Option<f64>,
/// Filter out spots from these callsign-UUID combinations (self spots, for example), comma-separated
#[arg(short = 't', long, value_delimiter = ',', num_args = 1..,)]
filter: Vec<String>,
} }
/// Convert a byte slice into a hex-dump string for debugging purposes. /// Convert a byte slice into a hex-dump string for debugging purposes.
@@ -450,7 +454,7 @@ fn handle_frame(frame: &AgwFrame, cli: &Cli, buffers: &mut BufferManager, loc_st
if cli.ui_only && summary != "UI" { if cli.ui_only && summary != "UI" {
return; return;
} }
// In non-debug mode, print the session line and any additional lines. // In non-debug mode, print the session line and any additional lines.
if !cli.debug { if !cli.debug {
print_session_line(&timestamp, &source, &final_destination, &summary); print_session_line(&timestamp, &source, &final_destination, &summary);
@@ -484,30 +488,33 @@ fn handle_frame(frame: &AgwFrame, cli: &Cli, buffers: &mut BufferManager, loc_st
//println!("Stored location: {} {}", stored_loc[0].clone(), stored_loc[1].clone()); //println!("Stored location: {} {}", stored_loc[0].clone(), stored_loc[1].clone());
// If Spothole is enabled // If Spothole is enabled and this is a UI frame and the source is not filtered
if summary == "UI" && cli.spothole { if summary == "UI" && cli.spothole && !cli.filter.contains(&source){
// POST JSON // POST JSON
let packet = json!({ let packet = json!({
"dx_call": &source_call, "dx_call": &source_call,
"dx_ssid": &source_ssid, "dx_ssid": &source_ssid,
"de_call": &my_call, "de_call": &my_call,
"de_latitude": &my_lat, "de_latitude": &my_lat,
"de_longitude": &my_lon, "de_longitude": &my_lon,
"freq": &freq, "freq": &freq,
"comment": &text, "comment": &text,
"dx_latitude": &json_lat, "dx_latitude": &json_lat,
"dx_longitude": &json_lon, "dx_longitude": &json_lon,
"mode": "PKT", "mode": "PKT",
"mode_type": "DATA", "mode_type": "DATA",
"mode_source": "SPOT", "mode_source": "SPOT",
"time": SystemTime::now().duration_since(UNIX_EPOCH).unwrap().as_secs(), "time": SystemTime::now().duration_since(UNIX_EPOCH).unwrap().as_secs(),
}); });
let client = reqwest::blocking::Client::new(); let client = reqwest::blocking::Client::new();
let res = client.post(spothole_url) let res = client.post(spothole_url)
.json(&packet) .json(&packet)
.send(); .send();
println!("sent to {} = {}", spothole_url, packet); println!("sent to {} = {}", spothole_url, packet);
println!("res = {res:?}"); println!("res = {res:?}");
// If it's filtered, send a note about that
} else if summary == "UI" && cli.spothole && cli.filter.contains(&source) {
println!("Filter {:?} applied, not spotted", cli.filter)
} }
// Send UDP // Send UDP
@@ -538,6 +545,13 @@ fn aprs_loc(packet: &str) -> (f64, f64) {
// Capture different pieces of the location string // Capture different pieces of the location string
let re_loc = Regex::new(r"(?P<latd>\d{2})(?P<latm>[\d ]{2}\.[\d ]{2})(?P<ns>[nsNS])/(?P<lond>\d{3})(?P<lonm>[\d ]{2}\.[\d ]{2})(?P<ew>[ewEW])").unwrap(); let re_loc = Regex::new(r"(?P<latd>\d{2})(?P<latm>[\d ]{2}\.[\d ]{2})(?P<ns>[nsNS])/(?P<lond>\d{3})(?P<lonm>[\d ]{2}\.[\d ]{2})(?P<ew>[ewEW])").unwrap();
// Check for 3rd party traffic, which may include location
let re_3p = Regex::new(r"(?<tcpip>TCPIP)").unwrap();
let mut loc3p: bool = false;
if re_3p.is_match(&packet) {
loc3p = true;
}
// Only proceed if there were captures // Only proceed if there were captures
match re_loc.captures(&packet) { match re_loc.captures(&packet) {
Some(_caps) => { Some(_caps) => {
@@ -565,7 +579,14 @@ fn aprs_loc(packet: &str) -> (f64, f64) {
// String to paste into map for testing // String to paste into map for testing
//println!("{}, {}", lat_dec, lon_dec); //println!("{}, {}", lat_dec, lon_dec);
// But if after all that, it's a 3rd-party location, toss it
// This is the only capture group we ID by numberf
if loc3p == true {
(lat_dec, lon_dec) = (-9999.0_f64, -9999.0_f64);
println!("Dropping location, looks like 3rd party");
}
// Return // Return
(lat_dec, lon_dec) (lat_dec, lon_dec)