Files
libsignal/rust/net/examples/dns_over_https.rs
2024-08-28 14:52:47 -04:00

83 lines
2.6 KiB
Rust

//
// Copyright 2024 Signal Messenger, LLC.
// SPDX-License-Identifier: AGPL-3.0-only
//
use std::convert::Infallible;
use std::num::NonZeroU16;
use std::sync::Arc;
use clap::Parser;
use futures_util::StreamExt;
use libsignal_net::infra::certs::RootCertificates;
use libsignal_net::infra::dns::custom_resolver::DnsTransport;
use libsignal_net::infra::dns::dns_lookup::DnsLookupRequest;
use libsignal_net::infra::dns::dns_transport_doh::DohTransport;
use libsignal_net::infra::host::Host;
use libsignal_net::infra::{
ConnectionParams, HttpRequestDecoratorSeq, RouteType, TransportConnectionParams,
};
#[derive(Parser, Debug)]
struct Args {
/// disable IPv6 address group
#[arg(long, default_value = "false")]
no_ipv6: bool,
/// domain to lookup
#[arg(long, default_value = "chat.signal.org")]
domain: String,
/// address of the name server
#[arg(long, default_value = "1.1.1.1", value_parser=parse_host)]
ns_address: Host<Arc<str>>,
/// port of the name server
#[arg(long, default_value = "443")]
ns_port: u16,
}
fn parse_host(s: &str) -> Result<Host<Arc<str>>, Infallible> {
Ok(Host::parse_as_ip_or_domain(s))
}
#[tokio::main]
async fn main() {
let _ = env_logger::builder()
.filter_level(log::LevelFilter::Debug)
.try_init();
let args = Args::parse();
let host = args.ns_address.to_string().into();
let connection_params = ConnectionParams {
route_type: RouteType::Direct,
http_request_decorator: HttpRequestDecoratorSeq::default(),
transport: TransportConnectionParams {
sni: Arc::clone(&host),
tcp_host: args.ns_address,
port: NonZeroU16::try_from(args.ns_port).expect("valid port value"),
certs: RootCertificates::Native,
},
http_host: host,
connection_confirmation_header: None,
};
let doh_transport = DohTransport::connect(connection_params.clone(), !args.no_ipv6)
.await
.expect("connected to the DNS server");
log::info!(
"successfully connected to the DNS server at {:?}",
connection_params
);
let request = DnsLookupRequest {
hostname: Arc::from(args.domain),
ipv6_enabled: !args.no_ipv6,
};
log::info!("sending DNS request: {:?}", request);
let mut stream = doh_transport.send_queries(request).await.unwrap();
let next_response = stream.next().await;
log::info!("received first response from DNS: [{:?}]", next_response);
let next_response = stream.next().await;
log::info!("received second response from DNS: [{:?}]", next_response);
}