BlastDNS
An async rust library for DNS lookups. Can be used to perform simple, one-off lookups or bulk lookups in parallel with many resolvers, similar to massdns.
Features
BlastDNS is simultaneously a:
Benchmark
100K DNS lookups against local dnsmasq, with 100 workers:
| Library | Time | QPS | Success | Failed | vs dnspython |
|---|---|---|---|---|---|
| massdns | 1.687s | 71,898 | 100,000 | 0 | 28.87x |
| blastdns-cli | 1.732s | 64,942 | 100,000 | 0 | 26.07x |
| blastdns-python | 3.903s | 25,623 | 100,000 | 0 | 10.29x |
| dnspython | 40.149s | 2,491 | 100,000 | 0 | 1.00x |
CLI
The CLI mass-resolves hosts using a specified list of resolvers. It outputs to JSON.
# send all results to jq
|
# print only the raw IPv4 addresses
|
# load from stdin
|
# skip empty responses (e.g., NXDOMAIN with no answers)
|
# skip error responses (e.g., timeouts, connection failures)
|
CLI Help
$ blastdns --help
BlastDNS - Async DNS spray client
Usage: blastdns [OPTIONS] --resolvers <FILE> [HOSTS_TO_RESOLVE]
Arguments:
[HOSTS_TO_RESOLVE] File containing hostnames to resolve (one per line). Reads from stdin if not specified
Options:
--rdtype <RECORD_TYPE>
Record type to query (A, AAAA, MX, ...) [default: A]
--resolvers <FILE>
File containing DNS nameservers (one per line)
--threads-per-resolver <THREADS_PER_RESOLVER>
Worker threads per resolver [default: 2]
--timeout-ms <TIMEOUT_MS>
Per-request timeout in milliseconds [default: 1000]
--retries <RETRIES>
Retry attempts after a resolver failure [default: 10]
--purgatory-threshold <PURGATORY_THRESHOLD>
Consecutive errors before a worker is put into timeout [default: 10]
--purgatory-sentence-ms <PURGATORY_SENTENCE_MS>
How many milliseconds a worker stays in timeout [default: 1000]
--skip-empty
Don't show responses with no answers
--skip-errors
Don't show error responses
-h, --help
Print help
-V, --version
Print version
Example JSON output
BlastDNS outputs to JSON by default:
Debug Logging
BlastDNS uses the standard Rust tracing ecosystem. Enable debug logging by setting the RUST_LOG environment variable:
# Show debug logs from blastdns only
RUST_LOG=blastdns=debug
# Show debug logs from everything
RUST_LOG=debug
# Show trace-level logs for detailed internal behavior
RUST_LOG=blastdns=trace
Valid log levels (from least to most verbose): error, warn, info, debug, trace
Rust API
use ;
use StreamExt;
use RecordType;
use Duration;
// read DNS resolvers from a file (one per line -> vector of strings)
let resolvers = read_to_string
.expect
.lines
.map
.;
// create a new blastdns client with default config
let client = new.await?;
// or with custom config
let mut config = default;
config.threads_per_resolver = 5;
config.request_timeout = from_secs;
let client = with_config.await?;
// lookup a domain
let result = client.resolve.await?;
// print the result as serde JSON
println!;
// resolve_batch: process many hosts in parallel with bounded concurrency
// streams results back as they complete
let wordlist = ;
let mut stream = client.resolve_batch;
while let Some = stream.next.await
// resolve_batch_basic: simplified batch resolution with minimal output
// returns only (host, record_type, Vec<rdata>) - no full DNS response structures
// automatically filters out errors and empty responses
let wordlist = ;
let mut stream = client.resolve_batch_basic;
while let Some = stream.next.await
// resolve_multi: resolve multiple record types for a single host in parallel
let record_types = vec!;
let results = client.resolve_multi.await?;
for in results
Python API
The blastdns Python package is a thin wrapper around the Rust library.
# install python dependencies
# build and install the rust->python bindings
# run tests
To use it in Python, you can use the Client class:
=
=
# resolve: lookup a single host, returns a Pydantic model
= await
# resolve_batch: process many hosts in parallel with bounded concurrency
# streams results back as they complete
=
# resolve_batch_basic: simplified batch resolution with minimal output
# returns only (host, record_type, list[rdata]) - no full DNS response structures
# automatically filters out errors and empty responses
=
# e.g., "93.184.216.34" for A records
# resolve_multi: resolve multiple record types for a single host in parallel
=
= await
Python API Methods
-
Client.resolve(host, record_type=None) -> DNSResult: Lookup a single hostname. Defaults toArecords. Returns a PydanticDNSResultmodel with typed fields for easy access to the response data. -
Client.resolve_batch(hosts, record_type=None, skip_empty=False, skip_errors=False): Resolve many hosts in parallel. Takes an iterable of hostnames and streams back(host, result)tuples as results complete. Each result is either aDNSResultorDNSErrorPydantic model. Setskip_empty=Trueto filter out successful responses with no answers. Setskip_errors=Trueto filter out error responses. Useful for processing large lists of hosts. -
Client.resolve_batch_basic(hosts, record_type=None): Simplified batch resolution that returns only the essential data. Takes an iterable of hostnames and streams back(host, record_type, answers)tuples whereanswersis a list of rdata strings (e.g.,["93.184.216.34"]for A records,["10 aspmx.l.google.com."]for MX records). Automatically filters out errors and empty responses. Perfect for simple use cases where you just need the IP addresses or other record data without the full DNS response structure. -
Client.resolve_multi(host, record_types) -> dict[str, DNSResultOrError]: Resolve multiple record types for a single hostname in parallel. Takes a list of record type strings (e.g.,["A", "AAAA", "MX"]) and returns a dictionary keyed by record type. Each value is either aDNSResult(success) orDNSError(failure) Pydantic model.
MockClient for Testing
MockClient provides a drop-in replacement for Client that returns fabricated DNS responses without making real network requests. This is useful for testing code that depends on DNS lookups.
=
return
# MockClient implements the same interface as Client
= await
assert
assert == 1
# Test error cases
= await
assert ==
# Works with all Client methods
# ["93.184.216.34"]
MockClient supports all the same methods as Client (resolve, resolve_batch, resolve_batch_basic, resolve_multi) and returns the same Pydantic models.
Response Models
All methods return Pydantic V2 models for type safety and IDE autocomplete:
DNSResult: Successful DNS response withhostandresponsefieldsDNSError: Failed DNS lookup with anerrorfieldResponse: DNS message withheader,queries,answers,name_servers, etc.
ClientConfig exposes the knobs shown above (threads_per_resolver, request_timeout_ms, max_retries, purgatory_threshold, purgatory_sentence_ms) and validates them before handing them to the Rust core.
Architecture
BlastDNS is built on top of hickory-dns, but only makes use of the low-level Client API, not the Resolver API.
BlastDNS is designed to be faster the more resolvers you give it.
Beneath the hood of the BlastDNSClient, each resolver gets its own ResolverWorker tasks, with a configurable number of workers per resolver (default: 2, configurable via BlastDNSConfig.threads_per_resolver).
When a user calls BlastDNSClient::resolve, a new WorkItem is created which contains the request (host + rdtype) and a oneshot channel to hold the result. This WorkItem is put into a crossfire MPMC queue, to be picked up by the first available ResolverWorker. Workers are spawned immediately during client instantiation.
Testing
To run the full test suite including integration tests, you'll need a local DNS server running on 127.0.0.1:5353 and [::1]:5353.
Install dnsmasq:
Start the test DNS server:
Then run tests with:
When done, stop the test DNS server:
Linting
Rust
# Run clippy for lints
# Run rustfmt for formatting
Python
# Run ruff for lints
# Run ruff for formatting