2010-11-30 16:27:59 +08:00
|
|
|
module Zip
|
2013-06-03 15:56:24 +08:00
|
|
|
class ExtraField < Hash
|
2010-11-30 16:27:59 +08:00
|
|
|
ID_MAP = {}
|
|
|
|
|
2013-07-01 04:52:18 +08:00
|
|
|
def initialize(binstr = nil)
|
|
|
|
binstr and merge(binstr)
|
2010-11-30 16:27:59 +08:00
|
|
|
end
|
|
|
|
|
2013-07-01 04:52:18 +08:00
|
|
|
def extra_field_type_exist(binstr, id, len, i)
|
|
|
|
field_name = ID_MAP[id].name
|
|
|
|
if self.member?(field_name)
|
|
|
|
self[field_name].merge(binstr[i, len + 4])
|
|
|
|
else
|
|
|
|
field_obj = ID_MAP[id].new(binstr[i, len + 4])
|
|
|
|
self[field_name] = field_obj
|
2010-11-30 16:27:59 +08:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-07-01 04:52:18 +08:00
|
|
|
def extra_field_type_unknown(binstr, len, i)
|
|
|
|
create_unknown_item unless self['Unknown']
|
|
|
|
if !len || len + 4 > binstr[i..-1].bytesize
|
|
|
|
self['Unknown'] << binstr[i..-1]
|
|
|
|
return
|
2010-11-30 16:27:59 +08:00
|
|
|
end
|
2013-07-01 04:52:18 +08:00
|
|
|
self['Unknown'] << binstr[i, len + 4]
|
2010-11-30 16:27:59 +08:00
|
|
|
end
|
|
|
|
|
2013-07-01 04:52:18 +08:00
|
|
|
def create_unknown_item
|
|
|
|
s = ''
|
|
|
|
class << s
|
|
|
|
alias_method :to_c_dir_bin, :to_s
|
|
|
|
alias_method :to_local_bin, :to_s
|
|
|
|
end
|
|
|
|
self['Unknown'] = s
|
2010-11-30 16:27:59 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
def merge(binstr)
|
2012-02-14 06:03:34 +08:00
|
|
|
return if binstr.empty?
|
2013-07-01 04:52:18 +08:00
|
|
|
i = 0
|
2012-02-14 00:55:08 +08:00
|
|
|
while i < binstr.bytesize
|
2013-07-01 04:52:18 +08:00
|
|
|
id = binstr[i, 2]
|
|
|
|
len = binstr[i + 2, 2].to_s.unpack('v').first
|
2010-11-30 16:27:59 +08:00
|
|
|
if id && ID_MAP.member?(id)
|
2013-07-01 04:52:18 +08:00
|
|
|
extra_field_type_exist(binstr, id, len, i)
|
2010-11-30 16:27:59 +08:00
|
|
|
elsif id
|
2013-07-01 04:52:18 +08:00
|
|
|
create_unknown_item unless self['Unknown']
|
|
|
|
break unless extra_field_type_unknown(binstr, len, i)
|
2010-11-30 16:27:59 +08:00
|
|
|
end
|
2012-02-14 00:55:08 +08:00
|
|
|
i += len + 4
|
2010-11-30 16:27:59 +08:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def create(name)
|
2013-07-01 04:52:18 +08:00
|
|
|
unless field_class = ID_MAP.values.find { |k| k.name == name }
|
2011-11-17 17:48:42 +08:00
|
|
|
raise ZipError, "Unknown extra field '#{name}'"
|
2010-11-30 16:27:59 +08:00
|
|
|
end
|
2013-07-01 04:52:18 +08:00
|
|
|
self[name] = field_class.new
|
2010-11-30 16:27:59 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
def to_local_bin
|
2013-09-18 01:34:19 +08:00
|
|
|
self.map { |_, v| v.to_local_bin.force_encoding('BINARY') }.join
|
2010-11-30 16:27:59 +08:00
|
|
|
end
|
2013-07-01 04:52:18 +08:00
|
|
|
|
2010-11-30 16:27:59 +08:00
|
|
|
alias :to_s :to_local_bin
|
|
|
|
|
|
|
|
def to_c_dir_bin
|
2013-07-01 04:52:18 +08:00
|
|
|
self.map { |_, v| v.to_c_dir_bin }.join
|
2010-11-30 16:27:59 +08:00
|
|
|
end
|
|
|
|
|
2013-08-27 04:26:14 +08:00
|
|
|
def c_dir_size
|
2012-02-14 00:55:08 +08:00
|
|
|
to_c_dir_bin.bytesize
|
2010-11-30 16:27:59 +08:00
|
|
|
end
|
2013-07-01 04:52:18 +08:00
|
|
|
|
2013-08-27 04:26:14 +08:00
|
|
|
def local_size
|
2012-02-14 00:55:08 +08:00
|
|
|
to_local_bin.bytesize
|
2010-11-30 16:27:59 +08:00
|
|
|
end
|
2013-07-01 04:52:18 +08:00
|
|
|
|
2013-08-27 04:26:14 +08:00
|
|
|
alias :length :local_size
|
|
|
|
alias :size :local_size
|
2010-11-30 16:27:59 +08:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-07-01 04:52:18 +08:00
|
|
|
require 'zip/extra_field/generic'
|
|
|
|
require 'zip/extra_field/universal_time'
|
|
|
|
require 'zip/extra_field/unix'
|
2013-08-27 04:26:14 +08:00
|
|
|
|
2010-11-30 16:27:59 +08:00
|
|
|
# Copyright (C) 2002, 2003 Thomas Sondergaard
|
|
|
|
# rubyzip is free software; you can redistribute it and/or
|
|
|
|
# modify it under the terms of the ruby license.
|