Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fix phoenix/elixir recipe #1252

Open
wants to merge 8 commits into
base: main
Choose a base branch
from
Open

fix phoenix/elixir recipe #1252

wants to merge 8 commits into from

Conversation

nbw
Copy link

@nbw nbw commented Jan 5, 2025

Summary

  • 1️⃣ Lock file problem when deploying a Phoenix Elixir app
  • 2️⃣ Improvements for the Phoenix/Elixir deploying recipe
  • 3️⃣ Fixes issue with copied files for the start phase
  • 3️⃣ Fixes issue with default values for ARGS in the docker file

👋 the FINAL dockerfile output is in the reference section ⬇️

1️⃣ Lock file problem

When deploying a Phoenix Elixir app to Railway there is a conflict with the lock file that cause the error: lock outdated: the lock is outdated compared to the options in your mix.exs[1]. This is because the we're copying too many files during multiple phases. The solution is to update the elixir dockerfile builder to only copy specific files.

https://elixirforum.com/t/docker-build-started-failing-unchecked-dependencies-for-environment-prod/28844/6

The Railway template for phoenix got around this problem by removing the lock file, which is not behaviour we want.

2️⃣ General updates to the Elixir build

  • run mix ecto.setup if ecto_sql is detected rather than postgrex
    • postgrex is a postgres specific adapter, but there are other adapters (for mysql, etc). detecting ecto_sql is more generic
  • only copy the necessary folders and assets
  • build hex, rebar, and other deps using MIX_ENV=prod

Future considerations

The way we run elixir is with mix phx.server but actually creating a mix release is the better approach (it creates a binary that can be run). More details here about why release are better. In the future, we should move away from mix phx.server and use releases instead, but it's beyond the scope of this PR. Here's an example of an Elixir dockerfile that uses releases.

3️⃣ Other fixes

  • Issue: setting only_include_files in the start phase was ignored. Instead COPY . /app was always added to the generated dockerfile
    • ✅ if only_include_files is set in the start phase, use it
  • Issue: default environment values aren't set in the Dockerfile
    • ✅ set default values for docker ARGs. Example output: ARG ELIXIR_ERL_OPTIONS="+fnu" LANG="en_US.UTF-8" LANGUAGE="en_US:en" MIX_ENV="prod" NIXPACKS_METADATA="elixir"

References

[1] Lock file problem

[+] Building 113.3s (15/18)                                                 docker:orbstack
 => [internal] load build definition from Dockerfile                                   0.0s
 => => transferring dockerfile: 858B                                                   0.0s
 => [internal] load metadata for ghcr.io/railwayapp/nixpacks:ubuntu-1733184274         0.9s
 => [internal] load .dockerignore                                                      0.0s
 => => transferring context: 2B                                                        0.0s
 => [ 1/14] FROM ghcr.io/railwayapp/nixpacks:ubuntu-1733184274@sha256:91b1c0ad82b8a7a  0.0s
 => [internal] load build context                                                      0.6s
 => => transferring context: 82.02MB                                                   0.5s
 => CACHED [ 2/14] WORKDIR /app/                                                       0.0s
 => CACHED [ 3/14] COPY .nixpacks/nixpkgs-c5702bd28cbde41a191a9c2a00501f18941efbd0.ni  0.0s
 => [ 4/14] RUN nix-env -if .nixpacks/nixpkgs-c5702bd28cbde41a191a9c2a00501f18941ef  102.7s
 => [ 5/14] COPY . /app/.                                                              0.9s
 => [ 6/14] RUN  mix local.hex --force                                                 2.0s
 => [ 7/14] RUN  echo test                                                             0.2s
 => [ 8/14] RUN  mix local.rebar --force                                               1.7s
 => [ 9/14] RUN  mix deps.get --only prod                                              2.5s
 => [10/14] COPY . /app/.                                                              1.1s
 => ERROR [11/14] RUN  mix compile                                                     0.4s
------
 > [11/14] RUN  mix compile:
0.382 Unchecked dependencies for environment prod:
0.382 * heroicons (https://github.com/tailwindlabs/heroicons.git - v2.1.1)
0.382   lock outdated: the lock is outdated compared to the options in your mix.exs. To fetch locked version run "mix deps.get"
0.384 ** (Mix) Can't continue due to errors on dependencies
------
Dockerfile:26
--------------------
  24 |     # build phase
  25 |     COPY . /app/.
  26 | >>> RUN  mix compile
  27 |     RUN  mix assets.deploy
  28 |     RUN  mix ecto.setup
--------------------
ERROR: failed to solve: process "/bin/bash -ol pipefail -c mix compile" did not complete successfully: exit code: 1

View build details: docker-desktop://dashboard/build/orbstack/orbstack/uxra2k6hpddi2mzjzo7r97fjr
Error: Docker build failed

[2] Final Dockerfile

output of nixpacks build --dockerfile . on a phoenix elixir project that uses a database:

FROM ghcr.io/railwayapp/nixpacks:ubuntu-1733184274

ENTRYPOINT ["/bin/bash", "-l", "-c"]
WORKDIR /app/


COPY .nixpacks/nixpkgs-c5702bd28cbde41a191a9c2a00501f18941efbd0.nix .nixpacks/nixpkgs-c5702bd28cbde41a191a9c2a00501f18941efbd0.nix
RUN nix-env -if .nixpacks/nixpkgs-c5702bd28cbde41a191a9c2a00501f18941efbd0.nix && nix-collect-garbage -d


ARG ELIXIR_ERL_OPTIONS="+fnu" LANG="en_US.UTF-8" LANGUAGE="en_US:en" MIX_ENV="prod" NIXPACKS_METADATA="elixir"
ENV ELIXIR_ERL_OPTIONS=$ELIXIR_ERL_OPTIONS LANG=$LANG LANGUAGE=$LANGUAGE MIX_ENV=$MIX_ENV NIXPACKS_METADATA=$NIXPACKS_METADATA

# setup phase
# noop

# install phase
COPY config/config.exs /app/config/config.exs
COPY config/${MIX_ENV}.exs /app/config/${MIX_ENV}.exs
COPY mix.exs /app/mix.exs
COPY mix.lock /app/mix.lock
RUN  mix local.hex --force
RUN  mix local.rebar --force
RUN  mix deps.get --only prod

# build phase
COPY config/runtime.exs /app/config/runtime.exs
COPY assets /app/assets
COPY priv /app/priv
COPY config /app/config
COPY lib /app/lib
RUN  mix compile
RUN  mix assets.deploy





# start


CMD ["mix ecto.setup && mix phx.server"]

nbw added 3 commits December 27, 2024 23:05
postgrex is just the adapter for postgres specifically. there are other
ones for different database (mysql, etc.):
https://hexdocs.pm/ecto_sql/Ecto.Adapters.SQL.html#module-built-in-adapters
- only copy certain files in each phase so that there isn't a conflict
  with the lock file

- move db migrations to happen at start time rather than bulding

- add more ENV defaults like LANG and LANGUAGE

let mut start_phase = StartPhase::new(start_phase_cmd.to_string());
// Skip copying any additional files for the start phase
start_phase.only_include_files = Some(vec![]);
Copy link
Author

@nbw nbw Jan 5, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

we don't want to copy any files before starting. if we do, the "lock" file problem occurs.

if mix_exs_content.contains("assets.deploy") {
build_phase.add_cmd("mix assets.deploy".to_string());
}

if mix_exs_content.contains("postgrex") && mix_exs_content.contains("ecto") {
Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

this won't work because the DATABASE_URL isn't available at this time. Railway's own docs suggesting adding mix ecto.setup to the start phase instead.

@@ -76,6 +96,8 @@ impl ElixirProvider {
fn default_elixir_environment_variables() -> EnvironmentVariables {
let var_map = vec![
("MIX_ENV", "prod"),
("LANG", "en_US.UTF-8"),
Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

LANG and LANGUAGE are required for runtime and good to set as defaults

@@ -148,7 +148,7 @@ impl DockerfileGenerator for BuildPlan {
// Pull the variables in from docker `--build-arg`
variables
.iter()
.map(|var| var.0.to_string())
.map(|var| format!("{}=\"{}\"", var.0.trim(), var.1.trim()))
Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This line changes the Dockerfile output from:

ARG ELIXIR_ERL_OPTIONS LANG LANGUAGE MIX_ENV NIXPACKS_METADATA

To:

ARG ELIXIR_ERL_OPTIONS="+fnu" LANG="en_US.UTF-8" LANGUAGE="en_US:en" MIX_ENV="prod" NIXPACKS_METADATA="elixir"

Looking at the docker arg docs and from my own testing, this simply sets a default value for the ARG but it can be overridden by setting a variable.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

without this line, the rest of the dockerfile doesn't actually inherit values like MIX_ENV for the build phases and then we need to find a different workaround (like prefixing commands with MIX_ENV=prod mix local.rebar --force)

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I believe we have intentionally not done this because then all variables will be logged in the build logs. The current way of injecting build variables prevents that.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

but this is injecting default values of environment variables. isn't it okay for these ones to be logged? like MIX_ENV= prod

there shouldn't be any user specific secrets set as defaults afaik.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@coffee-cup is this the only discussion point that's blocking? I will comb through the codebase a bit and see if there are any defaults that we might handle differently. the alternative here is to add some flag to explicitly say "in this case, please fill in the defaults"

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I believe these are all variables (see the screenshot)
screenshot-2025-01-08-14 02 17

The problem is that the variables returned from the providers are not available at build time?

@nbw nbw force-pushed the nbw/elixir-build branch from ada3548 to d82498a Compare January 5, 2025 10:21
@nbw nbw force-pushed the nbw/elixir-build branch from d82498a to 0b46f6a Compare January 5, 2025 11:04
start_cmd=start_cmd,}
} else {
// Copy over app files if provided, otherwise copy all files
let copy_cmds = match &self.only_include_files {
Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

START PHASE: if only_include_files is specified, then use it. Otherwise default to COPY . /app

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I believe only_include_files was only used when the runtime image was different from the build image. So this makes the uses the same 👍

@nbw nbw added release/minor Author minor release release/patch Author patch release and removed release/minor Author minor release labels Jan 5, 2025
let mut phase = StartPhase::new("./build/erlang-shipment/entrypoint.sh run");
phase.only_include_files = Some(vec!["build/erlang-shipment".into()]);
phase
StartPhase::new("./build/erlang-shipment/entrypoint.sh run")
Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

failed build

the phase.only_include_files = Some(vec!["build/erlang-shipment".into()]); was never working to begin with (since it was fixed in this PR). We can remove the line to keep the previous behaviour.

CleanShot 2025-01-05 at 20 55 45@2x

the only_include_files breaks the build and was never working to begin
with
@nbw nbw force-pushed the nbw/elixir-build branch from 7bd6ac9 to b32a1b2 Compare January 5, 2025 12:00
@nbw nbw requested a review from coffee-cup January 6, 2025 00:50
@nbw nbw changed the title fix phoenix/elixir build fix phoenix/elixir recipe Jan 6, 2025
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
release/patch Author patch release
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants