mdiez / pyCZDS

A Python library for ICANN's CZDS.
GNU General Public License v3.0
4 stars 1 forks source link
dns python tld zone-files

pyCZDS – A Python client for ICANN's Centralized Zone Data Service (CZDS) API

This package allows you to seamlessly interact with ICANN's CZDS in Python, and download zone files for participating top level domains (e.g., .com, .net, and many others).

The Centralized Zone Data Service (CZDS) is an online portal where any interested party can request access to the Zone Files provided by participating generic Top-Level Domains (gTLDs).

(source)

Relevant links:

Installation

This package requires Python 3 and has been tested with Python 3.10.8. It requires the requests package. The library implements a client against the official API documentation which can be found under this link.

Install pyCZDS with the command pip install pyCZDS.

Exemplary flow

1. Create a new CZDSClient

from pyczds.client import CZDSClient

# replace username and password with actual credentials
c = CZDSClient(username, password)

2. Retrieve all URLs for the zone files accessible under the account

zonefile_urls = c.get_zonefiles_list()

3. From the list of URLs, select the zone file of interest

com_zonefile_url = next((url for url in zonefile_urls if "com.zone" in url), None)

4. Optionally, inspect the current header of the selected zonefile

This is useful if, for example, you have downloaded the zonefile for a zone previously and want to understand if it has been updated in the meantime.

from datetime import datetime, timezone

com_zonefile_headers = c.head_zonefile(com_zonefile_url)

# check if the zonefile is from today
com_zonefile_headers['parsed']['last-modified'].date() == datetime.now(timezone.utc).date()

5. Download the zonefile

The following command will download the zone file for the .com zone in the current directory to the file "com.txt.gz".

c.get_zonefile(com_zonefile_url)

Detailed description

Usage

The library supports the following actions:

Instantiating a client

Use the following pyCZDS – An API client for ICANN's Centralized Zone Data Service (CZDS)code to create a new CZDSClient object:

from pyczds.client import CZDSClient

# replace username and password with actual credentials
c = CZDSClient(username, password)

The client handles the authentication with the API transparently. It will authenticate with the first call of any method, and will retain the acquired token for subsequent requests. When the token expires, it will renew the authentication automatically.

Getting zone file download URLs

The following command will retrieve a list of all zone files the account is authorized to access. It returns a list with the respective URLs.

print(c.get_zonefiles_list())
# [
    'https://czds-download-api.icann.org/czds/downloads/net.zone',
    ...
    'https://czds-download-api.icann.org/czds/downloads/com.zone'
]

Access to additional zone files can be requested online under this link.

Requesting the headers for a zone file

Using one of the links received via get_zonefiles_list(), the following command retrieves the headers for a specified zonefile. It returns a dict (more specifically, a requests.models.CaseInsensitiveDict):

print(c.head_zonefile('https://czds-download-api.icann.org/czds/downloads/vision.zone'))
# {
    'Date': 'Fri, 16 Dec 2022 19:42:58 GMT',
    ...
    'Last-Modified': 'Fri, 16 Dec 2022 01:29:08 GMT',
    ...
    'Content-Disposition': 'attachment;filename=vision.txt.gz',
    ...
    'Content-Length': '602034',
    ...
}

To facilitate further work with the metadata, the dictionary not only contains the raw HTTP headers, but also a subdict parsed (which is a requests.models.CaseInsensitiveDict, too), which contains a selection of the headers parsed in suitable data types:

print(c.head_zonefile('https://czds-download-api.icann.org/czds/downloads/vision.zone'))
# {
    'Last-Modified': 'Sat, 17 Dec 2022 00:17:25 GMT',
    'Content-Disposition': 'attachment;filename=net.txt.gz',
    'Content-Length': '481167941',
    ...
    'parsed': {
        'last-modified': datetime.datetime(2022, 12, 17, 0, 17, 25, tzinfo=datetime.timezone.utc),  # datetime.datetime
        'content-length': 481167941,  # int
        'filename': 'net.txt.gz'  # string
    }
}

Having the values available in parsed form makes it easier to compare different headers of a zone file. For example, you can easily determine whether a zonefile has been updated since the last retrieval by comparing:

headers_old['parsed']['last-modified'] < headers_new['parsed']['last_modified']
# True

Downloading a zone file

The following command will download a specified zone file:

c.get_zonefile('https://czds-download-api.icann.org/czds/downloads/vision.zone', download_dir='zonefiles/', filename='vision_zonefile')

The following parameters are optional:

Troubleshooting

Should you encounter any errors, a good first step is to increase the logging level to debug and then analyze the output.

import logging

from pyczds import client

logger = logging.getLogger()
logger.setLevel(logging.DEBUG)

# Run the problematic command
c = client.CZDSClient(username, password)

Tests

The entire codebase is covered. Run the test from the tests directory. Before you do so, make sure you set username and password in tests/test_pyczds.py:

class TestPyCZDS(unittest.TestCase):
    def setUp(self):
        # In order to run these tests, enter your valid username and password for the CZDS website here.
        # Please note that these tests do not change any data.

        # TODO PASTE USERNAME AND PASSWORD HERE
        username = ''
        password = ''
        # END TODO

Legal disclaimer

I am a hobby enthusiast and am neither affiliated with ICANN, nor is this library endorsed by ICANN.

Links and further information