Skip to content

Latest commit

 

History

History
256 lines (141 loc) · 14.1 KB

README.md

File metadata and controls

256 lines (141 loc) · 14.1 KB

Containerized APRS and ham radio services

Docker containers for running APRS on ham radio.

These containers primarely serve as templates that can easily be modified and repurposed into different applications for APRS and ham radio use. The goal is to have configurability via minimal code changes, minimal scripting, and rather than supporting all possible use cases with knobs, support a set of use cases while providing all flexibility via small code changes.

Goals

  • Easily modifiable and repurposable containers. Easy to repurpose via small code edits.
  • Easy, fast, and painless deployment of radio services to any new Linux system
    • Zero configuration or system alterations, only needs Docker
    • One click deployment on any Linux system
    • Only use Docker with no host scripting
  • Repeatability on any (Linux) host. Works whether you have 1 or 100 audio devices and serial ports, regardless of system resets, always use the correct audio device and serial ports. Based on work I have done for find_devices https://github.com/iontodirel/find_devices.
  • Reliability. Resilience across restarts, with health checks, handle crashes to provide 24/7 radio services.
  • Minimal internal scripting.

Limitations

  • Only Linux is currently supported. This work is targeted at small, cheap and compact Linux embedded devices, running radio services 24/7. Windows and OSX has limitations sharing hadware to Docker containers as they run in a VM.
  • The --privileged mode. Compose does not have hooks for host scripting, which means we need to run find_devices in the containers, and the containers need hardware access. We could simply run this in the host and only expose to the containers the devices required more granularly, but then we loose the usefulness and convinience of Docker Compose.

Configuration

These containers are not an E2E application, they are not meant to be taken as is without modifications and change a few flags and in a piece of GUI. These are written for maximum hackability with minimal changes.

If you don't need GPS, or repeater capabilities, or don't have the GPS hardware, simply comment these services in the compose.yaml file. Change and update anything else that you don't need. If you need a second direwolf container for RX only, you could add it in the compose file.

If you don't use a Digirig, write another configuration for find_devices to find your own sound card and/or serial port. find_devices should support any USB sound card and USB serial port device.

Running

Run from the root directory:

docker compose build
docker compose up

Troubleshooting

Build the container without caches

The Docker and Compose caches are lazy, sometimes after you make changes you need to force a cache update with:

docker compose build --no-cache

Running things into the container

This command will share a serial port and the sound subsystem to the Docker container, and run a prompt into it:

docker run -it --device /dev/ttyUSB0:/dev/ttyUSB0 --device /dev/snd --tty --entrypoint /bin/sh direwolf

NOTE: change the serial port as appropriate. You could also simply specify the --privileged instead of specifying each device to share access to.

This will build the gps container, and then run it with hardware access, and open a prompt in the container:

docker build -t gps .
docker run -it --privileged --tty --entrypoint /bin/sh gps

If you want to run everything as is with no changes to anything

This is primarily used as a documented use case, as the project was specifically designed to be hackable and accomodate many hardware and service variations.

Hardware requirements:

  • Digirig
  • Digirig cables for your radio
  • A GPS device with a USB serial port, I use a Sparkfun NEO-M9N, SMA
  • A computer to run Linux (Raspberry Pi etc.), I originally used a Pi 4, but now use a CTONE Micro PC clone from Amazon, with an Intel genuine chip
  • Your own radio choice and cables for Digirig, I use a Yaesu FT2980R
  • Power supply for the radio
  • GPS antenna, I use a Bingfu Waterproof Active SMA antenna, which is a clone from Amazon.
  • 2M antenna, I don't have space and use an HT Signal Stick SMA antenna, so I can run the radio at 30W.

Software requirements:

  • Ubuntu (have not tested other distros, but probably all works ok as long as Docker is supported)
  • Docker

Update the .env file, to set your callsign (MYCALL), and APRS-IS passcode (APRSIS_PASSCODE).

Use the find_devices utility to find your digirig serial port serial number, use the find_devices documentation https://github.com/iontodirel/find_devices/blob/main/README.md for more details.

Update the digirig_config.json file to find your sound card, based on the information from find_devices.

Run everything with docker compose build followed by docker compose up.

Volume level

Use alsamixer to one time adjust the sound level of your sound card. This is not typically necessary, and the defaults are just fine.

THAT'S IT THE END

Services

Direwolf

Provides the modem services to the Aprx service and to the host. With the Direwolf service, we are able to run an RF repeater, and we can also connect external clients like Xastir or YAAC to monitor RF ARPS traffic.

The Direwolf service has access to the host hardware, and exposes ports 8010 and 8020 to the host and outside the host. If desired, these ports can be changed in .env by updating the DIREWOLF_HOST_AGWP_PORT and DIREWOLF_HOST_KISS_PORT variables.

The service shares a private network with the Aprx service, so access between Aprx and Direwolf is not done throught the host.

The service is defined in compose.yaml.

gps

The gps service provides GPS data via GPSD.

The gps service has access to the host hardware, and exposes ports 8888 to the host. If desired, this ports can be changed in .env by updating the GPS_HOST_PORT variable.

The service shares a private network with the Aprx service, so access between Aprx and gps is not done throught the host.

A gps client like cgps can connect on the host or outside the host using cgps localhost:8888.

Aprx

Provides repeater capabilities. Uses the Direwolf and GPS services. Has access to the 14580 and 10152 ports on the host to access the APRS-IS internet service.

Containers

YOU DON'T NEED TO DO ANYTHING FROM THE BELOW, compose via compose.yaml runs and setups the accesses, ports, sharing etc. all automatically with zero configuration for the containers required.

Note that the containers can be ran standalone, but the (minimal) scripting around each container was written to support full automation and with compose in mind. You'll need to set the appropriate variables for the scripts when not using compose, or simply consume the .env file. This is pretty easy to follow, if you read the contents of the .env file defaults, and compose.yaml. You should not need to run the containers directly unless you are debugging an issue. If you don't want to use compose, you don't have to, but you'll need to create the appropriate command lines for yourself. Again, follow what compose.yaml is doing, if you want to follow the route without compose, it's not hard.

Direwolf

The direwolf container runs Direwolf, this is located in the direwolf directory. The start_direwolf.sh script is used for finding soundcard/serial port using find_devices https://github.com/iontodirel/find_devices.

The direwolf.conf contains a basic Direwolf modem configuration, the start_direwolf.sh script uses sed to set the soundcard and serial port within the configuration file.

The container needs hardware access to USB and serial ports, so we can enumerate the hardware and have Direwolf use it. You can give the container granular access to the hardware or use the --privileged mode. When used in conjunction with compose and running Direwolf as a service, only --privileged mode is supported, as compose has no host side scripting capability. If this is important to you can still use the contaner, but run find_devices externally and replace management of the container startup with your own scripting.

To access the TCP KISS and AWGPE servers from Direwolf, expose the appropriate ports when starting the container, so you have access to the services from the host.

Aprx

This container runs aprx. The aprx.conf is used to configure the aprx service and is an example of a fill-in TX igate. Change this configuration if you need more from your igate. The start_aprx.sh script uses sed to setup the callsign, KISS TCP port (from Direwolf), APRS-IS passcode, location etc. which are configured in the .env file.

The fetch_gps.sh script, uses the gps_util utility https://github.com/iontodirel/gps_util, a GPSD client, to create packets dynamically with the position fetched from a GPSD server. Thus when running the aprx container, if mobile/location capability is desired, give the container access to the port exposed by the gps container or a GPSD server (Note this happens automatically when using compose in the root). The aprx service can be configured to use a realtime GPS service, or a fixed location, based on whether you have a mobile or fixes igate. Simply comment or uncomment one of the two beacon lines in aprx.conf. When running the aprx container, give it access to ports 14580 and 10152 on the host so that aprx can access the APRS-IS service. As currently configured this is required as we beacon to APRS-IS, but if you don't want to beacon to APRS-IS and work offline with only radio RX/TX, simple update the aprx.conf.

GPS

This container runs gpsd in daemon mode. Just like the Direwolf container, find_devices is used to find the serial port corresponding to the GPS hardware. Just like the Direwolf container, this requires USB access, and when used with compose we use the --privileged mode.

The start_gpsd.sh script is used to find the serial port, and start gpsd. gpsd typically runs a server on port 2947, expose the appropriate port when running the container so you can access the service.

Connecting APRS clients

The APRS containers expose ports 8010 and 8020 for AGWP and KISS use. If you want to change these ports, change the the DIREWOLF_HOST_AGWP_PORT and DIREWOLF_HOST_KISS_PORT variables in the .env file.

Xastir

Go to Interface - Interface Control, and in the open modal Dialog, then add a new Networked AGWPE port.

image

YAAC

Go to File - Configure - Export Mode, in the open modal Dialog, then go to Ports, add a new AGWPE port, setting the hostname and port.

image

PinPoint

Go to Tools - Options, in the open modal Dialog, then go to TNC, set TNC Mode to network KISS mode, set IP address, and set Port to 8020, as this is a KISS TCP connection, and not AGWP.

image

Linux host machine configuration

The Linux system requires virtually no configuration, as our containers and compose handles everything. The only real requirements are git and Docker.

Raspberry Pi 4 with Raspbian 32-bit

Below is an example configuration done to setup the Docker APRS container to run on a fresh install Raspberry Pi.

1. Setup the system for first time use.

sudo apt-get update
sudo apt-get install git

2. Install Docker, if on Rasphbian, according to official Docker documentation https://docs.docker.com/engine/install/debian/#install-using-the-convenience-script.

curl -fsSL https://get.docker.com -o get-docker.sh
sudo sh ./get-docker.sh --dry-run

3. Configure Docker for rootless use according to official Docker documentation https://docs.docker.com/engine/security/rootless/.

sudo sh -eux <<EOF
apt-get install -y uidmap
EOF
dockerd-rootless-setuptool.sh install

4. Clone the container repo.

git clone https://github.com/iontodirel/ham_docker_container
cd ham_docker_container

5. Set or establish your configuration in the .env file.

Modify the other configuration files as needed for your application.

6. Build and run the containers. Creating the container will take a few minutes when done for the first time.

docker compose build
docker compose up

Ubuntu 64-bit on Generic Intel hardware

1. Setup the system for first time use.

sudo apt-get update
sudo apt-get install git

2. Install Docker and setup Docker for rootless use

Follow the official Docker instructions: https://docs.docker.com/engine/install/ubuntu/

Note instructions are similar to Raspbian, but are not explained to the same level of detail to keep things simple. Please just follow official Docker documentation.

3. Clone the container repo.

git clone https://github.com/iontodirel/ham_docker_container
cd ham_docker_container

4. Set or establish your configuration in the .env file.

Modify the other configuration files as needed for your application.

5. Build and run the containers. Creating the container will take a few minutes when done for the first time.

docker compose build docker compose up

Resilience

  • The audio soudcard can be connected to any USB port, depending on your audio configuration. Follow the instructions from find_devices about creating a find_devices configuration for your devices and circumstances: https://github.com/iontodirel/find_devices
  • In case of a USB disconnect, Docker will automatically handle container restart.
  • In case of Direwolf crashes, or other Direwolf failures, Docker will automatically handle container restart.
  • An internet connection is not required after setting up a system.

Handling system reboots and continuous operation

There are no needs for any hooks or scripts to run during system initialization. Docker runs automatically after boot by default, and the container policy to auto-restart, auto-restarts the container.

If you look in compose.yaml, this is handled with the restart: always policy.