Simplified Ham Alerts on a deGoogled Phone
Table of Contents
In my previous post I showed how to get notifications from HamAlert on a deGoogled phone. That solution requires setting up Node Red to template HamAlert spots and forward them the ntfy.
After I wrote that post, I noticed that ntfy had support to template JSON posted to it. This means that anyone with access to a ntfy server can receive HamAlert spots on their deGoogled phone without needing to set up Node Red.
If I send JSON and a template to a ntfy server.
$ curl \
--globoff \
--data '{"title": "SIMULATED spot N0CALL (28.420 SSB)"}' \
'https://ntfy.fediverse.radio/6RpjkaBXXK9SeQ48?tpl=1&m={{.title}}'
I get a formatted notification. I was so excited to see that there was a simpler way to get HamAlert spots on a deGoogled phone.
I tried to configure HamAlert with that URL, but at the time, HamAlert sent form data to ntfy. This is a sensible decision for HamAlert to make. Any web framework can parse form data.
Unfortunately, If I send form data to ntfy I receive an error. This is because ntfy doesn’t support form data. This was sad news. I would have to get ntfy to support form data or get HamAlert to send JSON.
$ curl \
--globoff \
--data 'SIMULATED%20spot%20N0CALL%20%2828.420%20SSB%29' \
'https://ntfy.fediverse.radio/6RpjkaBXXK9SeQ48?tpl=1&m={{.title}}'
{"code":40042,"http":400,"error":"invalid request: message body must be JSON if templating is enabled","link":"https://ntfy.sh/docs/publish/#message-templating"}
On a whim, I posted a feature request to HamAlert’s forum. I didn’t expect them to add support for JSON, but today I got an email that Manuel HB9DQM had implemented support for POSTing JSON in a URL destination!
So now anyone with access to a ntfy server can receive notifications without setting up Node Red or a custom service to convert HamAlert data to a ntfy request. You can now use ntfy with a carefully constructed URL right in HamAlert.
Does this mean that you still need to set up a ntfy server for this to work? Fortunately not. You can use the public ntfy server at https://ntfy.sh/app or the ntfy server that Christopher M0YNG set up for amateurs at https://ntfy.fediverse.radio/. While these servers are available for you to use, you might want to run your own if you can. The default IP based rate limits in ntfy may cause you to miss alerts (see the Caveat section below)
#
How-to
Let’s start by creating a topic on your ntfy server. I am going to use the fediverse.radio server for this example.
Visit https://ntfy.fediverse.radio and click “Subscribe to topic”. You can put in any topic name, but make sure that it can’t be guessed. If someone guesses your topic name, they can send notifications to your phone. You can click “Generate Name” if you don’t care what the name is.
Next we will have to create a template for the HamAlert JSON. ntfy uses Go Templates. You can use the Go Template Playground to experiment with. Here is an example POTA spot you can use. I will provide examples of other sources at the end of this post if you want to see how your template looks for different spot sources.
{
"fullCallsign":"N6T",
"callsign":"N6T",
"frequency":"14.343",
"band":"20m",
"mode":"ssb",
"modeDetail":"ssb",
"time":"18:50",
"spotter":"W7AYQ",
"rawText":"18:50 N6T in US-2838 (L.L. Stub Stewart State Park)",
"title":"POTA N6T in US-2838 (14.3430 SSB)",
"comment":"CQ",
"source":"pota",
"state":["US_OR"],
"spotterState":"US_OR",
"wwffRef":"US-2838",
"wwffDivision":"US",
"wwffName":"L.L. Stub Stewart State Park",
"dxcc":"291",
"entity":"United States of America",
"cq":"3",
"continent":"NA",
"homeDxcc":"291",
"homeEntity":"United States of America",
"spotterDxcc":"291",
"spotterEntity":"United States of America",
"spotterCq":"3",
"spotterContinent":"NA",
"triggerComment":"CQ Pride"
}
The simplest template that covers all the spot types is this template:
{{ .rawText }}
This renders a notification that looks like this. It isn’t the best template for this. You can come up with a better one in the Playground
18:50 N6T in US-2838 (L.L. Stub Stewart State Park)
The hardest part of this whole thing is turning that template into a URL parameter. If you go to https://www.urlencoder.org/ and paste in the template, it will produce this gnarly URL encoded string
%7B%7B%20.rawText%20%7D%7D
You will want to put the URL encoded string at the end of your topic URL. Note that the 6RpjkaBXXK9SeQ48
part needs to match your topic name
https://ntfy.fediverse.radio/6RpjkaBXXK9SeQ48?tpl=1&m=
So you should have this wild URL
https://ntfy.fediverse.radio/6RpjkaBXXK9SeQ48?tpl=1&m=%7B%7B%20.rawText%20%7D%7D
In HamAlert, open the Destinations page and scroll to the bottom. Paste in the gnarly URL, click POST (JSON)
, and click save.
Open the Ham Alert simulate page , fill out the form with sample data, and click Send.
You can then see if it worked by opening the ntfy URL https://ntfy.fediverse.radio/6RpjkaBXXK9SeQ48.
#
Caveat
One thing to consider when using publicly available ntfy servers is that they are rate-limited by IP. The default request limit per IP is pretty low. It uses a token bucket system. Each IP gets a bucket of 60 requests and the requests are refilled 1 per 5s. The result of this rate-limiting is that you will miss any spots HamAlert sends that are rate-limited.
The IP that ntfy sees is the IP of HamAlert’s worker nodes. I don’t know how many worker nodes HamAlert uses. If ntfy only sees one IP from HamAlert, that bucket is shared by everyone who uses this solution.
If someone configures HamAlert to spam the ntfy server, everyone using that ntfy server will not get alerts until HamAlert chills out. They may never realize that they’ve broken everyone’s alerts because the only feedback they receive is that their alerts have stopped.
You can avoid this issue by running your own ntfy server. I would recommend that you run your own nfty server if you have the ability. That way you can tune these limits yourself.
This will also reduce the administration burden for people that have graciously hosting a public ntfy server such as Christopher M0YNG’s ntfy.fediverse.radio. HamAlert has the potential to spam his ntfy server and there’s no de-duping of spots. If 50 people send spots to nfty.fediverse.radio whenever Jim N4JAW rides his bike to a park, that’s 50 spots sent to Christopher’s ntfy server.
If you configure HamAlert to send every POTA or FT8 spot to ntfy.fediverse.radio, you may overload his server. So be careful.
#
Conclusion
This is very exciting. I love it when a complicated solution can be simplified. Thanks to Manuel HB9DQM for adding support for JSON in HamAlert URL destinations. Thanks to Christopher M0YNG for hosing a ntfy server for amateurs to use.
If you need any help setting this up, don’t hesitant to reach out to me on mastodon.radio or by email via eric at this site’s domain.
#
Sample alert JSON
Here are some sample Ham Alert JSON packets that I captured from HamAlert itself. Each source is represented for you.
##
SOTAWatch
{
"fullCallsign": "N4AJJ",
"callsign": "N4AJJ",
"frequency": "7.213",
"band": "40m",
"mode": "ssb",
"modeDetail": "ssb",
"time": "15:40",
"spotter": "N4AJJ",
"rawText": "15:40 N4AJJ on W4C/CM-148 (Jumpoff Mountain, 957m, 4pt) 7.213 ssb: Cq cq cq",
"title": "SOTA N4AJJ on W4C/CM-148 (7.213 SSB)",
"comment": "Cq cq cq",
"source": "sotawatch",
"state": "US_KY",
"spotterState": "US_KY",
"summitName": "Jumpoff Mountain",
"summitHeight": 957,
"summitPoints": 4,
"summitRef": "W4C/CM-148",
"dxcc": 291,
"entity": "United States of America",
"cq": "4",
"continent": "NA",
"homeDxcc": 291,
"homeEntity": "United States of America",
"spotterDxcc": 291,
"spotterEntity": "United States of America",
"spotterCq": "4",
"spotterContinent": "NA",
"triggerComment": ""
}
##
Reverse Beacon Network
FT8
{
"fullCallsign": "VK3BAC",
"callsign": "VK3BAC",
"frequency": "21.074",
"band": "15m",
"mode": "ft8",
"modeDetail": "ft8",
"time": "04:48",
"spotter": "W3OA",
"snr": "-10",
"rawText": "DX de W3OA-#: 21074.0 VK3BAC FT8 -10 dB CQ 0448Z",
"title": "RBN spot VK3BAC (21.074 FT8)",
"source": "rbn",
"spotterState": "US_NC",
"dxcc": "150",
"entity": "Australia",
"cq": "30",
"continent": "OC",
"homeDxcc": "150",
"homeEntity": "Australia",
"spotterDxcc": "291",
"spotterEntity": "United States of America",
"spotterCq": "5",
"spotterContinent": "NA",
"triggerComment": ""
}
##
DX Cluster
{
"fullCallsign": "OE1AES",
"callsign": "OE1AES",
"frequency": "14.0811",
"band": "20m",
"mode": "ft4",
"modeDetail": "ft4",
"time": "04:47",
"spotter": "VK5WU",
"rawText": "DX de VK5WU: 14081.1 OE1AES FT4: PF95 <-=-> JN88, -03 TU 0447Z",
"title": "Cluster spot OE1AES (14.0811)",
"comment": "FT4: PF95 <-=-> JN88, -03 TU",
"source": "cluster",
"qsl": "eqsl",
"dxcc": "206",
"entity": "Austria",
"cq": "15",
"continent": "EU",
"homeDxcc": "206",
"homeEntity": "Austria",
"spotterDxcc": "150",
"spotterEntity": "Australia",
"spotterCq": "30",
"spotterContinent": "OC",
"triggerComment": ""
}
##
PSK Reporter
{
"fullCallsign": "VK2WL",
"callsign": "VK2WL",
"frequency": "14.076055",
"band": "20m",
"mode": "ft8",
"modeDetail": "ft8",
"time": "04:47",
"spotter": "ZL4CAT",
"snr": "0",
"rawText": "DX de ZL4CAT: 14076.1 VK2WL FT8 0 dB 0447Z (DigiSkimmer 0.35.1 KiwiSDR)",
"title": "PSK Reporter spot VK2WL (14.076055 FT8)",
"source": "pskreporter",
"dxcc": "150",
"entity": "Australia",
"cq": "30",
"continent": "OC",
"homeDxcc": "150",
"homeEntity": "Australia",
"spotterDxcc": "170",
"spotterEntity": "New Zealand",
"spotterCq": "32",
"spotterContinent": "OC",
"triggerComment": ""
}
##
POTA
{
"fullCallsign": "N6T",
"callsign": "N6T",
"frequency": "14.343",
"band": "20m",
"mode": "ssb",
"modeDetail": "ssb",
"time": "18:50",
"spotter": "W7AYQ",
"rawText": "18:50 N6T in US-2838 (L.L. Stub Stewart State Park)",
"title": "POTA N6T in US-2838 (14.3430 SSB)",
"comment": "CQ",
"source": "pota",
"state": [
"US_OR"
],
"spotterState": "US_OR",
"wwffRef": "US-2838",
"wwffDivision": "US",
"wwffName": "L.L. Stub Stewart State Park",
"dxcc": "291",
"entity": "United States of America",
"cq": "3",
"continent": "NA",
"homeDxcc": "291",
"homeEntity": "United States of America",
"spotterDxcc": "291",
"spotterEntity": "United States of America",
"spotterCq": "3",
"spotterContinent": "NA",
"triggerComment": "CQ Pride"
}