NostrEx
A Nostr client for Elixir applications. Connect to Nostr relays, manage subscriptions, send and receiving Nostr events.
Installation
Add nostr_ex to your list of dependencies in mix.exs:
def deps do
[
{:nostr_ex, "~> 0.2.2"}
]
endRequired dependencies:
nostr_ex depends on secp256k1, bitcoin-core's C implementation of the secp256k1 curve, via Sgiath's Elixir NIF. To compile the dependency successfully:
-
on Linux, you'll need
autotoolsinstalled -
on MacOS, you may need
make,autoconfandautobuild. -
using Nix, all you need is
autoreconfHookin your environment. It is included in the project devShell, seenix/shell.nix.
Usage
Connecting to Relays
# Connect to a relay
iex(1)> NostrEx.connect("wss://relay.example.com")
{:ok, "relay_example_com"}
Relays are tracked by names via the RelayRegistry. All public facing functions expect this name as input, so you don't have to worry about PIDs. See RelayManager.registered_names/0.
Receiving Events
Pass event filters to create_sub/1:
# Receive only new events
now = DateTime.utc_now() |> DateTime.to_unix()
NostrEx.create_sub(kinds: [1], since: now)
> {:ok, %NostrEx.Subscription{...}}
# Send the subscription
NostrEx.send_sub(sub)
> {:ok, "abc123f891..."}NostrEx receives events at the process that created the subscription. A simple event handler to print kind 1 notes might look like:
receive do
{:event, sub_id, %{kind: 1} = event} ->
IO.puts(event.content)
{:eose, sub_id, relay} ->
IO.puts("No more events for sub " <> sub_id <> " from relay " <> relay)
_ ->
IO.puts(:stderr, "Unexpected message received")
endThe Nostr events are received via PubSub, and it's up to you to implement how to handle those received events.
To subscribe to the given sub_id on a different process, call
NostrEx.listen(sub_id) from the process, a shorthand for
Registry.register(NostrEx.PubSub, sub_id, nil).
Similarly, unsubscribe the current process with Registry.unregister(NostrEx.PubSub, sub_id).
Sending Notes
# Create a private key, and send a simple note, returns the event ID
iex(2)> privkey = :crypto.strong_rand_bytes(32) |> Base.encode16(case: :lower)
"6dba065ffb6f51b4023d7d24a0c91c125c42ceff344d744d00f3c76e6cb5e03e"
# Create an event with kind and attrs
iex(4)> NostrEx.create_event(1, content: "hello joe")
{:ok, %Nostr.Event{
id: nil,
pubkey: nil,
kind: 1,
tags: [],
created_at: ~U[2025-08-03 15:29:15.261264Z],
content: "hello joe",
sig: nil
}}
# Sign the event with your hex-encoded private key
iex(3)> {:ok, signed} = NostrEx.sign_event(event, private_key)
{:ok,
%Nostr.Event{
id: "871a08bf8e1b6d286d92238ce44648a94f7397042dd01a4ecc6db0afed745ec3",
pubkey: "93155d8268a995888fe935ed9de633be690303ab37ba9d698c9f715076a99563",
kind: 1,
tags: [],
created_at: ~U[2025-08-03 15:33:30.652067Z],
content: "hello joe",
sig: "60278f60548d5fa49841e0b7518201625aba9a9cf1cdc6d72621290b1943c21971d90c5ca3c2fba49b00ef84f488bac8bc0932c8ccc5ba5e3af2121ce7ad67c9"
}}
# send it, returns the event ID
# and an error list
iex(4)> NostrEx.send_event(signed)
{:ok, "871a08bf8e1b6d286d92238ce44648a94f7397042dd01a4ecc6db0afed745ec3", []}
The send-type functions take a send_via option in opts to specify which relays to send the event to.
If not specified, all currently connected relays will be used.
Additionally, since most send operations usually happen towards multiple relays, the response is a tuple of the form {:ok, value, error_list} to send back partial failures where at least one send succeeded but others may not have.
NIP-05 Verification
# Verify a NIP-05 identifier
NostrEx.Nip05.verify("user@example.com")Architecture
NostrEx uses a supervision tree with the following components:
RelayManager: supervises WebSocket connections to relaysRelayAgent: manages subscription state across relaysSocket: handles individual WebSocket connectionsPubSub: use Registry to dispatch events to listenersRelayRegistry: Registry for mapping relay names to connection pids
This library is built on Sgiath's nostr_lib library. This dependency compiles the libsecp256k1 C library for cryptographic operations, therefore you will need a C compiler to build this project.
Contributing
Issues and pull requests are welcome! Please add tests for any new functionality.
License
MIT License - see LICENSE for details.