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
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(())
}
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()}")
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;
TODO
TODO
Authorization and Count Messages
TODO
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()}")
TODO
TODO
TODO
Error Messages
TODO
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()}")
TODO
TODO
TODO