require "json" module MiniMagick class Image # @private class Info ASCII_ENCODED_EXIF_KEYS = %w[ExifVersion FlashPixVersion] def initialize(path) @path = path @info = {} end def [](value, *args) case value when "format", "width", "height", "dimensions", "size", "human_size" cheap_info(value) when "colorspace" colorspace when "resolution" resolution(*args) when "signature" signature when /^EXIF\:/i raw_exif(value) when "exif" exif when "data" data else raw(value) end end def clear @info.clear end def cheap_info(value) @info.fetch(value) do format, width, height, size = parse_warnings(self["%m %w %h %b"]).split(" ") path = @path path = path.match(/\[\d+\]$/).pre_match if path =~ /\[\d+\]$/ @info.update( "format" => format, "width" => Integer(width), "height" => Integer(height), "dimensions" => [Integer(width), Integer(height)], "size" => File.size(path), "human_size" => size, ) @info.fetch(value) end rescue ArgumentError, TypeError raise MiniMagick::Invalid, "image data can't be read" end def parse_warnings(raw_info) return raw_info unless raw_info.split("\n").size > 1 raw_info.split("\n").each do |line| # must match "%m %w %h %b" return line if line.match?(/^[A-Z]+ \d+ \d+ \d+(|\.\d+)([KMGTPEZY]{0,1})B$/) end raise TypeError end def colorspace @info["colorspace"] ||= self["%r"] end def resolution(unit = nil) output = identify do |b| b.units unit if unit b.format "%x %y" end output.split(" ").map(&:to_i) end def raw_exif(value) self["%[#{value}]"] end def exif @info["exif"] ||= ( hash = {} output = self["%[EXIF:*]"] output.each_line do |line| line = line.chomp("\n") if match = line.match(/^exif:/) key, value = match.post_match.split("=", 2) value = decode_comma_separated_ascii_characters(value) if ASCII_ENCODED_EXIF_KEYS.include?(key) hash[key] = value else hash[hash.keys.last] << "\n#{line}" end end hash ) end def raw(value) @info["raw:#{value}"] ||= identify { |b| b.format(value) } end def signature @info["signature"] ||= self["%#"] end def data @info["data"] ||= ( json = MiniMagick.convert do |convert| convert << path convert << "json:" end data = JSON.parse(json) data = data.fetch(0) if data.is_a?(Array) data.fetch("image") ) end def identify MiniMagick.identify do |builder| yield builder if block_given? builder << path end end private def decode_comma_separated_ascii_characters(encoded_value) return encoded_value unless encoded_value.include?(',') encoded_value.scan(/\d+/).map(&:to_i).map(&:chr).join end def path value = @path value += "[0]" unless value =~ /\[\d+\]$/ value end end end end