Skip to main content

pubky_testnet/
testnet.rs

1#![doc = include_str!("../README.md")]
2//!
3
4#![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
17/// A local test network for Pubky Core development.
18/// Can create a flexible amount of pkarr relays, http relays and homeservers.
19///
20/// Keeps track of the components and can create new ones.
21/// Cleans up all resources when dropped.
22pub 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    /// Run a new testnet with a (fully-initialized) local DHT.
52    pub async fn new() -> Result<Self> {
53        Self::new_inner(true)
54    }
55
56    /// Run a new testnet with a (faster, but partially-initialized) local DHT.
57    pub async fn new_unseeded() -> Result<Self> {
58        Self::new_inner(false)
59    }
60
61    /// Run a new testnet with a local DHT.
62    /// Pass an optional postgres connection string to use for the homeserver.
63    /// If None, the default test connection string is used.
64    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    /// Extract the postgres connection string from the TEST_PUBKY_CONNECTION_STRING environment variable.
81    /// If the environment variable is not set, None is returned.
82    /// If the environment variable is set, but the connection string is invalid, a warning is logged and None is returned.
83    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    /// Run the full homeserver app with core and admin server.
95    ///
96    /// Uses [`ConfigToml::default_test_config()`] which enables the admin server.
97    /// Automatically listens on ephemeral ports and uses this Testnet's bootstrap nodes and relays.
98    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    /// Run the full homeserver app with core and admin server using a freshly generated random keypair.
108    ///
109    /// Uses [`ConfigToml::default_test_config()`] which enables the admin server.
110    /// Automatically listens on ephemeral ports and uses this Testnet's bootstrap nodes and relays.
111    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    /// Run the full homeserver app with core and admin server
121    /// Automatically listens on the configured ports.
122    /// Automatically uses the configured bootstrap nodes and relays in this Testnet.
123    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    /// Run an HTTP Relay
141    pub async fn create_http_relay(&mut self) -> Result<&HttpRelay> {
142        let relay = HttpRelay::builder()
143            .http_port(0) // Random available port
144            .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    /// Run a new Pkarr relay.
155    ///
156    /// You can access the list of relays at [Self::pkarr_relays].
157    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    // === Getters ===
177
178    /// Returns a list of DHT bootstrapping nodes.
179    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    /// Returns a list of pkarr relays.
192    pub fn dht_relay_urls(&self) -> Vec<Url> {
193        self.pkarr_relays.iter().map(|r| r.local_url()).collect()
194    }
195
196    /// Create a [pubky::PubkyHttpClientBuilder] and configure it to use this local test network.
197    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            // 100ms timeout for requests. This makes methods like `resolve_most_recent` fast
212            // because it doesn't need to wait the default 2s which would slow down the tests.
213            builder.request_timeout(Duration::from_millis(100));
214            builder
215        });
216
217        builder
218    }
219
220    /// Creates a [`pubky::PubkyHttpClient`] pre-configured to use this test network.
221    ///
222    /// This is a convenience method that builds a client from `Self::client_builder`.
223    pub fn client(&self) -> Result<pubky::PubkyHttpClient, pubky::BuildError> {
224        self.client_builder().build()
225    }
226
227    /// Creates a [`pubky::Pubky`] SDK facade pre-configured to use this test network.
228    ///
229    /// This is a convenience method that builds a client from `Self::client_builder`.
230    pub fn sdk(&self) -> Result<Pubky, pubky::BuildError> {
231        Ok(Pubky::with_client(self.client()?))
232    }
233
234    /// Create a [pkarr::ClientBuilder] and configure it to use this local test network.
235    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(); // Remove DHT bootstrap nodes and relays
239        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    /// Make sure the components are kept alive even when dropped.
256    #[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    /// Boostrap node conversion
267    #[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    /// Test that a user can signup in the testnet.
276    /// This is an e2e tests to check if everything is correct.
277    #[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    /// If everything is linked correctly, the hs_pubky should be resolvable from the pkarr client.
301    #[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        // Make sure the pkarr packet of the hs is resolvable.
307        let pkarr_client = testnet.pkarr_client_builder().build().unwrap();
308        let _packet = pkarr_client.resolve(&hs_pubky).await.unwrap();
309
310        // Make sure the pkarr can resolve the hs_pubky.
311        let pubkey = hs_pubky.z32();
312        let _endpoint = pkarr_client
313            .resolve_https_endpoint(pubkey.as_str())
314            .await
315            .unwrap();
316    }
317
318    /// Test relay resolvable.
319    /// This simulates pkarr clients in a browser.
320    /// Made due to https://github.com/pubky/pkarr/issues/140
321    #[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        // Publish packet on the DHT without using the relay.
330        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        // Resolve packet with a new client to prevent caching
335        // Only use the DHT, no relays
336        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        // Resolve packet with a new client to prevent caching
344        // Only use the relay, no DHT
345        // This simulates pkarr clients in a browser.
346        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}