diff options
author | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-07 19:33:14 +0000 |
---|---|---|
committer | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-07 19:33:14 +0000 |
commit | 36d22d82aa202bb199967e9512281e9a53db42c9 (patch) | |
tree | 105e8c98ddea1c1e4784a60a5a6410fa416be2de /third_party/rust/dns-parser/examples/sync_tcp_client.rs | |
parent | Initial commit. (diff) | |
download | firefox-esr-36d22d82aa202bb199967e9512281e9a53db42c9.tar.xz firefox-esr-36d22d82aa202bb199967e9512281e9a53db42c9.zip |
Adding upstream version 115.7.0esr.upstream/115.7.0esr
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'third_party/rust/dns-parser/examples/sync_tcp_client.rs')
-rw-r--r-- | third_party/rust/dns-parser/examples/sync_tcp_client.rs | 77 |
1 files changed, 77 insertions, 0 deletions
diff --git a/third_party/rust/dns-parser/examples/sync_tcp_client.rs b/third_party/rust/dns-parser/examples/sync_tcp_client.rs new file mode 100644 index 0000000000..732e8d3bfb --- /dev/null +++ b/third_party/rust/dns-parser/examples/sync_tcp_client.rs @@ -0,0 +1,77 @@ +extern crate dns_parser; + +use std::env; +use std::error::Error; +use std::io::{Read, Write}; +use std::net::TcpStream; +use std::process; + + +use dns_parser::{Builder, Packet, RData, ResponseCode}; +use dns_parser::rdata::a::Record; +use dns_parser::{QueryType, QueryClass}; + + +fn main() { + let mut code = 0; + for name in env::args().skip(1) { + match resolve(&name) { + Ok(()) => {}, + Err(e) => { + eprintln!("Error resolving {:?}: {}", name, e); + code = 1; + } + } + } + process::exit(code); +} + +fn resolve(name: &str) -> Result<(), Box<Error>> { + let mut conn = TcpStream::connect("127.0.0.1:53")?; + let mut builder = Builder::new_query(1, true); + builder.add_question(name, false, QueryType::A, QueryClass::IN); + let packet = builder.build().map_err(|_| "truncated packet")?; + let psize = [((packet.len() >> 8) & 0xFF) as u8, + (packet.len() & 0xFF) as u8]; + conn.write_all(&psize[..])?; + conn.write_all(&packet)?; + let mut buf = vec![0u8; 4096]; + let mut off = 0; + let pkt = loop { + if buf.len() - off < 4096 { + buf.extend(&[0u8; 4096][..]); + } + match conn.read(&mut buf[off..]) { + Ok(num) => { + off += num; + if off < 2 { continue; } + let bytes = ((buf[0] as usize) << 8) | buf[1] as usize; + if off < bytes + 2 { + continue; + } + if num == 0 { + return Err("Partial packet received".into()); + } + break Packet::parse(&buf[2..off])?; + } + Err(e) => { + return Err(Box::new(e)); + } + } + }; + if pkt.header.response_code != ResponseCode::NoError { + return Err(pkt.header.response_code.into()); + } + if pkt.answers.len() == 0 { + return Err("No records received".into()); + } + for ans in pkt.answers { + match ans.data { + RData::A(Record(ip)) => { + println!("{}", ip); + } + _ => {} // ignore + } + } + Ok(()) +} |