| 
									
										
										
										
											2016-07-22 18:47:04 -04:00
										 |  |  | import sys | 
					
						
							|  |  |  | import os | 
					
						
							|  |  |  | import struct | 
					
						
							|  |  |  | from array import array | 
					
						
							|  |  |  | from collections import namedtuple | 
					
						
							| 
									
										
										
										
											2017-03-27 16:05:26 +02:00
										 |  |  | from datetime import datetime | 
					
						
							| 
									
										
										
										
											2016-07-22 18:47:04 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | ttinfo = namedtuple('ttinfo', ['tt_gmtoff', 'tt_isdst', 'tt_abbrind']) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class TZInfo: | 
					
						
							|  |  |  |     def __init__(self, transitions, type_indices, ttis, abbrs): | 
					
						
							|  |  |  |         self.transitions = transitions | 
					
						
							|  |  |  |         self.type_indices = type_indices | 
					
						
							|  |  |  |         self.ttis = ttis | 
					
						
							|  |  |  |         self.abbrs = abbrs | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @classmethod | 
					
						
							|  |  |  |     def fromfile(cls, fileobj): | 
					
						
							|  |  |  |         if fileobj.read(4).decode() != "TZif": | 
					
						
							|  |  |  |             raise ValueError("not a zoneinfo file") | 
					
						
							|  |  |  |         fileobj.seek(20) | 
					
						
							|  |  |  |         header = fileobj.read(24) | 
					
						
							|  |  |  |         tzh = (tzh_ttisgmtcnt, tzh_ttisstdcnt, tzh_leapcnt, | 
					
						
							|  |  |  |                tzh_timecnt, tzh_typecnt, tzh_charcnt) = struct.unpack(">6l", header) | 
					
						
							|  |  |  |         transitions = array('i') | 
					
						
							|  |  |  |         transitions.fromfile(fileobj, tzh_timecnt) | 
					
						
							|  |  |  |         if sys.byteorder != 'big': | 
					
						
							|  |  |  |             transitions.byteswap() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         type_indices = array('B') | 
					
						
							|  |  |  |         type_indices.fromfile(fileobj, tzh_timecnt) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         ttis = [] | 
					
						
							|  |  |  |         for i in range(tzh_typecnt): | 
					
						
							|  |  |  |             ttis.append(ttinfo._make(struct.unpack(">lbb", fileobj.read(6)))) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         abbrs = fileobj.read(tzh_charcnt) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self = cls(transitions, type_indices, ttis, abbrs) | 
					
						
							|  |  |  |         self.tzh = tzh | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return self | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def dump(self, stream, start=None, end=None): | 
					
						
							|  |  |  |         for j, (trans, i) in enumerate(zip(self.transitions, self.type_indices)): | 
					
						
							|  |  |  |             utc = datetime.utcfromtimestamp(trans) | 
					
						
							|  |  |  |             tti = self.ttis[i] | 
					
						
							|  |  |  |             lmt = datetime.utcfromtimestamp(trans + tti.tt_gmtoff) | 
					
						
							|  |  |  |             abbrind = tti.tt_abbrind | 
					
						
							|  |  |  |             abbr = self.abbrs[abbrind:self.abbrs.find(0, abbrind)].decode() | 
					
						
							|  |  |  |             if j > 0: | 
					
						
							|  |  |  |                 prev_tti = self.ttis[self.type_indices[j - 1]] | 
					
						
							|  |  |  |                 shift = " %+g" % ((tti.tt_gmtoff - prev_tti.tt_gmtoff) / 3600) | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 shift = '' | 
					
						
							|  |  |  |             print("%s UTC = %s %-5s isdst=%d" % (utc, lmt, abbr, tti[1]) + shift, file=stream) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @classmethod | 
					
						
							|  |  |  |     def zonelist(cls, zonedir='/usr/share/zoneinfo'): | 
					
						
							|  |  |  |         zones = [] | 
					
						
							|  |  |  |         for root, _, files in os.walk(zonedir): | 
					
						
							|  |  |  |             for f in files: | 
					
						
							|  |  |  |                 p = os.path.join(root, f) | 
					
						
							|  |  |  |                 with open(p, 'rb') as o: | 
					
						
							|  |  |  |                     magic =  o.read(4) | 
					
						
							|  |  |  |                 if magic == b'TZif': | 
					
						
							|  |  |  |                     zones.append(p[len(zonedir) + 1:]) | 
					
						
							|  |  |  |         return zones | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if __name__ == '__main__': | 
					
						
							|  |  |  |     if len(sys.argv) < 2: | 
					
						
							|  |  |  |         zones = TZInfo.zonelist() | 
					
						
							|  |  |  |         for z in zones: | 
					
						
							|  |  |  |             print(z) | 
					
						
							|  |  |  |         sys.exit() | 
					
						
							|  |  |  |     filepath = sys.argv[1] | 
					
						
							|  |  |  |     if not filepath.startswith('/'): | 
					
						
							|  |  |  |         filepath = os.path.join('/usr/share/zoneinfo', filepath) | 
					
						
							|  |  |  |     with open(filepath, 'rb') as fileobj: | 
					
						
							|  |  |  |         tzi = TZInfo.fromfile(fileobj) | 
					
						
							|  |  |  |     tzi.dump(sys.stdout) |