Files
designate/functionaltests/api/v2/test_recordset.py
James Li 5879c90098 Fix data filtering with pagination
When listing recordsets with filtering on record data,
designate makes a first filtering against recordsets table
to retrieve a list of recordsets belonging to the specified
zone, followed by the second filtering against records table
on record data (e.g. IPs), then intersects the two.
Check https://github.com/openstack/designate/blob/master/designate/api/v2/controllers/recordsets.py

This approach does not work properly when pagination happens.
Imagine a zone has 21 A records with 10.* like IPs,
and 9 A records with 192.* like IPs. When requesting
"/v2/zone/{zone_id}/recordsets?data=10.*", designate makes
the first filtering and get the first page (i.e. 20 recordsets) which may
have both 192.* and 10.* records mixed together,
the second filtering actually excludes all 192.* records from the first page.
But that ends up with a page less than 20 records, so designate won't
include a 'next' link in the response.

This patch fixes the problem.

Closes-bug: #1561746
Change-Id: Ib06dd288d129ff4b39c388d80f24d179c6af28d8
2016-04-20 15:01:35 +00:00

225 lines
9.3 KiB
Python

"""
Copyright 2015 Rackspace
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
"""
import dns.rdatatype
from tempest_lib import exceptions
from functionaltests.common import datagen
from functionaltests.common import dnsclient
from functionaltests.common import utils
from functionaltests.api.v2.base import DesignateV2Test
from functionaltests.api.v2.clients.recordset_client import RecordsetClient
from functionaltests.api.v2.fixtures import ZoneFixture
from functionaltests.api.v2.fixtures import RecordsetFixture
RECORDSETS_DATASET = {
'A': dict(
make_recordset=lambda z: datagen.random_a_recordset(z.name)),
'AAAA': dict(
make_recordset=lambda z: datagen.random_aaaa_recordset(z.name)),
'CNAME': dict(
make_recordset=lambda z: datagen.random_cname_recordset(z.name)),
'MX': dict(
make_recordset=lambda z: datagen.random_mx_recordset(z.name)),
'SPF': dict(
make_recordset=lambda z: datagen.random_spf_recordset(z.name)),
'SRV': dict(
make_recordset=lambda z: datagen.random_srv_recordset(z.name)),
'SSHFP': dict(
make_recordset=lambda z: datagen.random_sshfp_recordset(z.name)),
'TXT': dict(
make_recordset=lambda z: datagen.random_txt_recordset(z.name)),
}
WILDCARD_RECORDSETS_DATASET = {
'A': dict(make_recordset=lambda z:
datagen.random_a_recordset(zone_name=z.name,
name="*.{0}".format(z.name))),
'AAAA': dict(make_recordset=lambda z:
datagen.random_aaaa_recordset(zone_name=z.name,
name="*.{0}".format(z.name))),
'CNAME': dict(make_recordset=lambda z:
datagen.random_cname_recordset(zone_name=z.name,
name="*.{0}".format(z.name))),
'MX': dict(make_recordset=lambda z:
datagen.random_mx_recordset(zone_name=z.name,
name="*.{0}".format(z.name))),
'SPF': dict(make_recordset=lambda z:
datagen.random_spf_recordset(zone_name=z.name,
name="*.{0}".format(z.name))),
'SSHFP': dict(make_recordset=lambda z:
datagen.random_sshfp_recordset(zone_name=z.name,
name="*.{0}".format(z.name))),
'TXT': dict(make_recordset=lambda z:
datagen.random_txt_recordset(zone_name=z.name,
name="*.{0}".format(z.name))),
}
@utils.parameterized_class
class RecordsetTest(DesignateV2Test):
def setUp(self):
super(RecordsetTest, self).setUp()
self.increase_quotas(user='default')
self.ensure_tld_exists('com')
self.zone = self.useFixture(ZoneFixture()).created_zone
def test_list_recordsets(self):
post_model = datagen.random_a_recordset(self.zone.name)
self.useFixture(RecordsetFixture(self.zone.id, post_model))
resp, model = RecordsetClient.as_user('default') \
.list_recordsets(self.zone.id)
self.assertEqual(200, resp.status)
self.assertGreater(len(model.recordsets), 0)
def test_list_recordsets_with_filtering(self):
# This test ensures the behavior in bug #1561746 won't happen
post_model = datagen.random_a_recordset(self.zone.name,
ip='192.168.1.2')
self.useFixture(RecordsetFixture(self.zone.id, post_model))
for i in range(1, 3):
post_model = datagen.random_a_recordset(self.zone.name,
ip='10.0.1.{}'.format(i))
self.useFixture(RecordsetFixture(self.zone.id, post_model))
# Add limit in filter to make response paginated
filters = {"data": "10.*", "limit": 2}
resp, model = RecordsetClient.as_user('default') \
.list_recordsets(self.zone.id, filters=filters)
self.assertEqual(200, resp.status)
self.assertEqual(2, model.metadata.total_count)
self.assertEqual(len(model.recordsets), 2)
self.assertIsNotNone(model.links.next)
def assert_dns(self, model):
results = dnsclient.query_servers(model.name, model.type)
model_data = model.to_dict()
if model.type == 'AAAA':
model_data['records'] = utils.shorten_ipv6_addrs(
model_data['records'])
for answer in results:
data = {
"type": dns.rdatatype.to_text(answer.rdtype),
"name": str(answer.canonical_name),
# DNSPython wraps TXT values in "" so '+all v=foo' becomes
# '"+all" "+v=foo"'
"records": [i.to_text().replace('"', '')
for i in answer.rrset.items]
}
if answer.rrset.ttl != 0:
data['ttl'] = answer.rrset.ttl
self.assertEqual(model_data, data)
@utils.parameterized(RECORDSETS_DATASET)
def test_crud_recordset(self, make_recordset):
post_model = make_recordset(self.zone)
fixture = self.useFixture(RecordsetFixture(self.zone.id, post_model))
recordset_id = fixture.created_recordset.id
self.assert_dns(fixture.post_model)
put_model = make_recordset(self.zone)
del put_model.name # don't try to update the name
resp, put_resp_model = RecordsetClient.as_user('default') \
.put_recordset(self.zone.id, recordset_id, put_model)
self.assertEqual(202, resp.status, "on put response")
self.assertEqual("PENDING", put_resp_model.status)
self.assertEqual(post_model.name, put_resp_model.name)
self.assertEqual(put_model.records, put_resp_model.records)
self.assertEqual(put_model.ttl, put_resp_model.ttl)
RecordsetClient.as_user('default').wait_for_recordset(
self.zone.id, recordset_id)
put_model.name = post_model.name
self.assert_dns(put_model)
resp, delete_resp_model = RecordsetClient.as_user('default') \
.delete_recordset(self.zone.id, recordset_id)
self.assertEqual(202, resp.status, "on delete response")
RecordsetClient.as_user('default').wait_for_404(
self.zone.id, recordset_id)
@utils.parameterized(WILDCARD_RECORDSETS_DATASET)
def test_can_create_and_query_wildcard_recordset(self, make_recordset):
post_model = make_recordset(self.zone)
self.useFixture(RecordsetFixture(self.zone.id, post_model))
verify_models = [
post_model.from_dict(post_model.to_dict()) for x in range(3)
]
verify_models[0].name = "abc.{0}".format(self.zone.name)
verify_models[1].name = "abc.def.{0}".format(self.zone.name)
verify_models[2].name = "abc.def.hij.{0}".format(self.zone.name)
for m in verify_models:
self.assert_dns(m)
class RecordsetOwnershipTest(DesignateV2Test):
def setUp(self):
super(RecordsetOwnershipTest, self).setUp()
self.increase_quotas(user='default')
self.increase_quotas(user='alt')
self.ensure_tld_exists('com')
def test_no_create_recordset_by_alt_tenant(self):
zone = self.useFixture(ZoneFixture(user='default')).created_zone
# try with name=A123456.zone.com.
recordset = datagen.random_a_recordset(zone_name=zone.name)
self.assertRaises(exceptions.RestClientException,
lambda: RecordsetClient.as_user('alt')
.post_recordset(zone.id, recordset))
# try with name=zone.com.
recordset.name = zone.name
self.assertRaises(exceptions.RestClientException,
lambda: RecordsetClient.as_user('alt')
.post_recordset(zone.id, recordset))
def test_no_create_super_recordsets(self):
# default creates zone a.b.c.example.com.
# alt fails to create record with name b.c.example.com
zone_data = datagen.random_zone_data()
recordset = datagen.random_a_recordset(zone_name=zone_data.name)
recordset.name = 'b.c.' + zone_data.name
zone_data.name = 'a.b.c.' + zone_data.name
fixture = self.useFixture(ZoneFixture(zone_data, user='default'))
self.assertRaises(exceptions.RestClientException,
lambda: RecordsetClient.as_user('alt')
.post_recordset(fixture.created_zone.id, recordset))
def test_no_create_recordset_via_alt_domain(self):
zone = self.useFixture(ZoneFixture(user='default')).created_zone
alt_zone = self.useFixture(ZoneFixture(user='alt')).created_zone
# alt attempts to create record with name A12345.{zone}
recordset = datagen.random_a_recordset(zone_name=zone.name)
self.assertRaises(exceptions.RestClientException,
lambda: RecordsetClient.as_user('alt')
.post_recordset(zone.id, recordset))
self.assertRaises(exceptions.RestClientException,
lambda: RecordsetClient.as_user('alt')
.post_recordset(alt_zone.id, recordset))