use std::{error::Error, net::SocketAddr, path::PathBuf, time::Duration};
use clap::Parser;
use tailscale::{Config, Device};
use tracing_subscriber::filter::LevelFilter;
#[derive(clap::Parser)]
#[command(version, about)]
struct Args {
#[arg(short = 'c', long, default_value = "tsrs_keys.json")]
key_file: PathBuf,
#[arg(short = 'k', long, env = "TS_AUTH_KEY")]
auth_key: Option<String>,
#[arg(short = 'H', long, default_value = "peer_ping_example")]
hostname: Option<String>,
#[arg(long, env = "TS_CONTROL_URL")]
control_url: Option<url::Url>,
#[clap(short, long)]
peer: SocketAddr,
#[clap(short = 'i', long, default_value_t = 1.0)]
ping_interval_secs: f64,
}
#[tokio::main(flavor = "multi_thread")]
async fn main() -> Result<(), Box<dyn Error>> {
tracing_subscriber::fmt()
.with_env_filter(
tracing_subscriber::EnvFilter::builder()
.with_default_directive(LevelFilter::INFO.into())
.from_env_lossy(),
)
.init();
let args = Args::parse();
let mut config = Config::default_with_key_file(&args.key_file).await?;
config.requested_hostname = args.hostname;
if let Some(url) = args.control_url {
config.control_server_url = url;
}
let dev = Device::new(&config, args.auth_key).await?;
let sock = dev.udp_bind((dev.ipv4_addr().await?, 1234).into()).await?;
let mut ticker = tokio::time::interval(Duration::from_secs_f64(args.ping_interval_secs));
loop {
sock.send_to(args.peer, b"hello").await?;
ticker.tick().await;
}
}