I am trying to include Ortex in a docker container I am building. First started with Alpine container, and that is a no go. Alpine is going to be problematic because of the musl compiler…
Compiling lib/ortex/native.ex (it's taking more than 10s)
41.10 Compiling ureq v2.9.6
41.61 Compiling ort-sys v2.0.0-rc.8
41.98 error: failed to run custom build command for `ort-sys v2.0.0-rc.8`
41.98
41.98 Caused by:
41.98 process didn't exit successfully: `/app/_build/prod/lib/ortex/native/ortex/release/build/ort-sys-d7347b282fba238c/build-script-build` (exit status: 101)
41.98 --- stdout
41.98 cargo:rerun-if-env-changed=LIBONNXRUNTIME_NO_PKG_CONFIG
41.98 cargo:rerun-if-env-changed=PKG_CONFIG_x86_64-alpine-linux-musl
41.98 cargo:rerun-if-env-changed=PKG_CONFIG_x86_64_alpine_linux_musl
41.98 cargo:rerun-if-env-changed=HOST_PKG_CONFIG
41.98 cargo:rerun-if-env-changed=PKG_CONFIG
41.98 cargo:rerun-if-env-changed=LIBONNXRUNTIME_STATIC
41.98 cargo:rerun-if-env-changed=LIBONNXRUNTIME_DYNAMIC
41.98 cargo:rerun-if-env-changed=PKG_CONFIG_ALL_STATIC
41.98 cargo:rerun-if-env-changed=PKG_CONFIG_ALL_DYNAMIC
41.98 cargo:rerun-if-env-changed=PKG_CONFIG_PATH_x86_64-alpine-linux-musl
41.98 cargo:rerun-if-env-changed=PKG_CONFIG_PATH_x86_64_alpine_linux_musl
41.98 cargo:rerun-if-env-changed=HOST_PKG_CONFIG_PATH
41.98 cargo:rerun-if-env-changed=PKG_CONFIG_PATH
41.98 cargo:rerun-if-env-changed=PKG_CONFIG_LIBDIR_x86_64-alpine-linux-musl
41.98 cargo:rerun-if-env-changed=PKG_CONFIG_LIBDIR_x86_64_alpine_linux_musl
41.98 cargo:rerun-if-env-changed=HOST_PKG_CONFIG_LIBDIR
41.98 cargo:rerun-if-env-changed=PKG_CONFIG_LIBDIR
41.98 cargo:rerun-if-env-changed=PKG_CONFIG_SYSROOT_DIR_x86_64-alpine-linux-musl
41.98 cargo:rerun-if-env-changed=PKG_CONFIG_SYSROOT_DIR_x86_64_alpine_linux_musl
41.98 cargo:rerun-if-env-changed=HOST_PKG_CONFIG_SYSROOT_DIR
41.98 cargo:rerun-if-env-changed=PKG_CONFIG_SYSROOT_DIR
41.98 onnxruntime not found using pkg-config, falling back to manual setup.
41.98 cargo:rerun-if-env-changed=ORT_LIB_LOCATION
41.98 cargo:rerun-if-env-changed=ORT_LIB_PROFILE
41.98 cargo:rerun-if-env-changed=ORT_PREFER_DYNAMIC_LINK
41.98 selected feature set: none
41.98
41.98 --- stderr
41.98 thread 'main' panicked at /app/.cargo/registry/src/index.crates.io-6f17d22bba15001f/ort-sys-2.0.0-rc.8/build.rs:360:17:
41.98 downloaded binaries not available for target x86_64-alpine-linux-musl
41.98 you may have to compile ONNX Runtime from source
41.98 note: run with `RUST_BACKTRACE=1` environment variable to display a backtrace
41.98 warning: build failed, waiting for other jobs to finish...
42.31
42.31 == Compilation error in file lib/ortex/native.ex ==
42.31 ** (RuntimeError) Rust NIF compile error (rustc exit code 101)
42.31 (rustler 0.35.1) lib/rustler/compiler.ex:36: Rustler.Compiler.compile_crate/3
42.31 lib/ortex/native.ex:11: (module)
42.31 could not compile dependency :ortex, "mix compile" failed. Errors may have been logged above. You can recompile this dependency with "mix deps.compile ortex --force", update it with "mix deps.update ortex" or clean it with "mix deps.clean ortex"
However, when I tried in a Debian container, I did not get much further…
.49 Compiling 6 files (.ex)
34.56 Generated rustler app
34.57 ==> ortex
34.57 Compiling 6 files (.ex)
34.60
34.60 == Compilation error in file lib/ortex/native.ex ==
34.60 ** (ErlangError) Erlang error: :enoent
34.60 (elixir 1.15.8) lib/system.ex:1111: System.cmd("cargo", ["metadata", "--format-version=1"], [cd: "native/ortex"])
34.60 (rustler 0.35.1) lib/rustler/compiler/config.ex:76: Rustler.Compiler.Config.metadata!/1
34.60 (rustler 0.35.1) lib/rustler/compiler/config.ex:63: Rustler.Compiler.Config.build/1
34.60 (rustler 0.35.1) lib/rustler/compiler.ex:8: Rustler.Compiler.compile_crate/3
34.60 lib/ortex/native.ex:11: (module)
34.60 could not compile dependency :ortex, "mix compile" failed. Errors may have been logged above. You can recompile this dependency with "mix deps.compile ortex --force", update it with "mix deps.update ortex" or clean it with "mix deps.clean ortex"
------
debian.Dockerfile:226
--------------------
Yes, it was not clear to me that the error was indicating that cargo was not installed. That is the issue. I did have rust installed, but it was not in the PATH. Now I have it in the path and but get the following error
error: rustup could not choose a version of cargo to run, because one wasn't specified explicitly, and no default is configured.
35.24 help: run 'rustup default stable' to download the latest stable release of Rust and set it as your default toolchain.
Here are the two lines in my Dockerfiles that make Rust work just fine:
RUN curl --proto '=https' --tlsv1.2 -sSf https://sh.rustup.rs | sh -s -- --default-toolchain stable --profile minimal --target x86_64-unknown-linux-gnu -y
ENV PATH="/root/.cargo/bin:${PATH}"
(Obviously make sure the target is the right one i.e. there are plenty of ARM containers out there.)
The key is to not only install Rust via rustup but also “activate” a toolchain – in this case stable – which is very akin to how version managers like asdf / mise work: you (1) install a tool and (2) make a version active / default.
In the example above the --default-toolchain stable does that, or you can do it manually like so (put this in your Dockerfile after installing rustup and Rust through it):
Thanks for looking at this. It just does not want to work.
I have tried to set the toolchain version explictly and also with the cmd you provided, same result.
ld not choose a version of cargo to run, because one wasn't specified explicitly, and no default is configured.
34.44 help: run 'rustup default stable' to download the latest stable release of Rust and set it as your default toolchain.
34.44
34.44 == Compilation error in file lib/ortex/native.ex ==
34.44 ** (RuntimeError) calling `cargo metadata` failed.
34.44
34.44 (rustler 0.35.1) lib/rustler/compiler/config.ex:81: Rustler.Compiler.Config.metadata!/1
34.44 (rustler 0.35.1) lib/rustler/compiler/config.ex:63: Rustler.Compiler.Config.build/1
34.44 (rustler 0.35.1) lib/rustler/compiler.ex:8: Rustler.Compiler.compile_crate/3
34.44 lib/ortex/native.ex:11: (module)
34.44 could not compile dependency :ortex, "mix compile" failed. Errors may have been logged above. You can recompile this dependency with "mix deps.compile ortex --force", update it with "mix deps.update ortex" or clean it with "mix deps.clean ortex"
Apparently this has to do with multi-stage docker build. If I install Rust in prior stage, it fails. However, if I install Rust just before the mix deps.compile it works ok.
Right, that is what I am thinking. Here is what I had to do to make it work. In the second stage, the ENV HOME… is messing up cargo
# First stage...
....
# Get Rust
RUN curl --proto '=https' --tlsv1.2 -sSf https://sh.rustup.rs | sh -s -- --default-toolchain stable --profile minimal --target x86_64-unknown-linux-gnu -y
ENV PATH="/root/.cargo/bin:${PATH}"
FROM build-os-deps AS build-deps-get
# second stage
ARG APP_DIR # /app
ENV HOME=$APP_DIR
# If rustup default stable is not run, mix build will FAIL
RUN rustup default stable
# mix compile...
The way you want to usually separate the docker stages is that the first stage compiles and creates a release, and the second one is a clear linux container + runtime dependencies required when you are running the elixir release.
What you are currently doing doesn’t make much sense and I am amazed it works at all.
First stage is basic OS stuff without any app data, second stage copies app and compiles mix. There are subsequent stages that create a clean container with just the app + runtime deps
Makes zero sense, the idea behind different stages is that you have a clean environment after each stage, you need rust to compile native dependencies, hence it’s a requirement to compile the project.