Start here: mix mob.doctor
Before diving into specific issues, run:
mix mob.doctor
This checks your entire environment in one go — required tools, mob.exs
configuration, OTP runtime caches, and connected devices — and prints specific
fix instructions for anything wrong. Most setup problems are caught here.
=== Mob Doctor ===
Tools
✓ adb — /usr/bin/adb
✗ xcrun — not found
Install Xcode command-line tools:
xcode-select --install
Project
✗ mob_dir — path not found: /Users/you/old/path/to/mob
Update mob.exs — the path must exist on this machine
OTP Cache
✗ OTP iOS simulator — directory exists but contains no erts-* — extraction was incomplete
Remove the stale directory and re-download:
rm -rf ~/.mob/cache/otp-ios-sim-73ba6e0f
mix mob.install
Devices
⚠ Android devices — none connected
Connect a device via USB (enable USB debugging) or start an emulator
3 failures — fix the issues above and re-run mix mob.doctor.The sections below cover issues that mix mob.doctor doesn't catch — runtime
behaviour, distribution quirks, and platform-specific edge cases.
Common issues encountered during development and how to resolve them.
Elixir or Hex version too old
Symptom: mix deps.get or mix mob.install fails with errors like
no matching version found, invalid requirement, or dependency resolution
failures that look unrelated to your code.
Cause: Mob requires Elixir 1.18 or later. Older versions of Hex (pre-2.0)
also have issues resolving some package requirements used by mob_dev.
Check:
mix mob.doctor # shows Elixir, OTP, and Hex versions with ✓/✗
elixir --version
mix hex --version
Fix — Hex (fast, no version manager needed):
mix local.hex --force
Fix — Elixir (choose the method that matches how you installed it):
# mise
mise install elixir@latest && mise use elixir@latest
# asdf
asdf install elixir latest && asdf global elixir latest
# Homebrew
brew upgrade elixir
# Nix / nix-shell: update your shell.nix or flake.nix to use elixir_1_18
# Official installer: https://elixir-lang.org/install.html
After upgrading, re-fetch deps:
mix deps.get
mix mob.doctor # confirm versions are green
OTP cache: "No erts-* directory found"
Symptom: mix mob.deploy --native fails with:
ERROR: No erts-* directory found in ~/.mob/cache/otp-ios-sim-73ba6e0f
Have you built OTP for iOS simulator?Cause: The OTP cache directory was created during a previous download attempt that failed partway through (network error, SSL failure, or curl exiting non-zero). Because the directory exists, subsequent runs skip re-downloading, so the problem persists across restarts.
This is particularly common on Nix-managed macOS setups, where Nix provides
its own curl binary that uses a different CA certificate store than macOS system
curl. The GitHub release download may fail with an SSL certificate error that isn't
surfaced clearly.
Fix:
mix mob.doctor # confirms the problem and shows the exact path
rm -rf ~/.mob/cache/otp-ios-sim-73ba6e0f # remove stale cache
mix mob.install # re-download
If the download fails again (Nix curl SSL), download the tarball manually using the system curl:
/usr/bin/curl -L https://github.com/GenericJam/mob/releases/download/otp-73ba6e0f/otp-ios-sim-73ba6e0f.tar.gz \
-o /tmp/otp-ios-sim.tar.gz
mkdir -p ~/.mob/cache/otp-ios-sim-73ba6e0f
tar xzf /tmp/otp-ios-sim.tar.gz -C ~/.mob/cache/otp-ios-sim-73ba6e0f --strip-components=1
Verify it worked:
ls ~/.mob/cache/otp-ios-sim-73ba6e0f/erts-* # should list erts-16.x
mix mob.doctor # should show ✓ for iOS simulator OTP
EPMD port conflict with adb (port 4369)
Symptom: App crashes on launch, Erlang distribution fails to start, or
mix mob.connect hangs indefinitely. Often surfaces as a silent failure with
no obvious error message — the node never comes online.
Cause: EPMD (Erlang Port Mapper Daemon) is registered with IANA on port
- The Android Debug Bridge also uses port 4369 in certain configurations.
When both are active on the same machine, EPMD fails to bind and Erlang
distribution cannot start — which means the device BEAM can't register itself
and
mix mob.connectcan never find it.
Fix: Move EPMD to a port nothing else uses. Port 4380 is a safe choice.
Set ERL_EPMD_PORT in both the device BEAM startup and your local dev
environment.
In mob.exs:
config :mob_dev, epmd_port: 4380In your app's application.ex, pass the port when starting distribution:
Mob.Dist.ensure_started(
node: :"my_app_android@127.0.0.1",
cookie: :mob_secret,
epmd_port: Application.get_env(:mob_dev, :epmd_port, 4369)
)mob_dev will update the adb reverse tunnel to use the configured port
automatically.
Why 4369 conflicts: EPMD's port 4369 dates from 1993 (predating Android by 15 years). The collision is coincidental and there is no Erlang inside the Android toolchain. Moving off the default port also has a secondary benefit: Mob's device nodes become isolated from any other Elixir processes running on your Mac.
Distribution in production
In development, Mob.Dist.ensure_started/1 runs so mix mob.connect can
reach the app. In production the picture is different but not simply "turn it
off" — it depends on whether you want OTA BEAM updates.
No OTA updates: gate distribution on environment and leave it off in prod.
Mob.Dist.ensure_started/1 is a no-op unless explicitly called, so production
builds are safe by default:
# lib/my_app/application.ex
if Application.get_env(:my_app, :env) == :dev do
Mob.Dist.ensure_started(node: :"my_app_ios@127.0.0.1", cookie: :mob_secret)
endWith OTA BEAM updates: distribution needs to be live, but only during the
update session. The recommended pattern is on-demand: the app polls your server
over HTTP for an update manifest, starts EPMD + distribution only when an
update is available, connects to your update server's BEAM node to receive new
BEAMs via :code.load_binary, then shuts distribution back down. Because the
phone initiates the outbound connection, no inbound ports need to be open and
the cookie can be rotated per session via the manifest.
mix mob.connect finds no nodes
Check in order:
Is the app running on the device?
mix mob.devices # confirms device is visible to adb / xcrunDid distribution start on the device? Check the device log for
[mob] distribution started— if absent, theMob.Dist.ensure_started/1call either wasn't reached or failed silently (often due to the EPMD port conflict above).Do cookies match? The cookie in your app's
Mob.Dist.ensure_started/1call must match the--cookieflag passed tomix mob.connect(default:mob_secret).iOS: is the simulator booted?
xcrun simctl list devices | grep BootedAndroid: are the adb tunnels up?
adb reverse --list # should show tcp:4369 tcp:4369 (or your custom port) adb forward --list # should show tcp:9100 tcp:9100If missing, re-run
mix mob.connect— it sets these up automatically on each run.
Hot-push succeeds but changes don't appear
nl(MyApp.SomeScreen) returns {:ok, [...]} but the running screen still
shows old behaviour.
Cause: The screen process is still executing the old version of the module.
Hot code loading in the BEAM takes effect on the next function call — if the
screen is in the middle of a handle_event/3 or handle_info/2 call, it
finishes with the old code first.
Fix: Trigger any event on the screen (a tap, a Mob.Test.tap/2) to force
the process to make a new function call, picking up the new code. For layout
changes, navigate away and back so render/1 is called fresh.
If you need a guaranteed clean reload, use mix mob.deploy (restarts the app)
rather than hot-push.
Android: app crashes on first distribution startup
Symptom: App starts successfully, then crashes 3–5 seconds later. Logcat shows a signal abort or mutex error.
Cause: On Android, starting Erlang distribution too early (before the hwui
thread pool is fully initialised) causes a pthread_mutex_lock on destroyed mutex SIGABRT. This is why Mob.Dist.ensure_started/1 defers Node.start/2
by 3 seconds on Android.
Fix: Make sure you are calling Mob.Dist.ensure_started/1 and not calling
Node.start/2 directly. If you need distribution earlier, increase the defer
delay:
Mob.Dist.ensure_started(node: :"my_app_android@127.0.0.1", cookie: :mob_secret, delay: 5000)iOS simulator: node connects but RPC calls fail
Symptom: Node.connect/1 returns true, Node.list/0 shows the device
node, but :rpc.call/4 returns {:badrpc, :nodedown} or hangs.
Cause: The iOS simulator shares the Mac's network stack, so EPMD registration works. But if the dist port (default 9101 for iOS) is blocked by macOS firewall or already in use, the actual distribution channel can't be established even though EPMD sees the node.
Fix: Check if 9101 is in use:
lsof -i :9101
If something else is using it, configure a different dist port in
Mob.Dist.ensure_started/1 and update mob.exs accordingly.