Relay Message

The backbone of the Nostr network is built on relays rather than application specific centralized databases. Clients use WebSockets as a means to connect to relays and pass relevant data back and forth around the network. In accordance with the protocol base specification (NIP-01) there are 5 main types of messages which relays construct as JSON arrays. This section is concerned with the construction of these message objects using the Relay Message Module.

For a more detailed explanation regarding the rules and handling of relay message objects please refer to the Nostr protocol documentation linked above.

Serialize/deserialize to/from JSON

Rust
use nostr::prelude::*;

pub fn relay_message() -> Result<()> {
    // Deserialize from json
    let json = r#"["EVENT", "random_string", {"id":"70b10f70c1318967eddf12527799411b1a9780ad9c43858f5e5fcd45486a13a5","pubkey":"379e863e8357163b5bce5d2688dc4f1dcc2d505222fb8d74db600f30535dfdfe","created_at":1612809991,"kind":1,"tags":[],"content":"test","sig":"273a9cd5d11455590f4359500bccb7a89428262b96b3ea87a756b770964472f8c3e87f5d5e64d8d2e859a71462a3f477b554565c4f2f326cb01dd7620db71502"}]"#;
    let msg = RelayMessage::from_json(json)?;

    // Serialize as json
    let json = msg.as_json();
    println!("{json}");

    Ok(())
}
Python

The RelayMessage class easily handles the construction of the 5 main message types EVENT, OK, EOSE (end of stored events), CLOSED and NOTICE. In the examples below we can utilize the relevant class methods event(), ok(), eose(), closed() and notice(), respectively, to create the relay message objects.

Once we have the RelayMessage objects we can use the as_enum() or as_json() methods to present their content. Note that when using as_enum() we unlock some additional methods associated with the RelayMessageEnum class. These allow for logical tests to be performed to establish the type of message object being assessed (for example, is_ok() will return a bool result assessing if the object represents an OK message type).

    # Create Event relay message
    print("  Event Relay Message:")
    relay_message = RelayMessage.event("subscription_ID_abc123", event)
    print(f"     - Event Message: {relay_message.as_enum().is_event_msg()}")
    print(f"     - JSON: {relay_message.as_json()}")
    # Create event acceptance relay message
    print("  Event Acceptance Relay Message:")
    relay_message = RelayMessage.ok(event.id(), False, "You have no power here, Gandalf The Grey")
    print(f"     - Event Acceptance Message: {relay_message.as_enum().is_ok()}")
    print(f"     - JSON: {relay_message.as_json()}")
    # Create End of Stored Events relay message
    print("  End of Stored Events Relay Message:")
    relay_message = RelayMessage.eose("subscription_ID_abc123")
    print(f"     - End of Stored Events Message: {relay_message.as_enum().is_end_of_stored_events()}")
    print(f"     - JSON: {relay_message.as_json()}")
    # Create Closed relay message
    print("  Closed Relay Message:")
    relay_message = RelayMessage.closed("subscription_ID_abc123", "So long and thanks for all the fish")
    print(f"     - Closed Message: {relay_message.as_enum().is_closed()}")
    print(f"     - JSON: {relay_message.as_json()}")
    # Create Notice relay message
    print("  Notice Relay Message:")
    relay_message = RelayMessage.notice("You have been served")
    print(f"     - Notice Message: {relay_message.as_enum().is_notice()}")
    print(f"     - JSON: {relay_message.as_json()}")

When presented with a relay message object as either a JSON or an instance of the RelayMessageEnum class we can parse these data using the from_json() or from_enum() methods, respectively.

    # Parse Messages from JSON and/or Enum
    print("  Parse Relay Messages:")
    relay_message = RelayMessage.from_json('["NOTICE","You have been served"]')
    print(f"     - ENUM: {relay_message.as_enum()}")
    relay_message = RelayMessage.from_enum(RelayMessageEnum.NOTICE("You have been served"))
    print(f"     - JSON: {relay_message.as_json()}")
JavaScript
const { loadWasmSync, RelayMessage } = require("@rust-nostr/nostr");

function relayMessageJson() {
    // Load WASM
    loadWasmSync();

    // Deserialize from json
    let json1 = '["EVENT", "random_string", {"id":"70b10f70c1318967eddf12527799411b1a9780ad9c43858f5e5fcd45486a13a5","pubkey":"379e863e8357163b5bce5d2688dc4f1dcc2d505222fb8d74db600f30535dfdfe","created_at":1612809991,"kind":1,"tags":[],"content":"test","sig":"273a9cd5d11455590f4359500bccb7a89428262b96b3ea87a756b770964472f8c3e87f5d5e64d8d2e859a71462a3f477b554565c4f2f326cb01dd7620db71502"}]'
    let msg = RelayMessage.fromJson(json1)

    // Serialize as json
    let json2 = msg.asJson()
    console.log(json2);
}

module.exports.relayMessageJson = relayMessageJson;
Kotlin

TODO

Swift

TODO

Authorization and Count Messages

Rust

TODO

Python

As an extension of the relay messaging section of the protocol NIP-42 and NIP-45 introduce two new messaging types AUTH and COUNT.

The AUTH type is designed to facilitate a method by which clients can authenticate with a given relay. Whereas the COUNT type offers a method for relays to provide simple counts of events to clients (upon request). These are constructed in much the same way as the earlier message examples, by using the RelayMessage class in conjunction with the relevant methods auth() and count(). As before the as_enum() method can be used to unlock logical test methods (e.g., is_auth()) associated with these message objects.

    # Create Authorization relay message (NIP42)
    print("  Auth Relay Message:")
    relay_message = RelayMessage.auth("I Challenge You To A Duel! (or some other challenge string)")
    print(f"     - Auth Message: {relay_message.as_enum().is_auth()}")
    print(f"     - JSON: {relay_message.as_json()}")
    # Create Count relay message (NIP45)
    print("  Count Relay Message:")
    relay_message = RelayMessage.count("subscription_ID_abc123", 42)
    print(f"     - Count Message: {relay_message.as_enum().is_count()}")
    print(f"     - JSON: {relay_message.as_json()}")
JavaScript

TODO

Kotlin

TODO

Swift

TODO

Error Messages

Rust

TODO

Python

Finally, the RelayMessageEnum class also opens up two additional message types NEG_ERR() and NEG_CODE(). These do not form part of the standard protocol specification but do have specific uses when it comes to providing methods by which error messaging (or error codes) can be handled by relays. To construct these we need to first create them as instance of the RelayMessageEnum class and then pass these into a RelayMessage object using the from_enum() method.

    # Negative Error Code
    print("  Negative Relay Message (code):")
    relay_message_neg = RelayMessageEnum.NEG_ERR("subscription_ID_abc123", "404")
    relay_message = RelayMessage.from_enum(relay_message_neg)
    print(f"     - Negative Error Code: {relay_message.as_enum().is_neg_err()}")
    print(f"     - JSON: {relay_message.as_json()}")
    # Negative Error Message
    print("  Negative Relay Message (message):")
    relay_message_neg = RelayMessageEnum.NEG_MSG("subscription_ID_abc123", "This is not the message you are looking for")
    relay_message = RelayMessage.from_enum(relay_message_neg)
    print(f"     - Negative Error Message: {relay_message.as_enum().is_neg_msg()}")
    print(f"     - JSON: {relay_message.as_json()}")
JavaScript

TODO

Kotlin

TODO

Swift

TODO