Files
Abtin Keshavarzian 20aefc2215 [dnssd] support generic record queries via discovery proxy and SRP (#11357)
This commit enhances the OpenThread DNSSD name server/resolver and its
native Discovery Proxy to support queries for arbitrary record
types.

To enable this, a new set of `otPlatDnssd` APIs are introduced for
generic `RecordQuerier`. These APIs mirror the existing APIs in the
OpenThread native mDNS module, allowing direct use of the native mDNS
implementation.

The discovery proxy implementation is updated to start and stop the
mDNS `RecordQuerier` when receiving a query for an arbitrary record
type, passing the first response record back to the client.

The unit tests `test_dnssd_discovery_proxy` and `test_dns_client`
are updated to cover all the newly added behaviors in discovery proxy.
2025-04-23 12:14:05 -07:00

298 lines
12 KiB
Python
Executable File

#!/usr/bin/env python3
#
# Copyright (c) 2021, The OpenThread Authors.
# All rights reserved.
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions are met:
# 1. Redistributions of source code must retain the above copyright
# notice, this list of conditions and the following disclaimer.
# 2. Redistributions in binary form must reproduce the above copyright
# notice, this list of conditions and the following disclaimer in the
# documentation and/or other materials provided with the distribution.
# 3. Neither the name of the copyright holder nor the
# names of its contributors may be used to endorse or promote products
# derived from this software without specific prior written permission.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
# POSSIBILITY OF SUCH DAMAGE.
#
import ipaddress
import typing
import unittest
import config
import thread_cert
# Test description:
#
# This test verifies DNS-SD server and DNS client behavior
# (browsing for services and/or subtype services, resolving an
# address, or resolving a service). It also indirectly covers the SRP
# client and server behavior and the interactions of DNS-SD server
# with SRP server).
#
# Topology:
# Four nodes, leader acting as SRP and DNS-SD servers, with 3 router
# nodes acting as SRP and DNS clients.
#
SERVER = 1
CLIENT1 = 2
CLIENT2 = 3
CLIENT3 = 4
DOMAIN = 'default.service.arpa.'
SERVICE = '_ipps._tcp'
class TestDnssd(thread_cert.TestCase):
SUPPORT_NCP = False
USE_MESSAGE_FACTORY = False
TOPOLOGY = {
SERVER: {
'mode': 'rdn',
},
CLIENT1: {
'mode': 'rdn',
},
CLIENT2: {
'mode': 'rdn',
},
CLIENT3: {
'mode': 'rdn',
}
}
def test(self):
server = self.nodes[SERVER]
client1 = self.nodes[CLIENT1]
client2 = self.nodes[CLIENT2]
client3 = self.nodes[CLIENT3]
#---------------------------------------------------------------
# Start the server & client devices.
server.start()
self.simulator.go(config.LEADER_STARTUP_DELAY)
self.assertEqual(server.get_state(), 'leader')
server.srp_server_set_enabled(True)
client1.start()
client2.start()
client3.start()
self.simulator.go(config.ROUTER_STARTUP_DELAY)
self.assertEqual(client1.get_state(), 'router')
self.assertEqual(client2.get_state(), 'router')
self.assertEqual(client3.get_state(), 'router')
#---------------------------------------------------------------
# Register services on clients
client1_addrs = [client1.get_mleid(), client1.get_rloc()]
client2_addrs = [client2.get_mleid(), client2.get_rloc()]
client3_addrs = [client3.get_mleid(), client2.get_rloc()]
self._config_srp_client_services(client1, server, 'ins1', 'host1', 11111, 1, 1, client1_addrs, ",_s1,_s2")
self._config_srp_client_services(client2, server, 'ins2', 'HOST2', 22222, 2, 2, client2_addrs)
self._config_srp_client_services(client3, server, 'ins3', 'host3', 33333, 3, 3, client3_addrs, ",_S1")
#---------------------------------------------------------------
# Resolve address (AAAA records)
answers = client1.dns_resolve(f"host1.{DOMAIN}".upper(), server.get_mleid(), 53)
self.assertEqual(set(ipaddress.IPv6Address(ip) for ip, _ in answers),
set(map(ipaddress.IPv6Address, client1_addrs)))
answers = client1.dns_resolve(f"host2.{DOMAIN}", server.get_mleid(), 53)
self.assertEqual(set(ipaddress.IPv6Address(ip) for ip, _ in answers),
set(map(ipaddress.IPv6Address, client2_addrs)))
#---------------------------------------------------------------
# Browsing for services
instance1_verify_info = {
'port': 11111,
'priority': 1,
'weight': 1,
'host': 'host1.default.service.arpa.',
'address': client1_addrs,
'txt_data': '',
'srv_ttl': lambda x: x > 0,
'txt_ttl': lambda x: x > 0,
'aaaa_ttl': lambda x: x > 0,
}
instance2_verify_info = {
'port': 22222,
'priority': 2,
'weight': 2,
'host': 'host2.default.service.arpa.',
'address': client2_addrs,
'txt_data': '',
'srv_ttl': lambda x: x > 0,
'txt_ttl': lambda x: x > 0,
'aaaa_ttl': lambda x: x > 0,
}
instance3_verify_info = {
'port': 33333,
'priority': 3,
'weight': 3,
'host': 'host3.default.service.arpa.',
'address': client3_addrs,
'txt_data': '',
'srv_ttl': lambda x: x > 0,
'txt_ttl': lambda x: x > 0,
'aaaa_ttl': lambda x: x > 0,
}
instance4_verify_info = {
'port': 44444,
'priority': 4,
'weight': 4,
'host': 'host3.default.service.arpa.',
'address': client3_addrs,
'txt_data': 'KEY=414243', # KEY=ABC
'srv_ttl': lambda x: x > 0,
'txt_ttl': lambda x: x > 0,
'aaaa_ttl': lambda x: x > 0,
}
# Browse for main service
service_instances = client1.dns_browse(f'{SERVICE}.{DOMAIN}'.upper(), server.get_mleid(), 53)
self.assertEqual({'ins1', 'ins2', 'ins3'}, set(service_instances.keys()))
# Browse for service sub-type _s1.
service_instances = client1.dns_browse(f'_s1._sub.{SERVICE}.{DOMAIN}'.upper(), server.get_mleid(), 53)
self.assertEqual({'ins1', 'ins3'}, set(service_instances.keys()))
# Browse for service sub-type _s2.
# Since there is only one matching instance, validate that
# server included the service info in additional section.
service_instances = client1.dns_browse(f'_s2._sub.{SERVICE}.{DOMAIN}'.upper(), server.get_mleid(), 53)
self.assertEqual({'ins1'}, set(service_instances.keys()))
self._assert_service_instance_equal(service_instances['ins1'], instance1_verify_info)
#---------------------------------------------------------------
# Resolve service
service_instance = client1.dns_resolve_service('ins1', f'{SERVICE}.{DOMAIN}'.upper(), server.get_mleid(), 53)
self._assert_service_instance_equal(service_instance, instance1_verify_info)
service_instance = client1.dns_resolve_service('ins2', f'{SERVICE}.{DOMAIN}'.upper(), server.get_mleid(), 53)
self._assert_service_instance_equal(service_instance, instance2_verify_info)
service_instance = client1.dns_resolve_service('ins3', f'{SERVICE}.{DOMAIN}'.upper(), server.get_mleid(), 53)
self._assert_service_instance_equal(service_instance, instance3_verify_info)
#---------------------------------------------------------------
# Add another service with TXT entries to the existing host and
# verify that it is properly merged.
client3.srp_client_add_service('ins4', (SERVICE + ",_s1").upper(), 44444, 4, 4, txt_entries=['KEY=ABC'])
self.simulator.go(5)
service_instances = client1.dns_browse(f'{SERVICE}.{DOMAIN}', server.get_mleid(), 53)
self.assertEqual({'ins1', 'ins2', 'ins3', 'ins4'}, set(service_instances.keys()))
service_instance = client1.dns_resolve_service('ins4', f'{SERVICE}.{DOMAIN}'.upper(), server.get_mleid(), 53)
self._assert_service_instance_equal(service_instance, instance4_verify_info)
#---------------------------------------------------------------
# Query for KEY record for `ins1` service name
records = client1.dns_query(25, 'ins1', f'{SERVICE}.{DOMAIN}'.upper(), server.get_mleid(), 53)
self.assertEqual(len(records), 1)
record = records[0]
self.assertEqual(int(record['RecordType']), 25)
self.assertEqual(int(record['RecordLength']), 78)
self.assertTrue(int(record['TTL']) > 0)
self.assertEqual(record['Section'], 'answer')
self.assertEqual(record['Name'].lower(), 'ins1._ipps._tcp.default.service.arpa.')
self.assertIn('RecordData', record)
#---------------------------------------------------------------
# Query for SRV record for `ins1` service name
records = client1.dns_query(33, 'ins1', f'{SERVICE}.{DOMAIN}'.upper(), server.get_mleid(), 53)
self.assertEqual(len(records), 4)
# SRV record in answer section
record = records[0]
self.assertEqual(int(record['RecordType']), 33)
self.assertTrue(int(record['RecordLength']) > 0)
self.assertTrue(int(record['TTL']) > 0)
self.assertEqual(record['Section'], 'answer')
self.assertEqual(record['Name'].lower(), 'ins1._ipps._tcp.default.service.arpa.')
self.assertIn('RecordData', record)
# Other records TXT and A in additional section
for record in records[1:]:
self.assertTrue(int(record['RecordLength']) > 0)
self.assertIn('RecordData', record)
self.assertTrue(int(record['TTL']) > 0)
self.assertEqual(record['Section'], 'additional')
rrtype = int(record['RecordType'])
self.assertIn(rrtype, [16, 28]) # TXT and AAAA
if rrtype == 16:
self.assertEqual(record['Name'].lower(), 'ins1._ipps._tcp.default.service.arpa.')
self.assertEqual(record['RecordData'], '[00]')
else:
self.assertEqual(record['Name'].lower(), 'host1.default.service.arpa.')
#---------------------------------------------------------------
# Query for non-existing A record for `ins1` service name
records = client1.dns_query(1, 'ins1', f'{SERVICE}.{DOMAIN}'.upper(), server.get_mleid(), 53)
self.assertEqual(len(records), 0)
def _assert_service_instance_equal(self, instance, info):
self.assertEqual(instance['host'].lower(), info['host'].lower(), instance)
for f in ('port', 'priority', 'weight', 'txt_data'):
self.assertEqual(instance[f], info[f], instance)
verify_addresses = info['address']
if not isinstance(verify_addresses, typing.Collection):
verify_addresses = [verify_addresses]
self.assertIn(ipaddress.IPv6Address(instance['address']), map(ipaddress.IPv6Address, verify_addresses),
instance)
for ttl_f in ('srv_ttl', 'txt_ttl', 'aaaa_ttl'):
check_ttl = info[ttl_f]
if not callable(check_ttl):
check_ttl = lambda x: x == check_ttl
self.assertTrue(check_ttl(instance[ttl_f]), instance)
def _config_srp_client_services(self,
client,
server,
instancename,
hostname,
port,
priority,
weight,
addrs,
subtypes=''):
client.srp_client_set_host_name(hostname)
client.srp_client_set_host_address(*addrs)
client.srp_client_add_service(instancename, SERVICE + subtypes, port, priority, weight)
self.simulator.go(5)
self.assertEqual(client.srp_client_get_host_state(), 'Registered')
if __name__ == '__main__':
unittest.main()