Skip to content

Commit fbcc5cc

Browse files
committed
Add storage adapter
1 parent 88cc9f2 commit fbcc5cc

23 files changed

+578
-180
lines changed

app/models/alchemy/attachment.rb

+1-3
Original file line numberDiff line numberDiff line change
@@ -114,9 +114,7 @@ def allowed_filetypes
114114
private
115115

116116
def file_types
117-
ActiveStorage::Blob.joins(:attachments).merge(
118-
ActiveStorage::Attachment.where(record_type: name)
119-
).distinct.pluck(:content_type)
117+
Alchemy.storage_adapter.file_formats(name)
120118
end
121119
end
122120

app/models/alchemy/picture.rb

+12-49
Original file line numberDiff line numberDiff line change
@@ -28,17 +28,6 @@ class Picture < BaseRecord
2828
large: "240x180"
2929
}.with_indifferent_access.freeze
3030

31-
TRANSFORMATION_OPTIONS = [
32-
:crop,
33-
:crop_from,
34-
:crop_size,
35-
:flatten,
36-
:format,
37-
:quality,
38-
:size,
39-
:upsample
40-
]
41-
4231
include Alchemy::Logger
4332
include Alchemy::NameConversions
4433
include Alchemy::Taggable
@@ -68,7 +57,7 @@ class Picture < BaseRecord
6857

6958
# Image preprocessing class
7059
def self.preprocessor_class
71-
@_preprocessor_class ||= Preprocessor
60+
@_preprocessor_class ||= Alchemy.storage_adapter.preprocessor_class
7261
end
7362

7463
# Set a image preprocessing class
@@ -80,31 +69,7 @@ def self.preprocessor_class=(klass)
8069
@_preprocessor_class = klass
8170
end
8271

83-
attr_readonly(
84-
:legacy_image_file_format,
85-
:legacy_image_file_height,
86-
:legacy_image_file_name,
87-
:legacy_image_file_size,
88-
:legacy_image_file_uid,
89-
:legacy_image_file_width
90-
)
91-
92-
deprecate(
93-
:legacy_image_file_format,
94-
:legacy_image_file_height,
95-
:legacy_image_file_name,
96-
:legacy_image_file_size,
97-
:legacy_image_file_uid,
98-
:legacy_image_file_width,
99-
deprecator: Alchemy::Deprecation
100-
)
101-
102-
# Use ActiveStorage image processing
103-
has_one_attached :image_file do |attachable|
104-
# Only works in Rails 7.1
105-
preprocessor_class.new(attachable).call
106-
Preprocessor.generate_thumbs!(attachable)
107-
end
72+
include Alchemy.storage_adapter.picture_class_methods
10873

10974
validates_presence_of :image_file
11075
validate :image_file_type_allowed, :image_file_not_too_big,
@@ -131,7 +96,7 @@ class << self
13196
#
13297
# @see Alchemy::Picture::Url
13398
def url_class
134-
@_url_class ||= Alchemy::Picture::Url
99+
@_url_class ||= Alchemy.storage_adapter.url_class
135100
end
136101

137102
# Set a different picture url class
@@ -177,9 +142,7 @@ def last_upload
177142
private
178143

179144
def file_formats
180-
ActiveStorage::Blob.joins(:attachments).merge(
181-
ActiveStorage::Attachment.where(record_type: name)
182-
).distinct.pluck(:content_type)
145+
Alchemy.storage_adapter.picture_file_formats
183146
end
184147
end
185148

@@ -196,7 +159,7 @@ def url(options = {})
196159
return unless image_file
197160

198161
self.class.url_class.new(self).call(options)
199-
rescue ::ActiveStorage::Error => e
162+
rescue Alchemy.storage_adapter.rescuable_errors => e
200163
log_warning(e.message)
201164
nil
202165
end
@@ -280,7 +243,7 @@ def convertible?
280243
# Returns true if the image can be converted into other formats
281244
#
282245
def has_convertible_format?
283-
image_file&.variable?
246+
Alchemy.storage_adapter.has_convertible_format?(self)
284247
end
285248

286249
# Checks if the picture is restricted.
@@ -302,27 +265,27 @@ def deletable?
302265
end
303266

304267
def image_file_name
305-
image_file&.filename&.to_s
268+
Alchemy.storage_adapter.image_file_name(self)
306269
end
307270

308271
def image_file_format
309-
image_file&.content_type
272+
Alchemy.storage_adapter.image_file_format(self)
310273
end
311274

312275
def image_file_size
313-
image_file&.byte_size
276+
Alchemy.storage_adapter.image_file_size(self)
314277
end
315278

316279
def image_file_width
317-
image_file&.metadata&.fetch(:width, nil)
280+
Alchemy.storage_adapter.image_file_width(self)
318281
end
319282

320283
def image_file_height
321-
image_file&.metadata&.fetch(:height, nil)
284+
Alchemy.storage_adapter.image_file_height(self)
322285
end
323286

324287
def image_file_extension
325-
image_file&.filename&.extension&.downcase
288+
Alchemy.storage_adapter.image_file_extension(self)
326289
end
327290

328291
alias_method :suffix, :image_file_extension
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,42 @@
1+
# frozen_string_literal: true
2+
3+
module Alchemy
4+
class Picture < BaseRecord
5+
class ActiveStoragePreprocessor
6+
def initialize(attachable)
7+
@attachable = attachable
8+
end
9+
10+
# Preprocess images after upload
11+
#
12+
# Define preprocessing options in the Alchemy::Config
13+
#
14+
# preprocess_image_resize [String] - Downsizing example: '1000x1000>'
15+
#
16+
def call
17+
max_image_size = Alchemy::Config.get(:preprocess_image_resize)
18+
if max_image_size.present?
19+
self.class.process_thumb(attachable, size: max_image_size)
20+
end
21+
end
22+
23+
attr_reader :attachable
24+
25+
class << self
26+
def generate_thumbs!(attachable)
27+
Alchemy::Picture::THUMBNAIL_SIZES.values.each do |size|
28+
process_thumb(attachable, size: size, flatten: true)
29+
end
30+
end
31+
32+
private
33+
34+
def process_thumb(attachable, options = {})
35+
attachable.variant :thumb,
36+
**Alchemy::DragonflyToImageProcessing.call(options),
37+
preprocessed: true
38+
end
39+
end
40+
end
41+
end
42+
end
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,55 @@
1+
# frozen_string_literal: true
2+
3+
module Alchemy
4+
class Picture < BaseRecord
5+
# Returns the URL to a variant of a picture using ActiveStorage
6+
class ActiveStorageUrl
7+
attr_reader :picture, :image_file
8+
9+
# @param [Alchemy::Picture]
10+
#
11+
def initialize(picture)
12+
@picture = picture
13+
@image_file = picture.image_file
14+
end
15+
16+
# The URL to a variant of a picture
17+
#
18+
# @return [String]
19+
#
20+
def call(options = {})
21+
variant_options = DragonflyToImageProcessing.call(options)
22+
variant_options[:format] = options[:format] || default_output_format
23+
variant = image_file&.variant(variant_options)
24+
return unless variant
25+
26+
Rails.application.routes.url_helpers.rails_blob_path(
27+
variant,
28+
{
29+
filename: filename(options),
30+
format: variant_options[:format],
31+
only_path: true
32+
}
33+
)
34+
end
35+
36+
private
37+
38+
def filename(options = {})
39+
if picture.name.presence
40+
picture.name.to_param
41+
else
42+
picture.image_file_name
43+
end
44+
end
45+
46+
def default_output_format
47+
if Alchemy::Config.get(:image_output_format) == "original"
48+
picture.image_file_extension
49+
else
50+
Alchemy::Config.get(:image_output_format)
51+
end
52+
end
53+
end
54+
end
55+
end

app/models/alchemy/picture/preprocessor.rb

+7-21
Original file line numberDiff line numberDiff line change
@@ -3,8 +3,8 @@
33
module Alchemy
44
class Picture < BaseRecord
55
class Preprocessor
6-
def initialize(attachable)
7-
@attachable = attachable
6+
def initialize(image_file)
7+
@image_file = image_file
88
end
99

1010
# Preprocess images after upload
@@ -15,28 +15,14 @@ def initialize(attachable)
1515
#
1616
def call
1717
max_image_size = Alchemy::Config.get(:preprocess_image_resize)
18-
if max_image_size.present?
19-
self.class.process_thumb(attachable, size: max_image_size)
20-
end
18+
image_file.thumb!(max_image_size) if max_image_size.present?
19+
# Auto orient the image so EXIF orientation data is taken into account
20+
image_file.auto_orient!
2121
end
2222

23-
attr_reader :attachable
23+
private
2424

25-
class << self
26-
def generate_thumbs!(attachable)
27-
Alchemy::Picture::THUMBNAIL_SIZES.values.each do |size|
28-
process_thumb(attachable, size: size, flatten: true)
29-
end
30-
end
31-
32-
private
33-
34-
def process_thumb(attachable, options = {})
35-
attachable.variant :thumb,
36-
**Alchemy::DragonflyToImageProcessing.call(options),
37-
preprocessed: true
38-
end
39-
end
25+
attr_reader :image_file
4026
end
4127
end
4228
end

app/models/alchemy/picture/url.rb

+38-22
Original file line numberDiff line numberDiff line change
@@ -3,50 +3,66 @@
33
module Alchemy
44
class Picture < BaseRecord
55
class Url
6-
attr_reader :picture, :image_file
6+
TRANSFORMATION_OPTIONS = [
7+
:crop,
8+
:crop_from,
9+
:crop_size,
10+
:flatten,
11+
:format,
12+
:quality,
13+
:size,
14+
:upsample
15+
]
16+
17+
attr_reader :picture, :variant, :thumb
718

819
# @param [Alchemy::Picture]
920
#
1021
def initialize(picture)
1122
@picture = picture
12-
@image_file = picture.image_file
1323
end
1424

1525
# The URL to a variant of a picture
1626
#
1727
# @return [String]
1828
#
1929
def call(options = {})
20-
variant_options = DragonflyToImageProcessing.call(options)
21-
variant_options[:format] = options[:format] || default_output_format
22-
variant = image_file&.variant(variant_options)
23-
return unless variant
24-
25-
Rails.application.routes.url_helpers.rails_blob_path(
26-
variant,
27-
{
28-
filename: filename(options),
29-
format: variant_options[:format],
30-
only_path: true
31-
}
30+
@variant = PictureVariant.new(picture, options.slice(*TRANSFORMATION_OPTIONS))
31+
params = options.except(*TRANSFORMATION_OPTIONS).merge(
32+
basename: picture.name,
33+
ext: variant.render_format,
34+
name: picture.name
3235
)
36+
37+
return variant.image.url(params) unless processible_image?
38+
39+
"/#{uid}"
3340
end
3441

3542
private
3643

37-
def filename(options = {})
38-
if picture.name.presence
39-
picture.name.to_param
44+
def processible_image?
45+
variant.image.is_a?(::Dragonfly::Job)
46+
end
47+
48+
def uid
49+
signature = PictureThumb::Signature.call(variant)
50+
if find_thumb_by(signature)
51+
thumb.uid
4052
else
41-
picture.image_file_name
53+
uid = PictureThumb::Uid.call(signature, variant)
54+
ActiveRecord::Base.connected_to(role: ActiveRecord.writing_role) do
55+
PictureThumb::Create.call(variant, signature, uid)
56+
end
57+
uid
4258
end
4359
end
4460

45-
def default_output_format
46-
if Alchemy::Config.get(:image_output_format) == "original"
47-
picture.image_file_extension
61+
def find_thumb_by(signature)
62+
@thumb = if picture.thumbs.loaded?
63+
picture.thumbs.find { |t| t.signature == signature }
4864
else
49-
Alchemy::Config.get(:image_output_format)
65+
picture.thumbs.find_by(signature: signature)
5066
end
5167
end
5268
end

0 commit comments

Comments
 (0)