1#![doc = include_str!("../README.md")]
2#![deny(missing_docs)]
5#![deny(rustdoc::broken_intra_doc_links)]
6#![cfg_attr(any(), deny(clippy::unwrap_used))]
7use anyhow::Result;
8use http_relay::HttpRelay;
9use pubky::{Keypair, Pubky};
10use pubky_homeserver::{
11 storage_config::StorageConfigToml, ConfigToml, ConnectionString, DomainPort, HomeserverApp,
12 MockDataDir,
13};
14use std::{str::FromStr, time::Duration};
15use url::Url;
16
17pub struct Testnet {
23 pub(crate) dht: pkarr::mainline::Testnet,
24 pub(crate) pkarr_relays: Vec<pkarr_relay::Relay>,
25 pub(crate) http_relays: Vec<HttpRelay>,
26 pub(crate) homeservers: Vec<HomeserverApp>,
27 pub(crate) postgres_connection_string: Option<ConnectionString>,
28
29 temp_dirs: Vec<tempfile::TempDir>,
30}
31
32impl Testnet {
33 fn new_inner(seeded: bool) -> Result<Self> {
34 let dht = pkarr::mainline::Testnet::builder(2)
35 .seeded(seeded)
36 .build()?;
37
38 let testnet = Self {
39 dht,
40 pkarr_relays: vec![],
41 http_relays: vec![],
42 homeservers: vec![],
43 temp_dirs: vec![],
44 postgres_connection_string: Self::extract_postgres_connection_string_from_env_variable(
45 ),
46 };
47
48 Ok(testnet)
49 }
50
51 pub async fn new() -> Result<Self> {
53 Self::new_inner(true)
54 }
55
56 pub async fn new_unseeded() -> Result<Self> {
58 Self::new_inner(false)
59 }
60
61 pub async fn new_with_custom_postgres(
65 postgres_connection_string: ConnectionString,
66 ) -> Result<Self> {
67 let dht = pkarr::mainline::Testnet::builder(2).build()?;
68 let testnet: Testnet = Self {
69 dht,
70 pkarr_relays: vec![],
71 http_relays: vec![],
72 homeservers: vec![],
73 temp_dirs: vec![],
74 postgres_connection_string: Some(postgres_connection_string),
75 };
76
77 Ok(testnet)
78 }
79
80 fn extract_postgres_connection_string_from_env_variable() -> Option<ConnectionString> {
84 if let Ok(raw_con_string) = std::env::var("TEST_PUBKY_CONNECTION_STRING") {
85 if let Ok(con_string) = ConnectionString::new(&raw_con_string) {
86 return Some(con_string);
87 } else {
88 tracing::warn!("Invalid database connection string in TEST_PUBKY_CONNECTION_STRING environment variable. Ignoring it.");
89 }
90 }
91 None
92 }
93
94 pub async fn create_homeserver(&mut self) -> Result<&HomeserverApp> {
99 let mut config = ConfigToml::default_test_config();
100 if let Some(connection_string) = self.postgres_connection_string.as_ref() {
101 config.general.database_url = connection_string.clone();
102 }
103 let mock_dir = MockDataDir::new(config, Some(Keypair::from_secret(&[0; 32])))?;
104 self.create_homeserver_app_with_mock(mock_dir).await
105 }
106
107 pub async fn create_random_homeserver(&mut self) -> Result<&HomeserverApp> {
112 let mut config = ConfigToml::default_test_config();
113 if let Some(connection_string) = self.postgres_connection_string.as_ref() {
114 config.general.database_url = connection_string.clone();
115 }
116 let mock_dir = MockDataDir::new(config, Some(Keypair::random()))?;
117 self.create_homeserver_app_with_mock(mock_dir).await
118 }
119
120 pub async fn create_homeserver_app_with_mock(
124 &mut self,
125 mut mock_dir: MockDataDir,
126 ) -> Result<&HomeserverApp> {
127 mock_dir.config_toml.pkdns.dht_bootstrap_nodes = Some(self.dht_bootstrap_nodes());
128 if !self.dht_relay_urls().is_empty() {
129 mock_dir.config_toml.pkdns.dht_relay_nodes = Some(self.dht_relay_urls().to_vec());
130 }
131 mock_dir.config_toml.storage.backend = StorageConfigToml::InMemory;
132 let homeserver = HomeserverApp::start_with_mock_data_dir(mock_dir).await?;
133 self.homeservers.push(homeserver);
134 Ok(self
135 .homeservers
136 .last()
137 .expect("homeservers should be non-empty"))
138 }
139
140 pub async fn create_http_relay(&mut self) -> Result<&HttpRelay> {
142 let relay = HttpRelay::builder()
143 .http_port(0) .cors_allow_all(true)
145 .run()
146 .await?;
147 self.http_relays.push(relay);
148 Ok(self
149 .http_relays
150 .last()
151 .expect("http relays should be non-empty"))
152 }
153
154 pub async fn create_pkarr_relay(&mut self) -> Result<Url> {
158 let dir = tempfile::tempdir()?;
159 let mut builder = pkarr_relay::Relay::builder();
160 builder
161 .disable_rate_limiter()
162 .http_port(0)
163 .storage(dir.path().to_path_buf())
164 .pkarr(|builder| {
165 builder.no_default_network();
166 builder.bootstrap(&self.dht.bootstrap);
167 builder
168 });
169 let relay = unsafe { builder.run().await? };
170 let url = relay.local_url();
171 self.pkarr_relays.push(relay);
172 self.temp_dirs.push(dir);
173 Ok(url)
174 }
175
176 pub fn dht_bootstrap_nodes(&self) -> Vec<DomainPort> {
180 self.dht
181 .nodes
182 .iter()
183 .map(|node| {
184 let addr = node.info().local_addr();
185 DomainPort::from_str(&format!("{}:{}", addr.ip(), addr.port()))
186 .expect("boostrap nodes from the pkarr dht are always valid domain:port pairs")
187 })
188 .collect()
189 }
190
191 pub fn dht_relay_urls(&self) -> Vec<Url> {
193 self.pkarr_relays.iter().map(|r| r.local_url()).collect()
194 }
195
196 pub fn client_builder(&self) -> pubky::PubkyHttpClientBuilder {
198 let relays = self.dht_relay_urls();
199
200 let mut builder = pubky::PubkyHttpClient::builder();
201 builder.pkarr(|builder| {
202 builder.no_default_network();
203 builder.bootstrap(&self.dht.bootstrap);
204 if relays.is_empty() {
205 builder.no_relays();
206 } else {
207 builder
208 .relays(&relays)
209 .expect("testnet relays should be valid urls");
210 }
211 builder.request_timeout(Duration::from_millis(100));
214 builder
215 });
216
217 builder
218 }
219
220 pub fn client(&self) -> Result<pubky::PubkyHttpClient, pubky::BuildError> {
224 self.client_builder().build()
225 }
226
227 pub fn sdk(&self) -> Result<Pubky, pubky::BuildError> {
231 Ok(Pubky::with_client(self.client()?))
232 }
233
234 pub fn pkarr_client_builder(&self) -> pkarr::ClientBuilder {
236 let relays = self.dht_relay_urls();
237 let mut builder = pkarr::Client::builder();
238 builder.no_default_network(); builder.bootstrap(&self.dht.bootstrap);
240 if !relays.is_empty() {
241 builder
242 .relays(&relays)
243 .expect("Testnet relays should be valid urls");
244 }
245
246 builder
247 }
248}
249
250#[cfg(test)]
251mod test {
252 use crate::Testnet;
253 use pubky::Keypair;
254
255 #[tokio::test]
257 #[crate::test]
258 async fn test_keep_relays_alive_even_when_dropped() {
259 let mut testnet = Testnet::new().await.unwrap();
260 {
261 let _relay = testnet.create_http_relay().await.unwrap();
262 }
263 assert_eq!(testnet.http_relays.len(), 1);
264 }
265
266 #[tokio::test]
268 #[crate::test]
269 async fn test_boostrap_node_conversion() {
270 let testnet = Testnet::new().await.unwrap();
271 let nodes = testnet.dht_bootstrap_nodes();
272 assert_eq!(nodes.len(), 2);
273 }
274
275 #[tokio::test]
278 #[crate::test]
279 async fn test_signup() {
280 let mut testnet = Testnet::new().await.unwrap();
281 testnet.create_homeserver().await.unwrap();
282
283 let hs = testnet.homeservers.first().unwrap();
284 let sdk = testnet.sdk().unwrap();
285
286 let signer = sdk.signer(Keypair::random());
287
288 let session = signer.signup(&hs.public_key(), None).await.unwrap();
289 assert_eq!(session.info().public_key(), &signer.public_key());
290 }
291
292 #[tokio::test]
293 async fn test_independent_dhts() {
294 let t1 = Testnet::new().await.unwrap();
295 let t2 = Testnet::new().await.unwrap();
296
297 assert_ne!(t1.dht.bootstrap, t2.dht.bootstrap);
298 }
299
300 #[tokio::test]
302 async fn test_homeserver_resolvable() {
303 let mut testnet = Testnet::new().await.unwrap();
304 let hs_pubky = testnet.create_homeserver().await.unwrap().public_key();
305
306 let pkarr_client = testnet.pkarr_client_builder().build().unwrap();
308 let _packet = pkarr_client.resolve(&hs_pubky).await.unwrap();
309
310 let pubkey = hs_pubky.z32();
312 let _endpoint = pkarr_client
313 .resolve_https_endpoint(pubkey.as_str())
314 .await
315 .unwrap();
316 }
317
318 #[tokio::test]
322 #[crate::test]
323 async fn test_pkarr_relay_resolvable() {
324 let mut testnet = Testnet::new().await.unwrap();
325 testnet.create_pkarr_relay().await.unwrap();
326
327 let keypair = Keypair::random();
328
329 let client = testnet.pkarr_client_builder().build().unwrap();
331 let signed = pkarr::SignedPacket::builder().sign(&keypair).unwrap();
332 client.publish(&signed, None).await.unwrap();
333
334 let client = testnet.pkarr_client_builder().no_relays().build().unwrap();
337 let packet = client.resolve(&keypair.public_key()).await;
338 assert!(
339 packet.is_some(),
340 "Published packet is not available over the DHT."
341 );
342
343 let client = testnet.pkarr_client_builder().no_dht().build().unwrap();
347 let packet = client.resolve(&keypair.public_key()).await;
348 assert!(
349 packet.is_some(),
350 "Published packet is not available over the relay only."
351 );
352 }
353}