Skip to content

Commit

Permalink
feat: trace all resolved ips (#743) - WIP
Browse files Browse the repository at this point in the history
  • Loading branch information
fujiapple852 committed Nov 3, 2023
1 parent bebb30c commit 1a585ae
Show file tree
Hide file tree
Showing 6 changed files with 97 additions and 28 deletions.
19 changes: 15 additions & 4 deletions src/config.rs
Original file line number Diff line number Diff line change
Expand Up @@ -215,6 +215,7 @@ pub struct TrippyConfig {
pub tui_bindings: TuiBindings,
pub mode: Mode,
pub privilege_mode: PrivilegeMode,
pub dns_resolve_all: bool,
pub report_cycles: usize,
pub geoip_mmdb_file: Option<String>,
pub max_rounds: Option<usize>,
Expand Down Expand Up @@ -276,13 +277,16 @@ impl TryFrom<(Args, &Platform)> for TrippyConfig {
cfg_file_trace.unprivileged,
constants::DEFAULT_UNPRIVILEGED,
);

let privilege_mode = if unprivileged {
PrivilegeMode::Unprivileged
} else {
PrivilegeMode::Privileged
};

let dns_resolve_all = cfg_layer_bool_flag(
args.dns_resolve_all,
cfg_file_dns.dns_resolve_all,
constants::DEFAULT_DNS_RESOLVE_ALL,
);
let verbose = args.verbose;
let log_format = cfg_layer(
args.log_format,
Expand Down Expand Up @@ -497,7 +501,7 @@ impl TryFrom<(Args, &Platform)> for TrippyConfig {
validate_privilege(privilege_mode, has_privileges, needs_privileges)?;
validate_logging(mode, verbose)?;
validate_strategy(multipath_strategy, unprivileged)?;
validate_multi(mode, protocol, &args.targets)?;
validate_multi(mode, protocol, &args.targets, dns_resolve_all)?;
validate_ttl(first_ttl, max_ttl)?;
validate_max_inflight(max_inflight)?;
validate_read_timeout(read_timeout)?;
Expand Down Expand Up @@ -552,6 +556,7 @@ impl TryFrom<(Args, &Platform)> for TrippyConfig {
tui_bindings,
mode,
privilege_mode,
dns_resolve_all,
report_cycles,
geoip_mmdb_file,
max_rounds,
Expand Down Expand Up @@ -638,7 +643,13 @@ fn validate_strategy(strategy: MultipathStrategy, unprivileged: bool) -> anyhow:
}

/// We only allow multiple targets to be specified for the Tui and for `Icmp` tracing.
fn validate_multi(mode: Mode, protocol: TracerProtocol, targets: &[String]) -> anyhow::Result<()> {
fn validate_multi(
mode: Mode,
protocol: TracerProtocol,
targets: &[String],
_all_resolved_ips: bool,
) -> anyhow::Result<()> {
// TODO validate dns_resolve_all
match (mode, protocol) {
(Mode::Stream | Mode::Pretty | Mode::Markdown | Mode::Csv | Mode::Json, _)
if targets.len() > 1 =>
Expand Down
4 changes: 4 additions & 0 deletions src/config/cmd.rs
Original file line number Diff line number Diff line change
Expand Up @@ -118,6 +118,10 @@ pub struct Args {
#[arg(value_enum, short = 'r', long)]
pub dns_resolve_method: Option<DnsResolveMethod>,

/// Trace to all IPs resolved from DNS lookup [default: false]
#[arg(short = 'y', long)]
pub dns_resolve_all: bool,

/// The maximum time to wait to perform DNS queries [default: 5s]
#[arg(long)]
pub dns_timeout: Option<String>,
Expand Down
3 changes: 3 additions & 0 deletions src/config/constants.rs
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,9 @@ pub const DEFAULT_MODE: Mode = Mode::Tui;
/// The default value for `unprivileged`.
pub const DEFAULT_UNPRIVILEGED: bool = false;

/// The default value for `all_resolved_ips`.
pub const DEFAULT_DNS_RESOLVE_ALL: bool = false;

/// The default value for `log-format`.
pub const DEFAULT_LOG_FORMAT: LogFormat = LogFormat::Pretty;

Expand Down
1 change: 1 addition & 0 deletions src/config/file.rs
Original file line number Diff line number Diff line change
Expand Up @@ -118,6 +118,7 @@ pub struct ConfigStrategy {
#[serde(rename_all = "kebab-case", deny_unknown_fields)]
pub struct ConfigDns {
pub dns_resolve_method: Option<DnsResolveMethod>,
pub dns_resolve_all: Option<bool>,
pub dns_lookup_as_info: Option<bool>,
pub dns_timeout: Option<String>,
}
Expand Down
92 changes: 68 additions & 24 deletions src/main.rs
Original file line number Diff line number Diff line change
Expand Up @@ -53,12 +53,8 @@ fn main() -> anyhow::Result<()> {
let _guard = configure_logging(&cfg);
let resolver = start_dns_resolver(&cfg)?;
let geoip_lookup = create_geoip_lookup(&cfg)?;
let traces: Vec<_> = cfg
.targets
.iter()
.enumerate()
.map(|(i, target_host)| start_tracer(&cfg, target_host, platform.pid + i as u16, &resolver))
.collect::<anyhow::Result<Vec<_>>>()?;
let addrs = resolve_targets(&cfg.targets, cfg.dns_resolve_all, &resolver)?;
let traces = start_tracers(&cfg, &addrs, platform.pid)?;
Platform::drop_privileges()?;
run_frontend(&cfg, resolver, geoip_lookup, traces)?;
Ok(())
Expand Down Expand Up @@ -128,30 +124,71 @@ fn configure_logging(cfg: &TrippyConfig) -> Option<FlushGuard> {
None
}

/// Resolve targets.
fn resolve_targets(
targets: &[String],
dns_resolve_all: bool,
resolver: &DnsResolver,
) -> anyhow::Result<Vec<TargetInfo>> {
targets
.iter()
.flat_map(|target| match resolver.lookup(target) {
Ok(addrs) => {
if dns_resolve_all {
addrs
.into_iter()
.enumerate()
.map(|(i, addr)| {
Ok(TargetInfo {
hostname: format!("{} [{}]", target, i + 1),
addr,
})
})
.collect::<Vec<anyhow::Result<TargetInfo>>>()
.into_iter()
} else {
addrs
.into_iter()
.map(|addr| {
Ok(TargetInfo {
hostname: target.to_string(),
addr,
})
})
.take(1)
.collect::<Vec<anyhow::Result<TargetInfo>>>()
.into_iter()
}
}
Err(e) => {
vec![Err(anyhow!("failed to resolve target: {} ({})", target, e))].into_iter()
}
})
.collect::<anyhow::Result<Vec<_>>>()
}

/// Start all tracers.
fn start_tracers(
cfg: &TrippyConfig,
addrs: &[TargetInfo],
pid: u16,
) -> anyhow::Result<Vec<TraceInfo>> {
addrs
.iter()
.enumerate()
.map(|(i, TargetInfo { hostname, addr })| {
start_tracer(cfg, hostname, *addr, pid + i as u16)
})
.collect::<anyhow::Result<Vec<_>>>()
}

/// Start a tracer to a given target.
fn start_tracer(
cfg: &TrippyConfig,
target_host: &str,
target_addr: IpAddr,
trace_identifier: u16,
resolver: &DnsResolver,
) -> Result<TraceInfo, Error> {
let target_addr: IpAddr = resolver
.lookup(target_host)
.map_err(|e| anyhow!("failed to resolve target: {} ({})", target_host, e))?
.into_iter()
.find(|addr| {
matches!(
(cfg.addr_family, addr),
(TracerAddrFamily::Ipv4, IpAddr::V4(_)) | (TracerAddrFamily::Ipv6, IpAddr::V6(_))
)
})
.ok_or_else(|| {
anyhow!(
"failed to find an {:?} address for target: {}",
cfg.addr_family,
target_host
)
})?;
let source_addr = match cfg.source_addr {
None => SourceAddr::discover(target_addr, cfg.port_direction, cfg.interface.as_deref())?,
Some(addr) => SourceAddr::validate(addr)?,
Expand Down Expand Up @@ -292,6 +329,13 @@ fn make_tui_config(args: &TrippyConfig) -> TuiConfig {
)
}

/// Information about a tracing target.
#[derive(Debug, Clone)]
pub struct TargetInfo {
pub hostname: String,
pub addr: IpAddr,
}

/// Information about a `Trace` needed for the Tui, stream and reports.
#[derive(Debug, Clone)]
pub struct TraceInfo {
Expand Down
6 changes: 6 additions & 0 deletions trippy-config-sample.toml
Original file line number Diff line number Diff line change
Expand Up @@ -176,6 +176,12 @@ read-timeout = "10ms"
# cloudflare - Resolve using the Cloudflare `1.1.1.1` DNS service
dns-resolve-method = "system"

# Trace to all IPs resolved from DNS lookup (ICMP only) [default: false]
#
# When set to true a trace will be started for all IPs resolved for all given targets.
# When set to false a trace will be started for one arbitrarily chosen IP per given target.
dns_resolve_all = false

# Whether to lookup AS information [default: false]
#
# If enabled, AS (autonomous system) information is retrieved during DNS
Expand Down

0 comments on commit 1a585ae

Please sign in to comment.