class Zip::ExtraField::NTFS

Only Tag 0x0001 is supported
PKWARE NTFS Extra Field (0x000a)

def ==(other)

def ==(other)
  @mtime == other.mtime &&
    @atime == other.atime &&
    @ctime == other.ctime
end

def from_ntfs_time(ntfs_time)

def from_ntfs_time(ntfs_time)
  ::Zip::DOSTime.at(ntfs_time / WINDOWS_TICK - SEC_TO_UNIX_EPOCH)
end

def initialize(binstr = nil)

def initialize(binstr = nil)
  @ctime = nil
  @mtime = nil
  @atime = nil
  binstr && merge(binstr)
end

def merge(binstr)

def merge(binstr)
  return if binstr.empty?
  size, content = initial_parse(binstr)
  (size && content) || return
  content = content[4..-1]
  tags = parse_tags(content)
  tag1 = tags[1]
  return unless tag1
  ntfs_mtime, ntfs_atime, ntfs_ctime = tag1.unpack('Q<Q<Q<')
  ntfs_mtime && @mtime ||= from_ntfs_time(ntfs_mtime)
  ntfs_atime && @atime ||= from_ntfs_time(ntfs_atime)
  ntfs_ctime && @ctime ||= from_ntfs_time(ntfs_ctime)
end

def pack_for_c_dir

But 7-zip for Windows only stores at central dir
def pack_for_c_dir
  # reserved 0 and tag 1
  s = [0, 1].pack('Vv')
  tag1 = ''.force_encoding(Encoding::BINARY)
  if @mtime
    tag1 << [to_ntfs_time(@mtime)].pack('Q<')
    if @atime
      tag1 << [to_ntfs_time(@atime)].pack('Q<')
      tag1 << [to_ntfs_time(@ctime)].pack('Q<') if @ctime
    end
  end
  s << [tag1.bytesize].pack('v') << tag1
  s
end

def pack_for_local

Info-ZIP note states this extra field is stored at local header
def pack_for_local
  pack_for_c_dir
end

def parse_tags(content)

def parse_tags(content)
  return {} if content.nil?
  tags = {}
  i = 0
  while i < content.bytesize
    tag, size = content[i, 4].unpack('vv')
    i += 4
    break unless tag && size
    value = content[i, size]
    i += size
    tags[tag] = value
  end
  tags
end

def to_ntfs_time(time)

def to_ntfs_time(time)
  ((time.to_f + SEC_TO_UNIX_EPOCH) * WINDOWS_TICK).to_i
end