pkarr 5.0.4

Public-Key Addressable Resource Records (Pkarr); publish and resolve DNS records over Mainline DHT
Documentation
//! Run an HTTP server listening on a Pkarr domain
//!
//! This server will _not_ be accessible from other networks
//! unless the provided IP is public and the port number is forwarded.

use axum::{routing::get, Router};
use axum_server::tls_rustls::RustlsConfig;
use tracing::Level;

use std::net::{SocketAddr, ToSocketAddrs};
use std::sync::Arc;

use clap::Parser;

use pkarr::{dns::rdata::SVCB, Client, Keypair, SignedPacket};

#[derive(Parser)]
#[command(author, version, about, long_about = None)]
struct Cli {
    /// IP address to listen on
    ip: String,
    /// Port number to listen no
    port: u16,
}

#[tokio::main]
async fn main() -> anyhow::Result<()> {
    tracing_subscriber::fmt().with_max_level(Level::INFO).init();

    let cli = Cli::parse();

    let addr = format!("{}:{}", cli.ip, cli.port)
        .to_socket_addrs()?
        .next()
        .ok_or(anyhow::anyhow!(
            "Could not convert IP and port to socket addresses"
        ))?;

    let keypair = Keypair::random();

    let client = Client::builder().build()?;

    // Run a server on Pkarr
    println!("Server listening on {addr}");

    // You should republish this every time the socket address change
    // and once an hour otherwise.
    publish_server_pkarr(&client, &keypair, &addr).await?;

    println!("Server running on https://{}", keypair.public_key());

    let server = axum_server::bind_rustls(
        addr,
        RustlsConfig::from_config(Arc::new(keypair.to_rpk_rustls_server_config())),
    );

    let app = Router::new().route("/", get(|| async { "Hello, world!" }));
    server.serve(app.into_make_service()).await?;

    Ok(())
}

async fn publish_server_pkarr(
    client: &Client,
    keypair: &Keypair,
    socket_addr: &SocketAddr,
) -> anyhow::Result<()> {
    let mut svcb = SVCB::new(0, ".".try_into().expect("infallible"));
    svcb.set_port(socket_addr.port());

    let signed_packet = SignedPacket::builder()
        .https(".".try_into().unwrap(), svcb, 60 * 60)
        .address(".".try_into().unwrap(), socket_addr.ip(), 60 * 60)
        .sign(keypair)?;

    client.publish(&signed_packet, None).await?;

    Ok(())
}