Take a look at the Phoenix’s deployment guide. I think people usually run the migrations right before the app starts. I run those as one Docker command:
How do folks handle it when deploying to multiple nodes/instances? I would want to only execute the migration script once per deployment, and not per started app instance.
For k8s - initContainer comes to mind.
Marking an instance to only run migrations there, is also an option. That’s what rails deployment with capistrano looked like back in the days.
Yes. The first instance to acquire the lock will run the migrations. All other instances will wait for the first instance to finish and release the lock - they will then acquire the lock in turn, see the database already migrated, and do nothing.
Hey everyone, I know this is an old thread but building on the accepted answer, I ran into an issue where docker compose stop was timing out on Elixir containers. The problem was the shell script (sh) handling SIGTERM as PID 1 instead of the BEAM.
To fix this, I updated the custom Phoenix server script (rel/overlays/bin/server):
#!/bin/sh
set -eu
cd -P -- "$(dirname -- "$0")"
# added this migration line
./my_app eval MyApp.Release.migrate
PHX_SERVER=true exec ./my_app start
With this, the Dockerfile stays:
CMD ["/app/bin/server"]
Now migrations run on startup, and SIGTERM signals are properly handled. Hope this helps someone that may be going through the same problem! Thanks! (: