| 
									
										
										
										
											1992-12-14 23:25:04 +00:00
										 |  |  | # Implement (a subset of) Sun XDR -- RFC1014. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1992-12-17 17:32:24 +00:00
										 |  |  | try: | 
					
						
							|  |  |  | 	import struct | 
					
						
							|  |  |  | except ImportError: | 
					
						
							|  |  |  | 	struct = None | 
					
						
							| 
									
										
										
										
											1992-12-14 23:25:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1992-12-15 21:43:59 +00:00
										 |  |  | Long = type(0L) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1992-12-14 23:25:04 +00:00
										 |  |  | class Packer: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-12-17 14:32:26 +00:00
										 |  |  | 	def __init__(self): | 
					
						
							| 
									
										
										
										
											1992-12-14 23:25:04 +00:00
										 |  |  | 		self.reset() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	def reset(self): | 
					
						
							|  |  |  | 		self.buf = '' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	def get_buf(self): | 
					
						
							|  |  |  | 		return self.buf | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	def pack_uint(self, x): | 
					
						
							|  |  |  | 		self.buf = self.buf + \ | 
					
						
							|  |  |  | 			(chr(int(x>>24 & 0xff)) + chr(int(x>>16 & 0xff)) + \ | 
					
						
							|  |  |  | 			 chr(int(x>>8 & 0xff)) + chr(int(x & 0xff))) | 
					
						
							| 
									
										
										
										
											1992-12-17 17:32:24 +00:00
										 |  |  | 	if struct and struct.pack('l', 1) == '\0\0\0\1': | 
					
						
							| 
									
										
										
										
											1992-12-14 23:25:04 +00:00
										 |  |  | 		def pack_uint(self, x): | 
					
						
							| 
									
										
										
										
											1992-12-15 21:43:59 +00:00
										 |  |  | 			if type(x) == Long: | 
					
						
							|  |  |  | 				x = int((x + 0x80000000L) % 0x100000000L \ | 
					
						
							|  |  |  | 					   - 0x80000000L) | 
					
						
							|  |  |  | 			self.buf = self.buf + struct.pack('l', x) | 
					
						
							| 
									
										
										
										
											1992-12-14 23:25:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	pack_int = pack_uint | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	pack_enum = pack_int | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	def pack_bool(self, x): | 
					
						
							|  |  |  | 		if x: self.buf = self.buf + '\0\0\0\1' | 
					
						
							|  |  |  | 		else: self.buf = self.buf + '\0\0\0\0' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	def pack_uhyper(self, x): | 
					
						
							| 
									
										
										
										
											1992-12-15 21:43:59 +00:00
										 |  |  | 		self.pack_uint(int(x>>32 & 0xffffffff)) | 
					
						
							|  |  |  | 		self.pack_uint(int(x & 0xffffffff)) | 
					
						
							| 
									
										
										
										
											1992-12-14 23:25:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	pack_hyper = pack_uhyper | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-12-17 14:32:26 +00:00
										 |  |  | 	def pack_float(self, x): | 
					
						
							|  |  |  | 		# XXX | 
					
						
							|  |  |  | 		self.buf = self.buf + struct.pack('f', x) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	def pack_double(self, x): | 
					
						
							|  |  |  | 		# XXX | 
					
						
							|  |  |  | 		self.buf = self.buf + struct.pack('d', x) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1992-12-14 23:25:04 +00:00
										 |  |  | 	def pack_fstring(self, n, s): | 
					
						
							|  |  |  | 		if n < 0: | 
					
						
							|  |  |  | 			raise ValueError, 'fstring size must be nonnegative' | 
					
						
							|  |  |  | 		n = ((n+3)/4)*4 | 
					
						
							|  |  |  | 		data = s[:n] | 
					
						
							|  |  |  | 		data = data + (n - len(data)) * '\0' | 
					
						
							|  |  |  | 		self.buf = self.buf + data | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	pack_fopaque = pack_fstring | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	def pack_string(self, s): | 
					
						
							|  |  |  | 		n = len(s) | 
					
						
							|  |  |  | 		self.pack_uint(n) | 
					
						
							|  |  |  | 		self.pack_fstring(n, s) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1995-10-11 18:54:15 +00:00
										 |  |  | 	pack_opaque = pack_string | 
					
						
							| 
									
										
										
										
											1992-12-14 23:25:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	def pack_list(self, list, pack_item): | 
					
						
							|  |  |  | 		for item in list: | 
					
						
							|  |  |  | 			self.pack_uint(1) | 
					
						
							| 
									
										
										
										
											1993-12-17 14:32:26 +00:00
										 |  |  | 			pack_item(item) | 
					
						
							| 
									
										
										
										
											1992-12-14 23:25:04 +00:00
										 |  |  | 		self.pack_uint(0) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1992-12-21 14:33:18 +00:00
										 |  |  | 	def pack_farray(self, n, list, pack_item): | 
					
						
							|  |  |  | 		if len(list) <> n: | 
					
						
							|  |  |  | 			raise ValueError, 'wrong array size' | 
					
						
							|  |  |  | 		for item in list: | 
					
						
							|  |  |  | 			pack_item(item) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	def pack_array(self, list, pack_item): | 
					
						
							|  |  |  | 		n = len(list) | 
					
						
							|  |  |  | 		self.pack_uint(n) | 
					
						
							| 
									
										
										
										
											1995-10-04 18:39:03 +00:00
										 |  |  | 		self.pack_farray(n, list, pack_item) | 
					
						
							| 
									
										
										
										
											1992-12-21 14:33:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1992-12-14 23:25:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | class Unpacker: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-12-17 14:32:26 +00:00
										 |  |  | 	def __init__(self, data): | 
					
						
							| 
									
										
										
										
											1992-12-14 23:25:04 +00:00
										 |  |  | 		self.reset(data) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	def reset(self, data): | 
					
						
							|  |  |  | 		self.buf = data | 
					
						
							|  |  |  | 		self.pos = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	def done(self): | 
					
						
							|  |  |  | 		if self.pos < len(self.buf): | 
					
						
							|  |  |  | 			raise RuntimeError, 'unextracted data remains' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	def unpack_uint(self): | 
					
						
							|  |  |  | 		i = self.pos | 
					
						
							|  |  |  | 		self.pos = j = i+4 | 
					
						
							|  |  |  | 		data = self.buf[i:j] | 
					
						
							| 
									
										
										
										
											1992-12-19 00:06:17 +00:00
										 |  |  | 		if len(data) < 4: | 
					
						
							|  |  |  | 			raise EOFError | 
					
						
							| 
									
										
										
										
											1992-12-14 23:25:04 +00:00
										 |  |  | 		x = long(ord(data[0]))<<24 | ord(data[1])<<16 | \ | 
					
						
							|  |  |  | 			ord(data[2])<<8 | ord(data[3]) | 
					
						
							|  |  |  | 		# Return a Python long only if the value is not representable | 
					
						
							|  |  |  | 		# as a nonnegative Python int | 
					
						
							|  |  |  | 		if x < 0x80000000L: x = int(x) | 
					
						
							|  |  |  | 		return x | 
					
						
							| 
									
										
										
										
											1992-12-17 17:32:24 +00:00
										 |  |  | 	if struct and struct.unpack('l', '\0\0\0\1') == 1: | 
					
						
							| 
									
										
										
										
											1992-12-14 23:25:04 +00:00
										 |  |  | 		def unpack_uint(self): | 
					
						
							|  |  |  | 			i = self.pos | 
					
						
							|  |  |  | 			self.pos = j = i+4 | 
					
						
							| 
									
										
										
										
											1992-12-19 00:06:17 +00:00
										 |  |  | 			data = self.buf[i:j] | 
					
						
							|  |  |  | 			if len(data) < 4: | 
					
						
							|  |  |  | 				raise EOFError | 
					
						
							|  |  |  | 			return struct.unpack('l', data) | 
					
						
							| 
									
										
										
										
											1992-12-14 23:25:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	def unpack_int(self): | 
					
						
							|  |  |  | 		x = self.unpack_uint() | 
					
						
							|  |  |  | 		if x >= 0x80000000L: x = x - 0x100000000L | 
					
						
							|  |  |  | 		return int(x) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	unpack_enum = unpack_int | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	unpack_bool = unpack_int | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	def unpack_uhyper(self): | 
					
						
							|  |  |  | 		hi = self.unpack_uint() | 
					
						
							|  |  |  | 		lo = self.unpack_uint() | 
					
						
							|  |  |  | 		return long(hi)<<32 | lo | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	def unpack_hyper(self): | 
					
						
							|  |  |  | 		x = self.unpack_uhyper() | 
					
						
							|  |  |  | 		if x >= 0x8000000000000000L: x = x - 0x10000000000000000L | 
					
						
							|  |  |  | 		return x | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-12-17 14:32:26 +00:00
										 |  |  | 	def unpack_float(self): | 
					
						
							|  |  |  | 		# XXX | 
					
						
							|  |  |  | 		i = self.pos | 
					
						
							|  |  |  | 		self.pos = j = i+4 | 
					
						
							|  |  |  | 		data = self.buf[i:j] | 
					
						
							|  |  |  | 		if len(data) < 4: | 
					
						
							|  |  |  | 			raise EOFError | 
					
						
							|  |  |  | 		return struct.unpack('f', data)[0] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	def unpack_double(self): | 
					
						
							|  |  |  | 		# XXX | 
					
						
							|  |  |  | 		i = self.pos | 
					
						
							|  |  |  | 		self.pos = j = i+8 | 
					
						
							|  |  |  | 		data = self.buf[i:j] | 
					
						
							|  |  |  | 		if len(data) < 8: | 
					
						
							|  |  |  | 			raise EOFError | 
					
						
							| 
									
										
										
										
											1998-01-28 14:59:48 +00:00
										 |  |  | 		return struct.unpack('d', data)[0] | 
					
						
							| 
									
										
										
										
											1993-12-17 14:32:26 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1992-12-14 23:25:04 +00:00
										 |  |  | 	def unpack_fstring(self, n): | 
					
						
							|  |  |  | 		if n < 0: | 
					
						
							|  |  |  | 			raise ValueError, 'fstring size must be nonnegative' | 
					
						
							|  |  |  | 		i = self.pos | 
					
						
							|  |  |  | 		j = i + (n+3)/4*4 | 
					
						
							|  |  |  | 		if j > len(self.buf): | 
					
						
							| 
									
										
										
										
											1992-12-19 00:06:17 +00:00
										 |  |  | 			raise EOFError | 
					
						
							| 
									
										
										
										
											1992-12-14 23:25:04 +00:00
										 |  |  | 		self.pos = j | 
					
						
							|  |  |  | 		return self.buf[i:i+n] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	unpack_fopaque = unpack_fstring | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	def unpack_string(self): | 
					
						
							|  |  |  | 		n = self.unpack_uint() | 
					
						
							|  |  |  | 		return self.unpack_fstring(n) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1995-10-11 18:54:15 +00:00
										 |  |  | 	unpack_opaque = unpack_string | 
					
						
							| 
									
										
										
										
											1992-12-14 23:25:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	def unpack_list(self, unpack_item): | 
					
						
							|  |  |  | 		list = [] | 
					
						
							|  |  |  | 		while 1: | 
					
						
							|  |  |  | 			x = self.unpack_uint() | 
					
						
							| 
									
										
										
										
											1993-12-17 14:32:26 +00:00
										 |  |  | 			if x == 0: break | 
					
						
							| 
									
										
										
										
											1992-12-14 23:25:04 +00:00
										 |  |  | 			if x <> 1: | 
					
						
							|  |  |  | 				raise RuntimeError, \ | 
					
						
							|  |  |  | 					'0 or 1 expected, got ' + `x` | 
					
						
							| 
									
										
										
										
											1993-12-17 14:32:26 +00:00
										 |  |  | 			item = unpack_item() | 
					
						
							|  |  |  | 			list.append(item) | 
					
						
							| 
									
										
										
										
											1992-12-14 23:25:04 +00:00
										 |  |  | 		return list | 
					
						
							| 
									
										
										
										
											1992-12-21 14:33:18 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	def unpack_farray(self, n, unpack_item): | 
					
						
							|  |  |  | 		list = [] | 
					
						
							|  |  |  | 		for i in range(n): | 
					
						
							|  |  |  | 			list.append(unpack_item()) | 
					
						
							|  |  |  | 		return list | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	def unpack_array(self, unpack_item): | 
					
						
							|  |  |  | 		n = self.unpack_uint() | 
					
						
							|  |  |  | 		return self.unpack_farray(n, unpack_item) |