Our goal is to have a combination of a dockerfile(s) and a docker-compose that will create a base image that contains the necessary MQTT modules installed along with restoring a Gateway backup on to the containers.
Where I am getting stuck is in two areas:
MQTT Module Installing (MQTT module shows up in the Gateway, but is not installed and must be installed after container is running manually - except EULA and Cert.).
Gateway back restore does not seem to work with the docker-compose I am using. The backup file mounts correctly and can be seen in the container in BASH, but the -r command does not seem to work or I have done this incorrectly.
Any suggestions out there?
Dockerfiles:
Transmitter Gateway
# Use the official Ignition image
FROM inductiveautomation/ignition:8.1.37
COPY MQTT-Transmission-signed.modl /usr/local/bin/ignition/user-lib/modules/
USER root
# Update the package list and install utilities
RUN apt-get update && apt-get install -y \
iputils-ping \
htop \
nano \
net-tools \
traceroute \
&& apt-get clean && rm -rf /var/lib/apt/lists/*
# Expose required ports
EXPOSE 8088 1883
Engine/Distriburtor Gateway
# Use the official Ignition image
FROM inductiveautomation/ignition:8.1.37
COPY MQTT-Engine-signed.modl /usr/local/bin/ignition/user-lib/modules/
COPY MQTT-Distributor-signed.modl /usr/local/bin/ignition/user-lib/modules/
USER root
# Update the package list and install utilities
RUN apt-get update && apt-get install -y \
iputils-ping \
htop \
nano \
net-tools \
traceroute \
&& apt-get clean && rm -rf /var/lib/apt/lists/*
# Expose required ports
EXPOSE 8088 1883
Docker Compose:
# Docker Compose Example for inductiveautomation/ignition
# Compose Spec: https://github.com/compose-spec/compose-spec/blob/master/spec.md
---
services:
sandbox_edge:
build:
context: .
dockerfile: Dockerfile.edge
# Transmitter Container
container_name: iX_SandBoxEdge
environment:
- ACCEPT_IGNITION_EULA=Y
- GATEWAY_ADMIN_USERNAME=admin
# - GATEWAY_ADMIN_PASSWORD_FILE=/run/secrets/gateway-admin-password
- GATEWAY_ADMIN_PASSWORD=password
- IGNITION_EDITION=edge
- TZ=America/Vancouver # see https://en.wikipedia.org/wiki/List_of_tz_database_time_zones#List
ports:
- "9088:8088" # Map HTTP port
- "9883:1883" # Map MQTT port
networks:
iX_Sandbox:
ipv4_address: 192.168.1.100
volumes:
- edge-data:/usr/local/bin/ignition/data # Mounts the local folder 'edge-data' to '/app/files' in the container
- ./:/app # Mounts the directory with your backup file
command: >
# This line does not seem to work???
-r /<your Gateway backup filename goes here>.gwbk
sandbox_cloud:
build:
context: .
dockerfile: Dockerfile.cloud
# Engine/Distributor Container
container_name: iX_SandBoxCloud
environment:
- ACCEPT_IGNITION_EULA=Y
- GATEWAY_ADMIN_USERNAME=admin
# - GATEWAY_ADMIN_PASSWORD_FILE=/run/secrets/gateway-admin-password
- GATEWAY_ADMIN_PASSWORD=password
- IGNITION_EDITION=standard
- TZ=America/Vancouver # see https://en.wikipedia.org/wiki/List_of_tz_database_time_zones#List
ports:
- "7088:8088" # Map HTTP port
- "7883:1883" # Map MQTT port
networks:
iX_Sandbox:
ipv4_address: 192.168.1.101
volumes:
- cloud-data:/usr/local/bin/ignition/data # Mounts the local folder 'cloud-data' to '/app/files' in the container
- ./:/app # Mounts the directory with your backup file
command: >
# This line does not seem to work???
-r /<your Gateway backup filename goes here>.gwbk
volumes:
cloud-data:
edge-data:
networks:
iX_Sandbox:
driver: bridge
ipam:
config:
- subnet: 192.168.1.0/24
So I seem to have made more progress with the docker-compose.yml file where I can get the MQTT Installed and the Gateway backup restored. However, I still have a "house keeping" issue with containers.
When I run this docker-compose.yml I seem to get one extra container instantiated for each of the two service (edge, cloud). Even though, I think I have explicitly prevented this from happening in the docker-compose.yml.
Once again, I may be showing my ignorance of how to properly compose things with docker. Any help would be appreciated.
# v0.1 Initial BETA iX Sandbox environment
# Compose Spec: https://github.com/compose-spec/compose-spec/blob/master/spec.md
---
services:
sandbox_edge:
build: # Build base Image
context: .
dockerfile: Dockerfile.edge
image: sandbox_edge_image:latest
entrypoint: ["echo", "This service is for building only"] # Prevent it from running as a real container
edge: # Instantiate container
image: sandbox_edge_image:latest # Use the built base image
container_name: iXSandBoxEdge
ports:
- "9088:8088" # Map HTTP port
- "9883:1883" # Map MQTT port
networks:
iXSandbox:
ipv4_address: 192.168.1.100
volumes:
- edge-data:/usr/local/bin/ignition/data # Mounts the local folder 'edge-data' to '/app/files' in the container
- ./:/app # Mounts the directory with your backup file
environment:
- ACCEPT_IGNITION_EULA=Y
- GATEWAY_ADMIN_USERNAME=admin
# - GATEWAY_ADMIN_PASSWORD_FILE=/run/secrets/gateway-admin-password
- GATEWAY_ADMIN_PASSWORD=Cogentind
- IGNITION_EDITION=edge
- TZ=America/Vancouver # see https://en.wikipedia.org/wiki/List_of_tz_database_time_zones#List
depends_on:
- sandbox_edge # Ensures the base image is built first
command: >
-r /<your Gateway backup filename goes here>.gwbk
sandbox_cloud:
build:
context: .
dockerfile: Dockerfile.cloud
image: sandbox_cloud_image:latest
entrypoint: ["echo", "This service is for building only"] # Prevent it from running as a real container
cloud:
image: sandbox_cloud_image:latest # Use the built base image
container_name: iXSandBoxCloud
ports:
- "7088:8088" # Map HTTP port
- "7883:1883" # Map MQTT port
networks:
iXSandbox:
ipv4_address: 192.168.1.101
volumes:
- cloud-data:/usr/local/bin/ignition/data # Mounts the local folder 'cloud-data' to '/app/files' in the container
- ./:/app # Mounts the directory with your backup file
environment:
- ACCEPT_IGNITION_EULA=Y
- GATEWAY_ADMIN_USERNAME=admin
# - GATEWAY_ADMIN_PASSWORD_FILE=/run/secrets/gateway-admin-password
- GATEWAY_ADMIN_PASSWORD=Cogentind
- IGNITION_EDITION=standard
- TZ=America/Vancouver # see https://en.wikipedia.org/wiki/List_of_tz_database_time_zones#List
depends_on:
- sandbox_cloud # Ensures the base image is built first
command: >
-r /<your Gateway backup filename goes here>.gwbk
volumes:
cloud-data:
edge-data:
networks:
iXSandbox:
driver: bridge
ipam:
config:
- subnet: 192.168.1.0/24
The only "state" that is tracked by Docker Compose is that which is available by inspecting labels of containers. This means it is very easy to orphan things; for example, changing service names in your compose YAML. I'm guessing that is probably what happened here. It looks like your docker-compose.yaml is probably under a very generically named folder called docker. If this lack of specificity is intentional (perhaps it is a folder under an existing repo), you can also set the Compose "project" name via the top-level name element; make sure to make these types of changes with your stack "down", again, to prevent orphaning existing containers (then requiring manual cleanup).
As to other interests you've mentioned, you might find some inspiration from these two repos I maintain:
Ok, I will use the GitHub 'iiot' package you have created and modify it according to fit our needs. One thing I did notice right away with using the files on Windows Docker, is that all the *.sh (shell script) files need to be opened and saved as (UNIX LF). Otherwise, you get strange errors that will lead you astray.
May want to mention this in the ReadMe potentially as it could take people some time to debug this issue?
Example error:
=> CACHED [gateway-edge1 prep 4/8] COPY --chmod=0755 retrieve-modules.sh . 0.0s
=> ERROR [gateway prep 5/8] RUN ./retrieve-modules.sh -m "mqttdistributor mqttengine" 2.3s
=> CACHED [gateway-edge1 prep 4/8] COPY --chmod=0755 retrieve-modules.sh . 0.0s
=> ERROR [gateway-edge1 prep 5/8] RUN ./retrieve-modules.sh -m "mqtttransmission" 2.3s
------
> [gateway prep 5/8] RUN ./retrieve-modules.sh -m "mqttdistributor mqttengine":
0.351 /usr/bin/env: ‘bash\r’: No such file or directory
------
------
> [gateway-edge1 prep 5/8] RUN ./retrieve-modules.sh -m "mqtttransmission":
0.346 /usr/bin/env: ‘bash\r’: No such file or directory
------
failed to solve: process "/usr/bin/env -S bash -euo pipefail -O inherit_errexit -c ./retrieve-modules.sh -m \"${SUPPLEMENTAL_MODULES:-}\"" did not complete successfully: exit code: 127
Ahhh, yes, good catch. I've put in .gitattributes files like over here before to ensure proper behavior (i.e. don't change shell scripts to CRLF on Windows). I'll update that for the future. Thanks!
Also, noticed that the Github 'iiot' package changing the password in the iiot\secrets\gateway-admin-password folder does not seem to take even with the
build with --no-cache to force a rebuild. I changed the gw-init/gateway.env file too as per ReadMe. I also tried setting it directly and that didn't seem to take either:
In addition, replacing the BASE_GWBK_NAME: arg with a different gateway backup does not seem to load. (assuming the iiot\gw-build is where you need to store the GW backup).
Is there a trick I am over looking here to change the password and load a gateway backup?
I think you're already aware of this, but without something like pull_policy of build, Docker Compose won't automatically rebuild the image with a docker compose up -d (alternatively, you can specify --build as an arg to up). Okay, now onto the real meat and potatoes.
This demo solution has some special handling for pre-loading the username password hash into the base gateway backup; namely, it will set that username/password against the first user source profile (id 1) and first user (id 1), ref here. If you're supplying a gateway backup that is perhaps already targeting a different user source, it might appear as though it isn't being set. Make sure that your base gwbk only has a single default internal user source (or alter the solution accordingly).
For some background, this was done in that example to show preloading a base username and password without leveraging the password reset functionality driven by specifying the GATEWAY_ADMIN_PASSWORD on a fresh launch, which will result in a temp user source.