1 # Copyright (C) 2004-2007, 2009, 2010 Nominum, Inc. 2 # 3 # Permission to use, copy, modify, and distribute this software and its 4 # documentation for any purpose with or without fee is hereby granted, 5 # provided that the above copyright notice and this permission notice 6 # appear in all copies. 7 # 8 # THE SOFTWARE IS PROVIDED "AS IS" AND NOMINUM DISCLAIMS ALL WARRANTIES 9 # WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF 10 # MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL NOMINUM BE LIABLE FOR 11 # ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES 12 # WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN 13 # ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT 14 # OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. 15 16 import cStringIO 17 18 import dns.exception 19 import dns.rdata 20 import dns.rdatatype 21 import dns.name 22 23 class NSEC(dns.rdata.Rdata): 24 """NSEC record 25 26 @ivar next: the next name 27 @type next: dns.name.Name object 28 @ivar windows: the windowed bitmap list 29 @type windows: list of (window number, string) tuples""" 30 31 __slots__ = ['next', 'windows'] 32 33 def __init__(self, rdclass, rdtype, next, windows): 34 super(NSEC, self).__init__(rdclass, rdtype) 35 self.next = next 36 self.windows = windows 37 38 def to_text(self, origin=None, relativize=True, **kw): 39 next = self.next.choose_relativity(origin, relativize) 40 text = '' 41 for (window, bitmap) in self.windows: 42 bits = [] 43 for i in xrange(0, len(bitmap)): 44 byte = ord(bitmap[i]) 45 for j in xrange(0, 8): 46 if byte & (0x80 >> j): 47 bits.append(dns.rdatatype.to_text(window * 256 + \ 48 i * 8 + j)) 49 text += (' ' + ' '.join(bits)) 50 return '%s%s' % (next, text) 51 52 def from_text(cls, rdclass, rdtype, tok, origin = None, relativize = True): 53 next = tok.get_name() 54 next = next.choose_relativity(origin, relativize) 55 rdtypes = [] 56 while 1: 57 token = tok.get().unescape() 58 if token.is_eol_or_eof(): 59 break 60 nrdtype = dns.rdatatype.from_text(token.value) 61 if nrdtype == 0: 62 raise dns.exception.SyntaxError("NSEC with bit 0") 63 if nrdtype > 65535: 64 raise dns.exception.SyntaxError("NSEC with bit > 65535") 65 rdtypes.append(nrdtype) 66 rdtypes.sort() 67 window = 0 68 octets = 0 69 prior_rdtype = 0 70 bitmap = ['\0'] * 32 71 windows = [] 72 for nrdtype in rdtypes: 73 if nrdtype == prior_rdtype: 74 continue 75 prior_rdtype = nrdtype 76 new_window = nrdtype // 256 77 if new_window != window: 78 windows.append((window, ''.join(bitmap[0:octets]))) 79 bitmap = ['\0'] * 32 80 window = new_window 81 offset = nrdtype % 256 82 byte = offset / 8 83 bit = offset % 8 84 octets = byte + 1 85 bitmap[byte] = chr(ord(bitmap[byte]) | (0x80 >> bit)) 86 windows.append((window, ''.join(bitmap[0:octets]))) 87 return cls(rdclass, rdtype, next, windows) 88 89 from_text = classmethod(from_text) 90 91 def to_wire(self, file, compress = None, origin = None): 92 self.next.to_wire(file, None, origin) 93 for (window, bitmap) in self.windows: 94 file.write(chr(window)) 95 file.write(chr(len(bitmap))) 96 file.write(bitmap) 97 98 def from_wire(cls, rdclass, rdtype, wire, current, rdlen, origin = None): 99 (next, cused) = dns.name.from_wire(wire[: current + rdlen], current) 100 current += cused 101 rdlen -= cused 102 windows = [] 103 while rdlen > 0: 104 if rdlen < 3: 105 raise dns.exception.FormError("NSEC too short") 106 window = ord(wire[current]) 107 octets = ord(wire[current + 1]) 108 if octets == 0 or octets > 32: 109 raise dns.exception.FormError("bad NSEC octets") 110 current += 2 111 rdlen -= 2 112 if rdlen < octets: 113 raise dns.exception.FormError("bad NSEC bitmap length") 114 bitmap = wire[current : current + octets] 115 current += octets 116 rdlen -= octets 117 windows.append((window, bitmap)) 118 if not origin is None: 119 next = next.relativize(origin) 120 return cls(rdclass, rdtype, next, windows) 121 122 from_wire = classmethod(from_wire) 123 124 def choose_relativity(self, origin = None, relativize = True): 125 self.next = self.next.choose_relativity(origin, relativize) 126 127 def _cmp(self, other): 128 v = cmp(self.next, other.next) 129 if v == 0: 130 b1 = cStringIO.StringIO() 131 for (window, bitmap) in self.windows: 132 b1.write(chr(window)) 133 b1.write(chr(len(bitmap))) 134 b1.write(bitmap) 135 b2 = cStringIO.StringIO() 136 for (window, bitmap) in other.windows: 137 b2.write(chr(window)) 138 b2.write(chr(len(bitmap))) 139 b2.write(bitmap) 140 v = cmp(b1.getvalue(), b2.getvalue()) 141 return v 142