# Copyright (C) 2003-2007, 2009-2011 Nominum, Inc. # # Permission to use, copy, modify, and distribute this software and its # documentation for any purpose with or without fee is hereby granted, # provided that the above copyright notice and this permission notice # appear in all copies. # # THE SOFTWARE IS PROVIDED "AS IS" AND NOMINUM DISCLAIMS ALL WARRANTIES # WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF # MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL NOMINUM BE LIABLE FOR # ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES # WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN # ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT # OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. """DNS Dynamic Update Support""" import dns.message import dns.name import dns.opcode import dns.rdata import dns.rdataclass import dns.rdataset import dns.tsig class Update(dns.message.Message): def __init__(self, zone, rdclass=dns.rdataclass.IN, keyring=None, keyname=None, keyalgorithm=dns.tsig.default_algorithm): """Initialize a new DNS Update object. @param zone: The zone which is being updated. @type zone: A dns.name.Name or string @param rdclass: The class of the zone; defaults to dns.rdataclass.IN. @type rdclass: An int designating the class, or a string whose value is the name of a class. @param keyring: The TSIG keyring to use; defaults to None. @type keyring: dict @param keyname: The name of the TSIG key to use; defaults to None. The key must be defined in the keyring. If a keyring is specified but a keyname is not, then the key used will be the first key in the keyring. Note that the order of keys in a dictionary is not defined, so applications should supply a keyname when a keyring is used, unless they know the keyring contains only one key. @type keyname: dns.name.Name or string @param keyalgorithm: The TSIG algorithm to use; defaults to dns.tsig.default_algorithm. Constants for TSIG algorithms are defined in dns.tsig, and the currently implemented algorithms are HMAC_MD5, HMAC_SHA1, HMAC_SHA224, HMAC_SHA256, HMAC_SHA384, and HMAC_SHA512. @type keyalgorithm: string """ super(Update, self).__init__() self.flags |= dns.opcode.to_flags(dns.opcode.UPDATE) if isinstance(zone, (str, unicode)): zone = dns.name.from_text(zone) self.origin = zone if isinstance(rdclass, str): rdclass = dns.rdataclass.from_text(rdclass) self.zone_rdclass = rdclass self.find_rrset(self.question, self.origin, rdclass, dns.rdatatype.SOA, create=True, force_unique=True) if not keyring is None: self.use_tsig(keyring, keyname, algorithm=keyalgorithm) def _add_rr(self, name, ttl, rd, deleting=None, section=None): """Add a single RR to the update section.""" if section is None: section = self.authority covers = rd.covers() rrset = self.find_rrset(section, name, self.zone_rdclass, rd.rdtype, covers, deleting, True, True) rrset.add(rd, ttl) def _add(self, replace, section, name, *args): """Add records. The first argument is the replace mode. If false, RRs are added to an existing RRset; if true, the RRset is replaced with the specified contents. The second argument is the section to add to. The third argument is always a name. The other arguments can be: - rdataset... - ttl, rdata... - ttl, rdtype, string...""" if isinstance(name, (str, unicode)): name = dns.name.from_text(name, None) if isinstance(args[0], dns.rdataset.Rdataset): for rds in args: if replace: self.delete(name, rds.rdtype) for rd in rds: self._add_rr(name, rds.ttl, rd, section=section) else: args = list(args) ttl = int(args.pop(0)) if isinstance(args[0], dns.rdata.Rdata): if replace: self.delete(name, args[0].rdtype) for rd in args: self._add_rr(name, ttl, rd, section=section) else: rdtype = args.pop(0) if isinstance(rdtype, str): rdtype = dns.rdatatype.from_text(rdtype) if replace: self.delete(name, rdtype) for s in args: rd = dns.rdata.from_text(self.zone_rdclass, rdtype, s, self.origin) self._add_rr(name, ttl, rd, section=section) def add(self, name, *args): """Add records. The first argument is always a name. The other arguments can be: - rdataset... - ttl, rdata... - ttl, rdtype, string...""" self._add(False, self.authority, name, *args) def delete(self, name, *args): """Delete records. The first argument is always a name. The other arguments can be: - I{nothing} - rdataset... - rdata... - rdtype, [string...]""" if isinstance(name, (str, unicode)): name = dns.name.from_text(name, None) if len(args) == 0: rrset = self.find_rrset(self.authority, name, dns.rdataclass.ANY, dns.rdatatype.ANY, dns.rdatatype.NONE, dns.rdatatype.ANY, True, True) elif isinstance(args[0], dns.rdataset.Rdataset): for rds in args: for rd in rds: self._add_rr(name, 0, rd, dns.rdataclass.NONE) else: args = list(args) if isinstance(args[0], dns.rdata.Rdata): for rd in args: self._add_rr(name, 0, rd, dns.rdataclass.NONE) else: rdtype = args.pop(0) if isinstance(rdtype, (str, unicode)): rdtype = dns.rdatatype.from_text(rdtype) if len(args) == 0: rrset = self.find_rrset(self.authority, name, self.zone_rdclass, rdtype, dns.rdatatype.NONE, dns.rdataclass.ANY, True, True) else: for s in args: rd = dns.rdata.from_text(self.zone_rdclass, rdtype, s, self.origin) self._add_rr(name, 0, rd, dns.rdataclass.NONE) def replace(self, name, *args): """Replace records. The first argument is always a name. The other arguments can be: - rdataset... - ttl, rdata... - ttl, rdtype, string... Note that if you want to replace the entire node, you should do a delete of the name followed by one or more calls to add.""" self._add(True, self.authority, name, *args) def present(self, name, *args): """Require that an owner name (and optionally an rdata type, or specific rdataset) exists as a prerequisite to the execution of the update. The first argument is always a name. The other arguments can be: - rdataset... - rdata... - rdtype, string...""" if isinstance(name, (str, unicode)): name = dns.name.from_text(name, None) if len(args) == 0: rrset = self.find_rrset(self.answer, name, dns.rdataclass.ANY, dns.rdatatype.ANY, dns.rdatatype.NONE, None, True, True) elif isinstance(args[0], dns.rdataset.Rdataset) or \ isinstance(args[0], dns.rdata.Rdata) or \ len(args) > 1: if not isinstance(args[0], dns.rdataset.Rdataset): # Add a 0 TTL args = list(args) args.insert(0, 0) self._add(False, self.answer, name, *args) else: rdtype = args[0] if isinstance(rdtype, (str, unicode)): rdtype = dns.rdatatype.from_text(rdtype) rrset = self.find_rrset(self.answer, name, dns.rdataclass.ANY, rdtype, dns.rdatatype.NONE, None, True, True) def absent(self, name, rdtype=None): """Require that an owner name (and optionally an rdata type) does not exist as a prerequisite to the execution of the update.""" if isinstance(name, (str, unicode)): name = dns.name.from_text(name, None) if rdtype is None: rrset = self.find_rrset(self.answer, name, dns.rdataclass.NONE, dns.rdatatype.ANY, dns.rdatatype.NONE, None, True, True) else: if isinstance(rdtype, (str, unicode)): rdtype = dns.rdatatype.from_text(rdtype) rrset = self.find_rrset(self.answer, name, dns.rdataclass.NONE, rdtype, dns.rdatatype.NONE, None, True, True) def to_wire(self, origin=None, max_size=65535): """Return a string containing the update in DNS compressed wire format. @rtype: string""" if origin is None: origin = self.origin return super(Update, self).to_wire(origin, max_size)
Name | Type | Size | Permission | Actions |
---|---|---|---|---|
rdtypes | Folder | 0755 |
|
|
.__init__.pyo.40009 | File | 765 B | 0644 |
|
.e164.pyo.40009 | File | 2.76 KB | 0644 |
|
.edns.pyo.40009 | File | 5.2 KB | 0644 |
|
.entropy.pyo.40009 | File | 3.7 KB | 0644 |
|
.exception.pyo.40009 | File | 1.56 KB | 0644 |
|
.flags.pyo.40009 | File | 2.58 KB | 0644 |
|
.hash.pyo.40009 | File | 2.11 KB | 0644 |
|
.inet.pyo.40009 | File | 2.61 KB | 0644 |
|
.ipv4.pyo.40009 | File | 1.37 KB | 0644 |
|
.ipv6.pyo.40009 | File | 3.44 KB | 0644 |
|
.message.pyo.40009 | File | 36.39 KB | 0644 |
|
.name.pyo.40009 | File | 23 KB | 0644 |
|
.namedict.pyo.40009 | File | 1.92 KB | 0644 |
|
.node.pyo.40009 | File | 6.15 KB | 0644 |
|
.opcode.pyo.40009 | File | 2.49 KB | 0644 |
|
.query.pyo.40009 | File | 15.31 KB | 0644 |
|
.rcode.pyo.40009 | File | 2.92 KB | 0644 |
|
.rdata.pyo.40009 | File | 16.59 KB | 0644 |
|
.rdataclass.pyo.40009 | File | 2.96 KB | 0644 |
|
.rdatatype.pyo.40009 | File | 5.29 KB | 0644 |
|
.resolver.pyo.40009 | File | 37.16 KB | 0644 |
|
.reversename.pyo.40009 | File | 2.4 KB | 0644 |
|
.rrset.pyo.40009 | File | 6.17 KB | 0644 |
|
.set.pyo.40009 | File | 9.81 KB | 0644 |
|
.tokenizer.pyo.40009 | File | 16.92 KB | 0644 |
|
.tsig.pyo.40009 | File | 8.08 KB | 0644 |
|
.tsigkeyring.pyo.40009 | File | 1.16 KB | 0644 |
|
.ttl.pyo.40009 | File | 1.48 KB | 0644 |
|
.update.pyo.40009 | File | 8.27 KB | 0644 |
|
.version.pyo.40009 | File | 560 B | 0644 |
|
.wiredata.pyo.40009 | File | 2.19 KB | 0644 |
|
.zone.pyo.40009 | File | 33.52 KB | 0644 |
|
__init__.py | File | 1.3 KB | 0644 |
|
__init__.pyc | File | 765 B | 0644 |
|
__init__.pyo | File | 765 B | 0644 |
|
dnssec.py | File | 14.37 KB | 0644 |
|
dnssec.pyc | File | 13.46 KB | 0644 |
|
dnssec.pyo | File | 13.38 KB | 0644 |
|
e164.py | File | 3.01 KB | 0644 |
|
e164.pyc | File | 2.76 KB | 0644 |
|
e164.pyo | File | 2.76 KB | 0644 |
|
edns.py | File | 4.21 KB | 0644 |
|
edns.pyc | File | 5.2 KB | 0644 |
|
edns.pyo | File | 5.2 KB | 0644 |
|
entropy.py | File | 3.79 KB | 0644 |
|
entropy.pyc | File | 3.7 KB | 0644 |
|
entropy.pyo | File | 3.7 KB | 0644 |
|
exception.py | File | 1.29 KB | 0644 |
|
exception.pyc | File | 1.56 KB | 0644 |
|
exception.pyo | File | 1.56 KB | 0644 |
|
flags.py | File | 2.62 KB | 0644 |
|
flags.pyc | File | 2.58 KB | 0644 |
|
flags.pyo | File | 2.58 KB | 0644 |
|
grange.py | File | 1.83 KB | 0644 |
|
grange.pyc | File | 1.15 KB | 0644 |
|
grange.pyo | File | 1.08 KB | 0644 |
|
hash.py | File | 2.34 KB | 0644 |
|
hash.pyc | File | 2.11 KB | 0644 |
|
hash.pyo | File | 2.11 KB | 0644 |
|
inet.py | File | 3.16 KB | 0644 |
|
inet.pyc | File | 2.61 KB | 0644 |
|
inet.pyo | File | 2.61 KB | 0644 |
|
ipv4.py | File | 1.81 KB | 0644 |
|
ipv4.pyc | File | 1.37 KB | 0644 |
|
ipv4.pyo | File | 1.37 KB | 0644 |
|
ipv6.py | File | 4.97 KB | 0644 |
|
ipv6.pyc | File | 3.44 KB | 0644 |
|
ipv6.pyo | File | 3.44 KB | 0644 |
|
message.py | File | 41.63 KB | 0644 |
|
message.pyc | File | 36.39 KB | 0644 |
|
message.pyo | File | 36.39 KB | 0644 |
|
name.py | File | 21.82 KB | 0644 |
|
name.pyc | File | 23 KB | 0644 |
|
name.pyo | File | 23 KB | 0644 |
|
namedict.py | File | 2.06 KB | 0644 |
|
namedict.pyc | File | 1.92 KB | 0644 |
|
namedict.pyo | File | 1.92 KB | 0644 |
|
node.py | File | 5.89 KB | 0644 |
|
node.pyc | File | 6.15 KB | 0644 |
|
node.pyo | File | 6.15 KB | 0644 |
|
opcode.py | File | 2.55 KB | 0644 |
|
opcode.pyc | File | 2.49 KB | 0644 |
|
opcode.pyo | File | 2.49 KB | 0644 |
|
query.py | File | 17.89 KB | 0644 |
|
query.pyc | File | 15.31 KB | 0644 |
|
query.pyo | File | 15.31 KB | 0644 |
|
rcode.py | File | 3.03 KB | 0644 |
|
rcode.pyc | File | 2.92 KB | 0644 |
|
rcode.pyo | File | 2.92 KB | 0644 |
|
rdata.py | File | 15.34 KB | 0644 |
|
rdata.pyc | File | 16.59 KB | 0644 |
|
rdata.pyo | File | 16.59 KB | 0644 |
|
rdataclass.py | File | 3.22 KB | 0644 |
|
rdataclass.pyc | File | 2.96 KB | 0644 |
|
rdataclass.pyo | File | 2.96 KB | 0644 |
|
rdataset.py | File | 11.28 KB | 0644 |
|
rdataset.pyc | File | 11.08 KB | 0644 |
|
rdataset.pyo | File | 11.03 KB | 0644 |
|
rdatatype.py | File | 5.07 KB | 0644 |
|
rdatatype.pyc | File | 5.29 KB | 0644 |
|
rdatatype.pyo | File | 5.29 KB | 0644 |
|
renderer.py | File | 11.63 KB | 0644 |
|
renderer.pyc | File | 10.85 KB | 0644 |
|
renderer.pyo | File | 10.75 KB | 0644 |
|
resolver.py | File | 45.35 KB | 0644 |
|
resolver.pyc | File | 37.16 KB | 0644 |
|
resolver.pyo | File | 37.16 KB | 0644 |
|
reversename.py | File | 3.03 KB | 0644 |
|
reversename.pyc | File | 2.4 KB | 0644 |
|
reversename.pyo | File | 2.4 KB | 0644 |
|
rrset.py | File | 5.76 KB | 0644 |
|
rrset.pyc | File | 6.17 KB | 0644 |
|
rrset.pyo | File | 6.17 KB | 0644 |
|
set.py | File | 7.66 KB | 0644 |
|
set.pyc | File | 9.81 KB | 0644 |
|
set.pyo | File | 9.81 KB | 0644 |
|
tokenizer.py | File | 17.54 KB | 0644 |
|
tokenizer.pyc | File | 16.92 KB | 0644 |
|
tokenizer.pyo | File | 16.92 KB | 0644 |
|
tsig.py | File | 8.1 KB | 0644 |
|
tsig.pyc | File | 8.08 KB | 0644 |
|
tsig.pyo | File | 8.08 KB | 0644 |
|
tsigkeyring.py | File | 1.61 KB | 0644 |
|
tsigkeyring.pyc | File | 1.16 KB | 0644 |
|
tsigkeyring.pyo | File | 1.16 KB | 0644 |
|
ttl.py | File | 2.13 KB | 0644 |
|
ttl.pyc | File | 1.48 KB | 0644 |
|
ttl.pyo | File | 1.48 KB | 0644 |
|
update.py | File | 9.92 KB | 0644 |
|
update.pyc | File | 8.27 KB | 0644 |
|
update.pyo | File | 8.27 KB | 0644 |
|
version.py | File | 1.24 KB | 0644 |
|
version.pyc | File | 560 B | 0644 |
|
version.pyo | File | 560 B | 0644 |
|
wiredata.py | File | 2.53 KB | 0644 |
|
wiredata.pyc | File | 2.19 KB | 0644 |
|
wiredata.pyo | File | 2.19 KB | 0644 |
|
zone.py | File | 37.94 KB | 0644 |
|
zone.pyc | File | 33.52 KB | 0644 |
|
zone.pyo | File | 33.52 KB | 0644 |
|