tor: fix service availability check

This commit is contained in:
ardocrat 2024-05-15 20:27:58 +03:00
parent 5749c5a367
commit 2bb51d6757
3 changed files with 112 additions and 89 deletions

View file

@ -18,6 +18,8 @@ use egui::{Align, Id, Layout, Margin, RichText, Rounding, ScrollArea};
use egui::os::OperatingSystem; use egui::os::OperatingSystem;
use egui::scroll_area::ScrollBarVisibility; use egui::scroll_area::ScrollBarVisibility;
use parking_lot::RwLock; use parking_lot::RwLock;
use tor_rtcompat::BlockOn;
use tor_rtcompat::tokio::TokioNativeTlsRuntime;
use grin_core::core::{amount_from_hr_string, amount_to_hr_string}; use grin_core::core::{amount_from_hr_string, amount_to_hr_string};
use grin_wallet_libwallet::SlatepackAddress; use grin_wallet_libwallet::SlatepackAddress;
@ -402,7 +404,7 @@ impl WalletTransport {
ui.add_space(6.0); ui.add_space(6.0);
} }
/// Draw Tor send content. /// Draw Tor receive content.
fn tor_receive_ui(&mut self, fn tor_receive_ui(&mut self,
ui: &mut egui::Ui, ui: &mut egui::Ui,
wallet: &Wallet, wallet: &Wallet,
@ -495,7 +497,7 @@ impl WalletTransport {
}); });
} }
/// Draw Tor receive content. /// Draw Tor send content.
fn tor_send_ui(&mut self, ui: &mut egui::Ui, cb: &dyn PlatformCallbacks) { fn tor_send_ui(&mut self, ui: &mut egui::Ui, cb: &dyn PlatformCallbacks) {
// Setup layout size. // Setup layout size.
let mut rect = ui.available_rect_before_wrap(); let mut rect = ui.available_rect_before_wrap();
@ -684,12 +686,13 @@ impl WalletTransport {
let send_success = self.tor_success.clone(); let send_success = self.tor_success.clone();
let mut wallet = wallet.clone(); let mut wallet = wallet.clone();
thread::spawn(move || { thread::spawn(move || {
tokio::runtime::Builder::new_multi_thread() let runtime = TokioNativeTlsRuntime::create().unwrap();
.enable_all() let runtime_tor = runtime.clone();
.build() runtime
.unwrap()
.block_on(async { .block_on(async {
if wallet.send_tor(a, &addr).await.is_some() { if wallet.send_tor(a, &addr, runtime_tor)
.await
.is_some() {
let mut w_send_success = send_success.write(); let mut w_send_success = send_success.write();
*w_send_success = true; *w_send_success = true;
} else { } else {
@ -742,15 +745,16 @@ impl WalletTransport {
let send_success = self.tor_success.clone(); let send_success = self.tor_success.clone();
let mut wallet = wallet.clone(); let mut wallet = wallet.clone();
thread::spawn(move || { thread::spawn(move || {
tokio::runtime::Builder::new_multi_thread() let runtime = TokioNativeTlsRuntime::create().unwrap();
.enable_all() let runtime_tor = runtime.clone();
.build() runtime
.unwrap()
.block_on(async { .block_on(async {
let addr_str = addr_text.as_str(); let addr_str = addr_text.as_str();
let addr = &SlatepackAddress::try_from(addr_str) let addr = &SlatepackAddress::try_from(addr_str)
.unwrap(); .unwrap();
if wallet.send_tor(a, &addr).await.is_some() { if wallet.send_tor(a, &addr, runtime_tor)
.await
.is_some() {
let mut w_send_success = send_success.write(); let mut w_send_success = send_success.write();
*w_send_success = true; *w_send_success = true;
} else { } else {

View file

@ -32,7 +32,7 @@ use sha2::Sha512;
use tokio::time::{sleep, sleep_until}; use tokio::time::{sleep, sleep_until};
use tor_config::CfgPath; use tor_config::CfgPath;
use tor_rtcompat::tokio::TokioNativeTlsRuntime; use tor_rtcompat::tokio::TokioNativeTlsRuntime;
use tor_rtcompat::Runtime; use tor_rtcompat::{BlockOn, PreferredRuntime, Runtime};
use tor_hsrproxy::OnionServiceReverseProxy; use tor_hsrproxy::OnionServiceReverseProxy;
use tor_hsrproxy::config::{Encapsulation, ProxyAction, ProxyPattern, ProxyRule, TargetAddr, ProxyConfigBuilder}; use tor_hsrproxy::config::{Encapsulation, ProxyAction, ProxyPattern, ProxyRule, TargetAddr, ProxyConfigBuilder};
use tor_hsservice::config::OnionServiceConfigBuilder; use tor_hsservice::config::OnionServiceConfigBuilder;
@ -68,7 +68,9 @@ pub struct Tor {
/// Starting Onion services identifiers. /// Starting Onion services identifiers.
starting_services: Arc<RwLock<BTreeSet<String>>>, starting_services: Arc<RwLock<BTreeSet<String>>>,
/// Failed Onion services identifiers. /// Failed Onion services identifiers.
failed_services: Arc<RwLock<BTreeSet<String>>> failed_services: Arc<RwLock<BTreeSet<String>>>,
/// Checking Onion services identifiers.
checking_services: Arc<RwLock<BTreeSet<String>>>
} }
impl Default for Tor { impl Default for Tor {
@ -76,14 +78,15 @@ impl Default for Tor {
Self { Self {
running_services: Arc::new(RwLock::new(BTreeMap::new())), running_services: Arc::new(RwLock::new(BTreeMap::new())),
starting_services: Arc::new(RwLock::new(BTreeSet::new())), starting_services: Arc::new(RwLock::new(BTreeSet::new())),
failed_services: Arc::new(RwLock::new(BTreeSet::new())) failed_services: Arc::new(RwLock::new(BTreeSet::new())),
checking_services: Arc::new(RwLock::new(BTreeSet::new()))
} }
} }
} }
impl Tor { impl Tor {
async fn build_client(runtime: TokioNativeTlsRuntime) async fn build_client(runtime: TokioNativeTlsRuntime)
-> (TorClient<TokioNativeTlsRuntime>, TorClientConfig) { -> (TorClient<TokioNativeTlsRuntime>, TorClientConfig) {
// Create Tor client config. // Create Tor client config.
let mut builder = let mut builder =
TorClientConfigBuilder::from_directories(TorConfig::state_path(), TorClientConfigBuilder::from_directories(TorConfig::state_path(),
@ -101,16 +104,15 @@ impl Tor {
// Create connected Tor client from config. // Create connected Tor client from config.
let config = builder.build().unwrap(); let config = builder.build().unwrap();
(TorClient::with_runtime(runtime) (TorClient::with_runtime(runtime)
.config(config.clone()) .config(config.clone())
.create_bootstrapped() .create_bootstrapped()
.await .await
.unwrap(), config) .unwrap(), config)
} }
/// Send post request using Tor. /// Send post request using Tor.
pub async fn post(body: String, url: String) -> Option<String> { pub async fn post(body: String, url: String, runtime: TokioNativeTlsRuntime) -> Option<String> {
// Create client. // Create client.
let runtime = TokioNativeTlsRuntime::create().unwrap();
let (client, _) = Self::build_client(runtime).await; let (client, _) = Self::build_client(runtime).await;
// Create http tor-powered client to post data. // Create http tor-powered client to post data.
let tls_connector = TlsConnector::builder().unwrap().build().unwrap(); let tls_connector = TlsConnector::builder().unwrap().build().unwrap();
@ -156,6 +158,12 @@ impl Tor {
r_services.contains(id) r_services.contains(id)
} }
/// Check if Onion service is checking.
pub fn is_service_checking(id: &String) -> bool {
let r_services = TOR_SERVER_STATE.checking_services.read().unwrap();
r_services.contains(id)
}
/// Stop running Onion service. /// Stop running Onion service.
pub fn stop_service(id: &String) { pub fn stop_service(id: &String) {
let mut w_services = TOR_SERVER_STATE.running_services.write().unwrap(); let mut w_services = TOR_SERVER_STATE.running_services.write().unwrap();
@ -180,70 +188,79 @@ impl Tor {
} }
let service_id = id.clone(); let service_id = id.clone();
let runtime = TokioNativeTlsRuntime::create().unwrap(); thread::spawn(move || {
let runtime_client = runtime.clone(); let runtime = TokioNativeTlsRuntime::create().unwrap();
runtime.spawn(async move { let runtime_client = runtime.clone();
let (client, config) = Self::build_client(runtime_client.clone()).await; runtime.spawn(async move {
// Add service key to keystore. let (client, config) = Self::build_client(runtime_client.clone()).await;
let hs_nickname = HsNickname::new(service_id.clone()).unwrap(); // Add service key to keystore.
Self::add_service_key(config.fs_mistrust(), &key, &hs_nickname); let hs_nickname = HsNickname::new(service_id.clone()).unwrap();
// Launch Onion service. Self::add_service_key(config.fs_mistrust(), &key, &hs_nickname);
let service_config = OnionServiceConfigBuilder::default() // Launch Onion service.
.nickname(hs_nickname.clone()) let service_config = OnionServiceConfigBuilder::default()
.build() .nickname(hs_nickname.clone())
.unwrap(); .build()
let (service, request) = client.launch_onion_service(service_config).unwrap(); .unwrap();
let (service, request) = client.launch_onion_service(service_config).unwrap();
// Launch service proxy.
let addr = SocketAddr::new(IpAddr::from(Ipv4Addr::LOCALHOST), port); // Launch service proxy.
tokio::spawn( let addr = SocketAddr::new(IpAddr::from(Ipv4Addr::LOCALHOST), port);
Self::run_service_proxy(addr, client, service.clone(), request, hs_nickname.clone()) tokio::spawn(
).await.unwrap(); Self::run_service_proxy(addr, client, service.clone(), request, hs_nickname.clone())
).await.unwrap();
// Check service availability.
let url = format!("http://{}", service.onion_name().unwrap().to_string()); // Check service availability if not checking.
std::thread::spawn(move || { if Self::is_service_checking(&service_id) {
thread::sleep(Duration::from_millis(3000)); return;
let runtime = TokioNativeTlsRuntime::create().unwrap(); }
let runtime_client = runtime.clone(); let url = format!("http://{}/v2/foreign", service.onion_name().unwrap().to_string());
runtime.spawn(async move { thread::spawn(move || {
loop { let runtime = TokioNativeTlsRuntime::create().unwrap();
if !Self::is_service_running(&service_id) && let client_runtime = runtime.clone();
!Self::is_service_starting(&service_id) { // Put service to checking.
break; {
} let mut w_services = TOR_SERVER_STATE.checking_services.write().unwrap();
// Create client. w_services.insert(service_id.clone());
let (client, _) = Self::build_client(runtime_client.clone()).await;
// Create http tor-powered client to ping service.
let tls_connector = TlsConnector::builder().unwrap().build().unwrap();
let tor_connector = ArtiHttpConnector::new(client, tls_connector);
let http = hyper::Client::builder().build::<_, Body>(tor_connector);
match http.get(Uri::from_str(url.as_str()).unwrap()).await {
Ok(_) => {
// Remove service from starting.
let mut w_services = TOR_SERVER_STATE.starting_services.write().unwrap();
w_services.remove(&service_id);
println!("check success");
},
Err(e) => {
// Put service to starting.
let mut w_services = TOR_SERVER_STATE.starting_services.write().unwrap();
w_services.insert(service_id.clone());
println!("check err: {}", e);
},
}
if !Self::is_service_running(&service_id) &&
!Self::is_service_starting(&service_id) {
break;
}
} }
}).unwrap(); runtime
}); .spawn(async move {
}).unwrap(); loop {
println!("start loop");
// Create client.
let (client, _) = Self::build_client(client_runtime.clone()).await;
// Create http tor-powered client to ping service.
let tls_connector = TlsConnector::builder().unwrap().build().unwrap();
let tor_connector = ArtiHttpConnector::new(client, tls_connector);
let http = hyper::Client::builder().build::<_, Body>(tor_connector);
println!("start get {}", url);
match http.get(Uri::from_str(url.as_str()).unwrap()).await {
Ok(_) => {
// Remove service from starting.
let mut w_services = TOR_SERVER_STATE.starting_services.write().unwrap();
w_services.remove(&service_id);
println!("OK");
},
Err(e) => {
// Put service to starting.
let mut w_services = TOR_SERVER_STATE.starting_services.write().unwrap();
w_services.insert(service_id.clone());
println!("err: {}", e);
},
}
if !Self::is_service_running(&service_id) &&
!Self::is_service_starting(&service_id) {
// Remove service from checking.
let mut w_services = TOR_SERVER_STATE.checking_services.write().unwrap();
w_services.remove(&service_id);
break;
}
}
}).unwrap();
});
}).unwrap();
});
} }
/// Launch Onion service proxy. /// Launch Onion service proxy.
@ -282,7 +299,6 @@ impl Tor {
.handle_requests(runtime, nickname.clone(), request) .handle_requests(runtime, nickname.clone(), request)
.await { .await {
Ok(()) => { Ok(()) => {
println!("service stopped");
// Remove service from running. // Remove service from running.
let mut w_services = let mut w_services =
TOR_SERVER_STATE.running_services.write().unwrap(); TOR_SERVER_STATE.running_services.write().unwrap();

View file

@ -24,6 +24,8 @@ use std::thread::Thread;
use std::time::Duration; use std::time::Duration;
use futures::channel::oneshot; use futures::channel::oneshot;
use serde_json::{json, Value}; use serde_json::{json, Value};
use tor_rtcompat::tokio::TokioNativeTlsRuntime;
use rand::Rng;
use grin_api::{ApiServer, Router}; use grin_api::{ApiServer, Router};
use grin_chain::SyncStatus; use grin_chain::SyncStatus;
@ -40,7 +42,6 @@ use grin_wallet_impls::{DefaultLCProvider, DefaultWalletImpl, HTTPNodeClient};
use grin_wallet_libwallet::{address, Error, InitTxArgs, IssueInvoiceTxArgs, NodeClient, RetrieveTxQueryArgs, RetrieveTxQuerySortField, RetrieveTxQuerySortOrder, Slate, SlatepackAddress, SlateState, SlateVersion, StatusMessage, TxLogEntry, TxLogEntryType, VersionedSlate, WalletInst, WalletLCProvider}; use grin_wallet_libwallet::{address, Error, InitTxArgs, IssueInvoiceTxArgs, NodeClient, RetrieveTxQueryArgs, RetrieveTxQuerySortField, RetrieveTxQuerySortOrder, Slate, SlatepackAddress, SlateState, SlateVersion, StatusMessage, TxLogEntry, TxLogEntryType, VersionedSlate, WalletInst, WalletLCProvider};
use grin_wallet_libwallet::api_impl::owner::{cancel_tx, retrieve_summary_info, retrieve_txs}; use grin_wallet_libwallet::api_impl::owner::{cancel_tx, retrieve_summary_info, retrieve_txs};
use grin_wallet_util::OnionV3Address; use grin_wallet_util::OnionV3Address;
use rand::Rng;
use crate::AppConfig; use crate::AppConfig;
use crate::node::{Node, NodeConfig}; use crate::node::{Node, NodeConfig};
@ -675,7 +676,9 @@ impl Wallet {
} }
/// Send amount to provided address with Tor transport. /// Send amount to provided address with Tor transport.
pub async fn send_tor(&mut self, amount: u64, addr: &SlatepackAddress) -> Option<Slate> { pub async fn send_tor(&mut self,
amount: u64, addr: &SlatepackAddress,
runtime: TokioNativeTlsRuntime) -> Option<Slate> {
// Initialize transaction. // Initialize transaction.
let send_res = self.send(amount); let send_res = self.send(amount);
@ -706,7 +709,7 @@ impl Wallet {
}).to_string(); }).to_string();
// Send request to receiver. // Send request to receiver.
let req_res = Tor::post(body, url).await; let req_res = Tor::post(body, url, runtime).await;
if req_res.is_none() { if req_res.is_none() {
cancel_tx(); cancel_tx();
return None; return None;