module MiniMagick
class Image
# @return [String] The location of the current working file
attr_writer :path
def path_for_windows_quote_space(path)
path = Pathname.new(@path).to_s
# For Windows, if a path contains space char, you need to quote it, otherwise you SHOULD NOT quote it.
# If you quote a path that does not contains space, it will not work.
@path.include?(' ') ? path.inspect : path
end
def path
run_queue if @command_queued
MiniMagick::Utilities.windows? ? path_for_windows_quote_space(@path) : @path
end
# Class Methods
# -------------
class << self
# This is the primary loading method used by all of the other class methods.
#
# Use this to pass in a stream object. Must respond to Object#read(size) or be a binary string object (BLOBBBB)
#
# As a change from the old API, please try and use IOStream objects.
# They are much, much better and more efficient!
#
# Probably easier to use the #open method if you want to open a file or a URL.
#
# @param stream [IOStream, String] Some kind of stream object that needs to be read or is a binary String blob!
# @param ext [String] A manual extension to use for reading the file. Not required, but if you are having issues,
# give this a try.
# @return [Image]
def read(stream, ext = nil)
if stream.is_a?(String)
stream = StringIO.new(stream)
elsif stream.is_a?(StringIO)
# Do nothing, we want a StringIO-object
elsif stream.respond_to? :path
if File.respond_to?(:binread)
stream = StringIO.new File.binread(stream.path.to_s)
else
stream = StringIO.new File.open(stream.path.to_s, 'rb') { |f| f.read }
end
end
create(ext) do |f|
while chunk = stream.read(8192)
f.write(chunk)
end
end
end
# @deprecated Please use Image.read instead!
def from_blob(blob, ext = nil)
warn 'Warning: MiniMagick::Image.from_blob method is deprecated. Instead, please use Image.read'
create(ext) { |f| f.write(blob) }
end
# Creates an image object from a binary string blob which contains raw pixel data (i.e. no header data).
#
# === Returns
#
# * [Image] The loaded image.
#
# === Parameters
#
# * [blob] String -- Binary string blob containing raw pixel data.
# * [columns] Integer -- Number of columns.
# * [rows] Integer -- Number of rows.
# * [depth] Integer -- Bit depth of the encoded pixel data.
# * [map] String -- A code for the mapping of the pixel data. Example: 'gray' or 'rgb'.
# * [format] String -- The file extension of the image format to be used when creating the image object.
# Defaults to 'png'.
#
def import_pixels(blob, columns, rows, depth, map, format = 'png')
# Create an image object with the raw pixel data string:
image = create('.dat', false) { |f| f.write(blob) }
# Use ImageMagick to convert the raw data file to an image file of the desired format:
converted_image_path = image.path[0..-4] + format
arguments = ['-size', "#{columns}x#{rows}", '-depth', "#{depth}", "#{map}:#{image.path}", "#{converted_image_path}"]
cmd = CommandBuilder.new('convert', *arguments) # Example: convert -size 256x256 -depth 16 gray:blob.dat blob.png
image.run(cmd)
# Update the image instance with the path of the properly formatted image, and return:
image.path = converted_image_path
image
end
# Opens a specific image file either on the local file system or at a URI.
#
# Use this if you don't want to overwrite the image file.
#
# Extension is either guessed from the path or you can specify it as a second parameter.
#
# If you pass in what looks like a URL, we require 'open-uri' before opening it.
#
# @param file_or_url [String] Either a local file path or a URL that open-uri can read
# @param ext [String] Specify the extension you want to read it as
# @return [Image] The loaded image
def open(file_or_url, ext = nil)
file_or_url = file_or_url.to_s # Force it to be a String... Hell or high water
if file_or_url.include?('://')
require 'open-uri'
ext ||= File.extname(URI.parse(file_or_url).path)
Kernel.open(file_or_url) do |f|
read(f, ext)
end
else
ext ||= File.extname(file_or_url)
File.open(file_or_url, 'rb') do |f|
read(f, ext)
end
end
end
# @deprecated Please use MiniMagick::Image.open(file_or_url) now
def from_file(file, ext = nil)
warn 'Warning: MiniMagick::Image.from_file is now deprecated. Please use Image.open'
open(file, ext)
end
# Used to create a new Image object data-copy. Not used to "paint" or that kind of thing.
#
# Takes an extension in a block and can be used to build a new Image object. Used
# by both #open and #read to create a new object! Ensures we have a good tempfile!
#
# @param ext [String] Specify the extension you want to read it as
# @param validate [Boolean] If false, skips validation of the created image. Defaults to true.
# @yield [IOStream] You can #write bits to this object to create the new Image
# @return [Image] The created image
def create(ext = nil, validate = MiniMagick.validate_on_create, &block)
begin
tempfile = Tempfile.new(['mini_magick', ext.to_s.downcase])
tempfile.binmode
block.call(tempfile)
tempfile.close
image = new(tempfile.path, tempfile)
fail MiniMagick::Invalid if validate && !image.valid?
return image
ensure
tempfile.close if tempfile
end
end
end
# Create a new MiniMagick::Image object
#
# _DANGER_: The file location passed in here is the *working copy*. That is, it gets *modified*.
# you can either copy it yourself or use the MiniMagick::Image.open(path) method which creates a
# temporary file for you and protects your original!
#
# @param input_path [String] The location of an image file
# @todo Allow this to accept a block that can pass off to Image#combine_options
def initialize(input_path, tempfile = nil)
@path = input_path
@tempfile = tempfile # ensures that the tempfile will stick around until this image is garbage collected.
@info = {}
reset_queue
end
def reset_queue
@command_queued = false
@queue = MiniMagick::CommandBuilder.new('mogrify')
@info.clear
end
def run_queue
return nil unless @command_queued
@queue << (MiniMagick::Utilities.windows? ? path_for_windows_quote_space(@path) : @path)
run(@queue)
reset_queue
end
# Checks to make sure that MiniMagick can read the file and understand it.
#
# This uses the 'identify' command line utility to check the file. If you are having
# issues with this, then please work directly with the 'identify' command and see if you
# can figure out what the issue is.
#
# @return [Boolean]
def valid?
run_command('identify', path)
true
rescue MiniMagick::Invalid
false
end
def info(key)
run_queue if @command_queued
@info[key]
end
# A rather low-level way to interact with the "identify" command. No nice API here, just
# the crazy stuff you find in ImageMagick. See the examples listed!
#
# @example
# image["format"] #=> "TIFF"
# image["height"] #=> 41 (pixels)
# image["width"] #=> 50 (pixels)
# image["colorspace"] #=> "DirectClassRGB"
# image["dimensions"] #=> [50, 41]
# image["size"] #=> 2050 (bits)
# image["original_at"] #=> 2005-02-23 23:17:24 +0000 (Read from Exif data)
# image["EXIF:ExifVersion"] #=> "0220" (Can read anything from Exif)
#
# @param format [String] A format for the "identify" command
# @see For reference see http://www.imagemagick.org/script/command-line-options.php#format
# @return [String, Numeric, Array, Time, Object] Depends on the method called! Defaults to String for unknown commands
def [](value)
retrieved = info(value)
return retrieved unless retrieved.nil?
# Why do I go to the trouble of putting in newlines? Because otherwise animated gifs screw everything up
retrieved = case value.to_s
when 'colorspace'
run_command('identify', '-format', '%r\n', path).split("\n")[0].strip
when 'format'
run_command('identify', '-format', '%m\n', path).split("\n")[0]
when 'dimensions', 'width', 'height'
width_height = run_command(
'identify', '-format', MiniMagick::Utilities.windows? ? '"%w %h\n"' : '%w %h\n', path
).split("\n")[0].split.map { |v| v.to_i }
@info[:width] = width_height[0]
@info[:height] = width_height[1]
@info[:dimensions] = width_height
nil
when 'size'
File.size(path) # Do this because calling identify -format "%b" on an animated gif fails!
when 'original_at'
# Get the EXIF original capture as a Time object
Time.local(*self['EXIF:DateTimeOriginal'].split(/:|\s+/)) rescue nil
when /^EXIF\:/i
result = run_command('identify', '-format', "%[#{value}]", path).chomp
if result.include?(',')
read_character_data(result)
else
result
end
else
run_command('identify', '-format', value, path).split("\n")[0]
end
@info[value] = retrieved unless retrieved.nil?
@info[value]
end
# Sends raw commands to imagemagick's `mogrify` command. The image path is automatically appended to the command.
#
# Remember, we are always acting on this instance of the Image when messing with this.
#
# @return [String] Whatever the result from the command line is. May not be terribly useful.
def <<(*args)
run_command('mogrify', *args << path)
end
# This is used to change the format of the image. That is, from "tiff to jpg" or something like that.
# Once you run it, the instance is pointing to a new file with a new extension!
#
# *DANGER*: This renames the file that the instance is pointing to. So, if you manually opened the
# file with Image.new(file_path)... Then that file is DELETED! If you used Image.open(file) then
# you are OK. The original file will still be there. But, any changes to it might not be...
#
# Formatting an animation into a non-animated type will result in ImageMagick creating multiple
# pages (starting with 0). You can choose which page you want to manipulate. We default to the
# first page.
#
# If you would like to convert between animated formats, pass nil as your
# page and ImageMagick will copy all of the pages.
#
# @param format [String] The target format... Like 'jpg', 'gif', 'tiff', etc.
# @param page [Integer] If this is an animated gif, say which 'page' you want
# with an integer. Default 0 will convert only the first page; 'nil' will
# convert all pages.
# @return [nil]
def format(format, page = 0)
run_queue if @command_queued
c = CommandBuilder.new('mogrify', '-format', format)
yield c if block_given?
c << (page ? "#{path}[#{page}]" : path)
run(c)
old_path = path
self.path = path.sub(/(\.\w*)?$/, (page ? ".#{format}" : "-0.#{format}"))
File.delete(old_path) if old_path != path
unless File.exist?(path)
fail MiniMagick::Error, "Unable to format to #{format}"
end
end
# Collapse images with sequences to the first frame (i.e. animated gifs) and
# preserve quality
def collapse!
run_command('mogrify', '-quality', '100', "#{path}[0]")
end
# Writes the temporary file out to either a file location (by passing in a String) or by
# passing in a Stream that you can #write(chunk) to repeatedly
#
# @param output_to [IOStream, String] Some kind of stream object that needs to be read or a file path as a String
# @return [IOStream, Boolean] If you pass in a file location [String] then you get a success boolean. If its a stream, you get it back.
# Writes the temporary image that we are using for processing to the output path
def write(output_to)
run_queue if @command_queued
if output_to.kind_of?(String) || output_to.kind_of?(Pathname) || !output_to.respond_to?(:write)
FileUtils.copy_file path, output_to
if MiniMagick.validate_on_write
run_command(
'identify', MiniMagick::Utilities.windows? ? path_for_windows_quote_space(output_to.to_s) : output_to.to_s
) # Verify that we have a good image
end
else # stream
File.open(path, 'rb') do |f|
f.binmode
while chunk = f.read(8192)
output_to.write(chunk)
end
end
output_to
end
end
# Gives you raw image data back
# @return [String] binary string
def to_blob
run_queue if @command_queued
f = File.new path
f.binmode
f.read
ensure
f.close if f
end
def mime_type
format = self[:format]
'image/' + format.to_s.downcase
end
# If an unknown method is called then it is sent through the mogrify program
# Look here to find all the commands (http://www.imagemagick.org/script/mogrify.php)
def method_missing(symbol, *args)
@queue.send(symbol, *args)
@command_queued = true
end
# You can use multiple commands together using this method. Very easy to use!
#
# @example
# image.combine_options do |c|
# c.draw "image Over 0,0 10,10 '#{MINUS_IMAGE_PATH}'"
# c.thumbnail "300x500>"
# c.background background
# end
#
# @yieldparam command [CommandBuilder]
def combine_options
if block_given?
yield @queue
@command_queued = true
end
end
def composite(other_image, output_extension = 'jpg', mask = nil, &block)
run_queue if @command_queued
begin
second_tempfile = Tempfile.new(output_extension)
second_tempfile.binmode
ensure
second_tempfile.close
end
command = CommandBuilder.new('composite')
block.call(command) if block
command.push(other_image.path)
command.push(path)
command.push(mask.path) unless mask.nil?
command.push(second_tempfile.path)
run(command)
Image.new(second_tempfile.path, second_tempfile)
end
def run_command(command, *args)
run_queue if @command_queued
if command == 'identify'
args.unshift '-ping' # -ping "efficiently determine image characteristics."
args.unshift '-quiet' if MiniMagick.mogrify? # graphicsmagick has no -quiet option.
end
run(CommandBuilder.new(command, *args))
end
def run(command_builder)
command = command_builder.command
sub = Subexec.run(command, :timeout => MiniMagick.timeout)
if sub.exitstatus != 0
# Clean up after ourselves in case of an error
destroy!
# Raise the appropriate error
if sub.output =~ /no decode delegate/i || sub.output =~ /did not return an image/i
fail Invalid, sub.output
else
# TODO: should we do something different if the command times out ...?
# its definitely better for logging.. Otherwise we don't really know
fail Error, "Command (#{command.inspect.gsub("\\", "")}) failed: #{{ :status_code => sub.exitstatus, :output => sub.output }.inspect}"
end
else
sub.output
end
end
def destroy!
return if @tempfile.nil?
File.unlink(path) if File.exist?(path)
@tempfile = nil
end
private
# Sometimes we get back a list of character values
def read_character_data(list_of_characters)
chars = list_of_characters.gsub(' ', '').split(',')
result = ''
chars.each do |val|
result << ('%c' % val.to_i)
end
result
end
end
end