Skip to content

Commit b02fb81

Browse files
Refactor mullvad-relay-selector
Implement a system for choosing appropriate relay(s) built on 'queries'. A query is a set of constraints which dictates which relay(s) that *can* be chosen by the relay selector. The user's settings can naturally be expressed as a query. The semantics of merging two queries in a way that always prefer user settings is defined by the new `Intersection` trait. Decrust `mullvad-relay-selector` by splitting it up into several modules - `query.rs`: Definition of a query on different types of relays. This module is integral to the new API of `mullvad-relay-selector` - `matcher.rs`: Logic for filtering out candidate relays based on a query. - `detailer.rs`: Logic for deriving connection details for the selected relay. - `tests/`: Integration tests for the new relay selector. These tests only use the public APIs of `RelaySelector` and make sure that the output matches the expected output in different scenarios.
1 parent 9c035ee commit b02fb81

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

46 files changed

+3847
-3128
lines changed

Cargo.lock

+15-3
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

Cargo.toml

+2
Original file line numberDiff line numberDiff line change
@@ -68,6 +68,8 @@ chrono = { version = "0.4.26", default-features = false}
6868
clap = { version = "4.4.18", features = ["cargo", "derive"] }
6969
once_cell = "1.13"
7070

71+
# Test dependencies
72+
proptest = "1.4"
7173

7274
[profile.release]
7375
opt-level = 3

mullvad-cli/src/cmds/bridge.rs

+3-2
Original file line numberDiff line numberDiff line change
@@ -2,9 +2,10 @@ use anyhow::{bail, Result};
22
use clap::Subcommand;
33
use mullvad_management_interface::MullvadProxyClient;
44
use mullvad_types::{
5+
constraints::Constraint,
56
relay_constraints::{
6-
BridgeConstraintsFormatter, BridgeState, BridgeType, Constraint, LocationConstraint,
7-
Ownership, Provider, Providers,
7+
BridgeConstraintsFormatter, BridgeState, BridgeType, LocationConstraint, Ownership,
8+
Provider, Providers,
89
},
910
relay_list::RelayEndpointData,
1011
};

mullvad-cli/src/cmds/custom_list.rs

+1-2
Original file line numberDiff line numberDiff line change
@@ -3,8 +3,7 @@ use anyhow::{anyhow, bail, Result};
33
use clap::Subcommand;
44
use mullvad_management_interface::MullvadProxyClient;
55
use mullvad_types::{
6-
relay_constraints::{Constraint, GeographicLocationConstraint},
7-
relay_list::RelayList,
6+
constraints::Constraint, relay_constraints::GeographicLocationConstraint, relay_list::RelayList,
87
};
98

109
#[derive(Subcommand, Debug)]

mullvad-cli/src/cmds/debug.rs

+4-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,9 @@
11
use anyhow::Result;
22
use mullvad_management_interface::MullvadProxyClient;
3-
use mullvad_types::relay_constraints::{Constraint, RelayConstraints, RelaySettings};
3+
use mullvad_types::{
4+
constraints::Constraint,
5+
relay_constraints::{RelayConstraints, RelaySettings},
6+
};
47

58
#[derive(clap::Subcommand, Debug)]
69
pub enum DebugCommands {

mullvad-cli/src/cmds/obfuscation.rs

+3-2
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,9 @@
11
use anyhow::Result;
22
use clap::Subcommand;
33
use mullvad_management_interface::MullvadProxyClient;
4-
use mullvad_types::relay_constraints::{
5-
Constraint, ObfuscationSettings, SelectedObfuscation, Udp2TcpObfuscationSettings,
4+
use mullvad_types::{
5+
constraints::Constraint,
6+
relay_constraints::{ObfuscationSettings, SelectedObfuscation, Udp2TcpObfuscationSettings},
67
};
78

89
#[derive(Subcommand, Debug)]

mullvad-cli/src/cmds/relay.rs

+5-4
Original file line numberDiff line numberDiff line change
@@ -3,10 +3,11 @@ use clap::Subcommand;
33
use itertools::Itertools;
44
use mullvad_management_interface::MullvadProxyClient;
55
use mullvad_types::{
6+
constraints::{Constraint, Match},
67
location::{CountryCode, Location},
78
relay_constraints::{
8-
Constraint, GeographicLocationConstraint, LocationConstraint, LocationConstraintFormatter,
9-
Match, OpenVpnConstraints, Ownership, Provider, Providers, RelayConstraints, RelayOverride,
9+
GeographicLocationConstraint, LocationConstraint, LocationConstraintFormatter,
10+
OpenVpnConstraints, Ownership, Provider, Providers, RelayConstraints, RelayOverride,
1011
RelaySettings, TransportPort, WireguardConstraints,
1112
},
1213
relay_list::{RelayEndpointData, RelayListCountry},
@@ -318,7 +319,7 @@ impl Relay {
318319

319320
print_option!(
320321
"Multihop state",
321-
if constraints.wireguard_constraints.use_multihop {
322+
if constraints.wireguard_constraints.multihop() {
322323
"enabled"
323324
} else {
324325
"disabled"
@@ -679,7 +680,7 @@ impl Relay {
679680
wireguard_constraints.ip_version = ipv;
680681
}
681682
if let Some(use_multihop) = use_multihop {
682-
wireguard_constraints.use_multihop = *use_multihop;
683+
wireguard_constraints.use_multihop(*use_multihop);
683684
}
684685
match entry_location {
685686
Some(EntryArgs::Location(location_args)) => {

mullvad-cli/src/cmds/relay_constraints.rs

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,8 @@
11
use clap::Args;
22
use mullvad_types::{
3+
constraints::Constraint,
34
location::{CityCode, CountryCode, Hostname},
4-
relay_constraints::{Constraint, GeographicLocationConstraint, LocationConstraint},
5+
relay_constraints::{GeographicLocationConstraint, LocationConstraint},
56
};
67

78
#[derive(Args, Debug, Clone)]

mullvad-cli/src/cmds/tunnel.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@ use anyhow::Result;
22
use clap::Subcommand;
33
use mullvad_management_interface::MullvadProxyClient;
44
use mullvad_types::{
5-
relay_constraints::Constraint,
5+
constraints::Constraint,
66
wireguard::{QuantumResistantState, RotationInterval, DEFAULT_ROTATION_INTERVAL},
77
};
88

mullvad-daemon/src/custom_list.rs

+3-4
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,8 @@
11
use crate::{new_selector_config, Daemon, Error, EventListener};
22
use mullvad_types::{
3+
constraints::Constraint,
34
custom_list::{CustomList, Id},
4-
relay_constraints::{
5-
BridgeState, Constraint, LocationConstraint, RelaySettings, ResolvedBridgeSettings,
6-
},
5+
relay_constraints::{BridgeState, LocationConstraint, RelaySettings, ResolvedBridgeSettings},
76
};
87
use talpid_types::net::TunnelType;
98

@@ -133,7 +132,7 @@ where
133132
{
134133
match endpoint.tunnel_type {
135134
TunnelType::Wireguard => {
136-
if relay_settings.wireguard_constraints.use_multihop {
135+
if relay_settings.wireguard_constraints.multihop() {
137136
if let Constraint::Only(LocationConstraint::CustomList { list_id }) =
138137
&relay_settings.wireguard_constraints.entry_location
139138
{

mullvad-daemon/src/device/mod.rs

+22-22
Original file line numberDiff line numberDiff line change
@@ -1348,12 +1348,10 @@ impl TunnelStateChangeHandler {
13481348
#[cfg(test)]
13491349
mod test {
13501350
use super::{Error, TunnelStateChangeHandler, WG_DEVICE_CHECK_THRESHOLD};
1351-
use mullvad_relay_selector::RelaySelector;
13521351
use std::sync::{
13531352
atomic::{AtomicBool, Ordering},
13541353
Arc,
13551354
};
1356-
use talpid_types::net::TunnelType;
13571355

13581356
const TIMEOUT_ERROR: Error = Error::OtherRestError(mullvad_api::rest::Error::TimeoutError);
13591357

@@ -1437,24 +1435,26 @@ mod test {
14371435
);
14381436
}
14391437

1440-
/// Test whether the relay selector selects wireguard often enough, given no special
1441-
/// constraints, to verify that the device is valid
1442-
#[test]
1443-
fn test_validates_by_default() {
1444-
for attempt in 0.. {
1445-
let should_validate =
1446-
TunnelStateChangeHandler::should_check_validity_on_attempt(attempt);
1447-
let (_, _, tunnel_type) =
1448-
RelaySelector::preferred_tunnel_constraints(attempt.try_into().unwrap());
1449-
assert_eq!(
1450-
tunnel_type,
1451-
TunnelType::Wireguard,
1452-
"failed on attempt {attempt}"
1453-
);
1454-
if should_validate {
1455-
// Now that we've triggered a device check, we can give up
1456-
break;
1457-
}
1458-
}
1459-
}
1438+
// TODO(markus): `preferred_tunnel_constraints` is slated for removal - consider writing a new test which
1439+
// does not depend on relay selector internals.
1440+
// /// Test whether the relay selector selects wireguard often enough, given no special
1441+
// /// constraints, to verify that the device is valid
1442+
// #[test]
1443+
// fn test_validates_by_default() {
1444+
// for attempt in 0.. {
1445+
// let should_validate =
1446+
// TunnelStateChangeHandler::should_check_validity_on_attempt(attempt);
1447+
// let (_, _, tunnel_type) =
1448+
// RelaySelector::preferred_tunnel_constraints(attempt.try_into().unwrap());
1449+
// assert_eq!(
1450+
// tunnel_type,
1451+
// TunnelType::Wireguard,
1452+
// "failed on attempt {attempt}"
1453+
// );
1454+
// if should_validate {
1455+
// // Now that we've triggered a device check, we can give up
1456+
// break;
1457+
// }
1458+
// }
1459+
// }
14601460
}

mullvad-daemon/src/lib.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1105,7 +1105,7 @@ where
11051105
// Note that `Constraint::Any` corresponds to just IPv4
11061106
matches!(
11071107
relay_constraints.wireguard_constraints.ip_version,
1108-
mullvad_types::relay_constraints::Constraint::Only(IpVersion::V6)
1108+
mullvad_types::constraints::Constraint::Only(IpVersion::V6)
11091109
)
11101110
} else {
11111111
false

mullvad-daemon/src/migrations/v1.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
use super::Result;
2-
use mullvad_types::{relay_constraints::Constraint, settings::SettingsVersion};
2+
use mullvad_types::{constraints::Constraint, settings::SettingsVersion};
33
use serde::{Deserialize, Serialize};
44

55
// ======================================================

mullvad-daemon/src/migrations/v4.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
use super::{Error, Result};
2-
use mullvad_types::{relay_constraints::Constraint, settings::SettingsVersion};
2+
use mullvad_types::{constraints::Constraint, settings::SettingsVersion};
33
use serde::{Deserialize, Serialize};
44

55
// ======================================================

mullvad-daemon/src/migrations/v5.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
use super::{Error, Result};
2-
use mullvad_types::{relay_constraints::Constraint, settings::SettingsVersion};
2+
use mullvad_types::{constraints::Constraint, settings::SettingsVersion};
33
use serde::{Deserialize, Serialize};
44

55
// ======================================================

mullvad-daemon/src/migrations/v6.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
use super::{Error, Result};
2-
use mullvad_types::{relay_constraints::Constraint, settings::SettingsVersion};
2+
use mullvad_types::{constraints::Constraint, settings::SettingsVersion};
33
use serde::{Deserialize, Serialize};
44

55
// ======================================================

mullvad-daemon/src/settings/mod.rs

+5-8
Original file line numberDiff line numberDiff line change
@@ -376,16 +376,13 @@ impl<'a> Display for SettingsSummary<'a> {
376376
write!(f, ", wg ip version: {ip_version}")?;
377377
}
378378

379-
let multihop = matches!(
380-
relay_settings,
379+
let multihop = match relay_settings {
381380
RelaySettings::Normal(RelayConstraints {
382-
wireguard_constraints: WireguardConstraints {
383-
use_multihop: true,
384-
..
385-
},
381+
wireguard_constraints,
386382
..
387-
})
388-
);
383+
}) => wireguard_constraints.multihop(),
384+
_ => false,
385+
};
389386

390387
write!(
391388
f,

0 commit comments

Comments
 (0)