|  | @@ -4,6 +4,7 @@ use std::time::Duration;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  use cdk_common::MintQuoteBolt12Response;
 | 
	
		
			
				|  |  |  use tokio::sync::{mpsc, RwLock};
 | 
	
		
			
				|  |  | +#[cfg(not(target_arch = "wasm32"))]
 | 
	
		
			
				|  |  |  use tokio::time;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  use super::WsSubscriptionBody;
 | 
	
	
		
			
				|  | @@ -84,6 +85,7 @@ async fn convert_subscription(
 | 
	
		
			
				|  |  |      Some(())
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +#[cfg(not(target_arch = "wasm32"))]
 | 
	
		
			
				|  |  |  #[inline]
 | 
	
		
			
				|  |  |  pub async fn http_main<S: IntoIterator<Item = SubId>>(
 | 
	
		
			
				|  |  |      initial_state: S,
 | 
	
	
		
			
				|  | @@ -94,7 +96,7 @@ pub async fn http_main<S: IntoIterator<Item = SubId>>(
 | 
	
		
			
				|  |  |      _wallet: Arc<Wallet>,
 | 
	
		
			
				|  |  |  ) {
 | 
	
		
			
				|  |  |      let mut interval = time::interval(Duration::from_secs(2));
 | 
	
		
			
				|  |  | -    let mut subscribed_to = HashMap::<UrlType, (mpsc::Sender<_>, _, AnyState)>::new();
 | 
	
		
			
				|  |  | +    let mut subscribed_to = SubscribedTo::new();
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      for sub_id in initial_state {
 | 
	
		
			
				|  |  |          convert_subscription(sub_id, &subscriptions, &mut subscribed_to).await;
 | 
	
	
		
			
				|  | @@ -103,76 +105,133 @@ pub async fn http_main<S: IntoIterator<Item = SubId>>(
 | 
	
		
			
				|  |  |      loop {
 | 
	
		
			
				|  |  |          tokio::select! {
 | 
	
		
			
				|  |  |              _ = interval.tick() => {
 | 
	
		
			
				|  |  | -                for (url, (sender, _, last_state)) in subscribed_to.iter_mut() {
 | 
	
		
			
				|  |  | -                    tracing::debug!("Polling: {:?}", url);
 | 
	
		
			
				|  |  | -                    match url {
 | 
	
		
			
				|  |  | -                        UrlType::MintBolt12(id) => {
 | 
	
		
			
				|  |  | -                            let response = http_client.get_mint_quote_bolt12_status(id).await;
 | 
	
		
			
				|  |  | -                            if let Ok(response) = response {
 | 
	
		
			
				|  |  | -                                if *last_state == AnyState::MintBolt12QuoteState(response.clone()) {
 | 
	
		
			
				|  |  | -                                    continue;
 | 
	
		
			
				|  |  | -                                }
 | 
	
		
			
				|  |  | -                                *last_state = AnyState::MintBolt12QuoteState(response.clone());
 | 
	
		
			
				|  |  | -                                if let Err(err) = sender.try_send(NotificationPayload::MintQuoteBolt12Response(response)) {
 | 
	
		
			
				|  |  | -                                    tracing::error!("Error sending mint quote response: {:?}", err);
 | 
	
		
			
				|  |  | -                                }
 | 
	
		
			
				|  |  | -                            }
 | 
	
		
			
				|  |  | -                        },
 | 
	
		
			
				|  |  | -                        UrlType::Mint(id) => {
 | 
	
		
			
				|  |  | +                poll_subscriptions(&http_client, &mut subscribed_to).await;
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +            Some(subid) = new_subscription_recv.recv() => {
 | 
	
		
			
				|  |  | +                convert_subscription(subid, &subscriptions, &mut subscribed_to).await;
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +            Some(id) = on_drop.recv() => {
 | 
	
		
			
				|  |  | +                subscribed_to.retain(|_, (_, sub_id, _)| *sub_id != id);
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -                            let response = http_client.get_mint_quote_status(id).await;
 | 
	
		
			
				|  |  | -                            if let Ok(response) = response {
 | 
	
		
			
				|  |  | -                                if *last_state == AnyState::MintQuoteState(response.state) {
 | 
	
		
			
				|  |  | -                                    continue;
 | 
	
		
			
				|  |  | -                                }
 | 
	
		
			
				|  |  | -                                *last_state = AnyState::MintQuoteState(response.state);
 | 
	
		
			
				|  |  | -                                if let Err(err) = sender.try_send(NotificationPayload::MintQuoteBolt11Response(response)) {
 | 
	
		
			
				|  |  | -                                    tracing::error!("Error sending mint quote response: {:?}", err);
 | 
	
		
			
				|  |  | -                                }
 | 
	
		
			
				|  |  | -                            }
 | 
	
		
			
				|  |  | -                        }
 | 
	
		
			
				|  |  | -                        UrlType::Melt(id) => {
 | 
	
		
			
				|  |  | +#[cfg(target_arch = "wasm32")]
 | 
	
		
			
				|  |  | +#[inline]
 | 
	
		
			
				|  |  | +pub async fn http_main<S: IntoIterator<Item = SubId>>(
 | 
	
		
			
				|  |  | +    initial_state: S,
 | 
	
		
			
				|  |  | +    http_client: Arc<dyn MintConnector + Send + Sync>,
 | 
	
		
			
				|  |  | +    subscriptions: Arc<RwLock<HashMap<SubId, WsSubscriptionBody>>>,
 | 
	
		
			
				|  |  | +    mut new_subscription_recv: mpsc::Receiver<SubId>,
 | 
	
		
			
				|  |  | +    mut on_drop: mpsc::Receiver<SubId>,
 | 
	
		
			
				|  |  | +    _wallet: Arc<Wallet>,
 | 
	
		
			
				|  |  | +) {
 | 
	
		
			
				|  |  | +    let mut subscribed_to = SubscribedTo::new();
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -                            let response = http_client.get_melt_quote_status(id).await;
 | 
	
		
			
				|  |  | -                            if let Ok(response) = response {
 | 
	
		
			
				|  |  | -                                if *last_state == AnyState::MeltQuoteState(response.state) {
 | 
	
		
			
				|  |  | -                                    continue;
 | 
	
		
			
				|  |  | -                                }
 | 
	
		
			
				|  |  | -                                *last_state = AnyState::MeltQuoteState(response.state);
 | 
	
		
			
				|  |  | -                                if let Err(err) =  sender.try_send(NotificationPayload::MeltQuoteBolt11Response(response)) {
 | 
	
		
			
				|  |  | -                                    tracing::error!("Error sending melt quote response: {:?}", err);
 | 
	
		
			
				|  |  | -                                }
 | 
	
		
			
				|  |  | -                            }
 | 
	
		
			
				|  |  | -                        }
 | 
	
		
			
				|  |  | -                        UrlType::PublicKey(id) => {
 | 
	
		
			
				|  |  | -                            let responses = http_client.post_check_state(CheckStateRequest {
 | 
	
		
			
				|  |  | -                                ys: vec![*id],
 | 
	
		
			
				|  |  | -                            }
 | 
	
		
			
				|  |  | -                            ).await;
 | 
	
		
			
				|  |  | -                            if let Ok(mut responses) = responses {
 | 
	
		
			
				|  |  | -                                let response = if let Some(state) = responses.states.pop() {
 | 
	
		
			
				|  |  | -                                    state
 | 
	
		
			
				|  |  | -                                } else {
 | 
	
		
			
				|  |  | -                                    continue;
 | 
	
		
			
				|  |  | -                                };
 | 
	
		
			
				|  |  | +    for sub_id in initial_state {
 | 
	
		
			
				|  |  | +        convert_subscription(sub_id, &subscriptions, &mut subscribed_to).await;
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -                                if *last_state == AnyState::PublicKey(response.state) {
 | 
	
		
			
				|  |  | -                                    continue;
 | 
	
		
			
				|  |  | -                                }
 | 
	
		
			
				|  |  | -                                *last_state = AnyState::PublicKey(response.state);
 | 
	
		
			
				|  |  | -                                if let Err(err) = sender.try_send(NotificationPayload::ProofState(response)) {
 | 
	
		
			
				|  |  | -                                    tracing::error!("Error sending proof state response: {:?}", err);
 | 
	
		
			
				|  |  | -                                }
 | 
	
		
			
				|  |  | -                            }
 | 
	
		
			
				|  |  | -                        }
 | 
	
		
			
				|  |  | +    loop {
 | 
	
		
			
				|  |  | +        tokio::select! {
 | 
	
		
			
				|  |  | +            _ = gloo_timers::future::sleep(Duration::from_secs(2)) => {
 | 
	
		
			
				|  |  | +                poll_subscriptions(&http_client, &mut subscribed_to).await;
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +            subid = new_subscription_recv.recv() => {
 | 
	
		
			
				|  |  | +                match subid {
 | 
	
		
			
				|  |  | +                    Some(subid) => {
 | 
	
		
			
				|  |  | +                        convert_subscription(subid, &subscriptions, &mut subscribed_to).await;
 | 
	
		
			
				|  |  | +                    }
 | 
	
		
			
				|  |  | +                    None => {
 | 
	
		
			
				|  |  | +                        // New subscription channel closed - SubscriptionClient was dropped, terminate worker
 | 
	
		
			
				|  |  | +                        break;
 | 
	
		
			
				|  |  |                      }
 | 
	
		
			
				|  |  |                  }
 | 
	
		
			
				|  |  |              }
 | 
	
		
			
				|  |  | -            Some(subid) = new_subscription_recv.recv() => {
 | 
	
		
			
				|  |  | -                convert_subscription(subid, &subscriptions, &mut subscribed_to).await;
 | 
	
		
			
				|  |  | +            id = on_drop.recv() => {
 | 
	
		
			
				|  |  | +                match id {
 | 
	
		
			
				|  |  | +                    Some(id) => {
 | 
	
		
			
				|  |  | +                        subscribed_to.retain(|_, (_, sub_id, _)| *sub_id != id);
 | 
	
		
			
				|  |  | +                    }
 | 
	
		
			
				|  |  | +                    None => {
 | 
	
		
			
				|  |  | +                        // Drop notification channel closed - SubscriptionClient was dropped, terminate worker
 | 
	
		
			
				|  |  | +                        break;
 | 
	
		
			
				|  |  | +                    }
 | 
	
		
			
				|  |  | +                }
 | 
	
		
			
				|  |  |              }
 | 
	
		
			
				|  |  | -            Some(id) = on_drop.recv() => {
 | 
	
		
			
				|  |  | -                subscribed_to.retain(|_, (_, sub_id, _)| *sub_id != id);
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +async fn poll_subscriptions(
 | 
	
		
			
				|  |  | +    http_client: &Arc<dyn MintConnector + Send + Sync>,
 | 
	
		
			
				|  |  | +    subscribed_to: &mut SubscribedTo,
 | 
	
		
			
				|  |  | +) {
 | 
	
		
			
				|  |  | +    for (url, (sender, _, last_state)) in subscribed_to.iter_mut() {
 | 
	
		
			
				|  |  | +        tracing::debug!("Polling: {:?}", url);
 | 
	
		
			
				|  |  | +        match url {
 | 
	
		
			
				|  |  | +            UrlType::MintBolt12(id) => {
 | 
	
		
			
				|  |  | +                let response = http_client.get_mint_quote_bolt12_status(id).await;
 | 
	
		
			
				|  |  | +                if let Ok(response) = response {
 | 
	
		
			
				|  |  | +                    if *last_state == AnyState::MintBolt12QuoteState(response.clone()) {
 | 
	
		
			
				|  |  | +                        continue;
 | 
	
		
			
				|  |  | +                    }
 | 
	
		
			
				|  |  | +                    *last_state = AnyState::MintBolt12QuoteState(response.clone());
 | 
	
		
			
				|  |  | +                    if let Err(err) =
 | 
	
		
			
				|  |  | +                        sender.try_send(NotificationPayload::MintQuoteBolt12Response(response))
 | 
	
		
			
				|  |  | +                    {
 | 
	
		
			
				|  |  | +                        tracing::error!("Error sending mint quote response: {:?}", err);
 | 
	
		
			
				|  |  | +                    }
 | 
	
		
			
				|  |  | +                }
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +            UrlType::Mint(id) => {
 | 
	
		
			
				|  |  | +                let response = http_client.get_mint_quote_status(id).await;
 | 
	
		
			
				|  |  | +                if let Ok(response) = response {
 | 
	
		
			
				|  |  | +                    if *last_state == AnyState::MintQuoteState(response.state) {
 | 
	
		
			
				|  |  | +                        continue;
 | 
	
		
			
				|  |  | +                    }
 | 
	
		
			
				|  |  | +                    *last_state = AnyState::MintQuoteState(response.state);
 | 
	
		
			
				|  |  | +                    if let Err(err) =
 | 
	
		
			
				|  |  | +                        sender.try_send(NotificationPayload::MintQuoteBolt11Response(response))
 | 
	
		
			
				|  |  | +                    {
 | 
	
		
			
				|  |  | +                        tracing::error!("Error sending mint quote response: {:?}", err);
 | 
	
		
			
				|  |  | +                    }
 | 
	
		
			
				|  |  | +                }
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +            UrlType::Melt(id) => {
 | 
	
		
			
				|  |  | +                let response = http_client.get_melt_quote_status(id).await;
 | 
	
		
			
				|  |  | +                if let Ok(response) = response {
 | 
	
		
			
				|  |  | +                    if *last_state == AnyState::MeltQuoteState(response.state) {
 | 
	
		
			
				|  |  | +                        continue;
 | 
	
		
			
				|  |  | +                    }
 | 
	
		
			
				|  |  | +                    *last_state = AnyState::MeltQuoteState(response.state);
 | 
	
		
			
				|  |  | +                    if let Err(err) =
 | 
	
		
			
				|  |  | +                        sender.try_send(NotificationPayload::MeltQuoteBolt11Response(response))
 | 
	
		
			
				|  |  | +                    {
 | 
	
		
			
				|  |  | +                        tracing::error!("Error sending melt quote response: {:?}", err);
 | 
	
		
			
				|  |  | +                    }
 | 
	
		
			
				|  |  | +                }
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +            UrlType::PublicKey(id) => {
 | 
	
		
			
				|  |  | +                let responses = http_client
 | 
	
		
			
				|  |  | +                    .post_check_state(CheckStateRequest { ys: vec![*id] })
 | 
	
		
			
				|  |  | +                    .await;
 | 
	
		
			
				|  |  | +                if let Ok(mut responses) = responses {
 | 
	
		
			
				|  |  | +                    let response = if let Some(state) = responses.states.pop() {
 | 
	
		
			
				|  |  | +                        state
 | 
	
		
			
				|  |  | +                    } else {
 | 
	
		
			
				|  |  | +                        continue;
 | 
	
		
			
				|  |  | +                    };
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +                    if *last_state == AnyState::PublicKey(response.state) {
 | 
	
		
			
				|  |  | +                        continue;
 | 
	
		
			
				|  |  | +                    }
 | 
	
		
			
				|  |  | +                    *last_state = AnyState::PublicKey(response.state);
 | 
	
		
			
				|  |  | +                    if let Err(err) = sender.try_send(NotificationPayload::ProofState(response)) {
 | 
	
		
			
				|  |  | +                        tracing::error!("Error sending proof state response: {:?}", err);
 | 
	
		
			
				|  |  | +                    }
 | 
	
		
			
				|  |  | +                }
 | 
	
		
			
				|  |  |              }
 | 
	
		
			
				|  |  |          }
 | 
	
		
			
				|  |  |      }
 |