Merge branch 'main' into 95-send-spots-to-xota

# Conflicts:
#	README.md
#	server/handlers/api/addspot.py
#	server/handlers/api/options.py
#	spotproviders/tiles.py
#	templates/about.html
#	templates/add_spot.html
#	templates/alerts.html
#	templates/api_only_home.html
#	templates/bands.html
#	templates/base.html
#	templates/conditions.html
#	templates/map.html
#	templates/spots.html
#	templates/status.html
#	webassets/css/style.css
#	webassets/js/add-spot.js
#	webassets/js/geo.js
#	webassets/js/ui-ham.js
#	webassets/js/utils.js
This commit is contained in:
Ian Renton
2026-06-19 21:48:10 +01:00
91 changed files with 1835 additions and 1261 deletions

View File

@@ -1,6 +1,48 @@
<component name="InspectionProjectProfileManager"> <component name="InspectionProjectProfileManager">
<profile version="1.0"> <profile version="1.0">
<option name="myName" value="Project Default" /> <option name="myName" value="Project Default" />
<inspection_tool class="Annotator" enabled="false" level="ERROR" enabled_by_default="false" />
<inspection_tool class="BadExpressionStatementJS" enabled="false" level="WEAK WARNING" enabled_by_default="false" />
<inspection_tool class="CssOverwrittenProperties" enabled="false" level="WARNING" enabled_by_default="false" />
<inspection_tool class="CssUnresolvedCustomProperty" enabled="false" level="ERROR" enabled_by_default="false" />
<inspection_tool class="CssUnusedSymbol" enabled="false" level="WARNING" enabled_by_default="false" />
<inspection_tool class="GrazieInspection" enabled="false" level="GRAMMAR_ERROR" enabled_by_default="false" />
<inspection_tool class="GrazieStyle" enabled="false" level="STYLE_SUGGESTION" enabled_by_default="false" />
<inspection_tool class="HtmlFormInputWithoutLabel" enabled="true" level="WEAK WARNING" enabled_by_default="true" editorAttributes="INFO_ATTRIBUTES" />
<inspection_tool class="HtmlUnknownAttribute" enabled="false" level="WARNING" enabled_by_default="false">
<option name="myValues">
<value>
<list size="0" />
</value>
</option>
<option name="myCustomValuesEnabled" value="true" />
</inspection_tool>
<inspection_tool class="HtmlUnknownTag" enabled="false" level="WARNING" enabled_by_default="false">
<option name="myValues">
<value>
<list size="6">
<item index="0" class="java.lang.String" itemvalue="nobr" />
<item index="1" class="java.lang.String" itemvalue="noembed" />
<item index="2" class="java.lang.String" itemvalue="comment" />
<item index="3" class="java.lang.String" itemvalue="noscript" />
<item index="4" class="java.lang.String" itemvalue="embed" />
<item index="5" class="java.lang.String" itemvalue="script" />
</list>
</value>
</option>
<option name="myCustomValuesEnabled" value="true" />
</inspection_tool>
<inspection_tool class="HtmlUnknownTarget" enabled="false" level="WARNING" enabled_by_default="false" />
<inspection_tool class="HttpUrlsUsage" enabled="false" level="WEAK WARNING" enabled_by_default="false" />
<inspection_tool class="JSDeprecatedSymbols" enabled="false" level="WEAK WARNING" enabled_by_default="false" />
<inspection_tool class="JSIgnoredPromiseFromCall" enabled="false" level="WEAK WARNING" enabled_by_default="false" />
<inspection_tool class="JSJQueryEfficiency" enabled="false" level="WARNING" enabled_by_default="false" />
<inspection_tool class="JSUnnecessarySemicolon" enabled="false" level="WARNING" enabled_by_default="false" />
<inspection_tool class="JSUnresolvedReference" enabled="false" level="WEAK WARNING" enabled_by_default="false" />
<inspection_tool class="JSUnusedGlobalSymbols" enabled="true" level="WEAK WARNING" enabled_by_default="true" editorAttributes="INFO_ATTRIBUTES" />
<inspection_tool class="LanguageDetectionInspection" enabled="false" level="WEAK WARNING" enabled_by_default="false" />
<inspection_tool class="OutdatedRequirementInspection" enabled="false" level="WEAK WARNING" enabled_by_default="false" />
<inspection_tool class="PyBroadExceptionInspection" enabled="false" level="WEAK WARNING" enabled_by_default="false" />
<inspection_tool class="SpellCheckingInspection" enabled="false" level="TYPO" enabled_by_default="false"> <inspection_tool class="SpellCheckingInspection" enabled="false" level="TYPO" enabled_by_default="false">
<option name="processCode" value="true" /> <option name="processCode" value="true" />
<option name="processLiterals" value="true" /> <option name="processLiterals" value="true" />

View File

@@ -1,6 +0,0 @@
<component name="InspectionProjectProfileManager">
<settings>
<option name="USE_PROJECT_PROFILE" value="false" />
<version value="1.0" />
</settings>
</component>

6
.idea/jsLibraryMappings.xml generated Normal file
View File

@@ -0,0 +1,6 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="JavaScriptLibraryMappings">
<file url="file://$PROJECT_DIR$" libraries="{redoc.standalone}" />
</component>
</project>

2
.idea/spothole.iml generated
View File

@@ -3,8 +3,10 @@
<component name="NewModuleRootManager"> <component name="NewModuleRootManager">
<content url="file://$MODULE_DIR$"> <content url="file://$MODULE_DIR$">
<excludeFolder url="file://$MODULE_DIR$/.venv" /> <excludeFolder url="file://$MODULE_DIR$/.venv" />
<excludeFolder url="file://$MODULE_DIR$/webassets/vendor" />
</content> </content>
<orderEntry type="jdk" jdkName="Python 3.13 virtualenv at ~/code/spothole/.venv" jdkType="Python SDK" /> <orderEntry type="jdk" jdkName="Python 3.13 virtualenv at ~/code/spothole/.venv" jdkType="Python SDK" />
<orderEntry type="sourceFolder" forTests="false" /> <orderEntry type="sourceFolder" forTests="false" />
<orderEntry type="library" name="redoc.standalone" level="application" />
</component> </component>
</module> </module>

227
README.md
View File

@@ -1,16 +1,23 @@
# ![Spothole](/webassets/img/logo.png) # ![Spothole](/webassets/img/logo.png)
Spothole is a utility to aggregate "spots" from amateur radio DX clusters and xOTA spotting sites, and provide an open JSON API as well as a website to browse the data. Spothole is a utility to aggregate "spots" from amateur radio DX clusters and xOTA spotting sites, and provide an open
JSON API as well as a website to browse the data.
![Screenshot](/images/screenshot.png) ![Screenshot](/images/screenshot.png)
While there are several other web-based interfaces to DX clusters, and sites that aggregate spots from various outdoor activity programmes for amateur radio, Spothole differentiates itself by supporting a large number of data sources, and by being "API first" rather than just providing a web front-end. This allows other software to be built on top of it. While there are several other web-based interfaces to DX clusters, and sites that aggregate spots from various outdoor
activity programmes for amateur radio, Spothole differentiates itself by supporting a large number of data sources, and
by being "API first" rather than just providing a web front-end. This allows other software to be built on top of it.
The API is deliberately well-defined with an OpenAPI specification and auto-generated API documentation. The API delivers spots in a consistent format regardless of the data source, freeing developers from needing to know how each individual data source presents its data. The API is deliberately well-defined with an OpenAPI specification and auto-generated API documentation. The API
delivers spots in a consistent format regardless of the data source, freeing developers from needing to know how each
individual data source presents its data.
Spothole itself is also open source, Public Domain licenced code that anyone can take and modify. Spothole itself is also open source, Public Domain licenced code that anyone can take and modify.
Supported data sources include DX Clusters, the Reverse Beacon Network (RBN), the APRS Internet Service (APRS-IS), POTA, SOTA, WWFF, GMA, WWBOTA, HEMA, Parks 'n' Peaks, ZLOTA, WOTA, BOTA, LLOTA, WWTOTA, Tiles on the Air, the UK Packet Repeater Network, NG3K, and any site based on the xOTA software by nischu. Supported data sources include DX Clusters, the Reverse Beacon Network (RBN), the APRS Internet Service (APRS-IS), POTA,
SOTA, WWFF, GMA, WWBOTA, HEMA, Parks 'n' Peaks, ZLOTA, WOTA, BOTA, LLOTA, WWTOTA, Tiles on the Air, the UK Packet
Repeater Network, NG3K, and any site based on the xOTA software by nischu.
![Screenshot](/images/screenshot2.png) ![Screenshot](/images/screenshot2.png)
@@ -18,65 +25,88 @@ Supported data sources include DX Clusters, the Reverse Beacon Network (RBN), th
## Accessing the public version ## Accessing the public version
You can access the public version's web interface at [https://spothole.app](https://spothole.app), and see [https://spothole.app/apidocs](https://spothole.app/apidocs) for the API details. You can access the public version's web interface at [https://spothole.app](https://spothole.app), and
see [https://spothole.app/apidocs](https://spothole.app/apidocs) for the API details.
This is a Progressive Web App, so you can also "install" it to your Android or iOS device by accessing it in Chrome or Safari respectively, and following the menu-driven process for installing PWAs. This is a Progressive Web App, so you can also "install" it to your Android or iOS device by accessing it in Chrome or
Safari respectively, and following the menu-driven process for installing PWAs.
You are more than welcome to use the data and the API that Spothole provides to power your own software. There are many ways to do this; see below. You are more than welcome to use the data and the API that Spothole provides to power your own software. There are many
ways to do this; see below.
## Embedding Spothole in another website ## Embedding Spothole in another website
You can embed Spothole's web interface in another website, e.g. for use as part of a ham radio custom dashboard. You can embed Spothole's web interface in another website, e.g. for use as part of a ham radio custom dashboard.
URL parameters can be used to trigger an "embedded" mode which hides the headers, footers and settings. In this mode, you provide configuration for the various filter and display options via additional URL parameters. Any settings that the user has set for Spothole are ignored. This is so that the embedding site can select, for example, their choice of dark mode or SIG filters, which will not impact how Spothole appears when the user accesses it directly. Effectively, it becomes separate to their normal Spothole settings. URL parameters can be used to trigger an "embedded" mode which hides the headers, footers and settings. In this mode,
you provide configuration for the various filter and display options via additional URL parameters. Any settings that
the user has set for Spothole are ignored. This is so that the embedding site can select, for example, their choice of
dark mode or SIG filters, which will not impact how Spothole appears when the user accesses it directly. Effectively, it
becomes separate to their normal Spothole settings.
Setting `embedded` to true is important for the rest of the settings to be applied; otherwise, the user's defaults will be used in preference to the URL params. Setting `embedded` to true is important for the rest of the settings to be applied; otherwise, the user's defaults will
be used in preference to the URL params.
These are supplied with the URL to the page you want to embed, for example for an embedded version of the band map in dark mode, use `https://spothole.app/bands?embedded=true&dark-mode=true`. For an embedded version of the main spots/home page in the system light/dark mode, use `https://spothole.app/?embedded=true`. For dark mode showing 70cm TOTA spots only, use `https://spothole.app/?embedded=true&dark-mode=true&sig=TOTA&band=70cm`. Providing no URL params causes the page to be loaded in the normal way it would when accessed directly in the user's browser. These are supplied with the URL to the page you want to embed, for example for an embedded version of the band map in
dark mode, use `https://spothole.app/bands?embedded=true&dark-mode=true`. For an embedded version of the main spots/home
page in the system light/dark mode, use `https://spothole.app/?embedded=true`. For dark mode showing 70cm TOTA spots
only, use `https://spothole.app/?embedded=true&dark-mode=true&sig=TOTA&band=70cm`. Providing no URL params causes the
page to be loaded in the normal way it would when accessed directly in the user's browser.
The supported parameters are as follows. Generally these match the equivalent parameters in the real Spothole API, where a mapping exists. The supported parameters are as follows. Generally these match the equivalent parameters in the real Spothole API, where
a mapping exists.
| Name | Allowed Values | Default | Example | Description | | Name | Allowed Values | Default | Example | Description |
|-------------------|-------------------------|---------|--------------------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------| |------------------|-------------------------|---------|-------------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| `embedded` | `true`, `false` | `false` | `?embedded=true` | Enables embedded mode. | | `embedded` | `true`, `false` | `false` | `?embedded=true` | Enables embedded mode. |
| `color-scheme` | `light`, `dark`, `auto` | `auto` | `?color-scheme=dark` | Forces light or dark mode in preference to the operating system default. | | `color-scheme` | `light`, `dark`, `auto` | `auto` | `?color-scheme=dark` | Forces light or dark mode in preference to the operating system default. |
| `time-zone` | `UTC`, `local` | `UTC` | `?time-zone=local` | Sets times to be in UTC or local time. | | `time-zone` | `UTC`, `local` | `UTC` | `?time-zone=local` | Sets times to be in UTC or local time. |
| `limit` | 10, 25, 50, 100 | 50 | `?limit=50` | Sets the number of spots that will be displayed on the main spots page | | `limit` | 10, 25, 50, 100 | 50 | `?limit=50` | Sets the number of spots that will be displayed on the main spots page |
| `limit` | 25, 50, 100, 200, 500 | 100 | `?limit=100` | Sets the number of alerts that will be displayed on the alerts page | | `limit` | 25, 50, 100, 200, 500 | 100 | `?limit=100` | Sets the number of alerts that will be displayed on the alerts page |
| `max_age` | 300, 600, 1800, 3600 | 1800 | `?max_age=1800` | Sets the maximum age of spots displayed on the map and bands pages, in seconds. | | `max_age` | 300, 600, 1800, 3600 | 1800 | `?max_age=1800` | Sets the maximum age of spots displayed on the map and bands pages, in seconds. |
| `band` | Comma-separated list | (all) | `?band=20m,40m` | Sets the list of bands that will be shown on the spots, bands and map pages. Available options match the labels of the buttons in the standard web interface. | | `band` | Comma-separated list | (all) | `?band=20m,40m` | Sets the list of bands that will be shown on the spots, bands and map pages. Available options match the labels of the buttons in the standard web interface. |
| `sig` | Comma-separated list | (all) | `?sig=POTA,SOTA,NO_SIG` | Sets the list of SIGs that will be shown on the spots, bands and map pages. Available options match the labels of the buttons in the standard web interface. | | `sig` | Comma-separated list | (all) | `?sig=POTA,SOTA,NO_SIG` | Sets the list of SIGs that will be shown on the spots, bands and map pages. Available options match the labels of the buttons in the standard web interface. |
| `source` | Comma-separated list | (all) | `?source=Cluster` | Sets the list of sources that will be shown on any spot or alert pages. Available options match the labels of the buttons in the standard web interface. | | `source` | Comma-separated list | (all) | `?source=Cluster` | Sets the list of sources that will be shown on any spot or alert pages. Available options match the labels of the buttons in the standard web interface. |
| `mode_type` | Comma-separated list | (all) | `?mode_type=PHONE,CW` | Sets the list of mode types that will be shown on the spots, bands and map pages. Available options match the labels of the buttons in the standard web interface. | | `mode_type` | Comma-separated list | (all) | `?mode_type=PHONE,CW` | Sets the list of mode types that will be shown on the spots, bands and map pages. Available options match the labels of the buttons in the standard web interface. |
| `dx_continent` | Comma-separated list | (all) | `?dx_continent=NA,SA` | Sets the list of DX Continents that will be shown on any spot or alert pages. Available options match the labels of the buttons in the standard web interface. | | `dx_continent` | Comma-separated list | (all) | `?dx_continent=NA,SA` | Sets the list of DX Continents that will be shown on any spot or alert pages. Available options match the labels of the buttons in the standard web interface. |
| `de_continent` | Comma-separated list | (all) | `?de_continent=EU` | Sets the list of DE Continents that will be shown on the spots, bands and map pages. Available options match the labels of the buttons in the standard web interface. | | `de_continent` | Comma-separated list | (all) | `?de_continent=EU` | Sets the list of DE Continents that will be shown on the spots, bands and map pages. Available options match the labels of the buttons in the standard web interface. |
| `map-center-lat` | Numeric (decimal) | (auto) | `?map-center-lat=51.5` | Sets the initial latitude of the map centre on the map page. If omitted, the map auto-fits to the loaded spots. | | `map-center-lat` | Numeric (decimal) | (auto) | `?map-center-lat=51.5` | Sets the initial latitude of the map centre on the map page. If omitted, the map auto-fits to the loaded spots. |
| `map-center-lon` | Numeric (decimal) | (auto) | `?map-center-lon=-0.1` | Sets the initial longitude of the map centre on the map page. If omitted, the map auto-fits to the loaded spots. | | `map-center-lon` | Numeric (decimal) | (auto) | `?map-center-lon=-0.1` | Sets the initial longitude of the map centre on the map page. If omitted, the map auto-fits to the loaded spots. |
| `map-zoom` | Numeric (integer) | (auto) | `?map-zoom=6` | Sets the initial zoom level of the map on the map page. If omitted, the map auto-fits to the loaded spots. | | `map-zoom` | Numeric (integer) | (auto) | `?map-zoom=6` | Sets the initial zoom level of the map on the map page. If omitted, the map auto-fits to the loaded spots. |
More will be added soon to allow customisation of filters and other display properties. More will be added soon to allow customisation of filters and other display properties.
## Writing your own client ## Writing your own client
One of the key strengths of Spothole is that the API is well-defined and open to anyone to use. This means you can build your own software that uses data from Spothole. One of the key strengths of Spothole is that the API is well-defined and open to anyone to use. This means you can build
your own software that uses data from Spothole.
As well as the main API endpoints to fetch spots and alerts, with various possible query parameters, there are also Server-Sent Events (SSE) API endpoints to receive a live feed, plus various utility lookup endpoints for things like callsign and park data. As well as the main API endpoints to fetch spots and alerts, with various possible query parameters, there are also
Server-Sent Events (SSE) API endpoints to receive a live feed, plus various utility lookup endpoints for things like
callsign and park data.
Various approaches exist to writing your own client, but in general: Various approaches exist to writing your own client, but in general:
* Refer to the API docs. These are built on an OpenAPI definition file (`/webassets/apidocs/openapi.yml`), which you can automatically use to generate a client skeleton using various software. * Refer to the API docs. These are built on an OpenAPI definition file (`/webassets/apidocs/openapi.yml`), which you can
automatically use to generate a client skeleton using various software.
* Call the main "spots" or "alerts" API endpoints to get the data you want. Apply filters if necessary. * Call the main "spots" or "alerts" API endpoints to get the data you want. Apply filters if necessary.
* Call the "options" API to get an idea of which bands, modes etc. the server knows about. You might want to do that first before calling the spots/alerts APIs, to allow you to populate your filters correctly. * Call the "options" API to get an idea of which bands, modes etc. the server knows about. You might want to do that
* Refer to the provided HTML/JS interface for a reference on different approaches. For example, the "map" and "bands" pages simply query the main spot API on a timer, whereas the main/spots page combines this approach with using the Server-Sent Events (SSE) endpoint to update live. first before calling the spots/alerts APIs, to allow you to populate your filters correctly.
* Refer to the provided HTML/JS interface for a reference on different approaches. For example, the "map" and "bands"
pages simply query the main spot API on a timer, whereas the main/spots page combines this approach with using the
Server-Sent Events (SSE) endpoint to update live.
* Let me know if you get stuck, I'm happy to help! * Let me know if you get stuck, I'm happy to help!
## Running your own copy ## Running your own copy
If you want to run a copy of Spothole with different configuration settings than the main instance, you can download it and run it on your own local machine or server. If you want to run a copy of Spothole with different configuration settings than the main instance, you can download it
and run it on your own local machine or server.
You will require Python version 3.8 or later. If you encounter an error about `gdal-config` during the following process, you will also need `libgdal-dev` installed. You will require Python version 3.8 or later. If you encounter an error about `gdal-config` during the following
process, you will also need `libgdal-dev` installed.
To download and set up Spothole on a Debian server, run the following commands. Other operating systems will likely be similar. To download and set up Spothole on a Debian server, run the following commands. Other operating systems will likely be
similar.
```bash ```bash
git clone ssh://git@git.ianrenton.com/ian/spothole.git git clone ssh://git@git.ianrenton.com/ian/spothole.git
@@ -88,13 +118,23 @@ deactivate
cp config-example.yml config.yml cp config-example.yml config.yml
``` ```
Then edit `config.yml` in your text editor of choice to set up the software as you like it. Mostly, this will involve enabling or disabling the various providers of spot and alert data. Then edit `config.yml` in your text editor of choice to set up the software as you like it. Mostly, this will involve
enabling or disabling the various providers of spot and alert data.
By default, all outdoor programme providers are enabled, as is one cluster node and the NG3K DXpedition data. The RBN spot providers are turned off by default due to the volume of traffic from CW/RTTY/FT8 skimmers, and the APRS and Packet spot providers are off by default on the assumption that Spothole users want a spot with a human at the other end of it, but all can be easily re-enabled. By default, all outdoor programme providers are enabled, as is one cluster node and the NG3K DXpedition data. The RBN
spot providers are turned off by default due to the volume of traffic from CW/RTTY/FT8 skimmers, and the APRS and Packet
spot providers are off by default on the assumption that Spothole users want a spot with a human at the other end of it,
but all can be easily re-enabled.
Other parameters you will want to update include the base URL to your instance, and whether you want to serve a full web-based DX cluster interface or just the API endpoints for client software to use. Other parameters you will want to update include the base URL to your instance, and whether you want to serve a full
web-based DX cluster interface or just the API endpoints for client software to use.
`config.yml` has an entry for a Clublog API key. If provided, this will allow Spothole to retrieve some more information about DX spots. The software will work just fine without it, but you may find a few country flags etc. are less accurate or missing. Clublog API keys are free, but you'll need to get your own by submitting a helpdesk ticket and explaining what you'll use it for. The admin team are happy with the rate of requests made by my Spothole server, so unless you change the source code of yours to radically increase the rate of querying Clublog, I'm sure they will be fine with your server too. `config.yml` has an entry for a Clublog API key. If provided, this will allow Spothole to retrieve some more information
about DX spots. The software will work just fine without it, but you may find a few country flags etc. are less accurate
or missing. Clublog API keys are free, but you'll need to get your own by submitting a helpdesk ticket and explaining
what you'll use it for. The admin team are happy with the rate of requests made by my Spothole server, so unless you
change the source code of yours to radically increase the rate of querying Clublog, I'm sure they will be fine with your
server too.
Once you're happy with the content of `config.yml`, you can proceed to running the software. Once you're happy with the content of `config.yml`, you can proceed to running the software.
@@ -105,13 +145,16 @@ source .venv/bin/activate
python3 spothole.py python3 spothole.py
``` ```
The software can take a few seconds to start up, mostly because it is downloading an updated file to match callsigns to countries. This is normal, don't panic! The software can take a few seconds to start up, mostly because it is downloading an updated file to match callsigns to
countries. This is normal, don't panic!
If you see some errors on startup, check your configuration, e.g. in case you have specified a port for the web server that is already in use by something else. If you see some errors on startup, check your configuration, e.g. in case you have specified a port for the web server
that is already in use by something else.
### Multiple cluster nodes with different settings ### Multiple cluster nodes with different settings
Dan, S50U has written in with his Spothole cluster settings. He is using a cluster node which provides RBN spots, and uses different SSIDs on his callsign to get different settings when logged into the same cluster node. For example: Dan, S50U has written in with his Spothole cluster settings. He is using a cluster node which provides RBN spots, and
uses different SSIDs on his callsign to get different settings when logged into the same cluster node. For example:
``` ```
- -
@@ -207,9 +250,12 @@ For each callsign-SSID, we also specify our basic information with commands:
### systemd configuration ### systemd configuration
If you want Spothole to run automatically on startup on a Linux distribution that uses `systemd`, follow the instructions here. For distros that don't use `systemd`, or Windows/OSX/etc., you can find generic instructions for your OS online. If you want Spothole to run automatically on startup on a Linux distribution that uses `systemd`, follow the
instructions here. For distros that don't use `systemd`, or Windows/OSX/etc., you can find generic instructions for your
OS online.
Create a file at `/etc/systemd/system/spothole.service`. Give it the following content, adjusting for the user you want to run it as and the directory in which you have installed it: Create a file at `/etc/systemd/system/spothole.service`. Give it the following content, adjusting for the user you want
to run it as and the directory in which you have installed it:
``` ```
[Unit] [Unit]
@@ -239,11 +285,19 @@ Check the service has started up correctly with `sudo journalctl -u spothole -f`
### nginx Reverse Proxy configuration ### nginx Reverse Proxy configuration
Web servers generally serve their pages from port 80. However, it's best not to serve Spothole's web interface directly on port 80, as that requires root privileges on a Linux system. It also and prevents us using HTTPS to serve a secure site, since Spothole itself doesn't directly support acting as an HTTPS server. The normal solution to this is to use a "reverse proxy" setup, where a general web server handles HTTP and HTTP requests (to port 80 & 443 respectively), then passes on the request to the back-end application (in this case Spothole). nginx is a common choice for this general web server. Web servers generally serve their pages from port 80. However, it's best not to serve Spothole's web interface directly
on port 80, as that requires root privileges on a Linux system. It also and prevents us using HTTPS to serve a secure
site, since Spothole itself doesn't directly support acting as an HTTPS server. The normal solution to this is to use
a "reverse proxy" setup, where a general web server handles HTTP and HTTP requests (to port 80 & 443 respectively), then
passes on the request to the back-end application (in this case Spothole). nginx is a common choice for this general web
server.
To set up nginx as a reverse proxy that sits in front of Spothole, first ensure it's installed e.g. `sudo apt install nginx`, and enabled e.g. `sudo systemd enable nginx`. To set up nginx as a reverse proxy that sits in front of Spothole, first ensure it's installed e.g.
`sudo apt install nginx`, and enabled e.g. `sudo systemd enable nginx`.
Create a file at `/etc/nginx/sites-available/` called `spothole`. Give it the following contents, replacing `spothole.app` with the domain name on which you want to run Spothole. If you changed the port on which Spothole runs, update that on the "proxy_pass" line too. Create a file at `/etc/nginx/sites-available/` called `spothole`. Give it the following contents, replacing
`spothole.app` with the domain name on which you want to run Spothole. If you changed the port on which Spothole runs,
update that on the "proxy_pass" line too.
```nginx ```nginx
server { server {
@@ -298,13 +352,19 @@ server {
} }
``` ```
One further change you might want to make to the file above is the `add_header Access-Control-Allow-Origin` statements. These are what's used on One further change you might want to make to the file above is the `add_header Access-Control-Allow-Origin` statements.
my own Spothole server to make sure that other third-party web-based software can get the data from my instance, and applies to any endpoint underneath `/api`. If you want These are what's used on
my own Spothole server to make sure that other third-party web-based software can get the data from my instance, and
applies to any endpoint underneath `/api`. If you want
*your* Spothole instance to be set up the same way, so that others can write software in JavaScript that can access it, *your* Spothole instance to be set up the same way, so that others can write software in JavaScript that can access it,
leave this intact. But if you want your Spothole instance to only be usable by scripts running on the web server you write, leave this intact. But if you want your Spothole instance to only be usable by scripts running on the web server you
you can remove these lines. (Note that this doesn't stop other people writing *non-web-based* software that accesses your write,
Spothole API&mdash;the enforcement of cross-origin headers only happens within the user's browser. If you need to lock your you can remove these lines. (Note that this doesn't stop other people writing *non-web-based* software that accesses
instance down so that no-one else can access it with *any* software, that's an aspect of nginx or firewall config that you will need your
Spothole API&mdash;the enforcement of cross-origin headers only happens within the user's browser. If you need to lock
your
instance down so that no-one else can access it with *any* software, that's an aspect of nginx or firewall config that
you will need
to find help with elsewhere.) to find help with elsewhere.)
Now, make a symbolic link to enable the site: Now, make a symbolic link to enable the site:
@@ -314,17 +374,22 @@ cd /etc/nginx/sites-enabled
sudo ln -sf ../sites-available/spothole sudo ln -sf ../sites-available/spothole
``` ```
Test that your nginx config isn't broken using `nginx -t`. If it works, restart nginx with `sudo systemctl restart nginx`. Test that your nginx config isn't broken using `nginx -t`. If it works, restart nginx with
`sudo systemctl restart nginx`.
If you haven't already done so, set up a DNS entry to make sure requests for your domain name end up at the server that's running Spothole. If you haven't already done so, set up a DNS entry to make sure requests for your domain name end up at the server
that's running Spothole.
You should now be able to access the web interface by going to the domain from your browser. You should now be able to access the web interface by going to the domain from your browser.
Once that's working, [install certbot](https://certbot.eff.org/instructions?ws=nginx&os=snap) onto your server. Run it as root, and when prompted pick your domain name from the list. After a few seconds, it should successfully provision a certificate and modify your nginx config files automatically. You should then be able to access the site via HTTPS. Once that's working, [install certbot](https://certbot.eff.org/instructions?ws=nginx&os=snap) onto your server. Run it
as root, and when prompted pick your domain name from the list. After a few seconds, it should successfully provision a
certificate and modify your nginx config files automatically. You should then be able to access the site via HTTPS.
## Modifying the source code ## Modifying the source code
Spothole is Public Domain licenced, so you can grab the source code and start modifying it for your own needs. Contributions of code back to the main repository are encouraged, but completely optional. Spothole is Public Domain licenced, so you can grab the source code and start modifying it for your own needs.
Contributions of code back to the main repository are encouraged, but completely optional.
### Code structure ### Code structure
@@ -356,24 +421,37 @@ To navigate your way around the source code, this list may help.
* `/` - Main script (`spothole.py`), pip `requirements.txt`, config, README, etc. * `/` - Main script (`spothole.py`), pip `requirements.txt`, config, README, etc.
* `/images` - Image sources * `/images` - Image sources
* `/datafiles` - Local data sources (differentiated from the majority of data files which are loaded from URLs and cached in `/cache`) * `/datafiles` - Local data sources (differentiated from the majority of data files which are loaded from URLs and
* `/cache` - Directory where static-ish data downloaded from the internet is cached to avoid rapid re-requests, and where spot/alert data is cached so that it survives a software restart. Created on first run. cached in `/cache`)
* `/cache` - Directory where static-ish data downloaded from the internet is cached to avoid rapid re-requests, and
where spot/alert data is cached so that it survives a software restart. Created on first run.
### Extending the server ### Extending the server
Spothole is designed to be easily extensible. If you want to write your own spot provider, for example, simply add a module to the `spotproviders` package containing your class. (Currently, in order to be loaded correctly, the module (file) name should be the same as the class name, but lower case.) Spothole is designed to be easily extensible. If you want to write your own spot provider, for example, simply add a
module to the `spotproviders` package containing your class. (Currently, in order to be loaded correctly, the module (
file) name should be the same as the class name, but lower case.)
Your class should extend "SpotProvider"; if it operates by polling an HTTP Server on a timer, it can instead extend "HTTPSpotProvider" where some of the work is done for you. Your class should extend "SpotProvider"; if it operates by polling an HTTP Server on a timer, it can instead extend "
HTTPSpotProvider" where some of the work is done for you.
The class will need to implement a constructor that takes in the `provider_config` and provides it to the superclass constructor, while also taking any other config parameters it needs. The class will need to implement a constructor that takes in the `provider_config` and provides it to the superclass
constructor, while also taking any other config parameters it needs.
If you're extending the base `SpotProvider` class, you will need to implement `start()` and `stop()` methods that start and stop a separate thread which handles the provider's processing needs. The thread should call `submit()` or `submit_batch()` when it has one or more spots to report. If you're extending the base `SpotProvider` class, you will need to implement `start()` and `stop()` methods that start
and stop a separate thread which handles the provider's processing needs. The thread should call `submit()` or
`submit_batch()` when it has one or more spots to report.
If you're extending the `HTTPSpotProvider` class, you will need to provide a URI to query and an interval to the superclass constructor. You'll then need to implement the `http_response_to_spots()` method which is called when new data is retrieved. Your implementation should then call `submit()` or `submit_batch()` when it has one or more spots to report. If you're extending the `HTTPSpotProvider` class, you will need to provide a URI to query and an interval to the
superclass constructor. You'll then need to implement the `http_response_to_spots()` method which is called when new
data is retrieved. Your implementation should then call `submit()` or `submit_batch()` when it has one or more spots to
report.
When constructing spots, use the comments in the Spot class and the existing implementations as an example. All parameters are optional, but you will at least want to provide a `time` (which must be timezone-aware) and a `dx_call`. When constructing spots, use the comments in the Spot class and the existing implementations as an example. All
parameters are optional, but you will at least want to provide a `time` (which must be timezone-aware) and a `dx_call`.
Finally, simply add the appropriate config to the `spot_providers` section of `config.yml`, and your provider should be instantiated on startup. Finally, simply add the appropriate config to the `spot_providers` section of `config.yml`, and your provider should be
instantiated on startup.
The same approach as above is also used for alert providers. The same approach as above is also used for alert providers.
@@ -381,7 +459,8 @@ The same approach as above is also used for alert providers.
As well as being my work, I have also gratefully received feature patches from Steven, M1SDH. As well as being my work, I have also gratefully received feature patches from Steven, M1SDH.
The project contains GeoJSON files for CQ and ITU zones, in the `/datafiles/` directory. These are MIT-licenced and, to my knowledge, created by HA8TKS for his CQ and ITU zone layers for Leaflet. The project contains GeoJSON files for CQ and ITU zones, in the `/datafiles/` directory. These are MIT-licenced and, to
my knowledge, created by HA8TKS for his CQ and ITU zone layers for Leaflet.
The project contains a set of flag icons generated using the "Noto Color Emoji" font on a Debian system, in the `/webassets/img/flags/` directory. The project contains a set of flag icons generated using the "Noto Color Emoji" font on a Debian system, in the `/webassets/img/flags/` directory.
@@ -391,8 +470,16 @@ The software uses a number of Python libraries as listed in `requirements.txt`,
A number of third-party libraries are self-hosted in the `/webassets/vendor/` directory. These files are subject to their own licences and are not covered by the overall licence declared in the `LICENSE` file. A number of third-party libraries are self-hosted in the `/webassets/vendor/` directory. These files are subject to their own licences and are not covered by the overall licence declared in the `LICENSE` file.
Particular thanks go to country-files.com for providing country lookup data for amateur radio, to K0SWE for [this JSON-formatted DXCC data](https://github.com/k0swe/dxcc-json/), and to the developers of `pyhamtools` for making it easy to use country-files.com data as well as QRZ.com and Clublog lookup. A number of third-party libraries are self-hosted in the `/webassets/vendor/` directory. These files are subject to
their own licences and are not covered by the overall licence declared in the `LICENSE` file.
Amateur radio clusters, outdoor programmes, propagation data providers etc. are almost all volunteer-run services that make no or little profit, and are done for the love of amateur radio. Services like Spothole, which build on top of them, are truly standing on the shoulders of giants. None of this would have been possible without the hard work and dedication of many other people within the amaetur radio community. Particular thanks go to country-files.com for providing country lookup data for amateur radio, to K0SWE
for [this JSON-formatted DXCC data](https://github.com/k0swe/dxcc-json/), and to the developers of `pyhamtools` for
making it easy to use country-files.com data as well as QRZ.com and Clublog lookup.
Amateur radio clusters, outdoor programmes, propagation data providers etc. are almost all volunteer-run services that
make no or little profit, and are done for the love of amateur radio. Services like Spothole, which build on top of
them, are truly standing on the shoulders of giants. None of this would have been possible without the hard work and
dedication of many other people within the amaetur radio community.
The project's name was suggested by Harm, DK4HAA. Thanks! The project's name was suggested by Harm, DK4HAA. Thanks!

View File

@@ -21,19 +21,30 @@ class BOTA(HTTPAlertProvider):
new_alerts = [] new_alerts = []
# Find the table of upcoming alerts # Find the table of upcoming alerts
bs = BeautifulSoup(http_response.content.decode(), features="lxml") bs = BeautifulSoup(http_response.content.decode(), features="lxml")
if not bs.body:
return new_alerts
div = bs.body.find('div', attrs={'class': 'view-activations-public'}) div = bs.body.find('div', attrs={'class': 'view-activations-public'})
if div: if div:
table = div.find('table', attrs={'class': 'views-table'}) table = div.find('table', attrs={'class': 'views-table'})
if table: if table:
tbody = table.find('tbody') tbody = table.find('tbody')
if not tbody:
return new_alerts
for row in tbody.find_all('tr'): for row in tbody.find_all('tr'):
cells = row.find_all('td') cells = row.find_all('td')
first_cell_text = str(cells[0].find('a').contents[0]).strip() first_cell_anchor = cells[0].find('a') if len(cells) > 0 else None
second_cell_anchor = cells[1].find('a') if len(cells) > 1 else None
if not first_cell_anchor or not second_cell_anchor:
continue
first_cell_text = first_cell_anchor.get_text().strip()
ref_name = first_cell_text.split(" by ")[0] ref_name = first_cell_text.split(" by ")[0]
dx_call = str(cells[1].find('a').contents[0]).strip().upper() dx_call = second_cell_anchor.get_text().strip().upper()
# Get the date, dealing with the fact we get no year so have to figure out if it's last year or next year # Get the date, dealing with the fact we get no year so have to figure out if it's last year or next year
date_text = str(cells[2].find('span').contents[0]).strip() date_span = cells[2].find('span') if len(cells) > 2 else None
if not date_span:
continue
date_text = date_span.get_text().strip()
date_time = datetime.strptime(date_text, "%d %b - %H:%M UTC").replace(tzinfo=pytz.UTC) date_time = datetime.strptime(date_text, "%d %b - %H:%M UTC").replace(tzinfo=pytz.UTC)
date_time = date_time.replace(year=datetime.now(pytz.UTC).year) date_time = date_time.replace(year=datetime.now(pytz.UTC).year)
# If this was more than a day ago, activation is actually next year # If this was more than a day ago, activation is actually next year

View File

@@ -1,8 +1,10 @@
import re import re
from datetime import datetime from datetime import datetime
from typing import cast
import pytz import pytz
from rss_parser import Parser from rss_parser import Parser
from rss_parser.models.rss import RSS
from alertproviders.http_alert_provider import HTTPAlertProvider from alertproviders.http_alert_provider import HTTPAlertProvider
from data.alert import Alert from data.alert import Alert
@@ -20,7 +22,7 @@ class NG3K(HTTPAlertProvider):
def _http_response_to_alerts(self, http_response): def _http_response_to_alerts(self, http_response):
new_alerts = [] new_alerts = []
rss = Parser.parse(http_response.content.decode()) rss = cast(RSS, Parser.parse(http_response.content.decode()))
# Iterate through source data # Iterate through source data
for source_alert in rss.channel.items: for source_alert in rss.channel.items:
# Deal with "the format"... # Deal with "the format"...

View File

@@ -37,6 +37,6 @@ class POTA(HTTPAlertProvider):
# Add to our list, but exclude any old spots that POTA can sometimes give us where even the end time is # Add to our list, but exclude any old spots that POTA can sometimes give us where even the end time is
# in the past. Don't worry about de-duping, removing old alerts etc. at this point; other code will do # in the past. Don't worry about de-duping, removing old alerts etc. at this point; other code will do
# that for us. # that for us.
if alert.end_time > datetime.now(pytz.UTC).timestamp(): if alert.end_time and alert.end_time > datetime.now(pytz.UTC).timestamp():
new_alerts.append(alert) new_alerts.append(alert)
return new_alerts return new_alerts

View File

@@ -1,7 +1,9 @@
from datetime import datetime from datetime import datetime
from typing import cast
import pytz import pytz
from rss_parser import Parser as RSSParser from rss_parser import Parser as RSSParser
from rss_parser.models.rss import RSS
from alertproviders.http_alert_provider import HTTPAlertProvider from alertproviders.http_alert_provider import HTTPAlertProvider
from data.alert import Alert from data.alert import Alert
@@ -20,7 +22,7 @@ class WOTA(HTTPAlertProvider):
def _http_response_to_alerts(self, http_response): def _http_response_to_alerts(self, http_response):
new_alerts = [] new_alerts = []
rss = RSSParser.parse(http_response.content.decode()) rss = cast(RSS, RSSParser.parse(http_response.content.decode()))
# Iterate through source data # Iterate through source data
for source_alert in rss.channel.items: for source_alert in rss.channel.items:
@@ -35,9 +37,9 @@ class WOTA(HTTPAlertProvider):
ref_name = None ref_name = None
if len(title_split) > 1: if len(title_split) > 1:
ref_split = title_split[1].split(" - ") ref_split = title_split[1].split(" - ")
ref = ref_split[0] ref = str(ref_split[0])
if len(ref_split) > 1: if len(ref_split) > 1:
ref_name = ref_split[1] ref_name = str(ref_split[1])
# Pick apart the description # Pick apart the description
desc_split = source_alert.description.split(". ") desc_split = source_alert.description.split(". ")

View File

@@ -24,60 +24,46 @@ base-url: "http://localhost:8080"
# for CW/RTTY and 7001 for FT8, so if you want both, you need two entries, as shown below. # for CW/RTTY and 7001 for FT8, so if you want both, you need two entries, as shown below.
# Feel free to write your own provider classes! There are details in the README. # Feel free to write your own provider classes! There are details in the README.
spot-providers: spot-providers:
- - class: "POTA"
class: "POTA"
name: "POTA" name: "POTA"
enabled: true enabled: true
- - class: "SOTA"
class: "SOTA"
name: "SOTA" name: "SOTA"
enabled: true enabled: true
- - class: "WWFF"
class: "WWFF"
name: "WWFF" name: "WWFF"
enabled: true enabled: true
- - class: "WWBOTA"
class: "WWBOTA"
name: "WWBOTA" name: "WWBOTA"
enabled: true enabled: true
- - class: "GMA"
class: "GMA"
name: "GMA" name: "GMA"
enabled: true enabled: true
- - class: "HEMA"
class: "HEMA"
name: "HEMA" name: "HEMA"
enabled: true enabled: true
- - class: "ParksNPeaks"
class: "ParksNPeaks"
name: "ParksNPeaks" name: "ParksNPeaks"
enabled: true enabled: true
- - class: "ZLOTA"
class: "ZLOTA"
name: "ZLOTA" name: "ZLOTA"
enabled: true enabled: true
- - class: "WOTA"
class: "WOTA"
name: "WOTA" name: "WOTA"
enabled: true enabled: true
- - class: "LLOTA"
class: "LLOTA"
name: "LLOTA" name: "LLOTA"
enabled: true enabled: true
- - class: "WWTOTA"
class: "WWTOTA"
name: "WWTOTA" name: "WWTOTA"
enabled: true enabled: true
- - class: "Tiles"
class: "Tiles"
name: "Tiles" name: "Tiles"
enabled: true enabled: true
- - class: "APRSIS"
class: "APRSIS"
name: "APRS-IS" name: "APRS-IS"
enabled: false enabled: false
- - class: "DXCluster"
class: "DXCluster"
name: "HRD Cluster" name: "HRD Cluster"
enabled: true enabled: true
host: "hrd.wa9pie.net" host: "hrd.wa9pie.net"
@@ -92,8 +78,7 @@ spot-providers:
# sure you aren't also separately connecting to RBN directly, otherwise you may get duplicate spots.) Note that not # sure you aren't also separately connecting to RBN directly, otherwise you may get duplicate spots.) Note that not
# all clusters sent RBN spots anyway. # all clusters sent RBN spots anyway.
allow_rbn_spots: false allow_rbn_spots: false
- - class: "DXCluster"
class: "DXCluster"
name: "W3LPL Cluster" name: "W3LPL Cluster"
enabled: false enabled: false
host: "w3lpl.net" host: "w3lpl.net"
@@ -108,8 +93,7 @@ spot-providers:
# sure you aren't also separately connecting to RBN directly, otherwise you may get duplicate spots.) Note that not # sure you aren't also separately connecting to RBN directly, otherwise you may get duplicate spots.) Note that not
# all clusters sent RBN spots anyway. # all clusters sent RBN spots anyway.
allow_rbn_spots: false allow_rbn_spots: false
- - class: "RBN"
class: "RBN"
name: "RBN CW/RTTY" name: "RBN CW/RTTY"
enabled: false enabled: false
port: 7000 port: 7000
@@ -118,19 +102,16 @@ spot-providers:
# received by Spothole but not shown on the web UI unless the user explicitly turns it on. For that behaviour, # received by Spothole but not shown on the web UI unless the user explicitly turns it on. For that behaviour,
# set enabled to true, but enabled-by-default-in-web-ui to false. # set enabled to true, but enabled-by-default-in-web-ui to false.
enabled-by-default-in-web-ui: false enabled-by-default-in-web-ui: false
- - class: "RBN"
class: "RBN"
name: "RBN FT8" name: "RBN FT8"
enabled: false enabled: false
port: 7001 port: 7001
enabled-by-default-in-web-ui: false enabled-by-default-in-web-ui: false
- - class: "UKPacketNet"
class: "UKPacketNet"
name: "UK Packet Radio Net" name: "UK Packet Radio Net"
enabled: false enabled: false
enabled-by-default-in-web-ui: false enabled-by-default-in-web-ui: false
- - class: "XOTA"
class: "XOTA"
name: "39C3 TOTA" name: "39C3 TOTA"
enabled: false enabled: false
url: "wss://39c3.totawatch.de/api/spot/live" url: "wss://39c3.totawatch.de/api/spot/live"
@@ -139,8 +120,7 @@ spot-providers:
# programmes and so different URLs provide different programmes. # programmes and so different URLs provide different programmes.
sig: "TOTA" sig: "TOTA"
locations-csv: "datafiles/39c3-tota.csv" locations-csv: "datafiles/39c3-tota.csv"
- - class: "XOTA"
class: "XOTA"
name: "EH23 TOTA" name: "EH23 TOTA"
enabled: true enabled: true
url: "wss://eh23.totawatch.de/api/spot/live" url: "wss://eh23.totawatch.de/api/spot/live"
@@ -150,32 +130,25 @@ spot-providers:
# Alert providers to use. Same setup as the spot providers list above. # Alert providers to use. Same setup as the spot providers list above.
alert-providers: alert-providers:
- - class: "POTA"
class: "POTA"
name: "POTA" name: "POTA"
enabled: true enabled: true
- - class: "SOTA"
class: "SOTA"
name: "SOTA" name: "SOTA"
enabled: true enabled: true
- - class: "WWFF"
class: "WWFF"
name: "WWFF" name: "WWFF"
enabled: true enabled: true
- - class: "ParksNPeaks"
class: "ParksNPeaks"
name: "ParksNPeaks" name: "ParksNPeaks"
enabled: true enabled: true
- - class: "WOTA"
class: "WOTA"
name: "WOTA" name: "WOTA"
enabled: true enabled: true
- - class: "BOTA"
class: "BOTA"
name: "BOTA" name: "BOTA"
enabled: true enabled: true
- - class: "NG3K"
class: "NG3K"
name: "NG3K" name: "NG3K"
enabled: true enabled: true
@@ -183,20 +156,16 @@ alert-providers:
# Solar condition providers to use. These poll external APIs for solar propagation data (SFI, A/K indices, band # Solar condition providers to use. These poll external APIs for solar propagation data (SFI, A/K indices, band
# conditions, etc.) and make it available via the /api/v1/solar endpoint. # conditions, etc.) and make it available via the /api/v1/solar endpoint.
solar-condition-providers: solar-condition-providers:
- - class: "HamQSL"
class: "HamQSL"
name: "HamQSL" name: "HamQSL"
enabled: true enabled: true
- - class: "NOAA3dayForecast"
class: "NOAA3dayForecast"
name: "NOAA 3-day Forecast" name: "NOAA 3-day Forecast"
enabled: true enabled: true
- - class: "GIROIonosonde"
class: "GIROIonosonde"
name: "GIRO Ionosonde Data" name: "GIRO Ionosonde Data"
enabled: true enabled: true
- - class: "KC2GProp"
class: "KC2GProp"
name: "KC2G Propagation Data" name: "KC2G Propagation Data"
enabled: true enabled: true
@@ -228,11 +197,11 @@ recaptcha-secret-key: ""
# Options for the web UI. # Options for the web UI.
web-ui-options: web-ui-options:
spot-count: [10, 25, 50, 100] spot-count: [ 10, 25, 50, 100 ]
spot-count-default: 50 spot-count-default: 50
max-spot-age: [5, 10, 30, 60] max-spot-age: [ 5, 10, 30, 60 ]
max-spot-age-default: 30 max-spot-age-default: 30
alert-count: [25, 50, 100, 200, 500] alert-count: [ 25, 50, 100, 200, 500 ]
alert-count-default: 100 alert-count-default: 100
# Default UI colour scheme. Supported values are "light", "dark" and "auto" (i.e. use the browser/OS colour scheme). # Default UI colour scheme. Supported values are "light", "dark" and "auto" (i.e. use the browser/OS colour scheme).
# Users can still override this in the UI to their own preference. # Users can still override this in the UI to their own preference.

View File

@@ -104,7 +104,7 @@ class LookupHelper:
self._hamqth_callsign_data_cache = Cache('cache/hamqth_callsign_lookup_cache') self._hamqth_callsign_data_cache = Cache('cache/hamqth_callsign_lookup_cache')
self._clublog_api_key = config["clublog-api-key"] self._clublog_api_key = str(config["clublog-api-key"])
self._clublog_cty_xml_cache = CachedSession("cache/clublog_cty_xml_cache", expire_after=timedelta(days=10)) self._clublog_cty_xml_cache = CachedSession("cache/clublog_cty_xml_cache", expire_after=timedelta(days=10))
self._clublog_api_available = self._clublog_api_key != "" self._clublog_api_available = self._clublog_api_key != ""
self._clublog_xml_download_location = "cache/cty.xml" self._clublog_xml_download_location = "cache/cty.xml"
@@ -184,6 +184,7 @@ class LookupHelper:
open(self._clublog_xml_download_location + ".gz", 'wb').write(response.content) open(self._clublog_xml_download_location + ".gz", 'wb').write(response.content)
with gzip.open(self._clublog_xml_download_location + ".gz", "rb") as uncompressed: with gzip.open(self._clublog_xml_download_location + ".gz", "rb") as uncompressed:
file_content = uncompressed.read() file_content = uncompressed.read()
assert isinstance(file_content, bytes)
logging.info("Caching Clublog cty.xml...") logging.info("Caching Clublog cty.xml...")
with open(self._clublog_xml_download_location, "wb") as f: with open(self._clublog_xml_download_location, "wb") as f:
f.write(file_content) f.write(file_content)
@@ -384,12 +385,12 @@ class LookupHelper:
data = self._get_qrz_data_for_callsign(call, credentials) data = self._get_qrz_data_for_callsign(call, credentials)
if data and "latitude" in data and "longitude" in data and ( if data and "latitude" in data and "longitude" in data and (
float(data["latitude"]) != 0 or float(data["longitude"]) != 0) and -89.9 < float( float(data["latitude"]) != 0 or float(data["longitude"]) != 0) and -89.9 < float(
data["latitude"]) < 89.9: data["latitude"]) < 89.9:
return [float(data["latitude"]), float(data["longitude"])] return [float(data["latitude"]), float(data["longitude"])]
data = self._get_hamqth_data_for_callsign(call, credentials) data = self._get_hamqth_data_for_callsign(call, credentials)
if data and "latitude" in data and "longitude" in data and ( if data and "latitude" in data and "longitude" in data and (
float(data["latitude"]) != 0 or float(data["longitude"]) != 0) and -89.9 < float( float(data["latitude"]) != 0 or float(data["longitude"]) != 0) and -89.9 < float(
data["latitude"]) < 89.9: data["latitude"]) < 89.9:
return [float(data["latitude"]), float(data["longitude"])] return [float(data["latitude"]), float(data["longitude"])]
else: else:
return None return None
@@ -446,15 +447,16 @@ class LookupHelper:
def infer_grid_from_callsign_dxcc(self, call): def infer_grid_from_callsign_dxcc(self, call):
"""Infer a grid locator from a callsign (using DXCC, probably very inaccurate)""" """Infer a grid locator from a callsign (using DXCC, probably very inaccurate)"""
latlon = self.infer_latlon_from_callsign_dxcc(call) latlon = self.infer_latlon_from_callsign_dxcc(call) or []
grid = None grid = None
try: if latlon:
grid = latlong_to_locator(latlon[0], latlon[1], 8) try:
except: grid = latlong_to_locator(latlon[0], latlon[1], 8)
logging.debug("Invalid lat/lon received for DXCC") except:
logging.debug("Invalid lat/lon received for DXCC")
return grid return grid
def _get_qrz_data_for_callsign(self, call, credentials): def _get_qrz_data_for_callsign(self, call, credentials) -> dict | None:
"""Utility method to get QRZ.com data from cache if possible, if not get it from the API and cache it. """Utility method to get QRZ.com data from cache if possible, if not get it from the API and cache it.
Returns None immediately if no credentials are provided.""" Returns None immediately if no credentials are provided."""
@@ -475,7 +477,7 @@ class LookupHelper:
login_data = xmltodict.parse(login_response) login_data = xmltodict.parse(login_response)
session = login_data.get("QRZDatabase", {}).get("Session", {}) session = login_data.get("QRZDatabase", {}).get("Session", {})
if "Key" in session: if "Key" in session:
session_key = session["Key"] session_key = str(session["Key"])
else: else:
logging.warning("QRZ.com login details incorrect, failed to look up with QRZ.") logging.warning("QRZ.com login details incorrect, failed to look up with QRZ.")
return None return None
@@ -512,7 +514,7 @@ class LookupHelper:
self._qrz_callsign_data_cache.add(call, None, expire=604800) # 1 week in seconds self._qrz_callsign_data_cache.add(call, None, expire=604800) # 1 week in seconds
return None return None
def _get_hamqth_data_for_callsign(self, call, credentials): def _get_hamqth_data_for_callsign(self, call, credentials) -> dict | None:
"""Utility method to get HamQTH data from cache if possible, if not get it from the API and cache it. """Utility method to get HamQTH data from cache if possible, if not get it from the API and cache it.
Returns None immediately if no credentials are provided.""" Returns None immediately if no credentials are provided."""
@@ -531,7 +533,7 @@ class LookupHelper:
"&p=" + urllib.parse.quote_plus(credentials.hamqth_password), headers=HTTP_HEADERS).content "&p=" + urllib.parse.quote_plus(credentials.hamqth_password), headers=HTTP_HEADERS).content
dict_data = xmltodict.parse(session_data) dict_data = xmltodict.parse(session_data)
if "session_id" in dict_data["HamQTH"]["session"]: if "session_id" in dict_data["HamQTH"]["session"]:
session_id = dict_data["HamQTH"]["session"]["session_id"] session_id = str(dict_data["HamQTH"]["session"]["session_id"])
else: else:
logging.warning("HamQTH login details incorrect, failed to look up with HamQTH.") logging.warning("HamQTH login details incorrect, failed to look up with HamQTH.")
return None return None
@@ -566,7 +568,7 @@ class LookupHelper:
self._hamqth_callsign_data_cache.add(call, None, expire=604800) # 1 week in seconds self._hamqth_callsign_data_cache.add(call, None, expire=604800) # 1 week in seconds
return None return None
def _get_clublog_api_data_for_callsign(self, call): def _get_clublog_api_data_for_callsign(self, call) -> dict | None:
"""Utility method to get Clublog API data from cache if possible, if not get it from the API and cache it""" """Utility method to get Clublog API data from cache if possible, if not get it from the API and cache it"""
# Fetch from cache if we can, otherwise fetch from the API and cache it # Fetch from cache if we can, otherwise fetch from the API and cache it
@@ -594,7 +596,7 @@ class LookupHelper:
else: else:
return None return None
def _get_clublog_xml_data_for_callsign(self, call): def _get_clublog_xml_data_for_callsign(self, call) -> dict | None:
"""Utility method to get Clublog XML data from file""" """Utility method to get Clublog XML data from file"""
if self._clublog_xml_available: if self._clublog_xml_available:
@@ -608,7 +610,7 @@ class LookupHelper:
else: else:
return None return None
def _get_dxcc_data_for_callsign(self, call): def _get_dxcc_data_for_callsign(self, call) -> dict | None:
"""Utility method to get generic DXCC data from our lookup table, if we can find it""" """Utility method to get generic DXCC data from our lookup table, if we can find it"""
for entry in self._dxcc_data.values(): for entry in self._dxcc_data.values():
@@ -627,6 +629,7 @@ class LookupHelper:
# Singleton object # Singleton object
lookup_helper = LookupHelper() lookup_helper = LookupHelper()
def infer_mode_from_comment(comment): def infer_mode_from_comment(comment):
"""Infer a mode from the comment""" """Infer a mode from the comment"""

View File

@@ -25,13 +25,13 @@ def populate_sig_ref_info(sig_ref):
if sig_ref.sig is None or sig_ref.id is None: if sig_ref.sig is None or sig_ref.id is None:
logging.warning("Failed to look up sig_ref info, sig or id were not set.") logging.warning("Failed to look up sig_ref info, sig or id were not set.")
sig = sig_ref.sig sig = sig_ref.sig or ""
ref_id = sig_ref.id ref_id = sig_ref.id
try: try:
if sig.upper() == "POTA": if sig.upper() == "POTA":
data = SEMI_STATIC_URL_DATA_CACHE.get("https://api.pota.app/park/" + ref_id, headers=HTTP_HEADERS).json() data = SEMI_STATIC_URL_DATA_CACHE.get("https://api.pota.app/park/" + ref_id, headers=HTTP_HEADERS).json()
if data: if data:
fullname = data["name"] if "name" in data else None fullname = str(data["name"]) if "name" in data else None
if fullname and "parktypeDesc" in data and data["parktypeDesc"] != "": if fullname and "parktypeDesc" in data and data["parktypeDesc"] != "":
fullname = fullname + " " + data["parktypeDesc"] fullname = fullname + " " + data["parktypeDesc"]
sig_ref.name = fullname sig_ref.name = fullname
@@ -81,7 +81,8 @@ def populate_sig_ref_info(sig_ref):
elif sig.upper() == "SIOTA": elif sig.upper() == "SIOTA":
siota_csv_data = SEMI_STATIC_URL_DATA_CACHE.get("https://www.silosontheair.com/data/silos.csv", siota_csv_data = SEMI_STATIC_URL_DATA_CACHE.get("https://www.silosontheair.com/data/silos.csv",
headers=HTTP_HEADERS) headers=HTTP_HEADERS)
siota_index = {row["SILO_CODE"]: row for row in csv.DictReader(siota_csv_data.content.decode().splitlines())} siota_index = {row["SILO_CODE"]: row for row in
csv.DictReader(siota_csv_data.content.decode().splitlines())}
row = siota_index.get(ref_id) row = siota_index.get(ref_id)
if row: if row:
sig_ref.name = row["NAME"] if "NAME" in row else None sig_ref.name = row["NAME"] if "NAME" in row else None
@@ -129,9 +130,9 @@ def populate_sig_ref_info(sig_ref):
if data: if data:
for ref in data: for ref in data:
if ref["reference_code"] == ref_id: if ref["reference_code"] == ref_id:
sig_ref.name = ref["name"] sig_ref.name = str(ref["name"])
sig_ref.url = "https://llota.app/list/ref/" + ref_id sig_ref.url = "https://llota.app/list/ref/" + ref_id
sig_ref.grid = ref["grid_locator"] sig_ref.grid = str(ref["grid_locator"])
ll = locator_to_latlong(sig_ref.grid) ll = locator_to_latlong(sig_ref.grid)
sig_ref.latitude = ll[0] sig_ref.latitude = ll[0]
sig_ref.longitude = ll[1] sig_ref.longitude = ll[1]
@@ -139,7 +140,7 @@ def populate_sig_ref_info(sig_ref):
elif sig.upper() == "WWTOTA": elif sig.upper() == "WWTOTA":
if not sig_ref.name: if not sig_ref.name:
sig_ref.name = sig_ref.id sig_ref.name = sig_ref.id
sig_ref.url = "https://wwtota.com/seznam/karta_rozhledny.php?ref=" + sig_ref.name sig_ref.url = "https://wwtota.com/seznam/karta_rozhledny.php?ref=" + str(sig_ref.name)
elif sig.upper() == "TILES": elif sig.upper() == "TILES":
# Tiles on the Air just uses Maidenhead 6-digit squares, so ID, Name and Grid are all the same # Tiles on the Air just uses Maidenhead 6-digit squares, so ID, Name and Grid are all the same
if not sig_ref.name: if not sig_ref.name:
@@ -147,7 +148,7 @@ def populate_sig_ref_info(sig_ref):
if not sig_ref.grid: if not sig_ref.grid:
sig_ref.grid = sig_ref.id sig_ref.grid = sig_ref.id
if sig_ref.grid and not sig_ref.latitude: if sig_ref.grid and not sig_ref.latitude:
ll = locator_to_latlong(sig_ref.grid) ll = locator_to_latlong(str(sig_ref.grid))
sig_ref.latitude = ll[0] sig_ref.latitude = ll[0]
sig_ref.longitude = ll[1] sig_ref.longitude = ll[1]
elif sig.upper() == "WAB" or sig.upper() == "WAI": elif sig.upper() == "WAB" or sig.upper() == "WAI":

View File

@@ -89,7 +89,8 @@ class StatusReporter:
"last_page_access_time"].replace( "last_page_access_time"].replace(
tzinfo=pytz.UTC).timestamp() if self._web_server.web_server_metrics[ tzinfo=pytz.UTC).timestamp() if self._web_server.web_server_metrics[
"last_page_access_time"] else 0, "last_page_access_time"] else 0,
"page_access_count": self._web_server.web_server_metrics["page_access_counter"]} "page_access_count": self._web_server.web_server_metrics[
"page_access_counter"]}
# Update Prometheus metrics # Update Prometheus metrics
memory_use_gauge.set(psutil.Process(os.getpid()).memory_info().rss) memory_use_gauge.set(psutil.Process(os.getpid()).memory_info().rss)

View File

@@ -15,51 +15,51 @@ class Alert:
"""Data class that defines an alert.""" """Data class that defines an alert."""
# Unique identifier for the alert # Unique identifier for the alert
id: str = None id: str | None = None
# Callsigns of the operators that has been alerted # Callsigns of the operators that has been alerted
dx_calls: list = None dx_calls: list | None = None
# Names of the operators that has been alerted # Names of the operators that has been alerted
dx_names: list = None dx_names: list | None = None
# Country of the DX operator # Country of the DX operator
dx_country: str = None dx_country: str | None = None
# Country flag of the DX operator # Country flag of the DX operator
dx_flag: str = None dx_flag: str | None = None
# Continent of the DX operator # Continent of the DX operator
dx_continent: str = None dx_continent: str | None = None
# DXCC ID of the DX operator # DXCC ID of the DX operator
dx_dxcc_id: int = None dx_dxcc_id: int | None = None
# CQ zone of the DX operator # CQ zone of the DX operator
dx_cq_zone: int = None dx_cq_zone: int | None = None
# ITU zone of the DX operator # ITU zone of the DX operator
dx_itu_zone: int = None dx_itu_zone: int | None = None
# Intended frequencies & modes of operation. Essentially just a different kind of comment field. # Intended frequencies & modes of operation. Essentially just a different kind of comment field.
freqs_modes: str = None freqs_modes: str | None = None
# Start time of the activation, UTC seconds since UNIX epoch # Start time of the activation, UTC seconds since UNIX epoch
start_time: float = None start_time: float | None = None
# Start time of the activation of the alert, ISO 8601 # Start time of the activation of the alert, ISO 8601
start_time_iso: str = None start_time_iso: str | None = None
# End time of the activation, UTC seconds since UNIX epoch. Optional # End time of the activation, UTC seconds since UNIX epoch. Optional
end_time: float = None end_time: float | None = None
# End time of the activation of the alert, ISO 8601 # End time of the activation of the alert, ISO 8601
end_time_iso: str = None end_time_iso: str | None = None
# Time that this software received the alert, UTC seconds since UNIX epoch. This is used with the "since_received" # Time that this software received the alert, UTC seconds since UNIX epoch. This is used with the "since_received"
# call to our API to receive all data that is new to us, even if by a quirk of the API it might be older than the # call to our API to receive all data that is new to us, even if by a quirk of the API it might be older than the
# list time the client polled the API. # list time the client polled the API.
received_time: float = None received_time: float | None = None
# Time that this software received the alert, ISO 8601 # Time that this software received the alert, ISO 8601
received_time_iso: str = None received_time_iso: str | None = None
# Comment made by the alerter, if any # Comment made by the alerter, if any
comment: str = None comment: str | None = None
# Special Interest Group (SIG), e.g. outdoor activity programme such as POTA # Special Interest Group (SIG), e.g. outdoor activity programme such as POTA
sig: str = None sig: str | None = None
# SIG references. We allow multiple here for e.g. n-fer activations, unlike ADIF SIG_INFO # SIG references. We allow multiple here for e.g. n-fer activations, unlike ADIF SIG_INFO
sig_refs: list = None sig_refs: list | None = None
# Whether this alert is for a DXpedition, as opposed to e.g. an xOTA programme. # Whether this alert is for a DXpedition, as opposed to e.g. an xOTA programme.
is_dxpedition: bool = False is_dxpedition: bool = False
# Where we got the alert from, e.g. "POTA", "SOTA"... # Where we got the alert from, e.g. "POTA", "SOTA"...
source: str = None source: str | None = None
# The ID the source gave it, if any. # The ID the source gave it, if any.
source_id: str = None source_id: str | None = None
def infer_missing(self, credentials=None): def infer_missing(self, credentials=None):
"""Infer missing parameters where possible""" """Infer missing parameters where possible"""

View File

@@ -6,10 +6,10 @@ class LookupCredentials:
"""Per-request credentials for QRZ.com and HamQTH online callsign lookups.""" """Per-request credentials for QRZ.com and HamQTH online callsign lookups."""
qrz_username: str = "" qrz_username: str = ""
qrz_password: str = "" qrz_password: str = ""
qrz_session_key: str = "" # alternative to username/password qrz_session_key: str = "" # alternative to username/password
hamqth_username: str = "" hamqth_username: str = ""
hamqth_password: str = "" hamqth_password: str = ""
hamqth_session_id: str = "" # alternative to username/password hamqth_session_id: str = "" # alternative to username/password
def extract_credentials(query_params): def extract_credentials(query_params):

View File

@@ -11,14 +11,14 @@ class SIGRef:
# SIG that this reference is in, e.g. "POTA". # SIG that this reference is in, e.g. "POTA".
sig: str sig: str
# Name of the reference, e.g. "Null Country Park", if known. # Name of the reference, e.g. "Null Country Park", if known.
name: str = None name: str | None = None
# URL to look up more information about the reference, if known. # URL to look up more information about the reference, if known.
url: str = None url: str | None = None
# Latitude of the reference, if known. # Latitude of the reference, if known.
latitude: float = None latitude: float | None = None
# Longitude of the reference, if known. # Longitude of the reference, if known.
longitude: float = None longitude: float | None = None
# Maidenhead grid reference of the reference, if known. # Maidenhead grid reference of the reference, if known.
grid: str = None grid: str | None = None
# Activation score. SOTA only # Activation score. SOTA only
activation_score: int = None activation_score: int | None = None

View File

@@ -110,11 +110,11 @@ class HFBandCondition:
"""Data class representing HF propagation conditions for certain bands and time of day.""" """Data class representing HF propagation conditions for certain bands and time of day."""
# Band name, e.g. "80m-40m", "20m-17m", "10m-6m" # Band name, e.g. "80m-40m", "20m-17m", "10m-6m"
band: str = None band: str | None = None
# Time of day: "day" or "night" # Time of day: "day" or "night"
time: str = None time: str | None = None
# Propagation condition: "Good", "Fair", or "Poor" # Propagation condition: "Good", "Fair", or "Poor"
condition: str = None condition: str | None = None
@dataclass @dataclass
@@ -122,66 +122,66 @@ class SolarConditions:
"""Data class representing current solar and propagation conditions.""" """Data class representing current solar and propagation conditions."""
# Time the data was last updated at the source, UTC seconds since UNIX epoch # Time the data was last updated at the source, UTC seconds since UNIX epoch
updated: float = None updated: float | None = None
# Solar Flux Index (SFI) # Solar Flux Index (SFI)
sfi: int = None sfi: int | None = None
# A-index (daily geomagnetic activity) # A-index (daily geomagnetic activity)
a_index: int = None a_index: int | None = None
# K-index (3-hour geomagnetic activity) # K-index (3-hour geomagnetic activity)
k_index: int = None k_index: int | None = None
# X-ray flux class, e.g. "B2.3", "C1.0" # X-ray flux class, e.g. "B2.3", "C1.0"
xray: str = None xray: str | None = None
# Proton flux # Proton flux
proton_flux: int = None proton_flux: int | None = None
# Electron flux # Electron flux
electron_flux: int = None electron_flux: int | None = None
# Aurora activity level # Aurora activity level
aurora: int = None aurora: int | None = None
# Latitude in degrees of the aurora boundary # Latitude in degrees of the aurora boundary
aurora_latitude: float = None aurora_latitude: float | None = None
# Sunspot count # Sunspot count
sunspots: int = None sunspots: int | None = None
# Solar wind speed in km/s # Solar wind speed in km/s
solar_wind: float = None solar_wind: float | None = None
# Interplanetary magnetic field strength in nT # Interplanetary magnetic field strength in nT
magnetic_field: float = None magnetic_field: float | None = None
# Geomagnetic field condition, e.g. "Quiet", "Unsettled", "Active", "Storm" # Geomagnetic field condition, e.g. "Quiet", "Unsettled", "Active", "Storm"
geomag_field: str = None geomag_field: str | None = None
# Geomagnetic background noise level, e.g. "S0", "S1", "S2" # Geomagnetic background noise level, e.g. "S0", "S1", "S2"
geomag_noise: str = None geomag_noise: str | None = None
# HF band propagation conditions, keyed by "{band}-{time}" e.g. "80m-40m-day" # HF band propagation conditions, keyed by "{band}-{time}" e.g. "80m-40m-day"
hf_conditions: dict = None hf_conditions: dict | None = None
# VHF propagation conditions, keyed by condition name # VHF propagation conditions, keyed by condition name
vhf_conditions: dict = None vhf_conditions: dict | None = None
# NOAA Kp index 3-day forecast, keyed by UNIX timestamp of the start of each 3-hour UTC period # NOAA Kp index 3-day forecast, keyed by UNIX timestamp of the start of each 3-hour UTC period
k_index_forecast: dict = None k_index_forecast: dict | None = None
# NOAA Solar Radiation Storm (S1 or greater) probability forecast, keyed by UNIX timestamp of start of day UTC # NOAA Solar Radiation Storm (S1 or greater) probability forecast, keyed by UNIX timestamp of start of day UTC
solar_storm_forecast: dict = None solar_storm_forecast: dict | None = None
# NOAA Radio Blackout (R1-R2) probability forecast, keyed by UNIX timestamp of start of day UTC # NOAA Radio Blackout (R1-R2) probability forecast, keyed by UNIX timestamp of start of day UTC
blackout_forecast_r1r2: dict = None blackout_forecast_r1r2: dict | None = None
# NOAA Radio Blackout (R3 or greater) probability forecast, keyed by UNIX timestamp of start of day UTC # NOAA Radio Blackout (R3 or greater) probability forecast, keyed by UNIX timestamp of start of day UTC
blackout_forecast_r3_or_greater: dict = None blackout_forecast_r3_or_greater: dict | None = None
# Ionosonde measurements, dict keyed by URSI code, values are dicts with keys: ursi, name, fof2, muf, luf, # Ionosonde measurements, dict keyed by URSI code, values are dicts with keys: ursi, name, fof2, muf, luf,
# band_states. Populated by GIROIonosonde or KC2GProp providers. # band_states. Populated by GIROIonosonde or KC2GProp providers.
ionosonde_data: dict = None ionosonde_data: dict | None = None
# Derived values (populated by infer_descriptions()) # Derived values (populated by infer_descriptions())
# HF radio blackout risk description, derived from xray # HF radio blackout risk description, derived from xray
xray_desc: str = None xray_desc: str | None = None
# HF radio blackout scale number (R0-R5), derived from xray # HF radio blackout scale number (R0-R5), derived from xray
radio_blackout_scale: int = None radio_blackout_scale: int | None = None
# Solar radiation storm level description, derived from proton_flux # Solar radiation storm level description, derived from proton_flux
proton_flux_desc: str = None proton_flux_desc: str | None = None
# Solar radiation storm scale number (S0-S5), derived from proton_flux # Solar radiation storm scale number (S0-S5), derived from proton_flux
solar_storm_scale: int = None solar_storm_scale: int | None = None
# Geomagnetic storm level description, derived from k_index # Geomagnetic storm level description, derived from k_index
geomag_storm_desc: str = None geomag_storm_desc: str | None = None
# Geomagnetic storm scale number (G0-G5), derived from k_index # Geomagnetic storm scale number (G0-G5), derived from k_index
geomag_storm_scale: int = None geomag_storm_scale: int | None = None
# Overall HF band conditions summary, derived from sfi # Overall HF band conditions summary, derived from sfi
band_conditions_desc: str = None band_conditions_desc: str | None = None
# Electron flux description, derived from electron_flux # Electron flux description, derived from electron_flux
electron_flux_desc: str = None electron_flux_desc: str | None = None
def infer_descriptions(self): def infer_descriptions(self):
"""Populate derived text description fields from the current numeric/raw field values.""" """Populate derived text description fields from the current numeric/raw field values."""

View File

@@ -23,38 +23,38 @@ class Spot:
"""Data class that defines a spot.""" """Data class that defines a spot."""
# Unique identifier for the spot # Unique identifier for the spot
id: str = None id: str | None = None
# DX (spotted) operator info # DX (spotted) operator info
# Callsign of the operator that has been spotted # Callsign of the operator that has been spotted
dx_call: str = None dx_call: str | None = None
# Name of the operator that has been spotted # Name of the operator that has been spotted
dx_name: str = None dx_name: str | None = None
# QTH of the operator that has been spotted. This could be from any SIG refs or could be from online lookup of their # QTH of the operator that has been spotted. This could be from any SIG refs or could be from online lookup of their
# home QTH. # home QTH.
dx_qth: str = None dx_qth: str | None = None
# Country of the DX operator # Country of the DX operator
dx_country: str = None dx_country: str | None = None
# Country flag of the DX operator # Country flag of the DX operator
dx_flag: str = None dx_flag: str | None = None
# Continent of the DX operator # Continent of the DX operator
dx_continent: str = None dx_continent: str | None = None
# DXCC ID of the DX operator # DXCC ID of the DX operator
dx_dxcc_id: int = None dx_dxcc_id: int | None = None
# CQ zone of the DX operator # CQ zone of the DX operator
dx_cq_zone: int = None dx_cq_zone: int | None = None
# ITU zone of the DX operator # ITU zone of the DX operator
dx_itu_zone: int = None dx_itu_zone: int | None = None
# If this is an APRS/Packet/etc spot, what SSID was the DX operator using? # If this is an APRS/Packet/etc spot, what SSID was the DX operator using?
dx_ssid: str = None dx_ssid: str | None = None
# Maidenhead grid locator for the DX. This could be from a geographical reference e.g. POTA, or just from the # Maidenhead grid locator for the DX. This could be from a geographical reference e.g. POTA, or just from the
# country # country
dx_grid: str = None dx_grid: str | None = None
# Latitude & longitude of the DX, in degrees. This could be from a geographical reference e.g. POTA, or from a QRZ # Latitude & longitude of the DX, in degrees. This could be from a geographical reference e.g. POTA, or from a QRZ
# lookup # lookup
dx_latitude: float = None dx_latitude: float | None = None
dx_longitude: float = None dx_longitude: float | None = None
# DX Location source. Indicates how accurate the location might be. Values: "SPOT", "WAB/WAI GRID", "HOME QTH", # DX Location source. Indicates how accurate the location might be. Values: "SPOT", "WAB/WAI GRID", "HOME QTH",
# "DXCC", "NONE" # "DXCC", "NONE"
dx_location_source: str = "NONE" dx_location_source: str = "NONE"
@@ -66,70 +66,70 @@ class Spot:
# DE (Spotter) info # DE (Spotter) info
# Callsign of the spotter # Callsign of the spotter
de_call: str = None de_call: str | None = None
# Country of the spotter # Country of the spotter
de_country: str = None de_country: str | None = None
# Country flag of the spotter # Country flag of the spotter
de_flag: str = None de_flag: str | None = None
# Continent of the spotter # Continent of the spotter
de_continent: str = None de_continent: str | None = None
# DXCC ID of the spotter # DXCC ID of the spotter
de_dxcc_id: int = None de_dxcc_id: int | None = None
# If this is an APRS/Packet/etc spot, what SSID was the spotter/receiver using? # If this is an APRS/Packet/etc spot, what SSID was the spotter/receiver using?
de_ssid: str = None de_ssid: str | None = None
# Maidenhead grid locator for the spotter. This is not going to be from a xOTA reference so it will likely just be # Maidenhead grid locator for the spotter. This is not going to be from a xOTA reference so it will likely just be
# a QRZ or DXCC lookup. If the spotter is also portable, this is probably wrong, but it's good enough for some # a QRZ or DXCC lookup. If the spotter is also portable, this is probably wrong, but it's good enough for some
# simple mapping. # simple mapping.
de_grid: str = None de_grid: str | None = None
# Latitude & longitude of the DX, in degrees. This is not going to be from a xOTA reference so it will likely just # Latitude & longitude of the DX, in degrees. This is not going to be from a xOTA reference so it will likely just
# be a QRZ or DXCC lookup. If the spotter is also portable, this is probably wrong, but it's good enough for some # be a QRZ or DXCC lookup. If the spotter is also portable, this is probably wrong, but it's good enough for some
# simple mapping. # simple mapping.
de_latitude: float = None de_latitude: float | None = None
de_longitude: float = None de_longitude: float | None = None
# General QSO info # General QSO info
# Reported mode, such as SSB, PHONE, CW, FT8... # Reported mode, such as SSB, PHONE, CW, FT8...
mode: str = None mode: str | None = None
# Inferred mode "family". One of "CW", "PHONE" or "DIGI". # Inferred mode "family". One of "CW", "PHONE" or "DIGI".
mode_type: str = None mode_type: str | None = None
# Source of the mode information. "SPOT", "COMMENT", "BANDPLAN" or "NONE" # Source of the mode information. "SPOT", "COMMENT", "BANDPLAN" or "NONE"
mode_source: str = "NONE" mode_source: str = "NONE"
# Frequency, in Hz # Frequency, in Hz
freq: float = None freq: float | None = None
# Band, defined by the frequency, e.g. "40m" or "70cm" # Band, defined by the frequency, e.g. "40m" or "70cm"
band: str = None band: str | None = None
# Comment left by the spotter, if any # Comment left by the spotter, if any
comment: str = None comment: str | None = None
# QRT state. Some APIs return spots marked as QRT. Otherwise we can check the comments. # QRT state. Some APIs return spots marked as QRT. Otherwise we can check the comments.
qrt: bool = False qrt: bool = False
# Special Interest Group info # Special Interest Group info
# Special Interest Group (SIG), e.g. outdoor activity programme such as POTA # Special Interest Group (SIG), e.g. outdoor activity programme such as POTA
sig: str = None sig: str | None = None
# SIG references. We allow multiple here for e.g. n-fer activations, unlike ADIF SIG_INFO # SIG references. We allow multiple here for e.g. n-fer activations, unlike ADIF SIG_INFO
sig_refs: list = None sig_refs: list | None = None
# Timing info # Timing info
# Time of the spot, UTC seconds since UNIX epoch # Time of the spot, UTC seconds since UNIX epoch
time: float = None time: float | None = None
# Time of the spot, ISO 8601 # Time of the spot, ISO 8601
time_iso: str = None time_iso: str | None = None
# Time that this software received the spot, UTC seconds since UNIX epoch. This is used with the "since_received" # Time that this software received the spot, UTC seconds since UNIX epoch. This is used with the "since_received"
# call to our API to receive all data that is new to us, even if by a quirk of the API it might be older than the # call to our API to receive all data that is new to us, even if by a quirk of the API it might be older than the
# list time the client polled the API. # list time the client polled the API.
received_time: float = None received_time: float | None = None
# Time that this software received the spot, ISO 8601 # Time that this software received the spot, ISO 8601
received_time_iso: str = None received_time_iso: str | None = None
# Source info # Source info
# Where we got the spot from, e.g. "POTA", "Cluster"... # Where we got the spot from, e.g. "POTA", "Cluster"...
source: str = None source: str | None = None
# The ID the source gave it, if any. # The ID the source gave it, if any.
source_id: str = None source_id: str | None = None
def infer_missing(self, credentials=None): def infer_missing(self, credentials=None):
"""Infer missing parameters where possible""" """Infer missing parameters where possible"""
@@ -342,9 +342,10 @@ class Spot:
# Determine a "QTH" string. If we have a SIG ref, pick the first one and turn it into a suitable string, # Determine a "QTH" string. If we have a SIG ref, pick the first one and turn it into a suitable string,
# otherwise see what they have set on an online lookup service. # otherwise see what they have set on an online lookup service.
if self.sig_refs and len(self.sig_refs) > 0: if self.sig_refs and len(self.sig_refs) > 0:
self.dx_qth = self.sig_refs[0].id qth = self.sig_refs[0].id
if self.sig_refs[0].name: if self.sig_refs[0].name:
self.dx_qth = self.dx_qth + " " + self.sig_refs[0].name qth += " " + self.sig_refs[0].name
self.dx_qth = qth
else: else:
self.dx_qth = lookup_helper.infer_qth_from_callsign_online_lookup(self.dx_call, credentials) self.dx_qth = lookup_helper.infer_qth_from_callsign_online_lookup(self.dx_call, credentials)
@@ -360,10 +361,11 @@ class Spot:
# It looks like we can sometimes get a string into lat/lon, so try to parse as float, reject if not valid # It looks like we can sometimes get a string into lat/lon, so try to parse as float, reject if not valid
if isinstance(self.dx_latitude, str) or isinstance(self.dx_longitude, str): if isinstance(self.dx_latitude, str) or isinstance(self.dx_longitude, str):
try: try:
self.dx_latitude = float(self.dx_latitude) self.dx_latitude = float(str(self.dx_latitude))
self.dx_longitude = float(self.dx_longitude) self.dx_longitude = float(str(self.dx_longitude))
except (TypeError, ValueError): except (TypeError, ValueError):
logging.warning("Received non-numeric strings in lat/lon (" + str(self.dx_latitude) + ", " + str(self.dx_longitude) + ") for call " + self.dx_call + ", rejecting it") logging.warning("Received non-numeric strings in lat/lon (" + str(self.dx_latitude) + ", " + str(
self.dx_longitude) + ") for call " + str(self.dx_call) + ", rejecting it")
self.dx_latitude = None self.dx_latitude = None
self.dx_longitude = None self.dx_longitude = None
@@ -381,10 +383,10 @@ class Spot:
# DX Location is "good" if it is from a spot, or from QRZ if the callsign doesn't contain a slash, so the operator # DX Location is "good" if it is from a spot, or from QRZ if the callsign doesn't contain a slash, so the operator
# is likely at home. # is likely at home.
self.dx_location_good = self.dx_latitude and self.dx_longitude and ( self.dx_location_good = bool(self.dx_latitude and self.dx_longitude and (
self.dx_location_source == "SPOT" or self.dx_location_source == "SIG REF LOOKUP" self.dx_location_source == "SPOT" or self.dx_location_source == "SIG REF LOOKUP"
or self.dx_location_source == "WAB/WAI GRID" or self.dx_location_source == "WAB/WAI GRID"
or (self.dx_location_source == "HOME QTH" and not "/" in self.dx_call)) or (self.dx_location_source == "HOME QTH" and "/" not in (self.dx_call or ""))))
# DE with no digits and APRS servers starting "T2" are not things we can look up location for # DE with no digits and APRS servers starting "T2" are not things we can look up location for
if self.de_call and any(char.isdigit() for char in self.de_call) and not ( if self.de_call and any(char.isdigit() for char in self.de_call) and not (
@@ -413,16 +415,16 @@ class Spot:
def _append_sig_ref_if_missing(self, new_sig_ref): def _append_sig_ref_if_missing(self, new_sig_ref):
"""Append a sig_ref to the list, so long as it's not already there.""" """Append a sig_ref to the list, so long as it's not already there."""
if not self.sig_refs: sig_refs = self.sig_refs or []
self.sig_refs = [] self.sig_refs = sig_refs
new_sig_ref.id = new_sig_ref.id.strip().upper() new_sig_ref.id = new_sig_ref.id.strip().upper()
new_sig_ref.sig = new_sig_ref.sig.strip().upper() new_sig_ref.sig = new_sig_ref.sig.strip().upper()
if new_sig_ref.id == "": if new_sig_ref.id == "":
return return
for sig_ref in self.sig_refs: for sig_ref in sig_refs:
if sig_ref.id == new_sig_ref.id and sig_ref.sig == new_sig_ref.sig: if sig_ref.id == new_sig_ref.id and sig_ref.sig == new_sig_ref.sig:
return return
self.sig_refs.append(new_sig_ref) sig_refs.append(new_sig_ref)
def expired(self): def expired(self):
"""Decide if this spot has expired (in which case it should not be added to the system in the first place, and not """Decide if this spot has expired (in which case it should not be added to the system in the first place, and not

View File

@@ -3,10 +3,13 @@ import logging
import re import re
import threading import threading
from datetime import datetime from datetime import datetime
from typing import Any
import pytz import pytz
import requests import requests
import tornado import tornado
from tornado import httputil
from tornado.web import Application
from core.config import ALLOW_SPOTTING, ALLOW_UPSTREAM_SPOTTING, MAX_SPOT_AGE, RECAPTCHA_SECRET_KEY from core.config import ALLOW_SPOTTING, ALLOW_UPSTREAM_SPOTTING, MAX_SPOT_AGE, RECAPTCHA_SECRET_KEY
from core.constants import UNKNOWN_BAND from core.constants import UNKNOWN_BAND
@@ -23,6 +26,11 @@ RECAPTCHA_VERIFY_URL = "https://www.google.com/recaptcha/api/siteverify"
class APISpotHandler(tornado.web.RequestHandler): class APISpotHandler(tornado.web.RequestHandler):
"""API request handler for /api/v1/spot (POST)""" """API request handler for /api/v1/spot (POST)"""
def __init__(self, application: "Application", request: httputil.HTTPServerRequest, **kwargs: Any):
self._spots = None
self._web_server_metrics = None
super().__init__(application, request, **kwargs)
def initialize(self, spots, web_server_metrics, spot_providers=None): def initialize(self, spots, web_server_metrics, spot_providers=None):
self._spots = spots self._spots = spots
self._web_server_metrics = web_server_metrics self._web_server_metrics = web_server_metrics

View File

@@ -3,16 +3,18 @@ import json
import logging import logging
from datetime import datetime from datetime import datetime
from queue import Queue from queue import Queue
from typing import Any
import pytz import pytz
import tornado import tornado
import tornado_eventsource.handler import tornado_eventsource.handler
from tornado import httputil
from tornado.web import Application
from core.prometheus_metrics_handler import api_requests_counter from core.prometheus_metrics_handler import api_requests_counter
from core.utils import serialize_everything, empty_queue from core.utils import serialize_everything, empty_queue
from data.lookup_credentials import extract_credentials from data.lookup_credentials import extract_credentials
SSE_HANDLER_MAX_QUEUE_SIZE = 100 SSE_HANDLER_MAX_QUEUE_SIZE = 100
SSE_HANDLER_QUEUE_CHECK_INTERVAL = 5000 SSE_HANDLER_QUEUE_CHECK_INTERVAL = 5000
@@ -20,6 +22,11 @@ SSE_HANDLER_QUEUE_CHECK_INTERVAL = 5000
class APIAlertsHandler(tornado.web.RequestHandler): class APIAlertsHandler(tornado.web.RequestHandler):
"""API request handler for /api/v1/alerts""" """API request handler for /api/v1/alerts"""
def __init__(self, application: "Application", request: httputil.HTTPServerRequest, **kwargs: Any):
self._alerts = None
self._web_server_metrics = None
super().__init__(application, request, **kwargs)
def initialize(self, alerts, web_server_metrics): def initialize(self, alerts, web_server_metrics):
self._alerts = alerts self._alerts = alerts
self._web_server_metrics = web_server_metrics self._web_server_metrics = web_server_metrics
@@ -67,6 +74,15 @@ class APIAlertsHandler(tornado.web.RequestHandler):
class APIAlertsStreamHandler(tornado_eventsource.handler.EventSourceHandler): class APIAlertsStreamHandler(tornado_eventsource.handler.EventSourceHandler):
"""API request handler for /api/v1/alerts/stream""" """API request handler for /api/v1/alerts/stream"""
def __init__(self, application, request, **kwargs: Any):
self._sse_alert_queues = None
self._web_server_metrics = None
self._query_params = None
self._credentials = None
self._alert_queue = None
self._heartbeat = None
super().__init__(application, request, **kwargs)
def initialize(self, sse_alert_queues, web_server_metrics): def initialize(self, sse_alert_queues, web_server_metrics):
self._sse_alert_queues = sse_alert_queues self._sse_alert_queues = sse_alert_queues
self._web_server_metrics = web_server_metrics self._web_server_metrics = web_server_metrics

View File

@@ -1,9 +1,12 @@
import json import json
from collections import Counter from collections import Counter
from datetime import datetime, timedelta from datetime import datetime, timedelta
from typing import Any
import pytz import pytz
import tornado import tornado
from tornado import httputil
from tornado.web import Application
from core.prometheus_metrics_handler import api_requests_counter from core.prometheus_metrics_handler import api_requests_counter
@@ -16,6 +19,11 @@ BANDS_SET = frozenset(BANDS)
class APIDxStatsHandler(tornado.web.RequestHandler): class APIDxStatsHandler(tornado.web.RequestHandler):
"""API request handler for /api/v1/dxstats""" """API request handler for /api/v1/dxstats"""
def __init__(self, application: "Application", request: httputil.HTTPServerRequest, **kwargs: Any):
self._spots = None
self._web_server_metrics = None
super().__init__(application, request, **kwargs)
def initialize(self, spots, web_server_metrics): def initialize(self, spots, web_server_metrics):
self._spots = spots self._spots = spots
self._web_server_metrics = web_server_metrics self._web_server_metrics = web_server_metrics

View File

@@ -2,9 +2,12 @@ import json
import logging import logging
import re import re
from datetime import datetime from datetime import datetime
from typing import Any
import pytz import pytz
import tornado import tornado
from tornado import httputil
from tornado.web import Application
from core.constants import SIGS from core.constants import SIGS
from core.geo_utils import lat_lon_for_grid_sw_corner_plus_size, lat_lon_to_cq_zone, lat_lon_to_itu_zone from core.geo_utils import lat_lon_for_grid_sw_corner_plus_size, lat_lon_to_cq_zone, lat_lon_to_itu_zone
@@ -19,6 +22,10 @@ from data.spot import Spot
class APILookupCallHandler(tornado.web.RequestHandler): class APILookupCallHandler(tornado.web.RequestHandler):
"""API request handler for /api/v1/lookup/call""" """API request handler for /api/v1/lookup/call"""
def __init__(self, application: "Application", request: httputil.HTTPServerRequest, **kwargs: Any):
self._web_server_metrics = None
super().__init__(application, request, **kwargs)
def initialize(self, web_server_metrics): def initialize(self, web_server_metrics):
self._web_server_metrics = web_server_metrics self._web_server_metrics = web_server_metrics
@@ -36,7 +43,7 @@ class APILookupCallHandler(tornado.web.RequestHandler):
# The "call" query param must exist and look like a callsign # The "call" query param must exist and look like a callsign
if "call" in query_params.keys(): if "call" in query_params.keys():
call = query_params.get("call").upper() call = str(query_params.get("call")).upper()
if re.match(r"^[A-Z0-9/\-]*$", call): if re.match(r"^[A-Z0-9/\-]*$", call):
# Take the callsign, make a "fake spot" so we can run infer_missing() on it, then repack the # Take the callsign, make a "fake spot" so we can run infer_missing() on it, then repack the
# resulting data in the correct way for the API response. # resulting data in the correct way for the API response.
@@ -80,6 +87,10 @@ class APILookupCallHandler(tornado.web.RequestHandler):
class APILookupSIGRefHandler(tornado.web.RequestHandler): class APILookupSIGRefHandler(tornado.web.RequestHandler):
"""API request handler for /api/v1/lookup/sigref""" """API request handler for /api/v1/lookup/sigref"""
def __init__(self, application: "Application", request: httputil.HTTPServerRequest, **kwargs: Any):
self._web_server_metrics = None
super().__init__(application, request, **kwargs)
def initialize(self, web_server_metrics): def initialize(self, web_server_metrics):
self._web_server_metrics = web_server_metrics self._web_server_metrics = web_server_metrics
@@ -98,8 +109,8 @@ class APILookupSIGRefHandler(tornado.web.RequestHandler):
# "sig" and "id" query params must exist, SIG must be known, and if we have a reference regex for that SIG, # "sig" and "id" query params must exist, SIG must be known, and if we have a reference regex for that SIG,
# the provided id must match it. # the provided id must match it.
if "sig" in query_params.keys() and "id" in query_params.keys(): if "sig" in query_params.keys() and "id" in query_params.keys():
sig = query_params.get("sig").upper() sig = str(query_params.get("sig")).upper()
ref_id = query_params.get("id").upper() ref_id = str(query_params.get("id")).upper()
if sig in list(map(lambda p: p.name, SIGS)): if sig in list(map(lambda p: p.name, SIGS)):
if not get_ref_regex_for_sig(sig) or re.match(get_ref_regex_for_sig(sig), ref_id): if not get_ref_regex_for_sig(sig) or re.match(get_ref_regex_for_sig(sig), ref_id):
data = populate_sig_ref_info(SIGRef(id=ref_id, sig=sig)) data = populate_sig_ref_info(SIGRef(id=ref_id, sig=sig))
@@ -107,8 +118,9 @@ class APILookupSIGRefHandler(tornado.web.RequestHandler):
else: else:
self.write( self.write(
json.dumps("Error - '" + ref_id + "' does not look like a valid reference ID for " + sig + ".", json.dumps(
default=serialize_everything)) "Error - '" + ref_id + "' does not look like a valid reference ID for " + sig + ".",
default=serialize_everything))
self.set_status(422) self.set_status(422)
else: else:
self.write(json.dumps("Error - sig '" + sig + "' is not known.", default=serialize_everything)) self.write(json.dumps("Error - sig '" + sig + "' is not known.", default=serialize_everything))
@@ -129,6 +141,10 @@ class APILookupSIGRefHandler(tornado.web.RequestHandler):
class APILookupGridHandler(tornado.web.RequestHandler): class APILookupGridHandler(tornado.web.RequestHandler):
"""API request handler for /api/v1/lookup/grid""" """API request handler for /api/v1/lookup/grid"""
def __init__(self, application: "Application", request: httputil.HTTPServerRequest, **kwargs: Any):
self._web_server_metrics = None
super().__init__(application, request, **kwargs)
def initialize(self, web_server_metrics): def initialize(self, web_server_metrics):
self._web_server_metrics = web_server_metrics self._web_server_metrics = web_server_metrics
@@ -146,7 +162,7 @@ class APILookupGridHandler(tornado.web.RequestHandler):
# "grid" query param must exist. # "grid" query param must exist.
if "grid" in query_params.keys(): if "grid" in query_params.keys():
grid = query_params.get("grid").upper() grid = str(query_params.get("grid")).upper()
lat, lon, lat_cell_size, lon_cell_size = lat_lon_for_grid_sw_corner_plus_size(grid) lat, lon, lat_cell_size, lon_cell_size = lat_lon_for_grid_sw_corner_plus_size(grid)
if lat is not None and lon is not None and lat_cell_size is not None and lon_cell_size is not None: if lat is not None and lon is not None and lat_cell_size is not None and lon_cell_size is not None:
center_lat = lat + lat_cell_size / 2.0 center_lat = lat + lat_cell_size / 2.0

View File

@@ -1,8 +1,11 @@
import json import json
from datetime import datetime from datetime import datetime
from typing import Any
import pytz import pytz
import tornado import tornado
from tornado import httputil
from tornado.web import Application
from core.config import MAX_SPOT_AGE, ALLOW_SPOTTING from core.config import MAX_SPOT_AGE, ALLOW_SPOTTING
from core.constants import BANDS, ALL_MODES, MODE_TYPES, SIGS, CONTINENTS from core.constants import BANDS, ALL_MODES, MODE_TYPES, SIGS, CONTINENTS
@@ -13,6 +16,11 @@ from core.utils import serialize_everything
class APIOptionsHandler(tornado.web.RequestHandler): class APIOptionsHandler(tornado.web.RequestHandler):
"""API request handler for /api/v1/options""" """API request handler for /api/v1/options"""
def __init__(self, application: "Application", request: httputil.HTTPServerRequest, **kwargs: Any):
self._status_data = None
self._web_server_metrics = None
super().__init__(application, request, **kwargs)
def initialize(self, status_data, web_server_metrics, spot_providers=None): def initialize(self, status_data, web_server_metrics, spot_providers=None):
self._status_data = status_data self._status_data = status_data
self._web_server_metrics = web_server_metrics self._web_server_metrics = web_server_metrics

View File

@@ -1,7 +1,10 @@
from datetime import datetime from datetime import datetime
from typing import Any
import pytz import pytz
import tornado import tornado
from tornado import httputil
from tornado.web import Application
from core.prometheus_metrics_handler import api_requests_counter from core.prometheus_metrics_handler import api_requests_counter
@@ -9,6 +12,11 @@ from core.prometheus_metrics_handler import api_requests_counter
class APISolarConditionsHandler(tornado.web.RequestHandler): class APISolarConditionsHandler(tornado.web.RequestHandler):
"""API request handler for /api/v1/solar""" """API request handler for /api/v1/solar"""
def __init__(self, application: "Application", request: httputil.HTTPServerRequest, **kwargs: Any):
self._solar_conditions = None
self._web_server_metrics = None
super().__init__(application, request, **kwargs)
def initialize(self, solar_conditions, web_server_metrics): def initialize(self, solar_conditions, web_server_metrics):
self._solar_conditions = solar_conditions self._solar_conditions = solar_conditions
self._web_server_metrics = web_server_metrics self._web_server_metrics = web_server_metrics

View File

@@ -3,16 +3,18 @@ import json
import logging import logging
from datetime import datetime, timedelta from datetime import datetime, timedelta
from queue import Queue from queue import Queue
from typing import Any
import pytz import pytz
import tornado import tornado
import tornado_eventsource.handler import tornado_eventsource.handler
from tornado import httputil
from tornado.web import Application
from core.prometheus_metrics_handler import api_requests_counter from core.prometheus_metrics_handler import api_requests_counter
from core.utils import serialize_everything, empty_queue from core.utils import serialize_everything, empty_queue
from data.lookup_credentials import extract_credentials from data.lookup_credentials import extract_credentials
SSE_HANDLER_MAX_QUEUE_SIZE = 1000 SSE_HANDLER_MAX_QUEUE_SIZE = 1000
SSE_HANDLER_QUEUE_CHECK_INTERVAL = 5000 SSE_HANDLER_QUEUE_CHECK_INTERVAL = 5000
@@ -20,6 +22,11 @@ SSE_HANDLER_QUEUE_CHECK_INTERVAL = 5000
class APISpotsHandler(tornado.web.RequestHandler): class APISpotsHandler(tornado.web.RequestHandler):
"""API request handler for /api/v1/spots""" """API request handler for /api/v1/spots"""
def __init__(self, application: "Application", request: httputil.HTTPServerRequest, **kwargs: Any):
self._spots = None
self._web_server_metrics = None
super().__init__(application, request, **kwargs)
def initialize(self, spots, web_server_metrics): def initialize(self, spots, web_server_metrics):
self._spots = spots self._spots = spots
self._web_server_metrics = web_server_metrics self._web_server_metrics = web_server_metrics
@@ -67,6 +74,15 @@ class APISpotsHandler(tornado.web.RequestHandler):
class APISpotsStreamHandler(tornado_eventsource.handler.EventSourceHandler): class APISpotsStreamHandler(tornado_eventsource.handler.EventSourceHandler):
"""API request handler for /api/v1/spots/stream""" """API request handler for /api/v1/spots/stream"""
def __init__(self, application, request, **kwargs: Any):
self._sse_spot_queues = None
self._web_server_metrics = None
self._query_params = None
self._credentials = None
self._spot_queue = None
self._heartbeat = None
super().__init__(application, request, **kwargs)
def initialize(self, sse_spot_queues, web_server_metrics): def initialize(self, sse_spot_queues, web_server_metrics):
self._sse_spot_queues = sse_spot_queues self._sse_spot_queues = sse_spot_queues
self._web_server_metrics = web_server_metrics self._web_server_metrics = web_server_metrics

View File

@@ -1,8 +1,11 @@
import json import json
from datetime import datetime from datetime import datetime
from typing import Any
import pytz import pytz
import tornado import tornado
from tornado import httputil
from tornado.web import Application
from core.prometheus_metrics_handler import api_requests_counter from core.prometheus_metrics_handler import api_requests_counter
from core.utils import serialize_everything from core.utils import serialize_everything
@@ -11,6 +14,11 @@ from core.utils import serialize_everything
class APIStatusHandler(tornado.web.RequestHandler): class APIStatusHandler(tornado.web.RequestHandler):
"""API request handler for /api/v1/status""" """API request handler for /api/v1/status"""
def __init__(self, application: "Application", request: httputil.HTTPServerRequest, **kwargs: Any):
self._status_data = None
self._web_server_metrics = None
super().__init__(application, request, **kwargs)
def initialize(self, status_data, web_server_metrics): def initialize(self, status_data, web_server_metrics):
self._status_data = status_data self._status_data = status_data
self._web_server_metrics = web_server_metrics self._web_server_metrics = web_server_metrics

View File

@@ -1,7 +1,10 @@
from datetime import datetime from datetime import datetime
from typing import Any
import pytz import pytz
import tornado import tornado
from tornado import httputil
from tornado.web import Application
from core.config import ALLOW_SPOTTING, WEB_UI_OPTIONS, BASE_URL, SERVER_OWNER_CALLSIGN from core.config import ALLOW_SPOTTING, WEB_UI_OPTIONS, BASE_URL, SERVER_OWNER_CALLSIGN
from core.constants import SOFTWARE_VERSION from core.constants import SOFTWARE_VERSION
@@ -11,6 +14,11 @@ from core.prometheus_metrics_handler import page_requests_counter
class PageTemplateHandler(tornado.web.RequestHandler): class PageTemplateHandler(tornado.web.RequestHandler):
"""Handler for all HTML pages generated from templates""" """Handler for all HTML pages generated from templates"""
def __init__(self, application: "Application", request: httputil.HTTPServerRequest, **kwargs: Any):
self._template_name = None
self._web_server_metrics = None
super().__init__(application, request, **kwargs)
def initialize(self, template_name, web_server_metrics): def initialize(self, template_name, web_server_metrics):
self._template_name = template_name self._template_name = template_name
self._web_server_metrics = web_server_metrics self._web_server_metrics = web_server_metrics

View File

@@ -18,6 +18,8 @@ from server.handlers.api.status import APIStatusHandler
from server.handlers.metrics import PrometheusMetricsHandler from server.handlers.metrics import PrometheusMetricsHandler
from server.handlers.pagetemplate import PageTemplateHandler from server.handlers.pagetemplate import PageTemplateHandler
_HERE = os.path.dirname(__file__ or "")
class WebServer: class WebServer:
"""Provides the public-facing web server.""" """Provides the public-facing web server."""
@@ -102,11 +104,11 @@ class WebServer:
misc_routes = [ misc_routes = [
(r"/apidocs", PageTemplateHandler, {"template_name": "apidocs", **handler_opts}), (r"/apidocs", PageTemplateHandler, {"template_name": "apidocs", **handler_opts}),
(r"/metrics", PrometheusMetricsHandler), (r"/metrics", PrometheusMetricsHandler),
(r"/(.*)", StaticFileHandler, {"path": os.path.join(os.path.dirname(__file__), "../webassets")}) (r"/(.*)", StaticFileHandler, {"path": os.path.join(_HERE, "../webassets")})
] ]
app = tornado.web.Application(api_routes + ui_routes + misc_routes, app = tornado.web.Application(api_routes + ui_routes + misc_routes,
template_path=os.path.join(os.path.dirname(__file__), "../templates"), template_path=os.path.join(_HERE, "../templates"),
debug=False) debug=False)
app.listen(self._port) app.listen(self._port)
logging.info("Web server running on port " + str(WEB_SERVER_PORT)) logging.info("Web server running on port " + str(WEB_SERVER_PORT))

View File

@@ -28,7 +28,8 @@ class GIROIonosonde(SolarConditionsProvider):
self._thread = None self._thread = None
self._stop_event = Event() self._stop_event = Event()
def _load_stations(self): @staticmethod
def _load_stations():
stations = [] stations = []
with open(STATIONS_INDEX, newline='') as f: with open(STATIONS_INDEX, newline='') as f:
for row in csv.reader(f): for row in csv.reader(f):

View File

@@ -4,7 +4,6 @@ from xml.etree import ElementTree
import pytz import pytz
from dateutil import parser as dateutil_parser, tz as dateutil_tz from dateutil import parser as dateutil_parser, tz as dateutil_tz
from solarconditionsproviders.http_solar_conditions_provider import HTTPSolarConditionsProvider from solarconditionsproviders.http_solar_conditions_provider import HTTPSolarConditionsProvider
POLL_INTERVAL = 3600 # 1 hour POLL_INTERVAL = 3600 # 1 hour
@@ -32,6 +31,9 @@ class HamQSL(HTTPSolarConditionsProvider):
# Some error checking functions in case the data is janky. # Some error checking functions in case the data is janky.
def text(tag, default=None): def text(tag, default=None):
if sd is None:
logging.warning("HamQSL solar conditions API returned unexpected XML structure")
return default
el = sd.find(tag) el = sd.find(tag)
return el.text.strip() if el is not None and el.text else default return el.text.strip() if el is not None and el.text else default
@@ -95,16 +97,17 @@ class HamQSL(HTTPSolarConditionsProvider):
"solar_wind": float_val("solarwind"), "solar_wind": float_val("solarwind"),
"magnetic_field": float_val("magneticfield"), "magnetic_field": float_val("magneticfield"),
"geomag_field": text("geomagfield").title() "geomag_field": text("geomagfield").title()
.replace("Vr Quiet", "Very Quiet") .replace("Vr Quiet", "Very Quiet")
.replace("Unsettld", "Unsettled") .replace("Unsettld", "Unsettled")
.replace("Min Strm", "Minor Storm") .replace("Min Strm", "Minor Storm")
.replace("Maj Strm", "Major Storm") .replace("Maj Strm", "Major Storm")
.replace("Sev Strm", "Severe Storm") .replace("Sev Strm", "Severe Storm")
.replace("Ext Strm", "Extreme Storm"), .replace("Ext Strm", "Extreme Storm"),
"geomag_noise": text("signalnoise"), "geomag_noise": text("signalnoise"),
"hf_conditions": hf_conditions, "hf_conditions": hf_conditions,
"vhf_conditions": { "vhf_conditions": {
"vhf_aurora_northern_hemi": vhf_map.get(("vhf-aurora", "northern_hemi")).title().replace("Lat Aur", "Latitude"), "vhf_aurora_northern_hemi": (vhf_map.get(("vhf-aurora", "northern_hemi")) or "").title().replace(
"Lat Aur", "Latitude") or None,
"es_2m_europe": vhf_map.get(("E-Skip", "europe")), "es_2m_europe": vhf_map.get(("E-Skip", "europe")),
"es_4m_europe": vhf_map.get(("E-Skip", "europe_4m")), "es_4m_europe": vhf_map.get(("E-Skip", "europe_4m")),
"es_6m_europe": vhf_map.get(("E-Skip", "europe_6m")), "es_6m_europe": vhf_map.get(("E-Skip", "europe_6m")),

View File

@@ -3,8 +3,8 @@ from core.constants import BANDS
HF_BANDS = [b for b in BANDS if b.is_ham_hf] HF_BANDS = [b for b in BANDS if b.is_ham_hf]
def _latest(d): def _latest(d) -> float | None:
return d[max(d.keys())] if d else None return float(d[max(d.keys())]) if d else None
def compute_band_states(fof2_dict, muf_dict, luf_dict): def compute_band_states(fof2_dict, muf_dict, luf_dict):

View File

@@ -4,7 +4,7 @@ from datetime import datetime, timezone
from solarconditionsproviders.http_solar_conditions_provider import HTTPSolarConditionsProvider from solarconditionsproviders.http_solar_conditions_provider import HTTPSolarConditionsProvider
POLL_INTERVAL = 10800 # Every 3 hours POLL_INTERVAL = 10800 # Every 3 hours
URL = "https://services.swpc.noaa.gov/text/3-day-forecast.txt" URL = "https://services.swpc.noaa.gov/text/3-day-forecast.txt"

View File

@@ -10,6 +10,7 @@ class SolarConditionsProvider:
def __init__(self, provider_config): def __init__(self, provider_config):
"""Constructor""" """Constructor"""
self._solar_conditions_cache = None
self.name = provider_config["name"] self.name = provider_config["name"]
self.enabled = provider_config["enabled"] self.enabled = provider_config["enabled"]
self.last_update_time = datetime.min.replace(tzinfo=pytz.UTC) self.last_update_time = datetime.min.replace(tzinfo=pytz.UTC)

View File

@@ -8,11 +8,11 @@ import sys
from diskcache import Cache from diskcache import Cache
from core.cleanup import CleanupTimer from core.cleanup import CleanupTimer
from data.solar_conditions import SolarConditions
from core.config import config, SERVER_OWNER_CALLSIGN from core.config import config, SERVER_OWNER_CALLSIGN
from core.constants import SOFTWARE_VERSION from core.constants import SOFTWARE_VERSION
from core.lookup_helper import lookup_helper from core.lookup_helper import lookup_helper
from core.status_reporter import StatusReporter from core.status_reporter import StatusReporter
from data.solar_conditions import SolarConditions
from server.webserver import WebServer from server.webserver import WebServer
# Globals # Globals
@@ -29,13 +29,14 @@ cleanup_timer = None
run = True run = True
def shutdown(sig, frame): def shutdown(_signum=None, _frame=None):
"""Shutdown function""" """Shutdown function"""
global run global run
logging.info("Stopping program...") logging.info("Stopping program...")
web_server.stop() if web_server:
web_server.stop()
for sp in spot_providers: for sp in spot_providers:
if sp.enabled: if sp.enabled:
sp.stop() sp.stop()
@@ -45,8 +46,10 @@ def shutdown(sig, frame):
for scp in solar_condition_providers: for scp in solar_condition_providers:
if scp.enabled: if scp.enabled:
scp.stop() scp.stop()
cleanup_timer.stop() if cleanup_timer:
lookup_helper.stop() cleanup_timer.stop()
if lookup_helper:
lookup_helper.stop()
spots.close() spots.close()
alerts.close() alerts.close()
solar_conditions_cache.close() solar_conditions_cache.close()

View File

@@ -37,20 +37,20 @@ class APRSIS(SpotProvider):
def _handle(self, data): def _handle(self, data):
# Split SSID in "from" call and store separately # Split SSID in "from" call and store separately
from_parts = data["from"].split("-").upper() from_parts = str(data["from"]).split("-")
dx_call = from_parts[0] dx_call = from_parts[0].upper()
dx_ssid = from_parts[1] if len(from_parts) > 1 else None dx_ssid = from_parts[1].upper() if len(from_parts) > 1 else None
via_parts = data["via"].split("-").upper() via_parts = str(data["via"]).split("-")
de_call = via_parts[0] de_call = via_parts[0].upper()
de_ssid = via_parts[1] if len(via_parts) > 1 else None de_ssid = via_parts[1].upper() if len(via_parts) > 1 else None
spot = Spot(source="APRS-IS", spot = Spot(source="APRS-IS",
dx_call=dx_call, dx_call=dx_call,
dx_ssid=dx_ssid, dx_ssid=dx_ssid,
de_call=de_call, de_call=de_call,
de_ssid=de_ssid, de_ssid=de_ssid,
comment=data["comment"] if "comment" in data else None, comment=str(data["comment"]) if "comment" in data else None,
dx_latitude=data["latitude"] if "latitude" in data else None, dx_latitude=float(data["latitude"]) if "latitude" in data else None,
dx_longitude=data["longitude"] if "longitude" in data else None, dx_longitude=float(data["longitude"]) if "longitude" in data else None,
time=datetime.now( time=datetime.now(
pytz.UTC).timestamp()) # APRS-IS spots are live so we can assume spot time is "now" pytz.UTC).timestamp()) # APRS-IS spots are live so we can assume spot time is "now"

View File

@@ -72,7 +72,8 @@ class DXCluster(SpotProvider):
match = self._spot_line_pattern.match(telnet_output.decode("latin-1")) match = self._spot_line_pattern.match(telnet_output.decode("latin-1"))
if match: if match:
spot_time = datetime.strptime(match.group(5), "%H%MZ") spot_time = datetime.strptime(match.group(5), "%H%MZ")
spot_datetime = datetime.combine(datetime.now(pytz.UTC).date(), spot_time.time(), tzinfo=pytz.UTC) spot_datetime = datetime.combine(datetime.now(pytz.UTC).date(), spot_time.time(),
tzinfo=pytz.UTC)
spot = Spot(source=self.name, spot = Spot(source=self.name,
dx_call=match.group(3), dx_call=match.group(3),
de_call=match.group(1), de_call=match.group(1),

View File

@@ -38,10 +38,10 @@ class GMA(HTTPSpotProvider):
time=datetime.strptime(source_spot["DATE"] + source_spot["TIME"], "%Y%m%d%H%M").replace( time=datetime.strptime(source_spot["DATE"] + source_spot["TIME"], "%Y%m%d%H%M").replace(
tzinfo=pytz.UTC).timestamp(), tzinfo=pytz.UTC).timestamp(),
dx_latitude=float(source_spot["LAT"]) if ( dx_latitude=float(source_spot["LAT"]) if (
source_spot["LAT"] and source_spot["LAT"] != "") else None, source_spot["LAT"] and source_spot["LAT"] != "") else None,
# Seen GMA spots with no (or empty) lat/lon # Seen GMA spots with no (or empty) lat/lon
dx_longitude=float(source_spot["LON"]) if ( dx_longitude=float(source_spot["LON"]) if (
source_spot["LON"] and source_spot["LON"] != "") else None) source_spot["LON"] and source_spot["LON"] != "") else None)
# GMA doesn't give what programme (SIG) the reference is for until we separately look it up. # GMA doesn't give what programme (SIG) the reference is for until we separately look it up.
if "REF" in source_spot: if "REF" in source_spot:
@@ -55,7 +55,7 @@ class GMA(HTTPSpotProvider):
# spots come through with reftype=POTA or reftype=WWFF. SOTA is harder to figure out because both SOTA # spots come through with reftype=POTA or reftype=WWFF. SOTA is harder to figure out because both SOTA
# and GMA summits come through with reftype=Summit, so we must check for the presence of a "sota" entry # and GMA summits come through with reftype=Summit, so we must check for the presence of a "sota" entry
# to determine if it's a SOTA summit. # to determine if it's a SOTA summit.
if "reftype" in ref_info and ref_info["reftype"] not in ["POTA", "WWFF"] and ( if spot.sig_refs and "reftype" in ref_info and ref_info["reftype"] not in ["POTA", "WWFF"] and (
ref_info["reftype"] != "Summit" or "sota" not in ref_info or ref_info["sota"] == ""): ref_info["reftype"] != "Summit" or "sota" not in ref_info or ref_info["sota"] == ""):
match ref_info["reftype"]: match ref_info["reftype"]:
case "Summit": case "Summit":

View File

@@ -45,6 +45,8 @@ class HEMA(HTTPSpotProvider):
# Fiddle with some data to extract bits we need. Freq/mode and spotter/comment come in combined fields. # Fiddle with some data to extract bits we need. Freq/mode and spotter/comment come in combined fields.
freq_mode_match = re.search(self.FREQ_MODE_PATTERN, spot_items[5]) freq_mode_match = re.search(self.FREQ_MODE_PATTERN, spot_items[5])
spotter_comment_match = re.search(self.SPOTTER_COMMENT_PATTERN, spot_items[6]) spotter_comment_match = re.search(self.SPOTTER_COMMENT_PATTERN, spot_items[6])
if not freq_mode_match or not spotter_comment_match:
continue
# Convert to our spot format # Convert to our spot format
spot = Spot(source=self.name, spot = Spot(source=self.name,

View File

@@ -22,8 +22,8 @@ class LLOTA(HTTPSpotProvider):
comment = None comment = None
spotter = None spotter = None
if "history" in source_spot and len(source_spot["history"]) > 0: if "history" in source_spot and len(source_spot["history"]) > 0:
comment = source_spot["history"][-1]["comment"] comment = str(source_spot["history"][-1]["comment"])
spotter = source_spot["history"][-1]["spotter_callsign"] spotter = str(source_spot["history"][-1]["spotter_callsign"])
# Convert to our spot format # Convert to our spot format
spot = Spot(source=self.name, spot = Spot(source=self.name,
source_id=source_spot["id"], source_id=source_spot["id"],

View File

@@ -43,9 +43,9 @@ class ParksNPeaks(HTTPSpotProvider):
tzinfo=pytz.UTC).timestamp()) tzinfo=pytz.UTC).timestamp())
# Extract a de_call if it's in the comment but not in the "actSpoter" field # Extract a de_call if it's in the comment but not in the "actSpoter" field
m = re.search(r"\(de ([A-Za-z0-9]*)\)", spot.comment) m = re.search(r"\(de ([A-Za-z0-9]*)\)", spot.comment or "")
if not spot.de_call and m: if not spot.de_call and m:
spot.de_call = m.group(1) spot.de_call = str(m.group(1))
# Record SIG information. Sometimes we get a "SIG" of "QRP", which we ignore as it's not a programme with a # Record SIG information. Sometimes we get a "SIG" of "QRP", which we ignore as it's not a programme with a
# defined set of references # defined set of references
@@ -53,11 +53,12 @@ class ParksNPeaks(HTTPSpotProvider):
sig_ref = source_spot["actSiteID"] sig_ref = source_spot["actSiteID"]
if sig and sig != "" and sig != "QRP" and sig_ref and sig_ref != "": if sig and sig != "" and sig != "QRP" and sig_ref and sig_ref != "":
spot.sig = sig spot.sig = sig
spot.sig_refs = [SIGRef(id=source_spot["actSiteID"], sig=source_spot["actClass"].upper())] sig_refs = [SIGRef(id=source_spot["actSiteID"], sig=source_spot["actClass"].upper())]
spot.sig_refs = sig_refs
# Free text location is not present in all spots, so only add it if it's set # Free text location is not present in all spots, so only add it if it's set
if "actLocation" in source_spot and source_spot["actLocation"] != "": if "actLocation" in source_spot and source_spot["actLocation"] != "":
spot.sig_refs[0].name = source_spot["actLocation"] sig_refs[0].name = source_spot["actLocation"]
# Log a warning for the developer if PnP gives us an unknown programme we've never seen before # Log a warning for the developer if PnP gives us an unknown programme we've never seen before
if sig not in ["POTA", "SOTA", "WWFF", "SIOTA", "ZLOTA", "KRMNPA"]: if sig not in ["POTA", "SOTA", "WWFF", "SIOTA", "ZLOTA", "KRMNPA"]:

View File

@@ -46,7 +46,7 @@ class RBN(SpotProvider):
self.status = "Connecting" self.status = "Connecting"
logging.info("RBN port " + str(self._port) + " connecting...") logging.info("RBN port " + str(self._port) + " connecting...")
self._telnet = telnetlib3.Telnet("telnet.reversebeacon.net", self._port) self._telnet = telnetlib3.Telnet("telnet.reversebeacon.net", self._port)
telnet_output = self._telnet.read_until("Please enter your call: ".encode("latin-1")) self._telnet.read_until("Please enter your call: ".encode("latin-1"))
self._telnet.write((SERVER_OWNER_CALLSIGN + "\n").encode("latin-1")) self._telnet.write((SERVER_OWNER_CALLSIGN + "\n").encode("latin-1"))
connected = True connected = True
logging.info("RBN port " + str(self._port) + " connected.") logging.info("RBN port " + str(self._port) + " connected.")
@@ -63,7 +63,8 @@ class RBN(SpotProvider):
match = self._LINE_PATTERN.match(telnet_output.decode("latin-1")) match = self._LINE_PATTERN.match(telnet_output.decode("latin-1"))
if match: if match:
spot_time = datetime.strptime(match.group(5), "%H%MZ") spot_time = datetime.strptime(match.group(5), "%H%MZ")
spot_datetime = datetime.combine(datetime.now(pytz.UTC).date(), spot_time.time(), tzinfo=pytz.UTC) spot_datetime = datetime.combine(datetime.now(pytz.UTC).date(), spot_time.time(),
tzinfo=pytz.UTC)
spot = Spot(source=self.name, spot = Spot(source=self.name,
dx_call=match.group(3), dx_call=match.group(3),
de_call=match.group(1), de_call=match.group(1),

View File

@@ -46,7 +46,7 @@ class SOTA(HTTPSpotProvider):
dx_name=source_spot["activatorName"], dx_name=source_spot["activatorName"],
de_call=source_spot["callsign"].upper(), de_call=source_spot["callsign"].upper(),
freq=(float(source_spot["frequency"]) * 1000000) if ( freq=(float(source_spot["frequency"]) * 1000000) if (
source_spot["frequency"] is not None) else None, source_spot["frequency"] is not None) else None,
# Seen SOTA spots with no frequency! # Seen SOTA spots with no frequency!
mode=source_spot["mode"].upper(), mode=source_spot["mode"].upper(),
comment=source_spot["comments"], comment=source_spot["comments"],

View File

@@ -35,7 +35,8 @@ class Tiles(HTTPSpotProvider):
sig="Tiles", sig="Tiles",
# Tiles spots can include POTA & SOTA references, but ignore those on the basis that we will get them separately from the POTA/SOTA providers anyway. # Tiles spots can include POTA & SOTA references, but ignore those on the basis that we will get them separately from the POTA/SOTA providers anyway.
# Just take the grid reference itself as the single Tiles SIG reference. # Just take the grid reference itself as the single Tiles SIG reference.
sig_refs=[SIGRef(id=source_spot["maidenhead_grid"], sig="Tiles", name=source_spot["maidenhead_grid"])], sig_refs=[SIGRef(id=source_spot["maidenhead_grid"], sig="Tiles",
name=source_spot["maidenhead_grid"])],
time=datetime.fromisoformat(source_spot["created_at"].replace("Z", "+00:00")).timestamp(), time=datetime.fromisoformat(source_spot["created_at"].replace("Z", "+00:00")).timestamp(),
dx_grid=source_spot["maidenhead_grid"], dx_grid=source_spot["maidenhead_grid"],
dx_latitude=source_spot["latitude"], dx_latitude=source_spot["latitude"],

View File

@@ -1,9 +1,11 @@
import logging import logging
import re import re
from datetime import datetime from datetime import datetime
from typing import cast
import pytz import pytz
from rss_parser import Parser from rss_parser import Parser
from rss_parser.models.rss import RSS
from data.sig_ref import SIGRef from data.sig_ref import SIGRef
from data.spot import Spot from data.spot import Spot
@@ -23,7 +25,7 @@ class WOTA(HTTPSpotProvider):
def _http_response_to_spots(self, http_response): def _http_response_to_spots(self, http_response):
new_spots = [] new_spots = []
rss = Parser.parse(http_response.content.decode()) rss = cast(RSS, Parser.parse(http_response.content.decode()))
# Iterate through source data # Iterate through source data
for source_spot in rss.channel.items: for source_spot in rss.channel.items:
@@ -39,15 +41,15 @@ class WOTA(HTTPSpotProvider):
ref_name = None ref_name = None
if len(title_split) > 1: if len(title_split) > 1:
ref_split = title_split[1].split(" - ") ref_split = title_split[1].split(" - ")
ref = ref_split[0] ref = str(ref_split[0])
if len(ref_split) > 1: if len(ref_split) > 1:
ref_name = ref_split[1] ref_name = str(ref_split[1])
# Pick apart the description # Pick apart the description
desc_split = source_spot.description.split(". ") desc_split = source_spot.description.split(". ")
freq_mode = desc_split[0].replace("Frequencies/modes:", "").strip() freq_mode = desc_split[0].replace("Frequencies/modes:", "").strip()
freq_mode_split = re.split(r'[\-\s]+', freq_mode) freq_mode_split = re.split(r'[\-\s]+', freq_mode)
freq_hz = float(freq_mode_split[0].replace("'",".")) * 1000000 freq_hz = float(freq_mode_split[0].replace("'", ".")) * 1000000
mode = None mode = None
if len(freq_mode_split) > 1: if len(freq_mode_split) > 1:
mode = freq_mode_split[1].upper() mode = freq_mode_split[1].upper()

View File

@@ -1,6 +1,5 @@
from datetime import datetime
import json import json
from datetime import datetime
from data.sig_ref import SIGRef from data.sig_ref import SIGRef
from data.spot import Spot from data.spot import Spot

View File

@@ -22,8 +22,8 @@ class XOTA(WebsocketSpotProvider):
def __init__(self, provider_config): def __init__(self, provider_config):
super().__init__(provider_config, provider_config["url"]) super().__init__(provider_config, provider_config["url"])
locations_csv = provider_config["locations-csv"] if "locations-csv" in provider_config else None locations_csv = str(provider_config["locations-csv"]) if "locations-csv" in provider_config else None
self.SIG = provider_config["sig"] if "sig" in provider_config else None self.SIG = str(provider_config["sig"]) if "sig" in provider_config else None
# Load location data # Load location data
if locations_csv: if locations_csv:
@@ -48,8 +48,9 @@ class XOTA(WebsocketSpotProvider):
freq=float(source_spot["freq"]) * 1000, freq=float(source_spot["freq"]) * 1000,
mode=source_spot["mode"].upper(), mode=source_spot["mode"].upper(),
sig=self.SIG, sig=self.SIG,
sig_refs=[SIGRef(id=ref_id, sig=self.SIG, url=source_spot["reference"]["website"], latitude=lat, sig_refs=[
longitude=lon)], SIGRef(id=ref_id, sig=self.SIG or "", url=source_spot["reference"]["website"], latitude=lat,
longitude=lon)],
time=datetime.now(pytz.UTC).timestamp(), time=datetime.now(pytz.UTC).timestamp(),
dx_latitude=lat, dx_latitude=lat,
dx_longitude=lon, dx_longitude=lon,

View File

@@ -3,72 +3,204 @@
<div id="info-container" class="mt-4"> <div id="info-container" class="mt-4">
<h2 class="mt-4 mb-4">About Spothole</h2> <h2 class="mt-4 mb-4">About Spothole</h2>
<p>Spothole is a utility to aggregate "spots" from amateur radio DX clusters and xOTA spotting sites, and provide an open JSON API as well as a website to browse the data.</p> <p>Spothole is a utility to aggregate "spots" from amateur radio DX clusters and xOTA spotting sites, and provide an
<p>While there are several other web-based interfaces to DX clusters, and sites that aggregate spots from various outdoor activity programmes for amateur radio, Spothole differentiates itself by supporting a larger number of data sources, and by being "API first" rather than just providing a web front-end. This allows other software to be built on top of it.</p> open JSON API as well as a website to browse the data.</p>
<p>The API is deliberately well-defined with an <a href="/apidocs/openapi.yml">OpenAPI specification</a> and <a href="/apidocs">API documentation</a>. The API delivers spots in a consistent format regardless of the data source, freeing developers from needing to know how each individual data source presents its data.</p> <p>While there are several other web-based interfaces to DX clusters, and sites that aggregate spots from various
<p>Spothole itself is also open source, Public Domain licenced code that anyone can take and modify. <a href="https://git.ianrenton.com/ian/metaspot/">The source code is here</a>.</p> outdoor activity programmes for amateur radio, Spothole differentiates itself by supporting a larger number of
<p>The software was written by <a href="https://ianrenton.com">Ian Renton, MØTRT</a> and other contributors. Full details are available in the <a href="https://git.ianrenton.com/ian/spothole/src/branch/main/README.md">README file</a>.</p> data sources, and by being "API first" rather than just providing a web front-end. This allows other software to
be built on top of it.</p>
<p>The API is deliberately well-defined with an <a href="/apidocs/openapi.yml">OpenAPI specification</a> and <a
href="/apidocs">API documentation</a>. The API delivers spots in a consistent format regardless of the data
source, freeing developers from needing to know how each individual data source presents its data.</p>
<p>Spothole itself is also open source, Public Domain licenced code that anyone can take and modify. <a
href="https://git.ianrenton.com/ian/metaspot/">The source code is here</a>.</p>
<p>The software was written by <a href="https://ianrenton.com">Ian Renton, MØTRT</a> and other contributors. Full
details are available in the <a href="https://git.ianrenton.com/ian/spothole/src/branch/main/README.md">README
file</a>.</p>
<p>This server is running Spothole version {{software_version}}.</p> <p>This server is running Spothole version {{software_version}}.</p>
<h2 class="mt-4 mb-4">Using Spothole</h2> <h2 class="mt-4 mb-4">Using Spothole</h2>
<p>There are a number of different ways to use Spothole, depending on what you want to do with it and your level of technical skill:</p> <p>There are a number of different ways to use Spothole, depending on what you want to do with it and your level of
<ol><li>You can <b>use it on the web</b>, like you are (probably) doing right now. This is how most people use it, to look up spots and alerts, and make interesting QSOs.</li> technical skill:</p>
<li>If you are using an Android or iOS device, you can <b>"install" it on your device</b>. Spothole is a Progressive Web App, meaning it's not delivered through app stores, but if you open the page on Chrome (Android) or Safari (iOS) there will be an option in the menu to install it. It will then appear in your main app menu.</li> <ol>
<li>You can <b>embed the web interface in another website</b> to show its spots in a custom dashboard or the like. The usage is explained in more detail in the <a href="https://git.ianrenton.com/ian/spothole/src/branch/main/README.md">README file</a>.</li> <li>You can <b>use it on the web</b>, like you are (probably) doing right now. This is how most people use it,
<li>You can <b>write your own client using the Spothole API</b>, using the main Spothole instance to provide data, and do whatever you like with it. The README contains guidance on how to do this, and the full API docs are linked above. You can also find reference implementations in the form of Spothole's own web-based front end, plus my other two tools built on Spothole: <a href="https://fieldspotter.radio">Field Spotter</a> and the <a href="https://qsomap.m0trt.radio">QSO Map Tool</a>.</li> to look up spots and alerts, and make interesting QSOs.
<li>If you want to <b>run your own version of Spothole</b> so you can customise the configuration, such as enabling sources that I disable on the main instance, you can do that too. The README contains not only advice on how to set up Spothole but how to get it auto-starting with systemd, using an nginx reverse proxy, and setting up HTTPS support with certbot.</li> </li>
<li>Finally, you can of course download the source code and <b>develop Spothole to meet your needs</b>. Whether you contribute your changes back to the main repository is up to you. As usual, the README file contains some advice on the structure of the repository, and how to get started writing your own spot provider.</li></ol> <li>If you are using an Android or iOS device, you can <b>"install" it on your device</b>. Spothole is a
Progressive Web App, meaning it's not delivered through app stores, but if you open the page on Chrome
(Android) or Safari (iOS) there will be an option in the menu to install it. It will then appear in your
main app menu.
</li>
<li>You can <b>embed the web interface in another website</b> to show its spots in a custom dashboard or the
like. The usage is explained in more detail in the <a
href="https://git.ianrenton.com/ian/spothole/src/branch/main/README.md">README file</a>.
</li>
<li>You can <b>write your own client using the Spothole API</b>, using the main Spothole instance to provide
data, and do whatever you like with it. The README contains guidance on how to do this, and the full API
docs are linked above. You can also find reference implementations in the form of Spothole's own web-based
front end, plus my other two tools built on Spothole: <a href="https://fieldspotter.radio">Field Spotter</a>
and the <a href="https://qsomap.m0trt.radio">QSO Map Tool</a>.
</li>
<li>If you want to <b>run your own version of Spothole</b> so you can customise the configuration, such as
enabling sources that I disable on the main instance, you can do that too. The README contains not only
advice on how to set up Spothole but how to get it auto-starting with systemd, using an nginx reverse proxy,
and setting up HTTPS support with certbot.
</li>
<li>Finally, you can of course download the source code and <b>develop Spothole to meet your needs</b>. Whether
you contribute your changes back to the main repository is up to you. As usual, the README file contains
some advice on the structure of the repository, and how to get started writing your own spot provider.
</li>
</ol>
<h2 id="faq" class="mt-4">FAQ</h2> <h2 id="faq" class="mt-4">FAQ</h2>
<h4 class="mt-4">"Spots"? "DX Clusters"? What does any of this mean?</h4> <h4 class="mt-4">"Spots"? "DX Clusters"? What does any of this mean?</h4>
<p>This is a tool for amateur ("ham") radio users. Many amateur radio operators like to make contacts with others who are doing something more interesting than sitting in their home "shack", such as people in rarely-seen countries, remote islands, or on mountaintops. Such operators are often "spotted", i.e. when someone speaks to them, they will put the details such as their operating frequency into an online system, to let others know where to find them. A DX Cluster is one type of those systems. Most outdoor radio awards programmes, such as "Parks on the Air" (POTA) have their own websites for posting spots.</p> <p>This is a tool for amateur ("ham") radio users. Many amateur radio operators like to make contacts with others
<p>Spothole is an "aggregator" for those spots, so it checks lots of different services for data, and brings it all together in one place. So no matter what kinds of interesting spots you are looking for, you can find them here.</p> who are doing something more interesting than sitting in their home "shack", such as people in rarely-seen
<p>As well as spots, it also provides a similar feed of "alerts". This is where amateur radio users who are going to interesting places soon will announce their intentions.</p> countries, remote islands, or on mountaintops. Such operators are often "spotted", i.e. when someone speaks to
them, they will put the details such as their operating frequency into an online system, to let others know
where to find them. A DX Cluster is one type of those systems. Most outdoor radio awards programmes, such as
"Parks on the Air" (POTA) have their own websites for posting spots.</p>
<p>Spothole is an "aggregator" for those spots, so it checks lots of different services for data, and brings it all
together in one place. So no matter what kinds of interesting spots you are looking for, you can find them
here.</p>
<p>As well as spots, it also provides a similar feed of "alerts". This is where amateur radio users who are going to
interesting places soon will announce their intentions.</p>
<h4 class="mt-4">What are "DX", "DE" and modes?</h4> <h4 class="mt-4">What are "DX", "DE" and modes?</h4>
<p>In amateur radio terminology, the "DX" contact is the "interesting" one that is using the frequency shown and looking for callers. They might be on a remote island or just in a local park, but either way it's interesting enough that someone has "spotted" them. The callsign listed under "DE" is the person who entered the spot of the "DX" operator. "Modes" are the type of communication they are using. For example you might see "CW" which is Morse Code, or voice "modes" like SSB or FM, or more exotic "data" modes which are used for computer-to-computer communication.</p> <p>In amateur radio terminology, the "DX" contact is the "interesting" one that is using the frequency shown and
looking for callers. They might be on a remote island or just in a local park, but either way it's interesting
enough that someone has "spotted" them. The callsign listed under "DE" is the person who entered the spot of the
"DX" operator. "Modes" are the type of communication they are using. For example you might see "CW" which is
Morse Code, or voice "modes" like SSB or FM, or more exotic "data" modes which are used for computer-to-computer
communication.</p>
<h4 class="mt-4">What data sources are supported?</h4> <h4 class="mt-4">What data sources are supported?</h4>
<p>Spothole can retrieve spots from: <a href="https://www.dxcluster.info/telnet/">Telnet-based DX clusters</a>, the <a href="https://www.reversebeacon.net/">Reverse Beacon Network (RBN)</a>, the <a href="https://www.aprs-is.net/">APRS Internet Service (APRS-IS)</a>, <a href="https://pota.app">POTA</a>, <a href="https://www.sota.org.uk/">SOTA</a>, <a href="https://wwff.co/">WWFF</a>, <a href="https://www.cqgma.org/">GMA</a>, <a href="https://wwbota.net/">WWBOTA</a>, <a href="http://www.hema.org.uk/">HEMA</a>, <a href="https://www.parksnpeaks.org/">Parks 'n' Peaks</a>, <a href="https://ontheair.nz">ZLOTA</a>, <a href="https://www.wota.org.uk/">WOTA</a>, <a href="https://llota.app">LLOTA</a>, <a href="https://wwtota.com">WWTOTA</a>, <a href="https://tilesontheair.com/">Tiles on the Air</a>, the <a href="https://ukpacketradio.network/">UK Packet Repeater Network</a>, and any site based on the <a href="https://github.com/nischu/xOTA">xOTA software by nischu</a>.</p> <p>Spothole can retrieve spots from: <a href="https://www.dxcluster.info/telnet/">Telnet-based DX clusters</a>, the
<p>Spothole can retrieve alerts from: <a href="https://www.ng3k.com/">NG3K</a>, <a href="https://pota.app">POTA</a>, <a href="https://www.sota.org.uk/">SOTA</a>, <a href="https://wwff.co/">WWFF</a>, <a href="https://www.parksnpeaks.org/">Parks 'n' Peaks</a>, <a href="https://www.wota.org.uk/">WOTA</a> and <a href="https://www.beachesontheair.com/">BOTA</a>.</p> <a href="https://www.reversebeacon.net/">Reverse Beacon Network (RBN)</a>, the <a
<p>Spothole can retrieve solar and propagation condition data from <a href="https://www.hamqsl.com">HamQSL</a>, the <a href="https://www.swpc.noaa.gov/">NOAA Space Weather Prediction Center</a>, the <a href="https://giro.uml.edu/">Lowell GIRO Data Center</a> and <a href="https://prop.kc2g.com/">prop.kc2g.com</a> by KC2G.</p> href="https://www.aprs-is.net/">APRS Internet Service (APRS-IS)</a>, <a href="https://pota.app">POTA</a>,
<p>Spothole can also perform lookups for callsign data on behalf of the user from <a href="https://qrz.com">QRZ.com</a> and <a href="https://hamqth.com">HamQTH</a>.</p> <a href="https://www.sota.org.uk/">SOTA</a>, <a href="https://wwff.co/">WWFF</a>, <a
<p>Note that the server owner has not necessarily enabled all these data sources. In particular it is common to disable RBN, to avoid the server being swamped with FT8 traffic, and to disable APRS-IS and UK Packet Net so that the server only displays stations where there is likely to be an operator physically present for a QSO.</p> href="https://www.cqgma.org/">GMA</a>, <a href="https://wwbota.net/">WWBOTA</a>, <a
<p>Between the various data sources, the following Special Interest Groups (SIGs) are supported: Parks on the Air (POTA), Summits on the Air (SOTA), Worldwide Flora & Fauna (WWFF), Global Mountain Activity (GMA), Worldwide Bunkers on the Air (WWBOTA), HuMPs Excluding Marilyns Award (HEMA), Islands on the Air (IOTA), Mills on the Air (MOTA), the Amateur Radio Lighthouse Socirty (ARLHS), International Lighthouse Lightship Weekend (ILLW), Silos on the Air (SIOTA), World Castles Award (WCA), New Zealand on the Air (ZLOTA), Keith Roget Memorial National Parks Award (KRMNPA), Wainwrights on the Air (WOTA), Beaches on the Air (BOTA), Lagos y Lagunas On the Air (LLOTA), Towers on the Air (WWTOTA), Tiles on the Air, Worked All Britain (WAB), Worked All Ireland (WAI), and Toilets on the Air (TOTA).</p> href="http://www.hema.org.uk/">HEMA</a>, <a href="https://www.parksnpeaks.org/">Parks 'n' Peaks</a>, <a
<p>As of the time of writing in November 2025, I think Spothole captures essentially all outdoor radio programmes that have a defined reference list, and almost certainly those that have a spotting/alerting API. If you know of one I've missed, please let me know!</p> href="https://ontheair.nz">ZLOTA</a>, <a href="https://www.wota.org.uk/">WOTA</a>, <a
href="https://llota.app">LLOTA</a>, <a href="https://wwtota.com">WWTOTA</a>, <a
href="https://tilesontheair.com/">Tiles on the Air</a>, the <a href="https://ukpacketradio.network/">UK
Packet Repeater Network</a>, and any site based on the <a href="https://github.com/nischu/xOTA">xOTA
software by nischu</a>.</p>
<p>Spothole can retrieve alerts from: <a href="https://www.ng3k.com/">NG3K</a>, <a href="https://pota.app">POTA</a>,
<a href="https://www.sota.org.uk/">SOTA</a>, <a href="https://wwff.co/">WWFF</a>, <a
href="https://www.parksnpeaks.org/">Parks 'n' Peaks</a>, <a href="https://www.wota.org.uk/">WOTA</a> and
<a href="https://www.beachesontheair.com/">BOTA</a>.</p>
<p>Spothole can retrieve solar and propagation condition data from <a href="https://www.hamqsl.com">HamQSL</a>, the
<a href="https://www.swpc.noaa.gov/">NOAA Space Weather Prediction Center</a>, the <a
href="https://giro.uml.edu/">Lowell GIRO Data Center</a> and <a href="https://prop.kc2g.com/">prop.kc2g.com</a>
by KC2G.</p>
<p>Spothole can also perform lookups for callsign data on behalf of the user from <a
href="https://qrz.com">QRZ.com</a> and <a href="https://hamqth.com">HamQTH</a>.</p>
<p>Note that the server owner has not necessarily enabled all these data sources. In particular it is common to
disable RBN, to avoid the server being swamped with FT8 traffic, and to disable APRS-IS and UK Packet Net so
that the server only displays stations where there is likely to be an operator physically present for a QSO.</p>
<p>Between the various data sources, the following Special Interest Groups (SIGs) are supported: Parks on the Air
(POTA), Summits on the Air (SOTA), Worldwide Flora & Fauna (WWFF), Global Mountain Activity (GMA), Worldwide
Bunkers on the Air (WWBOTA), HuMPs Excluding Marilyns Award (HEMA), Islands on the Air (IOTA), Mills on the Air
(MOTA), the Amateur Radio Lighthouse Socirty (ARLHS), International Lighthouse Lightship Weekend (ILLW), Silos
on the Air (SIOTA), World Castles Award (WCA), New Zealand on the Air (ZLOTA), Keith Roget Memorial National
Parks Award (KRMNPA), Wainwrights on the Air (WOTA), Beaches on the Air (BOTA), Lagos y Lagunas On the Air
(LLOTA), Towers on the Air (WWTOTA), Tiles on the Air, Worked All Britain (WAB), Worked All Ireland (WAI), and
Toilets on the Air (TOTA).</p>
<p>As of the time of writing in November 2025, I think Spothole captures essentially all outdoor radio programmes
that have a defined reference list, and almost certainly those that have a spotting/alerting API. If you know of
one I've missed, please let me know!</p>
<h4 class="mt-4">Why can I filter spots by both SIG and Source? Isn't that basically the same thing?</h4> <h4 class="mt-4">Why can I filter spots by both SIG and Source? Isn't that basically the same thing?</h4>
<p>Mostly, but not quite. While POTA spots generally come from the POTA source and so on, there are a few exceptions:</p> <p>Mostly, but not quite. While POTA spots generally come from the POTA source and so on, there are a few
<ol><li>Sources like GMA and Parks 'n' Peaks provide spots for multiple different programmes (SIGs).</li> exceptions:</p>
<li>Cluster spots may name SIGs in their comment, in which case the source remains the Cluster, but a SIG is assigned.</li> <ol>
<li>Some SIGs, such as Worked all Britain (WAB), don't have their own spotting site and can <em>only</em> be identified through comments on spots retrieved from other sources.</li> <li>Sources like GMA and Parks 'n' Peaks provide spots for multiple different programmes (SIGs).</li>
<li>SIGs have well-defined names, whereas the server owner may name the sources as they see fit.</li></ol> <li>Cluster spots may name SIGs in their comment, in which case the source remains the Cluster, but a SIG is
<p>Spothole's web interface exists not just for the end user, but also as a reference implementation for the API, so I have chosen to demonstrate both methods of filtering.</p> assigned.
</li>
<li>Some SIGs, such as Worked all Britain (WAB), don't have their own spotting site and can <em>only</em> be
identified through comments on spots retrieved from other sources.
</li>
<li>SIGs have well-defined names, whereas the server owner may name the sources as they see fit.</li>
</ol>
<p>Spothole's web interface exists not just for the end user, but also as a reference implementation for the API, so
I have chosen to demonstrate both methods of filtering.</p>
<h4 class="mt-4">How is this better than DXheat, DXsummit, POTA's own website, etc?</h4> <h4 class="mt-4">How is this better than DXheat, DXsummit, POTA's own website, etc?</h4>
<p>It's probably not? But it's nice to have choice.</p> <p>It's probably not? But it's nice to have choice.</p>
<p>I think it's got three key advantages over those sites:</p> <p>I think it's got three key advantages over those sites:</p>
<ol><li>It provides a public, <a href="/apidocs">well-documented API</a> with an <a href="/apidocs/openapi.yml">OpenAPI specification</a>. Other sites don't have official APIs or don't bother documenting them publicly, because they want people to use their web page. I like Spothole's web page, but you don't have to use it&mdash;if you're a programmer, you can build your own software on Spothole's API. Spothole does the hard work of taking all the various data sources and providing a consistent, well-documented data set. You can then do the fun bit of writing your own application.</li> <ol>
<li>It grabs data from a lot more sources. I've seen other sites that pull in DX Cluster and POTA spots together, but nothing on the scale of what Spothole supports.</li> <li>It provides a public, <a href="/apidocs">well-documented API</a> with an <a href="/apidocs/openapi.yml">OpenAPI
<li>Spothole is open source, so anyone can contribute the code to support a new data source or add new features, and share them with the community.</li></ol> specification</a>. Other sites don't have official APIs or don't bother documenting them publicly, because
they want people to use their web page. I like Spothole's web page, but you don't have to use it&mdash;if
you're a programmer, you can build your own software on Spothole's API. Spothole does the hard work of
taking all the various data sources and providing a consistent, well-documented data set. You can then do
the fun bit of writing your own application.
</li>
<li>It grabs data from a lot more sources. I've seen other sites that pull in DX Cluster and POTA spots
together, but nothing on the scale of what Spothole supports.
</li>
<li>Spothole is open source, so anyone can contribute the code to support a new data source or add new features,
and share them with the community.
</li>
</ol>
<h4 class="mt-4">Why does this website ask me if I want to install it?</h4> <h4 class="mt-4">Why does this website ask me if I want to install it?</h4>
<p>Spothole is a Progressive Web App, which means you can install it on an Android or iOS device by opening the site in Chrome or Safari respectively, and clicking "Install" on the pop-up panel. It'll only prompt you once, so if you dismiss the prompt and change your mind, you'll find an Install / Add to Home Screen option on your browser's menu.</p> <p>Spothole is a Progressive Web App, which means you can install it on an Android or iOS device by opening the site
<p>Installing Spothole on your phone is completely optional, the website works exactly the same way as the "app" does.</p> in Chrome or Safari respectively, and clicking "Install" on the pop-up panel. It'll only prompt you once, so if
you dismiss the prompt and change your mind, you'll find an Install / Add to Home Screen option on your
browser's menu.</p>
<p>Installing Spothole on your phone is completely optional, the website works exactly the same way as the "app"
does.</p>
<h4 class="mt-4">Why hasn't my spot/alert shown up yet?</h4> <h4 class="mt-4">Why hasn't my spot/alert shown up yet?</h4>
<p>To avoid putting too much load on the various servers that Spothole connects to, the Spothole server only polls them once every two minutes for spots, and once every 30 minutes for alerts. (Some sources, such as DX clusters, RBN, APRS-IS and WWBOTA use a non-polling mechanism, and their updates will therefore arrive more quickly.) Then if you are using the web interface, that has its own rate at which it fetches the data from Spothole. This is instant for the main spots list, with new spots appearing immediately at the top of the list, while the map and bands displays update once a minute, and the alerts display updates once every 5 minutes. So you could be waiting around three minutes to see a newly added spot, or 40 minutes to see a newly added alert.</p> <p>To avoid putting too much load on the various servers that Spothole connects to, the Spothole server only polls
them once every two minutes for spots, and once every 30 minutes for alerts. (Some sources, such as DX clusters,
RBN, APRS-IS and WWBOTA use a non-polling mechanism, and their updates will therefore arrive more quickly.) Then
if you are using the web interface, that has its own rate at which it fetches the data from Spothole. This is
instant for the main spots list, with new spots appearing immediately at the top of the list, while the map and
bands displays update once a minute, and the alerts display updates once every 5 minutes. So you could be
waiting around three minutes to see a newly added spot, or 40 minutes to see a newly added alert.</p>
<h4 class="mt-4">What licence does Spothole use?</h4> <h4 class="mt-4">What licence does Spothole use?</h4>
<p>Spothole's source code is licenced under the Public Domain. You can write a Spothole client, run your own server, modify it however you like, you can claim you wrote it and charge people £1000 for a copy, I don't really mind. (Please don't do the last one. But if you're using my code for something cool, it would be nice to hear from you!)</p> <p>Spothole's source code is licenced under the Public Domain. You can write a Spothole client, run your own server,
modify it however you like, you can claim you wrote it and charge people £1000 for a copy, I don't really mind.
(Please don't do the last one. But if you're using my code for something cool, it would be nice to hear from
you!)</p>
<h2 class="mt-4">Data Accuracy</h2> <h2 class="mt-4">Data Accuracy</h2>
<p>Please note that the data coming out of Spothole is only as good as the data going in. People mis-hear and make typos when spotting callsigns all the time. There are also plenty of cases where Spothole's data, particularly location data, may be inaccurate. For example, there are POTA parks that span multiple US states, countries that span multiple CQ zones, portable operators with no requirement to sign /P, etc. If you are doing something where accuracy is important, such as contesting, you should not rely on Spothole's data to fill in any gaps in your log.</p> <p>Please note that the data coming out of Spothole is only as good as the data going in. People mis-hear and make
typos when spotting callsigns all the time. There are also plenty of cases where Spothole's data, particularly
location data, may be inaccurate. For example, there are POTA parks that span multiple US states, countries that
span multiple CQ zones, portable operators with no requirement to sign /P, etc. If you are doing something where
accuracy is important, such as contesting, you should not rely on Spothole's data to fill in any gaps in your
log.</p>
<h2 id="privacy" class="mt-4">Privacy</h2> <h2 id="privacy" class="mt-4">Privacy</h2>
<p>Spothole collects no data about you on a permanent basis. All spots and alerts are "timed out" and deleted from the system after a set interval, which by default is one hour for spots and one week for alerts.</p> <p>Spothole collects no data about you on a permanent basis. All spots and alerts are "timed out" and deleted from
<p>Settings you select from Spothole's menus are sent to the server, in order to provide the data with the requested filters. They are also stored in your browser's local storage, so that your preferences are remembered between sessions.</p> the system after a set interval, which by default is one hour for spots and one week for alerts.</p>
<p>The data you provide can optionally include your login credentials for QRZ.com and HamQTH. You can provide these in the "Data" menu of most pages. If you do, Spothole will augment the data it produces with lookups from these services, which can for example provide more accurate markers on the map tab, and operator names when you mouse over a DX callsign. Spothole will still work fine if you don't provide these. The values you enter are sent to Spothole via HTTPS so are protected in transit, though of course you do have to trust Spothole with this sensitive data in order to use this feature.</p> <p>Settings you select from Spothole's menus are sent to the server, in order to provide the data with the requested
filters. They are also stored in your browser's local storage, so that your preferences are remembered between
sessions.</p>
<p>The data you provide can optionally include your login credentials for QRZ.com and HamQTH. You can provide these
in the "Data" menu of most pages. If you do, Spothole will augment the data it produces with lookups from these
services, which can for example provide more accurate markers on the map tab, and operator names when you mouse
over a DX callsign. Spothole will still work fine if you don't provide these. The values you enter are sent to
Spothole via HTTPS so are protected in transit, though of course you do have to trust Spothole with this
sensitive data in order to use this feature.</p>
<p>Spothole uses no trackers, no ads, and no cookies.</p> <p>Spothole uses no trackers, no ads, and no cookies.</p>
{% if len(web_ui_options["support-button-html"]) > 0 %} {% if len(web_ui_options["support-button-html"]) > 0 %}
<p><strong>Caveat: </strong> The owner of this server has chosen to inject their own content into the "spots" page. This is designed for a "donate" or "support this server" button. The functionality of this injected content is the responsibility of the server owner, rather than the Spothole software.</p> <p><strong>Caveat: </strong> The owner of this server has chosen to inject their own content into the "spots" page.
This is designed for a "donate" or "support this server" button. The functionality of this injected content is
the responsibility of the server owner, rather than the Spothole software.</p>
{% end %} {% end %}
<p>Spothole is open source, so you can audit <a href="https://git.ianrenton.com/ian/spothole">the code</a> if you like.</p> <p>Spothole is open source, so you can audit <a href="https://git.ianrenton.com/ian/spothole">the code</a> if you
like.</p>
<h2 class="mt-4">Thanks</h2> <h2 class="mt-4">Thanks</h2>
<p>This project would not have been possible without those volunteers who have taken it upon themselves to run DX clusters, xOTA programmes, DXpedition lists, callsign lookup databases, solar conditions and propagation modelling software, and other online tools on which Spothole's data is based. The vast majority of these are not profit-seeking and are made purely for the love of the hobby and to help others in the community. Spothole is standing on the shoulders of giants, who deserve a huge amount of thanks for all the work they put in.</p> <p>This project would not have been possible without those volunteers who have taken it upon themselves to run DX
<p>Spothole is also dependent on a number of Python libraries, in particular pyhamtools, and many JavaScript libraries, as well as the Font Awesome icon set and flag icons from the Noto Color Emoji set, and MIT-licenced GeoJSON files for CQ and ITU zones from HA8TKS.</p> clusters, xOTA programmes, DXpedition lists, callsign lookup databases, solar conditions and propagation
<p>This software is dedicated to the memory of Tom G1PJB, SK, a friend and colleague who sadly passed away around the time I started writing it in Autumn 2025. I was looking forward to showing it to you when it was done.</p> modelling software, and other online tools on which Spothole's data is based. The vast majority of these are not
profit-seeking and are made purely for the love of the hobby and to help others in the community. Spothole is
standing on the shoulders of giants, who deserve a huge amount of thanks for all the work they put in.</p>
<p>Spothole is also dependent on a number of Python libraries, in particular pyhamtools, and many JavaScript
libraries, as well as the Font Awesome icon set and flag icons from the Noto Color Emoji set, and MIT-licenced
GeoJSON files for CQ and ITU zones from HA8TKS.</p>
<p>This software is dedicated to the memory of Tom G1PJB, SK, a friend and colleague who sadly passed away around
the time I started writing it in Autumn 2025. I was looking forward to showing it to you when it was done.</p>
</div> </div>
<script>$(document).ready(function() { $("#nav-link-about").addClass("active"); }); <!-- highlight active page in nav --></script> <script>$(document).ready(function () {
$("#nav-link-about").addClass("active");
}); <!-- highlight active page in nav --></script>
{% end %} {% end %}

View File

@@ -2,10 +2,16 @@
{% block content %} {% block content %}
<div id="add-spot-intro-box" class="permanently-dismissible-box mt-3"> <div id="add-spot-intro-box" class="permanently-dismissible-box mt-3">
<div class="alert alert-primary alert-dismissible fade show" role="alert"> <div class="alert alert-primary alert-dismissible fade show" role="alert">
<i class="fa-solid fa-circle-info"></i> <strong>Adding spots to Spothole</strong><br/>This page is implemented as a proof of concept for adding spots to the Spothole system. Currently, spots added in this way are only visible within Spothole and are not sent "upstream" to DX clusters or xOTA spotting sites. The functionality might be extended to include this in future if there is demand for it. If you'd like this to be added, please give a thumbs-up on <a href="https://git.ianrenton.com/ian/spothole/issues/39" target="_new" class="alert-link">issue #39</a> or get in touch via email. <i class="fa-solid fa-circle-info"></i> <strong>Adding spots to Spothole</strong><br/>This page is implemented
<button type="button" id="add-spot-intro-box-dismiss" class="btn-close" data-bs-dismiss="alert" aria-label="Close"></button> as a proof of concept for adding spots to the Spothole system. Currently, spots added in this way are only
</div> visible within Spothole and are not sent "upstream" to DX clusters or xOTA spotting sites. The functionality
might be extended to include this in future if there is demand for it. If you'd like this to be added, please
give a thumbs-up on <a href="https://git.ianrenton.com/ian/spothole/issues/39" target="_new" class="alert-link">issue
#39</a> or get in touch via email.
<button type="button" id="add-spot-intro-box-dismiss" class="btn-close" data-bs-dismiss="alert"
aria-label="Close"></button>
</div>
</div> </div>
<div class="mt-3"> <div class="mt-3">
@@ -53,7 +59,8 @@
</div> </div>
<div class="col-auto"> <div class="col-auto">
<label for="de-call" class="form-label">Your Call *</label> <label for="de-call" class="form-label">Your Call *</label>
<input type="text" class="form-control storeable-text input-narrow" id="de-call" placeholder="N0CALL"> <input type="text" class="form-control storeable-text input-narrow" id="de-call"
placeholder="N0CALL">
</div> </div>
<div class="col-auto"> <div class="col-auto">
<button type="button" class="btn btn-primary mt-2em" onclick="addSpot();">Spot</button> <button type="button" class="btn btn-primary mt-2em" onclick="addSpot();">Spot</button>
@@ -69,7 +76,9 @@
</div> </div>
<script src="/js/add-spot.js?v=1781811406"></script> <script src="/js/add-spot.js?v=1781901371"></script>
<script>$(document).ready(function() { $("#nav-link-add-spot").addClass("active"); }); <!-- highlight active page in nav --></script> <script>$(document).ready(function () {
$("#nav-link-add-spot").addClass("active");
}); <!-- highlight active page in nav --></script>
{% end %} {% end %}

View File

@@ -65,12 +65,19 @@
</div> </div>
<div id="table-container"> <div id="table-container">
<table id="table" class="table"><thead><tr></tr></thead><tbody></tbody></table> <table id="table" class="table">
<thead>
<tr></tr>
</thead>
<tbody></tbody>
</table>
</div> </div>
</div> </div>
<script src="/js/alerts.js?v=1781811406"></script> <script src="/js/alerts.js?v=1781901371"></script>
<script>$(document).ready(function() { $("#nav-link-alerts").addClass("active"); }); <!-- highlight active page in nav --></script> <script>$(document).ready(function () {
$("#nav-link-alerts").addClass("active");
}); <!-- highlight active page in nav --></script>
{% end %} {% end %}

View File

@@ -1,6 +1,6 @@
{% extends "skeleton.html" %} {% extends "skeleton.html" %}
{% block head_extra %} {% block head_extra %}
<link href="/vendor/css/bootstrap-5.3.8.min.css" rel="stylesheet"> <link href="/vendor/css/bootstrap-5.3.8.min.css" rel="stylesheet">
{% end %} {% end %}
{% block body %} {% block body %}
<div class="container mt-5"> <div class="container mt-5">
@@ -11,9 +11,15 @@
</div> </div>
<div class="card"> <div class="card">
<div class="card-body"> <div class="card-body">
<p class="card-text">This server is running <strong>Spothole v{{software_version}}</strong>, and is operated by <strong>{{server_owner_callsign}}</strong>.</p> <p class="card-text">This server is running <strong>Spothole v{{software_version}}</strong>, and is
<p class="card-text">The web UI is not available on this instance because the server is running in API-only mode, intended for use by client software rather than visitors to the website. See the <a href="/apidocs">API documentation</a> for details of how client software can interact with the server.</p> operated by <strong>{{server_owner_callsign}}</strong>.</p>
<p class="card-text">Please see the <a href="https://git.ianrenton.com/ian/spothole#readme">README</a> for details of what Spothole is and how you can run it for yourself.</p> <p class="card-text">The web UI is not available on this instance because the server is running in
API-only mode, intended for use by client software rather than visitors to the website. See the
<a href="/apidocs">API documentation</a> for details of how client software can interact with
the server.</p>
<p class="card-text">Please see the <a
href="https://git.ianrenton.com/ian/spothole#readme">README</a> for details of what Spothole
is and how you can run it for yourself.</p>
</div> </div>
</div> </div>
</div> </div>

View File

@@ -1,5 +1,5 @@
{% extends "skeleton.html" %} {% extends "skeleton.html" %}
{% block body %} {% block body %}
<redoc spec-url="/apidocs/openapi.yml"></redoc> <redoc spec-url="/apidocs/openapi.yml"></redoc>
<script src="https://cdn.redoc.ly/redoc/latest/bundles/redoc.standalone.js"> </script> <script src="https://cdn.redoc.ly/redoc/latest/bundles/redoc.standalone.js"></script>
{% end %} {% end %}

View File

@@ -49,7 +49,8 @@
{% module Template("cards/spot-age.html", web_ui_options=web_ui_options) %} {% module Template("cards/spot-age.html", web_ui_options=web_ui_options) %}
</div> </div>
<div class="col"> <div class="col">
{% module Template("cards/color-scheme-and-band-color-scheme.html", web_ui_options=web_ui_options) %} {% module Template("cards/color-scheme-and-band-color-scheme.html", web_ui_options=web_ui_options)
%}
</div> </div>
</div> </div>
</div> </div>
@@ -74,10 +75,15 @@
</div> </div>
<script> <script>
let spotProvidersEnabledByDefault = {% raw json_encode(web_ui_options["spot-providers-enabled-by-default"]) %}; let spotProvidersEnabledByDefault = { % raw
json_encode(web_ui_options["spot-providers-enabled-by-default"]) %
}
;
</script> </script>
<script src="/js/spotsbandsandmap.js?v=1781811406"></script> <script src="/js/spotsbandsandmap.js?v=1781901371"></script>
<script src="/js/bands.js?v=1781811406"></script> <script src="/js/bands.js?v=1781901371"></script>
<script>$(document).ready(function() { $("#nav-link-bands").addClass("active"); }); <!-- highlight active page in nav --></script> <script>$(document).ready(function () {
$("#nav-link-bands").addClass("active");
}); <!-- highlight active page in nav --></script>
{% end %} {% end %}

View File

@@ -1,19 +1,19 @@
{% extends "skeleton.html" %} {% extends "skeleton.html" %}
{% block head_extra %} {% block head_extra %}
<link rel="stylesheet" href="/css/style.css?v=1781811406" type="text/css"> <link rel="stylesheet" href="/css/style.css?v=1781901371" type="text/css">
<link href="/vendor/css/bootstrap-5.3.8.min.css" rel="stylesheet"> <link href="/vendor/css/bootstrap-5.3.8.min.css" rel="stylesheet">
<link href="/vendor/css/fontawesome-6.7.2.min.css" rel="stylesheet"> <link href="/vendor/css/fontawesome-6.7.2.min.css" rel="stylesheet">
<link href="/vendor/css/solid-6.7.2.min.css" rel="stylesheet"> <link href="/vendor/css/solid-6.7.2.min.css" rel="stylesheet">
<script src="/vendor/js/jquery-3.7.1.min.js"></script> <script src="/vendor/js/jquery-3.7.1.min.js"></script>
<script src="/vendor/js/moment-2.29.4.min.js"></script> <script src="/vendor/js/moment-2.29.4.min.js"></script>
<script src="/vendor/js/bootstrap-5.3.8.bundle.min.js"></script> <script src="/vendor/js/bootstrap-5.3.8.bundle.min.js"></script>
<script src="/vendor/js/tinycolor2-1.6.0.min.js"></script> <script src="/vendor/js/tinycolor2-1.6.0.min.js"></script>
<script src="/js/utils.js?v=1781811406"></script> <script src="/js/utils.js?v=1781901371"></script>
<script src="/js/ui-ham.js?v=1781811406"></script> <script src="/js/ui-ham.js?v=1781901371"></script>
<script src="/js/geo.js?v=1781811406"></script> <script src="/js/geo.js?v=1781901371"></script>
<script src="/js/common.js?v=1781811406"></script> <script src="/js/common.js?v=1781901371"></script>
{% end %} {% end %}
{% block body %} {% block body %}
<div class="container"> <div class="container">
@@ -22,22 +22,33 @@
<a class="navbar-brand" href="/"> <a class="navbar-brand" href="/">
<img src="/img/logo.png" class="logo" width="192" height="60" alt="Spothole"> <img src="/img/logo.png" class="logo" width="192" height="60" alt="Spothole">
</a> </a>
<button class="navbar-toggler" type="button" data-bs-toggle="collapse" data-bs-target="#navbar-toggler-content" aria-controls="navbar-toggler-content" aria-expanded="false" aria-label="Toggle navigation"> <button class="navbar-toggler" type="button" data-bs-toggle="collapse"
<span class="navbar-toggler-icon"></span> data-bs-target="#navbar-toggler-content" aria-controls="navbar-toggler-content"
aria-expanded="false" aria-label="Toggle navigation">
<span class="navbar-toggler-icon"></span>
</button> </button>
<div class="collapse navbar-collapse" id="navbar-toggler-content"> <div class="collapse navbar-collapse" id="navbar-toggler-content">
<ul class="navbar-nav me-auto mb-2 mb-lg-0"> <ul class="navbar-nav me-auto mb-2 mb-lg-0">
<li class="nav-item ms-4"><a href="/" class="nav-link" id="nav-link-spots"><i class="fa-solid fa-tower-cell"></i> Spots</a></li> <li class="nav-item ms-4"><a href="/" class="nav-link" id="nav-link-spots"><i
<li class="nav-item ms-4"><a href="/map" class="nav-link" id="nav-link-map"><i class="fa-solid fa-map"></i> Map</a></li> class="fa-solid fa-tower-cell"></i> Spots</a></li>
<li class="nav-item ms-4"><a href="/bands" class="nav-link" id="nav-link-bands"><i class="fa-solid fa-ruler-vertical"></i> Bands</a></li> <li class="nav-item ms-4"><a href="/map" class="nav-link" id="nav-link-map"><i
<li class="nav-item ms-4"><a href="/alerts" class="nav-link" id="nav-link-alerts"><i class="fa-solid fa-clock"></i> Upcoming</a></li> class="fa-solid fa-map"></i> Map</a></li>
{% if allow_spotting %} <li class="nav-item ms-4"><a href="/bands" class="nav-link" id="nav-link-bands"><i
<li class="nav-item ms-4"><a href="/add-spot" class="nav-link" id="nav-link-add-spot"><i class="fa-solid fa-comment"></i> Add&nbsp;Spot</a></li> class="fa-solid fa-ruler-vertical"></i> Bands</a></li>
{% end %} <li class="nav-item ms-4"><a href="/alerts" class="nav-link" id="nav-link-alerts"><i
<li class="nav-item ms-4"><a href="/conditions" class="nav-link" id="nav-link-conditions"><i class="fa-solid fa-sun"></i> Conditions</a></li> class="fa-solid fa-clock"></i> Upcoming</a></li>
<li class="nav-item ms-4"><a href="/status" class="nav-link" id="nav-link-status"><i class="fa-solid fa-chart-simple"></i> Status</a></li> {% if allow_spotting %}
<li class="nav-item ms-4"><a href="/about" class="nav-link" id="nav-link-about"><i class="fa-solid fa-circle-info"></i> About</a></li> <li class="nav-item ms-4"><a href="/add-spot" class="nav-link" id="nav-link-add-spot"><i
<li class="nav-item ms-4"><a href="/apidocs" class="nav-link" id="nav-link-api"><i class="fa-solid fa-gear"></i> API</a></li> class="fa-solid fa-comment"></i> Add&nbsp;Spot</a></li>
{% end %}
<li class="nav-item ms-4"><a href="/conditions" class="nav-link" id="nav-link-conditions"><i
class="fa-solid fa-sun"></i> Conditions</a></li>
<li class="nav-item ms-4"><a href="/status" class="nav-link" id="nav-link-status"><i
class="fa-solid fa-chart-simple"></i> Status</a></li>
<li class="nav-item ms-4"><a href="/about" class="nav-link" id="nav-link-about"><i
class="fa-solid fa-circle-info"></i> About</a></li>
<li class="nav-item ms-4"><a href="/apidocs" class="nav-link" id="nav-link-api"><i
class="fa-solid fa-gear"></i> API</a></li>
</ul> </ul>
</div> </div>
</div> </div>
@@ -45,14 +56,17 @@
<main> <main>
{% block content %}{% end %} {% block content %}{% end %}
</main> </main>
<div id="footer" class="hideonmobile hideonmap"> <div id="footer" class="hideonmobile hideonmap">
<footer class="d-flex flex-wrap justify-content-between align-items-center py-3 my-4 border-top"> <footer class="d-flex flex-wrap justify-content-between align-items-center py-3 my-4 border-top">
<p class="col-md-4 mb-0 text-body-secondary">Made with love by <a href="https://ianrenton.com" class="text-body-secondary">Ian, MØTRT</a> and other contributors.</p> <p class="col-md-4 mb-0 text-body-secondary">Made with love by <a href="https://ianrenton.com"
<p class="col-md-4 mb-0 justify-content-center text-body-secondary text-center">Spothole v{{software_version}}</p> class="text-body-secondary">Ian, MØTRT</a>
and other contributors.</p>
<p class="col-md-4 mb-0 justify-content-center text-body-secondary text-center">Spothole
v{{software_version}}</p>
<ul class="nav col-md-4 justify-content-end"> <ul class="nav col-md-4 justify-content-end">
<li class="nav-item"> <li class="nav-item">
<a href="/about#faq" class="nav-link px-3 text-body-secondary">FAQ</a> <a href="/about#faq" class="nav-link px-3 text-body-secondary">FAQ</a>
@@ -61,16 +75,20 @@
<a href="/about#privacy" class="nav-link px-3 text-body-secondary">Privacy</a> <a href="/about#privacy" class="nav-link px-3 text-body-secondary">Privacy</a>
</li> </li>
<li class="nav-item"> <li class="nav-item">
<a href="https://git.ianrenton.com/ian/spothole" class="nav-link px-3 text-body-secondary">Source Code</a> <a href="https://git.ianrenton.com/ian/spothole" class="nav-link px-3 text-body-secondary">Source
Code</a>
</li> </li>
<li class="nav-item"> <li class="nav-item">
<a href="https://git.ianrenton.com/ian/spothole/issues" class="nav-link px-3 text-body-secondary">Issue Tracker</a> <a href="https://git.ianrenton.com/ian/spothole/issues" class="nav-link px-3 text-body-secondary">Issue
Tracker</a>
</li> </li>
</ul> </ul>
</footer> </footer>
</div> </div>
</div> </div>
<div id="embeddedModeFooter" class="text-body-secondary pt-2 px-3 pb-1">Powered by <img src="/img/logo.png" class="logo" width="96" height="30" alt="Spothole"></div> <div id="embeddedModeFooter" class="text-body-secondary pt-2 px-3 pb-1">Powered by <img src="/img/logo.png" class="logo"
width="96" height="30"
alt="Spothole"></div>
{% end %} {% end %}

View File

@@ -3,8 +3,9 @@
<h5 class="card-title mb-3">Audio</h5> <h5 class="card-title mb-3">Audio</h5>
<div class="form-group"> <div class="form-group">
<div class="form-check form-check-inline"> <div class="form-check form-check-inline">
<input class="form-check-input storeable-checkbox" type="checkbox" id="pingOnNewSpots" value="pingOnNewSpots" oninput="saveSettings();"> <input class="form-check-input storeable-checkbox" type="checkbox" id="pingOnNewSpots"
<label class="form-check-label" for="pingOnNewSpots">Ping on new spots</label> value="pingOnNewSpots" oninput="saveSettings();">
<label class="form-check-label" for="pingOnNewSpots">Ping on new spots</label>
</div> </div>
</div> </div>
</div> </div>

View File

@@ -1,9 +1,12 @@
<div class="card"> <div class="card">
<div class="card-body"> <div class="card-body">
<h5 class="card-title">Duration Limit <i class='fa-solid fa-circle-question' title='Some users create long-duration alerts for the period they will be generally in and around xOTA references, when they are not indending to be on the air most of the time. Use this control to restrict the maximum duration of spots that the software will display, and exclude any with a long duration, to avoid these filling up the list. By default, we allow DXpeditions to be displayed even if they are longer than this limit, because on a DXpedition the operators typically ARE on the air most of the time.'></i></h5> <h5 class="card-title">Duration Limit <i class='fa-solid fa-circle-question'
title='Some users create long-duration alerts for the period they will be generally in and around xOTA references, when they are not indending to be on the air most of the time. Use this control to restrict the maximum duration of spots that the software will display, and exclude any with a long duration, to avoid these filling up the list. By default, we allow DXpeditions to be displayed even if they are longer than this limit, because on a DXpedition the operators typically ARE on the air most of the time.'></i>
</h5>
<p class="card-text spothole-card-text"> <p class="card-text spothole-card-text">
<label for="max-duration" class="form-label">Hide any alerts lasting more than</label> <label for="max-duration" class="form-label">Hide any alerts lasting more than</label>
<select id="max-duration" class="storeable-select form-select" onclick="filtersUpdated();" style="width: 8em; display: inline-block;"> <select id="max-duration" class="storeable-select form-select" onclick="filtersUpdated();"
style="width: 8em; display: inline-block;">
<option value="10800">3 hours</option> <option value="10800">3 hours</option>
<option value="43200">12 hours</option> <option value="43200">12 hours</option>
<option value="86400" selected>24 hours</option> <option value="86400" selected>24 hours</option>
@@ -13,7 +16,10 @@
</select> </select>
</p> </p>
<p class='card-text spothole-card-text' style='line-height: 1.5em !important;'> <p class='card-text spothole-card-text' style='line-height: 1.5em !important;'>
<input class="form-check-input storeable-checkbox" type="checkbox" value="" onclick="filtersUpdated();" id="dxpeditions_skip_max_duration_check" checked><label class="form-check-label ms-2" for="dxpeditions_skip_max_duration_check">Allow DXpeditions that are longer</label> <input class="form-check-input storeable-checkbox" type="checkbox" value="" onclick="filtersUpdated();"
id="dxpeditions_skip_max_duration_check" checked><label class="form-check-label ms-2"
for="dxpeditions_skip_max_duration_check">Allow
DXpeditions that are longer</label>
</p> </p>
</div> </div>
</div> </div>

View File

@@ -3,21 +3,28 @@
<h5 class="card-title">HamQTH</h5> <h5 class="card-title">HamQTH</h5>
<div class="card-text spothole-card-text"> <div class="card-text spothole-card-text">
<div class="form-check mb-2"> <div class="form-check mb-2">
<input type="checkbox" class="storeable-checkbox form-check-input" id="hamqth-enabled" onchange="saveSettings();"> <input type="checkbox" class="storeable-checkbox form-check-input" id="hamqth-enabled"
onchange="saveSettings();">
<label for="hamqth-enabled" class="form-check-label">Use data from HamQTH</label> <label for="hamqth-enabled" class="form-check-label">Use data from HamQTH</label>
</div> </div>
<div class="mb-2"> <div class="mb-2">
<input type="text" class="storeable-text form-control" id="hamqth-username" placeholder="Username (Callsign)" onchange="saveSettings();" autocomplete="username"> <input type="text" class="storeable-text form-control" id="hamqth-username"
placeholder="Username (Callsign)" onchange="saveSettings();" autocomplete="username">
</div> </div>
<div class="mb-2"> <div class="mb-2">
<input type="password" class="password-field form-control" id="hamqth-password" placeholder="Password" data-remember-checkbox="hamqth-remember-password" onchange="saveSettings();" autocomplete="current-password"> <input type="password" class="password-field form-control" id="hamqth-password" placeholder="Password"
data-remember-checkbox="hamqth-remember-password" onchange="saveSettings();"
autocomplete="current-password">
</div> </div>
<div class="form-check"> <div class="form-check">
<input type="checkbox" class="storeable-checkbox form-check-input" id="hamqth-remember-password" onchange="saveSettings();"> <input type="checkbox" class="storeable-checkbox form-check-input" id="hamqth-remember-password"
onchange="saveSettings();">
<label for="hamqth-remember-password" class="form-check-label">Remember password</label> <label for="hamqth-remember-password" class="form-check-label">Remember password</label>
</div> </div>
<div class="mt-3"> <div class="mt-3">
<button type="button" class="btn btn-outline-secondary btn-sm" onclick="location.reload();">Reload with this data</button> <button type="button" class="btn btn-outline-secondary btn-sm" onclick="location.reload();">Reload with
this data
</button>
</div> </div>
<div class="mt-1"> <div class="mt-1">
<small>See <a href="/about#privacy">Privacy</a> for more information.</small> <small>See <a href="/about#privacy">Privacy</a> for more information.</small>

View File

@@ -1,9 +1,10 @@
<div class="card"> <div class="card">
<div class="card-body"> <div class="card-body">
<h5 class="card-title">Location</h5> <h5 class="card-title">Location</h5>
<div class="form-group spothole-card-text"> <div class="form-group spothole-card-text">
<label for="userGrid">Your grid:</label> <label for="userGrid">Your grid:</label>
<input type="text" class="storeable-text form-control" id="userGrid" placeholder="AA00aa" oninput="userGridUpdated();" style="width: 10em; display: inline-block;"> <input type="text" class="storeable-text form-control" id="userGrid" placeholder="AA00aa"
</div> oninput="userGridUpdated();" style="width: 10em; display: inline-block;">
</div>
</div> </div>
</div> </div>

View File

@@ -3,37 +3,43 @@
<h5 class="card-title mb-3">Map Features</h5> <h5 class="card-title mb-3">Map Features</h5>
<div class="form-group"> <div class="form-group">
<div class="form-check form-check-inline"> <div class="form-check form-check-inline">
<input class="form-check-input storeable-checkbox" type="checkbox" id="mapShowGeodesics" value="mapShowGeodesics" oninput="displayUpdated();"> <input class="form-check-input storeable-checkbox" type="checkbox" id="mapShowGeodesics"
<label class="form-check-label" for="mapShowGeodesics">Geodesic Lines</label> value="mapShowGeodesics" oninput="displayUpdated();">
<label class="form-check-label" for="mapShowGeodesics">Geodesic Lines</label>
</div> </div>
</div> </div>
<div class="form-group"> <div class="form-group">
<div class="form-check form-check-inline"> <div class="form-check form-check-inline">
<input class="form-check-input storeable-checkbox" type="checkbox" id="showTerminator" oninput="displayUpdated();" checked> <input class="form-check-input storeable-checkbox" type="checkbox" id="showTerminator"
oninput="displayUpdated();" checked>
<label class="form-check-label" for="showTerminator">Terminator / Greyline</label> <label class="form-check-label" for="showTerminator">Terminator / Greyline</label>
</div> </div>
</div> </div>
<div class="form-group"> <div class="form-group">
<div class="form-check form-check-inline"> <div class="form-check form-check-inline">
<input class="form-check-input storeable-checkbox" type="checkbox" id="showMaidenheadGrid" oninput="displayUpdated();"> <input class="form-check-input storeable-checkbox" type="checkbox" id="showMaidenheadGrid"
oninput="displayUpdated();">
<label class="form-check-label" for="showMaidenheadGrid">Maidenhead Grid</label> <label class="form-check-label" for="showMaidenheadGrid">Maidenhead Grid</label>
</div> </div>
</div> </div>
<div class="form-group"> <div class="form-group">
<div class="form-check form-check-inline"> <div class="form-check form-check-inline">
<input class="form-check-input storeable-checkbox" type="checkbox" id="showCQZones" oninput="displayUpdated();"> <input class="form-check-input storeable-checkbox" type="checkbox" id="showCQZones"
oninput="displayUpdated();">
<label class="form-check-label" for="showCQZones">CQ Zones</label> <label class="form-check-label" for="showCQZones">CQ Zones</label>
</div> </div>
</div> </div>
<div class="form-group"> <div class="form-group">
<div class="form-check form-check-inline"> <div class="form-check form-check-inline">
<input class="form-check-input storeable-checkbox" type="checkbox" id="showITUZones" oninput="displayUpdated();"> <input class="form-check-input storeable-checkbox" type="checkbox" id="showITUZones"
oninput="displayUpdated();">
<label class="form-check-label" for="showITUZones">ITU Zones</label> <label class="form-check-label" for="showITUZones">ITU Zones</label>
</div> </div>
</div> </div>
<div class="form-group"> <div class="form-group">
<div class="form-check form-check-inline"> <div class="form-check form-check-inline">
<input class="form-check-input storeable-checkbox" type="checkbox" id="showWABWAIGrid" oninput="displayUpdated();"> <input class="form-check-input storeable-checkbox" type="checkbox" id="showWABWAIGrid"
oninput="displayUpdated();">
<label class="form-check-label" for="showWABWAIGrid">WAB/WAI Grid</label> <label class="form-check-label" for="showWABWAIGrid">WAB/WAI Grid</label>
</div> </div>
</div> </div>

View File

@@ -2,10 +2,12 @@
<div class="card-body"> <div class="card-body">
<h5 class="card-title">Number of Alerts</h5> <h5 class="card-title">Number of Alerts</h5>
<p class="card-text spothole-card-text">Show up to <p class="card-text spothole-card-text">Show up to
<select id="alerts-to-fetch" class="storeable-select form-select ms-2 me-2" oninput="filtersUpdated();" style="width: 5em;display: inline-block;"> <select id="alerts-to-fetch" class="storeable-select form-select ms-2 me-2" oninput="filtersUpdated();"
{% for c in web_ui_options["alert-count"] %} style="width: 5em;display: inline-block;">
<option value="{{c}}" {% if web_ui_options["alert-count-default"] == c %}selected{% end %}>{{c}}</option> {% for c in web_ui_options["alert-count"] %}
{% end %} <option value="{{c}}" {% if web_ui_options[
"alert-count-default"] == c %}selected{% end %}>{{c}}</option>
{% end %}
</select> </select>
alerts alerts
</p> </p>

View File

@@ -2,10 +2,12 @@
<div class="card-body"> <div class="card-body">
<h5 class="card-title">Number of Spots</h5> <h5 class="card-title">Number of Spots</h5>
<p class="card-text spothole-card-text">Show up to <p class="card-text spothole-card-text">Show up to
<select id="spots-to-fetch" class="storeable-select form-select ms-2 me-2 d-inline-block" oninput="filtersUpdated();" style="width: 5em; display: inline-block;"> <select id="spots-to-fetch" class="storeable-select form-select ms-2 me-2 d-inline-block"
{% for c in web_ui_options["spot-count"] %} oninput="filtersUpdated();" style="width: 5em; display: inline-block;">
<option value="{{c}}" {% if web_ui_options["spot-count-default"] == c %}selected{% end %}>{{c}}</option> {% for c in web_ui_options["spot-count"] %}
{% end %} <option value="{{c}}" {% if web_ui_options[
"spot-count-default"] == c %}selected{% end %}>{{c}}</option>
{% end %}
</select> </select>
spots spots
</p> </p>

View File

@@ -3,21 +3,28 @@
<h5 class="card-title">QRZ.com</h5> <h5 class="card-title">QRZ.com</h5>
<div class="card-text spothole-card-text"> <div class="card-text spothole-card-text">
<div class="form-check mb-2"> <div class="form-check mb-2">
<input type="checkbox" class="storeable-checkbox form-check-input" id="qrz-enabled" onchange="saveSettings();"> <input type="checkbox" class="storeable-checkbox form-check-input" id="qrz-enabled"
onchange="saveSettings();">
<label for="qrz-enabled" class="form-check-label">Use data from QRZ.com</label> <label for="qrz-enabled" class="form-check-label">Use data from QRZ.com</label>
</div> </div>
<div class="mb-2"> <div class="mb-2">
<input type="text" class="storeable-text form-control" id="qrz-username" placeholder="Username (Callsign)" onchange="saveSettings();" autocomplete="username"> <input type="text" class="storeable-text form-control" id="qrz-username"
placeholder="Username (Callsign)" onchange="saveSettings();" autocomplete="username">
</div> </div>
<div class="mb-2"> <div class="mb-2">
<input type="password" class="password-field form-control" id="qrz-password" placeholder="Password" data-remember-checkbox="qrz-remember-password" onchange="saveSettings();" autocomplete="current-password"> <input type="password" class="password-field form-control" id="qrz-password" placeholder="Password"
data-remember-checkbox="qrz-remember-password" onchange="saveSettings();"
autocomplete="current-password">
</div> </div>
<div class="form-check"> <div class="form-check">
<input type="checkbox" class="storeable-checkbox form-check-input" id="qrz-remember-password" onchange="saveSettings();"> <input type="checkbox" class="storeable-checkbox form-check-input" id="qrz-remember-password"
onchange="saveSettings();">
<label for="qrz-remember-password" class="form-check-label">Remember password</label> <label for="qrz-remember-password" class="form-check-label">Remember password</label>
</div> </div>
<div class="mt-3"> <div class="mt-3">
<button type="button" class="btn btn-outline-secondary btn-sm" onclick="location.reload();">Reload with this data</button> <button type="button" class="btn btn-outline-secondary btn-sm" onclick="location.reload();">Reload with
this data
</button>
</div> </div>
<div class="mt-1"> <div class="mt-1">
<small>See <a href="/about#privacy">Privacy</a> for more information.</small> <small>See <a href="/about#privacy">Privacy</a> for more information.</small>

View File

@@ -2,9 +2,11 @@
<div class="card-body"> <div class="card-body">
<h5 class="card-title">Spot Age</h5> <h5 class="card-title">Spot Age</h5>
<p class="card-text spothole-card-text">Last <p class="card-text spothole-card-text">Last
<select id="max-spot-age" class="storeable-select form-select ms-2 me-2 d-inline-block" oninput="filtersUpdated();" style="width: 5em; display: inline-block;"> <select id="max-spot-age" class="storeable-select form-select ms-2 me-2 d-inline-block"
oninput="filtersUpdated();" style="width: 5em; display: inline-block;">
{% for a in web_ui_options["max-spot-age"] %} {% for a in web_ui_options["max-spot-age"] %}
<option value="{{a*60}}" {% if web_ui_options["max-spot-age-default"] == a*60 %}selected{% end %}>{{a}}</option> <option value="{{a*60}}" {% if web_ui_options[
"max-spot-age-default"] == a*60 %}selected{% end %}>{{a}}</option>
{% end %} {% end %}
</select> </select>
minutes minutes

View File

@@ -2,34 +2,55 @@
<div class="card-body"> <div class="card-body">
<h5 class="card-title">Table Columns</h5> <h5 class="card-title">Table Columns</h5>
<div class="row row-cols-2 g-1"> <div class="row row-cols-2 g-1">
<div class="col"><div class="form-check"> <div class="col">
<input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowStartTime" value="tableShowStartTime" oninput="columnsUpdated();" checked> <div class="form-check">
<label class="form-check-label" for="tableShowStartTime">Start Time</label> <input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowStartTime"
</div></div> value="tableShowStartTime" oninput="columnsUpdated();" checked>
<div class="col"><div class="form-check"> <label class="form-check-label" for="tableShowStartTime">Start Time</label>
<input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowEndTime" value="tableShowEndTime" oninput="columnsUpdated();" checked> </div>
<label class="form-check-label" for="tableShowEndTime">End Time</label> </div>
</div></div> <div class="col">
<div class="col"><div class="form-check"> <div class="form-check">
<input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowDX" value="tableShowDX" oninput="columnsUpdated();" checked> <input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowEndTime"
<label class="form-check-label" for="tableShowDX">DX</label> value="tableShowEndTime" oninput="columnsUpdated();" checked>
</div></div> <label class="form-check-label" for="tableShowEndTime">End Time</label>
<div class="col"><div class="form-check"> </div>
<input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowFreqsModes" value="tableShowFreqsModes" oninput="columnsUpdated();" checked> </div>
<label class="form-check-label" for="tableShowFreqsModes">Freq &amp; Mode</label> <div class="col">
</div></div> <div class="form-check">
<div class="col"><div class="form-check"> <input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowDX"
<input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowComment" value="tableShowComment" oninput="columnsUpdated();" checked> value="tableShowDX" oninput="columnsUpdated();" checked>
<label class="form-check-label" for="tableShowComment">Comment</label> <label class="form-check-label" for="tableShowDX">DX</label>
</div></div> </div>
<div class="col"><div class="form-check"> </div>
<input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowSource" value="tableShowSource" oninput="columnsUpdated();" checked> <div class="col">
<label class="form-check-label" for="tableShowSource">Source</label> <div class="form-check">
</div></div> <input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowFreqsModes"
<div class="col"><div class="form-check"> value="tableShowFreqsModes" oninput="columnsUpdated();" checked>
<input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowRef" value="tableShowRef" oninput="columnsUpdated();" checked> <label class="form-check-label" for="tableShowFreqsModes">Freq &amp; Mode</label>
<label class="form-check-label" for="tableShowRef">Ref.</label> </div>
</div></div> </div>
<div class="col">
<div class="form-check">
<input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowComment"
value="tableShowComment" oninput="columnsUpdated();" checked>
<label class="form-check-label" for="tableShowComment">Comment</label>
</div>
</div>
<div class="col">
<div class="form-check">
<input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowSource"
value="tableShowSource" oninput="columnsUpdated();" checked>
<label class="form-check-label" for="tableShowSource">Source</label>
</div>
</div>
<div class="col">
<div class="form-check">
<input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowRef"
value="tableShowRef" oninput="columnsUpdated();" checked>
<label class="form-check-label" for="tableShowRef">Ref.</label>
</div>
</div>
</div> </div>
</div> </div>
</div> </div>

View File

@@ -2,46 +2,76 @@
<div class="card-body"> <div class="card-body">
<h5 class="card-title">Table Columns</h5> <h5 class="card-title">Table Columns</h5>
<div class="row row-cols-2 g-1"> <div class="row row-cols-2 g-1">
<div class="col"><div class="form-check"> <div class="col">
<input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowTime" value="tableShowTime" oninput="columnsUpdated();" checked> <div class="form-check">
<label class="form-check-label" for="tableShowTime">Time</label> <input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowTime"
</div></div> value="tableShowTime" oninput="columnsUpdated();" checked>
<div class="col"><div class="form-check"> <label class="form-check-label" for="tableShowTime">Time</label>
<input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowDX" value="tableShowDX" oninput="columnsUpdated();" checked> </div>
<label class="form-check-label" for="tableShowDX">DX</label> </div>
</div></div> <div class="col">
<div class="col"><div class="form-check"> <div class="form-check">
<input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowFreq" value="tableShowFreq" oninput="columnsUpdated();" checked> <input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowDX"
<label class="form-check-label" for="tableShowFreq">Frequency</label> value="tableShowDX" oninput="columnsUpdated();" checked>
</div></div> <label class="form-check-label" for="tableShowDX">DX</label>
<div class="col"><div class="form-check"> </div>
<input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowMode" value="tableShowMode" oninput="columnsUpdated();" checked> </div>
<label class="form-check-label" for="tableShowMode">Mode</label> <div class="col">
</div></div> <div class="form-check">
<div class="col"><div class="form-check"> <input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowFreq"
<input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowComment" value="tableShowComment" oninput="columnsUpdated();" checked> value="tableShowFreq" oninput="columnsUpdated();" checked>
<label class="form-check-label" for="tableShowComment">Comment</label> <label class="form-check-label" for="tableShowFreq">Frequency</label>
</div></div> </div>
<div class="col"><div class="form-check"> </div>
<input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowBearing" value="tableShowBearing" oninput="columnsUpdated();"> <div class="col">
<label class="form-check-label" for="tableShowBearing">Bearing</label> <div class="form-check">
</div></div> <input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowMode"
<div class="col"><div class="form-check"> value="tableShowMode" oninput="columnsUpdated();" checked>
<input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowType" value="tableShowType" oninput="columnsUpdated();" checked> <label class="form-check-label" for="tableShowMode">Mode</label>
<label class="form-check-label" for="tableShowType">Type</label> </div>
</div></div> </div>
<div class="col"><div class="form-check"> <div class="col">
<input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowRef" value="tableShowRef" oninput="columnsUpdated();" checked> <div class="form-check">
<label class="form-check-label" for="tableShowRef">Ref.</label> <input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowComment"
</div></div> value="tableShowComment" oninput="columnsUpdated();" checked>
<div class="col"><div class="form-check"> <label class="form-check-label" for="tableShowComment">Comment</label>
<input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowDE" value="tableShowDE" oninput="columnsUpdated();" checked> </div>
<label class="form-check-label" for="tableShowDE">DE</label> </div>
</div></div> <div class="col">
<div class="col"><div class="form-check"> <div class="form-check">
<input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowWorkedCheckbox" value="tableShowWorkedCheckbox" oninput="columnsUpdated();" checked> <input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowBearing"
<label class="form-check-label" for="tableShowWorkedCheckbox">Worked?</label> value="tableShowBearing" oninput="columnsUpdated();">
</div></div> <label class="form-check-label" for="tableShowBearing">Bearing</label>
</div>
</div>
<div class="col">
<div class="form-check">
<input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowType"
value="tableShowType" oninput="columnsUpdated();" checked>
<label class="form-check-label" for="tableShowType">Type</label>
</div>
</div>
<div class="col">
<div class="form-check">
<input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowRef"
value="tableShowRef" oninput="columnsUpdated();" checked>
<label class="form-check-label" for="tableShowRef">Ref.</label>
</div>
</div>
<div class="col">
<div class="form-check">
<input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowDE"
value="tableShowDE" oninput="columnsUpdated();" checked>
<label class="form-check-label" for="tableShowDE">DE</label>
</div>
</div>
<div class="col">
<div class="form-check">
<input class="form-check-input storeable-checkbox" type="checkbox" id="tableShowWorkedCheckbox"
value="tableShowWorkedCheckbox" oninput="columnsUpdated();" checked>
<label class="form-check-label" for="tableShowWorkedCheckbox">Worked?</label>
</div>
</div>
</div> </div>
</div> </div>
</div> </div>

View File

@@ -2,7 +2,8 @@
<div class="card-body"> <div class="card-body">
<h5 class="card-title">Time Zone</h5> <h5 class="card-title">Time Zone</h5>
<p class="card-text spothole-card-text"> Use <p class="card-text spothole-card-text"> Use
<select id="timeZone" class="storeable-select form-select ms-2 me-2 d-inline-block" oninput="timeZoneUpdated();" style="width: 8em; display: inline-block;"> <select id="timeZone" class="storeable-select form-select ms-2 me-2 d-inline-block"
oninput="timeZoneUpdated();" style="width: 8em; display: inline-block;">
<option value="UTC" selected>UTC</option> <option value="UTC" selected>UTC</option>
<option value="local">Local time</option> <option value="local">Local time</option>
</select> </select>

View File

@@ -284,7 +284,7 @@
</div> </div>
<script src="/vendor/js/chart-4.4.9.umd.min.js"></script> <script src="/vendor/js/chart-4.4.9.umd.min.js"></script>
<script src="/js/conditions.js?v=1781811406"></script> <script src="/js/conditions.js?v=1781901371"></script>
<script>$(document).ready(function () { <script>$(document).ready(function () {
$("#nav-link-conditions").addClass("active"); $("#nav-link-conditions").addClass("active");
}); <!-- highlight active page in nav --></script> }); <!-- highlight active page in nav --></script>

View File

@@ -13,7 +13,7 @@
</div> </div>
<div id="filters-area" class="appearing-panel card mb-3"> <div id="filters-area" class="appearing-panel card mb-3">
{% module Template("widgets/filters-area-header.html", web_ui_options=web_ui_options) %} {% module Template("widgets/filters-area-header.html", web_ui_options=web_ui_options) %}
<div class="card-body"> <div class="card-body">
<div class="row row-cols-1 g-4 mb-4 row-cols-md-3"> <div class="row row-cols-1 g-4 mb-4 row-cols-md-3">
<div class="col"> <div class="col">
@@ -54,7 +54,8 @@
{% module Template("cards/map-features.html", web_ui_options=web_ui_options) %} {% module Template("cards/map-features.html", web_ui_options=web_ui_options) %}
</div> </div>
<div class="col"> <div class="col">
{% module Template("cards/color-scheme-and-band-color-scheme.html", web_ui_options=web_ui_options) %} {% module Template("cards/color-scheme-and-band-color-scheme.html",
web_ui_options=web_ui_options) %}
</div> </div>
</div> </div>
</div> </div>
@@ -92,10 +93,15 @@
<script src="/vendor/js/leaflet-workedallbritainireland.js"></script> <script src="/vendor/js/leaflet-workedallbritainireland.js"></script>
<script> <script>
let spotProvidersEnabledByDefault = {% raw json_encode(web_ui_options["spot-providers-enabled-by-default"]) %}; let spotProvidersEnabledByDefault = { % raw
json_encode(web_ui_options["spot-providers-enabled-by-default"]) %
}
;
</script> </script>
<script src="/js/spotsbandsandmap.js?v=1781811406"></script> <script src="/js/spotsbandsandmap.js?v=1781901371"></script>
<script src="/js/map.js?v=1781811406"></script> <script src="/js/map.js?v=1781901371"></script>
<script>$(document).ready(function() { $("#nav-link-map").addClass("active"); }); <!-- highlight active page in nav --></script> <script>$(document).ready(function () {
$("#nav-link-map").addClass("active");
}); <!-- highlight active page in nav --></script>
{% end %} {% end %}

View File

@@ -11,8 +11,10 @@
<meta property="og:title" content="Spothole"/> <meta property="og:title" content="Spothole"/>
<meta property="twitter:title" content="Spothole"/> <meta property="twitter:title" content="Spothole"/>
<meta name="description" content="An Amateur Radio spotting tool bringing together DX clusters and outdoor programmes, providing a universal JSON API and web interface."/> <meta name="description"
<meta property="og:description" content="An Amateur Radio spotting tool bringing together DX clusters and outdoor programmes, providing a universal JSON API and web interface."/> content="An Amateur Radio spotting tool bringing together DX clusters and outdoor programmes, providing a universal JSON API and web interface."/>
<meta property="og:description"
content="An Amateur Radio spotting tool bringing together DX clusters and outdoor programmes, providing a universal JSON API and web interface."/>
<link rel="canonical" href="https://spothole.app/"/> <link rel="canonical" href="https://spothole.app/"/>
<meta property="og:url" content="https://spothole.app/"/> <meta property="og:url" content="https://spothole.app/"/>
<meta property="og:image" content="https://spothole.app/img/banner.png"/> <meta property="og:image" content="https://spothole.app/img/banner.png"/>

View File

@@ -2,10 +2,16 @@
{% block content %} {% block content %}
<div id="intro-box" class="permanently-dismissible-box mt-3"> <div id="intro-box" class="permanently-dismissible-box mt-3">
<div class="alert alert-primary alert-dismissible fade show" role="alert"> <div class="alert alert-primary alert-dismissible fade show" role="alert">
<i class="fa-solid fa-circle-info"></i> <strong>What is Spothole?</strong><br/>Spothole is an aggregator of amateur radio spots from DX clusters and outdoor activity programmes. It's free for anyone to use and includes an API that developers can build other applications on. For more information, check out the <a href="/about" class="alert-link">"About" page</a>. If that sounds like nonsense to you, you can visit <a href="/about#faq" class="alert-link">the FAQ section</a> to learn more. <i class="fa-solid fa-circle-info"></i> <strong>What is Spothole?</strong><br/>Spothole is an aggregator of
<button type="button" id="intro-box-dismiss" class="btn-close" data-bs-dismiss="alert" aria-label="Close"></button> amateur radio spots from DX clusters and outdoor activity programmes. It's free for anyone to use and includes
</div> an API that developers can build other applications on. For more information, check out the <a href="/about"
class="alert-link">"About"
page</a>. If that sounds like nonsense to you, you can visit <a href="/about#faq" class="alert-link">the FAQ
section</a> to learn more.
<button type="button" id="intro-box-dismiss" class="btn-close" data-bs-dismiss="alert"
aria-label="Close"></button>
</div>
</div> </div>
<div class="mt-3"> <div class="mt-3">
@@ -63,7 +69,8 @@
{% module Template("cards/number-of-spots.html", web_ui_options=web_ui_options) %} {% module Template("cards/number-of-spots.html", web_ui_options=web_ui_options) %}
</div> </div>
<div class="col"> <div class="col">
{% module Template("cards/color-scheme-and-band-color-scheme.html", web_ui_options=web_ui_options) %} {% module Template("cards/color-scheme-and-band-color-scheme.html", web_ui_options=web_ui_options)
%}
</div> </div>
<div class="col"> <div class="col">
{% module Template("cards/table-columns-spots.html", web_ui_options=web_ui_options) %} {% module Template("cards/table-columns-spots.html", web_ui_options=web_ui_options) %}
@@ -96,16 +103,26 @@
</div> </div>
<div id="table-container"> <div id="table-container">
<table id="table" class="table"><thead><tr></tr></thead><tbody></tbody></table> <table id="table" class="table">
<thead>
<tr></tr>
</thead>
<tbody></tbody>
</table>
</div> </div>
</div> </div>
<script> <script>
let spotProvidersEnabledByDefault = {% raw json_encode(web_ui_options["spot-providers-enabled-by-default"]) %}; let spotProvidersEnabledByDefault = { % raw
json_encode(web_ui_options["spot-providers-enabled-by-default"]) %
}
;
</script> </script>
<script src="/js/spotsbandsandmap.js?v=1781811406"></script> <script src="/js/spotsbandsandmap.js?v=1781901371"></script>
<script src="/js/spots.js?v=1781811406"></script> <script src="/js/spots.js?v=1781901371"></script>
<script>$(document).ready(function() { $("#nav-link-spots").addClass("active"); }); <!-- highlight active page in nav --></script> <script>$(document).ready(function () {
$("#nav-link-spots").addClass("active");
}); <!-- highlight active page in nav --></script>
{% end %} {% end %}

View File

@@ -59,9 +59,11 @@
</div> </div>
</div> </div>
<script src="/js/status.js?v=1781811406"></script> <script src="/js/status.js?v=1781901371"></script>
<script> <script>
$(document).ready(function() { $("#nav-link-status").addClass("active"); }); <!-- highlight active page in nav --> $(document).ready(function () {
$("#nav-link-status").addClass("active");
}); <!-- highlight active page in nav -->
</script> </script>
{% end %} {% end %}

View File

@@ -1,13 +1,24 @@
<label class="form-check-label form-label" for="band-color-scheme">Band color scheme</label><br/> <label class="form-check-label form-label" for="band-color-scheme">Band color scheme</label><br/>
<select id="band-color-scheme" class="storeable-select form-select d-inline-block" oninput="setBandColorSchemeFromUI();" style="display: inline-block;"> <select id="band-color-scheme" class="storeable-select form-select d-inline-block" oninput="setBandColorSchemeFromUI();"
<option value="PSK Reporter" {% if web_ui_options["band-color-scheme-default"] == "PSK Reporter" %}selected{% end %}>PSK Reporter</option> style="display: inline-block;">
<option value="PSK Reporter (Adjusted)" {% if web_ui_options["band-color-scheme-default"] == "PSK Reporter (Adjusted)" %}selected{% end %}>PSK Reporter (Adjusted)</option> <option value="PSK Reporter" {% if web_ui_options[
<option value="RBN" {% if web_ui_options["band-color-scheme-default"] == "RBN" %}selected{% end %}>RBN</option> "band-color-scheme-default"] == "PSK Reporter" %}selected{% end %}>PSK Reporter</option>
<option value="Ham Rainbow" {% if web_ui_options["band-color-scheme-default"] == "Ham Rainbow" %}selected{% end %}>Ham Rainbow</option> <option value="PSK Reporter (Adjusted)" {% if web_ui_options[
<option value="Ham Rainbow (Reverse)" {% if web_ui_options["band-color-scheme-default"] == "Ham Rainbow (Reverse)" %}selected{% end %}>Ham Rainbow (Reverse)</option> "band-color-scheme-default"] == "PSK Reporter (Adjusted)" %}selected{% end %}>PSK Reporter (Adjusted)</option>
<option value="Kate Morley" {% if web_ui_options["band-color-scheme-default"] == "Kate Morley" %}selected{% end %}>Kate Morley</option> <option value="RBN" {% if web_ui_options[
<option value="ColorBrewer" {% if web_ui_options["band-color-scheme-default"] == "ColorBrewer" %}selected{% end %}>ColorBrewer</option> "band-color-scheme-default"] == "RBN" %}selected{% end %}>RBN</option>
<option value="IWantHue" {% if web_ui_options["band-color-scheme-default"] == "IWantHue" %}selected{% end %}>IWantHue</option> <option value="Ham Rainbow" {% if web_ui_options[
<option value="IWantHue (Color Blind)" {% if web_ui_options["band-color-scheme-default"] == "IWantHue (Color Blind)" %}selected{% end %}>IWantHue (Color Blind)</option> "band-color-scheme-default"] == "Ham Rainbow" %}selected{% end %}>Ham Rainbow</option>
<option value="Mokole" {% if web_ui_options["band-color-scheme-default"] == "Mokole" %}selected{% end %}>Mokole</option> <option value="Ham Rainbow (Reverse)" {% if web_ui_options[
"band-color-scheme-default"] == "Ham Rainbow (Reverse)" %}selected{% end %}>Ham Rainbow (Reverse)</option>
<option value="Kate Morley" {% if web_ui_options[
"band-color-scheme-default"] == "Kate Morley" %}selected{% end %}>Kate Morley</option>
<option value="ColorBrewer" {% if web_ui_options[
"band-color-scheme-default"] == "ColorBrewer" %}selected{% end %}>ColorBrewer</option>
<option value="IWantHue" {% if web_ui_options[
"band-color-scheme-default"] == "IWantHue" %}selected{% end %}>IWantHue</option>
<option value="IWantHue (Color Blind)" {% if web_ui_options[
"band-color-scheme-default"] == "IWantHue (Color Blind)" %}selected{% end %}>IWantHue (Color Blind)</option>
<option value="Mokole" {% if web_ui_options[
"band-color-scheme-default"] == "Mokole" %}selected{% end %}>Mokole</option>
</select> </select>

View File

@@ -1,6 +1,10 @@
<label class="form-check-label form-label" for="color-scheme">UI color scheme</label> <label class="form-check-label form-label" for="color-scheme">UI color scheme</label>
<select id="color-scheme" class="storeable-select form-select d-inline-block" oninput="setColorSchemeFromUI();" style="display: inline-block;"> <select id="color-scheme" class="storeable-select form-select d-inline-block" oninput="setColorSchemeFromUI();"
<option value="auto" {% if web_ui_options["color-scheme-default"] == "auto" %}selected{% end %}>Automatic</option> style="display: inline-block;">
<option value="light" {% if web_ui_options["color-scheme-default"] == "light" %}selected{% end %}>Light</option> <option value="auto" {% if web_ui_options[
<option value="dark" {% if web_ui_options["color-scheme-default"] == "dark" %}selected{% end %}>Dark</option> "color-scheme-default"] == "auto" %}selected{% end %}>Automatic</option>
<option value="light" {% if web_ui_options[
"color-scheme-default"] == "light" %}selected{% end %}>Light</option>
<option value="dark" {% if web_ui_options[
"color-scheme-default"] == "dark" %}selected{% end %}>Dark</option>
</select> </select>

View File

@@ -4,7 +4,8 @@
Data Data
</div> </div>
<div class="col-auto d-inline-flex"> <div class="col-auto d-inline-flex">
<button id="close-data-button" type="button" class="btn-close btn-close" aria-label="Close" onclick="closeDataPanel();"></button> <button id="close-data-button" type="button" class="btn-close btn-close" aria-label="Close"
onclick="closeDataPanel();"></button>
</div> </div>
</div> </div>
</div> </div>

View File

@@ -4,7 +4,8 @@
Display Display
</div> </div>
<div class="col-auto d-inline-flex"> <div class="col-auto d-inline-flex">
<button id="close-display-button" type="button" class="btn-close btn-close" aria-label="Close" onclick="closeDisplayPanel();"></button> <button id="close-display-button" type="button" class="btn-close btn-close" aria-label="Close"
onclick="closeDisplayPanel();"></button>
</div> </div>
</div> </div>
</div> </div>

View File

@@ -4,7 +4,8 @@
Filters Filters
</div> </div>
<div class="col-auto d-inline-flex"> <div class="col-auto d-inline-flex">
<button id="close-filters-button" type="button" class="btn-close btn-close" aria-label="Close" onclick="closeFiltersPanel();"></button> <button id="close-filters-button" type="button" class="btn-close btn-close" aria-label="Close"
onclick="closeFiltersPanel();"></button>
</div> </div>
</div> </div>
</div> </div>

View File

@@ -1,5 +1,11 @@
<div class="d-inline-flex gap-1"> <div class="d-inline-flex gap-1">
<button id="filters-button" type="button" class="btn btn-outline-secondary" data-bs-toggle="button" onclick="toggleFiltersPanel();"><i class="fa-solid fa-filter"></i><span class="hideonmobile">&nbsp;Filters</span></button> <button id="filters-button" type="button" class="btn btn-outline-secondary" data-bs-toggle="button"
<button id="display-button" type="button" class="btn btn-outline-secondary" data-bs-toggle="button" onclick="toggleDisplayPanel();"><i class="fa-solid fa-desktop"></i><span class="hideonmobile">&nbsp;Display</span></button> onclick="toggleFiltersPanel();"><i class="fa-solid fa-filter"></i><span
<button id="data-button" type="button" class="btn btn-outline-secondary" data-bs-toggle="button" onclick="toggleDataPanel();"><i class="fa-solid fa-database"></i><span class="hideonmobile">&nbsp;Your data</span></button> class="hideonmobile">&nbsp;Filters</span></button>
<button id="display-button" type="button" class="btn btn-outline-secondary" data-bs-toggle="button"
onclick="toggleDisplayPanel();"><i class="fa-solid fa-desktop"></i><span
class="hideonmobile">&nbsp;Display</span></button>
<button id="data-button" type="button" class="btn btn-outline-secondary" data-bs-toggle="button"
onclick="toggleDataPanel();"><i class="fa-solid fa-database"></i><span
class="hideonmobile">&nbsp;Your data</span></button>
</div> </div>

View File

@@ -691,7 +691,7 @@ components:
oneOf: oneOf:
- $ref: "#/components/schemas/SIGName" - $ref: "#/components/schemas/SIGName"
- type: string - type: string
enum: [NO_SIG] enum: [ NO_SIG ]
example: POTA example: POTA
Continent: Continent:
@@ -1510,7 +1510,7 @@ components:
for DX). Null if foF2 or MUF data is not yet available. for DX). Null if foF2 or MUF data is not yet available.
additionalProperties: additionalProperties:
type: string type: string
enum: [Closed, Short, Long] enum: [ Closed, Short, Long ]
example: example:
"160m": "Closed" "160m": "Closed"
"80m": "Short" "80m": "Short"

View File

@@ -1,10 +1,11 @@
/* NAVIGATION */ /* NAVIGATION */
.navbar-nav .nav-link.active { .navbar-nav .nav-link.active {
font-weight: bold; font-weight: bold;
} }
.navbar-nav .nav-link i { .navbar-nav .nav-link i {
margin-right: 0.2em; margin-right: 0.2em;
} }
/* In embedded mode, hide header/footer/settings. "#header div" is kind of janky but for some reason if we hide the /* In embedded mode, hide header/footer/settings. "#header div" is kind of janky but for some reason if we hide the
@@ -26,9 +27,11 @@ whole of #header, the map vertical sizing breaks. */
border-top: 1px solid grey; border-top: 1px solid grey;
border-left: 1px solid grey; border-left: 1px solid grey;
} }
[embedded-mode=true] #embeddedModeFooter { [embedded-mode=true] #embeddedModeFooter {
display: block; display: block;
} }
#embeddedModeFooter img.logo { #embeddedModeFooter img.logo {
position: relative; position: relative;
top: -2px; top: -2px;
@@ -50,15 +53,15 @@ whole of #header, the map vertical sizing breaks. */
/* GENERAL PAGE LAYOUT */ /* GENERAL PAGE LAYOUT */
div.container { div.container {
display:grid; display: grid;
grid-template-rows:auto 1fr auto; grid-template-rows:auto 1fr auto;
grid-template-columns:100%; grid-template-columns:100%;
/* fallback height */ /* fallback height */
min-height:100vh; min-height: 100vh;
/* new small viewport height for modern browsers */ /* new small viewport height for modern browsers */
min-height:100svh; min-height: 100svh;
} }
[embedded-mode=true] div.container { [embedded-mode=true] div.container {
@@ -69,15 +72,15 @@ div.container {
/* ABOUT PAGE */ /* ABOUT PAGE */
#info-container{ #info-container {
width: 100%; width: 100%;
} }
@media (min-width: 768px) { @media (min-width: 768px) {
#info-container{ #info-container {
max-width: 60em; max-width: 60em;
margin: 0 auto; margin: 0 auto;
} }
} }
@@ -90,7 +93,7 @@ input#search {
i#searchicon { i#searchicon {
position: absolute; position: absolute;
left: 0rem; left: 0;
top: 2px; top: 2px;
padding: 10px; padding: 10px;
pointer-events: none; pointer-events: none;
@@ -161,6 +164,7 @@ a.dx-link {
text-decoration: none; text-decoration: none;
font-weight: bold; font-weight: bold;
} }
a.sig-ref-link { a.sig-ref-link {
color: var(--bs-emphasis-color); color: var(--bs-emphasis-color);
text-decoration: none; text-decoration: none;
@@ -171,21 +175,23 @@ tr.table-faded td {
filter: grayscale(100%) opacity(30%) !important; filter: grayscale(100%) opacity(30%) !important;
text-decoration: line-through !important; text-decoration: line-through !important;
} }
tr.table-faded td span { tr.table-faded td span {
text-decoration: line-through !important; text-decoration: line-through !important;
} }
/* New spot styles */ /* New spot styles */
tr.new td { tr.new td {
animation: 2s linear newspotanim; animation: 2s linear newspotanim;
} }
@keyframes newspotanim { @keyframes newspotanim {
0% { 0% {
background-color: var(--bs-success-border-subtle); background-color: var(--bs-success-border-subtle);
} }
100% { 100% {
background-color: initial; background-color: initial;
} }
} }
@@ -196,7 +202,7 @@ tr.new td {
overflow: hidden; overflow: hidden;
} }
#table-container table{ #table-container table {
margin-bottom: 0; margin-bottom: 0;
} }
@@ -217,8 +223,9 @@ div#map {
} }
.leaflet-container { .leaflet-container {
font-family: var(--bs-body-font-family) !important; font-family: var(--bs-body-font-family) sans-serif !important;
} }
.leaflet-control-attribution { .leaflet-control-attribution {
background: none; background: none;
} }
@@ -343,6 +350,7 @@ div.band-spot:hover span.band-spot-info {
.input-narrow { .input-narrow {
max-width: 8em; max-width: 8em;
} }
.input-medium { .input-medium {
max-width: 12em; max-width: 12em;
} }
@@ -360,22 +368,27 @@ div.band-spot:hover span.band-spot-info {
.hideonmobile { .hideonmobile {
display: none !important; display: none !important;
} }
/* Make map stretch to horizontal screen edges */ /* Make map stretch to horizontal screen edges */
div#map, div#table-container, div#bands-container { div#map, div#table-container, div#bands-container {
margin-left: -1em; margin-left: -1em;
margin-right: -1em; margin-right: -1em;
} }
/* Avoid map page filters panel being larger than the map itself */ /* Avoid map page filters panel being larger than the map itself */
#settingsButtonRowMap .appearing-panel { #settingsButtonRowMap .appearing-panel {
max-height: 30em; max-height: 30em;
} }
#settingsButtonRowMap .appearing-panel .card-body { #settingsButtonRowMap .appearing-panel .card-body {
max-height: 26em; max-height: 26em;
overflow: scroll; overflow: scroll;
} }
input#search { input#search {
max-width: 7em; max-width: 7em;
} }
.table-fixed-on-desktop { .table-fixed-on-desktop {
table-layout: auto !important; table-layout: auto !important;
} }

View File

@@ -25,7 +25,7 @@ var PROVIDER_CREDENTIAL_SCHEMAS = {
// Load server options. Once a successful callback is made from this, we can populate the choice boxes in the form and load // Load server options. Once a successful callback is made from this, we can populate the choice boxes in the form and load
// any saved values from local storage. // any saved values from local storage.
function loadOptions() { function loadOptions() {
$.getJSON('/api/v1/options', function(jsonData) { $.getJSON('/api/v1/options', function (jsonData) {
// Store options // Store options
options = jsonData; options = jsonData;
@@ -33,7 +33,7 @@ function loadOptions() {
$.each(options["modes"], function (i, m) { $.each(options["modes"], function (i, m) {
$('#mode').append($('<option>', { $('#mode').append($('<option>', {
value: m, value: m,
text : m text: m
})); }));
}); });
@@ -41,7 +41,7 @@ function loadOptions() {
$.each(options["sigs"], function (i, sig) { $.each(options["sigs"], function (i, sig) {
$('#sig').append($('<option>', { $('#sig').append($('<option>', {
value: sig.name, value: sig.name,
text : sig.name text: sig.name
})); }));
}); });
@@ -190,45 +190,45 @@ function addSpot() {
saveSettings(); saveSettings();
// Unpack the user's entered values // Unpack the user's entered values
var dx = $("#dx-call").val().toUpperCase(); const dx = $("#dx-call").val().toUpperCase();
var freqStr = $("#freq").val(); const freqStr = $("#freq").val();
var mode = $("#mode")[0].value; const mode = $("#mode")[0].value;
var sig = $("#sig")[0].value; const sig = $("#sig")[0].value;
var sigRef = $("#sig-ref").val(); const sigRef = $("#sig-ref").val();
var dxGrid = $("#dx-grid").val(); const dxGrid = $("#dx-grid").val();
var comment = $("#comment").val(); const comment = $("#comment").val();
var de = $("#de-call").val().toUpperCase(); const de = $("#de-call").val().toUpperCase();
var spot = {} const spot = {};
if (dx != "") { if (dx !== "") {
spot["dx_call"] = dx; spot["dx_call"] = dx;
} else { } else {
// todo maybe for neatness just make all these error/rejections server side rather than having logic in two places // todo maybe for neatness just make all these error/rejections server side rather than having logic in two places
showAddSpotError("A DX callsign is required in order to spot."); showAddSpotError("A DX callsign is required in order to spot.");
return; return;
} }
if (freqStr != "") { if (freqStr !== "") {
spot["freq"] = parseFloat(freqStr) * 1000; spot["freq"] = parseFloat(freqStr) * 1000;
} else { } else {
showAddSpotError("A frequency is required in order to spot."); showAddSpotError("A frequency is required in order to spot.");
return; return;
} }
if (mode != "") { if (mode !== "") {
spot["mode"] = mode; spot["mode"] = mode;
} }
if (sig != "") { if (sig !== "") {
spot["sig"] = sig; spot["sig"] = sig;
} }
if (sigRef != "") { if (sigRef !== "") {
spot["sig_refs"] = [{id: sigRef}]; spot["sig_refs"] = [{id: sigRef}];
} }
if (dxGrid != "") { if (dxGrid !== "") {
spot["dx_grid"] = dxGrid; spot["dx_grid"] = dxGrid;
} }
if (comment != "") { if (comment !== "") {
spot["comment"] = comment; spot["comment"] = comment;
} }
if (de != "") { if (de !== "") {
spot["de_call"] = de; spot["de_call"] = de;
} else { } else {
showAddSpotError("A spotter callsign is required in order to spot."); showAddSpotError("A spotter callsign is required in order to spot.");
@@ -274,9 +274,9 @@ function addSpot() {
} }
$.ajax("/api/v1/spot", { $.ajax("/api/v1/spot", {
data : JSON.stringify(spot), data: JSON.stringify(spot),
contentType : 'application/json', contentType: 'application/json',
type : 'POST', type: 'POST',
timeout: 10000, timeout: 10000,
success: async function (result) { success: async function (result) {
// Reset CAPTCHA for next use // Reset CAPTCHA for next use
@@ -313,7 +313,7 @@ function addSpot() {
// Show an "add spot" error. // Show an "add spot" error.
function showAddSpotError(text) { function showAddSpotError(text) {
var div = $("<div class='alert alert-danger alert-dismissible fade show mb-0 mt-4' role='alert'></div>"); const div = $("<div class='alert alert-danger alert-dismissible fade show mb-0 mt-4' role='alert'></div>");
div.append("<i class='fa-solid fa-triangle-exclamation'></i> "); div.append("<i class='fa-solid fa-triangle-exclamation'></i> ");
div.append(document.createTextNode(text)); div.append(document.createTextNode(text));
div.append("<button type='button' class='btn-close' data-bs-dismiss='alert' aria-label='Close'></button>"); div.append("<button type='button' class='btn-close' data-bs-dismiss='alert' aria-label='Close'></button>");
@@ -342,7 +342,7 @@ $("#upstream-provider-select").change(function () {
}); });
// Startup // Startup
$(document).ready(function() { $(document).ready(function () {
// Load options // Load options
loadOptions(); loadOptions();
}); });

View File

@@ -2,11 +2,11 @@
const REFRESH_INTERVAL_SEC = 60 * 10; const REFRESH_INTERVAL_SEC = 60 * 10;
// Storage for the alert data that the server gives us. // Storage for the alert data that the server gives us.
var alerts = [] let alerts = [];
// Load alerts and populate the table. // Load alerts and populate the table.
function loadAlerts() { function loadAlerts() {
$.getJSON('/api/v1/alerts' + buildQueryString(false), function(jsonData) { $.getJSON('/api/v1/alerts' + buildQueryString(false), function (jsonData) {
// Store last updated time // Store last updated time
lastUpdateTime = moment.utc(); lastUpdateTime = moment.utc();
updateRefreshDisplay(); updateRefreshDisplay();
@@ -19,15 +19,15 @@ function loadAlerts() {
// Build a query string for the API, based on the filters that the user has selected. // Build a query string for the API, based on the filters that the user has selected.
function buildQueryString(includeCredentials) { function buildQueryString(includeCredentials) {
var str = "?"; let str = "?";
["dx_continent", "source"].forEach(fn => { ["dx_continent", "source"].forEach(fn => {
if (!allFilterOptionsSelected(fn)) { if (!allFilterOptionsSelected(fn)) {
str = str + getQueryStringFor(fn) + "&"; str = str + getQueryStringFor(fn) + "&";
} }
}); });
str = str + "limit=" + $("#alerts-to-fetch option:selected").val(); str = str + "limit=" + $("#alerts-to-fetch option:selected").val();
var maxDur = $("#max-duration option:selected").val(); const maxDur = $("#max-duration option:selected").val();
if (maxDur != "9999999999") { if (maxDur !== "9999999999") {
str = str + "&max_duration=" + maxDur; str = str + "&max_duration=" + maxDur;
} }
if ($("#dxpeditions_skip_max_duration_check")[0].checked) { if ($("#dxpeditions_skip_max_duration_check")[0].checked) {
@@ -42,16 +42,16 @@ function buildQueryString(includeCredentials) {
// Update the alerts table // Update the alerts table
function updateTable() { function updateTable() {
// Use local time instead of UTC? // Use local time instead of UTC?
var useLocalTime = $("#timeZone")[0].value == "local"; const useLocalTime = $("#timeZone")[0].value === "local";
// Table data toggles // Table data toggles
var showStartTime = $("#tableShowStartTime")[0].checked; const showStartTime = $("#tableShowStartTime")[0].checked;
var showEndTime = $("#tableShowEndTime")[0].checked; const showEndTime = $("#tableShowEndTime")[0].checked;
var showDX = $("#tableShowDX")[0].checked; const showDX = $("#tableShowDX")[0].checked;
var showFreqsModes = $("#tableShowFreqsModes")[0].checked; const showFreqsModes = $("#tableShowFreqsModes")[0].checked;
var showComment = $("#tableShowComment")[0].checked; const showComment = $("#tableShowComment")[0].checked;
var showSource = $("#tableShowSource")[0].checked; const showSource = $("#tableShowSource")[0].checked;
var showRef = $("#tableShowRef")[0].checked; const showRef = $("#tableShowRef")[0].checked;
// Populate table with headers // Populate table with headers
let table = $("#table"); let table = $("#table");
@@ -83,10 +83,10 @@ function updateTable() {
// Split alerts into three types, each of which will get its own table header: On now, next 24h, and later. "On now" // Split alerts into three types, each of which will get its own table header: On now, next 24h, and later. "On now"
// is considered to be events with an end_time where start<now<end, or events with no end time that started in the // is considered to be events with an end_time where start<now<end, or events with no end time that started in the
// last hour. // last hour.
onNow = alerts.filter(a => (a["end_time"] != null && a["end_time"] != 0 && moment.unix(a["end_time"]).utc().isSameOrAfter() && moment.unix(a["start_time"]).utc().isBefore()) const onNow = alerts.filter(a => (a["end_time"] != null && a["end_time"] !== 0 && moment.unix(a["end_time"]).utc().isSameOrAfter() && moment.unix(a["start_time"]).utc().isBefore())
|| ((a["end_time"] == null || a["end_time"] == 0) && moment.unix(a["start_time"]).utc().add(1, 'hours').isSameOrAfter() && moment.unix(a["start_time"]).utc().isBefore())); || ((a["end_time"] == null || a["end_time"] === 0) && moment.unix(a["start_time"]).utc().add(1, 'hours').isSameOrAfter() && moment.unix(a["start_time"]).utc().isBefore()));
next24h = alerts.filter(a => moment.unix(a["start_time"]).utc().isSameOrAfter() && moment.unix(a["start_time"]).utc().subtract(24, 'hours').isBefore()); const next24h = alerts.filter(a => moment.unix(a["start_time"]).utc().isSameOrAfter() && moment.unix(a["start_time"]).utc().subtract(24, 'hours').isBefore());
later = alerts.filter(a => moment.unix(a["start_time"]).utc().subtract(24, 'hours').isSameOrAfter()); const later = alerts.filter(a => moment.unix(a["start_time"]).utc().subtract(24, 'hours').isSameOrAfter());
if (onNow.length > 0) { if (onNow.length > 0) {
table.find('tbody').append('<tr><td colspan="100" class="bg-primary-subtle" style="text-align:center;">On Now</td></tr>'); table.find('tbody').append('<tr><td colspan="100" class="bg-primary-subtle" style="text-align:center;">On Now</td></tr>');
@@ -103,14 +103,14 @@ function updateTable() {
addAlertRowsToTable(table.find('tbody'), later); addAlertRowsToTable(table.find('tbody'), later);
} }
if (onNow.length == 0 && next24h.length == 0 && later.length == 0) { if (onNow.length === 0 && next24h.length === 0 && later.length === 0) {
table.find('tbody').append('<tr class="bg-danger-subtle"><td colspan="100" style="text-align:center;">No alerts match your filters.</td></tr>'); table.find('tbody').append('<tr class="bg-danger-subtle"><td colspan="100" style="text-align:center;">No alerts match your filters.</td></tr>');
} }
} }
// Add a row to tbody for each alert in the provided list // Add a row to tbody for each alert in the provided list
function addAlertRowsToTable(tbody, alerts) { function addAlertRowsToTable(tbody, alerts) {
var count = 0; let count = 0;
alerts.forEach(a => { alerts.forEach(a => {
// Create row // Create row
let $tr = $('<tr>'); let $tr = $('<tr>');
@@ -118,29 +118,29 @@ function addAlertRowsToTable(tbody, alerts) {
// Apply striping to the table. We can't just use Bootstrap's table-striped class because we have all sorts of // Apply striping to the table. We can't just use Bootstrap's table-striped class because we have all sorts of
// extra faff to deal with, like the mobile view having extra rows, and the On Now / Next 24h / Later banners // extra faff to deal with, like the mobile view having extra rows, and the On Now / Next 24h / Later banners
// which cause the table-striped colouring to go awry. // which cause the table-striped colouring to go awry.
if (count % 2 == 1) { if (count % 2 === 1) {
$tr.addClass("table-active"); $tr.addClass("table-active");
} }
// Use local time instead of UTC? // Use local time instead of UTC?
var useLocalTime = $("#timeZone")[0].value == "local"; const useLocalTime = $("#timeZone")[0].value === "local";
// Table data toggles // Table data toggles
var showStartTime = $("#tableShowStartTime")[0].checked; const showStartTime = $("#tableShowStartTime")[0].checked;
var showEndTime = $("#tableShowEndTime")[0].checked; const showEndTime = $("#tableShowEndTime")[0].checked;
var showDX = $("#tableShowDX")[0].checked; const showDX = $("#tableShowDX")[0].checked;
var showFreqsModes = $("#tableShowFreqsModes")[0].checked; const showFreqsModes = $("#tableShowFreqsModes")[0].checked;
var showComment = $("#tableShowComment")[0].checked; const showComment = $("#tableShowComment")[0].checked;
var showSource = $("#tableShowSource")[0].checked; const showSource = $("#tableShowSource")[0].checked;
var showRef = $("#tableShowRef")[0].checked; const showRef = $("#tableShowRef")[0].checked;
// Get times for the alert, and convert to local time if necessary. // Get times for the alert, and convert to local time if necessary.
var start_time_utc = moment.unix(a["start_time"]).utc(); const start_time_utc = moment.unix(a["start_time"]).utc();
var start_time_local = start_time_utc.clone().local(); const start_time_local = start_time_utc.clone().local();
start_time = useLocalTime ? start_time_local : start_time_utc; const start_time = useLocalTime ? start_time_local : start_time_utc;
var end_time_utc = moment.unix(a["end_time"]).utc(); const end_time_utc = moment.unix(a["end_time"]).utc();
var end_time_local = end_time_utc.clone().local(); const end_time_local = end_time_utc.clone().local();
end_time = useLocalTime ? end_time_local : end_time_utc; const end_time = useLocalTime ? end_time_local : end_time_utc;
// Format the times for display. Start time is displayed as e.g. 7 Oct 12:34 unless the time is in a // Format the times for display. Start time is displayed as e.g. 7 Oct 12:34 unless the time is in a
// different year to the current year, in which case the year is inserted between month and hour. // different year to the current year, in which case the year is inserted between month and hour.
@@ -150,22 +150,22 @@ function addAlertRowsToTable(tbody, alerts) {
// Overriding all of that, if the start time is 00:00 and the end time is 23:59 when considered in UTC, the // Overriding all of that, if the start time is 00:00 and the end time is 23:59 when considered in UTC, the
// hours and minutes are stripped out from the display, as we assume the server is just giving us full days. // hours and minutes are stripped out from the display, as we assume the server is just giving us full days.
// Finally, if there is no end date set, "---" is displayed. // Finally, if there is no end date set, "---" is displayed.
var whole_days = start_time_utc.format("HH:mm") == "00:00" && const whole_days = start_time_utc.format("HH:mm") === "00:00" &&
(end_time_utc != null || end_time_utc > 0 || end_time_utc.format("HH:mm") == "23:59"); (end_time_utc === 0 || end_time_utc.format("HH:mm") === "23:59");
var hours_minutes_format = whole_days ? "" : " HH:mm"; const hours_minutes_format = whole_days ? "" : " HH:mm";
var start_time_formatted = start_time.format("D MMM" + hours_minutes_format); let start_time_formatted = start_time.format("D MMM" + hours_minutes_format);
if (start_time.format("YYYY") != moment().format("YYYY")) { if (start_time.format("YYYY") !== moment().format("YYYY")) {
start_time_formatted = start_time.format("D MMM YYYY" + hours_minutes_format); start_time_formatted = start_time.format("D MMM YYYY" + hours_minutes_format);
} else if (useLocalTime && start_time.format("D MMM YYYY") == moment().format("D MMM YYYY")) { } else if (useLocalTime && start_time.format("D MMM YYYY") === moment().format("D MMM YYYY")) {
start_time_formatted = start_time.format("[Today]" + hours_minutes_format); start_time_formatted = start_time.format("[Today]" + hours_minutes_format);
} }
var end_time_formatted = "---"; let end_time_formatted = "---";
if (end_time_utc != null && end_time_utc > 0 && end_time != null) { if (end_time_utc > 0 && end_time != null) {
var end_time_formatted = whole_days ? start_time_formatted : end_time.format("HH:mm"); end_time_formatted = whole_days ? start_time_formatted : end_time.format("HH:mm");
if (end_time.format("D MMM") != start_time.format("D MMM")) { if (end_time.format("D MMM") !== start_time.format("D MMM")) {
if (end_time.format("YYYY") != moment().format("YYYY")) { if (end_time.format("YYYY") !== moment().format("YYYY")) {
end_time_formatted = end_time.format("D MMM YYYY" + hours_minutes_format); end_time_formatted = end_time.format("D MMM YYYY" + hours_minutes_format);
} else if (useLocalTime && end_time.format("D MMM YYYY") == moment().format("D MMM YYYY")) { } else if (useLocalTime && end_time.format("D MMM YYYY") === moment().format("D MMM YYYY")) {
end_time_formatted = end_time.format("[Today]" + hours_minutes_format); end_time_formatted = end_time.format("[Today]" + hours_minutes_format);
} else { } else {
end_time_formatted = end_time.format("D MMM" + hours_minutes_format); end_time_formatted = end_time.format("D MMM" + hours_minutes_format);
@@ -174,52 +174,52 @@ function addAlertRowsToTable(tbody, alerts) {
} }
// Format dx country // Format dx country
var dx_country = a["dx_country"] let dx_country = a["dx_country"];
if (dx_country == null) { if (dx_country == null) {
dx_country = "Unknown or not a country" dx_country = "Unknown or not a country"
} }
// Format DX flag // Format DX flag
var dx_flag = "<i class='fa-solid fa-globe-africa'></i>"; let dx_flag = "<i class='fa-solid fa-globe-africa'></i>";
if (a["dx_dxcc_id"] && a["dx_dxcc_id"] != null && a["dx_dxcc_id"] != 0) { if (a["dx_dxcc_id"] && a["dx_dxcc_id"] != null && a["dx_dxcc_id"] !== 0) {
dx_flag = `<img src="img/flags/${a['dx_dxcc_id']}.png" class="flag" width="24" alt="${dx_country}" title="${dx_country}"/>`; dx_flag = `<img src="img/flags/${a['dx_dxcc_id']}.png" class="flag" width="24" alt="${dx_country}" title="${dx_country}"/>`;
} }
// Format dx calls // Format dx calls
var dx_calls_html = ""; let dx_calls_html = "";
if (a["dx_calls"] != null) { if (a["dx_calls"] != null) {
dx_calls_html = a["dx_calls"].map(call => `<a class='dx-link' href='https://qrz.com/db/${call}' target='_new'>${call}</a>`).join(", "); dx_calls_html = a["dx_calls"].map(call => `<a class='dx-link' href='https://qrz.com/db/${call}' target='_new'>${call}</a>`).join(", ");
} }
// Format DXpedition country // Format DXpedition country
var dx_country_html = ""; let dx_country_html = "";
if (a["is_dxpedition"] == true && a["dx_country"] != null && a["dx_country"] != "") { if (a["is_dxpedition"] === true && a["dx_country"] != null && a["dx_country"] !== "") {
dx_country_html = `<br/>${a["dx_country"]}`; dx_country_html = `<br/>${a["dx_country"]}`;
} }
// Format freqs & modes // Format freqs & modes
var freqsModesText = ""; let freqsModesText = "";
if (a["freqs_modes"] != null) { if (a["freqs_modes"] != null) {
freqsModesText = escapeHtml(a["freqs_modes"]); freqsModesText = escapeHtml(a["freqs_modes"]);
} }
// Format comment // Format comment
var commentText = ""; let commentText = "";
if (a["comment"] != null) { if (a["comment"] != null) {
commentText = escapeHtml(a["comment"]); commentText = escapeHtml(a["comment"]);
} }
// Sig or fallback to source // Sig or fallback to source
var sigSourceText = a["source"]; let sigSourceText = a["source"];
if (a["sig"]) { if (a["sig"]) {
sigSourceText = a["sig"]; sigSourceText = a["sig"];
} }
// Format sig_refs // Format sig_refs
var sig_refs = ""; let sig_refs = "";
if (a["sig_refs"] != null) { if (a["sig_refs"] != null) {
var items = [] const items = [];
for (var i = 0; i < a["sig_refs"].length; i++) { for (let i = 0; i < a["sig_refs"].length; i++) {
if (a["sig_refs"][i]["url"] != null) { if (a["sig_refs"][i]["url"] != null) {
items[i] = `<a href='${encodeURI(a["sig_refs"][i]["url"])}' title='${escapeHtml(a["sig_refs"][i]["name"])}' target='_new' class='sig-ref-link'>${escapeHtml(a["sig_refs"][i]["id"])}</a>` items[i] = `<a href='${encodeURI(a["sig_refs"][i]["url"])}' title='${escapeHtml(a["sig_refs"][i]["name"])}' target='_new' class='sig-ref-link'>${escapeHtml(a["sig_refs"][i]["id"])}</a>`
} else { } else {
@@ -254,11 +254,11 @@ function addAlertRowsToTable(tbody, alerts) {
tbody.append($tr); tbody.append($tr);
// Second row for mobile view only, containing source, ref, freqs/modes & comment // Second row for mobile view only, containing source, ref, freqs/modes & comment
$tr2 = $("<tr class='hidenotonmobile'>"); const $tr2 = $("<tr class='hidenotonmobile'>");
if (count % 2 == 1) { if (count % 2 === 1) {
$tr2.addClass("table-active"); $tr2.addClass("table-active");
} }
$td2 = $("<td colspan='100'>"); const $td2 = $("<td colspan='100'>");
if (showSource) { if (showSource) {
$td2.append(`<span class='icon-wrapper'><i class='fa-solid ${sigToIcon(a["sig"], "fa-globe-africa")}'></i></span> `); $td2.append(`<span class='icon-wrapper'><i class='fa-solid ${sigToIcon(a["sig"], "fa-globe-africa")}'></i></span> `);
} }
@@ -280,7 +280,7 @@ function addAlertRowsToTable(tbody, alerts) {
// Load server options. Once a successful callback is made from this, we then query alerts. // Load server options. Once a successful callback is made from this, we then query alerts.
function loadOptions() { function loadOptions() {
$.getJSON('/api/v1/options', function(jsonData) { $.getJSON('/api/v1/options', function (jsonData) {
// Store options // Store options
options = jsonData; options = jsonData;
@@ -310,7 +310,7 @@ function filtersUpdated() {
} }
// Startup // Startup
$(document).ready(function() { $(document).ready(function () {
// Call loadOptions(), this will then trigger loading alerts and setting up timers. // Call loadOptions(), this will then trigger loading alerts and setting up timers.
loadOptions(); loadOptions();
// Update the refresh timing display every second // Update the refresh timing display every second
@@ -319,7 +319,7 @@ $(document).ready(function() {
// Reload alerts on becoming visible. This forces a refresh when used as a PWA and the user switches back to the PWA // Reload alerts on becoming visible. This forces a refresh when used as a PWA and the user switches back to the PWA
// after some time has passed with it in the background. // after some time has passed with it in the background.
addEventListener("visibilitychange", (event) => { addEventListener("visibilitychange", () => {
if (!document.hidden) { if (!document.hidden) {
loadAlerts(); loadAlerts();
} }

View File

@@ -12,7 +12,7 @@ BAND_COLUMN_SPOT_DIV_HEIGHT_PX = BAND_COLUMN_FONT_SIZE * 1.6;
// Load spots and populate the bands display. // Load spots and populate the bands display.
function loadSpots() { function loadSpots() {
$.getJSON('/api/v1/spots' + buildQueryString(false), function(jsonData) { $.getJSON('/api/v1/spots' + buildQueryString(false), function (jsonData) {
// Store last updated time // Store last updated time
lastUpdateTime = moment.utc(); lastUpdateTime = moment.utc();
updateRefreshDisplay(); updateRefreshDisplay();
@@ -25,7 +25,7 @@ function loadSpots() {
// Build a query string for the API, based on the filters that the user has selected. // Build a query string for the API, based on the filters that the user has selected.
function buildQueryString(includeCredentials) { function buildQueryString(includeCredentials) {
var str = "?"; let str = "?";
["dx_continent", "de_continent", "mode", "source", "band", "sig"].forEach(fn => { ["dx_continent", "de_continent", "mode", "source", "band", "sig"].forEach(fn => {
if (!allFilterOptionsSelected(fn)) { if (!allFilterOptionsSelected(fn)) {
str = str + getQueryStringFor(fn) + "&"; str = str + getQueryStringFor(fn) + "&";
@@ -43,7 +43,7 @@ function buildQueryString(includeCredentials) {
// Update the bands display // Update the bands display
function updateBands() { function updateBands() {
// Stop here if nothing to display // Stop here if nothing to display
var bandsContainer = $("#bands-container"); const bandsContainer = $("#bands-container");
if (spots.length === 0) { if (spots.length === 0) {
bandsContainer.html("<div class='alert alert-danger' role='alert'>No spots match your filters.</div>"); bandsContainer.html("<div class='alert alert-danger' role='alert'>No spots match your filters.</div>");
return; return;
@@ -52,7 +52,7 @@ function updateBands() {
// Do some harsher de-duping. Because we only display callsign, frequency and mode here, the previous // Do some harsher de-duping. Because we only display callsign, frequency and mode here, the previous
// de-duplication could have let some through that don't look like dupes on the map, but would do here. // de-duplication could have let some through that don't look like dupes on the map, but would do here.
// Typically that's a person activating two programs at the same time, e.g. POTA & WWFF. // Typically that's a person activating two programs at the same time, e.g. POTA & WWFF.
spotList = removeDuplicatesForBandPanel(spots); const spotList = removeDuplicatesForBandPanel(spots);
// Convert to a map of band names to the spots on that band. Bands with no // Convert to a map of band names to the spots on that band. Bands with no
// spots in view will not be present. // spots in view will not be present.
@@ -67,10 +67,10 @@ function updateBands() {
}); });
// Track if any columns end up taller than expected, so we can resize the container and avoid vertical scroll. // Track if any columns end up taller than expected, so we can resize the container and avoid vertical scroll.
var maxHeightBand = 0; let maxHeightBand = 0;
// Build up table content for each band // Build up table content for each band
var table = $('<table id="bands-table">').append('<thead><tr></tr></thead><tbody><tr></tr></tbody>'); const table = $('<table id="bands-table">').append('<thead><tr></tr></thead><tbody><tr></tr></tbody>');
bandToSpots.forEach(function (spotList, bandName) { bandToSpots.forEach(function (spotList, bandName) {
// Get the colours for the band from the first spot, and prepare the header // Get the colours for the band from the first spot, and prepare the header
table.find('thead tr').append(`<th style='background-color:${bandToColor(spotList[0].band)}; color:${bandToContrastColor(spotList[0].band)}'>${spotList[0].band}</th>`); table.find('thead tr').append(`<th style='background-color:${bandToColor(spotList[0].band)}; color:${bandToContrastColor(spotList[0].band)}'>${spotList[0].band}</th>`);
@@ -82,11 +82,11 @@ function updateBands() {
// Print the frequency band markers. This is 41 steps to divide the band evenly into 40 markers. One in every // Print the frequency band markers. This is 41 steps to divide the band evenly into 40 markers. One in every
// four will show the actual frequency, the others will just be dashes. // four will show the actual frequency, the others will just be dashes.
bandMarkersDiv = $('<div class="band-markers">'); const bandMarkersDiv = $('<div class="band-markers">');
const freqStep = (band.end_freq - band.start_freq) / 40.0; const freqStep = (band.end_freq - band.start_freq) / 40.0;
for (let i = 0; i <= 40; i++) { for (let i = 0; i <= 40; i++) {
if (i % 4 === 0) { if (i % 4 === 0) {
bandMarkersDiv.append("&mdash;" + ((band.start_freq + i * freqStep)/1000000).toFixed(3) + "<br/>"); bandMarkersDiv.append("&mdash;" + ((band.start_freq + i * freqStep) / 1000000).toFixed(3) + "<br/>");
} else if (i % 4 === 2) { } else if (i % 4 === 2) {
bandMarkersDiv.append("&ndash;<br/>"); bandMarkersDiv.append("&ndash;<br/>");
} else { } else {
@@ -95,10 +95,12 @@ function updateBands() {
} }
// Prepare the spots list // Prepare the spots list
var bandSpotsDiv = $("<div class='band-spots'>"); const bandSpotsDiv = $("<div class='band-spots'>");
var lastSpotPxDownBand = -999; let lastSpotPxDownBand = -999;
// Sort by frequency so have a consistent order in which to plan where they will appear on the band div. // Sort by frequency so have a consistent order in which to plan where they will appear on the band div.
spotList.sort(function(a, b) { return a.freq - b.freq; }); spotList.sort(function (a, b) {
return a.freq - b.freq;
});
// First calculate how we should be displaying the spots. There are three "modes" to try to place them in a // First calculate how we should be displaying the spots. There are three "modes" to try to place them in a
// visually appealing way: // visually appealing way:
// 1) Spaced normally, not going over the end of the band, so we populate them forwards. // 1) Spaced normally, not going over the end of the band, so we populate them forwards.
@@ -118,8 +120,8 @@ function updateBands() {
// Mode 1 or 2. Run through adding things to the list forwards as a test. // Mode 1 or 2. Run through adding things to the list forwards as a test.
spotList.forEach(s => { spotList.forEach(s => {
// Work out how far down the div to draw it // Work out how far down the div to draw it
var percentDownBand = (s.freq - band.start_freq) / (band.end_freq - band.start_freq) * 0.97; // not 100% due to fudge, the first and last dashes are not exactly at the top and bottom of the div as some space is needed for text const percentDownBand = (s.freq - band.start_freq) / (band.end_freq - band.start_freq) * 0.97; // not 100% due to fudge, the first and last dashes are not exactly at the top and bottom of the div as some space is needed for text
var pxDownBand = percentDownBand * BAND_COLUMN_HEIGHT_PX; let pxDownBand = percentDownBand * BAND_COLUMN_HEIGHT_PX;
if (pxDownBand < lastSpotPxDownBand + BAND_COLUMN_SPOT_DIV_HEIGHT_PX) { if (pxDownBand < lastSpotPxDownBand + BAND_COLUMN_SPOT_DIV_HEIGHT_PX) {
pxDownBand = lastSpotPxDownBand + BAND_COLUMN_SPOT_DIV_HEIGHT_PX; // Prevent overlap pxDownBand = lastSpotPxDownBand + BAND_COLUMN_SPOT_DIV_HEIGHT_PX; // Prevent overlap
} }
@@ -135,8 +137,8 @@ function updateBands() {
lastSpotPxDownBand = 999999; lastSpotPxDownBand = 999999;
spotList.reverse().forEach(s => { spotList.reverse().forEach(s => {
// Work out how far down the div to draw it // Work out how far down the div to draw it
var percentDownBand = (s.freq - band.start_freq) / (band.end_freq - band.start_freq) * 0.97; // not 100% due to fudge, the first and last dashes are not exactly at the top and bottom of the div as some space is needed for text const percentDownBand = (s.freq - band.start_freq) / (band.end_freq - band.start_freq) * 0.97; // not 100% due to fudge, the first and last dashes are not exactly at the top and bottom of the div as some space is needed for text
var pxDownBand = percentDownBand * BAND_COLUMN_HEIGHT_PX; let pxDownBand = percentDownBand * BAND_COLUMN_HEIGHT_PX;
if (pxDownBand > lastSpotPxDownBand - BAND_COLUMN_SPOT_DIV_HEIGHT_PX) { if (pxDownBand > lastSpotPxDownBand - BAND_COLUMN_SPOT_DIV_HEIGHT_PX) {
pxDownBand = lastSpotPxDownBand - BAND_COLUMN_SPOT_DIV_HEIGHT_PX; // Prevent overlap pxDownBand = lastSpotPxDownBand - BAND_COLUMN_SPOT_DIV_HEIGHT_PX; // Prevent overlap
} }
@@ -149,25 +151,25 @@ function updateBands() {
// Now each spot is tagged with how far down the div it should go, add them to the DOM. // Now each spot is tagged with how far down the div it should go, add them to the DOM.
spotList.forEach(s => { spotList.forEach(s => {
let worked = alreadyWorked(s["dx_call"], s["band"], s["mode"]); let worked = alreadyWorked(s["dx_call"], s["band"], s["mode"]);
bandSpotsDiv.append(`<div class="band-spot" style="top: ${s['pxDownBandLabel']}px; border-top: 1px solid ${bandToColor(s['band'])}; border-left: 5px solid ${bandToColor(s['band'])}; border-bottom: 1px solid ${bandToColor(s['band'])}; border-right: 1px solid ${bandToColor(s['band'])}; text-decoration: ${worked ? 'line-through' : 'none'};"><span class="band-spot-call">${s.dx_call}${s.dx_ssid != null ? "-" + s.dx_ssid : ""}</span><span class="band-spot-info">${s.dx_call}${s.dx_ssid != null ? "-" + s.dx_ssid : ""} ${(s.freq/1000000).toFixed(3)} ${s.mode}</span></div>`); bandSpotsDiv.append(`<div class="band-spot" style="top: ${s['pxDownBandLabel']}px; border-top: 1px solid ${bandToColor(s['band'])}; border-left: 5px solid ${bandToColor(s['band'])}; border-bottom: 1px solid ${bandToColor(s['band'])}; border-right: 1px solid ${bandToColor(s['band'])}; text-decoration: ${worked ? 'line-through' : 'none'};"><span class="band-spot-call">${s.dx_call}${s.dx_ssid != null ? "-" + s.dx_ssid : ""}</span><span class="band-spot-info">${s.dx_call}${s.dx_ssid != null ? "-" + s.dx_ssid : ""} ${(s.freq / 1000000).toFixed(3)} ${s.mode}</span></div>`);
}); });
// Work out how tall the canvas should be. Normally this is matching the normal band column height, but if some // Work out how tall the canvas should be. Normally this is matching the normal band column height, but if some
// spots have gone off the end of the band markers and stretched their div, we need to resize the canvas to // spots have gone off the end of the band markers and stretched their div, we need to resize the canvas to
// match, otherwise we have nowhere to draw their connecting lines. // match, otherwise we have nowhere to draw their connecting lines.
var canvasHeight = Math.max(BAND_COLUMN_HEIGHT_PX, lastSpotPxDownBand + BAND_COLUMN_SPOT_DIV_HEIGHT_PX); const canvasHeight = Math.max(BAND_COLUMN_HEIGHT_PX, lastSpotPxDownBand + BAND_COLUMN_SPOT_DIV_HEIGHT_PX);
maxHeightBand = Math.max(maxHeightBand, canvasHeight); maxHeightBand = Math.max(maxHeightBand, canvasHeight);
// Draw horizontal or diagonal lines to join up the "real" frequency with where the spot div ended up // Draw horizontal or diagonal lines to join up the "real" frequency with where the spot div ended up
var bandLinesCanvas = $(`<canvas class='band-lines-canvas' width='${BAND_COLUMN_CANVAS_WIDTH_PX}px' height='${canvasHeight}px' style='height:${canvasHeight}px !important;'>`); const bandLinesCanvas = $(`<canvas class='band-lines-canvas' width='${BAND_COLUMN_CANVAS_WIDTH_PX}px' height='${canvasHeight}px' style='height:${canvasHeight}px !important;'>`);
spotList.forEach(s => { spotList.forEach(s => {
// Work out how far down the div to draw it // Work out how far down the div to draw it
var percentDownBand = (s.freq - band.start_freq) / (band.end_freq - band.start_freq) * 0.97; // not 100% due to fudge, the first and last dashes are not exactly at the top and bottom of the div as some space is needed for text const percentDownBand = (s.freq - band.start_freq) / (band.end_freq - band.start_freq) * 0.97; // not 100% due to fudge, the first and last dashes are not exactly at the top and bottom of the div as some space is needed for text
var pxDownBandFreq = (percentDownBand + 0.015) * BAND_COLUMN_HEIGHT_PX; // same fudge but add half to put the left end of the line in the right place const pxDownBandFreq = (percentDownBand + 0.015) * BAND_COLUMN_HEIGHT_PX; // same fudge but add half to put the left end of the line in the right place
var pxDownBandLabel = s["pxDownBandLabel"] + (BAND_COLUMN_SPOT_DIV_HEIGHT_PX / 1.75); // line should be to the vertical text-centre spot, not to the top corner const pxDownBandLabel = s["pxDownBandLabel"] + (BAND_COLUMN_SPOT_DIV_HEIGHT_PX / 1.75); // line should be to the vertical text-centre spot, not to the top corner
// Draw the line on the canvas // Draw the line on the canvas
var ctx = bandLinesCanvas[0].getContext('2d'); const ctx = bandLinesCanvas[0].getContext('2d');
ctx.beginPath(); ctx.beginPath();
ctx.lineWidth = 2; ctx.lineWidth = 2;
ctx.lineCap = "round"; ctx.lineCap = "round";
@@ -178,8 +180,8 @@ function updateBands() {
}); });
// Assemble the table cell // Assemble the table cell
td = $("<td>"); const td = $("<td>");
container = $("<div class='band-container'>"); const container = $("<div class='band-container'>");
container.append(bandLinesCanvas); container.append(bandLinesCanvas);
container.append(bandMarkersDiv); container.append(bandMarkersDiv);
container.append(bandSpotsDiv); container.append(bandSpotsDiv);
@@ -213,7 +215,6 @@ function removeDuplicatesForBandPanel(spotList) {
if (s.dx_call === check.dx_call && s.freq === check.freq && s.mode === check.mode) { if (s.dx_call === check.dx_call && s.freq === check.freq && s.mode === check.mode) {
// Find which one to keep and which to delete // Find which one to keep and which to delete
const checkSpotNewer = check.time > s.time; const checkSpotNewer = check.time > s.time;
const keepSpot = checkSpotNewer ? check : s;
const deleteSpot = checkSpotNewer ? s : check; const deleteSpot = checkSpotNewer ? s : check;
// Aggregate list of spots to remove // Aggregate list of spots to remove
spotsToRemove.push(deleteSpot.uid); spotsToRemove.push(deleteSpot.uid);
@@ -228,7 +229,7 @@ function removeDuplicatesForBandPanel(spotList) {
// Load server options. Once a successful callback is made from this, we then query spots and set up the timer to query // Load server options. Once a successful callback is made from this, we then query spots and set up the timer to query
// spots repeatedly. // spots repeatedly.
function loadOptions() { function loadOptions() {
$.getJSON('/api/v1/options', function(jsonData) { $.getJSON('/api/v1/options', function (jsonData) {
// Store options // Store options
options = jsonData; options = jsonData;
@@ -269,7 +270,7 @@ function displayUpdated() {
} }
// Startup // Startup
$(document).ready(function() { $(document).ready(function () {
// Call loadOptions(), this will then trigger loading spots and setting up timers. // Call loadOptions(), this will then trigger loading spots and setting up timers.
loadOptions(); loadOptions();
// Update the refresh timing display every second // Update the refresh timing display every second

View File

@@ -1,7 +1,7 @@
// Storage for the options that the server gives us. This will define our filters. // Storage for the options that the server gives us. This will define our filters.
var options = {}; let options = {};
// Last time we updated the spots/alerts list on display. // Last time we updated the spots/alerts list on display.
var lastUpdateTime; let lastUpdateTime;
// Normally load user settings from local storage, unless embedded mode is in use // Normally load user settings from local storage, unless embedded mode is in use
let useLocalStorage = true; let useLocalStorage = true;
@@ -21,8 +21,8 @@ function saveSettings() {
}); });
// Password fields are only saved if the corresponding "remember password" checkbox is ticked. // Password fields are only saved if the corresponding "remember password" checkbox is ticked.
$(".password-field").each(function () { $(".password-field").each(function () {
var pwKey = "#" + $(this)[0].id + ":value"; const pwKey = "#" + $(this)[0].id + ":value";
var rememberCheckboxId = $(this).data("remember-checkbox"); const rememberCheckboxId = $(this).data("remember-checkbox");
if (rememberCheckboxId && $("#" + rememberCheckboxId)[0] && $("#" + rememberCheckboxId)[0].checked) { if (rememberCheckboxId && $("#" + rememberCheckboxId)[0] && $("#" + rememberCheckboxId)[0].checked) {
localStorage.setItem(pwKey, JSON.stringify($(this)[0].value)); localStorage.setItem(pwKey, JSON.stringify($(this)[0].value));
} else { } else {
@@ -39,7 +39,7 @@ function loadSettings() {
Object.keys(localStorage).forEach(function (key) { Object.keys(localStorage).forEach(function (key) {
if (key.startsWith("#") && key.includes(":")) { if (key.startsWith("#") && key.includes(":")) {
// Split the key back into an element ID and a property // Split the key back into an element ID and a property
var split = key.split(":"); const split = key.split(":");
$(split[0]).prop(split[1], JSON.parse(localStorage.getItem(key))); $(split[0]).prop(split[1], JSON.parse(localStorage.getItem(key)));
} }
}); });
@@ -76,21 +76,13 @@ function loadURLParams() {
updateFilterFromParam(params, "de_continent", "de_continent"); updateFilterFromParam(params, "de_continent", "de_continent");
} }
// Update an HTML checkbox element so that its selected matches the given parameter (which must have a true or false value)
function updateCheckboxFromParam(params, paramName, checkboxID) {
let v = params.get(paramName);
if (v != null) {
$("#" + checkboxID).prop("checked", (v === "true") ? true : false);
}
}
// Update an HTML select element so that its value matches the given parameter // Update an HTML select element so that its value matches the given parameter
function updateSelectFromParam(params, paramName, selectID) { function updateSelectFromParam(params, paramName, selectID) {
let v = params.get(paramName); let v = params.get(paramName);
if (v != null) { if (v != null) {
$("#" + selectID).prop("value", v); $("#" + selectID).prop("value", v);
// Extra check if this is the "color scheme" select // Extra check if this is the "color scheme" select
if (selectID == "color-scheme") { if (selectID === "color-scheme") {
setColorScheme(v); setColorScheme(v);
} }
} }
@@ -128,16 +120,16 @@ function getSelectedFilterOptions(parameter) {
// For a parameter, such as dx_continent, return true if all possible options are enabled. (In this case, we don't need // For a parameter, such as dx_continent, return true if all possible options are enabled. (In this case, we don't need
// to bother sending this as one of the query parameters to the API; no parameter provided implies "send everything".) // to bother sending this as one of the query parameters to the API; no parameter provided implies "send everything".)
function allFilterOptionsSelected(parameter) { function allFilterOptionsSelected(parameter) {
var filter = $(".filter-button-" + parameter).filter(function () { const filter = $(".filter-button-" + parameter).filter(function () {
return !this.checked; return !this.checked;
}).get(); }).get();
return filter.length == 0; return filter.length === 0;
} }
// Generate a filter card with inline checkboxes plus All/None links. // Generate a filter card with inline checkboxes plus All/None links.
function generateMultiToggleFilterCard(elementID, filterQuery, options) { function generateMultiToggleFilterCard(elementID, filterQuery, options) {
var $row = $('<div>'); const $row = $('<div>');
options.forEach(o => { options.forEach(o => {
$row.append(`<div class="form-check form-check-inline"><input type="checkbox" class="form-check-input filter-button-${filterQuery} storeable-checkbox" id="filter-button-${filterQuery}-${o}" value="${o}" autocomplete="off" onClick="filtersUpdated()" checked><label class="form-check-label" for="filter-button-${filterQuery}-${o}">${o}</label></div>`); $row.append(`<div class="form-check form-check-inline"><input type="checkbox" class="form-check-input filter-button-${filterQuery} storeable-checkbox" id="filter-button-${filterQuery}-${o}" value="${o}" autocomplete="off" onClick="filtersUpdated()" checked><label class="form-check-label" for="filter-button-${filterQuery}-${o}">${o}</label></div>`);
}); });
@@ -161,12 +153,13 @@ function updateRefreshDisplay() {
let updatingString = "Updating..." let updatingString = "Updating..."
if (secSinceUpdate < REFRESH_INTERVAL_SEC) { if (secSinceUpdate < REFRESH_INTERVAL_SEC) {
count = REFRESH_INTERVAL_SEC - secSinceUpdate; count = REFRESH_INTERVAL_SEC - secSinceUpdate;
let number;
if (count <= 60) { if (count <= 60) {
var number = count.toFixed(0); number = count.toFixed(0);
updatingString = "<span class='nowrap'>Updating in " + number + " second" + (number != "1" ? "s" : "") + ".</span>"; updatingString = "<span class='nowrap'>Updating in " + number + " second" + (number !== "1" ? "s" : "") + ".</span>";
} else { } else {
var number = Math.round(count / 60.0).toFixed(0); number = Math.round(count / 60.0).toFixed(0);
updatingString = "<span class='nowrap'>Updating in " + number + " minute" + (number != "1" ? "s" : "") + ".</span>"; updatingString = "<span class='nowrap'>Updating in " + number + " minute" + (number !== "1" ? "s" : "") + ".</span>";
} }
} }
$("#timing-container").html("Last updated at " + lastUpdateTime.format('HH:mm') + " UTC. " + updatingString); $("#timing-container").html("Last updated at " + lastUpdateTime.format('HH:mm') + " UTC. " + updatingString);
@@ -188,7 +181,7 @@ function columnsUpdated() {
// Function to set the colour scheme based on the state of the UI select box // Function to set the colour scheme based on the state of the UI select box
function setColorSchemeFromUI() { function setColorSchemeFromUI() {
let theme = $("#color-scheme option:selected").val(); let theme = $("#color-scheme option:selected").val();
if (theme != "") { if (theme !== "") {
setColorScheme(theme); setColorScheme(theme);
saveSettings(); saveSettings();
} }
@@ -196,8 +189,8 @@ function setColorSchemeFromUI() {
// Function to set the color scheme. Supported values: "dark", "light", "auto" // Function to set the color scheme. Supported values: "dark", "light", "auto"
function setColorScheme(mode) { function setColorScheme(mode) {
let effectiveModeDark = mode == "dark"; let effectiveModeDark = mode === "dark";
if (mode == "auto") { if (mode === "auto") {
effectiveModeDark = window.matchMedia('(prefers-color-scheme: dark)').matches effectiveModeDark = window.matchMedia('(prefers-color-scheme: dark)').matches
} }
$("html").attr("data-bs-theme", effectiveModeDark ? "dark" : "light"); $("html").attr("data-bs-theme", effectiveModeDark ? "dark" : "light");
@@ -283,16 +276,16 @@ function closeDataPanel() {
// Build a query string fragment containing any QRZ.com / HamQTH credentials the user has supplied, // Build a query string fragment containing any QRZ.com / HamQTH credentials the user has supplied,
// provided the corresponding "enabled" checkbox is ticked. // provided the corresponding "enabled" checkbox is ticked.
function getCredentialQueryString() { function getCredentialQueryString() {
var str = ""; let str = "";
if ($("#qrz-enabled")[0] && $("#qrz-enabled")[0].checked) { if ($("#qrz-enabled")[0] && $("#qrz-enabled")[0].checked) {
var qrzUsername = $("#qrz-username").val(); const qrzUsername = $("#qrz-username").val();
var qrzPassword = $("#qrz-password").val(); const qrzPassword = $("#qrz-password").val();
if (qrzUsername) str += "&qrz_username=" + encodeURIComponent(qrzUsername); if (qrzUsername) str += "&qrz_username=" + encodeURIComponent(qrzUsername);
if (qrzPassword) str += "&qrz_password=" + encodeURIComponent(qrzPassword); if (qrzPassword) str += "&qrz_password=" + encodeURIComponent(qrzPassword);
} }
if ($("#hamqth-enabled")[0] && $("#hamqth-enabled")[0].checked) { if ($("#hamqth-enabled")[0] && $("#hamqth-enabled")[0].checked) {
var hamqthUsername = $("#hamqth-username").val(); const hamqthUsername = $("#hamqth-username").val();
var hamqthPassword = $("#hamqth-password").val(); const hamqthPassword = $("#hamqth-password").val();
if (hamqthUsername) str += "&hamqth_username=" + encodeURIComponent(hamqthUsername); if (hamqthUsername) str += "&hamqth_username=" + encodeURIComponent(hamqthUsername);
if (hamqthPassword) str += "&hamqth_password=" + encodeURIComponent(hamqthPassword); if (hamqthPassword) str += "&hamqth_password=" + encodeURIComponent(hamqthPassword);
} }

View File

@@ -420,7 +420,10 @@ function renderIonosondeData() {
$('#ionosonde-data-rows').hide(); $('#ionosonde-data-rows').hide();
$('#ionosonde-band-state').hide(); $('#ionosonde-band-state').hide();
$('#ionosonde-chart').hide(); $('#ionosonde-chart').hide();
if (ionosondeChart) { ionosondeChart.destroy(); ionosondeChart = null; } if (ionosondeChart) {
ionosondeChart.destroy();
ionosondeChart = null;
}
return; return;
} }
$('#ionosonde-no-data').hide(); $('#ionosonde-no-data').hide();

View File

@@ -9,12 +9,14 @@ function calcBearing(lat1, lon1, lat2, lon2) {
lon1 *= Math.PI / 180; lon1 *= Math.PI / 180;
lat2 *= Math.PI / 180; lat2 *= Math.PI / 180;
lon2 *= Math.PI / 180; lon2 *= Math.PI / 180;
var lonDelta = lon2 - lon1; const lonDelta = lon2 - lon1;
var y = Math.sin(lonDelta) * Math.cos(lat2); const y = Math.sin(lonDelta) * Math.cos(lat2);
var x = Math.cos(lat1) * Math.sin(lat2) - Math.sin(lat1) * Math.cos(lat2) * Math.cos(lonDelta); const x = Math.cos(lat1) * Math.sin(lat2) - Math.sin(lat1) * Math.cos(lat2) * Math.cos(lonDelta);
var bearing = Math.atan2(y, x); let bearing = Math.atan2(y, x);
bearing = bearing * (180 / Math.PI); bearing = bearing * (180 / Math.PI);
if ( bearing < 0 ) { bearing += 360; } if (bearing < 0) {
bearing += 360;
}
return bearing; return bearing;
} }
@@ -96,7 +98,7 @@ function latLonForGridSWCornerPlusSize(grid) {
lat -= 90.0; lat -= 90.0;
// Return nulls on maths errors // Return nulls on maths errors
if (isNaN(lat) || isNaN(lon) || isNaN(latCellSize) || isNaN(lonCellSize)) { if (isNaN(lat) || isNaN(lon) || isNaN(latCellSize) || isNaN(lonCellSize)) {
return [null, null, null, null]; return [null, null, null, null];
} }

View File

@@ -12,23 +12,23 @@ const ITU_ZONES_COLOR_DARK = 'rgba(120, 120, 60, 1.0)';
const WAB_WAI_GRID_COLOR_DARK = 'rgba(60, 60, 120, 1.0)'; const WAB_WAI_GRID_COLOR_DARK = 'rgba(60, 60, 120, 1.0)';
// Map layers // Map layers
var backgroundTileLayer; let backgroundTileLayer;
var markersLayer; let markersLayer;
var geodesicsLayer; let geodesicsLayer;
var oms; let oms;
var terminator; let terminator;
var maidenheadGrid; let maidenheadGrid;
var cqZones; let cqZones;
var ituZones; let ituZones;
var wabwaiGrid; let wabwaiGrid;
// Tracks the currently-loaded basemap provider string to avoid unnecessary tile reloads // Tracks the currently-loaded basemap provider string to avoid unnecessary tile reloads
var loadedBasemap; let loadedBasemap;
// Tracks whether this is the first display of markers after page load // Tracks whether this is the first display of markers after page load
var firstLoad = true; let firstLoad = true;
// Load spots and populate the map. // Load spots and populate the map.
function loadSpots() { function loadSpots() {
$.getJSON('/api/v1/spots' + buildQueryString(true), function(jsonData) { $.getJSON('/api/v1/spots' + buildQueryString(true), function (jsonData) {
// Store data // Store data
spots = jsonData; spots = jsonData;
// Update map // Update map
@@ -41,7 +41,7 @@ function loadSpots() {
// Build a query string for the API, based on the filters that the user has selected. // Build a query string for the API, based on the filters that the user has selected.
function buildQueryString(includeCredentials) { function buildQueryString(includeCredentials) {
var str = "?"; let str = "?";
["dx_continent", "de_continent", "mode", "source", "band", "sig"].forEach(fn => { ["dx_continent", "de_continent", "mode", "source", "band", "sig"].forEach(fn => {
if (!allFilterOptionsSelected(fn)) { if (!allFilterOptionsSelected(fn)) {
str = str + getQueryStringFor(fn) + "&"; str = str + getQueryStringFor(fn) + "&";
@@ -65,7 +65,7 @@ function updateMap() {
// Make new markers for all spots that match the filter // Make new markers for all spots that match the filter
spots.forEach(function (s) { spots.forEach(function (s) {
var m = L.marker([s["dx_latitude"], s["dx_longitude"]], {icon: getIcon(s)}); const m = L.marker([s["dx_latitude"], s["dx_longitude"]], {icon: getIcon(s)});
m.bindPopup(getTooltipText(s)); m.bindPopup(getTooltipText(s));
markersLayer.addLayer(m); markersLayer.addLayer(m);
oms.addMarker(m); oms.addMarker(m);
@@ -73,7 +73,7 @@ function updateMap() {
// Create geodesics if required // Create geodesics if required
if ($("#mapShowGeodesics")[0].checked && s["de_latitude"] != null && s["de_longitude"] != null) { if ($("#mapShowGeodesics")[0].checked && s["de_latitude"] != null && s["de_longitude"] != null) {
try { try {
var geodesic = L.geodesic([[s["de_latitude"], s["de_longitude"]], m.getLatLng()], { const geodesic = L.geodesic([[s["de_latitude"], s["de_longitude"]], m.getLatLng()], {
color: bandToColor(s['band']), color: bandToColor(s['band']),
wrap: false, wrap: false,
steps: 5 steps: 5
@@ -88,7 +88,7 @@ function updateMap() {
// On first load, zoom to the extent of the markers // On first load, zoom to the extent of the markers
if (firstLoad) { if (firstLoad) {
if (markersLayer.getLayers().length >= 2) { if (markersLayer.getLayers().length >= 2) {
var group = new L.featureGroup(markersLayer.getLayers()); const group = new L.featureGroup(markersLayer.getLayers());
map.fitBounds(group.getBounds().pad(0.1)); map.fitBounds(group.getBounds().pad(0.1));
} }
firstLoad = false; firstLoad = false;
@@ -110,48 +110,50 @@ function getIcon(s) {
// Tooltip text for the markers // Tooltip text for the markers
function getTooltipText(s) { function getTooltipText(s) {
// Format DX call // Format DX call
var dx_call = s["dx_call"]; let dx_call = s["dx_call"];
if (dx_call == null) { if (dx_call == null) {
dx_call = ""; dx_call = "";
dx_flag = "";
} }
if (s["dx_ssid"] != null) { if (s["dx_ssid"] != null) {
dx_call = dx_call + "-" + s["dx_ssid"]; dx_call = dx_call + "-" + s["dx_ssid"];
} }
// Format DX flag // Format DX flag
var dx_flag = "<i class='fa-solid fa-globe-africa'></i>"; let dx_flag = "<i class='fa-solid fa-globe-africa'></i>";
if (s["dx_flag"] && s["dx_flag"] != null && s["dx_flag"] != "") { if (dx_call == null) {
dx_flag = "";
}
if (s["dx_flag"] && s["dx_flag"] != null && s["dx_flag"] !== "") {
dx_flag = s["dx_flag"]; dx_flag = s["dx_flag"];
} }
// Format the frequency // Format the frequency
var freq_string = "Unknown" let freq_string = "Unknown";
if (s["freq"] != null) { if (s["freq"] != null) {
var mhz = Math.floor(s["freq"] / 1000000.0); const mhz = Math.floor(s["freq"] / 1000000.0);
var khz = Math.floor((s["freq"] - (mhz * 1000000.0)) / 1000.0); const khz = Math.floor((s["freq"] - (mhz * 1000000.0)) / 1000.0);
var hz = Math.floor(s["freq"] - (mhz * 1000000.0) - (khz * 1000.0)); const hz = Math.floor(s["freq"] - (mhz * 1000000.0) - (khz * 1000.0));
var hz_string = (hz > 0) ? hz.toFixed(0)[0] : ""; const hz_string = (hz > 0) ? hz.toFixed(0)[0] : "";
freq_string = `<span class='freq-mhz freq-mhz-pad'>${mhz.toFixed(0)}</span><span class='freq-khz'>${khz.toFixed(0).padStart(3, '0')}</span><span class='freq-hz hideonmobile'>${hz_string}</span>` freq_string = `<span class='freq-mhz freq-mhz-pad'>${mhz.toFixed(0)}</span><span class='freq-khz'>${khz.toFixed(0).padStart(3, '0')}</span><span class='freq-hz hideonmobile'>${hz_string}</span>`
} }
// Format comment // Format comment
var commentText = ""; let commentText = "";
if (s["comment"] != null) { if (s["comment"] != null) {
commentText = escapeHtml(s["comment"]); commentText = escapeHtml(s["comment"]);
} }
// Sig or fallback to source // Sig or fallback to source
var sigSourceText = s["source"]; let sigSourceText = s["source"];
if (s["sig"]) { if (s["sig"]) {
sigSourceText = s["sig"]; sigSourceText = s["sig"];
} }
// Format sig_refs // Format sig_refs
var sig_refs = ""; let sig_refs = "";
if (s["sig_refs"] != null) { if (s["sig_refs"] != null) {
var items = [] const items = [];
for (var i = 0; i < s["sig_refs"].length; i++) { for (let i = 0; i < s["sig_refs"].length; i++) {
if (s["sig_refs"][i]["url"] != null) { if (s["sig_refs"][i]["url"] != null) {
items[i] = `<a href='${s["sig_refs"][i]["url"]}' title='${s["sig_refs"][i]["name"]}' target='_new' class='sig-ref-link'>${s["sig_refs"][i]["id"]}</a>` items[i] = `<a href='${s["sig_refs"][i]["url"]}' title='${s["sig_refs"][i]["name"]}' target='_new' class='sig-ref-link'>${s["sig_refs"][i]["id"]}</a>`
} else { } else {
@@ -162,7 +164,7 @@ function getTooltipText(s) {
} }
// DX // DX
ttt = `<span class='nowrap'><span class='icon-wrapper'>${dx_flag}</span> <a href='https://www.qrz.com/db/${dx_call}' target='_blank' class="dx-link">${dx_call}</a></span><br/>`; let ttt = `<span class='nowrap'><span class='icon-wrapper'>${dx_flag}</span> <a href='https://www.qrz.com/db/${dx_call}' target='_blank' class="dx-link">${dx_call}</a></span><br/>`;
// Frequency & band // Frequency & band
ttt += `<span class='icon-wrapper'><i class='fa-solid fa-radio markerPopupIcon'></i></span>&nbsp;${freq_string}`; ttt += `<span class='icon-wrapper'><i class='fa-solid fa-radio markerPopupIcon'></i></span>&nbsp;${freq_string}`;
@@ -192,7 +194,7 @@ function getTooltipText(s) {
// Load server options. Once a successful callback is made from this, we then query spots and set up the timer to query // Load server options. Once a successful callback is made from this, we then query spots and set up the timer to query
// spots repeatedly. // spots repeatedly.
function loadOptions() { function loadOptions() {
$.getJSON('/api/v1/options', function(jsonData) { $.getJSON('/api/v1/options', function (jsonData) {
// Store options // Store options
options = jsonData; options = jsonData;
@@ -273,7 +275,7 @@ function setBasemap(basemapname) {
backgroundTileLayer.addTo(map); backgroundTileLayer.addTo(map);
backgroundTileLayer.bringToBack(); backgroundTileLayer.bringToBack();
if (basemapname === "OpenStreetMap.Mapnik.Dark") { if (basemapname === "OpenStreetMap.Mapnik.Dark") {
var container = backgroundTileLayer.getContainer(); const container = backgroundTileLayer.getContainer();
if (container) { if (container) {
container.style.filter = 'invert(100%) hue-rotate(180deg) brightness(80%)'; container.style.filter = 'invert(100%) hue-rotate(180deg) brightness(80%)';
} }
@@ -409,7 +411,7 @@ function setUpMap() {
backgroundTileLayer.addTo(map); backgroundTileLayer.addTo(map);
backgroundTileLayer.bringToBack(); backgroundTileLayer.bringToBack();
if (loadedBasemap === "OpenStreetMap.Mapnik.Dark") { if (loadedBasemap === "OpenStreetMap.Mapnik.Dark") {
var container = backgroundTileLayer.getContainer(); const container = backgroundTileLayer.getContainer();
if (container) { if (container) {
container.style.filter = 'invert(100%) hue-rotate(180deg) brightness(80%)'; container.style.filter = 'invert(100%) hue-rotate(180deg) brightness(80%)';
} }
@@ -421,7 +423,7 @@ function setUpMap() {
// Set up spiderfy for overlapping markers // Set up spiderfy for overlapping markers
oms = new OverlappingMarkerSpiderfier(map, {keepSpiderfied: true}); oms = new OverlappingMarkerSpiderfier(map, {keepSpiderfied: true});
oms.addListener('click', function(marker) { oms.addListener('click', function (marker) {
marker.openPopup(); marker.openPopup();
}); });
@@ -440,7 +442,7 @@ function setUpMap() {
// Add Maidenhead grid (toggleable) // Add Maidenhead grid (toggleable)
maidenheadGrid = L.maidenhead({ maidenheadGrid = L.maidenhead({
color : MAIDENHEAD_GRID_COLOR_LIGHT color: MAIDENHEAD_GRID_COLOR_LIGHT
}); });
if ($("#showMaidenheadGrid")[0].checked) { if ($("#showMaidenheadGrid")[0].checked) {
maidenheadGrid.addTo(map); maidenheadGrid.addTo(map);
@@ -449,7 +451,7 @@ function setUpMap() {
// Add CQ zone layer (toggleable) // Add CQ zone layer (toggleable)
cqZones = L.cqzones({ cqZones = L.cqzones({
color : CQ_ZONES_COLOR_LIGHT color: CQ_ZONES_COLOR_LIGHT
}); });
if ($("#showCQZones")[0].checked) { if ($("#showCQZones")[0].checked) {
cqZones.addTo(map); cqZones.addTo(map);
@@ -458,7 +460,7 @@ function setUpMap() {
// Add ITU zone layer (toggleable) // Add ITU zone layer (toggleable)
ituZones = L.ituzones({ ituZones = L.ituzones({
color : ITU_ZONES_COLOR_LIGHT color: ITU_ZONES_COLOR_LIGHT
}); });
if ($("#showITUZones")[0].checked) { if ($("#showITUZones")[0].checked) {
ituZones.addTo(map); ituZones.addTo(map);
@@ -467,7 +469,7 @@ function setUpMap() {
// Add WAB/WAI grid layer (toggleable) // Add WAB/WAI grid layer (toggleable)
wabwaiGrid = L.workedAllBritainIreland({ wabwaiGrid = L.workedAllBritainIreland({
color : WAB_WAI_GRID_COLOR_LIGHT color: WAB_WAI_GRID_COLOR_LIGHT
}); });
if ($("#showWABWAIGrid")[0].checked) { if ($("#showWABWAIGrid")[0].checked) {
wabwaiGrid.addTo(map); wabwaiGrid.addTo(map);
@@ -480,7 +482,7 @@ function setUpMap() {
} }
// Startup // Startup
$(document).ready(function() { $(document).ready(function () {
// Hide the extra things that need to be hidden on this page // Hide the extra things that need to be hidden on this page
$(".hideonmap").hide(); $(".hideonmap").hide();
// Set up map // Set up map

View File

@@ -6,7 +6,7 @@ let rowCount = 0;
// Set up a listener to close the SSE connection nicely when we navigate away from the page, to prevent console errors // Set up a listener to close the SSE connection nicely when we navigate away from the page, to prevent console errors
// and keep things nice and tidy for the server. // and keep things nice and tidy for the server.
window.addEventListener('beforeunload', function() { window.addEventListener('beforeunload', function () {
if (evtSource != null) { if (evtSource != null) {
evtSource.close(); evtSource.close();
} }
@@ -20,7 +20,7 @@ function loadSpots() {
} }
// Make the new query // Make the new query
$.getJSON('/api/v1/spots' + buildQueryString(false), function(jsonData) { $.getJSON('/api/v1/spots' + buildQueryString(false), function (jsonData) {
// Store data // Store data
spots = jsonData; spots = jsonData;
// Update table // Update table
@@ -41,9 +41,9 @@ function startSSEConnection() {
} }
evtSource = new EventSource('/api/v1/spots/stream' + buildQueryString(true)); evtSource = new EventSource('/api/v1/spots/stream' + buildQueryString(true));
evtSource.onmessage = function(event) { evtSource.onmessage = function (event) {
// Get the new spot // Get the new spot
newSpot = JSON.parse(event.data); const newSpot = JSON.parse(event.data);
// Awful fudge to ensure new incoming spots at the top of the list don't have timestamps that make them look // Awful fudge to ensure new incoming spots at the top of the list don't have timestamps that make them look
// like they belong further down the list. If the spot is older than the latest one we already have, bump its // like they belong further down the list. If the spot is older than the latest one we already have, bump its
// time up to match it. This isn't great but since we poll spot providers every 2 minutes anyway, it shouldn't // time up to match it. This isn't great but since we poll spot providers every 2 minutes anyway, it shouldn't
@@ -63,7 +63,7 @@ function startSSEConnection() {
} }
// If we had zero spots before (i.e. one now), the table will have a "No spots" row that we need to remove now // If we had zero spots before (i.e. one now), the table will have a "No spots" row that we need to remove now
// that we have one. // that we have one.
if (spots.length == 1) { if (spots.length === 1) {
$("#table tbody tr").last().remove(); $("#table tbody tr").last().remove();
} }
@@ -76,7 +76,7 @@ function startSSEConnection() {
} }
}; };
evtSource.onerror = function(err) { evtSource.onerror = function () {
if (evtSource != null) { if (evtSource != null) {
evtSource.close(); evtSource.close();
} }
@@ -87,14 +87,14 @@ function startSSEConnection() {
// Build a query string for the API, based on the filters that the user has selected. // Build a query string for the API, based on the filters that the user has selected.
function buildQueryString(includeCredentials) { function buildQueryString(includeCredentials) {
var str = "?"; let str = "?";
["dx_continent", "de_continent", "mode", "source", "band", "sig"].forEach(fn => { ["dx_continent", "de_continent", "mode", "source", "band", "sig"].forEach(fn => {
if (!allFilterOptionsSelected(fn)) { if (!allFilterOptionsSelected(fn)) {
str = str + getQueryStringFor(fn) + "&"; str = str + getQueryStringFor(fn) + "&";
} }
}); });
str = str + "limit=" + $("#spots-to-fetch option:selected").val(); str = str + "limit=" + $("#spots-to-fetch option:selected").val();
if ($("#search").val() != "") { if ($("#search").val() !== "") {
str = str + "&text_includes=" + encodeURIComponent($("#search").val()); str = str + "&text_includes=" + encodeURIComponent($("#search").val());
} }
if (includeCredentials) { if (includeCredentials) {
@@ -106,22 +106,22 @@ function buildQueryString(includeCredentials) {
// Update the spots table // Update the spots table
function updateTable() { function updateTable() {
// Use local time instead of UTC? // Use local time instead of UTC?
var useLocalTime = $("#timeZone")[0].value == "local"; const useLocalTime = $("#timeZone")[0].value === "local";
// Get user grid if valid, this will be null if it's not. // Get user grid if valid, this will be null if it's not.
var userPos = latLonForGridCentre($("#userGrid").val()); const userPos = latLonForGridCentre($("#userGrid").val());
// Table data toggles // Table data toggles
var showTime = $("#tableShowTime")[0].checked; const showTime = $("#tableShowTime")[0].checked;
var showDX = $("#tableShowDX")[0].checked; const showDX = $("#tableShowDX")[0].checked;
var showFreq = $("#tableShowFreq")[0].checked; const showFreq = $("#tableShowFreq")[0].checked;
var showMode = $("#tableShowMode")[0].checked; const showMode = $("#tableShowMode")[0].checked;
var showComment = $("#tableShowComment")[0].checked; const showComment = $("#tableShowComment")[0].checked;
var showBearing = $("#tableShowBearing")[0].checked && userPos != null; const showBearing = $("#tableShowBearing")[0].checked && userPos != null;
var showType = $("#tableShowType")[0].checked; const showType = $("#tableShowType")[0].checked;
var showRef = $("#tableShowRef")[0].checked; const showRef = $("#tableShowRef")[0].checked;
var showDE = $("#tableShowDE")[0].checked; const showDE = $("#tableShowDE")[0].checked;
var showWorkedCheckbox = $("#tableShowWorkedCheckbox")[0].checked; const showWorkedCheckbox = $("#tableShowWorkedCheckbox")[0].checked;
// Populate table with headers // Populate table with headers
let table = $("#table"); let table = $("#table");
@@ -158,7 +158,7 @@ function updateTable() {
} }
table.find('tbody').empty(); table.find('tbody').empty();
if (spots.length == 0) { if (spots.length === 0) {
table.find('tbody').append('<tr class="bg-danger-subtle"><td colspan="100" style="text-align:center;">No spots match your filters.</td></tr>'); table.find('tbody').append('<tr class="bg-danger-subtle"><td colspan="100" style="text-align:center;">No spots match your filters.</td></tr>');
} }
@@ -182,22 +182,22 @@ function addSpotToTopOfTable(s, highlightNew) {
// highlightNew = false for an initial load, true for new SSE-loaded spots // highlightNew = false for an initial load, true for new SSE-loaded spots
function createNewTableRowsForSpot(s, highlightNew) { function createNewTableRowsForSpot(s, highlightNew) {
// Use local time instead of UTC? // Use local time instead of UTC?
var useLocalTime = $("#timeZone")[0].value == "local"; const useLocalTime = $("#timeZone")[0].value === "local";
// Get user grid if valid, this will be null if it's not. // Get user grid if valid, this will be null if it's not.
var userPos = latLonForGridCentre($("#userGrid").val()); const userPos = latLonForGridCentre($("#userGrid").val());
// Table data toggles // Table data toggles
var showTime = $("#tableShowTime")[0].checked; const showTime = $("#tableShowTime")[0].checked;
var showDX = $("#tableShowDX")[0].checked; const showDX = $("#tableShowDX")[0].checked;
var showFreq = $("#tableShowFreq")[0].checked; const showFreq = $("#tableShowFreq")[0].checked;
var showMode = $("#tableShowMode")[0].checked; const showMode = $("#tableShowMode")[0].checked;
var showComment = $("#tableShowComment")[0].checked; const showComment = $("#tableShowComment")[0].checked;
var showBearing = $("#tableShowBearing")[0].checked && userPos != null; const showBearing = $("#tableShowBearing")[0].checked && userPos != null;
var showType = $("#tableShowType")[0].checked; const showType = $("#tableShowType")[0].checked;
var showRef = $("#tableShowRef")[0].checked; const showRef = $("#tableShowRef")[0].checked;
var showDE = $("#tableShowDE")[0].checked; const showDE = $("#tableShowDE")[0].checked;
var showWorkedCheckbox = $("#tableShowWorkedCheckbox")[0].checked; const showWorkedCheckbox = $("#tableShowWorkedCheckbox")[0].checked;
// Create row // Create row
let $tr = $('<tr>'); let $tr = $('<tr>');
@@ -205,13 +205,13 @@ function createNewTableRowsForSpot(s, highlightNew) {
// Apply striping to the table. We can't just use Bootstrap's table-striped class because we have all sorts of // Apply striping to the table. We can't just use Bootstrap's table-striped class because we have all sorts of
// extra faff to deal with, like the mobile view having extra rows, and the On Now / Next 24h / Later banners // extra faff to deal with, like the mobile view having extra rows, and the On Now / Next 24h / Later banners
// which cause the table-striped colouring to go awry. // which cause the table-striped colouring to go awry.
if (rowCount % 2 == 1) { if (rowCount % 2 === 1) {
$tr.addClass("table-active"); $tr.addClass("table-active");
} }
// Show faded out if QRT or already worked // Show faded out if QRT or already worked
let alreadyWorkedThis = alreadyWorked(s["dx_call"], s["band"], s["mode"]); let alreadyWorkedThis = alreadyWorked(s["dx_call"], s["band"], s["mode"]);
if (s["qrt"] == true || alreadyWorkedThis) { if (s["qrt"] === true || alreadyWorkedThis) {
$tr.addClass("table-faded"); $tr.addClass("table-faded");
} }
@@ -222,65 +222,67 @@ function createNewTableRowsForSpot(s, highlightNew) {
} }
// Format a UTC or local time for display // Format a UTC or local time for display
var time = moment.unix(s["time"]).utc(); const time = moment.unix(s["time"]).utc();
if (useLocalTime) { if (useLocalTime) {
time.local(); time.local();
} }
var time_formatted = time.format("HH:mm"); const time_formatted = time.format("HH:mm");
// Format DX call // Format DX call
var dx_call = s["dx_call"]; let dx_call = s["dx_call"];
if (dx_call == null) { if (dx_call == null) {
dx_call = ""; dx_call = "";
dx_flag = "";
} }
if (s["dx_ssid"] != null) { if (s["dx_ssid"] != null) {
dx_call = dx_call + "-" + s["dx_ssid"]; dx_call = dx_call + "-" + s["dx_ssid"];
} }
// Format dx country // Format dx country
var dx_country = s["dx_country"]; let dx_country = s["dx_country"];
if (dx_country == null) { if (dx_country == null) {
dx_country = "Unknown or not a country"; dx_country = "Unknown or not a country";
} }
// Format DX flag // Format DX flag
var dx_flag = "<i class='fa-solid fa-globe-africa'></i>"; let dx_flag = "<i class='fa-solid fa-globe-africa'></i>";
if (s["dx_dxcc_id"] && s["dx_dxcc_id"] != null && s["dx_dxcc_id"] != 0) { if (dx_call == null) {
dx_flag = "";
}
if (s["dx_dxcc_id"] && s["dx_dxcc_id"] != null && s["dx_dxcc_id"] !== 0) {
dx_flag = `<img src="img/flags/${s['dx_dxcc_id']}.png" class="flag" width="24" alt="${dx_country}" title="${dx_country}"/>`; dx_flag = `<img src="img/flags/${s['dx_dxcc_id']}.png" class="flag" width="24" alt="${dx_country}" title="${dx_country}"/>`;
} }
// Format the frequency // Format the frequency
var freq_string = "Unknown" let freq_string = "Unknown";
if (s["freq"] != null) { if (s["freq"] != null) {
var mhz = Math.floor(s["freq"] / 1000000.0); const mhz = Math.floor(s["freq"] / 1000000.0);
var khz = Math.floor((s["freq"] - (mhz * 1000000.0)) / 1000.0); const khz = Math.floor((s["freq"] - (mhz * 1000000.0)) / 1000.0);
var hz = Math.floor(s["freq"] - (mhz * 1000000.0) - (khz * 1000.0)); const hz = Math.floor(s["freq"] - (mhz * 1000000.0) - (khz * 1000.0));
var hz_string = (hz > 0) ? hz.toFixed(0)[0] : ""; const hz_string = (hz > 0) ? hz.toFixed(0)[0] : "";
freq_string = `<span class='freq-mhz freq-mhz-pad'>${mhz.toFixed(0)}</span><span class='freq-khz'>${khz.toFixed(0).padStart(3, '0')}</span><span class='freq-hz hideonmobile'>${hz_string}</span>` freq_string = `<span class='freq-mhz freq-mhz-pad'>${mhz.toFixed(0)}</span><span class='freq-khz'>${khz.toFixed(0).padStart(3, '0')}</span><span class='freq-hz hideonmobile'>${hz_string}</span>`
} }
// Format the mode // Format the mode
mode_string = s["mode"]; let mode_string = s["mode"];
if (s["mode"] == null) { if (s["mode"] == null) {
mode_string = ""; mode_string = "";
} else if (s["mode_source"] == "BANDPLAN") { } else if (s["mode_source"] === "BANDPLAN") {
mode_string = mode_string + "<span class='mode-q hideonmobile'><i class='fa-solid fa-circle-question' title='The mode was not reported via the spotting service. This is a guess based on the frequency.'></i></span>"; mode_string = mode_string + "<span class='mode-q hideonmobile'><i class='fa-solid fa-circle-question' title='The mode was not reported via the spotting service. This is a guess based on the frequency.'></i></span>";
} }
// Format comment // Format comment
var commentText = ""; let commentText = "";
if (s["comment"] != null) { if (s["comment"] != null) {
commentText = escapeHtml(s["comment"]); commentText = escapeHtml(s["comment"]);
} }
// Format bearing text // Format bearing text
var bearingText = "---<span class='bearing-q hideonmobile'><i class='fa-solid fa-circle-question' title='The position was not reported via the spotting service, and we could not determine one. A bearing to this DX is not available.'></i></span>"; let bearingText = "---<span class='bearing-q hideonmobile'><i class='fa-solid fa-circle-question' title='The position was not reported via the spotting service, and we could not determine one. A bearing to this DX is not available.'></i></span>";
if (userPos != null && s["dx_latitude"] != null && s["dx_longitude"] != null) { if (userPos != null && s["dx_latitude"] != null && s["dx_longitude"] != null) {
var bearing = calcBearing(userPos[0], userPos[1], s["dx_latitude"], s["dx_longitude"]); const bearing = calcBearing(userPos[0], userPos[1], s["dx_latitude"], s["dx_longitude"]);
bearingText = bearing.toFixed(0).padStart(3, '0') + "°"; bearingText = bearing.toFixed(0).padStart(3, '0') + "°";
if (s["dx_location_good"] == null || s["dx_location_good"] == false) { if (s["dx_location_good"] == null || s["dx_location_good"] === false) {
if (s["dx_location_source"] == "HOME QTH") { if (s["dx_location_source"] === "HOME QTH") {
bearingText = bearingText + "<span class='bearing-q hideonmobile'><i class='fa-solid fa-circle-question' title='The position was not reported via the spotting service. We had to fall back to a QRZ \"home\" location for a portable/mobile/alternative spot, so this bearing may not be accurate if the DX is close to you..'></i></span>"; bearingText = bearingText + "<span class='bearing-q hideonmobile'><i class='fa-solid fa-circle-question' title='The position was not reported via the spotting service. We had to fall back to a QRZ \"home\" location for a portable/mobile/alternative spot, so this bearing may not be accurate if the DX is close to you..'></i></span>";
} else { } else {
bearingText = bearingText + "<span class='bearing-q hideonmobile'><i class='fa-solid fa-circle-question' title='The position was not reported via the spotting service. We had to fall back to just using the centre of a DXCC entity, so this bearing may not be accurate if the DX is close to you.'></i></span>"; bearingText = bearingText + "<span class='bearing-q hideonmobile'><i class='fa-solid fa-circle-question' title='The position was not reported via the spotting service. We had to fall back to just using the centre of a DXCC entity, so this bearing may not be accurate if the DX is close to you.'></i></span>";
@@ -289,16 +291,16 @@ function createNewTableRowsForSpot(s, highlightNew) {
} }
// Format "type" (Sig or fallback to source) // Format "type" (Sig or fallback to source)
var typeText = s["source"]; let typeText = s["source"];
if (s["sig"]) { if (s["sig"]) {
typeText = s["sig"]; typeText = s["sig"];
} }
// Format sig_refs // Format sig_refs
var sig_refs = ""; let sig_refs = "";
if (s["sig_refs"] != null) { if (s["sig_refs"] != null) {
var items = [] const items = [];
for (var i = 0; i < s["sig_refs"].length; i++) { for (let i = 0; i < s["sig_refs"].length; i++) {
if (s["sig_refs"][i]["url"] != null) { if (s["sig_refs"][i]["url"] != null) {
items[i] = `<span style="white-space: nowrap;"><a href='${encodeURI(s["sig_refs"][i]["url"])}' title='${escapeHtml(s["sig_refs"][i]["name"])}' target='_new' class='sig-ref-link'>${escapeHtml(s["sig_refs"][i]["id"])}</a></span>` items[i] = `<span style="white-space: nowrap;"><a href='${encodeURI(s["sig_refs"][i]["url"])}' title='${escapeHtml(s["sig_refs"][i]["name"])}' target='_new' class='sig-ref-link'>${escapeHtml(s["sig_refs"][i]["id"])}</a></span>`
} else { } else {
@@ -309,19 +311,19 @@ function createNewTableRowsForSpot(s, highlightNew) {
} }
// Format de country // Format de country
var de_country = s["de_country"]; let de_country = s["de_country"];
if (de_country == null) { if (de_country == null) {
de_country = "Unknown or not a country"; de_country = "Unknown or not a country";
} }
// Format DE flag // Format DE flag
var de_flag = "<i class='fa-solid fa-circle-question'></i>"; let de_flag = "<i class='fa-solid fa-circle-question'></i>";
if (s["de_dxcc_id"] && s["de_dxcc_id"] != null && s["de_dxcc_id"] != 0) { if (s["de_dxcc_id"] && s["de_dxcc_id"] != null && s["de_dxcc_id"] !== 0) {
de_flag = `<img src="img/flags/${s['de_dxcc_id']}.png" class="flag" width="24" alt="${de_country}" title="${de_country}"/>`; de_flag = `<img src="img/flags/${s['de_dxcc_id']}.png" class="flag" width="24" alt="${de_country}" title="${de_country}"/>`;
} }
// Format de call // Format de call
var de_call = s["de_call"]; let de_call = s["de_call"];
if (de_call == null) { if (de_call == null) {
de_call = ""; de_call = "";
de_flag = ""; de_flag = "";
@@ -331,10 +333,10 @@ function createNewTableRowsForSpot(s, highlightNew) {
} }
// Format band name // Format band name
var bandFullName = s['band'] ? s['band'] + " band": "Unknown band"; const bandFullName = s['band'] ? s['band'] + " band" : "Unknown band";
// Format "worked" checkbox // Format "worked" checkbox
var workedCheckbox = `<input type="checkbox" ${alreadyWorkedThis ? "checked" : ""} onClick="setWorkedState('${s['dx_call']}', '${s['band']}', '${s['mode']}', ${alreadyWorkedThis ? "false" : "true"});" title="Check this box to record that you have worked this callsign on their current band and mode.">`; const workedCheckbox = `<input type="checkbox" ${alreadyWorkedThis ? "checked" : ""} onClick="setWorkedState('${s['dx_call']}', '${s['band']}', '${s['mode']}', ${alreadyWorkedThis ? "false" : "true"});" title="Check this box to record that you have worked this callsign on their current band and mode.">`;
// Populate the row // Populate the row
if (showTime) { if (showTime) {
@@ -369,21 +371,21 @@ function createNewTableRowsForSpot(s, highlightNew) {
} }
// Second row for mobile view only, containing type, ref & comment // Second row for mobile view only, containing type, ref & comment
$tr2 = $("<tr class='hidenotonmobile'>"); const $tr2 = $("<tr class='hidenotonmobile'>");
// Apply styles as per the first row // Apply styles as per the first row
if (rowCount % 2 == 1) { if (rowCount % 2 === 1) {
$tr2.addClass("table-active"); $tr2.addClass("table-active");
} }
if (s["qrt"] == true || alreadyWorkedThis) { if (s["qrt"] === true || alreadyWorkedThis) {
$tr2.addClass("table-faded"); $tr2.addClass("table-faded");
} }
if (highlightNew) { if (highlightNew) {
$tr2.addClass("new"); $tr2.addClass("new");
} }
$td2 = $("<td colspan='100'>"); const $td2 = $("<td colspan='100'>");
$td2floatleft = $(`<div style="float: left;">`); const $td2floatleft = $(`<div style="float: left;">`);
if (showType) { if (showType) {
$td2floatleft.append(`<span class='icon-wrapper'><i class='fa-solid ${sigToIcon(s["sig"], "fa-tower-cell")}'></i></span> ${typeText} `); $td2floatleft.append(`<span class='icon-wrapper'><i class='fa-solid ${sigToIcon(s["sig"], "fa-tower-cell")}'></i></span> ${typeText} `);
} }
@@ -391,7 +393,7 @@ function createNewTableRowsForSpot(s, highlightNew) {
$td2floatleft.append(`${sig_refs} `); $td2floatleft.append(`${sig_refs} `);
} }
$td2.append($td2floatleft); $td2.append($td2floatleft);
$td2floatright = $(`<div style="float: right;">`); const $td2floatright = $(`<div style="float: right;">`);
if (showBearing) { if (showBearing) {
$td2floatright.append(`${bearingText} &nbsp;`); $td2floatright.append(`${bearingText} &nbsp;`);
} }
@@ -416,7 +418,7 @@ function createNewTableRowsForSpot(s, highlightNew) {
// Load server options. Once a successful callback is made from this, we then query spots and set up the timer to query // Load server options. Once a successful callback is made from this, we then query spots and set up the timer to query
// spots repeatedly. // spots repeatedly.
function loadOptions() { function loadOptions() {
$.getJSON('/api/v1/options', function(jsonData) { $.getJSON('/api/v1/options', function (jsonData) {
// Store options // Store options
options = jsonData; options = jsonData;
@@ -459,13 +461,13 @@ function loadOptions() {
// Work out if the user's entered grid is a valid Maidenhead grid // Work out if the user's entered grid is a valid Maidenhead grid
function isUserGridValid() { function isUserGridValid() {
userGrid = $("#userGrid").val().toUpperCase(); const userGrid = $("#userGrid").val().toUpperCase();
return latLonForGridCentre(userGrid) != null; return latLonForGridCentre(userGrid) != null;
} }
// Method called when the user's grid input is changed. // Method called when the user's grid input is changed.
function userGridUpdated() { function userGridUpdated() {
var userGridValid = isUserGridValid(); const userGridValid = isUserGridValid();
if (userGridValid) { if (userGridValid) {
updateTable(); updateTable();
} }
@@ -483,7 +485,7 @@ function displayIntroBox() {
if (localStorage.getItem("intro-box-dismissed") == null) { if (localStorage.getItem("intro-box-dismissed") == null) {
$("#intro-box").show(); $("#intro-box").show();
} }
$("#intro-box-dismiss").click(function() { $("#intro-box-dismiss").click(function () {
localStorage.setItem("intro-box-dismissed", true); localStorage.setItem("intro-box-dismissed", true);
}); });
} }
@@ -510,19 +512,19 @@ function clearWorked() {
} }
// Startup // Startup
$(document).ready(function() { $(document).ready(function () {
// Call loadOptions(), this will then trigger loading spots and setting up timers. // Call loadOptions(), this will then trigger loading spots and setting up timers.
loadOptions(); loadOptions();
// Display intro box // Display intro box
displayIntroBox(); displayIntroBox();
// Set up run/pause toggles // Set up run/pause toggles
$("#runButton").change(function() { $("#runButton").change(function () {
// Need to start the SSE connection but also do a full re-query to catch up anything that we missed, so we // Need to start the SSE connection but also do a full re-query to catch up anything that we missed, so we
// might as well just call loadSpots again which will trigger it all // might as well just call loadSpots again which will trigger it all
loadSpots(); loadSpots();
}); });
$("#pauseButton").change(function() { $("#pauseButton").change(function () {
// If we are pausing and have an open SSE connection, stop it // If we are pausing and have an open SSE connection, stop it
if (evtSource != null) { if (evtSource != null) {
evtSource.close(); evtSource.close();

View File

@@ -1,5 +1,5 @@
// Storage for the spot data that the server gives us. // Storage for the spot data that the server gives us.
var spots = [] let spots = [];
// List of people the user has worked. Each entry has the format callsign-band-mode. These can be added to the list by // List of people the user has worked. Each entry has the format callsign-band-mode. These can be added to the list by
// ticking the checkbox on a row of the table, and cleared from the Display menu. Where a row would be added to the // ticking the checkbox on a row of the table, and cleared from the Display menu. Where a row would be added to the
// table and the callsign-band-mode is in this list, it is shown struck through as already worked. This is persisted // table and the callsign-band-mode is in this list, it is shown struck through as already worked. This is persisted
@@ -9,9 +9,9 @@ let worked = []
// Dynamically add CSS code for the band checkboxes to show in the appropriate colour. // Dynamically add CSS code for the band checkboxes to show in the appropriate colour.
// Some band names contain decimal points which are not allowed in CSS classes, so we text-replace them to "p". // Some band names contain decimal points which are not allowed in CSS classes, so we text-replace them to "p".
function addBandToggleColourCSS(band_options) { function addBandToggleColourCSS(band_options) {
var $style = $('<style>'); const $style = $('<style>');
band_options.forEach(o => { band_options.forEach(o => {
var domSafeName = o["name"].replace(/^[^A-Za-z0-9]+|[^\w]+/gi, ""); const domSafeName = o["name"].replace(/^[^A-Za-z0-9]+|[^\w]+/gi, "");
$style.append(`#filter-button-label-band-${domSafeName} { padding-left: 0.3em; border-left: 5px solid ${bandToColor(o['name'])};}`); $style.append(`#filter-button-label-band-${domSafeName} { padding-left: 0.3em; border-left: 5px solid ${bandToColor(o['name'])};}`);
}); });
$('html > head').append($style); $('html > head').append($style);
@@ -19,9 +19,9 @@ function addBandToggleColourCSS(band_options) {
// Generate bands filter card. This one is a special case. // Generate bands filter card. This one is a special case.
function generateBandsMultiToggleFilterCard(band_options) { function generateBandsMultiToggleFilterCard(band_options) {
var $grid = $('<div class="row row-cols-3 row-cols-md-2 row-cols-lg-3 row-cols-xxl-4 g-1 mb-1">'); const $grid = $('<div class="row row-cols-3 row-cols-md-2 row-cols-lg-3 row-cols-xxl-4 g-1 mb-1">');
band_options.forEach(o => { band_options.forEach(o => {
var domSafeName = o["name"].replace(/^[^A-Za-z0-9]+|[^\w]+/gi, ""); const domSafeName = o["name"].replace(/^[^A-Za-z0-9]+|[^\w]+/gi, "");
$grid.append(`<div class="col"><div class="form-check"><input type="checkbox" class="form-check-input filter-button-band storeable-checkbox" id="filter-button-band-${domSafeName}" value="${o['name']}" autocomplete="off" onClick="filtersUpdated()" checked> <label class="form-check-label" id="filter-button-label-band-${domSafeName}" for="filter-button-band-${domSafeName}">${o['name']}</label></div></div>`); $grid.append(`<div class="col"><div class="form-check"><input type="checkbox" class="form-check-input filter-button-band storeable-checkbox" id="filter-button-band-${domSafeName}" value="${o['name']}" autocomplete="off" onClick="filtersUpdated()" checked> <label class="form-check-label" id="filter-button-label-band-${domSafeName}" for="filter-button-band-${domSafeName}">${o['name']}</label></div></div>`);
}); });
$("#band-options").append($grid); $("#band-options").append($grid);
@@ -32,7 +32,7 @@ function generateBandsMultiToggleFilterCard(band_options) {
// widely expected by hams to be included. Special case of toggleFilterButtons(). // widely expected by hams to be included. Special case of toggleFilterButtons().
function setHamHFBandToggles() { function setHamHFBandToggles() {
const hamHFBands = ["160m", "80m", "60m", "40m", "30m", "20m", "17m", "15m", "12m", "10m", "6m"]; const hamHFBands = ["160m", "80m", "60m", "40m", "30m", "20m", "17m", "15m", "12m", "10m", "6m"];
$(".filter-button-band").each(function() { $(".filter-button-band").each(function () {
$(this).prop('checked', hamHFBands.includes($(this).val().replace("filter-button-band-", ""))); $(this).prop('checked', hamHFBands.includes($(this).val().replace("filter-button-band-", "")));
}); });
filtersUpdated(); filtersUpdated();
@@ -40,9 +40,9 @@ function setHamHFBandToggles() {
// Generate SIGs filter card. This one is also a special case. // Generate SIGs filter card. This one is also a special case.
function generateSIGsMultiToggleFilterCard(sig_options) { function generateSIGsMultiToggleFilterCard(sig_options) {
var $grid = $('<div class="row row-cols-2 row-cols-xxl-3 g-1 mb-1">'); const $grid = $('<div class="row row-cols-2 row-cols-xxl-3 g-1 mb-1">');
sig_options.forEach(o => { sig_options.forEach(o => {
var domSafeName = o["name"].replace(/^[^A-Za-z0-9]+|[^\w]+/gi, ""); const domSafeName = o["name"].replace(/^[^A-Za-z0-9]+|[^\w]+/gi, "");
$grid.append(`<div class="col"><div class="form-check"><input type="checkbox" class="form-check-input filter-button-sig storeable-checkbox" id="filter-button-sig-${domSafeName}" value="${o['name']}" autocomplete="off" onClick="filtersUpdated()" checked><label class="form-check-label" id="filter-button-label-sig-${domSafeName}" for="filter-button-sig-${domSafeName}" title="${o['description']}"><i class="fa-solid ${sigToIcon(o['name'], 'fa-tower-cell')}"></i> ${o['name']}</label></div></div>`); $grid.append(`<div class="col"><div class="form-check"><input type="checkbox" class="form-check-input filter-button-sig storeable-checkbox" id="filter-button-sig-${domSafeName}" value="${o['name']}" autocomplete="off" onClick="filtersUpdated()" checked><label class="form-check-label" id="filter-button-label-sig-${domSafeName}" for="filter-button-sig-${domSafeName}" title="${o['description']}"><i class="fa-solid ${sigToIcon(o['name'], 'fa-tower-cell')}"></i> ${o['name']}</label></div></div>`);
}); });
// Bonus "NO_SIG" / "General DX" option // Bonus "NO_SIG" / "General DX" option
@@ -53,9 +53,9 @@ function generateSIGsMultiToggleFilterCard(sig_options) {
// Generate modes filter card. This one is also a special case. // Generate modes filter card. This one is also a special case.
function generateModesMultiToggleFilterCard(mode_options) { function generateModesMultiToggleFilterCard(mode_options) {
var $grid = $('<div class="row row-cols-3 row-cols-md-2 row-cols-lg-3 g-1 mb-1">'); const $grid = $('<div class="row row-cols-3 row-cols-md-2 row-cols-lg-3 g-1 mb-1">');
mode_options.forEach(o => { mode_options.forEach(o => {
var domSafeName = o.replace(/^[^A-Za-z0-9]+|[^\w]+/gi, ""); const domSafeName = o.replace(/^[^A-Za-z0-9]+|[^\w]+/gi, "");
$grid.append(`<div class="col"><div class="form-check"><input type="checkbox" class="form-check-input filter-button-mode storeable-checkbox" id="filter-button-mode-${domSafeName}" value="${o}" autocomplete="off" onClick="filtersUpdated()" checked><label class="form-check-label" id="filter-button-label-mode-${domSafeName}" for="filter-button-mode-${domSafeName}">${o}</label></div></div>`); $grid.append(`<div class="col"><div class="form-check"><input type="checkbox" class="form-check-input filter-button-mode storeable-checkbox" id="filter-button-mode-${domSafeName}" value="${o}" autocomplete="off" onClick="filtersUpdated()" checked><label class="form-check-label" id="filter-button-label-mode-${domSafeName}" for="filter-button-mode-${domSafeName}">${o}</label></div></div>`);
}); });
$("#mode-options").append($grid); $("#mode-options").append($grid);
@@ -65,7 +65,7 @@ function generateModesMultiToggleFilterCard(mode_options) {
// Set the mode toggles that relate to Analog Voice. // Set the mode toggles that relate to Analog Voice.
function setVoiceModeToggles() { function setVoiceModeToggles() {
const modes = ["PHONE", "SSB", "LSB", "USB", "AM", "FM", "DV", "DMR", "DSTAR", "C4FM", "M17"]; const modes = ["PHONE", "SSB", "LSB", "USB", "AM", "FM", "DV", "DMR", "DSTAR", "C4FM", "M17"];
$(".filter-button-mode").each(function() { $(".filter-button-mode").each(function () {
$(this).prop('checked', modes.includes($(this).val().replace("filter-button-mode-", ""))); $(this).prop('checked', modes.includes($(this).val().replace("filter-button-mode-", "")));
}); });
filtersUpdated(); filtersUpdated();
@@ -74,7 +74,7 @@ function setVoiceModeToggles() {
// Set the mode toggles that relate to Digimodes. // Set the mode toggles that relate to Digimodes.
function setDigiModeToggles() { function setDigiModeToggles() {
const modes = ["DATA", "FT8", "FT4", "RTTY", "SSTV", "JS8", "HELL", "PSK", "OLIVIA", "PKT", "MSK144"]; const modes = ["DATA", "FT8", "FT4", "RTTY", "SSTV", "JS8", "HELL", "PSK", "OLIVIA", "PKT", "MSK144"];
$(".filter-button-mode").each(function() { $(".filter-button-mode").each(function () {
$(this).prop('checked', modes.includes($(this).val().replace("filter-button-mode-", ""))); $(this).prop('checked', modes.includes($(this).val().replace("filter-button-mode-", "")));
}); });
filtersUpdated(); filtersUpdated();
@@ -84,10 +84,10 @@ function setDigiModeToggles() {
// set which ones are enabled by default based on config rather than having them all enabled by default. We also sanitise // set which ones are enabled by default based on config rather than having them all enabled by default. We also sanitise
// names here for HTML elements. // names here for HTML elements.
function generateSourcesMultiToggleFilterCard(source_options, sources_enabled_by_default) { function generateSourcesMultiToggleFilterCard(source_options, sources_enabled_by_default) {
var $grid = $('<div class="row row-cols-2 row-cols-xxl-3 g-1 mb-1">'); const $grid = $('<div class="row row-cols-2 row-cols-xxl-3 g-1 mb-1">');
source_options.forEach(o => { source_options.forEach(o => {
var enable = sources_enabled_by_default.includes(o); const enable = sources_enabled_by_default.includes(o);
var domSafeName = o.replace(/^[^A-Za-z0-9]+|[^\w]+/gi, ""); const domSafeName = o.replace(/^[^A-Za-z0-9]+|[^\w]+/gi, "");
$grid.append(`<div class="col"><div class="form-check"><input type="checkbox" class="form-check-input filter-button-source storeable-checkbox" id="filter-button-source-${domSafeName}" value="${o}" autocomplete="off" onClick="filtersUpdated()" ${enable ? "checked" : ""}><label class="form-check-label" for="filter-button-source-${domSafeName}">${o}</label></div></div>`); $grid.append(`<div class="col"><div class="form-check"><input type="checkbox" class="form-check-input filter-button-source storeable-checkbox" id="filter-button-source-${domSafeName}" value="${o}" autocomplete="off" onClick="filtersUpdated()" ${enable ? "checked" : ""}><label class="form-check-label" for="filter-button-source-${domSafeName}">${o}</label></div></div>`);
}); });
$("#source-options").append($grid); $("#source-options").append($grid);
@@ -115,16 +115,16 @@ function alreadyWorked(callsign, band, mode) {
// Reload spots on becoming visible. This forces a refresh when used as a PWA and the user switches back to the PWA // Reload spots on becoming visible. This forces a refresh when used as a PWA and the user switches back to the PWA
// after some time has passed with it in the background. // after some time has passed with it in the background.
addEventListener("visibilitychange", (event) => { addEventListener("visibilitychange", () => {
if (!document.hidden) { if (!document.hidden) {
loadSpots(); loadSpots();
} }
}); });
// Startup // Startup
$(document).ready(function() { $(document).ready(function () {
// Load worked list // Load worked list
var tmpWorked = JSON.parse(localStorage.getItem("worked")); const tmpWorked = JSON.parse(localStorage.getItem("worked"));
if (tmpWorked) { if (tmpWorked) {
worked = tmpWorked; worked = tmpWorked;
} }

View File

@@ -1,6 +1,6 @@
// Load server status // Load server status
function loadStatus() { function loadStatus() {
$.getJSON('/api/v1/status', function(jsonData) { $.getJSON('/api/v1/status', function (jsonData) {
$("#software-version").text(jsonData["software-version"]); $("#software-version").text(jsonData["software-version"]);
$("#server-owner-callsign").text(jsonData["server-owner-callsign"]); $("#server-owner-callsign").text(jsonData["server-owner-callsign"]);
$("#up-since").text(moment().subtract(jsonData["uptime"], 'seconds').fromNow()); $("#up-since").text(moment().subtract(jsonData["uptime"], 'seconds').fromNow());
@@ -46,6 +46,6 @@ function loadStatus() {
} }
// Startup // Startup
$(document).ready(function() { $(document).ready(function () {
loadStatus(); loadStatus();
}); });

View File

@@ -4,286 +4,286 @@
// //
const BAND_COLOR_SCHEMES = { const BAND_COLOR_SCHEMES = {
"PSK Reporter": { "PSK Reporter": {
"2200m": "#ff4500", "2200m": "#ff4500",
"600m": "#1e90ff", "600m": "#1e90ff",
"160m": "#7cfc00", "160m": "#7cfc00",
"80m": "#e550e5", "80m": "#e550e5",
"60m": "#00008b", "60m": "#00008b",
"40m": "#5959ff", "40m": "#5959ff",
"30m": "#62d962", "30m": "#62d962",
"20m": "#f2c40c", "20m": "#f2c40c",
"17m": "#f2f261", "17m": "#f2f261",
"15m": "#cca166", "15m": "#cca166",
"12m": "#b22222", "12m": "#b22222",
"11m": "#00ff00", "11m": "#00ff00",
"10m": "#ff69b4", "10m": "#ff69b4",
"6m": "#FF0000", "6m": "#FF0000",
"5m": "#e0e0e0", "5m": "#e0e0e0",
"4m": "#cc0044", "4m": "#cc0044",
"2m": "#FF1493", "2m": "#FF1493",
"1.25m": "#CCFF00", "1.25m": "#CCFF00",
"70cm": "#999900", "70cm": "#999900",
"23cm": "#5AB8C7", "23cm": "#5AB8C7",
"13cm": "#FF7F50", "13cm": "#FF7F50",
"5.8GHz": "#cc0099", "5.8GHz": "#cc0099",
"10GHz": "#696969", "10GHz": "#696969",
"24GHz": "#f3edc6", "24GHz": "#f3edc6",
"47GHz": "#ffe786", "47GHz": "#ffe786",
"76GHz": "#baf9d8" "76GHz": "#baf9d8"
}, },
"PSK Reporter (Adjusted)": { "PSK Reporter (Adjusted)": {
"2200m": "#ff4500", "2200m": "#ff4500",
"600m": "#1e90ff", "600m": "#1e90ff",
"160m": "#7cfc00", "160m": "#7cfc00",
"80m": "#b33fb3", "80m": "#b33fb3",
"60m": "#00008b", "60m": "#00008b",
"40m": "#5959ff", "40m": "#5959ff",
"30m": "#62d962", "30m": "#62d962",
"20m": "#f2c40c", "20m": "#f2c40c",
"17m": "#f2f261", "17m": "#f2f261",
"15m": "#cca166", "15m": "#cca166",
"12m": "#b22222", "12m": "#b22222",
"11m": "#00ff00", "11m": "#00ff00",
"10m": "#ff7eb4", "10m": "#ff7eb4",
"6m": "#FF0000", "6m": "#FF0000",
"5m": "#e0e0e0", "5m": "#e0e0e0",
"4m": "#cc0044", "4m": "#cc0044",
"2m": "#FF1493", "2m": "#FF1493",
"1.25m": "#CCFF00", "1.25m": "#CCFF00",
"70cm": "#999900", "70cm": "#999900",
"23cm": "#5AB8C7", "23cm": "#5AB8C7",
"13cm": "#FF7F50", "13cm": "#FF7F50",
"5.8GHz": "#cc0099", "5.8GHz": "#cc0099",
"10GHz": "#696969", "10GHz": "#696969",
"24GHz": "#f3edc6", "24GHz": "#f3edc6",
"47GHz": "#ffe786", "47GHz": "#ffe786",
"76GHz": "#baf9d8" "76GHz": "#baf9d8"
}, },
"RBN": { "RBN": {
"2200m": "#000000", "2200m": "#000000",
"600m": "#aaaaaa", "600m": "#aaaaaa",
"160m": "#ffe000", "160m": "#ffe000",
"80m": "#093F00", "80m": "#093F00",
"60m": "#777777", "60m": "#777777",
"40m": "#ffa500", "40m": "#ffa500",
"30m": "#ff0000", "30m": "#ff0000",
"20m": "#800080", "20m": "#800080",
"17m": "#0000ff", "17m": "#0000ff",
"15m": "#444444", "15m": "#444444",
"12m": "#00ffff", "12m": "#00ffff",
"11m": "#000000", "11m": "#000000",
"10m": "#ff00ff", "10m": "#ff00ff",
"6m": "#ffc0cb", "6m": "#ffc0cb",
"5m": "#000000", "5m": "#000000",
"4m": "#a276ff", "4m": "#a276ff",
"2m": "#92FF7F", "2m": "#92FF7F",
"1.25m": "#000000", "1.25m": "#000000",
"70cm": "#000000", "70cm": "#000000",
"23cm": "#000000", "23cm": "#000000",
"13cm": "#000000", "13cm": "#000000",
"5.8GHz": "#000000", "5.8GHz": "#000000",
"10GHz": "#000000", "10GHz": "#000000",
"24GHz": "#000000", "24GHz": "#000000",
"47GHz": "#000000", "47GHz": "#000000",
"76GHz": "#000000" "76GHz": "#000000"
}, },
"Ham Rainbow": { "Ham Rainbow": {
"2200m": "#8e4f37", "2200m": "#8e4f37",
"600m": "#8e4f37", "600m": "#8e4f37",
"160m": "#8e3737", "160m": "#8e3737",
"80m": "#da2f93", "80m": "#da2f93",
"60m": "#792fda", "60m": "#792fda",
"40m": "#2f4bda", "40m": "#2f4bda",
"30m": "#2fdad2", "30m": "#2fdad2",
"20m": "#68da2f", "20m": "#68da2f",
"17m": "#dad52f", "17m": "#dad52f",
"15m": "#da832f", "15m": "#da832f",
"12m": "#da5c2f", "12m": "#da5c2f",
"11m": "#8e8e8e", "11m": "#8e8e8e",
"10m": "#da2f2f", "10m": "#da2f2f",
"6m": "#8e377a", "6m": "#8e377a",
"5m": "#8e8e8e", "5m": "#8e8e8e",
"4m": "#42378e", "4m": "#42378e",
"2m": "#37748e", "2m": "#37748e",
"1.25m": "#8e8e8e", "1.25m": "#8e8e8e",
"70cm": "#378e65", "70cm": "#378e65",
"23cm": "#8e8e37", "23cm": "#8e8e37",
"13cm": "#8e6037", "13cm": "#8e6037",
"5.8GHz": "#8e6037", "5.8GHz": "#8e6037",
"10GHz": "#8e6037", "10GHz": "#8e6037",
"24GHz": "#8e6037", "24GHz": "#8e6037",
"47GHz": "#8e6037", "47GHz": "#8e6037",
"76GHz": "#8e6037" "76GHz": "#8e6037"
}, },
"Ham Rainbow (Reverse)": { "Ham Rainbow (Reverse)": {
"2200m": "#42378e", "2200m": "#42378e",
"600m": "#42378e", "600m": "#42378e",
"160m": "#8e377a", "160m": "#8e377a",
"80m": "#da2f2f", "80m": "#da2f2f",
"60m": "#da5c2f", "60m": "#da5c2f",
"40m": "#da832f", "40m": "#da832f",
"30m": "#dad52f", "30m": "#dad52f",
"20m": "#68da2f", "20m": "#68da2f",
"17m": "#2fdad2", "17m": "#2fdad2",
"15m": "#2f4bda", "15m": "#2f4bda",
"12m": "#792fda", "12m": "#792fda",
"11m": "#8e8e8e", "11m": "#8e8e8e",
"10m": "#da2f93", "10m": "#da2f93",
"6m": "#8e3737", "6m": "#8e3737",
"5m": "#8e8e8e", "5m": "#8e8e8e",
"4m": "#8e4f37", "4m": "#8e4f37",
"2m": "#8e6037", "2m": "#8e6037",
"1.25m": "#8e8e8e", "1.25m": "#8e8e8e",
"70cm": "#8e8e37", "70cm": "#8e8e37",
"23cm": "#378e65", "23cm": "#378e65",
"13cm": "#37748e", "13cm": "#37748e",
"5.8GHz": "#37748e", "5.8GHz": "#37748e",
"10GHz": "#37748e", "10GHz": "#37748e",
"24GHz": "#37748e", "24GHz": "#37748e",
"47GHz": "#37748e", "47GHz": "#37748e",
"76GHz": "#37748e", "76GHz": "#37748e",
}, },
"Kate Morley": { "Kate Morley": {
"2200m": "#817", "2200m": "#817",
"600m": "#817", "600m": "#817",
"160m": "#817", "160m": "#817",
"80m": "#a35", "80m": "#a35",
"60m": "#c66", "60m": "#c66",
"40m": "#e94", "40m": "#e94",
"30m": "#ed0", "30m": "#ed0",
"20m": "#9d5", "20m": "#9d5",
"17m": "#4d8", "17m": "#4d8",
"15m": "#2cb", "15m": "#2cb",
"12m": "#0bc", "12m": "#0bc",
"11m": "#09c", "11m": "#09c",
"10m": "#09c", "10m": "#09c",
"6m": "#36b", "6m": "#36b",
"5m": "#36b", "5m": "#36b",
"4m": "#36b", "4m": "#36b",
"2m": "#36b", "2m": "#36b",
"1.25m": "#36b", "1.25m": "#36b",
"70cm": "#639", "70cm": "#639",
"23cm": "#639", "23cm": "#639",
"13cm": "#639", "13cm": "#639",
"5.8GHz": "#639", "5.8GHz": "#639",
"10GHz": "#639", "10GHz": "#639",
"24GHz": "#639", "24GHz": "#639",
"47GHz": "#639", "47GHz": "#639",
"76GHz": "#639", "76GHz": "#639",
}, },
"ColorBrewer": { "ColorBrewer": {
"2200m": "#54278f", "2200m": "#54278f",
"600m": "#756bb1", "600m": "#756bb1",
"160m": "#9e9ac8", "160m": "#9e9ac8",
"80m": "#cbc9e2", "80m": "#cbc9e2",
"60m": "#08519c", "60m": "#08519c",
"40m": "#3182bd", "40m": "#3182bd",
"30m": "#6baed6", "30m": "#6baed6",
"20m": "#bdd7e7", "20m": "#bdd7e7",
"17m": "#006d2c", "17m": "#006d2c",
"15m": "#31a354", "15m": "#31a354",
"12m": "#74c476", "12m": "#74c476",
"11m": "#bae4b3", "11m": "#bae4b3",
"10m": "#a63603", "10m": "#a63603",
"6m": "#e6550d", "6m": "#e6550d",
"5m": "#fd8d3c", "5m": "#fd8d3c",
"4m": "#fdbe85", "4m": "#fdbe85",
"2m": "#a50f15", "2m": "#a50f15",
"1.25m": "#de2d26", "1.25m": "#de2d26",
"70cm": "#fb6a4a", "70cm": "#fb6a4a",
"23cm": "#fcae91", "23cm": "#fcae91",
"13cm": "#636363", "13cm": "#636363",
"5.8GHz": "#636363", "5.8GHz": "#636363",
"10GHz": "#969696", "10GHz": "#969696",
"24GHz": "#969696", "24GHz": "#969696",
"47GHz": "#cccccc", "47GHz": "#cccccc",
"76GHz": "#cccccc", "76GHz": "#cccccc",
}, },
"IWantHue": { "IWantHue": {
"2200m": "#409271", "2200m": "#409271",
"600m": "#b03ce1", "600m": "#b03ce1",
"160m": "#50c640", "160m": "#50c640",
"80m": "#d545b7", "80m": "#d545b7",
"60m": "#99b936", "60m": "#99b936",
"40m": "#7260db", "40m": "#7260db",
"30m": "#60af57", "30m": "#60af57",
"20m": "#d54788", "20m": "#d54788",
"17m": "#58c79f", "17m": "#58c79f",
"15m": "#e2462a", "15m": "#e2462a",
"12m": "#49b1d3", "12m": "#49b1d3",
"11m": "#df872f", "11m": "#df872f",
"10m": "#506bb0", "10m": "#506bb0",
"6m": "#c6a639", "6m": "#c6a639",
"5m": "#9554a3", "5m": "#9554a3",
"4m": "#36783c", "4m": "#36783c",
"2m": "#da405b", "2m": "#da405b",
"1.25m": "#657527", "1.25m": "#657527",
"70cm": "#8c97e2", "70cm": "#8c97e2",
"23cm": "#b44f2f", "23cm": "#b44f2f",
"13cm": "#d386c8", "13cm": "#d386c8",
"5.8GHz": "#aaac66", "5.8GHz": "#aaac66",
"10GHz": "#9d4760", "10GHz": "#9d4760",
"24GHz": "#90672c", "24GHz": "#90672c",
"47GHz": "#e08086", "47GHz": "#e08086",
"76GHz": "#dc9769", "76GHz": "#dc9769",
}, },
"IWantHue (Color Blind)": { "IWantHue (Color Blind)": {
"2200m": "#bf9e3d", "2200m": "#bf9e3d",
"600m": "#9d2fec", "600m": "#9d2fec",
"160m": "#79df39", "160m": "#79df39",
"80m": "#d445db", "80m": "#d445db",
"60m": "#5dd175", "60m": "#5dd175",
"40m": "#814dd8", "40m": "#814dd8",
"30m": "#d7ce2f", "30m": "#d7ce2f",
"20m": "#657af1", "20m": "#657af1",
"17m": "#8cc34a", "17m": "#8cc34a",
"15m": "#d635aa", "15m": "#d635aa",
"12m": "#6cbd80", "12m": "#6cbd80",
"11m": "#b860c1", "11m": "#b860c1",
"10m": "#e48721", "10m": "#e48721",
"6m": "#686ccc", "6m": "#686ccc",
"5m": "#d44e2b", "5m": "#d44e2b",
"4m": "#51b3db", "4m": "#51b3db",
"2m": "#d74058", "2m": "#d74058",
"1.25m": "#56c5ad", "1.25m": "#56c5ad",
"70cm": "#d0478d", "70cm": "#d0478d",
"23cm": "#708940", "23cm": "#708940",
"13cm": "#c380c2", "13cm": "#c380c2",
"5.8GHz": "#cab775", "5.8GHz": "#cab775",
"10GHz": "#7a7fc2", "10GHz": "#7a7fc2",
"24GHz": "#b87148", "24GHz": "#b87148",
"47GHz": "#bd678c", "47GHz": "#bd678c",
"76GHz": "#c3666b", "76GHz": "#c3666b",
}, },
"Mokole": { "Mokole": {
"2200m": "#8b4513", "2200m": "#8b4513",
"600m": "#006400", "600m": "#006400",
"160m": "#808000", "160m": "#808000",
"80m": "#483d8b", "80m": "#483d8b",
"60m": "#5f9ea0", "60m": "#5f9ea0",
"40m": "#000080", "40m": "#000080",
"30m": "#9acd32", "30m": "#9acd32",
"20m": "#8b008b", "20m": "#8b008b",
"17m": "#ff0000", "17m": "#ff0000",
"15m": "#ff8c00", "15m": "#ff8c00",
"12m": "#ffd700", "12m": "#ffd700",
"11m": "#7fff00", "11m": "#7fff00",
"10m": "#8a2be2", "10m": "#8a2be2",
"6m": "#00ff7f", "6m": "#00ff7f",
"5m": "#dc143c", "5m": "#dc143c",
"4m": "#00bfff", "4m": "#00bfff",
"2m": "#0000ff", "2m": "#0000ff",
"1.25m": "#d8bfd8", "1.25m": "#d8bfd8",
"70cm": "#ff00ff", "70cm": "#ff00ff",
"23cm": "#1e90ff", "23cm": "#1e90ff",
"13cm": "#db7093", "13cm": "#db7093",
"5.8GHz": "#f0e68c", "5.8GHz": "#f0e68c",
"10GHz": "#ff1493", "10GHz": "#ff1493",
"24GHz": "#ffa07a", "24GHz": "#ffa07a",
"47GHz": "#ee82ee", "47GHz": "#ee82ee",
"76GHz": "#7fffd4", "76GHz": "#7fffd4",
} }
}; };
let bandColorScheme = "PSK Reporter (Adjusted)"; let bandColorScheme = "PSK Reporter (Adjusted)";
@@ -320,7 +320,7 @@ function bandToColor(band) {
// possible with the band colour. If the band is unknown, white will be returned. // possible with the band colour. If the band is unknown, white will be returned.
function bandToContrastColor(band) { function bandToContrastColor(band) {
const rgb = hexToRGB(bandToColor(band)); const rgb = hexToRGB(bandToColor(band));
const lum = 0.2126*rgb[0] + 0.7152*rgb[1] + 0.0722*rgb[2]; const lum = 0.2126 * rgb[0] + 0.7152 * rgb[1] + 0.0722 * rgb[2];
return (lum > 128) ? "#000000" : "#ffffff"; return (lum > 128) ? "#000000" : "#ffffff";
} }
@@ -341,28 +341,28 @@ function modeTypeToColor(modeType) {
} }
const SIG_ICONS = { const SIG_ICONS = {
"POTA": "fa-tree", "POTA": "fa-tree",
"SOTA": "fa-mountain-sun", "SOTA": "fa-mountain-sun",
"WWFF": "fa-seedling", "WWFF": "fa-seedling",
"GMA": "fa-person-hiking", "GMA": "fa-person-hiking",
"WWBOTA": "fa-radiation", "WWBOTA": "fa-radiation",
"HEMA": "fa-mound", "HEMA": "fa-mound",
"IOTA": "fa-book-atlas", "IOTA": "fa-book-atlas",
"MOTA": "fa-fan", "MOTA": "fa-fan",
"ARLHS": "fa-house-flood-water", "ARLHS": "fa-house-flood-water",
"ILLW": "fa-house-flood-water", "ILLW": "fa-house-flood-water",
"SIOTA": "fa-wheat-awn", "SIOTA": "fa-wheat-awn",
"WCA": "fa-chess-rook", "WCA": "fa-chess-rook",
"ZLOTA": "fa-kiwi-bird", "ZLOTA": "fa-kiwi-bird",
"WOTA": "fa-w", "WOTA": "fa-w",
"BOTA": "fa-umbrella-beach", "BOTA": "fa-umbrella-beach",
"KRMNPA": "fa-earth-oceania", "KRMNPA": "fa-earth-oceania",
"LLOTA": "fa-water", "LLOTA": "fa-water",
"WWTOTA": "fa-tower-observation", "WWTOTA": "fa-tower-observation",
"WAB": "fa-table-cells-large", "WAB": "fa-table-cells-large",
"WAI": "fa-table-cells-large", "WAI": "fa-table-cells-large",
"Tiles": "fa-square", "Tiles": "fa-square",
"TOTA": "fa-toilet" "TOTA": "fa-toilet"
} }
const SIG_NAMES = { const SIG_NAMES = {
@@ -396,12 +396,12 @@ function sigToIcon(sig, defaultIcon) {
if (col) { if (col) {
return col; return col;
} else { } else {
let col = (sig != null) ? SIG_ICONS[sig.toUpperCase()] : null; let col = (sig != null) ? SIG_ICONS[sig.toUpperCase()] : null;
if (col) { if (col) {
return col; return col;
} else { } else {
return defaultIcon; return defaultIcon;
} }
} }
} }

View File

@@ -11,13 +11,20 @@ function escapeHtml(str) {
const escapeCharacter = (match) => { const escapeCharacter = (match) => {
switch (match) { switch (match) {
case '&': return '&amp;'; case '&':
case '<': return '&lt;'; return '&amp;';
case '>': return '&gt;'; case '<':
case '"': return '&quot;'; return '&lt;';
case '\'': return '&#039;'; case '>':
case '`': return '&#096;'; return '&gt;';
default: return match; case '"':
return '&quot;';
case '\'':
return '&#039;';
case '`':
return '&#096;';
default:
return match;
} }
}; };
@@ -27,7 +34,7 @@ function escapeHtml(str) {
// Converts an HTML hex colour to an array of [R, G, B] where each is 0-255. // Converts an HTML hex colour to an array of [R, G, B] where each is 0-255.
function hexToRGB(hex) { function hexToRGB(hex) {
return hex.replace(/^#?([a-f\d])([a-f\d])([a-f\d])$/i return hex.replace(/^#?([a-f\d])([a-f\d])([a-f\d])$/i
,(m, r, g, b) => '#' + r + r + g + g + b + b) , (m, r, g, b) => '#' + r + r + g + g + b + b)
.substring(1).match(/.{2}/g) .substring(1).match(/.{2}/g)
.map(x => parseInt(x, 16)); .map(x => parseInt(x, 16));
} }

View File

@@ -1,49 +1,49 @@
const CACHE_NAME = 'Spothole'; const CACHE_NAME = 'Spothole';
const CACHE_URLS = [ const CACHE_URLS = [
'index.html', 'index.html',
'./', './',
'apidocs', 'apidocs',
'apidocs/openapi.yml', 'apidocs/openapi.yml',
'about', 'about',
'css/style.css', 'css/style.css',
'js/add-spot.js', 'js/add-spot.js',
'js/alerts.js', 'js/alerts.js',
'js/bands.js', 'js/bands.js',
'js/common.js', 'js/common.js',
'js/map.js', 'js/map.js',
'js/spots.js', 'js/spots.js',
'js/spotsbandsandmap.js', 'js/spotsbandsandmap.js',
'js/status.js', 'js/status.js',
'img/logo.png', 'img/logo.png',
'img/favicon.ico', 'img/favicon.ico',
'img/icon-32.png', 'img/icon-32.png',
'img/icon-192.png', 'img/icon-192.png',
'img/icon-512.png', 'img/icon-512.png',
'fa/css/fontawesome.min.css', 'fa/css/fontawesome.min.css',
'fa/css/solid.min.css', 'fa/css/solid.min.css',
'fa/webfonts/fa-solid-900.ttf', 'fa/webfonts/fa-solid-900.ttf',
'fa/webfonts/fa-solid-900.woff2' 'fa/webfonts/fa-solid-900.woff2'
]; ];
self.addEventListener('fetch', (event) => { self.addEventListener('fetch', (event) => {
// Is this an asset we can cache? // Is this an asset we can cache?
const url = new URL(event.request.url); const url = new URL(event.request.url);
const isCacheableRequest = CACHE_URLS.includes(url.pathname); const isCacheableRequest = CACHE_URLS.includes(url.pathname);
if (isCacheableRequest) { if (isCacheableRequest) {
// Open the cache // Open the cache
event.respondWith(caches.open(CACHE_NAME).then((cache) => { event.respondWith(caches.open(CACHE_NAME).then((cache) => {
// Go to the network first, cacheing the response // Go to the network first, cacheing the response
return fetch(event.request.url).then((fetchedResponse) => { return fetch(event.request.url).then((fetchedResponse) => {
cache.put(event.request, fetchedResponse.clone()); cache.put(event.request, fetchedResponse.clone());
return fetchedResponse; return fetchedResponse;
}).catch(() => { }).catch(() => {
// If the network is unavailable, get from cache. // If the network is unavailable, get from cache.
return cache.match(event.request.url); return cache.match(event.request.url);
}); });
})); }));
} else { } else {
// Not a cacheable request, must be a call to the API, so no cache involved just go to the network // Not a cacheable request, must be a call to the API, so no cache involved just go to the network
} }
}); });