mirror of
https://github.com/github/octodns.git
synced 2024-05-11 05:55:00 +00:00
Fix for EnsureTrailingDots reverting value types back to strings
This commit is contained in:
@ -4,6 +4,8 @@
|
||||
where lots of changes are expected frequently to live along side zones
|
||||
where little or no churn is expected.
|
||||
* AutoArpa gained support for prioritizing values
|
||||
* Fix for EnsureTrailingDots reverting value types back to strings which then
|
||||
failed when rr methods were used on them (e.g. w/octodns-bind)
|
||||
|
||||
## v1.6.1 - 2024-03-17 - Didn't we do this already
|
||||
|
||||
|
@ -14,7 +14,9 @@ def _ensure_trailing_dots(record, prop):
|
||||
for value in new.values:
|
||||
val = getattr(value, prop)
|
||||
if val[-1] != '.':
|
||||
setattr(value, prop, f'{val}.')
|
||||
# these will generally be str, but just in case we'll use the
|
||||
# constructor
|
||||
setattr(value, prop, val.__class__(f'{val}.'))
|
||||
return new
|
||||
|
||||
|
||||
@ -24,14 +26,18 @@ class EnsureTrailingDots(BaseProcessor):
|
||||
_type = record._type
|
||||
if _type in ('ALIAS', 'CNAME', 'DNAME') and record.value[-1] != '.':
|
||||
new = record.copy()
|
||||
new.value = f'{new.value}.'
|
||||
# we need to preserve the value type (class) here and there's no
|
||||
# way to change a strings value, these all inherit from string,
|
||||
# so we need to create a new one of the same type
|
||||
new.value = new.value.__class__(f'{new.value}.')
|
||||
desired.add_record(new, replace=True)
|
||||
elif _type in ('NS', 'PTR') and any(
|
||||
v[-1] != '.' for v in record.values
|
||||
):
|
||||
new = record.copy()
|
||||
klass = new.values[0].__class__
|
||||
new.values = [
|
||||
v if v[-1] == '.' else f'{v}.' for v in record.values
|
||||
v if v[-1] == '.' else klass(f'{v}.') for v in record.values
|
||||
]
|
||||
desired.add_record(new, replace=True)
|
||||
elif _type == 'MX' and _no_trailing_dot(record, 'exchange'):
|
||||
|
@ -10,6 +10,11 @@ from octodns.processor.trailing_dots import (
|
||||
_no_trailing_dot,
|
||||
)
|
||||
from octodns.record import Record
|
||||
from octodns.record.alias import AliasValue
|
||||
from octodns.record.cname import CnameValue
|
||||
from octodns.record.dname import DnameValue
|
||||
from octodns.record.ns import NsValue
|
||||
from octodns.record.ptr import PtrValue
|
||||
from octodns.zone import Zone
|
||||
|
||||
|
||||
@ -37,22 +42,61 @@ class EnsureTrailingDotsTest(TestCase):
|
||||
zone.add_record(missing)
|
||||
|
||||
got = etd.process_source_zone(zone, None)
|
||||
|
||||
self.assertEqual('absolute.target.', _find(got, 'has').value)
|
||||
self.assertEqual('relative.target.', _find(got, 'missing').value)
|
||||
# ensure types were preserved
|
||||
self.assertIsInstance(_find(got, 'has').value, CnameValue)
|
||||
self.assertIsInstance(_find(got, 'missing').value, CnameValue)
|
||||
|
||||
def test_alias(self):
|
||||
etd = EnsureTrailingDots('test')
|
||||
|
||||
zone = Zone('unit.tests.', [])
|
||||
has = Record.new(
|
||||
zone,
|
||||
'has',
|
||||
{'type': 'ALIAS', 'ttl': 42, 'value': 'absolute.target.'},
|
||||
lenient=True,
|
||||
)
|
||||
zone.add_record(has)
|
||||
missing = Record.new(
|
||||
zone,
|
||||
'missing',
|
||||
{'type': 'ALIAS', 'ttl': 42, 'value': 'relative.target'},
|
||||
lenient=True,
|
||||
)
|
||||
zone.add_record(missing)
|
||||
|
||||
# HACK: this should never be done to records outside of specific testing
|
||||
# situations like this
|
||||
has._type = 'ALIAS'
|
||||
missing._type = 'ALIAS'
|
||||
got = etd.process_source_zone(zone, None)
|
||||
self.assertEqual('absolute.target.', _find(got, 'has').value)
|
||||
self.assertEqual('relative.target.', _find(got, 'missing').value)
|
||||
self.assertIsInstance(_find(got, 'has').value, AliasValue)
|
||||
self.assertIsInstance(_find(got, 'missing').value, AliasValue)
|
||||
|
||||
def test_dname(self):
|
||||
etd = EnsureTrailingDots('test')
|
||||
|
||||
zone = Zone('unit.tests.', [])
|
||||
has = Record.new(
|
||||
zone,
|
||||
'has',
|
||||
{'type': 'DNAME', 'ttl': 42, 'value': 'absolute.target.'},
|
||||
)
|
||||
zone.add_record(has)
|
||||
missing = Record.new(
|
||||
zone,
|
||||
'missing',
|
||||
{'type': 'DNAME', 'ttl': 42, 'value': 'relative.target'},
|
||||
lenient=True,
|
||||
)
|
||||
zone.add_record(missing)
|
||||
|
||||
has._type = 'DNAME'
|
||||
missing._type = 'DNAME'
|
||||
got = etd.process_source_zone(zone, None)
|
||||
self.assertEqual('absolute.target.', _find(got, 'has').value)
|
||||
self.assertEqual('relative.target.', _find(got, 'missing').value)
|
||||
self.assertIsInstance(_find(got, 'has').value, DnameValue)
|
||||
self.assertIsInstance(_find(got, 'missing').value, DnameValue)
|
||||
|
||||
def test_mx(self):
|
||||
etd = EnsureTrailingDots('test')
|
||||
@ -114,13 +158,48 @@ class EnsureTrailingDotsTest(TestCase):
|
||||
got = etd.process_source_zone(zone, None)
|
||||
got = next(iter(got.records))
|
||||
self.assertEqual(['absolute.target.', 'relative.target.'], got.values)
|
||||
self.assertIsInstance(got.values[0], NsValue)
|
||||
self.assertIsInstance(got.values[1], NsValue)
|
||||
|
||||
# again, but this time nothing to fix so that we fully use up the
|
||||
# generator
|
||||
zone = Zone('unit.tests.', [])
|
||||
record = Record.new(
|
||||
zone,
|
||||
'record',
|
||||
{
|
||||
'type': 'NS',
|
||||
'ttl': 42,
|
||||
'values': ['absolute.target.', 'another.target.'],
|
||||
},
|
||||
)
|
||||
zone.add_record(record)
|
||||
|
||||
got = etd.process_source_zone(zone, None)
|
||||
got = next(iter(got.records))
|
||||
self.assertEqual(['absolute.target.', 'another.target.'], got.values)
|
||||
|
||||
def test_ptr(self):
|
||||
etd = EnsureTrailingDots('test')
|
||||
|
||||
zone = Zone('unit.tests.', [])
|
||||
record = Record.new(
|
||||
zone,
|
||||
'record',
|
||||
{
|
||||
'type': 'PTR',
|
||||
'ttl': 42,
|
||||
'values': ['absolute.target.', 'relative.target'],
|
||||
},
|
||||
lenient=True,
|
||||
)
|
||||
zone.add_record(record)
|
||||
|
||||
# HACK: this should never be done to records outside of specific testing
|
||||
# situations like this
|
||||
record._type = 'PTR'
|
||||
got = etd.process_source_zone(zone, None)
|
||||
got = next(iter(got.records))
|
||||
self.assertEqual(['absolute.target.', 'relative.target.'], got.values)
|
||||
self.assertIsInstance(got.values[0], PtrValue)
|
||||
self.assertIsInstance(got.values[1], PtrValue)
|
||||
|
||||
def test_srv(self):
|
||||
etd = EnsureTrailingDots('test')
|
||||
|
Reference in New Issue
Block a user