You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					86 lines
				
				2.5 KiB
			
		
		
			
		
	
	
					86 lines
				
				2.5 KiB
			| 
								 
											3 years ago
										 
									 | 
							
								# Copyright (C) Dnspython Contributors, see LICENSE for text of ISC license
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								import contextlib
							 | 
						||
| 
								 | 
							
								import struct
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								import dns.exception
							 | 
						||
| 
								 | 
							
								import dns.name
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								class Parser:
							 | 
						||
| 
								 | 
							
								    def __init__(self, wire, current=0):
							 | 
						||
| 
								 | 
							
								        self.wire = wire
							 | 
						||
| 
								 | 
							
								        self.current = 0
							 | 
						||
| 
								 | 
							
								        self.end = len(self.wire)
							 | 
						||
| 
								 | 
							
								        if current:
							 | 
						||
| 
								 | 
							
								            self.seek(current)
							 | 
						||
| 
								 | 
							
								        self.furthest = current
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    def remaining(self):
							 | 
						||
| 
								 | 
							
								        return self.end - self.current
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    def get_bytes(self, size):
							 | 
						||
| 
								 | 
							
								        if size > self.remaining():
							 | 
						||
| 
								 | 
							
								            raise dns.exception.FormError
							 | 
						||
| 
								 | 
							
								        output = self.wire[self.current:self.current + size]
							 | 
						||
| 
								 | 
							
								        self.current += size
							 | 
						||
| 
								 | 
							
								        self.furthest = max(self.furthest, self.current)
							 | 
						||
| 
								 | 
							
								        return output
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    def get_counted_bytes(self, length_size=1):
							 | 
						||
| 
								 | 
							
								        length = int.from_bytes(self.get_bytes(length_size), 'big')
							 | 
						||
| 
								 | 
							
								        return self.get_bytes(length)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    def get_remaining(self):
							 | 
						||
| 
								 | 
							
								        return self.get_bytes(self.remaining())
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    def get_uint8(self):
							 | 
						||
| 
								 | 
							
								        return struct.unpack('!B', self.get_bytes(1))[0]
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    def get_uint16(self):
							 | 
						||
| 
								 | 
							
								        return struct.unpack('!H', self.get_bytes(2))[0]
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    def get_uint32(self):
							 | 
						||
| 
								 | 
							
								        return struct.unpack('!I', self.get_bytes(4))[0]
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    def get_uint48(self):
							 | 
						||
| 
								 | 
							
								        return int.from_bytes(self.get_bytes(6), 'big')
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    def get_struct(self, format):
							 | 
						||
| 
								 | 
							
								        return struct.unpack(format, self.get_bytes(struct.calcsize(format)))
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    def get_name(self, origin=None):
							 | 
						||
| 
								 | 
							
								        name = dns.name.from_wire_parser(self)
							 | 
						||
| 
								 | 
							
								        if origin:
							 | 
						||
| 
								 | 
							
								            name = name.relativize(origin)
							 | 
						||
| 
								 | 
							
								        return name
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    def seek(self, where):
							 | 
						||
| 
								 | 
							
								        # Note that seeking to the end is OK!  (If you try to read
							 | 
						||
| 
								 | 
							
								        # after such a seek, you'll get an exception as expected.)
							 | 
						||
| 
								 | 
							
								        if where < 0 or where > self.end:
							 | 
						||
| 
								 | 
							
								            raise dns.exception.FormError
							 | 
						||
| 
								 | 
							
								        self.current = where
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @contextlib.contextmanager
							 | 
						||
| 
								 | 
							
								    def restrict_to(self, size):
							 | 
						||
| 
								 | 
							
								        if size > self.remaining():
							 | 
						||
| 
								 | 
							
								            raise dns.exception.FormError
							 | 
						||
| 
								 | 
							
								        saved_end = self.end
							 | 
						||
| 
								 | 
							
								        try:
							 | 
						||
| 
								 | 
							
								            self.end = self.current + size
							 | 
						||
| 
								 | 
							
								            yield
							 | 
						||
| 
								 | 
							
								            # We make this check here and not in the finally as we
							 | 
						||
| 
								 | 
							
								            # don't want to raise if we're already raising for some
							 | 
						||
| 
								 | 
							
								            # other reason.
							 | 
						||
| 
								 | 
							
								            if self.current != self.end:
							 | 
						||
| 
								 | 
							
								                raise dns.exception.FormError
							 | 
						||
| 
								 | 
							
								        finally:
							 | 
						||
| 
								 | 
							
								            self.end = saved_end
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    @contextlib.contextmanager
							 | 
						||
| 
								 | 
							
								    def restore_furthest(self):
							 | 
						||
| 
								 | 
							
								        try:
							 | 
						||
| 
								 | 
							
								            yield None
							 | 
						||
| 
								 | 
							
								        finally:
							 | 
						||
| 
								 | 
							
								            self.current = self.furthest
							 |