A Python3 library for parsing and generating DNS messages
Project description
DNS-Messages
A Python3 library for parsing and generating DNS messages. Additionally, it provides a dns server which enables receiving and sending dns messages.
Currently, it only supports some types of resource records (A, AAAA, CNAME, PTR, TXT, SOA, MX, HINFO) but can be easily extended - feel free to create a pull request :wink: If a message contains an unsupported RR, the parser will ignore the data of this RR but will not break :smiley:
The logic to encode a message is very simple - no compression etc. is done.
Installation
Install the package with pip
pip3 install dns-messages
Parsing DNS Messages
Use the function from_bytes
of the DnsMessage
class to parse dns messages in raw bytes.
from dns_messages import DnsMessage
raw_bytes: bytes
parsed_message: DnsMessage = DnsMessage.from_bytes(raw_bytes)
Generating DNS Messages
Build a message using the provided classes DnsMessage
, OPCODE
, Question
, A
, RRType
etc.:
from dns_messages import DnsMessage, OPCODE, RRType, RRClass, Question
# build a simple query for the A record of "example.com"
message = DnsMessage(message_id=1, qr=1, op_code=OPCODE.QUERY)
question = Question(qname="example.com", qtype=RRType.A, qclass=RRClass.IN)
message.questions.append(question)
message_in_bytes: bytes = message.to_bytes()
DNS server
The package also provides a dns server DnsServer
which can easily be extended with your own logic by overwriting the functions _handle_broken_message
and _handle_received_message
.
from dns_messages import DnsServer, DnsPacketParsingException, DnsMessage
class TestServer(DnsServer):
def __init__(self, ip_address: str, port: int = 53):
super().__init__(ip_address=ip_address, port=port)
def _handle_broken_message(self, exception: DnsPacketParsingException, remote_ip: str, remote_port: int) -> None:
print('received package from {}:{} which could not be parsed: {}'.format(remote_ip, remote_port, exception))
def _handle_received_message(self, message: DnsMessage, remote_ip: str, remote_port: int) -> None:
# print out names of questions
for question in message.questions:
print(question.name)
Simple demo server
The package includes a simple demo for using the built-in dns server. You can start the dns demo server with the following three lines of code. The server prints out information about incoming dns packages.
from dns_messages.dns_server import DnsDemoServer
demo = DnsDemoServer(ip_address='127.0.0.1')
demo.start()
Here is the complete code of the demo server:
from .dns_server import DnsServer
from .. import DnsMessage, Question, ResourceRecord, A, AAAA
class DnsDemoServer(DnsServer):
def __init__(self, ip_address: str):
super().__init__(ip_address)
def _handle_received_message(self, message: DnsMessage, remote_ip: str, remote_port: int) -> None:
print(30*'-')
print('a new dns message was parsed')
print('message id: ', message.message_id)
print('is a query: ', message.is_query())
print('number of questions: ', message.number_of_questions())
print('number of questions: ', message.number_of_answer_records())
print('Questions:')
question: Question
for i, question in enumerate(message.questions):
print('\tquestion {}'.format(i+1))
print('\t- name: {}'.format(question.name))
print('\t- type: {}'.format(question.rr_type.name))
print('\t- class: {}'.format(question.rr_class.name))
print('Answers:')
answer: ResourceRecord
for i, answer in enumerate(message.answers_RRs):
print('\tanswer {}'.format(i + 1))
print('\t- name: {}'.format(answer.name))
print('\t- type: {}'.format(answer.get_RR_type()))
print('\t- class: {}'.format(answer.rr_class.name))
print('\t- ttl: {}'.format(answer.rr_class.name))
if isinstance(answer, A) or isinstance(answer, AAAA):
print('\t- IP: {}'.format(answer.ip_address))
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
File details
Details for the file dns-messages-1.0.1.tar.gz
.
File metadata
- Download URL: dns-messages-1.0.1.tar.gz
- Upload date:
- Size: 13.2 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/3.8.0 pkginfo/1.8.2 readme-renderer/32.0 requests/2.27.1 requests-toolbelt/0.9.1 urllib3/1.26.8 tqdm/4.62.3 importlib-metadata/4.11.1 keyring/23.5.0 rfc3986/2.0.0 colorama/0.4.4 CPython/3.9.0
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | dcc959efb78c6685ba079a0cef586f3e4619be48ae1332c0611681b39212285b |
|
MD5 | 2273d364d4de656b9369eb0e7c4182e6 |
|
BLAKE2b-256 | ee4d00f225f63ed2c239b956c8f882b71c979f3c28c437d3467dad90593a531c |
File details
Details for the file dns_messages-1.0.1-py3-none-any.whl
.
File metadata
- Download URL: dns_messages-1.0.1-py3-none-any.whl
- Upload date:
- Size: 22.5 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/3.8.0 pkginfo/1.8.2 readme-renderer/32.0 requests/2.27.1 requests-toolbelt/0.9.1 urllib3/1.26.8 tqdm/4.62.3 importlib-metadata/4.11.1 keyring/23.5.0 rfc3986/2.0.0 colorama/0.4.4 CPython/3.9.0
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 680a84adbb85aacefe35778041b0afec01caeba66aa3ccb7863086242f20cb59 |
|
MD5 | 782b23b2b4286c9ad2fab10ad5d5a17d |
|
BLAKE2b-256 | 358b764359c69062c895a149d7d43db06faf08ae42542fe43a8d01594d264d9d |