1
//! A general interface for Tor client usage.
2
//!
3
//! To construct a client, run the [`TorClient::create_bootstrapped`] method.
4
//! Once the client is bootstrapped, you can make anonymous
5
//! connections ("streams") over the Tor network using
6
//! [`TorClient::connect`].
7

            
8
#[cfg(feature = "rpc")]
9
use {derive_deftly::Deftly, tor_rpcbase::templates::*};
10

            
11
use crate::address::{IntoTorAddr, ResolveInstructions, StreamInstructions};
12

            
13
use crate::config::{ClientAddrConfig, StreamTimeoutConfig, TorClientConfig};
14
use crate::status::BootstrapStatus;
15
use safelog::{Sensitive, sensitive};
16
use tor_async_utils::{DropNotifyWatchSender, PostageWatchSenderExt};
17
use tor_chanmgr::ChanMgrConfig;
18
use tor_circmgr::ClientDataTunnel;
19
use tor_circmgr::isolation::{Isolation, StreamIsolation};
20
use tor_circmgr::{IsolationToken, TargetPort, isolation::StreamIsolationBuilder};
21
use tor_config::MutCfg;
22
#[cfg(feature = "bridge-client")]
23
use tor_dirmgr::bridgedesc::BridgeDescMgr;
24
use tor_dirmgr::{DirMgrStore, Timeliness};
25
use tor_error::{Bug, error_report, internal};
26
use tor_guardmgr::{GuardMgr, RetireCircuits};
27
use tor_keymgr::Keystore;
28
use tor_memquota::MemoryQuotaTracker;
29
use tor_netdir::{NetDirProvider, params::NetParameters};
30
use tor_persist::StateMgr;
31
#[cfg(all(target_arch = "wasm32", target_os = "unknown"))]
32
use tor_persist::TestingStateMgr;
33
#[cfg(feature = "onion-service-service")]
34
use tor_persist::state_dir::StateDirectory;
35
use tor_proto::client::stream::{DataStream, IpVersionPreference, StreamParameters};
36
#[cfg(all(
37
    any(feature = "native-tls", feature = "rustls"),
38
    any(feature = "async-std", feature = "tokio"),
39
))]
40
use tor_rtcompat::PreferredRuntime;
41
use tor_rtcompat::{Runtime, SleepProviderExt};
42
#[cfg(feature = "onion-service-client")]
43
use {
44
    tor_config::BoolOrAuto,
45
    tor_hsclient::{HsClientConnector, HsClientDescEncKeypairSpecifier, HsClientSecretKeysBuilder},
46
    tor_hscrypto::pk::{HsClientDescEncKey, HsClientDescEncKeypair, HsClientDescEncSecretKey},
47
    tor_netdir::DirEvent,
48
};
49

            
50
#[cfg(all(feature = "onion-service-service", feature = "experimental-api"))]
51
use tor_hsservice::HsIdKeypairSpecifier;
52
#[cfg(all(feature = "onion-service-client", feature = "experimental-api"))]
53
use {tor_hscrypto::pk::HsId, tor_hscrypto::pk::HsIdKeypair, tor_keymgr::KeystoreSelector};
54

            
55
use tor_keymgr::{ArtiNativeKeystore, KeyMgr, KeyMgrBuilder, config::ArtiKeystoreKind};
56

            
57
#[cfg(feature = "ephemeral-keystore")]
58
use tor_keymgr::ArtiEphemeralKeystore;
59

            
60
#[cfg(feature = "ctor-keystore")]
61
use tor_keymgr::{CTorClientKeystore, CTorServiceKeystore};
62

            
63
use futures::StreamExt as _;
64
use futures::lock::Mutex as AsyncMutex;
65
use std::net::IpAddr;
66
use std::result::Result as StdResult;
67
use std::sync::{Arc, Mutex};
68
use tor_rtcompat::SpawnExt;
69

            
70
use crate::err::ErrorDetail;
71
use crate::{TorClientBuilder, status, util};
72
#[cfg(feature = "geoip")]
73
use tor_geoip::CountryCode;
74
use tor_rtcompat::scheduler::TaskHandle;
75
use tracing::{debug, info, instrument, warn};
76

            
77
#[cfg(not(all(target_arch = "wasm32", target_os = "unknown")))]
78
use tor_persist::FsStateMgr as UsingStateMgr;
79

            
80
// TODO wasm: This is not the right choice, but at least it compiles.
81
#[cfg(all(target_arch = "wasm32", target_os = "unknown"))]
82
use tor_persist::TestingStateMgr as UsingStateMgr;
83

            
84
/// An active client session on the Tor network.
85
///
86
/// While it's running, it will fetch directory information, build
87
/// circuits, and make connections for you.
88
///
89
/// # In the Arti RPC System
90
///
91
/// An open client on the Tor network.
92
///
93
/// A `TorClient` can be used to open anonymous connections,
94
/// and (eventually) perform other activities.
95
///
96
/// You can use an `RpcSession` as a `TorClient`, or use the `isolated_client` method
97
/// to create a new `TorClient` whose stream will not share circuits with any other Tor client.
98
///
99
/// This ObjectID for this object can be used as the target of a SOCKS stream.
100
#[cfg_attr(
101
    feature = "rpc",
102
    derive(Deftly),
103
    derive_deftly(Object),
104
    deftly(rpc(expose_outside_of_session))
105
)]
106
pub struct TorClient<R: Runtime> {
107
    /// Default isolation token for streams through this client.
108
    ///
109
    /// This is eventually used for `owner_token` in `tor-circmgr/src/usage.rs`, and is orthogonal
110
    /// to the `stream_isolation` which comes from `connect_prefs` (or a passed-in `StreamPrefs`).
111
    /// (ie, both must be the same to share a circuit).
112
    client_isolation: IsolationToken,
113
    /// Connection preferences.  Starts out as `Default`,  Inherited by our clones.
114
    connect_prefs: StreamPrefs,
115

            
116
    /// Inner structure respresenting all components shared across different
117
    /// TorClients.
118
    client: Arc<ClientShared<R>>,
119
}
120

            
121
/// Shared pieces of a `TorClient`, used to implement client functionality.
122
///
123
/// In the future, we might choose to expose this along with APIs.
124
struct ClientShared<R: Runtime> {
125
    /// Asynchronous runtime object.
126
    runtime: R,
127

            
128
    /// Inner typestate object to represent the parts of the ClientShared that may be absent
129
    /// depending on whether we are running.
130
    inner: Mutex<Inner<R>>,
131

            
132
    /// Memory quota tracker
133
    memquota: Arc<MemoryQuotaTracker>,
134

            
135
    /// A handle to this client's [`InertTorClient`].
136
    ///
137
    /// Used for accessing the key manager and other persistent state.
138
    inert_client: InertTorClient,
139

            
140
    /// Location on disk where we store persistent data containing both location and Mistrust information.
141
    ///
142
    ///
143
    /// This path is configured via `[storage]` in the config but is not used directly as a
144
    /// StateDirectory in most places. Instead, its path and Mistrust information are copied
145
    /// to subsystems like `dirmgr`, `keymgr`, and `statemgr` during `TorClient` creation.
146
    #[cfg(feature = "onion-service-service")]
147
    state_directory: StateDirectory,
148
    /// Location on disk where we store persistent data (cooked state manager).
149
    statemgr: UsingStateMgr,
150

            
151
    /// Directory manager persistent storage.
152
    dirmgr_store: DirMgrStore<R>,
153

            
154
    /// Client address configuration
155
    addrcfg: MutCfg<ClientAddrConfig>,
156
    /// Client DNS configuration
157
    timeoutcfg: MutCfg<StreamTimeoutConfig>,
158
    /// Mutex used to serialize concurrent attempts to reconfigure a TorClient.
159
    ///
160
    /// See [`TorClient::reconfigure`] for more information on its use.
161
    reconfigure_lock: Arc<Mutex<()>>,
162

            
163
    /// A stream of bootstrap messages that we can clone when a client asks for
164
    /// it.
165
    ///
166
    /// (We don't need to observe this stream ourselves, since it drops each
167
    /// unobserved status change when the next status change occurs.)
168
    status_receiver: status::BootstrapEvents,
169

            
170
    /// mutex used to prevent two tasks from trying to bootstrap at once.
171
    bootstrap_in_progress: AsyncMutex<()>,
172

            
173
    /// Sender used to update changes in our bootstrap settings.
174
    bootstrap_setting_sender: Mutex<postage::watch::Sender<BootstrapSetting>>,
175

            
176
    /// Whether or not we should call `bootstrap` before doing things that require
177
    /// bootstrapping.
178
    ///
179
    /// If this is [`BootstrapBehavior::OnDemand`], we wait for the client to bootstrap
180
    /// (launching a bootstrap if necessary) before performing any operation that needs circuits.
181
    /// If this is [`BootstrapBehavior::Manual`], we give an error if we are told to do
182
    /// something that needs circuits and we have not been told to bootstrap.
183
    should_bootstrap: BootstrapBehavior,
184

            
185
    /// Shared boolean for whether we're currently in "dormant mode" or not.
186
    //
187
    // The sent value is `Option`, so that `None` is sent when the sender, here,
188
    // is dropped,.  That shuts down the monitoring task.
189
    dormant: Mutex<DropNotifyWatchSender<Option<DormantMode>>>,
190

            
191
    /// The path resolver given to us by a [`TorClientConfig`].
192
    ///
193
    /// We must not add our own variables to it since `TorClientConfig` uses it to perform its own
194
    /// path expansions. If we added our own variables, it would introduce an inconsistency where
195
    /// paths expanded by the `TorClientConfig` would expand differently than when expanded by us.
196
    path_resolver: Arc<tor_config_path::CfgPathResolver>,
197
}
198

            
199
/// A typestate object holding the parts of the client state that we may or may not have
200
/// depending on whether we are running.
201
enum Inner<R: Runtime> {
202
    /// The client is not constructed.
203
    ///
204
    /// In this state, the client won't try to connect to the network.
205
    NotConstructed(Box<NotConstructedInner<R>>),
206

            
207
    /// The client is either bootstrapped or trying to bootstrap.
208
    Running(Arc<RunningInner<R>>),
209

            
210
    /// The client has failed in a non-recoverable way.
211
    Poisoned(Box<ErrorDetail>),
212
}
213

            
214
/// Information stored by a never-bootstrapped [`TorClient`],
215
/// used to eventually construct a [`RunningInner`] and bootstrap.
216
struct NotConstructedInner<R: Runtime> {
217
    /// The client's configuration.
218
    config: TorClientConfig,
219

            
220
    /// A receiver to give to various tasks that want to monitor our dormant status.
221
    dormant_recv: postage::watch::Receiver<Option<DormantMode>>,
222

            
223
    /// A sender used to produce updates about our bootstrapping status.
224
    ///
225
    /// NOTE: The fact that this type is not Clone is the only reason
226
    /// that [`RunningInner::new`] needs to take NotConstructedInner by value.
227
    /// With some redesign we could simplify this, and do away with [`Inner::Poisoned`].
228
    status_sender: postage::watch::Sender<BootstrapStatus>,
229

            
230
    /// A receiver used to inform the bootstrap status processor about changes in our settings.
231
    bootstrap_setting_receiver: postage::watch::Receiver<BootstrapSetting>,
232

            
233
    /// A (possibly user-provided) builder used to construct our NetDirProvider.
234
    dirmgr_builder: Arc<dyn crate::builder::DirProviderBuilder<R>>,
235

            
236
    /// A (possibly user-provided) set of in-process extensions for our NetDirProvider.
237
    dirmgr_extensions: tor_dirmgr::config::DirMgrExtensions,
238
}
239

            
240
/// Data structures for a "running" client.
241
///
242
/// A running client is one that is either bootstrapped, or potentially trying to bootstrap.
243
///
244
/// All structures that potentially interact with the network belong here.
245
///
246
/// We defer the creation of this structure and its members until bootstrap time,
247
/// to make sure that before we are bootstrapping, nothing will try to connect to the network
248
/// or launch expensive background tasks.
249
struct RunningInner<R: Runtime> {
250
    /// Channel manager, used by circuits etc.,
251
    ///
252
    /// Used directly by client only for reconfiguration.
253
    chanmgr: Arc<tor_chanmgr::ChanMgr<R>>,
254
    /// Circuit manager for keeping our circuits up to date and building
255
    /// them on-demand.
256
    circmgr: Arc<tor_circmgr::CircMgr<R>>,
257
    /// Directory manager for keeping our directory material up to date.
258
    dirmgr: Arc<dyn tor_dirmgr::DirProvider>,
259
    /// Bridge descriptor manager
260
    ///
261
    /// None until we have bootstrapped.
262
    ///
263
    /// Lock hierarchy: don't acquire this before dormant
264
    //
265
    // TODO: after or as part of https://gitlab.torproject.org/tpo/core/arti/-/issues/634
266
    // this can be   bridge_desc_mgr: BridgeDescMgr<R>>
267
    // since BridgeDescMgr is Clone and all its methods take `&self` (it has a lock inside)
268
    // Or maybe BridgeDescMgr should not be Clone, since we want to make Weaks of it,
269
    // which we can't do when the Arc is inside.
270
    #[cfg(feature = "bridge-client")]
271
    bridge_desc_mgr: Arc<Mutex<Option<Arc<BridgeDescMgr<R>>>>>,
272
    /// Pluggable transport manager.
273
    #[cfg(feature = "pt-client")]
274
    pt_mgr: Arc<tor_ptmgr::PtMgr<R>>,
275
    /// HS client connector
276
    #[cfg(feature = "onion-service-client")]
277
    hsclient: HsClientConnector<R>,
278
    /// Circuit pool for providing onion services with circuits.
279
    #[cfg(any(feature = "onion-service-client", feature = "onion-service-service"))]
280
    hs_circ_pool: Arc<tor_circmgr::hspool::HsCircPool<R>>,
281
    /// Guard manager
282
    #[cfg_attr(not(feature = "bridge-client"), allow(dead_code))]
283
    guardmgr: GuardMgr<R>,
284
}
285

            
286
/// A Tor client that is not runnable.
287
///
288
/// Can be used to access the state that would be used by a running [`TorClient`].
289
///
290
/// An `InertTorClient` never connects to the network.
291
#[derive(Clone)]
292
pub struct InertTorClient {
293
    /// The key manager.
294
    ///
295
    /// This is used for retrieving private keys, certificates, and other sensitive data (for
296
    /// example, for retrieving the keys necessary for connecting to hidden services that are
297
    /// running in restricted discovery mode).
298
    ///
299
    /// If this crate is compiled _with_ the `keymgr` feature, [`TorClient`] will use a functional
300
    /// key manager implementation.
301
    ///
302
    /// If this crate is compiled _without_ the `keymgr` feature, then [`TorClient`] will use a
303
    /// no-op key manager implementation instead.
304
    ///
305
    /// See the [`KeyMgr`] documentation for more details.
306
    keymgr: Option<Arc<KeyMgr>>,
307
}
308

            
309
impl InertTorClient {
310
    /// Create an `InertTorClient` from a `TorClientConfig`.
311
1746
    pub(crate) fn new(config: &TorClientConfig) -> StdResult<Self, ErrorDetail> {
312
1746
        let keymgr = Self::create_keymgr(config)?;
313

            
314
1746
        Ok(Self { keymgr })
315
1746
    }
316

            
317
    /// Create a [`KeyMgr`] using the specified configuration.
318
    ///
319
    /// Returns `Ok(None)` if keystore use is disabled.
320
1746
    fn create_keymgr(config: &TorClientConfig) -> StdResult<Option<Arc<KeyMgr>>, ErrorDetail> {
321
1746
        let keystore = config.storage.keystore();
322
1746
        let permissions = config.storage.permissions();
323
1746
        let primary_store: Box<dyn Keystore> = match keystore.primary_kind() {
324
            Some(ArtiKeystoreKind::Native) => {
325
1746
                let (state_dir, _mistrust) = config.state_dir()?;
326
1746
                let key_store_dir = state_dir.join("keystore");
327

            
328
1746
                let native_store =
329
1746
                    ArtiNativeKeystore::from_path_and_mistrust(&key_store_dir, permissions)?;
330
                // Should only log fs paths at debug level or lower,
331
                // unless they're part of a diagnostic message.
332
1746
                debug!("Using keystore from {key_store_dir:?}");
333

            
334
1746
                Box::new(native_store)
335
            }
336
            #[cfg(feature = "ephemeral-keystore")]
337
            Some(ArtiKeystoreKind::Ephemeral) => {
338
                // TODO: make the keystore ID somehow configurable
339
                let ephemeral_store: ArtiEphemeralKeystore =
340
                    ArtiEphemeralKeystore::new("ephemeral".to_string());
341
                Box::new(ephemeral_store)
342
            }
343
            None => {
344
                info!("Running without a keystore");
345
                return Ok(None);
346
            }
347
            ty => return Err(internal!("unrecognized keystore type {ty:?}").into()),
348
        };
349

            
350
1746
        let mut builder = KeyMgrBuilder::default().primary_store(primary_store);
351

            
352
        #[cfg(feature = "ctor-keystore")]
353
1746
        for config in config.storage.keystore().ctor_svc_stores() {
354
476
            let store: Box<dyn Keystore> = Box::new(CTorServiceKeystore::from_path_and_mistrust(
355
476
                config.path(),
356
476
                permissions,
357
476
                config.id().clone(),
358
                // TODO: these nicknames should be cross-checked with configured
359
                // svc nicknames as part of config validation!!!
360
476
                config.nickname().clone(),
361
            )?);
362

            
363
476
            builder.secondary_stores().push(store);
364
        }
365

            
366
        #[cfg(feature = "ctor-keystore")]
367
1746
        for config in config.storage.keystore().ctor_client_stores() {
368
204
            let store: Box<dyn Keystore> = Box::new(CTorClientKeystore::from_path_and_mistrust(
369
204
                config.path(),
370
204
                permissions,
371
204
                config.id().clone(),
372
            )?);
373

            
374
204
            builder.secondary_stores().push(store);
375
        }
376

            
377
1746
        let keymgr = builder
378
1746
            .build()
379
1746
            .map_err(|_| internal!("failed to build keymgr"))?;
380
1746
        Ok(Some(Arc::new(keymgr)))
381
1746
    }
382

            
383
    /// Generate a service discovery keypair for connecting to a hidden service running in
384
    /// "restricted discovery" mode.
385
    ///
386
    /// See [`TorClient::generate_service_discovery_key`].
387
    //
388
    // TODO: decide whether this should use get_or_generate before making it
389
    // non-experimental
390
    #[cfg(all(
391
        feature = "onion-service-client",
392
        feature = "experimental-api",
393
        feature = "keymgr"
394
    ))]
395
    #[cfg_attr(
396
        docsrs,
397
        doc(cfg(all(
398
            feature = "onion-service-client",
399
            feature = "experimental-api",
400
            feature = "keymgr"
401
        )))
402
    )]
403
102
    pub fn generate_service_discovery_key(
404
102
        &self,
405
102
        selector: KeystoreSelector,
406
102
        hsid: HsId,
407
102
    ) -> crate::Result<HsClientDescEncKey> {
408
102
        let mut rng = tor_llcrypto::rng::CautiousRng;
409
102
        let spec = HsClientDescEncKeypairSpecifier::new(hsid);
410
102
        let key = self
411
102
            .keymgr
412
102
            .as_ref()
413
102
            .ok_or(ErrorDetail::KeystoreRequired {
414
102
                action: "generate client service discovery key",
415
102
            })?
416
102
            .generate::<HsClientDescEncKeypair>(
417
102
                &spec, selector, &mut rng, false, /* overwrite */
418
            )?;
419

            
420
102
        Ok(key.public().clone())
421
102
    }
422

            
423
    /// Rotate the service discovery keypair for connecting to a hidden service running in
424
    /// "restricted discovery" mode.
425
    ///
426
    /// See [`TorClient::rotate_service_discovery_key`].
427
    #[cfg(all(
428
        feature = "onion-service-client",
429
        feature = "experimental-api",
430
        feature = "keymgr"
431
    ))]
432
34
    pub fn rotate_service_discovery_key(
433
34
        &self,
434
34
        selector: KeystoreSelector,
435
34
        hsid: HsId,
436
34
    ) -> crate::Result<HsClientDescEncKey> {
437
34
        let mut rng = tor_llcrypto::rng::CautiousRng;
438
34
        let spec = HsClientDescEncKeypairSpecifier::new(hsid);
439
34
        let key = self
440
34
            .keymgr
441
34
            .as_ref()
442
34
            .ok_or(ErrorDetail::KeystoreRequired {
443
34
                action: "rotate client service discovery key",
444
34
            })?
445
34
            .generate::<HsClientDescEncKeypair>(
446
34
                &spec, selector, &mut rng, true, /* overwrite */
447
            )?;
448

            
449
34
        Ok(key.public().clone())
450
34
    }
451

            
452
    /// Insert a service discovery secret key for connecting to a hidden service running in
453
    /// "restricted discovery" mode
454
    ///
455
    /// See [`TorClient::insert_service_discovery_key`].
456
    #[cfg(all(
457
        feature = "onion-service-client",
458
        feature = "experimental-api",
459
        feature = "keymgr"
460
    ))]
461
    #[cfg_attr(
462
        docsrs,
463
        doc(cfg(all(
464
            feature = "onion-service-client",
465
            feature = "experimental-api",
466
            feature = "keymgr"
467
        )))
468
    )]
469
    pub fn insert_service_discovery_key(
470
        &self,
471
        selector: KeystoreSelector,
472
        hsid: HsId,
473
        hs_client_desc_enc_secret_key: HsClientDescEncSecretKey,
474
    ) -> crate::Result<HsClientDescEncKey> {
475
        let spec = HsClientDescEncKeypairSpecifier::new(hsid);
476
        let client_desc_enc_key = HsClientDescEncKey::from(&hs_client_desc_enc_secret_key);
477
        let client_desc_enc_keypair =
478
            HsClientDescEncKeypair::new(client_desc_enc_key.clone(), hs_client_desc_enc_secret_key);
479
        let _key = self
480
            .keymgr
481
            .as_ref()
482
            .ok_or(ErrorDetail::KeystoreRequired {
483
                action: "insert client service discovery key",
484
            })?
485
            .insert::<HsClientDescEncKeypair>(client_desc_enc_keypair, &spec, selector, false)?;
486
        Ok(client_desc_enc_key)
487
    }
488

            
489
    /// Return the service discovery public key for the service with the specified `hsid`.
490
    ///
491
    /// See [`TorClient::get_service_discovery_key`].
492
    #[cfg(all(feature = "onion-service-client", feature = "experimental-api"))]
493
    #[cfg_attr(
494
        docsrs,
495
        doc(cfg(all(feature = "onion-service-client", feature = "experimental-api")))
496
    )]
497
476
    pub fn get_service_discovery_key(
498
476
        &self,
499
476
        hsid: HsId,
500
476
    ) -> crate::Result<Option<HsClientDescEncKey>> {
501
476
        let spec = HsClientDescEncKeypairSpecifier::new(hsid);
502
476
        let key = self
503
476
            .keymgr
504
476
            .as_ref()
505
476
            .ok_or(ErrorDetail::KeystoreRequired {
506
476
                action: "get client service discovery key",
507
476
            })?
508
476
            .get::<HsClientDescEncKeypair>(&spec)?
509
483
            .map(|key| key.public().clone());
510

            
511
476
        Ok(key)
512
476
    }
513

            
514
    /// Removes the service discovery keypair for the service with the specified `hsid`.
515
    ///
516
    /// See [`TorClient::remove_service_discovery_key`].
517
    #[cfg(all(
518
        feature = "onion-service-client",
519
        feature = "experimental-api",
520
        feature = "keymgr"
521
    ))]
522
    #[cfg_attr(
523
        docsrs,
524
        doc(cfg(all(
525
            feature = "onion-service-client",
526
            feature = "experimental-api",
527
            feature = "keymgr"
528
        )))
529
    )]
530
34
    pub fn remove_service_discovery_key(
531
34
        &self,
532
34
        selector: KeystoreSelector,
533
34
        hsid: HsId,
534
34
    ) -> crate::Result<Option<()>> {
535
34
        let spec = HsClientDescEncKeypairSpecifier::new(hsid);
536
34
        let result = self
537
34
            .keymgr
538
34
            .as_ref()
539
34
            .ok_or(ErrorDetail::KeystoreRequired {
540
34
                action: "remove client service discovery key",
541
34
            })?
542
34
            .remove::<HsClientDescEncKeypair>(&spec, selector)?;
543
34
        match result {
544
34
            Some(_) => Ok(Some(())),
545
            None => Ok(None),
546
        }
547
34
    }
548

            
549
    /// Getter for keymgr.
550
    #[cfg(feature = "onion-service-cli-extra")]
551
748
    pub fn keymgr(&self) -> crate::Result<&KeyMgr> {
552
748
        Ok(self.keymgr.as_ref().ok_or(ErrorDetail::KeystoreRequired {
553
748
            action: "get key manager handle",
554
748
        })?)
555
748
    }
556

            
557
    /// Create (but do not launch) a new
558
    /// [`OnionService`](tor_hsservice::OnionService)
559
    /// using the given configuration.
560
    ///
561
    /// See [`TorClient::create_onion_service`].
562
    #[cfg(feature = "onion-service-service")]
563
    #[instrument(skip_all, level = "trace")]
564
272
    pub fn create_onion_service(
565
272
        &self,
566
272
        config: &TorClientConfig,
567
272
        svc_config: tor_hsservice::OnionServiceConfig,
568
272
    ) -> crate::Result<tor_hsservice::OnionService> {
569
272
        let keymgr = self.keymgr.as_ref().ok_or(ErrorDetail::KeystoreRequired {
570
272
            action: "create onion service",
571
272
        })?;
572

            
573
272
        let (state_dir, mistrust) = config.state_dir()?;
574
272
        let state_dir =
575
272
            self::StateDirectory::new(state_dir, mistrust).map_err(ErrorDetail::StateAccess)?;
576

            
577
272
        Ok(tor_hsservice::OnionService::builder()
578
272
            .config(svc_config)
579
272
            .keymgr(keymgr.clone())
580
272
            .state_dir(state_dir)
581
272
            .build()
582
272
            .map_err(ErrorDetail::OnionServiceSetup)?)
583
272
    }
584
}
585

            
586
/// Preferences for whether a [`TorClient`] should bootstrap on its own or not.
587
#[derive(Debug, Default, Copy, Clone, PartialEq, Eq)]
588
#[non_exhaustive]
589
pub enum BootstrapBehavior {
590
    /// Bootstrap the client automatically when requests are made that require the client to be
591
    /// bootstrapped.
592
    #[default]
593
    OnDemand,
594
    /// Make no attempts to automatically bootstrap. [`TorClient::bootstrap`] must be manually
595
    /// invoked in order for the [`TorClient`] to become useful.
596
    ///
597
    /// Attempts to use the client (e.g. by creating connections or resolving hosts over the Tor
598
    /// network) before calling [`bootstrap`](TorClient::bootstrap) will fail, and
599
    /// return an error that has kind [`ErrorKind::BootstrapRequired`](crate::ErrorKind::BootstrapRequired).
600
    Manual,
601
}
602

            
603
/// A representation of whether a [`TorClient`] is allowed to bootstrap, and whether it
604
/// has begun to do so.
605
#[derive(Debug, Clone, Copy)]
606
pub(crate) struct BootstrapSetting {
607
    /// The configured [`BootstrapBehavior`] for the `TorClient`.
608
    behavior: BootstrapBehavior,
609

            
610
    /// If true, we have a [`RunningInner`] in the `TorClient`,
611
    /// indicating that we are trying to bootstrap it.
612
    running_inner_is_present: bool,
613
}
614

            
615
impl Default for BootstrapSetting {
616
182
    fn default() -> Self {
617
182
        Self {
618
182
            behavior: BootstrapBehavior::Manual,
619
182
            running_inner_is_present: false,
620
182
        }
621
182
    }
622
}
623

            
624
impl BootstrapSetting {
625
    /// Return true if this [`BootstrapSetting`]
626
    /// indicates that the client is not trying to bootstrap,
627
    /// and will not try until it is told explicitly to do so.
628
    pub(crate) fn blocked(&self) -> bool {
629
        use BootstrapBehavior::*;
630
        match (self.behavior, self.running_inner_is_present) {
631
            (OnDemand, _) => false,
632
            (Manual, true) => false,
633
            (Manual, false) => true,
634
        }
635
    }
636
}
637

            
638
/// What level of sleep to put a Tor client into.
639
#[derive(Debug, Default, Copy, Clone, PartialEq, Eq)]
640
#[non_exhaustive]
641
pub enum DormantMode {
642
    /// The client functions as normal, and background tasks run periodically.
643
    #[default]
644
    Normal,
645
    /// Background tasks are suspended, conserving CPU usage. Attempts to use the client will
646
    /// wake it back up again.
647
    Soft,
648
}
649

            
650
/// Preferences for how to route a stream over the Tor network.
651
#[derive(Debug, Default, Clone)]
652
pub struct StreamPrefs {
653
    /// What kind of IPv6/IPv4 we'd prefer, and how strongly.
654
    ip_ver_pref: IpVersionPreference,
655
    /// How should we isolate connection(s)?
656
    isolation: StreamIsolationPreference,
657
    /// Whether to return the stream optimistically.
658
    optimistic_stream: bool,
659
    // TODO GEOIP Ideally this would be unconditional, with CountryCode maybe being Void
660
    // This probably applies in many other places, so probably:   git grep 'cfg.*geoip'
661
    // and consider each one with a view to making it unconditional.  Background:
662
    //   https://gitlab.torproject.org/tpo/core/arti/-/merge_requests/1537#note_2935256
663
    //   https://gitlab.torproject.org/tpo/core/arti/-/merge_requests/1537#note_2942214
664
    #[cfg(feature = "geoip")]
665
    /// A country to restrict the exit relay's location to.
666
    country_code: Option<CountryCode>,
667
    /// Whether to try to make connections to onion services.
668
    ///
669
    /// `Auto` means to use the client configuration.
670
    #[cfg(feature = "onion-service-client")]
671
    pub(crate) connect_to_onion_services: BoolOrAuto,
672
}
673

            
674
/// Record of how we are isolating connections
675
#[derive(Debug, Default, Clone)]
676
enum StreamIsolationPreference {
677
    /// No additional isolation
678
    #[default]
679
    None,
680
    /// Isolation parameter to use for connections
681
    Explicit(Box<dyn Isolation>),
682
    /// Isolate every connection!
683
    EveryStream,
684
}
685

            
686
impl From<DormantMode> for tor_chanmgr::Dormancy {
687
    fn from(dormant: DormantMode) -> tor_chanmgr::Dormancy {
688
        match dormant {
689
            DormantMode::Normal => tor_chanmgr::Dormancy::Active,
690
            DormantMode::Soft => tor_chanmgr::Dormancy::Dormant,
691
        }
692
    }
693
}
694
#[cfg(feature = "bridge-client")]
695
impl From<DormantMode> for tor_dirmgr::bridgedesc::Dormancy {
696
    fn from(dormant: DormantMode) -> tor_dirmgr::bridgedesc::Dormancy {
697
        match dormant {
698
            DormantMode::Normal => tor_dirmgr::bridgedesc::Dormancy::Active,
699
            DormantMode::Soft => tor_dirmgr::bridgedesc::Dormancy::Dormant,
700
        }
701
    }
702
}
703

            
704
impl StreamPrefs {
705
    /// Construct a new StreamPrefs.
706
18
    pub fn new() -> Self {
707
18
        Self::default()
708
18
    }
709

            
710
    /// Indicate that a stream may be made over IPv4 or IPv6, but that
711
    /// we'd prefer IPv6.
712
2
    pub fn ipv6_preferred(&mut self) -> &mut Self {
713
2
        self.ip_ver_pref = IpVersionPreference::Ipv6Preferred;
714
2
        self
715
2
    }
716

            
717
    /// Indicate that a stream may only be made over IPv6.
718
    ///
719
    /// When this option is set, we will only pick exit relays that
720
    /// support IPv6, and we will tell them to only give us IPv6
721
    /// connections.
722
2
    pub fn ipv6_only(&mut self) -> &mut Self {
723
2
        self.ip_ver_pref = IpVersionPreference::Ipv6Only;
724
2
        self
725
2
    }
726

            
727
    /// Indicate that a stream may be made over IPv4 or IPv6, but that
728
    /// we'd prefer IPv4.
729
    ///
730
    /// This is the default.
731
2
    pub fn ipv4_preferred(&mut self) -> &mut Self {
732
2
        self.ip_ver_pref = IpVersionPreference::Ipv4Preferred;
733
2
        self
734
2
    }
735

            
736
    /// Indicate that a stream may only be made over IPv4.
737
    ///
738
    /// When this option is set, we will only pick exit relays that
739
    /// support IPv4, and we will tell them to only give us IPv4
740
    /// connections.
741
2
    pub fn ipv4_only(&mut self) -> &mut Self {
742
2
        self.ip_ver_pref = IpVersionPreference::Ipv4Only;
743
2
        self
744
2
    }
745

            
746
    /// Indicate that a stream should appear to come from the given country.
747
    ///
748
    /// When this option is set, we will only pick exit relays that
749
    /// have an IP address that matches the country in our GeoIP database.
750
    #[cfg(feature = "geoip")]
751
    pub fn exit_country(&mut self, country_code: CountryCode) -> &mut Self {
752
        self.country_code = Some(country_code);
753
        self
754
    }
755

            
756
    /// Indicate that we don't care which country a stream appears to come from.
757
    ///
758
    /// This is available even in the case where GeoIP support is compiled out,
759
    /// to make things easier.
760
    pub fn any_exit_country(&mut self) -> &mut Self {
761
        #[cfg(feature = "geoip")]
762
        {
763
            self.country_code = None;
764
        }
765
        self
766
    }
767

            
768
    /// Indicate that the stream should be opened "optimistically".
769
    ///
770
    /// By default, streams are not "optimistic". When you call
771
    /// [`TorClient::connect()`], it won't give you a stream until the
772
    /// exit node has confirmed that it has successfully opened a
773
    /// connection to your target address.  It's safer to wait in this
774
    /// way, but it is slower: it takes an entire round trip to get
775
    /// your confirmation.
776
    ///
777
    /// If a stream _is_ configured to be "optimistic", on the other
778
    /// hand, then `TorClient::connect()` will return the stream
779
    /// immediately, without waiting for an answer from the exit.  You
780
    /// can start sending data on the stream right away, though of
781
    /// course this data will be lost if the connection is not
782
    /// actually successful.
783
2
    pub fn optimistic(&mut self) -> &mut Self {
784
2
        self.optimistic_stream = true;
785
2
        self
786
2
    }
787

            
788
    /// Return true if this stream has been configured as "optimistic".
789
    ///
790
    /// See [`StreamPrefs::optimistic`] for more info.
791
    pub fn is_optimistic(&self) -> bool {
792
        self.optimistic_stream
793
    }
794

            
795
    /// Indicate whether connection to a hidden service (`.onion` service) should be allowed
796
    ///
797
    /// If `Explicit(false)`, attempts to connect to Onion Services will be forced to fail with
798
    /// an error of kind [`InvalidStreamTarget`](crate::ErrorKind::InvalidStreamTarget).
799
    ///
800
    /// If `Explicit(true)`, Onion Service connections are enabled.
801
    ///
802
    /// If `Auto`, the behaviour depends on the `address_filter.allow_onion_addrs`
803
    /// configuration option, which is in turn enabled by default.
804
    #[cfg(feature = "onion-service-client")]
805
42
    pub fn connect_to_onion_services(
806
42
        &mut self,
807
42
        connect_to_onion_services: BoolOrAuto,
808
42
    ) -> &mut Self {
809
42
        self.connect_to_onion_services = connect_to_onion_services;
810
42
        self
811
42
    }
812
    /// Return a TargetPort to describe what kind of exit policy our
813
    /// target circuit needs to support.
814
4
    fn wrap_target_port(&self, port: u16) -> TargetPort {
815
4
        match self.ip_ver_pref {
816
            IpVersionPreference::Ipv6Only => TargetPort::ipv6(port),
817
4
            _ => TargetPort::ipv4(port),
818
        }
819
4
    }
820

            
821
    /// Return a new StreamParameters based on this configuration.
822
4
    fn stream_parameters(&self) -> StreamParameters {
823
4
        let mut params = StreamParameters::default();
824
4
        params
825
4
            .ip_version(self.ip_ver_pref)
826
4
            .optimistic(self.optimistic_stream);
827
4
        params
828
4
    }
829

            
830
    /// Indicate that connections with these preferences should have their own isolation group
831
    ///
832
    /// This is a convenience method which creates a fresh [`IsolationToken`]
833
    /// and sets it for these preferences.
834
    ///
835
    /// This connection preference is orthogonal to isolation established by
836
    /// [`TorClient::isolated_client`].  Connections made with an `isolated_client`
837
    ///  will not share circuits with the original client, even if the same
838
    /// `isolation` is specified via the `ConnectionPrefs` in force.
839
2
    pub fn new_isolation_group(&mut self) -> &mut Self {
840
2
        self.isolation = StreamIsolationPreference::Explicit(Box::new(IsolationToken::new()));
841
2
        self
842
2
    }
843

            
844
    /// Indicate which other connections might use the same circuit
845
    /// as this one.
846
    ///
847
    /// By default all connections made on a `TorClient` may share connections.
848
    /// Connections made with a particular `isolation` may share circuits with each other.
849
    ///
850
    /// This connection preference is orthogonal to isolation established by
851
    /// [`TorClient::isolated_client`].  Connections made with an `isolated_client`
852
    /// will not share circuits with the original client, even if the same
853
    /// `isolation` is specified via the `ConnectionPrefs` in force.
854
2
    pub fn set_isolation<T>(&mut self, isolation: T) -> &mut Self
855
2
    where
856
2
        T: Into<Box<dyn Isolation>>,
857
    {
858
2
        self.isolation = StreamIsolationPreference::Explicit(isolation.into());
859
2
        self
860
2
    }
861

            
862
    /// Indicate that no connection should share a circuit with any other.
863
    ///
864
    /// **Use with care:** This is likely to have poor performance, and imposes a much greater load
865
    /// on the Tor network.  Use this option only to make small numbers of connections each of
866
    /// which needs to be isolated from all other connections.
867
    ///
868
    /// (Don't just use this as a "get more privacy!!" method: the circuits
869
    /// that it put connections on will have no more privacy than any other
870
    /// circuits.  The only benefit is that these circuits will not be shared
871
    /// by multiple streams.)
872
    ///
873
    /// This can be undone by calling `set_isolation` or `new_isolation_group` on these
874
    /// preferences.
875
2
    pub fn isolate_every_stream(&mut self) -> &mut Self {
876
2
        self.isolation = StreamIsolationPreference::EveryStream;
877
2
        self
878
2
    }
879

            
880
    /// Return an [`Isolation`] which separates according to these `StreamPrefs` (only)
881
    ///
882
    /// This describes which connections or operations might use
883
    /// the same circuit(s) as this one.
884
    ///
885
    /// Since this doesn't have access to the `TorClient`,
886
    /// it doesn't separate streams which ought to be separated because of
887
    /// the way their `TorClient`s are isolated.
888
    /// For that, use [`TorClient::isolation`].
889
    fn prefs_isolation(&self) -> Option<Box<dyn Isolation>> {
890
        use StreamIsolationPreference as SIP;
891
        match self.isolation {
892
            SIP::None => None,
893
            SIP::Explicit(ref ig) => Some(ig.clone()),
894
            SIP::EveryStream => Some(Box::new(IsolationToken::new())),
895
        }
896
    }
897

            
898
    // TODO: Add some way to be IPFlexible, and require exit to support both.
899
}
900

            
901
#[cfg(all(
902
    any(feature = "native-tls", feature = "rustls"),
903
    any(feature = "async-std", feature = "tokio")
904
))]
905
impl TorClient<PreferredRuntime> {
906
    /// Bootstrap a connection to the Tor network, using the provided `config`.
907
    ///
908
    /// Returns a client once there is enough directory material to
909
    /// connect safely over the Tor network.
910
    ///
911
    /// Consider using [`TorClient::builder`] for more fine-grained control.
912
    ///
913
    /// # Panics
914
    ///
915
    /// If Tokio is being used (the default), panics if created outside the context of a currently
916
    /// running Tokio runtime. See the documentation for [`PreferredRuntime::current`] for
917
    /// more information.
918
    ///
919
    /// If using `async-std`, either take care to ensure Arti is not compiled with Tokio support,
920
    /// or manually create an `async-std` runtime using [`tor_rtcompat`] and use it with
921
    /// [`TorClient::with_runtime`].
922
    ///
923
    /// # Do not fork
924
    ///
925
    /// The process [**may not fork**](tor_rtcompat#do-not-fork)
926
    /// (except, very carefully, before exec)
927
    /// after calling this function, because it creates a [`PreferredRuntime`].
928
    pub async fn create_bootstrapped(config: TorClientConfig) -> crate::Result<Arc<Self>> {
929
        let runtime = PreferredRuntime::current()
930
            .expect("TorClient could not get an asynchronous runtime; are you running in the right context?");
931

            
932
        Self::with_runtime(runtime)
933
            .config(config)
934
            .create_bootstrapped()
935
            .await
936
    }
937

            
938
    /// Return a new builder for creating TorClient objects.
939
    ///
940
    /// If you want to make a [`TorClient`] synchronously, this is what you want; call
941
    /// `TorClientBuilder::create_unbootstrapped` on the returned builder.
942
    ///
943
    /// # Panics
944
    ///
945
    /// If Tokio is being used (the default), panics if created outside the context of a currently
946
    /// running Tokio runtime. See the documentation for `tokio::runtime::Handle::current` for
947
    /// more information.
948
    ///
949
    /// If using `async-std`, either take care to ensure Arti is not compiled with Tokio support,
950
    /// or manually create an `async-std` runtime using [`tor_rtcompat`] and use it with
951
    /// [`TorClient::with_runtime`].
952
    ///
953
    /// # Do not fork
954
    ///
955
    /// The process [**may not fork**](tor_rtcompat#do-not-fork)
956
    /// (except, very carefully, before exec)
957
    /// after calling this function, because it creates a [`PreferredRuntime`].
958
    pub fn builder() -> TorClientBuilder<PreferredRuntime> {
959
        let runtime = PreferredRuntime::current()
960
            .expect("TorClient could not get an asynchronous runtime; are you running in the right context?");
961

            
962
        TorClientBuilder::new(runtime)
963
    }
964
}
965

            
966
impl<R: Runtime> TorClient<R> {
967
    /// Return a new builder for creating TorClient objects, with a custom provided [`Runtime`].
968
    ///
969
    /// See the [`tor_rtcompat`] crate for more information on custom runtimes.
970
250
    pub fn with_runtime(runtime: R) -> TorClientBuilder<R> {
971
250
        TorClientBuilder::new(runtime)
972
250
    }
973

            
974
    /// Implementation of `create_unbootstrapped`, split out in order to avoid manually specifying
975
    /// double error conversions.
976
    #[instrument(skip_all, level = "trace")]
977
22
    pub(crate) fn create_impl(
978
22
        runtime: R,
979
22
        config: &TorClientConfig,
980
22
        autobootstrap: BootstrapBehavior,
981
22
        dirmgr_builder: Arc<dyn crate::builder::DirProviderBuilder<R>>,
982
22
        dirmgr_extensions: tor_dirmgr::config::DirMgrExtensions,
983
22
    ) -> StdResult<Arc<Self>, ErrorDetail> {
984
22
        if crate::util::running_as_setuid() {
985
            return Err(tor_error::bad_api_usage!(
986
                "Arti does not support running in a setuid or setgid context."
987
            )
988
            .into());
989
22
        }
990

            
991
22
        let memquota = MemoryQuotaTracker::new(&runtime, config.system.memory.clone())?;
992

            
993
22
        let path_resolver = Arc::new(config.path_resolver.clone());
994

            
995
22
        let (state_dir, mistrust) = config.state_dir()?;
996
        #[cfg(feature = "onion-service-service")]
997
22
        let state_directory =
998
22
            StateDirectory::new(&state_dir, mistrust).map_err(ErrorDetail::StateAccess)?;
999

            
22
        let dormant = DormantMode::Normal;
22
        let statemgr = Self::statemgr_from_config(config)?;
        // Try to take state ownership early, so we'll know if we have it.
        // Note that this `try_lock()` may return `Ok` even if we can't acquire the lock.
        // (At this point we don't yet care if we have it.)
22
        let _ignore_status = statemgr.try_lock().map_err(ErrorDetail::StateMgrSetup)?;
22
        let addr_cfg = config.address_filter.clone();
22
        let bootstrap_setting = BootstrapSetting {
22
            behavior: autobootstrap,
22
            running_inner_is_present: false,
22
        };
22
        let (bootstrap_setting_sender, bootstrap_setting_receiver) =
22
            postage::watch::channel_with(bootstrap_setting);
22
        let bootstrap_setting_sender = Mutex::new(bootstrap_setting_sender);
22
        let (status_sender, status_receiver) =
22
            postage::watch::channel_with(BootstrapStatus::from_setting(bootstrap_setting));
22
        let status_receiver = status::BootstrapEvents {
22
            inner: status_receiver,
22
        };
22
        let timeout_cfg = config.stream_timeouts.clone();
22
        let (dormant_send, dormant_recv) = postage::watch::channel_with(Some(dormant));
22
        let dormant_send = DropNotifyWatchSender::new(dormant_send);
22
        let client_isolation = IsolationToken::new();
22
        let inert_client = InertTorClient::new(config)?;
22
        let dirmgr_store = DirMgrStore::new(&config.dir_mgr_config()?, runtime.clone(), false)
22
            .map_err(ErrorDetail::DirMgrSetup)?;
22
        let inner = Box::new(NotConstructedInner {
22
            config: config.clone(),
22
            dormant_recv,
22
            status_sender,
22
            bootstrap_setting_receiver,
22
            dirmgr_builder,
22
            dirmgr_extensions,
22
        });
22
        let inner = Mutex::new(Inner::NotConstructed(inner));
22
        let client = Arc::new(ClientShared {
22
            runtime,
22
            inner,
22
            memquota,
22
            inert_client,
22
            statemgr,
22
            dirmgr_store,
22
            addrcfg: addr_cfg.into(),
22
            timeoutcfg: timeout_cfg.into(),
22
            reconfigure_lock: Arc::new(Mutex::new(())),
22
            status_receiver,
22
            bootstrap_in_progress: AsyncMutex::new(()),
22
            bootstrap_setting_sender,
22
            should_bootstrap: autobootstrap,
22
            dormant: Mutex::new(dormant_send),
22
            #[cfg(feature = "onion-service-service")]
22
            state_directory,
22
            path_resolver,
22
        });
22
        Ok(Arc::new(TorClient {
22
            client_isolation,
22
            connect_prefs: Default::default(),
22
            client,
22
        }))
22
    }
    /// Construct a state manager from the client configuration.
22
    fn statemgr_from_config(config: &TorClientConfig) -> Result<UsingStateMgr, ErrorDetail> {
        #[cfg(not(all(target_arch = "wasm32", target_os = "unknown")))]
        {
            use tor_persist::FsStateMgr;
22
            let (state_dir, mistrust) = config.state_dir()?;
22
            FsStateMgr::from_path_and_mistrust(state_dir, mistrust)
22
                .map_err(ErrorDetail::StateMgrSetup)
        }
        #[cfg(all(target_arch = "wasm32", target_os = "unknown"))]
        {
            unimplemented!()
        }
22
    }
    /// Bootstrap a connection to the Tor network, with a client created by `create_unbootstrapped`.
    ///
    /// Returns once there is enough directory material to connect safely over the Tor network.
    /// If the client has already been bootstrapped, returns immediately with
    /// success. If a bootstrap is in progress, waits for it to finish, then retries it if it
    /// failed (returning success if it succeeded).
    ///
    /// Bootstrap progress can be tracked by listening to the event receiver returned by
    /// [`bootstrap_events`](TorClient::bootstrap_events).
    ///
    /// # Failures
    ///
    /// If the bootstrapping process fails, returns an error. This function can safely be called
    /// again later to attempt to bootstrap another time.
    #[instrument(skip_all, level = "trace")]
    pub async fn bootstrap(&self) -> crate::Result<()> {
        self.client
            .bootstrap_inner()
            .await
            .map_err(ErrorDetail::into)
    }
}
impl<R: Runtime> NotConstructedInner<R> {
    /// Replace the configuration for this unconstructed client.
    ///
    /// Since most of the client's internals are not yet constructed,
    /// we can still replace nearly all of the items.
8
    fn reconfigure(
8
        &mut self,
8
        new_config: &TorClientConfig,
8
        how: tor_config::Reconfigure,
8
    ) -> StdResult<(), ErrorDetail> {
        // We _do_ have to check the cache_dir, since we can't and won't change that
        // while we're running.
        // (We already checked the state_dir in ClientShared::reconfigure_inner.)
8
        if new_config.storage.cache_dir != self.config.storage.cache_dir {
            how.cannot_change("storage.cache_dir")?;
8
        }
8
        if how == tor_config::Reconfigure::CheckAllOrNothing {
4
            return Ok(());
4
        }
4
        self.config = new_config.clone();
4
        Ok(())
8
    }
}
impl<R: Runtime> RunningInner<R> {
    /// Construct a new [`RunningInner`] and launch its associated tasks.
    fn new(
        pending: NotConstructedInner<R>,
        client: &ClientShared<R>,
    ) -> StdResult<Arc<Self>, ErrorDetail> {
        let NotConstructedInner {
            config,
            dormant_recv,
            status_sender,
            bootstrap_setting_receiver,
            dirmgr_builder,
            dirmgr_extensions,
        } = pending;
        let runtime = client.runtime.clone();
        let dormant = dormant_recv
            .borrow()
            .expect("Client somehow dropped while creating RunningInner");
        let memquota = &client.memquota;
        let statemgr = &client.statemgr;
        let path_resolver = &client.path_resolver;
        let (state_dir, _) = config.state_dir()?;
        let chanmgr = Arc::new(
            tor_chanmgr::ChanMgr::new(
                runtime.clone(),
                ChanMgrConfig::new(config.channel.clone()),
                dormant.into(),
                &NetParameters::from_map(&config.override_net_params),
                memquota.clone(),
            )
            .map_err(ErrorDetail::ChanMgrSetup)?,
        );
        let guardmgr = tor_guardmgr::GuardMgr::new(runtime.clone(), statemgr.clone(), &config)
            .map_err(ErrorDetail::GuardMgrSetup)?;
        #[cfg(feature = "pt-client")]
        let pt_mgr = {
            let pt_state_dir = state_dir.as_path().join("pt_state");
            config.storage.permissions().make_directory(&pt_state_dir)?;
            let mgr = Arc::new(tor_ptmgr::PtMgr::new(
                config.bridges.transports.clone(),
                pt_state_dir,
                Arc::clone(path_resolver),
                config.channel.outbound_proxy().cloned(),
                runtime.clone(),
            )?);
            chanmgr.set_pt_mgr(mgr.clone());
            mgr
        };
        let circmgr = Arc::new(
            tor_circmgr::CircMgr::new(
                &config,
                statemgr.clone(),
                &runtime,
                Arc::clone(&chanmgr),
                &guardmgr,
            )
            .map_err(ErrorDetail::CircMgrSetup)?,
        );
        let dir_cfg = {
            let mut c: tor_dirmgr::DirMgrConfig = config.dir_mgr_config()?;
            c.extensions = dirmgr_extensions;
            c
        };
        let dirmgr = dirmgr_builder
            .build(
                runtime.clone(),
                client.dirmgr_store.clone(),
                Arc::clone(&circmgr),
                dir_cfg,
            )
            .map_err(crate::Error::into_detail)?;
        let mut periodic_task_handles = circmgr
            .launch_background_tasks(&runtime, &dirmgr, statemgr.clone())
            .map_err(ErrorDetail::CircMgrSetup)?;
        periodic_task_handles.extend(dirmgr.download_task_handle());
        periodic_task_handles.extend(
            chanmgr
                .launch_background_tasks(&runtime, dirmgr.clone().upcast_arc())
                .map_err(ErrorDetail::ChanMgrSetup)?,
        );
        #[cfg(feature = "bridge-client")]
        // TODO: We can just construct this.
        let bridge_desc_mgr = Arc::new(Mutex::new(None));
        #[cfg(any(feature = "onion-service-client", feature = "onion-service-service"))]
        let hs_circ_pool = {
            let circpool = Arc::new(tor_circmgr::hspool::HsCircPool::new(&circmgr));
            circpool
                .launch_background_tasks(&runtime, &dirmgr.clone().upcast_arc())
                .map_err(ErrorDetail::CircMgrSetup)?;
            circpool
        };
        #[cfg(feature = "onion-service-client")]
        let hsclient = {
            // Prompt the hs connector to do its data housekeeping when we get a new consensus.
            // That's a time we're doing a bunch of thinking anyway, and it's not very frequent.
            let housekeeping = dirmgr.events().filter_map(|event| async move {
                match event {
                    DirEvent::NewConsensus => Some(()),
                    _ => None,
                }
            });
            let housekeeping = Box::pin(housekeeping);
            HsClientConnector::new(runtime.clone(), hs_circ_pool.clone(), &config, housekeeping)?
        };
        let conn_status = chanmgr.bootstrap_events();
        let dir_status = dirmgr.bootstrap_events();
        let skew_status = circmgr.skew_events();
        let rtclone = runtime.clone();
        // TODO: It might be a good idea to check this earlier, in `create_impl`,
        // when we have only the DirMgrStore.
        // But if we do that we need to add a method to DirMgrStore
        // to look at the protocol recommentations.
        #[allow(clippy::print_stderr)]
        crate::protostatus::enforce_protocol_recommendations(
            &runtime,
            Arc::clone(&dirmgr),
            crate::software_release_date(),
            crate::supported_protocols(),
            // TODO #1932: It would be nice to have a cleaner shutdown mechanism here,
            // but that will take some work.
            |fatal| async move {
                use tor_error::ErrorReport as _;
                // We already logged this error, but let's tell stderr too.
                eprintln!(
                    "Shutting down because of unsupported software version.\nError was:\n{}",
                    fatal.report(),
                );
                if let Some(hint) = crate::err::Error::from(fatal).hint() {
                    eprintln!("{}", hint);
                }
                // Give the tracing module a while to flush everything, since it has no built-in
                // flush function.
                rtclone.sleep(std::time::Duration::new(5, 0)).await;
                std::process::exit(1);
            },
        )?;
        runtime
            .spawn(status::report_status(
                status_sender,
                conn_status,
                dir_status,
                skew_status,
                bootstrap_setting_receiver,
            ))
            .map_err(|e| ErrorDetail::from_spawn("top-level status reporter", e))?;
        runtime
            .spawn(tasks_monitor_dormant(
                dormant_recv.clone(),
                dirmgr.clone().upcast_arc(),
                chanmgr.clone(),
                #[cfg(feature = "bridge-client")]
                bridge_desc_mgr.clone(),
                periodic_task_handles,
            ))
            .map_err(|e| ErrorDetail::from_spawn("periodic task dormant monitor", e))?;
        let running_inner = Arc::new(RunningInner {
            chanmgr,
            circmgr,
            dirmgr,
            #[cfg(feature = "bridge-client")]
            bridge_desc_mgr,
            #[cfg(feature = "pt-client")]
            pt_mgr,
            #[cfg(feature = "onion-service-client")]
            hsclient,
            #[cfg(any(feature = "onion-service-client", feature = "onion-service-service"))]
            hs_circ_pool,
            guardmgr,
        });
        Ok(running_inner)
    }
    /// Tell the parts of this [`RunningInner`] to reconfigure themselves
    /// (or to check the new configuration, if `how == CheckAllOrNothing`).
    fn reconfigure(
        &self,
        new_config: &TorClientConfig,
        how: tor_config::Reconfigure,
    ) -> crate::Result<()> {
        let dir_cfg = new_config.dir_mgr_config().map_err(wrap_err)?;
        let retire_circuits = self
            .circmgr
            .reconfigure(new_config, how)
            .map_err(wrap_err)?;
        #[cfg(any(feature = "onion-service-client", feature = "onion-service-service"))]
        if retire_circuits != RetireCircuits::None {
            self.hs_circ_pool.retire_all_circuits().map_err(wrap_err)?;
        }
        self.dirmgr.reconfigure(&dir_cfg, how).map_err(wrap_err)?;
        let netparams = self.dirmgr.params();
        self.chanmgr
            .reconfigure(&new_config.channel, how, netparams)
            .map_err(wrap_err)?;
        #[cfg(feature = "pt-client")]
        self.pt_mgr
            .reconfigure(
                how,
                new_config.bridges.transports.clone(),
                new_config.channel.outbound_proxy().cloned(),
            )
            .map_err(wrap_err)?;
        Ok(())
    }
}
impl<R: Runtime> TorClient<R> {
    /// Change the configuration of this TorClient to `new_config`.
    ///
    /// The `how` describes whether to perform an all-or-nothing
    /// reconfiguration: either all of the configuration changes will be
    /// applied, or none will. If you have disabled all-or-nothing changes, then
    /// only fatal errors will be reported in this function's return value.
    ///
    /// When performing a reconfiguration,
    /// a returned error may indicate that the client is now in an inconsistent state.
    ///
    /// This function applies its changes to **all** TorClient instances derived
    /// from the same call to `TorClient::create_*`: even ones whose circuits
    /// are isolated from this handle.
    ///
    /// # Limitations
    ///
    /// Although most options are reconfigurable, there are some whose values
    /// can't be changed on an a running TorClient.  Those options (or their
    /// sections) are explicitly documented not to be changeable.
    /// NOTE: Currently, not all of these non-reconfigurable options are
    /// documented. See [arti#1721][arti-1721].
    ///
    /// [arti-1721]: https://gitlab.torproject.org/tpo/core/arti/-/issues/1721
    ///
    /// Changing some options do not take effect immediately on all open streams
    /// and circuits, but rather affect only future streams and circuits.  Those
    /// are also explicitly documented.
    #[instrument(skip_all, level = "trace")]
    #[allow(clippy::cognitive_complexity)]
4
    pub fn reconfigure(
4
        &self,
4
        new_config: &TorClientConfig,
4
        how: tor_config::Reconfigure,
4
    ) -> crate::Result<()> {
        // We need to hold this lock while we're reconfiguring the client: even
        // though the individual fields have their own synchronization, we can't
        // safely let two threads change them at once.  If we did, then we'd
        // introduce time-of-check/time-of-use bugs in checking our configuration,
        // deciding how to change it, then applying the changes.
4
        let guard = self.client.reconfigure_lock.lock().expect("Poisoned lock");
        use tor_config::Reconfigure::*;
4
        match how {
            AllOrNothing => {
                // We have to check before we make any changes.
4
                self.client
4
                    .reconfigure_inner(new_config, CheckAllOrNothing, &guard)?;
                // Hopefully this doesn't fail,
                // otherwise we may have returned early from the reconfiguration
                // and its no longer "all-or-nothing".
4
                let result = self
4
                    .client
4
                    .reconfigure_inner(new_config, AllOrNothing, &guard);
4
                if result.is_err() {
                    warn!(
                        "Attempted an \"all-or-nothing\" reconfigure, but unexpectedly failed. \
                        The client will continue to run in an inconsistent state."
                    );
4
                }
4
                result
            }
            WarnOnFailures => {
                let result = self.client.reconfigure_inner(new_config, how, &guard);
                // If there's a fatal error,
                // we may have reconfigured some components and not others.
                if result.is_err() {
                    warn!(
                        "Attempted a reconfigure, but failed. \
                        The client will continue to run in an inconsistent state."
                    );
                }
                result
            }
            CheckAllOrNothing => self.client.reconfigure_inner(new_config, how, &guard),
            _ => self.client.reconfigure_inner(new_config, how, &guard),
        }
4
    }
    /// Return a new isolated `TorClient` handle.
    ///
    /// The two `TorClient`s will share internal state and configuration, but
    /// their streams will never share circuits with one another.
    ///
    /// Use this function when you want separate parts of your program to
    /// each have a TorClient handle, but where you don't want their
    /// activities to be linkable to one another over the Tor network.
    ///
    /// Calling this function is usually preferable to creating a
    /// completely separate TorClient instance, since it can share its
    /// internals with the existing `TorClient`.
    #[must_use]
2
    pub fn isolated_client(&self) -> Arc<TorClient<R>> {
2
        let result = TorClient {
2
            client_isolation: IsolationToken::new(),
2
            connect_prefs: self.connect_prefs.clone(),
2
            client: Arc::clone(&self.client),
2
        };
2
        Arc::new(result)
2
    }
    /// Launch an anonymized connection to the provided address and port over
    /// the Tor network.
    ///
    /// Note that because Tor prefers to do DNS resolution on the remote side of
    /// the network, this function takes its address as a string:
    ///
    /// ```no_run
    /// # use arti_client::*;use tor_rtcompat::Runtime;
    /// # async fn ex<R:Runtime>(tor_client: TorClient<R>) -> Result<()> {
    /// // The most usual way to connect is via an address-port tuple.
    /// let socket = tor_client.connect(("www.example.com", 443)).await?;
    ///
    /// // You can also specify an address and port as a colon-separated string.
    /// let socket = tor_client.connect("www.example.com:443").await?;
    /// # Ok(())
    /// # }
    /// ```
    ///
    /// Hostnames are _strongly_ preferred here: if this function allowed the
    /// caller here to provide an IPAddr or [`IpAddr`] or
    /// [`SocketAddr`](std::net::SocketAddr) address, then
    ///
    /// ```no_run
    /// # use arti_client::*; use tor_rtcompat::Runtime;
    /// # async fn ex<R:Runtime>(tor_client: TorClient<R>) -> Result<()> {
    /// # use std::net::ToSocketAddrs;
    /// // BAD: We're about to leak our target address to the local resolver!
    /// let address = "www.example.com:443".to_socket_addrs().unwrap().next().unwrap();
    /// // 🤯 Oh no! Now any eavesdropper can tell where we're about to connect! 🤯
    ///
    /// // Fortunately, this won't compile, since SocketAddr doesn't implement IntoTorAddr.
    /// // let socket = tor_client.connect(address).await?;
    /// //                                 ^^^^^^^ the trait `IntoTorAddr` is not implemented for `std::net::SocketAddr`
    /// # Ok(())
    /// # }
    /// ```
    ///
    /// If you really do need to connect to an IP address rather than a
    /// hostname, and if you're **sure** that the IP address came from a safe
    /// location, there are a few ways to do so.
    ///
    /// ```no_run
    /// # use arti_client::{TorClient,Result};use tor_rtcompat::Runtime;
    /// # use std::net::{SocketAddr,IpAddr};
    /// # async fn ex<R:Runtime>(tor_client: TorClient<R>) -> Result<()> {
    /// # use std::net::ToSocketAddrs;
    /// // ⚠️This is risky code!⚠️
    /// // (Make sure your addresses came from somewhere safe...)
    ///
    /// // If we have a fixed address, we can just provide it as a string.
    /// let socket = tor_client.connect("192.0.2.22:443").await?;
    /// let socket = tor_client.connect(("192.0.2.22", 443)).await?;
    ///
    /// // If we have a SocketAddr or an IpAddr, we can use the
    /// // DangerouslyIntoTorAddr trait.
    /// use arti_client::DangerouslyIntoTorAddr;
    /// let sockaddr = SocketAddr::from(([192, 0, 2, 22], 443));
    /// let ipaddr = IpAddr::from([192, 0, 2, 22]);
    /// let socket = tor_client.connect(sockaddr.into_tor_addr_dangerously().unwrap()).await?;
    /// let socket = tor_client.connect((ipaddr, 443).into_tor_addr_dangerously().unwrap()).await?;
    /// # Ok(())
    /// # }
    /// ```
    #[instrument(skip_all, level = "trace")]
4
    pub async fn connect<A: IntoTorAddr>(&self, target: A) -> crate::Result<DataStream> {
        self.connect_with_prefs(target, &self.connect_prefs).await
4
    }
    /// Launch an anonymized connection to the provided address and
    /// port over the Tor network, with explicit connection preferences.
    ///
    /// Note that because Tor prefers to do DNS resolution on the remote
    /// side of the network, this function takes its address as a string.
    /// (See [`TorClient::connect()`] for more information.)
    #[instrument(skip_all, level = "trace")]
4
    pub async fn connect_with_prefs<A: IntoTorAddr>(
4
        &self,
4
        target: A,
4
        prefs: &StreamPrefs,
4
    ) -> crate::Result<DataStream> {
        let addr = target.into_tor_addr().map_err(wrap_err)?;
        let mut stream_parameters = prefs.stream_parameters();
        // This macro helps prevent code duplication in the match below.
        //
        // Ideally, the match should resolve to a tuple consisting of the
        // tunnel, and the address, port and stream params,
        // but that's not currently possible because
        // the Exit and Hs branches use different tunnel types.
        //
        // TODO: replace with an async closure (when our MSRV allows it),
        // or with a more elegant approach.
        macro_rules! begin_stream {
            ($tunnel:expr, $addr:expr, $port:expr, $stream_params:expr) => {{
                let fut = $tunnel.begin_stream($addr, $port, $stream_params);
                self.client
                    .runtime
                    .timeout(self.client.timeoutcfg.get().connect_timeout, fut)
                    .await
                    .map_err(|_| ErrorDetail::ExitTimeout)?
                    .map_err(|cause| ErrorDetail::StreamFailed {
                        cause,
                        kind: "data",
                    })
            }};
        }
        let stream = match addr.into_stream_instructions(&self.client.addrcfg.get(), prefs)? {
            StreamInstructions::Exit {
                hostname: addr,
                port,
            } => {
                let exit_ports = [prefs.wrap_target_port(port)];
                let tunnel = self
                    .get_or_launch_exit_tunnel(&exit_ports, prefs)
                    .await
                    .map_err(wrap_err)?;
                debug!(
                    tunnel_id = %tunnel.unique_id(),
                    "Got a circuit for {}:{}", sensitive(&addr), port);
                begin_stream!(tunnel, &addr, port, Some(stream_parameters))
            }
            #[cfg(not(feature = "onion-service-client"))]
            #[allow(unused_variables)] // for hostname and port
            StreamInstructions::Hs {
                hsid,
                hostname,
                port,
            } => void::unreachable(hsid.0),
            #[cfg(feature = "onion-service-client")]
            StreamInstructions::Hs {
                hsid,
                hostname,
                port,
            } => {
                use safelog::DisplayRedacted as _;
                let running = self
                    .client
                    .wait_for_bootstrap_running("connect to hidden service")
                    .await?;
                let netdir = self.netdir(Timeliness::Timely, "connect to a hidden service")?;
                let mut hs_client_secret_keys_builder = HsClientSecretKeysBuilder::default();
                if let Some(keymgr) = &self.client.inert_client.keymgr {
                    let desc_enc_key_spec = HsClientDescEncKeypairSpecifier::new(hsid);
                    let ks_hsc_desc_enc =
                        keymgr.get::<HsClientDescEncKeypair>(&desc_enc_key_spec)?;
                    if let Some(ks_hsc_desc_enc) = ks_hsc_desc_enc {
                        debug!(
                            "Found descriptor decryption key for {}",
                            hsid.display_redacted()
                        );
                        hs_client_secret_keys_builder.ks_hsc_desc_enc(ks_hsc_desc_enc);
                    }
                };
                let hs_client_secret_keys = hs_client_secret_keys_builder
                    .build()
                    .map_err(ErrorDetail::Configuration)?;
                let tunnel = running
                    .hsclient
                    .get_or_launch_tunnel(
                        &netdir,
                        hsid,
                        hs_client_secret_keys,
                        self.isolation(prefs),
                    )
                    .await
                    .map_err(|cause| ErrorDetail::ObtainHsCircuit { cause, hsid })?;
                // On connections to onion services, we have to suppress
                // everything except the port from the BEGIN message.  We also
                // disable optimistic data.
                stream_parameters
                    .suppress_hostname()
                    .suppress_begin_flags()
                    .optimistic(false);
                begin_stream!(tunnel, &hostname, port, Some(stream_parameters))
            }
        };
        Ok(stream?)
4
    }
    /// Provides a new handle on this client, but with adjusted default preferences.
    ///
    /// Connections made with e.g. [`connect`](TorClient::connect) on the returned handle will use
    /// `connect_prefs`.
    #[must_use]
    pub fn with_prefs(&self, connect_prefs: StreamPrefs) -> Arc<Self> {
        let result = TorClient {
            client_isolation: self.client_isolation,
            connect_prefs,
            client: Arc::clone(&self.client),
        };
        Arc::new(result)
    }
    /// On success, return a list of IP addresses.
    #[instrument(skip_all, level = "trace")]
    pub async fn resolve(&self, hostname: &str) -> crate::Result<Vec<IpAddr>> {
        self.resolve_with_prefs(hostname, &self.connect_prefs).await
    }
    /// On success, return a list of IP addresses, but use prefs.
    #[instrument(skip_all, level = "trace")]
    pub async fn resolve_with_prefs(
        &self,
        hostname: &str,
        prefs: &StreamPrefs,
    ) -> crate::Result<Vec<IpAddr>> {
        // TODO This dummy port is only because `address::Host` is not pub(crate),
        // but I see no reason why it shouldn't be?  Then `into_resolve_instructions`
        // should be a method on `Host`, not `TorAddr`.  -Diziet.
        let addr = (hostname, 1).into_tor_addr().map_err(wrap_err)?;
        match addr.into_resolve_instructions(&self.client.addrcfg.get(), prefs)? {
            ResolveInstructions::Exit(hostname) => {
                let circ = self.get_or_launch_exit_tunnel(&[], prefs).await?;
                let resolve_future = circ.resolve(&hostname);
                let addrs = self
                    .client
                    .runtime
                    .timeout(self.client.timeoutcfg.get().resolve_timeout, resolve_future)
                    .await
                    .map_err(|_| ErrorDetail::ExitTimeout)?
                    .map_err(|cause| ErrorDetail::StreamFailed {
                        cause,
                        kind: "DNS lookup",
                    })?;
                Ok(addrs)
            }
            ResolveInstructions::Return(addrs) => Ok(addrs),
        }
    }
    /// Perform a remote DNS reverse lookup with the provided IP address.
    ///
    /// On success, return a list of hostnames.
    #[instrument(skip_all, level = "trace")]
    pub async fn resolve_ptr(&self, addr: IpAddr) -> crate::Result<Vec<String>> {
        self.resolve_ptr_with_prefs(addr, &self.connect_prefs).await
    }
    /// Perform a remote DNS reverse lookup with the provided IP address.
    ///
    /// On success, return a list of hostnames.
    #[instrument(level = "trace", skip_all)]
    pub async fn resolve_ptr_with_prefs(
        &self,
        addr: IpAddr,
        prefs: &StreamPrefs,
    ) -> crate::Result<Vec<String>> {
        let circ = self.get_or_launch_exit_tunnel(&[], prefs).await?;
        let resolve_ptr_future = circ.resolve_ptr(addr);
        let hostnames = self
            .client
            .runtime
            .timeout(
                self.client.timeoutcfg.get().resolve_ptr_timeout,
                resolve_ptr_future,
            )
            .await
            .map_err(|_| ErrorDetail::ExitTimeout)?
            .map_err(|cause| ErrorDetail::StreamFailed {
                cause,
                kind: "reverse DNS lookup",
            })?;
        Ok(hostnames)
    }
    /// Return a reference to this client's directory manager.
    ///
    /// This function is unstable. It is only enabled if the crate was
    /// built with the `experimental-api` feature.
    #[cfg(feature = "experimental-api")]
    pub fn dirmgr(&self) -> crate::Result<Arc<dyn tor_dirmgr::DirProvider>> {
        Ok(self
            .client
            .running_inner("access internal functionality")?
            .dirmgr
            .clone())
    }
    /// Return a reference to this client's circuit manager.
    ///
    /// This function is unstable. It is only enabled if the crate was
    /// built with the `experimental-api` feature.
    #[cfg(feature = "experimental-api")]
    pub fn circmgr(&self) -> crate::Result<Arc<tor_circmgr::CircMgr<R>>> {
        Ok(self
            .client
            .running_inner("access internal functionality")?
            .circmgr
            .clone())
    }
    /// Return a reference to this client's channel manager.
    ///
    /// This function is unstable. It is only enabled if the crate was
    /// built with the `experimental-api` feature.
    #[cfg(feature = "experimental-api")]
    pub fn chanmgr(&self) -> crate::Result<Arc<tor_chanmgr::ChanMgr<R>>> {
        Ok(self
            .client
            .running_inner("access internal functionality")?
            .chanmgr
            .clone())
    }
    /// Return a reference to this client's circuit pool.
    ///
    /// This function is unstable. It is only enabled if the crate was
    /// built with the `experimental-api` feature and any of `onion-service-client`
    /// or `onion-service-service` features. This method is required to invoke
    /// tor_hsservice::OnionService::launch()
    #[cfg(all(
        feature = "experimental-api",
        any(feature = "onion-service-client", feature = "onion-service-service")
    ))]
    pub fn hs_circ_pool(&self) -> crate::Result<Arc<tor_circmgr::hspool::HsCircPool<R>>> {
        Ok(self
            .client
            .running_inner("access internal functionality")?
            .hs_circ_pool
            .clone())
    }
    /// Return a reference to the runtime being used by this client.
    //
    // This API is not a hostage to fortune since we already require that R: Clone,
    // and necessarily a TorClient must have a clone of it.
    //
    // We provide it simply to save callers who have a TorClient from
    // having to separately keep their own handle,
4
    pub fn runtime(&self) -> &R {
4
        &self.client.runtime
4
    }
    /// Return a netdir that is timely according to the rules of `timeliness`.
    ///
    /// The `action` string is a description of what we wanted to do with the
    /// directory, to be put into the error message if we couldn't find a directory.
    fn netdir(
        &self,
        timeliness: Timeliness,
        action: &'static str,
    ) -> StdResult<Arc<tor_netdir::NetDir>, ErrorDetail> {
        use tor_netdir::Error as E;
        // TODO: Conceivably we could take a NetDir from our DirMgrStore.
        match self.client.running_inner(action)?.dirmgr.netdir(timeliness) {
            Ok(netdir) => Ok(netdir),
            Err(E::NoInfo) | Err(E::NotEnoughInfo) => {
                Err(ErrorDetail::BootstrapRequired { action })
            }
            Err(error) => Err(ErrorDetail::NoDir { error, action }),
        }
    }
    /// Get or launch an exit-suitable circuit with a given set of
    /// exit ports.
    #[instrument(skip_all, level = "trace")]
4
    async fn get_or_launch_exit_tunnel(
4
        &self,
4
        exit_ports: &[TargetPort],
4
        prefs: &StreamPrefs,
4
    ) -> StdResult<ClientDataTunnel, ErrorDetail> {
        let running = self
            .client
            .wait_for_bootstrap_running("build a circuit")
            .await?;
        // TODO HS probably this netdir ought to be made in connect_with_prefs
        // like for StreamInstructions::Hs.
        let dir = self.netdir(Timeliness::Timely, "build a circuit")?;
        let tunnel = running
            .circmgr
            .get_or_launch_exit(
                dir.as_ref().into(),
                exit_ports,
                self.isolation(prefs),
                #[cfg(feature = "geoip")]
                prefs.country_code,
            )
            .await
            .map_err(|cause| ErrorDetail::ObtainExitCircuit {
                cause,
                exit_ports: Sensitive::new(exit_ports.into()),
            })?;
        drop(dir); // This decreases the refcount on the netdir.
        Ok(tunnel)
4
    }
    /// Return an overall [`Isolation`] for this `TorClient` and a `StreamPrefs`.
    ///
    /// This describes which operations might use
    /// circuit(s) with this one.
    ///
    /// This combines isolation information from
    /// [`StreamPrefs::prefs_isolation`]
    /// and the `TorClient`'s isolation (eg from [`TorClient::isolated_client`]).
    fn isolation(&self, prefs: &StreamPrefs) -> StreamIsolation {
        let mut b = StreamIsolationBuilder::new();
        // Always consider our client_isolation.
        b.owner_token(self.client_isolation);
        // Consider stream isolation too, if it's set.
        if let Some(tok) = prefs.prefs_isolation() {
            b.stream_isolation(tok);
        }
        // Failure should be impossible with this builder.
        b.build().expect("Failed to construct StreamIsolation")
    }
    /// Try to launch an onion service with a given configuration.
    ///
    /// Returns `Ok(None)` if the service specified is disabled in the config.
    ///
    /// This onion service will not actually handle any requests on its own: you
    /// will need to
    /// pull [`RendRequest`](tor_hsservice::RendRequest) objects from the returned stream,
    /// [`accept`](tor_hsservice::RendRequest::accept) the ones that you want to
    /// answer, and then wait for them to give you [`StreamRequest`](tor_hsservice::StreamRequest)s.
    ///
    /// You may find the [`tor_hsservice::handle_rend_requests`] API helpful for
    /// translating `RendRequest`s into `StreamRequest`s.
    ///
    /// If you want to forward all the requests from an onion service to a set
    /// of local ports, you may want to use the `tor-hsrproxy` crate.
    #[cfg(feature = "onion-service-service")]
    #[instrument(skip_all, level = "trace")]
    pub fn launch_onion_service(
        &self,
        config: tor_hsservice::OnionServiceConfig,
    ) -> crate::Result<
        Option<(
            Arc<tor_hsservice::RunningOnionService>,
            impl futures::Stream<Item = tor_hsservice::RendRequest> + use<R>,
        )>,
    > {
        let nickname = config.nickname();
        if !config.enabled() {
            info!(
                nickname=%nickname,
                "Skipping onion service because it was disabled in the config"
            );
            return Ok(None);
        }
        let running = self
            .client
            .initiate_bootstrap_if_needed("launch onion service")?;
        let keymgr = self
            .client
            .inert_client
            .keymgr
            .as_ref()
            .ok_or(ErrorDetail::KeystoreRequired {
                action: "launch onion service",
            })?
            .clone();
        let state_dir = self.client.state_directory.clone();
        let service = tor_hsservice::OnionService::builder()
            .config(config) // TODO #1186: Allow override of KeyMgr for "ephemeral" operation?
            .keymgr(keymgr)
            // TODO #1186: Allow override of StateMgr for "ephemeral" operation?
            .state_dir(state_dir)
            .build()
            .map_err(ErrorDetail::LaunchOnionService)?;
        Ok(service
            .launch(
                self.client.runtime.clone(),
                running.dirmgr.clone().upcast_arc(),
                running.hs_circ_pool.clone(),
                Arc::clone(&self.client.path_resolver),
            )
            .map_err(ErrorDetail::LaunchOnionService)?)
    }
    /// Try to launch an onion service with a given configuration and provided
    /// [`HsIdKeypair`]. If an onion service with the given nickname already has an
    /// associated `HsIdKeypair`  in this `TorClient`'s `KeyMgr`, then this operation
    /// fails rather than overwriting the existing key.
    ///
    /// Returns `Ok(None)` if the service specified is disabled in the config.
    ///
    /// The specified `HsIdKeypair` will be inserted in the primary keystore.
    ///
    /// **Important**: depending on the configuration of your
    /// [primary keystore](tor_keymgr::config::PrimaryKeystoreConfig),
    /// the `HsIdKeypair` **may** get persisted to disk.
    /// By default, Arti's primary keystore is the [native](ArtiKeystoreKind::Native),
    /// disk-based keystore.
    ///
    /// This onion service will not actually handle any requests on its own: you
    /// will need to
    /// pull [`RendRequest`](tor_hsservice::RendRequest) objects from the returned stream,
    /// [`accept`](tor_hsservice::RendRequest::accept) the ones that you want to
    /// answer, and then wait for them to give you [`StreamRequest`](tor_hsservice::StreamRequest)s.
    ///
    /// You may find the [`tor_hsservice::handle_rend_requests`] API helpful for
    /// translating `RendRequest`s into `StreamRequest`s.
    ///
    /// If you want to forward all the requests from an onion service to a set
    /// of local ports, you may want to use the `tor-hsrproxy` crate.
    #[cfg(all(feature = "onion-service-service", feature = "experimental-api"))]
    #[instrument(skip_all, level = "trace")]
    pub fn launch_onion_service_with_hsid(
        &self,
        config: tor_hsservice::OnionServiceConfig,
        id_keypair: HsIdKeypair,
    ) -> crate::Result<
        Option<(
            Arc<tor_hsservice::RunningOnionService>,
            impl futures::Stream<Item = tor_hsservice::RendRequest> + use<R>,
        )>,
    > {
        let nickname = config.nickname();
        let hsid_spec = HsIdKeypairSpecifier::new(nickname.clone());
        let selector = KeystoreSelector::Primary;
        let _kp = self
            .client
            .inert_client
            .keymgr
            .as_ref()
            .ok_or(ErrorDetail::KeystoreRequired {
                action: "launch onion service ex",
            })?
            .insert::<HsIdKeypair>(id_keypair, &hsid_spec, selector, false)?;
        self.launch_onion_service(config)
    }
    /// Generate a service discovery keypair for connecting to a hidden service running in
    /// "restricted discovery" mode.
    ///
    /// The `selector` argument is used for choosing the keystore in which to generate the keypair.
    /// While most users will want to write to the [`Primary`](KeystoreSelector::Primary), if you
    /// have configured this `TorClient` with a non-default keystore and wish to generate the
    /// keypair in it, you can do so by calling this function with a [KeystoreSelector::Id]
    /// specifying the keystore ID of your keystore.
    ///
    // Note: the selector argument exists for future-proofing reasons. We don't currently support
    // configuring custom or non-default keystores (see #1106).
    ///
    /// Returns an error if the key already exists in the specified key store.
    ///
    /// Important: the public part of the generated keypair must be shared with the service, and
    /// the service needs to be configured to allow the owner of its private counterpart to
    /// discover its introduction points. The caller is responsible for sharing the public part of
    /// the key with the hidden service.
    ///
    /// This function does not require the `TorClient` to be running or bootstrapped.
    //
    // TODO: decide whether this should use get_or_generate before making it
    // non-experimental
    #[cfg(all(
        feature = "onion-service-client",
        feature = "experimental-api",
        feature = "keymgr"
    ))]
    pub fn generate_service_discovery_key(
        &self,
        selector: KeystoreSelector,
        hsid: HsId,
    ) -> crate::Result<HsClientDescEncKey> {
        self.client
            .inert_client
            .generate_service_discovery_key(selector, hsid)
    }
    /// Rotate the service discovery keypair for connecting to a hidden service running in
    /// "restricted discovery" mode.
    ///
    /// **If the specified keystore already contains a restricted discovery keypair
    /// for the service, it will be overwritten.** Otherwise, a new keypair is generated.
    ///
    /// The `selector` argument is used for choosing the keystore in which to generate the keypair.
    /// While most users will want to write to the [`Primary`](KeystoreSelector::Primary), if you
    /// have configured this `TorClient` with a non-default keystore and wish to generate the
    /// keypair in it, you can do so by calling this function with a [KeystoreSelector::Id]
    /// specifying the keystore ID of your keystore.
    ///
    // Note: the selector argument exists for future-proofing reasons. We don't currently support
    // configuring custom or non-default keystores (see #1106).
    ///
    /// Important: the public part of the generated keypair must be shared with the service, and
    /// the service needs to be configured to allow the owner of its private counterpart to
    /// discover its introduction points. The caller is responsible for sharing the public part of
    /// the key with the hidden service.
    ///
    /// This function does not require the `TorClient` to be running or bootstrapped.
    #[cfg(all(
        feature = "onion-service-client",
        feature = "experimental-api",
        feature = "keymgr"
    ))]
    #[cfg_attr(
        docsrs,
        doc(cfg(all(
            feature = "onion-service-client",
            feature = "experimental-api",
            feature = "keymgr"
        )))
    )]
    pub fn rotate_service_discovery_key(
        &self,
        selector: KeystoreSelector,
        hsid: HsId,
    ) -> crate::Result<HsClientDescEncKey> {
        self.client
            .inert_client
            .rotate_service_discovery_key(selector, hsid)
    }
    /// Insert a service discovery secret key for connecting to a hidden service running in
    /// "restricted discovery" mode
    ///
    /// The `selector` argument is used for choosing the keystore in which to generate the keypair.
    /// While most users will want to write to the [`Primary`](KeystoreSelector::Primary), if you
    /// have configured this `TorClient` with a non-default keystore and wish to insert the
    /// key in it, you can do so by calling this function with a [KeystoreSelector::Id]
    ///
    // Note: the selector argument exists for future-proofing reasons. We don't currently support
    // configuring custom or non-default keystores (see #1106).
    ///
    /// Returns an error if the key already exists in the specified key store.
    ///
    /// Important: the public part of the generated keypair must be shared with the service, and
    /// the service needs to be configured to allow the owner of its private counterpart to
    /// discover its introduction points. The caller is responsible for sharing the public part of
    /// the key with the hidden service.
    ///
    /// This function does not require the `TorClient` to be running or bootstrapped.
    #[cfg(all(
        feature = "onion-service-client",
        feature = "experimental-api",
        feature = "keymgr"
    ))]
    #[cfg_attr(
        docsrs,
        doc(cfg(all(
            feature = "onion-service-client",
            feature = "experimental-api",
            feature = "keymgr"
        )))
    )]
    pub fn insert_service_discovery_key(
        &self,
        selector: KeystoreSelector,
        hsid: HsId,
        hs_client_desc_enc_secret_key: HsClientDescEncSecretKey,
    ) -> crate::Result<HsClientDescEncKey> {
        self.client.inert_client.insert_service_discovery_key(
            selector,
            hsid,
            hs_client_desc_enc_secret_key,
        )
    }
    /// Return the service discovery public key for the service with the specified `hsid`.
    ///
    /// Returns `Ok(None)` if no such key exists.
    ///
    /// This function does not require the `TorClient` to be running or bootstrapped.
    #[cfg(all(feature = "onion-service-client", feature = "experimental-api"))]
    #[cfg_attr(
        docsrs,
        doc(cfg(all(feature = "onion-service-client", feature = "experimental-api")))
    )]
    pub fn get_service_discovery_key(
        &self,
        hsid: HsId,
    ) -> crate::Result<Option<HsClientDescEncKey>> {
        self.client.inert_client.get_service_discovery_key(hsid)
    }
    /// Removes the service discovery keypair for the service with the specified `hsid`.
    ///
    /// Returns an error if the selected keystore is not the default keystore or one of the
    /// configured secondary stores.
    ///
    /// Returns `Ok(None)` if no such keypair exists whereas `Ok(Some()) means the keypair was successfully removed.
    ///
    /// Returns `Err` if an error occurred while trying to remove the key.
    #[cfg(all(
        feature = "onion-service-client",
        feature = "experimental-api",
        feature = "keymgr"
    ))]
    #[cfg_attr(
        docsrs,
        doc(cfg(all(
            feature = "onion-service-client",
            feature = "experimental-api",
            feature = "keymgr"
        )))
    )]
    pub fn remove_service_discovery_key(
        &self,
        selector: KeystoreSelector,
        hsid: HsId,
    ) -> crate::Result<Option<()>> {
        self.client
            .inert_client
            .remove_service_discovery_key(selector, hsid)
    }
    /// Create (but do not launch) a new
    /// [`OnionService`](tor_hsservice::OnionService)
    /// using the given configuration.
    ///
    /// This is useful for managing an onion service without needing to start a `TorClient` or the
    /// onion service itself.
    /// If you only wish to run the onion service, see
    /// [`TorClient::launch_onion_service()`]
    /// which allows you to launch an onion service from a running `TorClient`.
    ///
    /// The returned `OnionService` can be launched using
    /// [`OnionService::launch()`](tor_hsservice::OnionService::launch).
    /// Note that `launch()` requires a [`NetDirProvider`],
    /// [`HsCircPool`](tor_circmgr::hspool::HsCircPool), etc,
    /// which you should obtain from a running `TorClient`.
    /// But these are only accessible from a `TorClient` if the "experimental-api" feature is
    /// enabled.
    /// The behaviour is not specified if you create the `OnionService` with
    /// `create_onion_service()` using one [`TorClientConfig`],
    /// but launch it using a `TorClient` generated from a different `TorClientConfig`.
    // TODO #2249: Look into this behaviour more, and possibly error if there is a different config.
    #[cfg(feature = "onion-service-service")]
    #[instrument(skip_all, level = "trace")]
48
    pub fn create_onion_service(
48
        config: &TorClientConfig,
48
        svc_config: tor_hsservice::OnionServiceConfig,
48
    ) -> crate::Result<tor_hsservice::OnionService> {
48
        let inert_client = InertTorClient::new(config)?;
48
        inert_client.create_onion_service(config, svc_config)
48
    }
    /// Return a current [`status::BootstrapStatus`] describing how close this client
    /// is to being ready for user traffic.
    pub fn bootstrap_status(&self) -> status::BootstrapStatus {
        self.client.status_receiver.inner.borrow().clone()
    }
    /// Return a stream of [`status::BootstrapStatus`] events that will be updated
    /// whenever the client's status changes.
    ///
    /// The receiver might not receive every update sent to this stream, though
    /// when it does poll the stream it should get the most recent one.
    //
    // TODO(nickm): will this also need to implement Send and 'static?
    pub fn bootstrap_events(&self) -> status::BootstrapEvents {
        self.client.status_receiver.clone()
    }
    /// Change the client's current dormant mode, putting background tasks to sleep
    /// or waking them up as appropriate.
    ///
    /// This can be used to conserve CPU usage if you aren't planning on using the
    /// client for a while, especially on mobile platforms.
    ///
    /// See the [`DormantMode`] documentation for more details.
    pub fn set_dormant(&self, mode: DormantMode) {
        *self
            .client
            .dormant
            .lock()
            .expect("dormant lock poisoned")
            .borrow_mut() = Some(mode);
    }
    /// Return a [`Future`] which resolves
    /// once this TorClient has stopped.
    #[cfg(feature = "experimental-api")]
    #[instrument(skip_all, level = "trace")]
    pub fn wait_for_stop(
        &self,
    ) -> impl futures::Future<Output = ()> + Send + Sync + 'static + use<R> {
        // We defer to the "wait for unlock" handle on our statemgr.
        //
        // The statemgr won't actually be unlocked until it is finally
        // dropped, which will happen when this TorClient is
        // dropped—which is what we want.
        self.client.statemgr.wait_for_unlock()
    }
    /// Getter for keymgr.
    #[cfg(feature = "onion-service-cli-extra")]
    pub fn keymgr(&self) -> crate::Result<&KeyMgr> {
        self.client.inert_client.keymgr()
    }
}
impl<R: Runtime> ClientShared<R> {
    /// Used by `bootstrap_inner`: Return a `RunningInner`, constructing it if necessary.
    fn instantiate_running_inner(
        &self,
        mut inner_guard: std::sync::MutexGuard<'_, Inner<R>>,
    ) -> Result<Arc<RunningInner<R>>, ErrorDetail> {
        match &*inner_guard {
            Inner::Running(running_inner) => Ok(Arc::clone(running_inner)),
            Inner::Poisoned(e) => Err(e.as_ref().clone()),
            Inner::NotConstructed(_) => {
                let error = ErrorDetail::from(internal!("Client under construction"));
                let mut pending = Inner::Poisoned(Box::new(error));
                std::mem::swap(&mut pending, &mut *inner_guard);
                let Inner::NotConstructed(pending) = pending else {
                    panic!("Surprising type change");
                };
                match RunningInner::new(*pending, self) {
                    Ok(running_inner) => {
                        *inner_guard = Inner::Running(Arc::clone(&running_inner));
                        self.bootstrap_setting_sender
                            .lock()
                            .expect("lock poisoned")
                            .borrow_mut()
                            .running_inner_is_present = true;
                        Ok(running_inner)
                    }
                    Err(e) => {
                        *inner_guard = Inner::Poisoned(Box::new(e.clone()));
                        Err(e)
                    }
                }
            }
        }
    }
    /// Implementation of `bootstrap`, split out in order to avoid manually specifying
    /// double error conversions.
    async fn bootstrap_inner(&self) -> StdResult<(), ErrorDetail> {
        // Wait for an existing bootstrap attempt to finish first.
        //
        // This is a futures::lock::Mutex, so it's okay to await while we hold it.
        let _bootstrap_lock = self.bootstrap_in_progress.lock().await;
        let running = self.instantiate_running_inner(self.inner.lock().expect("lock poisoned"))?;
        // Make sure we have a bridge descriptor manager, which is active iff required
        #[cfg(feature = "bridge-client")]
        {
            let mut dormant = self.dormant.lock().expect("dormant lock poisoned");
            let dormant = dormant.borrow();
            let dormant = dormant.ok_or_else(|| internal!("dormant dropped"))?.into();
            let mut bdm = running.bridge_desc_mgr.lock().expect("bdm lock poisoned");
            if bdm.is_none() {
                let new_bdm = Arc::new(BridgeDescMgr::new(
                    &Default::default(),
                    self.runtime.clone(),
                    self.dirmgr_store.clone(),
                    running.circmgr.clone(),
                    dormant,
                )?);
                running
                    .guardmgr
                    .install_bridge_desc_provider(&(new_bdm.clone() as _))
                    .map_err(ErrorDetail::GuardMgrSetup)?;
                // If ^ that fails, we drop the BridgeDescMgr again.  It may do some
                // work but will hopefully eventually quit.
                *bdm = Some(new_bdm);
            }
        }
        if self
            .statemgr
            .try_lock()
            .map_err(ErrorDetail::StateAccess)?
            .held()
        {
            debug!("It appears we have the lock on our state files.");
        } else {
            info!(
                "Another process has the lock on our state files. We'll proceed in read-only mode."
            );
        }
        // If we fail to bootstrap (i.e. we return before the disarm() point below), attempt to
        // unlock the state files.
        let unlock_guard = util::StateMgrUnlockGuard::new(&self.statemgr);
        running
            .dirmgr
            .bootstrap()
            .await
            .map_err(ErrorDetail::DirMgrBootstrap)?;
        // Since we succeeded, disarm the unlock guard.
        unlock_guard.disarm();
        Ok(())
    }
    /// Ensure that this client is running and bootstrapped, and return a [`RunningInner`] if it is.
    ///
    /// If we're not bootstrapped,
    /// we either try to bootstrap or return an error,
    /// depending on `self.should_bootstrap`:
    ///
    /// ## For `BootstrapBehavior::OnDemand` clients
    ///
    /// Initiate a bootstrap by calling `bootstrap_inner`
    /// (which is idempotent, so attempts to bootstrap twice will just do nothing).
    ///
    /// ## For `BootstrapBehavior::Manual` clients
    ///
    /// Check whether a bootstrap is in progress; if one is, wait until it finishes.
    /// Then see whether we're bootstrapped, and return either a success or a failure.
    #[instrument(skip_all, level = "trace")]
4
    async fn wait_for_bootstrap_running(
4
        &self,
4
        action: &'static str,
4
    ) -> StdResult<Arc<RunningInner<R>>, ErrorDetail> {
4
        match self.should_bootstrap {
            BootstrapBehavior::OnDemand => {
                self.bootstrap_inner().await?;
            }
            BootstrapBehavior::Manual => {
                // Grab the lock, and immediately release it.  That will ensure that nobody else is trying to bootstrap.
                self.bootstrap_in_progress.lock().await;
            }
        }
        self.dormant
            .lock()
            .map_err(|_| internal!("dormant poisoned"))?
4
            .try_maybe_send(|dormant| {
                Ok::<_, Bug>(Some({
4
                    match dormant.ok_or_else(|| internal!("dormant dropped"))? {
                        DormantMode::Soft => DormantMode::Normal,
4
                        other @ DormantMode::Normal => other,
                    }
                }))
4
            })?;
        self.running_inner(action)
4
    }
    /// If we are currently bootstrapping or running, return a [`RunningInner`].
4
    fn running_inner(&self, action: &'static str) -> StdResult<Arc<RunningInner<R>>, ErrorDetail> {
4
        let guard = self.inner.lock().expect("Lock poisoned");
4
        match &*guard {
4
            Inner::NotConstructed(_) => Err(ErrorDetail::BootstrapRequired { action }),
            Inner::Running(running_inner) => Ok(Arc::clone(running_inner)),
            Inner::Poisoned(e) => Err(e.as_ref().clone()),
        }
4
    }
    /// Ensure that our bootstrap state is [`RunningInner`], if possible.
    ///
    /// Return an error if our [`BootstrapBehavior`] is `Manual` and have not created a
    /// [`RunningInner`].
    fn initiate_bootstrap_if_needed(
        &self,
        action: &'static str,
    ) -> StdResult<Arc<RunningInner<R>>, ErrorDetail> {
        let guard = self.inner.lock().expect("Lock poisoned");
        match &*guard {
            Inner::Running(running_inner) => Ok(Arc::clone(running_inner)),
            Inner::Poisoned(e) => Err(e.as_ref().clone()),
            Inner::NotConstructed(_) => match self.should_bootstrap {
                BootstrapBehavior::Manual => Err(ErrorDetail::BootstrapRequired { action }),
                BootstrapBehavior::OnDemand => self.instantiate_running_inner(guard),
            },
        }
    }
    /// This is split out from `reconfigure` so we can do the all-or-nothing
    /// check without recursion. the caller to this method must hold the
    /// `reconfigure_lock`.
    #[instrument(level = "trace", skip_all)]
8
    fn reconfigure_inner(
8
        &self,
8
        new_config: &TorClientConfig,
8
        how: tor_config::Reconfigure,
8
        _reconfigure_lock_guard: &std::sync::MutexGuard<'_, ()>,
8
    ) -> crate::Result<()> {
        // We ignore 'new_config.path_resolver' here since CfgPathResolver does not impl PartialEq
        // and we have no way to compare them, but this field is explicitly documented as being
        // non-reconfigurable anyways.
8
        let addr_cfg = &new_config.address_filter;
8
        let timeout_cfg = &new_config.stream_timeouts;
8
        let state_cfg = new_config
8
            .storage
8
            .expand_state_dir(&self.path_resolver)
8
            .map_err(wrap_err)?;
        // TODO wasm: This ins't really how things should be long term,
        // but once we have a more generic notion of configuring storage
        // we can change this to comply with it.
        #[cfg(not(all(target_arch = "wasm32", target_os = "unknown")))]
        {
8
            if state_cfg != self.statemgr.path() {
                how.cannot_change("storage.state_dir").map_err(wrap_err)?;
8
            }
        }
8
        self.memquota
8
            .reconfigure(new_config.system.memory.clone(), how)
8
            .map_err(wrap_err)?;
8
        let mut inner_lock = self.inner.lock().expect("Lock poisoned");
8
        match &mut *inner_lock {
            Inner::Poisoned(e) => return Err(e.as_ref().clone().into()),
8
            Inner::NotConstructed(nc) => nc.reconfigure(new_config, how)?,
            Inner::Running(r) => {
                let running = Arc::clone(r);
                drop(inner_lock);
                running.reconfigure(new_config, how)?;
            }
        }
8
        if how == tor_config::Reconfigure::CheckAllOrNothing {
4
            return Ok(());
4
        }
4
        self.addrcfg.replace(addr_cfg.clone());
4
        self.timeoutcfg.replace(timeout_cfg.clone());
4
        Ok(())
8
    }
}
/// Monitor `dormant_mode` and enable/disable periodic tasks as applicable
///
/// This function is spawned as a task during client construction.
// TODO should this perhaps be done by each TaskHandle?
async fn tasks_monitor_dormant<R: Runtime>(
    mut dormant_rx: postage::watch::Receiver<Option<DormantMode>>,
    netdir: Arc<dyn NetDirProvider>,
    chanmgr: Arc<tor_chanmgr::ChanMgr<R>>,
    #[cfg(feature = "bridge-client")] bridge_desc_mgr: Arc<Mutex<Option<Arc<BridgeDescMgr<R>>>>>,
    periodic_task_handles: Vec<TaskHandle>,
) {
    while let Some(Some(mode)) = dormant_rx.next().await {
        let netparams = netdir.params();
        chanmgr
            .set_dormancy(mode.into(), netparams)
            .unwrap_or_else(|e| error_report!(e, "couldn't set dormancy"));
        // IEFI simplifies handling of exceptional cases, as "never mind, then".
        #[cfg(feature = "bridge-client")]
        (|| {
            let mut bdm = bridge_desc_mgr.lock().ok()?;
            let bdm = bdm.as_mut()?;
            bdm.set_dormancy(mode.into());
            Some(())
        })();
        let is_dormant = matches!(mode, DormantMode::Soft);
        for task in periodic_task_handles.iter() {
            if is_dormant {
                task.cancel();
            } else {
                task.fire();
            }
        }
    }
}
/// Alias for TorError::from(Error)
4
pub(crate) fn wrap_err<T>(err: T) -> crate::Error
4
where
4
    ErrorDetail: From<T>,
{
4
    ErrorDetail::from(err).into()
4
}
#[cfg(test)]
mod test {
    // @@ begin test lint list maintained by maint/add_warning @@
    #![allow(clippy::bool_assert_comparison)]
    #![allow(clippy::clone_on_copy)]
    #![allow(clippy::dbg_macro)]
    #![allow(clippy::mixed_attributes_style)]
    #![allow(clippy::print_stderr)]
    #![allow(clippy::print_stdout)]
    #![allow(clippy::single_char_pattern)]
    #![allow(clippy::unwrap_used)]
    #![allow(clippy::unchecked_time_subtraction)]
    #![allow(clippy::useless_vec)]
    #![allow(clippy::needless_pass_by_value)]
    #![allow(clippy::string_slice)] // See arti#2571
    //! <!-- @@ end test lint list maintained by maint/add_warning @@ -->
    use tor_config::Reconfigure;
    use super::*;
    use crate::config::TorClientConfigBuilder;
    use crate::{ErrorKind, HasKind};
    #[test]
    fn create_unbootstrapped() {
        tor_rtcompat::test_with_one_runtime!(|rt| async {
            let state_dir = tempfile::tempdir().unwrap();
            let cache_dir = tempfile::tempdir().unwrap();
            let cfg = TorClientConfigBuilder::from_directories(state_dir, cache_dir)
                .build()
                .unwrap();
            let _ = TorClient::with_runtime(rt)
                .config(cfg)
                .bootstrap_behavior(BootstrapBehavior::Manual)
                .create_unbootstrapped()
                .unwrap();
        });
        tor_rtcompat::test_with_one_runtime!(|rt| async {
            let state_dir = tempfile::tempdir().unwrap();
            let cache_dir = tempfile::tempdir().unwrap();
            let cfg = TorClientConfigBuilder::from_directories(state_dir, cache_dir)
                .build()
                .unwrap();
            let _ = TorClient::with_runtime(rt)
                .config(cfg)
                .bootstrap_behavior(BootstrapBehavior::Manual)
                .create_unbootstrapped_async()
                .await
                .unwrap();
        });
    }
    #[test]
    fn unbootstrapped_client_unusable() {
        tor_rtcompat::test_with_one_runtime!(|rt| async {
            let state_dir = tempfile::tempdir().unwrap();
            let cache_dir = tempfile::tempdir().unwrap();
            let cfg = TorClientConfigBuilder::from_directories(state_dir, cache_dir)
                .build()
                .unwrap();
            // Test sync
            let client = TorClient::with_runtime(rt)
                .config(cfg)
                .bootstrap_behavior(BootstrapBehavior::Manual)
                .create_unbootstrapped()
                .unwrap();
            let result = client.connect("example.com:80").await;
            assert!(result.is_err());
            assert_eq!(result.err().unwrap().kind(), ErrorKind::BootstrapRequired);
        });
        // Need a separate test for async because Runtime and TorClientConfig are consumed by the
        // builder
        tor_rtcompat::test_with_one_runtime!(|rt| async {
            let state_dir = tempfile::tempdir().unwrap();
            let cache_dir = tempfile::tempdir().unwrap();
            let cfg = TorClientConfigBuilder::from_directories(state_dir, cache_dir)
                .build()
                .unwrap();
            // Test sync
            let client = TorClient::with_runtime(rt)
                .config(cfg)
                .bootstrap_behavior(BootstrapBehavior::Manual)
                .create_unbootstrapped_async()
                .await
                .unwrap();
            let result = client.connect("example.com:80").await;
            assert!(result.is_err());
            assert_eq!(result.err().unwrap().kind(), ErrorKind::BootstrapRequired);
        });
    }
    #[test]
    fn streamprefs_isolate_every_stream() {
        let mut observed = StreamPrefs::new();
        observed.isolate_every_stream();
        match observed.isolation {
            StreamIsolationPreference::EveryStream => (),
            _ => panic!("unexpected isolation: {:?}", observed.isolation),
        };
    }
    #[test]
    fn streamprefs_new_has_expected_defaults() {
        let observed = StreamPrefs::new();
        assert_eq!(observed.ip_ver_pref, IpVersionPreference::Ipv4Preferred);
        assert!(!observed.optimistic_stream);
        // StreamIsolationPreference does not implement Eq, check manually.
        match observed.isolation {
            StreamIsolationPreference::None => (),
            _ => panic!("unexpected isolation: {:?}", observed.isolation),
        };
    }
    #[test]
    fn streamprefs_new_isolation_group() {
        let mut observed = StreamPrefs::new();
        observed.new_isolation_group();
        match observed.isolation {
            StreamIsolationPreference::Explicit(_) => (),
            _ => panic!("unexpected isolation: {:?}", observed.isolation),
        };
    }
    #[test]
    fn streamprefs_ipv6_only() {
        let mut observed = StreamPrefs::new();
        observed.ipv6_only();
        assert_eq!(observed.ip_ver_pref, IpVersionPreference::Ipv6Only);
    }
    #[test]
    fn streamprefs_ipv6_preferred() {
        let mut observed = StreamPrefs::new();
        observed.ipv6_preferred();
        assert_eq!(observed.ip_ver_pref, IpVersionPreference::Ipv6Preferred);
    }
    #[test]
    fn streamprefs_ipv4_only() {
        let mut observed = StreamPrefs::new();
        observed.ipv4_only();
        assert_eq!(observed.ip_ver_pref, IpVersionPreference::Ipv4Only);
    }
    #[test]
    fn streamprefs_ipv4_preferred() {
        let mut observed = StreamPrefs::new();
        observed.ipv4_preferred();
        assert_eq!(observed.ip_ver_pref, IpVersionPreference::Ipv4Preferred);
    }
    #[test]
    fn streamprefs_optimistic() {
        let mut observed = StreamPrefs::new();
        observed.optimistic();
        assert!(observed.optimistic_stream);
    }
    #[test]
    fn streamprefs_set_isolation() {
        let mut observed = StreamPrefs::new();
        observed.set_isolation(IsolationToken::new());
        match observed.isolation {
            StreamIsolationPreference::Explicit(_) => (),
            _ => panic!("unexpected isolation: {:?}", observed.isolation),
        };
    }
    #[test]
    fn reconfigure_all_or_nothing() {
        tor_rtcompat::test_with_one_runtime!(|rt| async {
            let state_dir = tempfile::tempdir().unwrap();
            let cache_dir = tempfile::tempdir().unwrap();
            let cfg = TorClientConfigBuilder::from_directories(state_dir, cache_dir)
                .build()
                .unwrap();
            let tor_client = TorClient::with_runtime(rt)
                .config(cfg.clone())
                .bootstrap_behavior(BootstrapBehavior::Manual)
                .create_unbootstrapped()
                .unwrap();
            tor_client
                .reconfigure(&cfg, Reconfigure::AllOrNothing)
                .unwrap();
        });
        tor_rtcompat::test_with_one_runtime!(|rt| async {
            let state_dir = tempfile::tempdir().unwrap();
            let cache_dir = tempfile::tempdir().unwrap();
            let cfg = TorClientConfigBuilder::from_directories(state_dir, cache_dir)
                .build()
                .unwrap();
            let tor_client = TorClient::with_runtime(rt)
                .config(cfg.clone())
                .bootstrap_behavior(BootstrapBehavior::Manual)
                .create_unbootstrapped_async()
                .await
                .unwrap();
            tor_client
                .reconfigure(&cfg, Reconfigure::AllOrNothing)
                .unwrap();
        });
    }
}