ntp_usg 1.1.0

Library for parsing and communicating over Network Time Protocol.
Documentation
ntp_usg-1.1.0 has been yanked.

ntp_usg

Documentation Crates.io License CI

A Network Time Protocol (NTP) packet parsing and client library written in Rust.

Features

  • 🔒 Safe & Secure: Zero unsafe code with #![forbid(unsafe_code)]
  • 📚 Well Documented: Comprehensive API documentation with examples
  • Configurable Timeouts: Control request timeouts for different network conditions
  • 🔄 Async Ready: Optional async support via Tokio (features = ["tokio"])
  • 🕐 Y2036 Safe: Era-aware timestamp handling for the NTP 32-bit rollover
  • 🌍 Multi-Server Support: Query multiple NTP servers for improved reliability
  • 🔐 Network Time Security: NTS (RFC 8915) with TLS 1.3 key establishment and AEAD authentication (features = ["nts"])
  • 📡 Continuous Client: Adaptive poll interval, multi-peer, and interleaved mode (RFC 9769) (features = ["tokio"])
  • 🌐 IPv6 Dual-Stack: Automatic IPv4/IPv6 socket binding
  • 🦀 Modern Rust: Edition 2024 with MSRV 1.93
  • Well Tested: CI/CD on Linux, macOS, and Windows

Installation

Add this to your Cargo.toml:

[dependencies]
ntp_usg = "1.0"

Minimum Supported Rust Version (MSRV): 1.93 Edition: 2024

Usage

Basic Example

use chrono::TimeZone;

fn main() {
    let address = "pool.ntp.org:123";
    let response = ntp::request(address).unwrap();
    let unix_time = ntp::unix_time::Instant::from(response.transmit_timestamp);
    let local_time = chrono::Local
        .timestamp_opt(unix_time.secs(), unix_time.subsec_nanos() as _)
        .unwrap();
    println!("Current time: {}", local_time);
}

Custom Timeout

use std::time::Duration;

let response = ntp::request_with_timeout("pool.ntp.org:123", Duration::from_secs(10))?;

Async with Tokio

Enable the tokio feature:

[dependencies]
ntp_usg = { version = "1.0", features = ["tokio"] }
tokio = { version = "1", features = ["rt-multi-thread", "macros"] }
#[tokio::main]
async fn main() -> Result<(), Box<dyn std::error::Error>> {
    let result = ntp::async_ntp::request("pool.ntp.org:123").await?;
    println!("Offset: {:.6} seconds", result.offset_seconds);
    Ok(())
}

Continuous Client

The continuous client polls servers with adaptive intervals and supports interleaved mode (RFC 9769):

use ntp::client::NtpClient;

#[tokio::main]
async fn main() -> Result<(), Box<dyn std::error::Error>> {
    let (client, mut state_rx) = NtpClient::builder()
        .server("pool.ntp.org:123")
        .min_poll(4)
        .max_poll(10)
        .build()
        .await?;

    tokio::spawn(client.run());

    // Wait for sync state updates.
    while state_rx.changed().await.is_ok() {
        let state = state_rx.borrow();
        println!("Offset: {:.6}s, Delay: {:.6}s", state.offset, state.delay);
    }
    Ok(())
}

NTS (Network Time Security)

Enable the nts feature for authenticated NTP:

[dependencies]
ntp_usg = { version = "1.0", features = ["nts"] }
tokio = { version = "1", features = ["rt-multi-thread", "macros"] }
use ntp::nts::NtsSession;

#[tokio::main]
async fn main() -> Result<(), Box<dyn std::error::Error>> {
    let mut session = NtsSession::from_ke("time.cloudflare.com").await?;
    let result = session.request().await?;
    println!("NTS offset: {:.6}s", result.offset_seconds);
    Ok(())
}

NTS Continuous Client

Combine NTS authentication with the continuous polling client:

use ntp::client::NtpClient;

#[tokio::main]
async fn main() -> Result<(), Box<dyn std::error::Error>> {
    let (client, mut state_rx) = NtpClient::builder()
        .nts_server("time.cloudflare.com")
        .min_poll(4)
        .max_poll(10)
        .build()
        .await?;

    tokio::spawn(client.run());

    while state_rx.changed().await.is_ok() {
        let state = state_rx.borrow();
        println!("Offset: {:.6}s, NTS: {}", state.offset, state.nts_authenticated);
    }
    Ok(())
}

Multiple Servers

See examples/multiple_servers.rs for a complete example of querying multiple NTP servers.

Examples

Run the included examples to see the library in action:

# Basic request example
cargo run --example request

# Custom timeout demonstration
cargo run --example timeout

# Query multiple servers
cargo run --example multiple_servers

# Detailed packet information
cargo run --example packet_details

# Async concurrent queries (requires tokio feature)
cargo run --example async_request --features tokio

# Continuous client with poll management (requires tokio feature)
cargo run --example continuous --features tokio

# NTS-authenticated request (requires nts feature)
cargo run --example nts_request --features nts

# NTS continuous client (requires nts feature)
cargo run --example nts_continuous --features nts

Roadmap

  • async support (tokio)
  • NTP era handling (Y2036)
  • IPv6 dual-stack support
  • Continuous client with adaptive polling
  • Interleaved mode (RFC 9769)
  • Network Time Security (RFC 8915)
  • no-std support
  • io-independent parsing
  • async-std support
  • setting system clocks
  • NTP server functionality

Contributing

Pull requests and issues are welcome! Please see our GitHub repository for more information.

License

ntp_usg is distributed under the terms of both the MIT license and the Apache License (Version 2.0).

See LICENSE-APACHE and LICENSE-MIT for details.