class Gem::Package::TarHeader

tar 文件的头部信息

常量

FIELDS

tar 头部中的字段

PACK_FORMAT

tar 头部的打包格式

UNPACK_FORMAT

tar 头部的解包格式

公共类方法

from(stream) 点击切换源码

从 IO stream 创建一个 tar 头部

# File rubygems/package/tar_header.rb, line 103
def self.from(stream)
  header = stream.read 512
  return EMPTY if header == EMPTY_HEADER

  fields = header.unpack UNPACK_FORMAT

  new name: fields.shift,
      mode: strict_oct(fields.shift),
      uid: oct_or_256based(fields.shift),
      gid: oct_or_256based(fields.shift),
      size: strict_oct(fields.shift),
      mtime: strict_oct(fields.shift),
      checksum: strict_oct(fields.shift),
      typeflag: fields.shift,
      linkname: fields.shift,
      magic: fields.shift,
      version: strict_oct(fields.shift),
      uname: fields.shift,
      gname: fields.shift,
      devmajor: strict_oct(fields.shift),
      devminor: strict_oct(fields.shift),
      prefix: fields.shift,

      empty: false
end
new(vals) 点击切换源码

使用 vals 创建一个新的 TarHeader

# File rubygems/package/tar_header.rb, line 149
def initialize(vals)
  unless vals[:name] && vals[:size] && vals[:prefix] && vals[:mode]
    raise ArgumentError, ":name, :size, :prefix and :mode required"
  end

  @checksum = vals[:checksum] || ""
  @devmajor = vals[:devmajor] || 0
  @devminor = vals[:devminor] || 0
  @gid = vals[:gid] || 0
  @gname = vals[:gname] || "wheel"
  @linkname = vals[:linkname]
  @magic = vals[:magic] || "ustar"
  @mode = vals[:mode]
  @mtime = vals[:mtime] || 0
  @name = vals[:name]
  @prefix = vals[:prefix]
  @size = vals[:size]
  @typeflag = vals[:typeflag]
  @typeflag = "0" if @typeflag.nil? || @typeflag.empty?
  @uid = vals[:uid] || 0
  @uname = vals[:uname] || "wheel"
  @version = vals[:version] || "00"

  @empty = vals[:empty]
end
oct_or_256based(str) 点击切换源码
# File rubygems/package/tar_header.rb, line 136
def self.oct_or_256based(str)
  # \x80 flags a positive 256-based number
  # \ff flags a negative 256-based number
  # In case we have a match, parse it as a signed binary value
  # in big-endian order, except that the high-order bit is ignored.

  return str.unpack1("@4N") if /\A[\x80\xff]/n.match?(str)
  strict_oct(str)
end
strict_oct(str) 点击切换源码
# File rubygems/package/tar_header.rb, line 129
def self.strict_oct(str)
  str.strip!
  return str.oct if /\A[0-7]*\z/.match?(str)

  raise ArgumentError, "#{str.inspect} is not an octal string"
end

公共实例方法

empty?() 点击切换源码

tar 条目是否为空?

# File rubygems/package/tar_header.rb, line 194
def empty?
  @empty
end
full_name() 点击切换源码

头部的完整名称,包括前缀

# File rubygems/package/tar_header.rb, line 234
def full_name
  if prefix != ""
    File.join prefix, name
  else
    name
  end
end
update_checksum() 点击切换源码

更新 TarHeader 的校验和

# File rubygems/package/tar_header.rb, line 226
def update_checksum
  header = header " " * 8
  @checksum = oct calculate_checksum(header), 6
end

私有实例方法

calculate_checksum(header) 点击切换源码
# File rubygems/package/tar_header.rb, line 244
def calculate_checksum(header)
  header.sum(0)
end
header(checksum = @checksum) 点击切换源码
# File rubygems/package/tar_header.rb, line 248
def header(checksum = @checksum)
  header = [
    name,
    oct(mode, 7),
    oct(uid, 7),
    oct(gid, 7),
    oct(size, 11),
    oct(mtime, 11),
    checksum,
    " ",
    typeflag,
    linkname,
    magic,
    oct(version, 2),
    uname,
    gname,
    oct(devmajor, 7),
    oct(devminor, 7),
    prefix,
  ]

  header = header.pack PACK_FORMAT

  header.ljust 512, "\0"
end
oct(num, len) 点击切换源码
# File rubygems/package/tar_header.rb, line 274
def oct(num, len)
  format("%0#{len}o", num)
end