1use std::collections::{BTreeMap, BTreeSet, HashMap};
2use std::pin::Pin;
3use std::sync::Arc;
4
5use anyhow::anyhow;
6use axum::extract::{Path, Query};
7use axum::{Extension, Json};
8use bitcoin::hashes::sha256;
9use fedimint_core::module::{FEDIMINT_GATEWAY_ALPN, IrohGatewayRequest, IrohGatewayResponse};
10use fedimint_core::net::iroh::build_iroh_endpoint;
11use fedimint_core::task::TaskGroup;
12use fedimint_gateway_common::STOP_ENDPOINT;
13use fedimint_logging::LOG_GATEWAY;
14use iroh::endpoint::Incoming;
15use reqwest::StatusCode;
16use serde::de::DeserializeOwned;
17use serde_json::json;
18use tracing::info;
19use url::Url;
20
21use crate::Gateway;
22use crate::error::{GatewayError, PublicGatewayError};
23use crate::rpc_server::verify_bolt11_preimage_v2_get;
24
25type GetHandler = Box<
28 dyn Fn(
29 Extension<Arc<Gateway>>,
30 )
31 -> Pin<Box<dyn Future<Output = Result<Json<serde_json::Value>, GatewayError>> + Send>>
32 + Send
33 + Sync,
34>;
35
36type PostHandler = Box<
39 dyn Fn(
40 Extension<Arc<Gateway>>,
41 serde_json::Value,
42 )
43 -> Pin<Box<dyn Future<Output = Result<Json<serde_json::Value>, GatewayError>> + Send>>
44 + Send
45 + Sync,
46>;
47
48fn make_get_handler<F, Fut>(f: F) -> GetHandler
50where
51 F: Fn(Extension<Arc<Gateway>>) -> Fut + Clone + Send + Sync + 'static,
52 Fut: Future<Output = Result<Json<serde_json::Value>, GatewayError>> + Send + 'static,
53{
54 Box::new(move |gateway: Extension<Arc<Gateway>>| {
55 let f = f.clone();
56 Box::pin(async move {
57 let res = f(gateway).await?;
58 Ok(res)
59 })
60 })
61}
62
63fn make_post_handler<P, F, Fut>(f: F) -> PostHandler
65where
66 P: DeserializeOwned + Send + 'static,
67 F: Fn(Extension<Arc<Gateway>>, Json<P>) -> Fut + Clone + Send + Sync + 'static,
68 Fut: Future<Output = Result<Json<serde_json::Value>, GatewayError>> + Send + 'static,
69{
70 Box::new(
71 move |gateway: Extension<Arc<Gateway>>, value: serde_json::Value| {
72 let f = f.clone();
73 Box::pin(async move {
74 let payload: P = serde_json::from_value(value)
75 .map_err(|e| PublicGatewayError::Unexpected(anyhow!(e.to_string())))?;
76 let res = f(gateway, Json(payload)).await?;
77 Ok(res)
78 })
79 },
80 )
81}
82
83pub struct Handlers {
89 get_handlers: BTreeMap<String, GetHandler>,
90 post_handlers: BTreeMap<String, PostHandler>,
91 authenticated_routes: BTreeSet<String>,
92}
93
94impl Handlers {
95 pub fn new() -> Self {
96 let mut authenticated_routes = BTreeSet::new();
97 authenticated_routes.insert(STOP_ENDPOINT.to_string());
98 Handlers {
99 get_handlers: BTreeMap::new(),
100 post_handlers: BTreeMap::new(),
101 authenticated_routes,
102 }
103 }
104
105 pub fn add_handler<F, Fut>(&mut self, route: &str, f: F, is_authenticated: bool)
106 where
107 F: Fn(Extension<Arc<Gateway>>) -> Fut + Clone + Send + Sync + 'static,
108 Fut: Future<Output = Result<Json<serde_json::Value>, GatewayError>> + Send + 'static,
109 {
110 if is_authenticated {
111 self.authenticated_routes.insert(route.to_string());
112 }
113 self.get_handlers
114 .insert(route.to_string(), make_get_handler(f));
115 }
116
117 pub fn get_handler(&self, route: &str) -> Option<&GetHandler> {
118 self.get_handlers.get(route)
119 }
120
121 pub fn add_handler_with_payload<P, F, Fut>(&mut self, route: &str, f: F, is_authenticated: bool)
122 where
123 P: DeserializeOwned + Send + 'static,
124 F: Fn(Extension<Arc<Gateway>>, Json<P>) -> Fut + Clone + Send + Sync + 'static,
125 Fut: Future<Output = Result<Json<serde_json::Value>, GatewayError>> + Send + 'static,
126 {
127 if is_authenticated {
128 self.authenticated_routes.insert(route.to_string());
129 }
130
131 self.post_handlers
132 .insert(route.to_string(), make_post_handler(f));
133 }
134
135 pub fn get_handler_with_payload(&self, route: &str) -> Option<&PostHandler> {
136 self.post_handlers.get(route)
137 }
138
139 pub fn is_authenticated(&self, route: &str) -> bool {
140 self.authenticated_routes.contains(route)
141 }
142}
143
144pub async fn start_iroh_endpoint(
147 gateway: &Arc<Gateway>,
148 task_group: TaskGroup,
149 handlers: Arc<Handlers>,
150) -> anyhow::Result<()> {
151 info!("Building Iroh Endpoint...");
152 let iroh_endpoint = build_iroh_endpoint(
153 gateway.iroh_sk.clone(),
154 gateway.iroh_listen,
155 gateway.iroh_dns.clone(),
156 gateway.iroh_relays.clone(),
157 FEDIMINT_GATEWAY_ALPN,
158 )
159 .await?;
160 let gw_clone = gateway.clone();
161 let tg_clone = task_group.clone();
162 let handlers_clone = handlers.clone();
163 info!("Spawning accept loop...");
164 task_group.spawn("Gateway Iroh", |_| async move {
165 while let Some(incoming) = iroh_endpoint.accept().await {
166 info!("Accepted new connection. Spawning handler...");
167 tg_clone.spawn_cancellable_silent(
168 "handle endpoint accept",
169 handle_incoming_iroh_request(
170 incoming,
171 gw_clone.clone(),
172 handlers_clone.clone(),
173 tg_clone.clone(),
174 ),
175 );
176 }
177 });
178
179 info!(target: LOG_GATEWAY, "Successfully started iroh endpoint");
180
181 Ok(())
182}
183
184async fn handle_incoming_iroh_request(
187 incoming: Incoming,
188 gateway: Arc<Gateway>,
189 handlers: Arc<Handlers>,
190 task_group: TaskGroup,
191) -> anyhow::Result<()> {
192 let connection = incoming.accept()?.await?;
193 let remote_node_id = &connection.remote_node_id()?;
194 info!(%remote_node_id, "Handler received connection");
195 while let Ok((mut send, mut recv)) = connection.accept_bi().await {
196 let request = recv.read_to_end(100_000).await?;
197 let request = serde_json::from_slice::<IrohGatewayRequest>(&request)?;
198
199 let (status, body) = handle_request(
200 &request,
201 gateway.clone(),
202 handlers.clone(),
203 task_group.clone(),
204 )
205 .await?;
206
207 let response = IrohGatewayResponse {
208 status: status.as_u16(),
209 body: body.0,
210 };
211 let response = serde_json::to_vec(&response)?;
212
213 send.write_all(&response).await?;
214 send.finish()?;
215 }
216 Ok(())
217}
218
219async fn handle_request(
224 request: &IrohGatewayRequest,
225 gateway: Arc<Gateway>,
226 handlers: Arc<Handlers>,
227 task_group: TaskGroup,
228) -> anyhow::Result<(StatusCode, Json<serde_json::Value>)> {
229 if handlers.is_authenticated(&request.route) && iroh_verify_password(&gateway, request).is_err()
230 {
231 return Ok((StatusCode::UNAUTHORIZED, Json(json!(()))));
232 }
233
234 if request.route == STOP_ENDPOINT {
237 let body = crate::rpc_server::stop(Extension(task_group), Extension(gateway)).await?;
238 return Ok((StatusCode::OK, body));
239 }
240
241 if request.route.starts_with("/verify") {
245 let url = Url::parse(&format!("http://localhost{}", request.route))?;
247 let mut segments = url.path_segments().unwrap();
249 let hash_str = segments.next();
250
251 let payment_hash: sha256::Hash = hash_str.ok_or(anyhow!("No has present"))?.parse()?;
252
253 let query_map: HashMap<String, String> = url.query_pairs().into_owned().collect();
255
256 let body =
257 verify_bolt11_preimage_v2_get(Extension(gateway), Path(payment_hash), Query(query_map))
258 .await?;
259
260 return Ok((StatusCode::OK, body));
261 }
262
263 let (status, body) = match &request.params {
264 Some(params) => {
265 if let Some(handler) = handlers.get_handler_with_payload(&request.route) {
266 (
267 StatusCode::OK,
268 handler(Extension(gateway), params.clone()).await?,
269 )
270 } else {
271 return Err(anyhow!("Iroh handler received request with unknown route"));
272 }
273 }
274 None => {
275 if let Some(handler) = handlers.get_handler(&request.route) {
276 (StatusCode::OK, handler(Extension(gateway)).await?)
277 } else {
278 return Err(anyhow!("Iroh handler received request with unknown route"));
279 }
280 }
281 };
282
283 Ok((status, body))
284}
285
286fn iroh_verify_password(
289 gateway: &Arc<Gateway>,
290 request: &IrohGatewayRequest,
291) -> anyhow::Result<()> {
292 if let Some(password) = request.password.as_ref()
293 && bcrypt::verify(password, &gateway.bcrypt_password_hash.to_string())?
294 {
295 return Ok(());
296 }
297
298 Err(anyhow!("Invalid password"))
299}