1
0
Fork 0
mirror of https://github.com/rails/rails.git synced 2022-11-09 12:12:34 -05:00
rails--rails/activestorage/app/models/active_storage/variant.rb

83 lines
3.9 KiB
Ruby
Raw Normal View History

# frozen_string_literal: true
2017-07-21 17:34:28 -04:00
# Image blobs can have variants that are the result of a set of transformations applied to the original.
2017-07-24 13:05:15 -04:00
# These variants are used to create thumbnails, fixed-size avatars, or any other derivative image from the
# original.
#
# Variants rely on {MiniMagick}[https://github.com/minimagick/minimagick] for the actual transformations
# of the file, so you must add <tt>gem "mini_magick"</tt> to your Gemfile if you wish to use variants.
2017-07-24 13:05:15 -04:00
#
# Note that to create a variant it's necessary to download the entire blob file from the service and load it
# into memory. The larger the image, the more memory is used. Because of this process, you also want to be
# considerate about when the variant is actually processed. You shouldn't be processing variants inline in a
# template, for example. Delay the processing to an on-demand controller, like the one provided in
# ActiveStorage::VariantsController.
2017-07-24 13:05:15 -04:00
#
# To refer to such a delayed on-demand variant, simply link to the variant through the resolved route provided
# by Active Storage like so:
#
# <%= image_tag url_for(Current.user.avatar.variant(resize: "100x100")) %>
#
# This will create a URL for that specific blob with that specific variant, which the ActiveStorage::VariantsController
2017-07-24 13:05:15 -04:00
# can then produce on-demand.
#
# When you do want to actually produce the variant needed, call +processed+. This will check that the variant
2017-07-24 13:05:15 -04:00
# has already been processed and uploaded to the service, and, if so, just return that. Otherwise it will perform
# the transformations, upload the variant to the service, and return itself again. Example:
#
# avatar.variant(resize: "100x100").processed.service_url
#
# This will create and process a variant of the avatar blob that's constrained to a height and width of 100.
# Then it'll upload said variant to the service according to a derivative key of the blob and the transformations.
#
# A list of all possible transformations is available at https://www.imagemagick.org/script/mogrify.php. You can
# combine as many as you like freely:
#
# avatar.variant(resize: "100x100", monochrome: true, flip: "-90")
2017-07-11 12:53:17 -04:00
class ActiveStorage::Variant
attr_reader :blob, :variation
delegate :service, to: :blob
2017-09-28 16:43:37 -04:00
def initialize(blob, variation_or_variation_key)
@blob, @variation = blob, ActiveStorage::Variation.wrap(variation_or_variation_key)
2017-07-11 12:53:17 -04:00
end
2017-07-24 13:05:15 -04:00
# Returns the variant instance itself after it's been processed or an existing processing has been found on the service.
2017-07-20 18:35:15 -04:00
def processed
2017-07-23 17:28:45 -04:00
process unless processed?
2017-07-20 18:35:15 -04:00
self
end
2017-07-24 13:05:15 -04:00
# Returns a combination key of the blob and the variation that together identifies a specific variant.
2017-07-21 16:51:31 -04:00
def key
"variants/#{blob.key}/#{Digest::SHA256.hexdigest(variation.key)}"
2017-07-11 12:53:17 -04:00
end
2017-07-24 13:05:15 -04:00
# Returns the URL of the variant on the service. This URL is intended to be short-lived for security and not used directly
# with users. Instead, the +service_url+ should only be exposed as a redirect from a stable, possibly authenticated URL.
# Hiding the +service_url+ behind a redirect also gives you the power to change services without updating all URLs. And
# it allows permanent URLs that redirect to the +service_url+ to be cached in the view.
2017-07-24 13:05:15 -04:00
#
# Use <tt>url_for(variant)</tt> (or the implied form, like +link_to variant+ or +redirect_to variant+) to get the stable URL
# for a variant that points to the ActiveStorage::VariantsController, which in turn will use this +service_call+ method
2017-07-24 13:05:15 -04:00
# for its redirection.
2017-09-28 16:43:37 -04:00
def service_url(expires_in: service.url_expires_in, disposition: :inline)
service.url key, expires_in: expires_in, disposition: disposition, filename: blob.filename, content_type: blob.content_type
2017-07-11 12:53:17 -04:00
end
2017-07-11 12:53:17 -04:00
private
2017-07-23 17:28:45 -04:00
def processed?
service.exist?(key)
end
2017-07-20 18:35:15 -04:00
def process
2017-07-21 16:51:31 -04:00
service.upload key, transform(service.download(blob.key))
2017-07-11 12:53:17 -04:00
end
def transform(io)
2017-07-23 17:06:45 -04:00
require "mini_magick"
2017-07-21 16:51:31 -04:00
File.open MiniMagick::Image.read(io).tap { |image| variation.transform(image) }.path
2017-07-11 12:53:17 -04:00
end
end