Client Message

One of the biggest strengths of the Nostr network is the almost limitless possibilities for interoperable user-facing applications. In protocol terminology these applications are often referred to as Clients. Where relays provide a data housing mechanism the clients get that data in front of users in myriad of wild and wonderful ways. 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 3 main types of messages which clients construct and pass to relays as JSON arrays. This section is concerned with the construction of these message objects using the Client Message Module.

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

Serialize/deserialize to/from JSON

Rust

TODO

Python

The ClientMessage class easily handles the construction of the 3 main message types EVENT, REQ, and CLOSE. In the examples below we can utilize the relevant class methods event(), req() and close(), respectively, to create the client message objects.

Once we have the ClientMessage 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 ClientMessageEnum class. These allow for logical tests to be performed to establish the type of message object being assessed (for example, is_req() will return a bool result assessing if the object represents an REQ message type).

    # Event client message
    print("  Event Client Message:")
    client_message = ClientMessage.event(event)
    print(f"     - Event Message: {client_message.as_enum().is_event_msg()}")
    print(f"     - JSON: {client_message.as_json()}")

Note that when constructing a REQ we want to pass through a Filter object which will allow the relay to return data meeting a given set of criteria. Please jump to the Filter section for more details on how to construct these objects.

    # Request client message
    print("  Request Client Message:")
    f = Filter().id(event.id())
    client_message = ClientMessage.req(subscription_id="ABC123", filters=[f])
    print(f"     - Request Message: {client_message.as_enum().is_req()}")
    print(f"     - JSON: {client_message.as_json()}")
    # Close client message
    print("  Close Client Message:")
    client_message = ClientMessage.close("ABC123")
    print(f"     - Close Message: {client_message.as_enum().is_close()}")
    print(f"     - JSON: {client_message.as_json()}")

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

    # Parse Messages from JSON and/or Enum
    print("  Parse Client Messages:")
    client_message = ClientMessage.from_json('["REQ","ABC123",{"#p":["421a4dd67be773903f805bcb7975b4d3377893e0e09d7563b8972ee41031f551"]}]')
    print(f"     - ENUM: {client_message.as_enum()}")
    f = Filter().pubkey(keys.public_key())
    client_message = ClientMessage.from_enum(ClientMessageEnum.REQ("ABC123", filters=[f]))
    print(f"     - JSON: {client_message.as_json()}")
JavaScript

TODO

Kotlin

TODO

Swift

TODO

Authorization and Count Messages

Rust

TODO

Python

As an extension of the client 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 clients can request simple counts of events from relays. These are constructed in much the same way as the earlier message examples, by using the ClientMessage 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.

    # Auth client message  (NIP42)
    print("  Auth Client Message:")
    client_message = ClientMessage.auth(event)
    print(f"     - Auth Message: {client_message.as_enum().is_auth()}")
    print(f"     - JSON: {client_message.as_json()}")

Note that COUNT is effectively a specific type of REQ message therefore it utilizes the Filter object in constructing the criteria which should be used by the relay to return the count value.

    # Count client message (NIP45)
    print("  Count Client Message:")
    f = Filter().pubkey(keys.public_key())
    client_message = ClientMessage.count(subscription_id="ABC123", filters=[f])
    print(f"     - Count Message: {client_message.as_enum().is_count()}")
    print(f"     - JSON: {client_message.as_json()}")
JavaScript

TODO

Kotlin

TODO

Swift

TODO

Error Messages

Rust

TODO

Python

Finally, the ClientMessageEnum class also opens up three additional message types NEG_OPEN(), NEG_CLOSE() and NEG_MSG(). 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 can be handled by clients. To construct these we need to first create them as instance of the ClientMessageEnum class and then pass these into a ClientMessage object using the from_enum() method.

    # Negative Open Message
    print("  Negative Client Message (open):")
    client_message = ClientMessage.from_enum(ClientMessageEnum.NEG_OPEN("ABC123", filter=f, id_size=32, initial_message="<hex-msg>"))
    print(f"     - Negative Error Open: {client_message.as_enum().is_neg_open()}")
    print(f"     - JSON: {client_message.as_json()}")
    # Negative Close Message
    print("  Negative Client Message (close):")
    client_message = ClientMessage.from_enum(ClientMessageEnum.NEG_CLOSE("ABC123"))
    print(f"     - Negative Error Close: {client_message.as_enum().is_neg_close()}")
    print(f"     - JSON: {client_message.as_json()}")
    # Negative Error Message
    print("  Negative Client Message (message):")
    client_message = ClientMessage.from_enum(ClientMessageEnum.NEG_MSG("ABC123", message="This is not the message you are looking for"))
    print(f"     - JSON: {client_message.as_json()}")
    print(f"     - Negative Error Message: {client_message.as_enum().is_neg_msg()}")
JavaScript

TODO

Kotlin

TODO

Swift

TODO