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