Mirrors decnet/geoip/ end-to-end: paths/base/factory/lookup at the package level, iptoasn/ subpackage holds the data-source-specific fetch+parse+provider. AsnLookup is bisect-indexed over (start, end, AsnInfo) ranges with a pickled cache invalidated on raw-file mtime bump. Why iptoasn (and not bgp.tools / Team Cymru): public-domain dump, zero attribution, no UA mandate, daily refresh — keeps DECNET stealth intact (the geoip/rir module's "never identify as DECNET" comment applies the same way here). bgp.tools' ToS would have required an identifying UA, conflicting with feedback_stealth. Public surface: decnet.asn.enrich_ip(ip) -> (asn, name, source) or all-None on miss/disabled. Same shape as decnet.geoip.enrich_ip so the profiler can compose them in one call site.
34 lines
1.0 KiB
Python
34 lines
1.0 KiB
Python
"""ASN provider protocol — mirror of :mod:`decnet.geoip.base`.
|
|
|
|
Concrete providers (e.g. :mod:`decnet.asn.iptoasn`) implement this.
|
|
Callers must go through :func:`decnet.asn.factory.get_provider`; never
|
|
import a concrete provider class directly.
|
|
"""
|
|
from __future__ import annotations
|
|
|
|
from abc import ABC, abstractmethod
|
|
from pathlib import Path
|
|
from typing import Sequence
|
|
|
|
from decnet.asn.lookup import AsnLookup
|
|
|
|
|
|
class Provider(ABC):
|
|
"""Abstract IP→ASN data provider."""
|
|
|
|
#: Short tag written to ``Attacker.asn_source`` (e.g. ``'iptoasn'``).
|
|
name: str
|
|
|
|
@abstractmethod
|
|
def refresh(self) -> None:
|
|
"""Download / regenerate the provider's raw data files."""
|
|
|
|
@abstractmethod
|
|
def build_lookup(self) -> AsnLookup:
|
|
"""Parse the on-disk data files and return a ready-to-query lookup."""
|
|
|
|
@abstractmethod
|
|
def data_paths(self) -> Sequence[Path]:
|
|
"""Return the list of files this provider manages — used for staleness
|
|
detection. Order is not significant."""
|