diff --git a/.cargo/config.toml b/.cargo/config.toml
new file mode 100644
index 00000000..c3b54de3
--- /dev/null
+++ b/.cargo/config.toml
@@ -0,0 +1,3 @@
+[target.aarch64-unknown-linux-gnu]
+linker = "aarch64-linux-gnu-gcc"
+rustflags = [ "-C", "target-feature=+crt-static", "-C", "link-arg=-lgcc" ]
diff --git a/.github/workflows/check-spelling.yml b/.github/workflows/check-spelling.yml
index 086c23a8..61768720 100644
--- a/.github/workflows/check-spelling.yml
+++ b/.github/workflows/check-spelling.yml
@@ -21,6 +21,5 @@ jobs:
./tools/check_spelling.sh ./core/common/README.md
./tools/check_spelling.sh ./dtdl-parser/README.md
./tools/check_spelling.sh ./docs/design/README.md
- ./tools/check_spelling.sh ./samples/container/README.md
./tools/check_spelling.sh ./samples/managed_subscribe/README.md
shell: bash
diff --git a/Dockerfile b/Dockerfile.amd64
similarity index 64%
rename from Dockerfile
rename to Dockerfile.amd64
index 41e3848c..d5740afc 100644
--- a/Dockerfile
+++ b/Dockerfile.amd64
@@ -10,24 +10,37 @@
# Create a stage for building the application.
ARG RUST_VERSION=1.72.1
-FROM docker.io/library/rust:${RUST_VERSION}-slim-bullseye AS build
ARG APP_NAME=invehicle-digital-twin
+ARG FEATURES=""
+ARG UID=10001
+
+FROM docker.io/library/rust:${RUST_VERSION}-slim-bullseye AS build
+ARG APP_NAME
+ARG FEATURES
WORKDIR /sdv
COPY ./ .
+# Check that APP_NAME argument is valid.
+RUN /sdv/container/scripts/argument_sanitizer.sh \
+ --arg-value "${APP_NAME}" \
+ --regex "^[a-zA-Z_0-9-]+$" || \
+ ( echo "Argument sanitizer failed for ARG 'APP_NAME'"; exit 1 )
+
+# Check that FEATURES argument is valid if the argument is not empty.
+# The regex checks if there is one or more features separated by a single space.
+RUN if [ -n "${FEATURES}" ]; then \
+ /sdv/container/scripts/argument_sanitizer.sh \
+ --arg-value "${FEATURES}" \
+ --regex "^[a-zA-Z_0-9-]+(?: [a-zA-Z_0-9-]+)*$" || \
+ ( echo "Argument sanitizer failed for ARG 'FEATURES'"; exit 1 ) \
+ fi
+
# Add Build dependencies.
RUN apt update && apt upgrade -y && apt install -y protobuf-compiler
-# Check that APP_NAME argument is valid.
-RUN sanitized=$(echo "${APP_NAME}" | tr -dc '^[a-zA-Z_0-9-]+$'); \
-[ "$sanitized" = "${APP_NAME}" ] || { \
- echo "ARG 'APP_NAME' is invalid. APP_NAME='${APP_NAME}' sanitized='${sanitized}'"; \
- exit 1; \
-}
-
-# Build the application with the 'containerize' feature.
-RUN cargo build --release -p "${APP_NAME}" --features "containerize"
+# Build the application (with features if provided).
+RUN cargo build --release -p "${APP_NAME}" --features "${FEATURES}"
# Copy the built application to working directory.
RUN cp ./target/release/"${APP_NAME}" /sdv/service
@@ -44,6 +57,16 @@ RUN cp ./target/release/"${APP_NAME}" /sdv/service
# reproducability is important, consider using a digest
# (e.g., debian@sha256:ac707220fbd7b67fc19b112cee8170b41a9e97f703f588b2cdbbcdcecdd8af57).
FROM docker.io/library/debian:bullseye-slim AS final
+ARG UID
+
+# Copy container scripts.
+COPY ./container/scripts/*.sh /sdv/scripts/
+
+# Check that UID argument is valid.
+RUN /sdv/scripts/argument_sanitizer.sh \
+ --arg-value "${UID}" \
+ --regex "^[0-9]+$" || \
+ ( echo "Argument sanitizer failed for ARG 'UID'"; exit 1 )
# Create a non-privileged user that the app will run under.
# See https://docs.docker.com/develop/develop-images/dockerfile_best-practices/#user
@@ -56,6 +79,14 @@ RUN adduser \
--no-create-home \
--uid "${UID}" \
appuser
+
+# Create and add user ownership to config directory.
+RUN mkdir -p /sdv/config
+RUN chown appuser /sdv/config
+
+# Create mnt directory to copy override configs into.
+RUN mkdir -p /mnt/config
+
USER appuser
WORKDIR /sdv
@@ -73,4 +104,4 @@ COPY --from=build /sdv/container/config/standalone/ /sdv/config
EXPOSE 5010
# What the container should run when it is started.
-CMD ["/sdv/service"]
+CMD ["/sdv/scripts/container_startup.sh"]
diff --git a/Dockerfile.integrated b/Dockerfile.arm64
similarity index 55%
rename from Dockerfile.integrated
rename to Dockerfile.arm64
index 05c335f3..244b4e97 100644
--- a/Dockerfile.integrated
+++ b/Dockerfile.arm64
@@ -10,27 +10,44 @@
# Create a stage for building the application.
ARG RUST_VERSION=1.72.1
-FROM docker.io/library/rust:${RUST_VERSION}-slim-bullseye AS build
ARG APP_NAME=invehicle-digital-twin
+ARG FEATURES=""
+ARG UID=10001
+
+FROM docker.io/library/rust:${RUST_VERSION}-slim-bullseye AS build
+ARG APP_NAME
+ARG FEATURES
WORKDIR /sdv
COPY ./ .
+# Check that APP_NAME argument is valid.
+RUN /sdv/container/scripts/argument_sanitizer.sh \
+ --arg-value "${APP_NAME}" \
+ --regex "^[a-zA-Z_0-9-]+$" || \
+ ( echo "Argument sanitizer failed for ARG 'APP_NAME'"; exit 1 )
+
+# Check that FEATURES argument is valid if the argument is not empty.
+# The regex checks if there is one or more features separated by a single space.
+RUN if [ -n "${FEATURES}" ]; then \
+ /sdv/container/scripts/argument_sanitizer.sh \
+ --arg-value "${FEATURES}" \
+ --regex "^[a-zA-Z_0-9-]+(?: [a-zA-Z_0-9-]+)*$" || \
+ ( echo "Argument sanitizer failed for ARG 'FEATURES'"; exit 1 ) \
+ fi
+
# Add Build dependencies.
-RUN apt update && apt upgrade -y && apt install -y protobuf-compiler
+RUN apt update && apt upgrade -y && apt install -y \
+ protobuf-compiler \
+ gcc-aarch64-linux-gnu
-# Check that APP_NAME argument is valid.
-RUN sanitized=$(echo "${APP_NAME}" | tr -dc '^[a-zA-Z_0-9-]+$'); \
-[ "$sanitized" = "${APP_NAME}" ] || { \
- echo "ARG 'APP_NAME' is invalid. APP_NAME='${APP_NAME}' sanitized='${sanitized}'"; \
- exit 1; \
-}
+RUN rustup target add aarch64-unknown-linux-gnu
-# Build the application with the 'containerize' feature.
-RUN cargo build --release -p "${APP_NAME}" --features "containerize managed_subscribe"
+# Build the application (with features if provided).
+RUN cargo build --release --target=aarch64-unknown-linux-gnu -p "${APP_NAME}" --features "${FEATURES}"
# Copy the built application to working directory.
-RUN cp ./target/release/"${APP_NAME}" /sdv/service
+RUN cp ./target/aarch64-unknown-linux-gnu/release/"${APP_NAME}" /sdv/service
################################################################################
# Create a new stage for running the application that contains the minimal
@@ -43,7 +60,17 @@ RUN cp ./target/release/"${APP_NAME}" /sdv/service
# most recent version of that tag when you build your Dockerfile. If
# reproducability is important, consider using a digest
# (e.g., debian@sha256:ac707220fbd7b67fc19b112cee8170b41a9e97f703f588b2cdbbcdcecdd8af57).
-FROM docker.io/library/debian:bullseye-slim AS final
+FROM docker.io/arm64v8/debian:bullseye-slim AS final
+ARG UID
+
+# Copy container scripts.
+COPY ./container/scripts/*.sh /sdv/scripts/
+
+# Check that UID argument is valid.
+RUN /sdv/scripts/argument_sanitizer.sh \
+ --arg-value "${UID}" \
+ --regex "^[0-9]+$" || \
+ ( echo "Argument sanitizer failed for ARG 'UID'"; exit 1 )
# Create a non-privileged user that the app will run under.
# See https://docs.docker.com/develop/develop-images/dockerfile_best-practices/#user
@@ -56,6 +83,14 @@ RUN adduser \
--no-create-home \
--uid "${UID}" \
appuser
+
+# Create and add user ownership to config directory.
+RUN mkdir -p /sdv/config
+RUN chown appuser /sdv/config
+
+# Create mnt directory to copy override configs into.
+RUN mkdir -p /mnt/config
+
USER appuser
WORKDIR /sdv
@@ -67,10 +102,10 @@ ENV IBEJI_HOME=/sdv/config
COPY --from=build /sdv/service /sdv/
# Copy configuration for service.
-COPY --from=build /sdv/container/config/integrated/ /sdv/config
+COPY --from=build /sdv/container/config/standalone/ /sdv/config
# Expose the port that the application listens on.
EXPOSE 5010
# What the container should run when it is started.
-CMD ["/sdv/service"]
+CMD ["/sdv/scripts/container_startup.sh"]
diff --git a/README.md b/README.md
index 75ce3775..1f8e0d5b 100644
--- a/README.md
+++ b/README.md
@@ -310,9 +310,8 @@ rather than having it statically provided in their respective config file, then
## Running in a Container
-To run the In-Vehicle Digital Twin Service in a container, please refer to [Ibeji Containerization](./container/README.md).
-
-To run the samples in a container, please refer to [Samples Containerization](./samples/container/README.md).
+Please refer to [Ibeji Containers](./container/README.md#ibeji-containers) for information on how
+build and run the In-Vehicle Digital Twin Service or the Sample Applications in a container.
## Trademarks
diff --git a/container/.accepted_words.txt b/container/.accepted_words.txt
index 4e86eebb..5158ffe4 100644
--- a/container/.accepted_words.txt
+++ b/container/.accepted_words.txt
@@ -1,5 +1,9 @@
+aarch
Agemo
agemo
+amd
+APP
+arg
br
build
cargo
@@ -54,6 +58,7 @@ loopback
md
microsoft
minimalistic
+mnt
mosquitto
Mosquitto
mqtt
diff --git a/container/README.md b/container/README.md
index 06d9363e..a2fe9716 100644
--- a/container/README.md
+++ b/container/README.md
@@ -1,119 +1,271 @@
-# Ibeji Containerization
+# Ibeji Containers
-This document covers how to containerize the In-Vehicle Digital Twin Service. To run the samples as
-a container please refer to [Samples Containerization](../samples/container/README.md).
+This document covers how to containerize the services provided in this repository.
-## Running the In-Vehicle Digital Twin Service in a Container
+## Dockerfile Selection
-Below are the steps for running the service in a container. Note that the configuration files used
-by the containerized service are cloned from [/container/config](./config/) defined in the
-project's root.
+### In-Vehicle Digital Twin Service
-### Dockerfile
+To containerize the [In-Vehicle Digital Twin Service](../core/invehicle-digital-twin/) use one of
+the following dockerfiles:
-There are currently two dockerfiles provided in the root directory of the project that can be built:
+- [Dockerfile.amd64](../Dockerfile.amd64) - For x86-64 architecture.
+- [Dockerfile.arm64](../Dockerfile.arm64) - For aarch64 architecture.
-- Dockerfile - A standalone version of the In-Vehicle Digital Twin Service
-- Dockerfile.integrated - A version of the In-Vehicle Digital Twin Service that communicates with
-the [Chariott Service](https://github.com/eclipse-chariott/chariott) and the
-[Agemo Service](https://github.com/eclipse-chariott/Agemo).
+### Sample Applications
-### Docker
+To containerize one of the [Ibeji Sample Applications](../samples/) use one of the following
+dockerfiles:
-#### Prerequisites
+- [./samples/Dockerfile.samples.amd64](../samples/Dockerfile.samples.amd64) - For x86-64
+architecture.
+- [./samples/Dockerfile.samples.arm64](../samples/Dockerfile.samples.arm64) - For aarch64
+architecture.
+
+## Configuration Defaults
+
+### In-Vehicle Digital Twin Service
+
+This repository provides default configuration files for the In-Vehicle Digital Twin Service
+running in [Standalone](../container/config/standalone/) and
+[Integrated](../container/config/integrated) modes.
+
+>Note: Integrated mode runs the In-Vehicle Digital Twin Service with the
+[Chariott Service](https://github.com/eclipse-chariott/chariott) and the
+[Agemo Service](https://github.com/eclipse-chariott/Agemo). Other combinations may require config
+changes.
+
+### Sample Applications
+
+For the samples, this repository provides
+default configuration files for the [Property Provider](../samples/container/config/provider/) and
+[Property Consumer](../samples/container/config/consumer/) samples.
+
+>Note: Other samples may need config changes (see
+[Running the Samples](../README.md#running-the-samples) for config changes).
+
+## Docker Containers
+
+### Prerequisites
[Install Docker](https://docs.docker.com/engine/install/)
-#### Running in Docker
+### Running in Docker
To run the service in a Docker container:
-1. Run the following command in the project's root directory to build the docker container from the
+>Note: Before running any of the following commands, replace all placeholders (wrapped with `<>`).
+
+1. Run the following command in the project root directory to build the docker container from the
Dockerfile:
```shell
- docker build -t invehicle_digital_twin -f Dockerfile .
+ docker build -t -f [--build-arg=APP_NAME=] [--build-arg=FEATURES=""] .
+ ```
+
+ >Note: The `FEATURES` build arg only applies to the dockerfiles for In-Vehicle Digital Twin Service.
+
+ For example, to build an image for the `invehicle-digital-twin` project in `Standalone` mode:
+
+ ```shell
+ docker build -t invehicle_digital_twin -f Dockerfile.amd64 .
+ ```
+
+ Or to build an image for the `invehicle-digital-twin` project with the `Managed Subscribe`
+ module:
+
+ ```shell
+ docker build -t invehicle_digital_twin -f Dockerfile.amd64 --build-arg=FEATURES="managed_subscribe" .
+ ```
+
+ >Note: Modules for Ibeji are enabled via the `--features` flag. The `FEATURES` build arg passes
+ one or more features to the cargo build argument. If you want to add multiple features, add
+ each feature followed by a space e.g. `"module_1 module_2"`.
+
+ Or to build an image for the `property-provider` sample for aarch64:
+
+ ```shell
+ docker build -t property_provider -f ./samples/Dockerfile.samples.arm64 --build-arg=APP_NAME=property-provider .
```
+ >Note: The build arg `APP_NAME` needs to be passed in for all sample applications to build the
+ correct sample.
+
1. Once the container has been built, start the container in interactive mode with the following
-command in the project's root directory:
+command in the project root directory:
+
+ ```shell
+ docker run --name --network=host -it --rm
+ ```
+
+ >Note: Most images built will require configuration overrides. See
+ [Running in Docker with overridden configuration](#running-in-docker-with-overridden-configuration)
+ for more details.
+
+ For example, to run the `invehicle_digital_twin` standalone image built in step 1:
```shell
- docker run --name invehicle_digital_twin -p 5010:5010 --env-file=./container/config/docker.env --add-host=host.docker.internal:host-gateway -it --rm invehicle_digital_twin
+ docker run --name invehicle_digital_twin --network=host -it --rm invehicle_digital_twin
```
+ >Note: A custom network is recommended when using a container for anything but testing.
+
1. To detach from the container, enter:
Ctrl + p, Ctrl + q
1. To stop the container, enter:
+ ```shell
+ docker stop
+ ```
+
+ For example, to stop the `invehicle_digital_twin` container started in step 2:
+
```shell
docker stop invehicle_digital_twin
```
-#### Running in Docker with overridden configuration
+### Running in Docker with overridden configuration
Follow the steps in [Running in Docker](#running-in-docker) to build the container.
1. To run the container with overridden configuration, create your config file and set an
-environment variable called IBEJI_HOME to the absolute path of the directory containing the
-config file:
+environment variable called CONFIG_HOME to the path to the config file:
```shell
- export IBEJI_HOME={absolute path of the directory containing the config file}
+ export CONFIG_HOME={path to directory containing config file}
```
+ For example, to set the configuration for the
+ [Property Provider](../samples/property/provider/) sample, run:
+
+ ```shell
+ export CONFIG_HOME={path-to-repo-root}/samples/container/config/provider
+ ```
+
+ >Note: See [Configuration Defaults](#configuration-defaults) for more information. If running
+ a sample other than the `Property` sample you will need to create your own configuration files.
+
1. Then run the container with the following command:
```shell
- docker run -v ${IBEJI_HOME}:/sdv/config --name invehicle_digital_twin -p 5010:5010 --env-file=./container/config/docker.env --add-host=host.docker.internal:host-gateway -it --rm invehicle_digital_twin
+ docker run -v ${CONFIG_HOME}:/mnt/config --name --network=host -it --rm
```
-### Podman
+ For example, to run the `property_provider` image with overridden configuration:
-#### Prerequisites
+ ```shell
+ docker run -v ${CONFIG_HOME}:/mnt/config --name property_provider --network=host -it --rm property_provider
+ ```
+
+## Podman
+
+### Prerequisites
[Install Podman](https://podman.io/docs/installation)
-#### Running in Podman
+### Running in Podman
To run the service in a Podman container:
-1. Run the following command in the project's root directory to build the podman container from the
+>Note: Before running any of the following commands, replace all placeholders (wrapped with `<>`).
+
+1. Run the following command in the project root directory to build the podman container from the
Dockerfile:
```shell
- podman build -t invehicle_digital_twin:latest -f Dockerfile .
+ podman build -t -f [--build-arg=APP_NAME=] [--build-arg=FEATURES=""] .
+ ```
+
+ >Note: The `FEATURES` build arg only applies to the dockerfiles for In-Vehicle Digital Twin Service.
+
+ For example, to build an image for the `invehicle-digital-twin` project in `Standalone` mode:
+
+ ```shell
+ podman build -t invehicle_digital_twin -f Dockerfile.amd64 .
```
-1. Once the container has been built, start the container with the following command in the
-project's root directory:
+ Or to build an image for the `invehicle-digital-twin` project with the `Managed Subscribe`
+ module:
+
+ ```shell
+ podman build -t invehicle_digital_twin -f Dockerfile.amd64 --build-arg=FEATURES="managed_subscribe" .
+ ```
+
+ >Note: Modules for Ibeji are enabled via the `--features` flag. The `FEATURES` build arg passes
+ one or more features to the cargo build argument. If you want to add multiple features, add
+ each feature followed by a space e.g. `"module_1 module_2"`.
+
+ Or to build an image for the `property-provider` sample for aarch64:
+
+ ```shell
+ podman build -t property_provider -f ./samples/Dockerfile.samples.arm64 --build-arg=APP_NAME=property-provider .
+ ```
+
+ >Note: The build arg `APP_NAME` needs to be passed in for all sample applications to build the
+ correct sample.
+
+1. Once the container has been built, start the container with the following command in the project
+root directory:
```shell
- podman run -p 5010:5010 --env-file=./container/config/podman.env --network=slirp4netns:allow_host_loopback=true localhost/invehicle_digital_twin
+ podman run --network=host
```
+ >Note: Most images built will require configuration overrides. See
+ [Running in Podman with overridden configuration](#running-in-podman-with-overridden-configuration)
+ for more details.
+
+ For example, to run the `invehicle_digital_twin` image built in step 1:
+
+ ```shell
+ podman run --network=host invehicle_digital_twin
+ ```
+
+ >Note: A custom network is recommended when using a container for anything but testing.
+
1. To stop the container, run:
+ ```shell
+ podman ps -f ancestor= --format="{{.Names}}" | xargs podman stop
+ ```
+
+ For example, to stop the `invehicle_digital_twin` container started in step 2:
+
```shell
podman ps -f ancestor=localhost/invehicle_digital_twin:latest --format="{{.Names}}" | xargs podman stop
```
-#### Running in Podman with overridden configuration
+### Running in Podman with overridden configuration
Follow the steps in [Running in Podman](#running-in-podman) to build the container.
1. To run the container with overridden configuration, create your config file and set an
-environment variable called IBEJI_HOME to the absolute path of the directory containing the
-config file:
+environment variable called CONFIG_HOME to the path to the config file:
```shell
- export IBEJI_HOME={absolute path of the directory containing the config file}
+ export CONFIG_HOME={path to directory containing config file}
```
+ For example, to set the configuration for the
+ [Property Provider](../samples/property/provider/) sample, run:
+
+ ```shell
+ export CONFIG_HOME={path-to-repo-root}/samples/container/config/provider
+ ```
+
+ >Note: See [Configuration Defaults](#configuration-defaults) for more information. If running
+ a sample other than the `Property` sample you will need to create your own configuration files.
+
1. Then run the container with the following command:
```shell
- podman run --mount=type=bind,src=${IBEJI_HOME},dst=/sdv/config,ro=true -p 5010:5010 --env-file=./container/config/podman.env --network=slirp4netns:allow_host_loopback=true localhost/invehicle_digital_twin:latest
+ podman run --mount=type=bind,src=${CONFIG_HOME},dst=/mnt/config,ro=true --network=host
+ ```
+
+ For example, to run the `property_provider` image with overridden configuration:
+
+ ```shell
+ podman run --mount=type=bind,src=${CONFIG_HOME},dst=/mnt/config,ro=true --network=host property_provider
```
diff --git a/container/config/docker.env b/container/config/docker.env
deleted file mode 100644
index 56bafc6e..00000000
--- a/container/config/docker.env
+++ /dev/null
@@ -1,9 +0,0 @@
-# Copyright (c) Microsoft Corporation.
-# Licensed under the MIT license.
-# SPDX-License-Identifier: MIT
-
-# DNS name used by the container to communicate with host.
-HOST_GATEWAY=host.docker.internal
-
-# Alias for localhost to be replaced by HOST_GATEWAY if run in a container.
-LOCALHOST_ALIAS=0.0.0.0
diff --git a/container/config/podman.env b/container/config/podman.env
deleted file mode 100644
index 1abc4a35..00000000
--- a/container/config/podman.env
+++ /dev/null
@@ -1,9 +0,0 @@
-# Copyright (c) Microsoft Corporation.
-# Licensed under the MIT license.
-# SPDX-License-Identifier: MIT
-
-# DNS name used by the container to communicate with host.
-HOST_GATEWAY=host.containers.internal
-
-# Alias for localhost to be replaced by HOST_GATEWAY if run in a container.
-LOCALHOST_ALIAS=0.0.0.0
diff --git a/container/scripts/argument_sanitizer.sh b/container/scripts/argument_sanitizer.sh
new file mode 100755
index 00000000..d31613d4
--- /dev/null
+++ b/container/scripts/argument_sanitizer.sh
@@ -0,0 +1,61 @@
+#!/bin/bash
+
+# Copyright (c) Microsoft Corporation.
+# Licensed under the MIT license.
+# SPDX-License-Identifier: MIT
+
+# Exits immediately on failure.
+set -eu
+
+# Function to display usage information
+usage() {
+ echo "Usage: $0 [-a|--arg-value] [-r|--regex] "
+ echo "Example:"
+ echo " $0 -a \"\${APP_NAME}\" -r \"^[a-zA-Z_0-9-]+$\""
+}
+
+# Parse command line arguments
+while [[ $# -gt 0 ]]
+do
+ key="$1"
+
+ case $key in
+ -a|--arg-value)
+ arg_value="$2"
+ shift # past argument
+ shift # past value
+ ;;
+ -r|--regex)
+ regex="$2"
+ shift # past argument
+ shift # past value
+ ;;
+ -h|--help)
+ usage
+ exit 0
+ ;;
+ *)
+ echo "Unknown argument: $key"
+ usage
+ exit 1
+ esac
+done
+
+# Check if all required arguments have been set
+if [[ -z "${arg_value}" || -z "${regex}" ]]; then
+ echo "Error: Missing required arguments:"
+ [[ -z "${arg_value}" ]] && echo " -a|--arg-value"
+ [[ -z "${regex}" ]] && echo " -r|--regex"
+ echo -e "\n"
+ usage
+ exit 1
+fi
+
+sanitized=$(echo "${arg_value}" | tr -dc "${regex}");
+[ "$sanitized" = "${arg_value}" ] || {
+ echo "ARG is invalid. ARG='${arg_value}' sanitized='${sanitized}'";
+ exit 1
+}
+
+echo -e "\nARG with value '${arg_value}' is sanitized"
+exit 0
\ No newline at end of file
diff --git a/container/scripts/container_startup.sh b/container/scripts/container_startup.sh
new file mode 100755
index 00000000..c5f6350a
--- /dev/null
+++ b/container/scripts/container_startup.sh
@@ -0,0 +1,16 @@
+#!/bin/bash
+
+# Copyright (c) Microsoft Corporation.
+# Licensed under the MIT license.
+# SPDX-License-Identifier: MIT
+
+# Exits immediately on failure.
+set -e
+
+# Copy any configuration files present to service configuration.
+# If there is a configuration file with the same name at `/sdv/config` this will overwrite
+# that file with the mounted configuration file.
+cp -rf /mnt/config /sdv
+
+# Start the Ibeji service.
+/sdv/service
\ No newline at end of file
diff --git a/core/common/Cargo.toml b/core/common/Cargo.toml
index 8fa85569..e226d4b7 100644
--- a/core/common/Cargo.toml
+++ b/core/common/Cargo.toml
@@ -35,6 +35,3 @@ url = { workspace = true }
[build-dependencies]
tonic-build = { workspace = true }
-
-[features]
-containerize = []
diff --git a/core/common/src/utils.rs b/core/common/src/utils.rs
index ab732d06..7acfdc03 100644
--- a/core/common/src/utils.rs
+++ b/core/common/src/utils.rs
@@ -116,10 +116,9 @@ pub async fn discover_service_using_chariott(
expected_communication_kind: &str,
expected_communication_reference: &str,
) -> Result {
- let uri = get_uri(chariott_uri)?;
-
- let mut client =
- ServiceRegistryClient::connect(uri).await.map_err(|e| Status::internal(e.to_string()))?;
+ let mut client = ServiceRegistryClient::connect(chariott_uri.to_owned())
+ .await
+ .map_err(|e| Status::internal(e.to_string()))?;
let request = Request::new(DiscoverRequest {
namespace: namespace.to_string(),
@@ -183,39 +182,7 @@ pub async fn get_service_uri(
}
};
- let uri = get_uri(&result)?;
-
- Ok(uri)
-}
-
-/// If feature 'containerize' is set, will modify a localhost uri to point to container's localhost
-/// DNS alias. Otherwise, returns the uri as a String.
-///
-/// # Arguments
-/// * `uri` - The uri to potentially modify.
-pub fn get_uri(uri: &str) -> Result {
- #[cfg(feature = "containerize")]
- let uri = {
- // Container env variable names.
- let host_gateway_env_var: &str = "HOST_GATEWAY";
- let host_alias_env_var: &str = "LOCALHOST_ALIAS";
-
- // Return an error if container env variables are not set.
- let host_gateway = env::var(host_gateway_env_var).map_err(|err| {
- Status::failed_precondition(format!(
- "Unable to get environment var '{host_gateway_env_var}' with error: {err}"
- ))
- })?;
- let host_alias = env::var(host_alias_env_var).map_err(|err| {
- Status::failed_precondition(format!(
- "Unable to get environment var '{host_alias_env_var}' with error: {err}"
- ))
- })?;
-
- uri.replace(&host_alias, &host_gateway)
- };
-
- Ok(uri.to_string())
+ Ok(result)
}
#[cfg(test)]
diff --git a/core/invehicle-digital-twin/Cargo.toml b/core/invehicle-digital-twin/Cargo.toml
index 79509142..213a8bd3 100644
--- a/core/invehicle-digital-twin/Cargo.toml
+++ b/core/invehicle-digital-twin/Cargo.toml
@@ -39,5 +39,4 @@ tonic-build = { workspace = true }
[features]
managed_subscribe = ["dep:managed_subscribe"]
-containerize = ["common/containerize"]
-tokio_console = ["dep:tokio-console-subscriber", "tokio/tracing"]
\ No newline at end of file
+tokio_console = ["dep:tokio-console-subscriber", "tokio/tracing"]
diff --git a/core/invehicle-digital-twin/src/main.rs b/core/invehicle-digital-twin/src/main.rs
index 1949c858..bf6f38ed 100644
--- a/core/invehicle-digital-twin/src/main.rs
+++ b/core/invehicle-digital-twin/src/main.rs
@@ -5,7 +5,6 @@
// Module references behind feature flags. Add any necessary module references here.
// Start: Module references.
-use common::utils;
#[cfg(feature = "managed_subscribe")]
use managed_subscribe::managed_subscribe_module::ManagedSubscribeModule;
@@ -55,10 +54,9 @@ async fn register_invehicle_digital_twin_service_with_chariott(
chariott_uri: &str,
invehicle_digital_twin_uri: &str,
) -> Result<(), Status> {
- let uri = utils::get_uri(chariott_uri)?;
-
- let mut client =
- ServiceRegistryClient::connect(uri).await.map_err(|e| Status::internal(e.to_string()))?;
+ let mut client = ServiceRegistryClient::connect(chariott_uri.to_owned())
+ .await
+ .map_err(|e| Status::internal(e.to_string()))?;
let service = Some(ServiceMetadata {
namespace: INVEHICLE_DIGITAL_TWIN_SERVICE_NAMESPACE.to_string(),
diff --git a/core/module/managed_subscribe/Cargo.toml b/core/module/managed_subscribe/Cargo.toml
index 81b63cc2..184a9758 100644
--- a/core/module/managed_subscribe/Cargo.toml
+++ b/core/module/managed_subscribe/Cargo.toml
@@ -27,6 +27,3 @@ yaml-rust = { workspace = true }
[build-dependencies]
tonic-build = { workspace = true }
-
-[features]
-containerize = ["common/containerize"]
diff --git a/core/module/managed_subscribe/src/managed_subscribe_interceptor.rs b/core/module/managed_subscribe/src/managed_subscribe_interceptor.rs
index 21edf3cd..3efbc1d2 100644
--- a/core/module/managed_subscribe/src/managed_subscribe_interceptor.rs
+++ b/core/module/managed_subscribe/src/managed_subscribe_interceptor.rs
@@ -9,7 +9,7 @@ use parking_lot::RwLock;
use prost::Message;
use std::{collections::HashMap, error::Error, sync::Arc};
-use common::{grpc_interceptor::GrpcInterceptor, utils};
+use common::grpc_interceptor::GrpcInterceptor;
use crate::managed_subscribe_store::{CallbackInfo, EntityMetadata, ManagedSubscribeStore};
@@ -77,7 +77,7 @@ impl GrpcInterceptor for ManagedSubscribeInterceptor {
for endpoint in &mut endpoints {
if endpoint.operations.contains(&Self::MANAGED_SUBSCRIBE_OPERATION.to_string()) {
- let entity_callback = utils::get_uri(&endpoint.uri)?;
+ let entity_callback = endpoint.uri.clone();
let callback_protocol = endpoint.protocol.clone();
// Set endpoint information to the managed subscribe module.
diff --git a/samples/Dockerfile.consumer b/samples/Dockerfile.samples.amd64
similarity index 75%
rename from samples/Dockerfile.consumer
rename to samples/Dockerfile.samples.amd64
index 425a1b30..0b5def23 100644
--- a/samples/Dockerfile.consumer
+++ b/samples/Dockerfile.samples.amd64
@@ -10,12 +10,21 @@
# Create a stage for building the application.
ARG RUST_VERSION=1.72.1
-FROM docker.io/library/rust:${RUST_VERSION}-slim-bullseye AS build
ARG APP_NAME=property-consumer
+ARG UID=10001
+
+FROM docker.io/library/rust:${RUST_VERSION}-slim-bullseye AS build
+ARG APP_NAME
WORKDIR /sdv
COPY ./ .
+# Check that APP_NAME argument is valid.
+RUN /sdv/container/scripts/argument_sanitizer.sh \
+ --arg-value "${APP_NAME}" \
+ --regex "^[a-zA-Z_0-9-]+$" || \
+ ( echo "Argument sanitizer failed for ARG 'APP_NAME'"; exit 1 )
+
# Add Build dependencies.
RUN apt update && apt upgrade -y && apt install -y \
cmake \
@@ -24,15 +33,8 @@ RUN apt update && apt upgrade -y && apt install -y \
pkg-config \
protobuf-compiler
-# Check that APP_NAME argument is valid.
-RUN sanitized=$(echo "${APP_NAME}" | tr -dc '^[a-zA-Z_0-9-]+$'); \
-[ "$sanitized" = "${APP_NAME}" ] || { \
- echo "ARG 'APP_NAME' is invalid. APP_NAME='${APP_NAME}' sanitized='${sanitized}'"; \
- exit 1; \
-}
-
-# Build the application with the 'containerize' feature.
-RUN cargo build --release --bin "${APP_NAME}" --features "containerize"
+# Build the application.
+RUN cargo build --release --bin "${APP_NAME}"
# Copy the built application to working directory.
RUN cp ./target/release/"${APP_NAME}" /sdv/service
@@ -49,12 +51,21 @@ RUN cp ./target/release/"${APP_NAME}" /sdv/service
# reproducability is important, consider using a digest
# (e.g., debian@sha256:ac707220fbd7b67fc19b112cee8170b41a9e97f703f588b2cdbbcdcecdd8af57).
FROM docker.io/library/debian:bullseye-slim AS final
+ARG UID
+
+# Copy container scripts.
+COPY ./container/scripts/*.sh /sdv/scripts/
+
+# Check that UID argument is valid.
+RUN /sdv/scripts/argument_sanitizer.sh \
+ --arg-value "${UID}" \
+ --regex "^[0-9]+$" || \
+ ( echo "Argument sanitizer failed for ARG 'UID'"; exit 1 )
RUN apt update && apt upgrade -y && apt install -y libsdl2-dev
# Create a non-privileged user that the app will run under.
# See https://docs.docker.com/develop/develop-images/dockerfile_best-practices/#user
-ARG UID=10001
RUN adduser \
--disabled-password \
--gecos "" \
@@ -63,6 +74,14 @@ RUN adduser \
--no-create-home \
--uid "${UID}" \
appuser
+
+# Create and add user ownership to config directory.
+RUN mkdir -p /sdv/config
+RUN chown appuser /sdv/config
+
+# Create mnt directory to copy override configs into.
+RUN mkdir -p /mnt/config
+
USER appuser
WORKDIR /sdv
@@ -76,8 +95,5 @@ COPY --from=build /sdv/service /sdv/
# Copy configuration for service.
COPY --from=build /sdv/samples/container/config/consumer/ /sdv/config
-# Expose the port that the application listens on.
-EXPOSE 6010
-
# What the container should run when it is started.
-CMD ["/sdv/service"]
+CMD ["/sdv/scripts/container_startup.sh"]
diff --git a/samples/Dockerfile.provider b/samples/Dockerfile.samples.arm64
similarity index 64%
rename from samples/Dockerfile.provider
rename to samples/Dockerfile.samples.arm64
index 34cad0ca..cc108117 100644
--- a/samples/Dockerfile.provider
+++ b/samples/Dockerfile.samples.arm64
@@ -10,32 +10,38 @@
# Create a stage for building the application.
ARG RUST_VERSION=1.72.1
+ARG APP_NAME=property-consumer
+ARG UID=10001
+
FROM docker.io/library/rust:${RUST_VERSION}-slim-bullseye AS build
-ARG APP_NAME=property-provider
+ARG APP_NAME
WORKDIR /sdv
COPY ./ .
+# Check that APP_NAME argument is valid.
+RUN /sdv/container/scripts/argument_sanitizer.sh \
+ --arg-value "${APP_NAME}" \
+ --regex "^[a-zA-Z_0-9-]+$" || \
+ ( echo "Argument sanitizer failed for ARG 'APP_NAME'"; exit 1 )
+
# Add Build dependencies.
RUN apt update && apt upgrade -y && apt install -y \
cmake \
libsdl2-dev \
libssl-dev \
pkg-config \
- protobuf-compiler
+ protobuf-compiler \
+ gcc-aarch64-linux-gnu \
+ g++-aarch64-linux-gnu
-# Check that APP_NAME argument is valid.
-RUN sanitized=$(echo "${APP_NAME}" | tr -dc '^[a-zA-Z_0-9-]+$'); \
-[ "$sanitized" = "${APP_NAME}" ] || { \
- echo "ARG 'APP_NAME' is invalid. APP_NAME='${APP_NAME}' sanitized='${sanitized}'"; \
- exit 1; \
-}
+RUN rustup target add aarch64-unknown-linux-gnu
-# Build the application with the 'containerize' feature.
-RUN cargo build --release --bin "${APP_NAME}" --features "containerize"
+# Build the application.
+RUN cargo build --release --target=aarch64-unknown-linux-gnu --bin "${APP_NAME}"
# Copy the built application to working directory.
-RUN cp ./target/release/"${APP_NAME}" /sdv/service
+RUN cp ./target/aarch64-unknown-linux-gnu/release/"${APP_NAME}" /sdv/service
################################################################################
# Create a new stage for running the application that contains the minimal
@@ -48,13 +54,20 @@ RUN cp ./target/release/"${APP_NAME}" /sdv/service
# most recent version of that tag when you build your Dockerfile. If
# reproducability is important, consider using a digest
# (e.g., debian@sha256:ac707220fbd7b67fc19b112cee8170b41a9e97f703f588b2cdbbcdcecdd8af57).
-FROM docker.io/library/debian:bullseye-slim AS final
+FROM docker.io/arm64v8/debian:bullseye-slim AS final
+ARG UID
+
+# Copy container scripts.
+COPY ./container/scripts/*.sh /sdv/scripts/
-RUN apt update && apt upgrade -y && apt install -y libsdl2-dev
+# Check that UID argument is valid.
+RUN /sdv/scripts/argument_sanitizer.sh \
+ --arg-value "${UID}" \
+ --regex "^[0-9]+$" || \
+ ( echo "Argument sanitizer failed for ARG 'UID'"; exit 1 )
# Create a non-privileged user that the app will run under.
# See https://docs.docker.com/develop/develop-images/dockerfile_best-practices/#user
-ARG UID=10001
RUN adduser \
--disabled-password \
--gecos "" \
@@ -63,6 +76,14 @@ RUN adduser \
--no-create-home \
--uid "${UID}" \
appuser
+
+# Create and add user ownership to config directory.
+RUN mkdir -p /sdv/config
+RUN chown appuser /sdv/config
+
+# Create mnt directory to copy override configs into.
+RUN mkdir -p /mnt/config
+
USER appuser
WORKDIR /sdv
@@ -74,10 +95,7 @@ ENV IBEJI_HOME=/sdv/config
COPY --from=build /sdv/service /sdv/
# Copy configuration for service.
-COPY --from=build /sdv/samples/container/config/provider/ /sdv/config
-
-# Expose the port that the application listens on.
-EXPOSE 4010
+COPY --from=build /sdv/samples/container/config/consumer/ /sdv/config
# What the container should run when it is started.
-CMD ["/sdv/service"]
+CMD ["/sdv/scripts/container_startup.sh"]
diff --git a/samples/common/.cargo/config.toml b/samples/common/.cargo/config.toml
new file mode 100644
index 00000000..47b518be
--- /dev/null
+++ b/samples/common/.cargo/config.toml
@@ -0,0 +1,3 @@
+[target.aarch64-unknown-linux-gnu]
+linker = "aarch64-linux-gnu-g++"
+rustflags = [ "-C", "target-feature=+crt-static", "-C", "link-arg=-lg++" ]
diff --git a/samples/common/Cargo.toml b/samples/common/Cargo.toml
index dc5b7659..e398c20b 100644
--- a/samples/common/Cargo.toml
+++ b/samples/common/Cargo.toml
@@ -20,5 +20,6 @@ tokio = { workspace = true }
tonic = { workspace = true }
yaml-rust = { workspace = true }
-[features]
-containerize = []
+# If built for aarch64, enable the 'vendored-ssl' feature.
+[target.'cfg(target_arch = "aarch64")'.dependencies]
+sdl2 = { workspace = true, features = [ "bundled", "static-link" ] }
diff --git a/samples/common/src/utils.rs b/samples/common/src/utils.rs
index 0b4ee2f5..f8497f4f 100644
--- a/samples/common/src/utils.rs
+++ b/samples/common/src/utils.rs
@@ -129,15 +129,12 @@ pub async fn discover_digital_twin_provider_using_ibeji(
})
.cloned()
{
- Some(mut result) => {
+ Some(result) => {
info!(
"Found a matching endpoint for entity id {entity_id} that has URI {}",
result.uri
);
- result.uri = get_uri(&result.uri)
- .map_err(|err| format!("Failed to get provider URI due to error: {err}"))?;
-
Ok(result)
}
None => Err("Did not find an endpoint that met our requirements".to_string()),
@@ -161,10 +158,9 @@ pub async fn discover_service_using_chariott(
communication_kind: &str,
communication_reference: &str,
) -> Result {
- let uri = get_uri(chariott_uri)?;
-
- let mut client =
- ServiceRegistryClient::connect(uri).await.map_err(|e| Status::internal(e.to_string()))?;
+ let mut client = ServiceRegistryClient::connect(chariott_uri.to_owned())
+ .await
+ .map_err(|e| Status::internal(e.to_string()))?;
let request = Request::new(DiscoverRequest {
namespace: namespace.to_string(),
@@ -233,39 +229,7 @@ pub async fn retrieve_invehicle_digital_twin_uri(
}
};
- get_uri(&result).map_err(|err| {
- format!("Failed to retrieve the in-vehicle digital twin service's URI due to error: {err}")
- })
-}
-
-/// If the 'containerize' feature is set, this function will modify the localhost URI to point to
-/// the container's localhost DNS alias. Otherwise, returns the URI as a string.
-///
-/// # Arguments
-/// * `uri` - The uri to potentially modify.
-pub fn get_uri(uri: &str) -> Result {
- #[cfg(feature = "containerize")]
- let uri = {
- // Container env variable names.
- let host_gateway_env_var: &str = "HOST_GATEWAY";
- let host_alias_env_var: &str = "LOCALHOST_ALIAS";
-
- // Return an error if container env variables are not set.
- let host_gateway = std::env::var(host_gateway_env_var).map_err(|err| {
- Status::failed_precondition(format!(
- "Unable to get environment var '{host_gateway_env_var}' with error: {err}"
- ))
- })?;
- let host_alias = std::env::var(host_alias_env_var).map_err(|err| {
- Status::failed_precondition(format!(
- "Unable to get environment var '{host_alias_env_var}' with error: {err}"
- ))
- })?;
-
- uri.replace(&host_alias, &host_gateway)
- };
-
- Ok(uri.to_string())
+ Ok(result)
}
#[cfg(test)]
diff --git a/samples/container/.accepted_words.txt b/samples/container/.accepted_words.txt
deleted file mode 100644
index 49d4301f..00000000
--- a/samples/container/.accepted_words.txt
+++ /dev/null
@@ -1,93 +0,0 @@
-Agemo
-agemo
-APP
-arg
-br
-build
-cargo
-cd
-CHARIOTT
-Chariott
-Chariott's
-chariott
-com
-config
-containerize
-containerized
-Containerization
-Ctrl
-dev
-digitaltwins
-dir
-Dockerfile
-dockerfile
-dockerfiles
-dotnet
-dst
-dt
-dtdl
-DTDL
-en
-env
-fontconfig
-gcc
-gcc's
-github
-hardcheese
-haudebourg
-https
-ibeji
-Ibeji
-Ibeji's
-intellectualproperty
-invehicle
-io
-iot
-js
-json
-JSON
-kbd
-ld
-LD
-libfontconfig
-libsdl
-localhost
-loopback
-md
-microsoft
-minimalistic
-mosquitto
-Mosquitto
-mqtt
-MQTT
-netns
-opendigitaltwins
-plugandplay
-Podman
-podman
-protobuf
-Protobuf
-ps
-repo
-Repo
-rm
-ro
-rustup
-sdk
-sdl
-SDL
-sdv
-slirp
-snapd
-src
-standalone
-sudo
-timothee
-toml
-Tonic
-toolchain
-uri
-URI
-www
-xargs
-yaml
diff --git a/samples/container/README.md b/samples/container/README.md
deleted file mode 100644
index f3467046..00000000
--- a/samples/container/README.md
+++ /dev/null
@@ -1,157 +0,0 @@
-# Samples Containerization
-
-This document covers how to containerize the samples for Ibeji. To run the in-vehicle digital twin
-service as a container or to get more in depth instruction for building and running a container
-please refer to [Ibeji Containerization](../../container/README.md).
-
-## Running the Samples in a Container
-
-Below are the steps for running the sample providers and consumers in a container. Note that the
-configuration files used by the containerized sample are cloned from
-[/samples/container/config](./config/) defined in the project's root.
-
-### Provider
-
-Provider containers utilize the dockerfile [Dockerfile.provider](../Dockerfile.provider).
-
-#### Build
-
-By default, the sample provider built in the Dockerfile is the
-[property-provider](../property/provider/). To change the provider the container builds, use
-the following argument:
-
- Docker
-
- ```shell
- --build-arg APP_NAME={name of the provider}
- ```
-
- For example:
-
- ```shell
- docker build -t provider -f ./samples/Dockerfile.provider . --build-arg APP_NAME=managed-subscribe-provider
- ```
-
- Podman
-
- ```shell
- --build-arg=APP_NAME={name of the provider}
- ```
-
- For example:
-
- ```shell
- podman build -t provider:latest -f ./samples/Dockerfile.provider . --build-arg=APP_NAME=managed-subscribe-provider
- ```
-
-#### Run
-
-If you selected a different provider than the default provider built in the dockerfile, you may
-need to override the configuration file [provider_settings.yaml](./config/provider/provider_settings.yaml).
-To override the configuration of a provider container, follow the below steps after you have built
-the container:
-
- Docker
-
- 1. To run the container with overridden configuration, create your config file and set an
- environment variable called IBEJI_HOME to the absolute path of the directory containing the
- config file:
-
- ```shell
- export IBEJI_HOME={absolute path of the directory containing the config file}
- ```
-
- 1. Then run the container from the project's root with the following command:
-
- ```shell
- docker run -v ${IBEJI_HOME}:/sdv/config --name provider -p 5010:5010 --env-file=./samples/container/config/docker.env --add-host=host.docker.internal:host-gateway -it --rm invehicle_digital_twin
- ```
-
- Podman
-
- 1. To run the container with overridden configuration, create your config file and set an
- environment variable called IBEJI_HOME to the absolute path of the directory containing the
- config file:
-
- ```shell
- export IBEJI_HOME={absolute path of the directory containing the config file}
- ```
-
- 1. Then run the container from the project's root with the following command:
-
- ```shell
- podman run --mount=type=bind,src=${IBEJI_HOME},dst=/sdv/config,ro=true -p 5010:5010 --env-file=./samples/container/config/podman.env --network=slirp4netns:allow_host_loopback=true localhost/provider:latest
- ```
-
-### Consumer
-
-Consumer containers utilize the dockerfile [Dockerfile.consumer](../Dockerfile.consumer).
-
-#### Build
-
-By default, the sample consumer built in the Dockerfile is the
-[property-consumer](../property/consumer/). To change the consumer the container builds, use
-the following argument:
-
- Docker
-
- ```shell
- --build-arg APP_NAME={name of the consumer}
- ```
-
- For example:
-
- ```shell
- docker build -t consumer -f ./samples/Dockerfile.consumer . --build-arg APP_NAME=managed-subscribe-consumer
- ```
-
- Podman
-
- ```shell
- --build-arg=APP_NAME={name of the consumer}
- ```
-
- For example:
-
- ```shell
- podman build -t consumer:latest -f ./samples/Dockerfile.consumer . --build-arg=APP_NAME=managed-subscribe-consumer
- ```
-
-#### Run
-
-If you selected a different consumer than the default consumer built in the dockerfile, you may
-need to override the configuration file [consumer_settings.yaml](./config/consumer/consumer_settings.yaml).
-To override the configuration of a consumer container, follow the below steps after you have built
-the container:
-
- Docker
-
- 1. To run the container with overridden configuration, create your config file and set an
- environment variable called IBEJI_HOME to the absolute path of the directory containing the
- config file:
-
- ```shell
- export IBEJI_HOME={absolute path of the directory containing the config file}
- ```
-
- 1. Then run the container from the project's root with the following command:
-
- ```shell
- docker run -v ${IBEJI_HOME}:/sdv/config --name consumer -p 5010:5010 --env-file=./samples/container/config/docker.env --add-host=host.docker.internal:host-gateway -it --rm invehicle_digital_twin
- ```
-
- Podman
-
- 1. To run the container with overridden configuration, create your config file and set an
- environment variable called IBEJI_HOME to the absolute path of the directory containing the
- config file:
-
- ```shell
- export IBEJI_HOME={absolute path of the directory containing the config file}
- ```
-
- 1. Then run the container from the project's root with the following command:
-
- ```shell
- podman run --mount=type=bind,src=${IBEJI_HOME},dst=/sdv/config,ro=true -p 5010:5010 --env-file=./samples/container/config/podman.env --network=slirp4netns:allow_host_loopback=true localhost/consumer:latest
- ```
diff --git a/samples/container/config/docker.env b/samples/container/config/docker.env
deleted file mode 100644
index 56bafc6e..00000000
--- a/samples/container/config/docker.env
+++ /dev/null
@@ -1,9 +0,0 @@
-# Copyright (c) Microsoft Corporation.
-# Licensed under the MIT license.
-# SPDX-License-Identifier: MIT
-
-# DNS name used by the container to communicate with host.
-HOST_GATEWAY=host.docker.internal
-
-# Alias for localhost to be replaced by HOST_GATEWAY if run in a container.
-LOCALHOST_ALIAS=0.0.0.0
diff --git a/samples/container/config/podman.env b/samples/container/config/podman.env
deleted file mode 100644
index 1abc4a35..00000000
--- a/samples/container/config/podman.env
+++ /dev/null
@@ -1,9 +0,0 @@
-# Copyright (c) Microsoft Corporation.
-# Licensed under the MIT license.
-# SPDX-License-Identifier: MIT
-
-# DNS name used by the container to communicate with host.
-HOST_GATEWAY=host.containers.internal
-
-# Alias for localhost to be replaced by HOST_GATEWAY if run in a container.
-LOCALHOST_ALIAS=0.0.0.0
diff --git a/samples/managed_subscribe/Cargo.toml b/samples/managed_subscribe/Cargo.toml
index f82adb36..63ed3e7d 100644
--- a/samples/managed_subscribe/Cargo.toml
+++ b/samples/managed_subscribe/Cargo.toml
@@ -39,5 +39,6 @@ path = "provider/src/main.rs"
name = "managed-subscribe-consumer"
path = "consumer/src/main.rs"
-[features]
-containerize = ["samples-common/containerize"]
+# If built for aarch64, enable the 'vendored-ssl' feature.
+[target.'cfg(target_arch = "aarch64")'.dependencies]
+paho-mqtt = { workspace = true, features = ["vendored-ssl"] }
diff --git a/samples/managed_subscribe/consumer/src/main.rs b/samples/managed_subscribe/consumer/src/main.rs
index e8ec255d..44b467a5 100644
--- a/samples/managed_subscribe/consumer/src/main.rs
+++ b/samples/managed_subscribe/consumer/src/main.rs
@@ -11,7 +11,7 @@ use paho_mqtt as mqtt;
use samples_common::constants::{constraint_type, digital_twin_operation, digital_twin_protocol};
use samples_common::consumer_config;
use samples_common::utils::{
- discover_digital_twin_provider_using_ibeji, get_uri, retrieve_invehicle_digital_twin_uri,
+ discover_digital_twin_provider_using_ibeji, retrieve_invehicle_digital_twin_uri,
};
use samples_protobuf_data_access::module::managed_subscribe::v1::managed_subscribe_client::ManagedSubscribeClient;
use samples_protobuf_data_access::module::managed_subscribe::v1::{
@@ -180,7 +180,7 @@ async fn main() -> Result<(), Box> {
.await?;
// Deconstruct subscription information.
- let broker_uri = get_uri(&subscription_info.uri)?;
+ let broker_uri = subscription_info.uri;
let topic = subscription_info.context;
info!("The broker URI for the AmbientAirTemperature property's provider is {broker_uri}");
diff --git a/samples/managed_subscribe/provider/src/provider_impl.rs b/samples/managed_subscribe/provider/src/provider_impl.rs
index 9fe889a4..a0151042 100644
--- a/samples/managed_subscribe/provider/src/provider_impl.rs
+++ b/samples/managed_subscribe/provider/src/provider_impl.rs
@@ -16,7 +16,6 @@ use log::{debug, info, warn};
use paho_mqtt as mqtt;
use parking_lot::RwLock;
use samples_common::constants::constraint_type;
-use samples_common::utils;
use serde_derive::{Deserialize, Serialize};
use strum_macros::{Display, EnumString};
use tokio::sync::{mpsc, watch};
@@ -131,9 +130,7 @@ impl ProviderImpl {
let min_interval_ms = self.min_interval_ms;
// This should not be empty.
- let mut subscription_info = payload.subscription_info.unwrap();
-
- subscription_info.uri = utils::get_uri(&subscription_info.uri).unwrap();
+ let subscription_info = payload.subscription_info.unwrap();
// Create stop publish channel.
let (sender, mut reciever) = mpsc::channel(10);
diff --git a/samples/property/Cargo.toml b/samples/property/Cargo.toml
index e24508d1..fe7b9743 100644
--- a/samples/property/Cargo.toml
+++ b/samples/property/Cargo.toml
@@ -37,5 +37,6 @@ path = "provider/src/main.rs"
name = "property-consumer"
path = "consumer/src/main.rs"
-[features]
-containerize = ["samples-common/containerize"]
+# If built for aarch64, enable the 'vendored-ssl' feature.
+[target.'cfg(target_arch = "aarch64")'.dependencies]
+paho-mqtt = { workspace = true, features = ["vendored-ssl"] }
diff --git a/samples/property/provider/src/main.rs b/samples/property/provider/src/main.rs
index 0808174a..21f9c052 100644
--- a/samples/property/provider/src/main.rs
+++ b/samples/property/provider/src/main.rs
@@ -8,9 +8,7 @@ use log::{debug, info, warn, LevelFilter};
use paho_mqtt as mqtt;
use samples_common::constants::{digital_twin_operation, digital_twin_protocol};
use samples_common::provider_config;
-use samples_common::utils::{
- get_uri, retrieve_invehicle_digital_twin_uri, retry_async_based_on_status,
-};
+use samples_common::utils::{retrieve_invehicle_digital_twin_uri, retry_async_based_on_status};
use samples_protobuf_data_access::invehicle_digital_twin::v1::invehicle_digital_twin_client::InvehicleDigitalTwinClient;
use samples_protobuf_data_access::invehicle_digital_twin::v1::{
EndpointInfo, EntityAccessInfo, RegisterRequest,
@@ -209,9 +207,7 @@ async fn main() -> Result<(), Box> {
})
.await?;
- let stream_uri = get_uri(&broker_uri)?;
-
- start_ambient_air_temperature_data_stream(stream_uri, topic);
+ start_ambient_air_temperature_data_stream(broker_uri, topic);
signal::ctrl_c().await.expect("Failed to listen for control-c event");
diff --git a/samples/streaming/.cargo/config.toml b/samples/streaming/.cargo/config.toml
new file mode 100644
index 00000000..47b518be
--- /dev/null
+++ b/samples/streaming/.cargo/config.toml
@@ -0,0 +1,3 @@
+[target.aarch64-unknown-linux-gnu]
+linker = "aarch64-linux-gnu-g++"
+rustflags = [ "-C", "target-feature=+crt-static", "-C", "link-arg=-lg++" ]
diff --git a/samples/streaming/Cargo.toml b/samples/streaming/Cargo.toml
index 1a1b8b41..431917a8 100644
--- a/samples/streaming/Cargo.toml
+++ b/samples/streaming/Cargo.toml
@@ -45,3 +45,7 @@ path = "provider/src/main.rs"
[[bin]]
name = "streaming-consumer"
path = "consumer/src/main.rs"
+
+# If built for aarch64, enable the 'vendored-ssl' feature.
+[target.'cfg(target_arch = "aarch64")'.dependencies]
+sdl2 = { workspace = true, features = [ "bundled", "static-link" ] }