Skip to content

Navigation Menu

Sign in
Appearance settings

Search code, repositories, users, issues, pull requests...

Provide feedback

We read every piece of feedback, and take your input very seriously.

Saved searches

Use saved searches to filter your results more quickly

Sign up
Appearance settings

filipsalo/surblclient

Repository files navigation

PyPI - Version Test suite status

surblclient

A small client library for querying SURBL, URIBL, and other RBL-style blocklists over DNS.

Give it a domain (or URL host, or IP address) and it tells you whether the domain is listed, and on which sub-lists.

Installation

uv add surblclient

Or, from a checkout:

uv pip install .

Usage

>>> from surblclient import surbl
>>> "test.surbl.org" in surbl
True
>>> surbl.lookup("test.surbl.org")
('test.surbl.org', ['ph', 'mw', 'abuse', 'cr'])
>>> "google.com" in surbl
False

lookup resolves the base domain before querying, so subdomains work too:

>>> surbl.lookup("foo.bar.baz.test.surbl.org")
('test.surbl.org', ['ph', 'mw', 'abuse', 'cr'])

It returns a (base_domain, lists) tuple on a hit, False when the domain is confirmed not listed, and None when the answer is unknown — a temporary DNS error, or the service refusing the query (see below).

The same interface is available for URIBL and the Spamhaus DBL:

>>> from surblclient import uribl, spamhausdbl
>>> "test.uribl.com" in uribl
True
>>> spamhausdbl.lookup("dbltest.com")
('dbltest.com', ['bad'])

The DBL labels a hit as either "bad" (inherently bad / safe to block) or "abused-legit" (an otherwise-good domain seen in abuse — meant for scoring, not outright blocking), reflecting Spamhaus's two return-code ranges.

Note that in can only return a bool, so it collapses the None (unknown/refused) case into False. If you need to distinguish "not listed" from "couldn't check", use lookup() and test for None:

result = surbl.lookup(domain)
if result is None:
 ... # unknown — do NOT treat as clean (often a blocked resolver)
elif result is False:
 ... # confirmed not listed
else:
 base, lists = result # listed

A single query can return several 127.0.0.x records (one per sublist); per RFC 5782 §6 a client must treat any returned record as a listing, so this library reads them all and combines them (bit masks for SURBL/URIBL, value-range tests for the DBL — both per §6).

Resolver requirements

SURBL, URIBL, and the Spamhaus DBL all refuse queries that arrive via public/shared DNS resolvers (Google Public DNS, OpenDNS, Cloudflare, Quad9, and most ISP caching resolvers) and rate-limit heavy users. A refused query comes back as an error sentinel (127.0.0.1 for SURBL/URIBL; 127.255.255.254 "public resolver" or 127.255.255.255 "excessive queries" for the DBL), which this library reports as None (unknown) — so on a public resolver every lookup silently returns "unknown" and the library can't do its job.

This is the services' documented anti-abuse policy, not a bug in this library:

  • URIBL — "All queries that we refuse, we return a 127.0.0.1 response to ... Public DNS providers such as OpenDNS or Google Public DNS are effected due to the high volume of queries they generate, as are many other internet service providers (ISP) that use caching nameservers ..." (uribl.com/refused.shtml)
  • SURBL — "If you get a result of 127.0.0.1 when doing a DNS query into the public nameservers, then it means your access is blocked ... A good administrative solution is to run a local caching nameserver ..." (surbl.org/faq/guidelines)
  • Spamhaus — public resolvers are blocked with return code 127.255.255.254 ("Query via public/open resolver"); high-volume use needs the Data Query Service (DQS) or rsync feed. (spamhaus.org/faqs/dnsbl-usage)

To use this library reliably, run your own recursive resolver (e.g. unbound) on the machine doing the checks and point it at the DNS roots, then resolve through 127.0.0.1. High volume use needs a data feed / datafeed instead of the public DNS mirrors; the free public service has usage limits (broadly, fewer than 1,000 users or 250,000 messages/day).

Development

This project is managed with uv.

uv sync # set up the environment
uv run python -m unittest -v # run the tests (mocked; no network)
uv build # build the wheel and sdist

The default test run mocks DNS, so it is deterministic and offline. The live integration tests in tests/test_live.py hit the real services and are skipped unless you opt in — and they only pass through a non-public resolver (see Resolver requirements):

SURBL_LIVE_TESTS=1 uv run python -m unittest -v

License

MIT — see LICENSE.

About

Read-only mirror of https://codeberg.org/filipsalo/surblclient — canonical repo and issues live there.

Resources

License

Stars

Watchers

Forks

Packages

Contributors

Languages

AltStyle によって変換されたページ (->オリジナル) /