Amarula

Elixir CIHex.pmHexdocsLicense: MIT

A WhatsApp Web client for Elixir — connect to WhatsApp the way the web/desktop app does: pair once by scanning a QR code with your phone, then send and receive messages from your own Elixir code.

Amarula is a faithful port of Baileys (the TypeScript WhatsApp Web library) to idiomatic Elixir/OTP. It speaks the real protocol end to end: the Noise handshake, the Signal Protocol for end-to-end encryption, WhatsApp's binary message format, multi-device (LID), groups, and history sync.

⚠️ Unofficial — use at your own risk. Amarula is not affiliated with, endorsed by, or sponsored by WhatsApp or Meta. WhatsApp does not support third-party clients, and automating an account can violate WhatsApp's Terms of Service. WhatsApp may ban any number you use with it, with no warning and no appeal. Only use accounts you own and can afford to lose; never use it for spam, bulk messaging, or anything against WhatsApp's terms. The maintainers provide this software as-is (see LICENSE) and take no responsibility for banned accounts or any other consequences of its use.

Features

Install

def deps do
[
{:amarula, "~> 0.1.0"}
]
end

Quick start

# Start a connection. Events (the QR code, incoming messages) are delivered to
# parent_pid — here, the current process.
{:ok, conn} =
Amarula.new(%{profile: :me})
|> Amarula.connect(parent_pid: self())
# First run: you get a QR code. Scan it on your phone:
# WhatsApp → Settings → Linked Devices → Link a device
receive do
{:whatsapp, :connection_update, %{qr: qr}} when is_binary(qr) ->
IO.puts(qr) # render this as a QR code
end
# Once linked you get an :open update — now you can send.
receive do
{:whatsapp, :connection_update, %{connection: :open}} -> :ready
end
Amarula.send_text(conn, "5511999999999@s.whatsapp.net", "hello from Elixir!")

:profile names this account's stored credentials, so the next run reconnects without a new QR. See Amarula (the public API) for the full set of send/receive functions.

The QR code

qr is a plain string — you render it to a scannable image however you like (terminal art, an eqrcode PNG, an HTML <img>). It's four comma-separated fields, ref,noiseKeyB64,identityKeyB64,advSecretKeyB64, where ref rotates every ~20s (each rotation emits a fresh :connection_update — re-render on each). Render it as-is; don't reformat. Example with eqrcode:

{:whatsapp, :connection_update, %{qr: qr}} ->
qr |> EQRCode.encode() |> EQRCode.png() |> then(&File.write!("qr.png", &1))

Events & connection flow

Everything reaches you as {:whatsapp, type, data} messages at parent_pid. You never poll — you react to events. Here's what to expect, and when.

sequenceDiagram
participant App as Your app
participant A as Amarula
participant WA as WhatsApp
App->>A: Amarula.new(%{profile}) |> connect(parent_pid: self())
A->>WA: WebSocket + Noise handshake
A-->>App: {:connection_update, %{connection: :connecting}}
WA-->>A: pair-device (ref)
A-->>App: {:connection_update, %{qr: "ref,...keys"}}
Note over App,WA: QR rotates ~20s → a new qr event each time, until scanned
WA-->>A: pair-success (phone scanned)
A-->>App: {:pairing_success, %{jid, lid, platform}}
Note over A,WA: Amarula persists credentials itself, scoped to the profile
Note over A,WA: stream restarts 515, Amarula re-handshakes automatically — no consumer action
A-->>App: {:connection_update, %{connection: :open}}
WA-->>A: offline batch + history
A-->>App: {:history_sync, %{chats, contacts, ...}}
A-->>App: {:chats_update, [...]}, {:contacts_update, [...]}
A-->>App: {:connection_update, %{received_pending_notifications: true}}

You never handle credentials. Amarula persists them itself, scoped to the connection's :profile (via the pluggable storage). The next boot with the same profile reconnects without a QR — no :creds_update event, no saving on your side.

The 515 stream restart after pairing is handled internally — Amarula reconnects and re-handshakes with the new credentials on its own. You don't handle it; just wait for connection: :open.

Re-login (already paired)

sequenceDiagram
participant App as Your app
participant A as Amarula
participant WA as WhatsApp
App->>A: connect(parent_pid: self()) %% same :profile
A->>WA: WebSocket + Noise handshake (saved creds)
A-->>App: {:connection_update, %{connection: :connecting}}
Note over A,WA: no QR — credentials already exist
A-->>App: {:connection_update, %{connection: :open}}
A-->>App: {:history_sync, ...} (incremental), {:chats_update, ...}
A-->>App: {:connection_update, %{received_pending_notifications: true}}

Steady state (messaging)

sequenceDiagram
participant App as Your app
participant A as Amarula
participant WA as WhatsApp
Note over App,WA: Incoming
WA-->>A: encrypted message
A-->>App: {:messages_upsert, %{from, id, messages: [%Amarula.Msg{}]}}
Note right of App: read msg.type and msg.content, download_media for files
Note over App,WA: Outgoing
App->>A: Amarula.send_text(conn, jid, "hi")
A->>WA: encrypt + send
WA-->>A: receipt (delivered / read / played)
A-->>App: {:receipt_update, %{message_ids, status, ...}}
Note over App,WA: Background, whenever they change
WA-->>A: group change / block / contact photo / app-state
A-->>App: {:group_update | :blocklist_update | :contacts_update | :chats_update, ...}

Sending (synchronous to you, concurrent underneath)

Amarula.send_text/3 (and friends) block until the send actually completes — you get the real {:ok, msg_id} or {:error, reason}, not a fire-and-forget acknowledgement. But under the hood sends are non-blocking and concurrent:

The consequence: if you fire two sends in parallel (from two processes, or two Tasks), you may get the second one's result before the first's — each returns when its own send finishes, not in call order. Within a single sequential caller it still looks plain synchronous; the concurrency only shows when you actually send in parallel.

It's a bar counter: you place your order and step aside (the counter takes the next order); your drink is made in parallel; you're called back when yours is ready — fast orders come out first.

sequenceDiagram
participant App as Your app
participant S as Socket
participant SA as SenderAlice
participant SB as SenderBob
App->>S: send_text bob ... slow, new recipient
S-->>SB: dispatch, Socket returns at once
App->>S: send_text alice ... fast, cached session
S-->>SA: dispatch, Socket still free
Note over SB: USync + bundle fetch, slow
SA-->>App: {:ok, alice_msg_id} Alice finishes first
SB-->>App: {:ok, bob_msg_id} Bob finishes later

Want true fire-and-forget? Wrap the call in your own Task — the library gives you the honest result and lets you choose the concurrency.

Event reference

EventDataWhen
:connection_update%{connection: :connecting|:open, qr, received_pending_notifications} (partial)lifecycle transitions; qr during pairing
:pairing_success%{jid, lid, platform}phone scanned the QR (first link only)
:messages_upsert%{from, id, messages: [%Amarula.Msg{}]}an incoming message (see Amarula.Msg)
:receipt_update%{message_ids, from, participant, status, timestamp}a message you sent was delivered/read/played
:history_sync%{chats, contacts, ...}initial + incremental history download
:chats_update / :contacts_update[%Amarula.Chat{}] / [%Amarula.Contact{}]history / app-state sync
:group_update%{group, author, action}a group's membership/metadata changed
:blocklist_update[%{jid, action}]you blocked/unblocked someone
:errora reason terma connection error

Try it

Runnable examples live in examples/:

# Pair a device and listen (shows a QR, then prints incoming messages)
mix run examples/pair.exs my_profile
# Send one message through a supervised connection, then exit
mix run examples/send_message.exs 5511999999999 "hello from amarula"

examples/connection.ex is a small supervised GenServer wrapper you can copy into a real app.

Configuration

Most settings are per-connection, passed to Amarula.new/1 (you usually only set :profile):

Amarula.new(%{
profile: :me, # required — names + scopes stored state
storage: {Amarula.Storage.File, root: "./data"},# storage backend (defaults to File)
sync_full_history: false, # skip the full history download
max_retries: 5,
connect_timeout_ms: 30_000
})

The full key list (with defaults) is in Amarula.Config. Only the pluggable backends are app-global:

config :amarula, :default_storage_adapter, Amarula.Storage.File
config :amarula, :retry_cache_adapter, Amarula.RetryCache.ETS

Logging

Amarula logs through Logger. Almost everything is :debug; only connection lifecycle, pairing, and errors are :info+. So at config :logger, level: :info your console won't be flooded. To silence Amarula specifically:

Logger.put_module_level(Amarula.Protocol.Socket.ConnectionManager, :warning)

For production observability prefer Amarula.Telemetry (structured :telemetry events) over log scraping.

Documentation

Development

mix deps.get # install dependencies
mix compile # compile
mix test # run the test suite
mix format # format
mix credo # lint
mix dialyzer # type checking

Protocol Buffers

When the WhatsApp protocol definitions in proto/wa_proto.proto change, recompile them:

protoc --elixir_opt=package_prefix=Amarula.Protocol:lib proto/wa_proto.proto

This regenerates lib/amarula/protocol/proto/wa_proto.pb.ex under the Amarula.Protocol.Proto.* namespace.

License & credits

Amarula is released under the MIT License, © 2026 Roberto Trevisan.

It is a port of Baileys (© 2025 Rajeh Taher/WhiskeySockets), also MIT-licensed — that license permits this use, and Baileys' copyright + permission notice are retained in LICENSE and NOTICE as it requires. Huge thanks to the Baileys authors for the reference implementation.

Unofficial. Not affiliated with, endorsed by, or sponsored by WhatsApp or Meta. Use it on accounts you control and in line with WhatsApp's terms.