0x04 - NTP Request using Rust
I recently had some downtime at work and instead of dissociating I decided to have a go at learning Rust as it’s been on my list for a while. At a first glance there is an overwhelming amount of stuff to take in, even for an experienced developer. Nevertheless I decided to dive in and give it a go with a quick project. I’m one of those learners that requires a project to sink my teeth into rather than just reading the manual.
The Project
I was recently exposed to the NTP (Network Time Protocol) Pool project, which is essentially an API for time synchronisation which is provided by a cluster of servers worldwide. 1 A review of the protocol’s RFC documentation2 (and some Stack Overflow3) reveals that a timestamp can be queried by first sending a mostly empty 48-byte packet over UDP to the NTP server. The server then responds with a 48-byte packet with the same structure, now populated with various timestamp information outlined in the RFC spec. Neat!
The Code
Armed with a basic understanding of how the Network Time Protocol functioned, below is some Rust that requests and prints a timestamp provided by an NTP pool server.
Upon compiling and running the above code you should get the following output:
Note: The NTP pool will provide a different server every time you send a request, so expect the IP address to change.
Code Breakdown
To begin the code breakdown, I’ll first go through the main function sequentially, and then each function individually.
Main
These first few lines are defining an empty buffer of bytes (unsigned char) to send and receive the NTP packet. The buffer is defined as mutable so that values can be changed later in the program (by default variables in Rust are immutable, which means they cannot be modified). The first line is a constant, defined like you would use a #define
in C. The first entry in the buffer is then modified to contain the value 0x23
, which according to the spec means NTP Version 4 and Client mode (See Figures 8 and 102).
These lines are performing a DNS request to resolve an IP address from pool.ntp.org
, the result is then printed out.
Note: NTP operates on port 123.
This line allows the program to listen to all UDP requests on port 12345
, if this operation fails then a message is printed to the user.
These lines of code send the buffer to the ip address resolved earlier in the program, an error is printed upon failure. An assertion checks that the entire packet is sent.
These lines subsequently receive the packet back from the NTP pool server. An assertion checks that all 48 bytes are received.
Finally, these lines convert the received data to a unix time which is then printed to the user.
Functions/Libraries
For this project I’ve used the standard library’s networking library which will perform the DNS resolution and the UDP transactions. I’ve also used the chrono
module for converting the unix time to a readable string.
In this function we pass a url which is then converted to an IP address in addrs. This is plural because this call can resolve multiple address if an array is passed. The unwrap()
clause is a catch-all for any errors or panics that may occur. The second line then converts the first addr in the collection to a string, which is then returned.
This function first creates a slice of 4 bytes from the original buffer, specifically the number of seconds in the Transmit Timestamp from the server2. This is then converted to an unsigned int 32 which is then returned.
The function converts the NTP timestamp to a UNIX timestamp, the constant was lifted from the perl script here 4. It essentially subtracts the number of seconds between 1900 and 1970 from the NTP time, which begins at year 1900 rather than Unix’s 1970.
Finally, this function converts the unsigned int 32 unix time to a human readable string which is then printed.
Conclusion
This was a fun little project to dip my toes into Rust. I particularly liked how strict the compiler was, essentially not allowing you to compile any old shite. I look forward to continuing to use Rust in future projects!