aboutsummaryrefslogtreecommitdiff
path: root/include
diff options
context:
space:
mode:
authorLibravatar buttercat1791 <mjjurkoic@gmail.com>2024-05-26 12:31:12 -0500
committerLibravatar buttercat1791 <mjjurkoic@gmail.com>2024-05-26 12:31:12 -0500
commit1d6b704c15ee289037447fb566e7583962496650 (patch)
treee1cd3322d4f30746c32e7f46e58060a66c5abc23 /include
parenta7d3e3d2ce1797fc2b7fc918684777ae75adb822 (diff)
Refactor to separate interface from implementation
Diffstat (limited to 'include')
-rw-r--r--include/client/web_socket_client.hpp4
-rw-r--r--include/client/websocketpp_client.hpp44
-rw-r--r--include/data/data.hpp3
-rw-r--r--include/nostr.hpp153
-rw-r--r--include/nostr_service_base.hpp104
-rw-r--r--include/signer/noscrypt_signer.hpp55
-rw-r--r--include/signer/signer.hpp7
7 files changed, 225 insertions, 145 deletions
diff --git a/include/client/web_socket_client.hpp b/include/client/web_socket_client.hpp
index eca8e24..19fc949 100644
--- a/include/client/web_socket_client.hpp
+++ b/include/client/web_socket_client.hpp
@@ -10,14 +10,14 @@ namespace nostr
{
namespace client
{
-class IWebSocketClient;
-
/**
* @brief An interface for a WebSocket client singleton.
*/
class IWebSocketClient
{
public:
+ virtual ~IWebSocketClient() = default;
+
/**
* @brief Starts the client.
* @remark This method must be called before any other client methods.
diff --git a/include/client/websocketpp_client.hpp b/include/client/websocketpp_client.hpp
new file mode 100644
index 0000000..becf4fa
--- /dev/null
+++ b/include/client/websocketpp_client.hpp
@@ -0,0 +1,44 @@
+#pragma once
+
+#include "web_socket_client.hpp"
+
+namespace nostr
+{
+namespace client
+{
+/**
+ * @brief An implementation of the `IWebSocketClient` interface that uses the WebSocket++ library.
+ */
+class WebsocketppClient : public IWebSocketClient
+{
+public:
+ void start() override;
+
+ void stop() override;
+
+ void openConnection(std::string uri) override;
+
+ bool isConnected(std::string uri) override;
+
+ std::tuple<std::string, bool> send(std::string message, std::string uri) override;
+
+ std::tuple<std::string, bool> send(
+ std::string message,
+ std::string uri,
+ std::function<void(const std::string&)> messageHandler) override;
+
+ void receive(std::string uri, std::function<void(const std::string&)> messageHandler) override;
+
+ void closeConnection(std::string uri) override;
+
+private:
+ typedef websocketpp::client<websocketpp::config::asio_client> websocketpp_client;
+ typedef std::unordered_map<std::string, websocketpp::connection_hdl>::iterator connection_hdl_iterator;
+
+ websocketpp_client _client;
+ std::unordered_map<std::string, websocketpp::connection_hdl> _connectionHandles;
+ std::mutex _propertyMutex;
+};
+} // namespace client
+} // namespace nostr
+
diff --git a/include/data/data.hpp b/include/data/data.hpp
index 78c17c4..46156cd 100644
--- a/include/data/data.hpp
+++ b/include/data/data.hpp
@@ -13,9 +13,6 @@ namespace nostr
{
namespace data
{
-class Event;
-class Filters;
-
/**
* @brief A Nostr event.
* @remark All data transmitted over the Nostr protocol is encoded in JSON blobs. This struct
diff --git a/include/nostr.hpp b/include/nostr.hpp
index d21a86d..c555bbe 100644
--- a/include/nostr.hpp
+++ b/include/nostr.hpp
@@ -21,52 +21,35 @@
namespace nostr
{
-class NostrService;
-
// TODO: Create custom exception types for the nostr namespace.
-class NostrService
+class INostrServiceBase
{
public:
- NostrService(
- std::shared_ptr<plog::IAppender> appender,
- std::shared_ptr<client::IWebSocketClient> client,
- std::shared_ptr<signer::ISigner> signer);
- NostrService(
- std::shared_ptr<plog::IAppender> appender,
- std::shared_ptr<client::IWebSocketClient> client,
- std::shared_ptr<signer::ISigner> signer,
- std::vector<std::string> relays);
- ~NostrService();
-
- std::vector<std::string> defaultRelays() const;
-
- std::vector<std::string> activeRelays() const;
-
- std::unordered_map<std::string, std::vector<std::string>> subscriptions() const;
+ virtual ~INostrServiceBase() = default;
/**
* @brief Opens connections to the default Nostr relays of the instance, as specified in
* the constructor.
* @return A list of the relay URLs to which connections were successfully opened.
*/
- std::vector<std::string> openRelayConnections();
+ virtual std::vector<std::string> openRelayConnections() = 0;
/**
* @brief Opens connections to the specified Nostr relays.
* @returns A list of the relay URLs to which connections were successfully opened.
*/
- std::vector<std::string> openRelayConnections(std::vector<std::string> relays);
+ virtual std::vector<std::string> openRelayConnections(std::vector<std::string> relays) = 0;
/**
* @brief Closes all open relay connections.
*/
- void closeRelayConnections();
+ virtual void closeRelayConnections() = 0;
/**
* @brief Closes any open connections to the specified Nostr relays.
*/
- void closeRelayConnections(std::vector<std::string> relays);
+ virtual void closeRelayConnections(std::vector<std::string> relays) = 0;
/**
* @brief Publishes a Nostr event to all open relay connections.
@@ -74,7 +57,8 @@ public:
* to which relays the event was published successfully, and to which relays the event failed
* to publish.
*/
- std::tuple<std::vector<std::string>, std::vector<std::string>> publishEvent(std::shared_ptr<data::Event> event);
+ virtual std::tuple<std::vector<std::string>, std::vector<std::string>> publishEvent(
+ std::shared_ptr<data::Event> event) = 0;
/**
* @brief Queries all open relay connections for events matching the given set of filters, and
@@ -88,7 +72,8 @@ public:
* set on the filters in the range 1-64, inclusive. If no valid limit is given, it will be
* defaulted to 16.
*/
- std::vector<std::shared_ptr<data::Event>> queryRelays(std::shared_ptr<data::Filters> filters);
+ virtual std::vector<std::shared_ptr<data::Event>> queryRelays(
+ std::shared_ptr<data::Filters> filters) = 0;
/**
* @brief Queries all open relay connections for events matching the given set of filters.
@@ -104,11 +89,11 @@ public:
* events returned from the relay for the given filters. The service will not store the
* events, and they will not be accessible via `getNewEvents`.
*/
- std::string queryRelays(
+ virtual std::string queryRelays(
std::shared_ptr<data::Filters> filters,
std::function<void(const std::string&, std::shared_ptr<data::Event>)> eventHandler,
std::function<void(const std::string&)> eoseHandler,
- std::function<void(const std::string&, const std::string&)> closeHandler);
+ std::function<void(const std::string&, const std::string&)> closeHandler) = 0;
/**
* @brief Closes the subscription with the given ID on all open relay connections.
@@ -116,7 +101,8 @@ public:
* to which relays the message was sent successfully, and which relays failed to receive the
* message.
*/
- std::tuple<std::vector<std::string>, std::vector<std::string>> closeSubscription(std::string subscriptionId);
+ virtual std::tuple<std::vector<std::string>, std::vector<std::string>> closeSubscription(
+ std::string subscriptionId) = 0;
/**
* @brief Closes the subscription with the given ID on the given relay.
@@ -124,119 +110,12 @@ public:
* @remark If the subscription does not exist on the given relay, or if the relay is not
* connected, the method will do nothing and return false.
*/
- bool closeSubscription(std::string subscriptionId, std::string relay);
+ virtual bool closeSubscription(std::string subscriptionId, std::string relay) = 0;
/**
* @brief Closes all open subscriptions on all open relay connections.
* @returns A list of any subscription IDs that failed to close.
*/
- std::vector<std::string> closeSubscriptions();
-
- /**
- * @brief Closes all open subscriptions on the given relays.
- * @returns A list of any subscription IDs that failed to close.
- */
- std::vector<std::string> closeSubscriptions(std::vector<std::string> relays);
-
-private:
- ///< The maximum number of events the service will store for each subscription.
- const int MAX_EVENTS_PER_SUBSCRIPTION = 128;
-
- ///< The WebSocket client used to communicate with relays.
- std::shared_ptr<client::IWebSocketClient> _client;
- ///< The signer used to sign Nostr events.
- std::shared_ptr<signer::ISigner> _signer;
-
- ///< A mutex to protect the instance properties.
- std::mutex _propertyMutex;
- ///< The default set of Nostr relays to which the service will attempt to connect.
- std::vector<std::string> _defaultRelays;
- ///< The set of Nostr relays to which the service is currently connected.
- std::vector<std::string> _activeRelays;
- ///< A map from subscription IDs to the relays on which each subscription is open.
- std::unordered_map<std::string, std::vector<std::string>> _subscriptions;
-
- /**
- * @brief Determines which of the given relays are currently connected.
- * @returns A list of the URIs of currently-open relay connections from the given list.
- */
- std::vector<std::string> getConnectedRelays(std::vector<std::string> relays);
-
- /**
- * @brief Determines which of the given relays are not currently connected.
- * @returns A list of the URIs of currently-unconnected relays from the given list.
- */
- std::vector<std::string> getUnconnectedRelays(std::vector<std::string> relays);
-
- /**
- * @brief Determines whether the given relay is currently connected.
- * @returns True if the relay is connected, false otherwise.
- */
- bool isConnected(std::string relay);
-
- /**
- * @brief Removes the given relay from the instance's list of active relays.
- */
- void eraseActiveRelay(std::string relay);
-
- /**
- * @brief Opens a connection from the client to the given relay.
- */
- void connect(std::string relay);
-
- /**
- * @brief Closes the connection from the client to the given relay.
- */
- void disconnect(std::string relay);
-
- /**
- * @brief Generates a unique subscription ID that may be used to identify event requests.
- * @returns A stringified UUID.
- */
- std::string generateSubscriptionId();
-
- /**
- * @brief Generates a message requesting a relay to close the subscription with the given ID.
- * @returns A stringified JSON object representing the close request.
- */
- std::string generateCloseRequest(std::string subscriptionId);
-
- /**
- * @brief Indicates whether the the service has an open subscription with the given ID.
- * @returns True if the service has the subscription, false otherwise.
- */
- bool hasSubscription(std::string subscriptionId);
-
- /**
- * @brief Indicates whether the service has an open subscription with the given ID on the given
- * relay.
- * @returns True if the subscription exists on the relay, false otherwise.
- */
- bool hasSubscription(std::string subscriptionId, std::string relay);
-
- /**
- * @brief Parses EVENT messages received from the relay and invokes the given event handler.
- * @param message The raw message received from the relay.
- * @param eventHandler A callable object that will be invoked with the subscription ID and the
- * payload of the event.
- * @param eoseHandler A callable object that will be invoked with the subscription ID when the
- * relay sends an EOSE message, indicating it has reached the end of stored events for the
- * given query.
- * @param closeHandler A callable object that will be invoked with the subscription ID and the
- * message sent by the relay if the subscription is ended by the relay.
- */
- void onSubscriptionMessage(
- std::string message,
- std::function<void(const std::string&, std::shared_ptr<data::Event>)> eventHandler,
- std::function<void(const std::string&)> eoseHandler,
- std::function<void(const std::string&, const std::string&)> closeHandler);
-
- /**
- * @brief Parses OK messages received from the relay and invokes the given acceptance handler.
- * @remark The OK message type is sent to indicate whether the relay has accepted an event sent
- * by the client. Note that this is distinct from whether the message was successfully sent to
- * the relay over the WebSocket connection.
- */
- void onAcceptance(std::string message, std::function<void(const bool)> acceptanceHandler);
+ virtual std::vector<std::string> closeSubscriptions() = 0;
};
} // namespace nostr
diff --git a/include/nostr_service_base.hpp b/include/nostr_service_base.hpp
new file mode 100644
index 0000000..6abfedc
--- /dev/null
+++ b/include/nostr_service_base.hpp
@@ -0,0 +1,104 @@
+#pragma once
+
+#include "nostr.hpp"
+
+namespace nostr
+{
+class NostrServiceBase : public INostrServiceBase
+{
+public:
+ NostrServiceBase(
+ std::shared_ptr<plog::IAppender> appender,
+ std::shared_ptr<client::IWebSocketClient> client);
+
+ NostrServiceBase(
+ std::shared_ptr<plog::IAppender> appender,
+ std::shared_ptr<client::IWebSocketClient> client,
+ std::vector<std::string> relays);
+
+ ~NostrServiceBase() override;
+
+ std::vector<std::string> defaultRelays() const;
+
+ std::vector<std::string> activeRelays() const;
+
+ std::unordered_map<std::string, std::vector<std::string>> subscriptions() const;
+
+ std::vector<std::string> openRelayConnections() override;
+
+ std::vector<std::string> openRelayConnections(std::vector<std::string> relays) override;
+
+ void closeRelayConnections() override;
+
+ void closeRelayConnections(std::vector<std::string> relays) override;
+
+ // TODO: Make this method return a promise.
+ std::tuple<std::vector<std::string>, std::vector<std::string>> publishEvent(
+ std::shared_ptr<data::Event> event) override;
+
+ // TODO: Make this method return a promise.
+ // TODO: Add a timeout to this method to prevent hanging while waiting for the relay.
+ std::vector<std::shared_ptr<data::Event>> queryRelays(
+ std::shared_ptr<data::Filters> filters) override;
+
+ std::string queryRelays(
+ std::shared_ptr<data::Filters> filters,
+ std::function<void(const std::string&, std::shared_ptr<data::Event>)> eventHandler,
+ std::function<void(const std::string&)> eoseHandler,
+ std::function<void(const std::string&, const std::string&)> closeHandler) override;
+
+ std::tuple<std::vector<std::string>, std::vector<std::string>> closeSubscription(
+ std::string subscriptionId) override;
+
+ bool closeSubscription(std::string subscriptionId, std::string relay) override;
+
+ std::vector<std::string> closeSubscriptions() override;
+
+private:
+ ///< The maximum number of events the service will store for each subscription.
+ const int MAX_EVENTS_PER_SUBSCRIPTION = 128;
+
+ ///< The WebSocket client used to communicate with relays.
+ std::shared_ptr<client::IWebSocketClient> _client;
+
+ ///< A mutex to protect the instance properties.
+ std::mutex _propertyMutex;
+
+ ///< The default set of Nostr relays to which the service will attempt to connect.
+ std::vector<std::string> _defaultRelays;
+
+ ///< The set of Nostr relays to which the service is currently connected.
+ std::vector<std::string> _activeRelays;
+
+ ///< A map from subscription IDs to the relays on which each subscription is open.
+ std::unordered_map<std::string, std::vector<std::string>> _subscriptions;
+
+ std::vector<std::string> _getConnectedRelays(std::vector<std::string> relays);
+
+ std::vector<std::string> _getUnconnectedRelays(std::vector<std::string> relays);
+
+ bool _isConnected(std::string relay);
+
+ void _eraseActiveRelay(std::string relay);
+
+ void _connect(std::string relay);
+
+ void _disconnect(std::string relay);
+
+ std::string _generateSubscriptionId();
+
+ std::string _generateCloseRequest(std::string subscriptionId);
+
+ bool _hasSubscription(std::string subscriptionId);
+
+ bool _hasSubscription(std::string subscriptionId, std::string relay);
+
+ void _onSubscriptionMessage(
+ std::string message,
+ std::function<void(const std::string&, std::shared_ptr<data::Event>)> eventHandler,
+ std::function<void(const std::string&)> eoseHandler,
+ std::function<void(const std::string&, const std::string&)> closeHandler);
+
+ void _onAcceptance(std::string message, std::function<void(const bool)> acceptanceHandler);
+};
+} // namespace nostr
diff --git a/include/signer/noscrypt_signer.hpp b/include/signer/noscrypt_signer.hpp
new file mode 100644
index 0000000..1476303
--- /dev/null
+++ b/include/signer/noscrypt_signer.hpp
@@ -0,0 +1,55 @@
+#pragma once
+
+extern "C"
+{
+#include <noscrypt.h>
+}
+
+#include "signer.hpp"
+
+namespace nostr
+{
+namespace signer
+{
+class NoscryptSigner : public INostrConnectSigner
+{
+public:
+ NoscryptSigner(std::shared_ptr<plog::IAppender> appender);
+
+ ~NoscryptSigner();
+
+ void receiveConnection(std::string connectionToken) override;
+
+ void initiateConnection(
+ std::string relay,
+ std::string name,
+ std::string url,
+ std::string description) override;
+
+ void sign(std::shared_ptr<data::Event> event) override;
+
+private:
+ std::shared_ptr<NCContext> noscryptContext;
+
+ std::string localPrivateKey;
+ std::string localPublicKey;
+
+ /**
+ * @brief Initializes the noscrypt library context into the class's `context` property.
+ * @returns `true` if successful, `false` otherwise.
+ */
+ std::shared_ptr<NCContext> _initNoscryptContext();
+
+ void _logNoscryptResult(NCResult result);
+
+ /**
+ * @brief Generates a private/public key pair for local use.
+ * @returns The generated keypair of the form `[privateKey, publicKey]`, or a pair of empty
+ * strings if the function failed.
+ * @remarks This keypair is intended for temporary use, and should not be saved or used outside
+ * of this class.
+ */
+ std::tuple<std::string, std::string> _createLocalKeypair();
+};
+} // namespace signer
+} // namespace nostr
diff --git a/include/signer/signer.hpp b/include/signer/signer.hpp
index cc4fd03..e16aa38 100644
--- a/include/signer/signer.hpp
+++ b/include/signer/signer.hpp
@@ -14,15 +14,14 @@ namespace nostr
{
namespace signer
{
-class ISigner;
-class INostrConnectSigner;
-
/**
* @brief An interface for Nostr event signing that implements NIP-46.
*/
class ISigner
{
public:
+ virtual ~ISigner() = default;
+
/**
* @brief Signs the given Nostr event.
* @param event The event to sign.
@@ -34,6 +33,8 @@ public:
class INostrConnectSigner : public ISigner
{
public:
+ virtual ~INostrConnectSigner() = default;
+
virtual void receiveConnection(std::string connectionToken) = 0;
virtual void initiateConnection(