Hi Elixir community
I’m having some trouble getting the BEAM VM to exit with exit code 0 upon auto-shutdown of the top-level supervisor.
For context, I’m writing an application that runs a cron job in cloud environment in a Docker container. So periodically, the cloud environment starts the container, which starts up my application, which runs a function to do some stuff. After that function exits (without raising), I want my application to exit and subsequently, I want the container to exit with exit code 0, so that the cloud environment knows that the run was successful.
Here’s my application module:
defmodule OpiniLead.Scrape.Application do
@moduledoc """
The application module for `OpiniLead.Scrape`.
"""
use Task, significant: true
use Application
def start_link(_) do
scrapers = Application.fetch_env!(:opinilead, :scrapers)
Task.start_link(OpiniLead.Scrape, :run, [scrapers])
end
def start(_type, _args) do
Supervisor.start_link([__MODULE__],
strategy: :one_for_one,
auto_shutdown: :all_significant,
name: OpiniLead.Scrape.Supervisor
)
end
end
When I start this application, the task runs successfully and the supervisor exits with reason :shutdown
, as expected. However, the BEAM VM seems to interpret this as a crash and exits with error code 1:
{"message":"Application opinilead exited: shutdown","@timestamp":"2025-03-03T10:56:46.693Z","ecs.version":"8.11.0","log.level":"notice","log.logger":"application_controller","log.origin":{"function":"info_exited/3","file.line":2125,"file.name":"application_controller.erl"}}
Kernel pid terminated (application_controller) ("{application_terminated,opinilead,shutdown}")
Crash dump is being written to: erl_crash.dump...done
How do I let the BEAM VM interpret :shutdown
as a normal / successful exit reason so it exits without a crash (and with exit code 0)? Or should I use an entirely different approach to release a Docker image that just runs a function until completion and then exits?
Note: I’m currently using mix releases to build my application into an executable for the Docker image, which is then started with bin/opinilead start
. I also have a few tests which call Application.start(:opinilead)
and then verify that the supervisor starts, runs and exits, which prevents me from straight-up calling System.stop(0)
in the application’s stop/1
callback.