#![forbid(unsafe_code)]
#![warn(missing_docs)]
#![warn(clippy::large_futures)]
#![warn(rustdoc::bare_urls)]
#![allow(clippy::arc_with_non_send_sync)]
use std::collections::HashMap;
use std::future::Future;
use std::sync::Arc;
use std::sync::atomic::{AtomicBool, Ordering};
use std::time::Duration;
pub extern crate nostr;
use nostr::nips::nip47::{Notification, Request, Response};
use nostr_sdk::prelude::*;
mod api;
pub mod builder;
pub mod error;
pub mod prelude;
pub use self::api::*;
use self::builder::NostrWalletConnectBuilder;
#[doc(hidden)]
pub use self::error::Error;
const NOTIFICATIONS_ID: &str = "nwc-notifications";
#[allow(missing_docs)]
#[deprecated(since = "0.45.0", note = "Use NostrWalletConnect instead")]
pub type NWC = NostrWalletConnect;
#[derive(Debug, Clone)]
pub struct NostrWalletConnect {
uri: NostrWalletConnectUri,
client: Client,
timeout: Duration,
relay_opts: RelayOptions,
bootstrapped: Arc<AtomicBool>,
notifications_subscribed: Arc<AtomicBool>,
}
impl NostrWalletConnect {
#[inline]
pub fn new(uri: NostrWalletConnectUri) -> Self {
Self::builder(uri).build()
}
#[inline]
pub fn builder(uri: NostrWalletConnectUri) -> NostrWalletConnectBuilder {
NostrWalletConnectBuilder::new(uri)
}
fn from_builder(builder: NostrWalletConnectBuilder) -> Self {
let client: Client = match builder.monitor {
Some(monitor) => Client::builder().monitor(monitor).build(),
None => Client::default(),
};
Self {
uri: builder.uri,
client,
timeout: builder.timeout,
relay_opts: builder.relay,
bootstrapped: Arc::new(AtomicBool::new(false)),
notifications_subscribed: Arc::new(AtomicBool::new(false)),
}
}
#[inline]
pub fn uri(&self) -> &NostrWalletConnectUri {
&self.uri
}
pub async fn status(&self) -> HashMap<RelayUrl, RelayStatus> {
let relays = self.client.relays().await;
relays.into_iter().map(|(u, r)| (u, r.status())).collect()
}
async fn bootstrap(&self) -> Result<(), Error> {
if self.bootstrapped.load(Ordering::SeqCst) {
return Ok(());
}
for url in self.uri.relays.iter() {
self.client
.add_relay(url)
.opts(self.relay_opts.clone())
.await?;
}
self.client.connect().await;
self.bootstrapped.store(true, Ordering::SeqCst);
Ok(())
}
async fn send_request(&self, req: Request, timeout: Duration) -> Result<Response, Error> {
self.bootstrap().await?;
tracing::debug!("Sending request '{}'", req.as_json());
let event: Event = req.to_event(&self.uri)?;
let filter = Filter::new()
.author(self.uri.public_key)
.kind(Kind::WalletConnectResponse)
.event(event.id);
let mut stream = self
.client
.stream_events(filter)
.timeout(timeout)
.policy(ReqExitPolicy::WaitForEvents(1))
.await?;
self.client.send_event(&event).await?;
let (_, res) = stream.next().await.ok_or(Error::ResponseNotReceived)?;
let received_event: Event = res?;
let response: Response = Response::from_event(&self.uri, &received_event)?;
Ok(response)
}
#[inline]
pub fn pay_invoice(&self, request: PayInvoiceRequest) -> PayInvoice<'_> {
PayInvoice::new(self, request)
}
#[inline]
pub fn pay_keysend(&self, request: PayKeysendRequest) -> PayKeysend<'_> {
PayKeysend::new(self, request)
}
#[inline]
pub fn make_invoice(&self, request: MakeInvoiceRequest) -> MakeInvoice<'_> {
MakeInvoice::new(self, request)
}
#[inline]
pub fn lookup_invoice(&self, request: LookupInvoiceRequest) -> LookupInvoice<'_> {
LookupInvoice::new(self, request)
}
#[inline]
pub fn list_transactions(&self, params: ListTransactionsRequest) -> ListTransactions<'_> {
ListTransactions::new(self, params)
}
#[inline]
pub fn get_balance(&self) -> GetBalance<'_> {
GetBalance::new(self)
}
#[inline]
pub fn get_info(&self) -> GetInfo<'_> {
GetInfo::new(self)
}
pub async fn subscribe_to_notifications(&self) -> Result<(), Error> {
if self.notifications_subscribed.load(Ordering::SeqCst) {
tracing::debug!("Already subscribed to notifications");
return Ok(());
}
tracing::info!("Subscribing to wallet notifications...");
self.bootstrap().await?;
let client_keys = Keys::new(self.uri.secret.clone());
let client_pubkey = client_keys.public_key();
tracing::debug!("Client pubkey: {}", client_pubkey);
tracing::debug!("Wallet service pubkey: {}", self.uri.public_key);
let notification_filter = Filter::new()
.author(self.uri.public_key)
.pubkey(client_pubkey)
.kind(Kind::WalletConnectNotification)
.since(Timestamp::now());
tracing::debug!("Notification filter: {:?}", notification_filter);
self.client
.subscribe(notification_filter)
.with_id(SubscriptionId::new(NOTIFICATIONS_ID))
.await?;
self.notifications_subscribed.store(true, Ordering::SeqCst);
tracing::info!("Successfully subscribed to notifications");
Ok(())
}
pub async fn handle_notifications<F, Fut>(&self, func: F) -> Result<(), Error>
where
F: Fn(Notification) -> Fut,
Fut: Future<Output = Result<bool>>,
{
let mut notifications = self.client.notifications();
while let Some(notification) = notifications.next().await {
tracing::trace!("Received a client notification: {:?}", notification);
match notification {
ClientNotification::Event {
subscription_id,
event,
..
} => {
tracing::debug!(
"Received event: kind={}, author={}, id={}",
event.kind,
event.pubkey,
event.id
);
if subscription_id.as_str() != NOTIFICATIONS_ID {
tracing::trace!("Ignoring event with subscription id: {}", subscription_id);
continue;
}
if event.kind != Kind::WalletConnectNotification {
tracing::trace!("Ignoring event with kind: {}", event.kind);
continue;
}
tracing::info!("Processing wallet notification event");
match Notification::from_event(&self.uri, &event) {
Ok(nip47_notification) => {
tracing::info!(
"Successfully parsed notification: {:?}",
nip47_notification.notification_type
);
let exit: bool = func(nip47_notification)
.await
.map_err(|e| Error::Handler(e.to_string()))?;
if exit {
break;
}
}
Err(e) => {
tracing::error!("Failed to parse notification: {}", e);
tracing::debug!("Event content: {}", event.content);
return Err(Error::from(e));
}
}
}
ClientNotification::Shutdown => break,
_ => {}
}
}
Ok(())
}
pub async fn unsubscribe_from_notifications(&self) -> Result<(), Error> {
self.client
.unsubscribe(&SubscriptionId::new(NOTIFICATIONS_ID))
.await?;
self.notifications_subscribed.store(false, Ordering::SeqCst);
Ok(())
}
pub async fn reconnect_relay<'a, U>(&self, url: U) -> Result<(), Error>
where
U: Into<RelayUrlArg<'a>>,
{
if !self.bootstrapped.load(Ordering::SeqCst) {
return Ok(());
}
Ok(self.client.connect_relay(url).await?)
}
#[inline]
pub async fn shutdown(self) {
self.client.disconnect().await
}
}