All patches and comments are welcome. Please squash your changes to logical
commits before using git-format-patch and git-send-email to
patches@git.madduck.net.
If you'd read over the Git project's submission guidelines and adhered to them,
I'd be especially grateful.
   1 local tz_info_mt = require "luatz.tzinfo".tz_info_mt
 
   2 local tt_info_mt = require "luatz.tzinfo".tt_info_mt
 
   4 local read_int32be, read_int64be
 
   6 if string.unpack then -- Only available in Lua 5.3+
 
   7         function read_int32be(fd)
 
   8                 local data, err = fd:read(4)
 
   9                 if data == nil then return nil, err end
 
  10                 return string.unpack(">i4", data)
 
  13         function read_int64be(fd)
 
  14                 local data, err = fd:read(8)
 
  15                 if data == nil then return nil, err end
 
  16                 return string.unpack(">i8", data)
 
  19         function read_int32be(fd)
 
  20                 local data, err = fd:read(4)
 
  21                 if data == nil then return nil, err end
 
  22                 local o1, o2, o3, o4 = data:byte(1, 4)
 
  24                 local unsigned = o4 + o3*2^8 + o2*2^16 + o1*2^24
 
  25                 if unsigned >= 2^31 then
 
  26                         return unsigned - 2^32
 
  32         function read_int64be(fd)
 
  33                 local data, err = fd:read(8)
 
  34                 if data == nil then return nil, err end
 
  35                 local o1, o2, o3, o4, o5, o6, o7, o8 = data:byte(1, 8)
 
  37                 local unsigned = o8 + o7*2^8 + o6*2^16 + o5*2^24 + o4*2^32 + o3*2^40 + o2*2^48 + o1*2^56
 
  38                 if unsigned >= 2^63 then
 
  39                         return unsigned - 2^64
 
  46 local function read_flags ( fd , n )
 
  47         local data , err = fd:read ( n )
 
  48         if data == nil then return nil , err end
 
  52                 res[i] = data:byte(i,i) ~= 0
 
  57 local fifteen_nulls = ("\0"):rep(15)
 
  58 local function read_tz ( fd )
 
  59         assert ( fd:read(4) == "TZif" , "Invalid TZ file" )
 
  60         local version = assert ( fd:read(1) )
 
  61         if version == "\0" or version == "2" or version == "3" then
 
  62                 local MIN_TIME = -2^32+1
 
  64                 assert ( assert ( fd:read(15) ) == fifteen_nulls , "Expected 15 nulls" )
 
  66                 -- The number of UTC/local indicators stored in the file.
 
  67                 local tzh_ttisgmtcnt = assert ( read_int32be ( fd ) )
 
  69                 -- The number of standard/wall indicators stored in the file.
 
  70                 local tzh_ttisstdcnt = assert ( read_int32be ( fd ) )
 
  72                 -- The number of leap seconds for which data is stored in the file.
 
  73                 local tzh_leapcnt = assert ( read_int32be ( fd ) )
 
  75                 -- The number of "transition times" for which data is stored in the file.
 
  76                 local tzh_timecnt = assert ( read_int32be ( fd ) )
 
  78                 -- The number of "local time types" for which data is stored in the file (must not be zero).
 
  79                 local tzh_typecnt = assert ( read_int32be ( fd ) )
 
  81                 -- The number of characters of "timezone abbreviation strings" stored in the file.
 
  82                 local tzh_charcnt = assert ( read_int32be ( fd ) )
 
  84                 local transition_times = { }
 
  85                 for i=1, tzh_timecnt do
 
  86                         transition_times [ i ] = assert ( read_int32be ( fd ) )
 
  88                 local transition_time_ind = { assert ( fd:read ( tzh_timecnt ) ):byte ( 1 , -1 ) }
 
  91                 for i=1, tzh_typecnt do
 
  93                                 gmtoff = assert ( read_int32be ( fd ) ) ;
 
  94                                 isdst  = assert ( fd:read ( 1 ) ) ~= "\0" ;
 
  95                                 abbrind = assert ( fd:read ( 1 ) ):byte ( ) ;
 
  99                 local abbreviations = assert ( fd:read ( tzh_charcnt ) )
 
 101                 local leap_seconds = { }
 
 102                 for i=1, tzh_leapcnt do
 
 103                         leap_seconds [ i ] = {
 
 104                                 offset = assert ( read_int32be ( fd ) ) ;
 
 105                                 n = assert ( read_int32be ( fd ) ) ;
 
 109                 local isstd = assert ( read_flags ( fd , tzh_ttisstdcnt ) )
 
 111                 local isgmt = assert ( read_flags ( fd , tzh_ttisgmtcnt ) )
 
 115                 if version == "2" or version == "3" then
 
 117                         For version-2-format timezone files, the above header and data is followed by a second header and data,
 
 118                         identical in format except that eight bytes are used for each transition time or leap-second time.
 
 120                         assert(fd:read(4) == "TZif")
 
 121                         assert(fd:read(1) == version)
 
 122                         assert ( assert ( fd:read(15) ) == fifteen_nulls , "Expected 15 nulls" )
 
 126                         -- The number of UTC/local indicators stored in the file.
 
 127                         tzh_ttisgmtcnt = assert ( read_int32be ( fd ) )
 
 129                         -- The number of standard/wall indicators stored in the file.
 
 130                         tzh_ttisstdcnt = assert ( read_int32be ( fd ) )
 
 132                         -- The number of leap seconds for which data is stored in the file.
 
 133                         tzh_leapcnt = assert ( read_int32be ( fd ) )
 
 135                         -- The number of "transition times" for which data is stored in the file.
 
 136                         tzh_timecnt = assert ( read_int32be ( fd ) )
 
 138                         -- The number of "local time types" for which data is stored in the file (must not be zero).
 
 139                         tzh_typecnt = assert ( read_int32be ( fd ) )
 
 141                         -- The number of characters of "timezone abbreviation strings" stored in the file.
 
 142                         tzh_charcnt = assert ( read_int32be ( fd ) )
 
 144                         transition_times = { }
 
 145                         for i=1, tzh_timecnt do
 
 146                                 transition_times [ i ] = assert ( read_int64be ( fd ) )
 
 148                         transition_time_ind = { assert ( fd:read ( tzh_timecnt ) ):byte ( 1 , -1 ) }
 
 151                         for i=1, tzh_typecnt do
 
 153                                         gmtoff = assert ( read_int32be ( fd ) ) ;
 
 154                                         isdst  = assert ( fd:read ( 1 ) ) ~= "\0" ;
 
 155                                         abbrind = assert ( fd:read ( 1 ) ):byte ( ) ;
 
 159                         abbreviations = assert ( fd:read ( tzh_charcnt ) )
 
 162                         for i=1, tzh_leapcnt do
 
 163                                 leap_seconds [ i ] = {
 
 164                                         offset = assert ( read_int64be ( fd ) ) ;
 
 165                                         n = assert ( read_int32be ( fd ) ) ;
 
 169                         isstd = assert ( read_flags ( fd , tzh_ttisstdcnt ) )
 
 171                         isgmt = assert ( read_flags ( fd , tzh_ttisgmtcnt ) )
 
 174                         After the second header and data comes a newline-enclosed, POSIX-TZ-environment-variable-style string
 
 175                         for use in handling instants after the last transition time stored in the file
 
 176                         (with nothing between the newlines if there is no POSIX representation for such instants).
 
 180                         For version-3-format time zone files, the POSIX-TZ-style string may
 
 181                         use two minor extensions to the POSIX TZ format, as described in newtzset (3).
 
 182                         First, the hours part of its transition times may be signed and range from
 
 183                         -167 through 167 instead of the POSIX-required unsigned values
 
 184                         from 0 through 24.  Second, DST is in effect all year if it starts
 
 185                         January 1 at 00:00 and ends December 31 at 24:00 plus the difference
 
 186                         between daylight saving and standard time.
 
 189                         assert ( assert ( fd:read ( 1 ) ) == "\n" , "Expected newline at end of version 2 header" )
 
 191                         TZ = assert ( fd:read ( "*l" ) )
 
 197                 for i=1, tzh_typecnt do
 
 198                         local v = ttinfos [ i ]
 
 199                         v.abbr = abbreviations:sub ( v.abbrind+1 , v.abbrind+3 )
 
 200                         v.isstd = isstd [ i ] or false
 
 201                         v.isgmt = isgmt [ i ] or false
 
 202                         setmetatable ( v , tt_info_mt )
 
 206                 Use the first standard-time ttinfo structure in the file
 
 207                 (or simply the first ttinfo structure in the absence of a standard-time structure)
 
 208                 if either tzh_timecnt is zero or the time argument is less than the first transition time recorded in the file.
 
 212                         for i=1, tzh_ttisstdcnt do
 
 223                                 transition_time = MIN_TIME ;
 
 224                                 info = ttinfos [ first ] ;
 
 227                 for i=1, tzh_timecnt do
 
 229                                 transition_time = transition_times [ i ] ;
 
 230                                 info = ttinfos [ transition_time_ind [ i ]+1 ] ;
 
 233                 return setmetatable ( res , tz_info_mt )
 
 235                 error ( "Unsupported version" )
 
 239 local function read_tzfile ( path )
 
 240         local fd = assert ( io.open ( path , "rb" ) )
 
 241         local tzinfo = read_tz ( fd )
 
 248         read_tzfile = read_tzfile ;