shrine 3.0.1 → 3.3.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/CHANGELOG.md +82 -0
- data/LICENSE.txt +1 -1
- data/README.md +15 -5
- data/doc/advantages.md +33 -16
- data/doc/attacher.md +2 -2
- data/doc/carrierwave.md +78 -34
- data/doc/changing_derivatives.md +39 -39
- data/doc/design.md +134 -85
- data/doc/direct_s3.md +1 -0
- data/doc/external/articles.md +57 -45
- data/doc/external/extensions.md +41 -35
- data/doc/external/misc.md +23 -8
- data/doc/getting_started.md +177 -112
- data/doc/metadata.md +79 -43
- data/doc/multiple_files.md +6 -4
- data/doc/paperclip.md +119 -42
- data/doc/plugins/activerecord.md +1 -1
- data/doc/plugins/add_metadata.md +112 -35
- data/doc/plugins/atomic_helpers.md +41 -3
- data/doc/plugins/backgrounding.md +12 -2
- data/doc/plugins/column.md +36 -7
- data/doc/plugins/data_uri.md +2 -2
- data/doc/plugins/default_url.md +6 -3
- data/doc/plugins/derivation_endpoint.md +26 -28
- data/doc/plugins/derivatives.md +238 -171
- data/doc/plugins/determine_mime_type.md +2 -2
- data/doc/plugins/download_endpoint.md +5 -5
- data/doc/plugins/dynamic_storage.md +1 -1
- data/doc/plugins/form_assign.md +5 -5
- data/doc/plugins/included.md +25 -5
- data/doc/plugins/infer_extension.md +11 -2
- data/doc/plugins/instrumentation.md +1 -1
- data/doc/plugins/metadata_attributes.md +22 -10
- data/doc/plugins/mirroring.md +1 -1
- data/doc/plugins/persistence.md +11 -1
- data/doc/plugins/refresh_metadata.md +5 -4
- data/doc/plugins/remote_url.md +8 -3
- data/doc/plugins/remove_invalid.md +9 -1
- data/doc/plugins/signature.md +11 -2
- data/doc/plugins/store_dimensions.md +12 -2
- data/doc/plugins/type_predicates.md +96 -0
- data/doc/plugins/upload_endpoint.md +7 -11
- data/doc/plugins/upload_options.md +1 -1
- data/doc/plugins/url_options.md +4 -4
- data/doc/plugins/validation.md +14 -4
- data/doc/plugins/validation_helpers.md +3 -3
- data/doc/plugins/versions.md +7 -7
- data/doc/processing.md +290 -127
- data/doc/refile.md +39 -18
- data/doc/release_notes/2.19.0.md +1 -1
- data/doc/release_notes/2.8.0.md +1 -1
- data/doc/release_notes/3.0.0.md +1 -1
- data/doc/release_notes/3.0.1.md +4 -0
- data/doc/release_notes/3.1.0.md +73 -0
- data/doc/release_notes/3.2.0.md +96 -0
- data/doc/release_notes/3.2.1.md +31 -0
- data/doc/release_notes/3.2.2.md +14 -0
- data/doc/release_notes/3.3.0.md +105 -0
- data/doc/securing_uploads.md +3 -3
- data/doc/storage/file_system.md +1 -1
- data/doc/storage/memory.md +19 -0
- data/doc/storage/s3.md +105 -82
- data/doc/testing.md +2 -2
- data/doc/upgrading_to_3.md +97 -49
- data/doc/validation.md +3 -2
- data/lib/shrine.rb +8 -8
- data/lib/shrine/attacher.rb +24 -14
- data/lib/shrine/attachment.rb +5 -5
- data/lib/shrine/plugins.rb +22 -0
- data/lib/shrine/plugins/activerecord.rb +1 -1
- data/lib/shrine/plugins/add_metadata.rb +18 -7
- data/lib/shrine/plugins/backgrounding.rb +2 -2
- data/lib/shrine/plugins/default_storage.rb +6 -6
- data/lib/shrine/plugins/default_url.rb +1 -1
- data/lib/shrine/plugins/derivation_endpoint.rb +12 -7
- data/lib/shrine/plugins/derivatives.rb +61 -29
- data/lib/shrine/plugins/determine_mime_type.rb +3 -3
- data/lib/shrine/plugins/entity.rb +6 -6
- data/lib/shrine/plugins/mirroring.rb +8 -8
- data/lib/shrine/plugins/model.rb +3 -3
- data/lib/shrine/plugins/presign_endpoint.rb +16 -4
- data/lib/shrine/plugins/pretty_location.rb +1 -1
- data/lib/shrine/plugins/processing.rb +1 -1
- data/lib/shrine/plugins/refresh_metadata.rb +2 -2
- data/lib/shrine/plugins/remote_url.rb +3 -3
- data/lib/shrine/plugins/remove_attachment.rb +5 -0
- data/lib/shrine/plugins/remove_invalid.rb +10 -5
- data/lib/shrine/plugins/sequel.rb +1 -1
- data/lib/shrine/plugins/signature.rb +7 -6
- data/lib/shrine/plugins/store_dimensions.rb +22 -11
- data/lib/shrine/plugins/type_predicates.rb +113 -0
- data/lib/shrine/plugins/upload_endpoint.rb +10 -5
- data/lib/shrine/plugins/upload_options.rb +2 -2
- data/lib/shrine/plugins/url_options.rb +2 -2
- data/lib/shrine/plugins/validation.rb +9 -7
- data/lib/shrine/storage/linter.rb +4 -4
- data/lib/shrine/storage/memory.rb +5 -3
- data/lib/shrine/storage/s3.rb +117 -38
- data/lib/shrine/uploaded_file.rb +0 -1
- data/lib/shrine/version.rb +2 -2
- data/shrine.gemspec +7 -8
- metadata +25 -31
data/doc/plugins/derivatives.md
CHANGED
@@ -2,7 +2,7 @@
|
|
2
2
|
title: Derivatives
|
3
3
|
---
|
4
4
|
|
5
|
-
The derivatives plugin allows storing processed files ("derivatives") alongside
|
5
|
+
The [`derivatives`][derivatives] plugin allows storing processed files ("derivatives") alongside
|
6
6
|
the main attached file. The processed file data will be saved together with the
|
7
7
|
main attachment data in the same record attribute.
|
8
8
|
|
@@ -10,12 +10,11 @@ main attachment data in the same record attribute.
|
|
10
10
|
Shrine.plugin :derivatives
|
11
11
|
```
|
12
12
|
|
13
|
-
##
|
13
|
+
## Quick start
|
14
14
|
|
15
|
-
|
16
|
-
|
17
|
-
|
18
|
-
to create derivatives.
|
15
|
+
You'll usually want to create derivatives from an attached file. The simplest
|
16
|
+
way to do this is to define a processor which returns the processed files, and
|
17
|
+
then trigger it when you want to create derivatives.
|
19
18
|
|
20
19
|
Here is an example of generating image thumbnails:
|
21
20
|
|
@@ -27,7 +26,7 @@ gem "image_processing", "~> 1.8"
|
|
27
26
|
require "image_processing/mini_magick"
|
28
27
|
|
29
28
|
class ImageUploader < Shrine
|
30
|
-
Attacher.
|
29
|
+
Attacher.derivatives do |original|
|
31
30
|
magick = ImageProcessing::MiniMagick.source(original)
|
32
31
|
|
33
32
|
{
|
@@ -39,56 +38,149 @@ class ImageUploader < Shrine
|
|
39
38
|
end
|
40
39
|
```
|
41
40
|
```rb
|
42
|
-
class Photo < Model(:image_data)
|
43
|
-
include ImageUploader::Attachment(:image)
|
44
|
-
end
|
45
|
-
```
|
46
|
-
```rb
|
47
41
|
photo = Photo.new(image: file)
|
48
|
-
photo.image_derivatives! #
|
42
|
+
photo.image_derivatives! # creates derivatives
|
49
43
|
photo.save
|
50
44
|
```
|
51
45
|
|
52
|
-
|
53
|
-
route make sure to create derivatives for new attachments:
|
46
|
+
You can then retrieve the URL of a processed derivative:
|
54
47
|
|
55
48
|
```rb
|
56
|
-
photo.
|
49
|
+
photo.image_url(:large) #=> "https://s3.amazonaws.com/path/to/large.jpg"
|
57
50
|
```
|
58
51
|
|
59
|
-
|
60
|
-
|
52
|
+
The derivatives data is stored in the `<attachment>_data` column alongside the
|
53
|
+
main file:
|
61
54
|
|
62
55
|
```rb
|
63
56
|
photo.image_data #=>
|
64
57
|
# {
|
65
|
-
# "id": "original.jpg",
|
58
|
+
# "id": "path/to/original.jpg",
|
66
59
|
# "store": "store",
|
67
60
|
# "metadata": { ... },
|
68
61
|
# "derivatives": {
|
69
|
-
# "small": { "id": "small.jpg", "storage": "store", "metadata": { ... } },
|
70
|
-
# "medium": { "id": "medium.jpg", "storage": "store", "metadata": { ... } },
|
71
|
-
# "large": { "id": "large.jpg", "storage": "store", "metadata": { ... } },
|
62
|
+
# "small": { "id": "path/to/small.jpg", "storage": "store", "metadata": { ... } },
|
63
|
+
# "medium": { "id": "path/to/medium.jpg", "storage": "store", "metadata": { ... } },
|
64
|
+
# "large": { "id": "path/to/large.jpg", "storage": "store", "metadata": { ... } },
|
72
65
|
# }
|
73
66
|
# }
|
74
67
|
```
|
75
68
|
|
76
|
-
|
69
|
+
And they can be retrieved as `Shrine::UploadedFile` objects:
|
77
70
|
|
78
71
|
```rb
|
79
|
-
photo.image(:large) #=> #<Shrine::UploadedFile>
|
72
|
+
photo.image(:large) #=> #<Shrine::UploadedFile id="path/to/large.jpg" storage=:store metadata={...}>
|
80
73
|
photo.image(:large).url #=> "https://s3.amazonaws.com/path/to/large.jpg"
|
81
|
-
photo.image(:large).size #=>
|
74
|
+
photo.image(:large).size #=> 5825949
|
82
75
|
photo.image(:large).mime_type #=> "image/jpeg"
|
83
76
|
```
|
84
77
|
|
85
|
-
|
86
|
-
|
87
|
-
|
78
|
+
## Retrieving derivatives
|
79
|
+
|
80
|
+
The list of stored derivatives can be retrieved with `#<name>_derivatives`:
|
81
|
+
|
82
|
+
```rb
|
83
|
+
photo.image_derivatives #=>
|
84
|
+
# {
|
85
|
+
# small: #<Shrine::UploadedFile ...>,
|
86
|
+
# medium: #<Shrine::UploadedFile ...>,
|
87
|
+
# large: #<Shrine::UploadedFile ...>,
|
88
|
+
# }
|
89
|
+
```
|
90
|
+
|
91
|
+
A specific derivative can be retrieved in any of the following ways:
|
92
|
+
|
93
|
+
```rb
|
94
|
+
photo.image_derivatives[:small] #=> #<Shrine::UploadedFile ...>
|
95
|
+
photo.image_derivatives(:small) #=> #<Shrine::UploadedFile ...>
|
96
|
+
photo.image(:small) #=> #<Shrine::UploadedFile ...>
|
97
|
+
```
|
98
|
+
|
99
|
+
Or with nested derivatives:
|
100
|
+
|
101
|
+
```rb
|
102
|
+
photo.image_derivatives #=> { thumbnail: { small: ..., medium: ..., large: ... } }
|
103
|
+
|
104
|
+
photo.image_derivatives.dig(:thumbnail, :small) #=> #<Shrine::UploadedFile ...>
|
105
|
+
photo.image_derivatives(:thumbnail, :small) #=> #<Shrine::UploadedFile ...>
|
106
|
+
photo.image(:thumbnails, :small) #=> #<Shrine::UploadedFile ...>
|
107
|
+
```
|
108
|
+
|
109
|
+
### Derivative URL
|
110
|
+
|
111
|
+
You can retrieve the URL of a derivative URL with `#<name>_url`:
|
112
|
+
|
113
|
+
```rb
|
114
|
+
photo.image_url(:small) #=> "https://example.com/small.jpg"
|
115
|
+
photo.image_url(:medium) #=> "https://example.com/medium.jpg"
|
116
|
+
photo.image_url(:large) #=> "https://example.com/large.jpg"
|
117
|
+
```
|
118
|
+
|
119
|
+
For nested derivatives you can pass multiple keys:
|
120
|
+
|
121
|
+
```rb
|
122
|
+
photo.image_derivatives #=> { thumbnail: { small: ..., medium: ..., large: ... } }
|
123
|
+
|
124
|
+
photo.image_url(:thumbnail, :medium) #=> "https://example.com/medium.jpg"
|
125
|
+
```
|
126
|
+
|
127
|
+
By default, `#<name>_url` method will return `nil` if derivative is not found.
|
128
|
+
You can use the [`default_url`][default_url] plugin to set up URL fallbacks:
|
129
|
+
|
130
|
+
```rb
|
131
|
+
Attacher.default_url do |derivative: nil, **|
|
132
|
+
"/fallbacks/#{derivative}.jpg" if derivative
|
133
|
+
end
|
134
|
+
```
|
135
|
+
```rb
|
136
|
+
photo.image_url(:medium) #=> "https://example.com/fallbacks.com/medium.jpg"
|
137
|
+
```
|
138
|
+
|
139
|
+
Any additional URL options passed to `#<name>_url` will be forwarded to the
|
140
|
+
storage:
|
141
|
+
|
142
|
+
```rb
|
143
|
+
photo.image_url(:small, response_content_disposition: "attachment")
|
144
|
+
```
|
145
|
+
|
146
|
+
You can also retrieve the derivative URL via `UploadedFile#url`:
|
147
|
+
|
148
|
+
```rb
|
149
|
+
photo.image_derivatives[:large].url
|
150
|
+
```
|
151
|
+
|
152
|
+
## Attacher API
|
153
|
+
|
154
|
+
The derivatives API is primarily defined on the `Shrine::Attacher` class, with
|
155
|
+
some important methods also being exposed through the `Shrine::Attachment`
|
156
|
+
module.
|
157
|
+
|
158
|
+
Here is a model example with equivalent attacher code:
|
159
|
+
|
160
|
+
```rb
|
161
|
+
photo.image_derivatives!(:thumbnails)
|
162
|
+
photo.image_derivatives #=> { ... }
|
163
|
+
|
164
|
+
photo.image_url(:large) #=> "https://..."
|
165
|
+
photo.image(:large) #=> #<Shrine::UploadedFile ...>
|
166
|
+
```
|
167
|
+
```rb
|
168
|
+
attacher.create_derivatives(:thumbnails)
|
169
|
+
attacher.get_derivatives #=> { ... }
|
170
|
+
|
171
|
+
attacher.url(:large) #=> "https://..."
|
172
|
+
attacher.get(:large) #=> "#<Shrine::UploadedFile>"
|
173
|
+
```
|
174
|
+
|
175
|
+
## Creating derivatives
|
176
|
+
|
177
|
+
By default, the `Attacher#create_derivatives` method downloads the attached
|
178
|
+
file, calls the processor, uploads results to attacher's permanent storage, and
|
179
|
+
saves uploaded files on the attacher.
|
88
180
|
|
89
181
|
```rb
|
90
182
|
attacher.file #=> #<Shrine::UploadedFile id="original.jpg" storage=:store ...>
|
91
|
-
attacher.create_derivatives # calls
|
183
|
+
attacher.create_derivatives # calls default processor and uploads results
|
92
184
|
attacher.derivatives #=>
|
93
185
|
# {
|
94
186
|
# small: #<Shrine::UploadedFile id="small.jpg" storage=:store ...>,
|
@@ -97,9 +189,7 @@ attacher.derivatives #=>
|
|
97
189
|
# }
|
98
190
|
```
|
99
191
|
|
100
|
-
|
101
|
-
file, calls the processor, uploads results to attacher's permanent storage, and
|
102
|
-
saves uploaded files on the attacher. Any additional arguments are forwarded to
|
192
|
+
Any additional arguments are forwarded to
|
103
193
|
[`Attacher#process_derivatives`](#processing-derivatives):
|
104
194
|
|
105
195
|
```rb
|
@@ -107,35 +197,59 @@ attacher.create_derivatives(different_source) # pass a different source file
|
|
107
197
|
attacher.create_derivatives(foo: "bar") # pass custom options to the processor
|
108
198
|
```
|
109
199
|
|
200
|
+
### Create on promote
|
201
|
+
|
202
|
+
You can also have derivatives created automatically on promotion:
|
203
|
+
|
204
|
+
```rb
|
205
|
+
Shrine.plugin :derivatives, create_on_promote: true
|
206
|
+
```
|
207
|
+
```rb
|
208
|
+
attacher.assign(file)
|
209
|
+
attacher.finalize # creates derivatives on promotion
|
210
|
+
attacher.derivatives #=> { small: ..., medium: ..., large: ... }
|
211
|
+
```
|
212
|
+
|
110
213
|
### Naming processors
|
111
214
|
|
112
215
|
If you want to have multiple processors for an uploader, you can assign each
|
113
|
-
processor a name
|
114
|
-
the desired processor.
|
216
|
+
processor a name:
|
115
217
|
|
116
218
|
```rb
|
117
219
|
class ImageUploader < Shrine
|
118
|
-
Attacher.
|
119
|
-
|
220
|
+
Attacher.derivatives :thumbnails do |original|
|
221
|
+
{ large: ..., medium: ..., small: ... }
|
120
222
|
end
|
121
223
|
|
122
|
-
Attacher.
|
123
|
-
|
224
|
+
Attacher.derivatives :crop do |original|
|
225
|
+
{ cropped: ... }
|
124
226
|
end
|
125
|
-
|
126
|
-
# ...
|
127
227
|
end
|
128
228
|
```
|
229
|
+
|
230
|
+
Then when creating derivatives you can specify the name of the desired
|
231
|
+
processor. New derivatives will be merged with any existing ones.
|
232
|
+
|
129
233
|
```rb
|
130
|
-
photo.image_derivatives!(:thumbnails)
|
131
|
-
# or
|
132
234
|
attacher.create_derivatives(:thumbnails)
|
235
|
+
attacher.derivatives #=> { large: ..., medium: ..., small: ... }
|
236
|
+
|
237
|
+
attacher.create_derivatives(:crop)
|
238
|
+
attacher.derivatives #=> { large: ..., medium: ..., small: ..., cropped: ... }
|
133
239
|
```
|
134
240
|
|
135
241
|
### Derivatives storage
|
136
242
|
|
137
243
|
By default, derivatives are uploaded to the permanent storage of the attacher.
|
138
|
-
You can change the
|
244
|
+
You can change the destination storage by passing `:storage` to the creation
|
245
|
+
call:
|
246
|
+
|
247
|
+
```rb
|
248
|
+
attacher.create_derivatives(storage: :cache) # will be promoted together with main file
|
249
|
+
attacher.create_derivatives(storage: :other_store)
|
250
|
+
```
|
251
|
+
|
252
|
+
You can also change the default destination storage with the `:storage` plugin
|
139
253
|
option:
|
140
254
|
|
141
255
|
```rb
|
@@ -172,11 +286,14 @@ The storage block is evaluated in the context of a `Shrine::Attacher` instance:
|
|
172
286
|
|
173
287
|
```rb
|
174
288
|
Attacher.derivatives_storage do |derivative|
|
175
|
-
self
|
289
|
+
self #=> #<Shrine::Attacher>
|
176
290
|
|
291
|
+
file #=> #<Shrine::UploadedFile>
|
177
292
|
record #=> #<Photo>
|
178
293
|
name #=> :image
|
179
294
|
context #=> { ... }
|
295
|
+
|
296
|
+
# ...
|
180
297
|
end
|
181
298
|
```
|
182
299
|
|
@@ -186,7 +303,7 @@ Derivatives can be nested to any level, using both hashes and arrays, but the
|
|
186
303
|
top-level object must be a hash.
|
187
304
|
|
188
305
|
```rb
|
189
|
-
Attacher.
|
306
|
+
Attacher.derivatives :tiff do |original|
|
190
307
|
{
|
191
308
|
thumbnail: {
|
192
309
|
small: small,
|
@@ -201,114 +318,29 @@ Attacher.derivatives_processor :tiff do |original|
|
|
201
318
|
}
|
202
319
|
end
|
203
320
|
```
|
204
|
-
|
205
|
-
## Retrieving derivatives
|
206
|
-
|
207
|
-
If you're using the `Shrine::Attachment` module, you can retrieve stored
|
208
|
-
derivatives by calling `#<name>_derivatives` on your model/entity.
|
209
|
-
|
210
|
-
```rb
|
211
|
-
class Photo < Model(:image_data)
|
212
|
-
include ImageUploader::Attachment(:image)
|
213
|
-
end
|
214
|
-
```
|
215
|
-
```rb
|
216
|
-
photo.image_derivatives #=>
|
217
|
-
# {
|
218
|
-
# small: #<Shrine::UploadedFile>,
|
219
|
-
# medium: #<Shrine::UploadedFile>,
|
220
|
-
# large: #<Shrine::UploadedFile>,
|
221
|
-
# }
|
222
|
-
```
|
223
|
-
|
224
|
-
A specific derivative can be retrieved in any of the following ways:
|
225
|
-
|
226
|
-
```rb
|
227
|
-
photo.image_derivatives[:small] #=> #<Shrine::UploadedFile>
|
228
|
-
photo.image_derivatives(:small) #=> #<Shrine::UploadedFile>
|
229
|
-
photo.image(:small) #=> #<Shrine::UploadedFile>
|
230
|
-
```
|
231
|
-
|
232
|
-
And with nested derivatives:
|
233
|
-
|
234
|
-
```rb
|
235
|
-
photo.image_derivatives #=> { thumbnail: { small: ..., medium: ..., large: ... } }
|
236
|
-
|
237
|
-
photo.image_derivatives.dig(:thumbnail, :small) #=> #<Shrine::UploadedFile>
|
238
|
-
photo.image_derivatives(:thumbnail, :small) #=> #<Shrine::UploadedFile>
|
239
|
-
photo.image(:thumbnails, :small) #=> #<Shrine::UploadedFile>
|
240
|
-
```
|
241
|
-
|
242
|
-
When using `Shrine::Attacher` directly, you can retrieve derivatives using
|
243
|
-
`Attacher#derivatives`:
|
244
|
-
|
245
321
|
```rb
|
246
322
|
attacher.derivatives #=>
|
247
323
|
# {
|
248
|
-
#
|
249
|
-
#
|
250
|
-
#
|
324
|
+
# thumbnail: {
|
325
|
+
# small: #<Shrine::UploadedFile ...>,
|
326
|
+
# medium: #<Shrine::UploadedFile ...>,
|
327
|
+
# large: #<Shrine::UploadedFile ...>,
|
328
|
+
# },
|
329
|
+
# layers: [
|
330
|
+
# #<Shrine::UploadedFile ...>,
|
331
|
+
# #<Shrine::UploadedFile ...>,
|
332
|
+
# # ...
|
333
|
+
# ]
|
251
334
|
# }
|
252
335
|
```
|
253
336
|
|
254
|
-
## Derivative URL
|
255
|
-
|
256
|
-
If you're using the `Shrine::Attachment` module, you can use the `#<name>_url`
|
257
|
-
method to retrieve the URL of a derivative.
|
258
|
-
|
259
|
-
```rb
|
260
|
-
class Photo < Model(:image_data)
|
261
|
-
include ImageUploader::Attachment(:image)
|
262
|
-
end
|
263
|
-
```
|
264
|
-
```rb
|
265
|
-
photo.image_url(:small) #=> "https://example.com/small.jpg"
|
266
|
-
photo.image_url(:medium) #=> "https://example.com/medium.jpg"
|
267
|
-
photo.image_url(:large) #=> "https://example.com/large.jpg"
|
268
|
-
```
|
269
|
-
|
270
|
-
For nested derivatives you can pass multiple keys:
|
271
|
-
|
272
|
-
```rb
|
273
|
-
photo.image_derivatives #=> { thumbnail: { small: ..., medium: ..., large: ... } }
|
274
|
-
|
275
|
-
photo.image_url(:thumbnail, :medium) #=> "https://example.com/medium.jpg"
|
276
|
-
```
|
277
|
-
|
278
|
-
By default, `#<name>_url` method will return `nil` if derivative is not found.
|
279
|
-
You can use the [`default_url`][default_url] plugin to set up URL fallbacks:
|
280
|
-
|
281
|
-
```rb
|
282
|
-
Attacher.default_url do |derivative: nil, **|
|
283
|
-
"https://my-app.com/fallbacks/#{derivative}.jpg" if derivative
|
284
|
-
end
|
285
|
-
```
|
286
|
-
```rb
|
287
|
-
photo.image_url(:medium) #=> "https://example.com/fallbacks.com/medium.jpg"
|
288
|
-
```
|
289
|
-
|
290
|
-
Any additional URL options passed to `#<name>_url` will be forwarded to the
|
291
|
-
storage:
|
292
|
-
|
293
|
-
```rb
|
294
|
-
photo.image_url(:small, response_content_disposition: "attachment")
|
295
|
-
```
|
296
|
-
|
297
|
-
You can also retrieve the derivative URL via `UploadedFile#url`:
|
298
|
-
|
299
|
-
```rb
|
300
|
-
photo.image_derivatives[:large].url
|
301
|
-
# or
|
302
|
-
attacher.derivatives[:large].url
|
303
|
-
```
|
304
|
-
|
305
337
|
## Processing derivatives
|
306
338
|
|
307
339
|
A derivatives processor block takes the original file, and is expected to
|
308
340
|
return a hash of processed files (it can be [nested](#nesting-derivatives)).
|
309
341
|
|
310
342
|
```rb
|
311
|
-
Attacher.
|
343
|
+
Attacher.derivatives :my_processor do |original|
|
312
344
|
# return a hash of processed files
|
313
345
|
end
|
314
346
|
```
|
@@ -327,9 +359,10 @@ The processor block is evaluated in context of the `Shrine::Attacher` instance,
|
|
327
359
|
which allows you to change your processing logic based on the record data.
|
328
360
|
|
329
361
|
```rb
|
330
|
-
Attacher.
|
362
|
+
Attacher.derivatives :my_processor do |original|
|
331
363
|
self #=> #<Shrine::Attacher>
|
332
364
|
|
365
|
+
file #=> #<Shrine::UploadedFile>
|
333
366
|
record #=> #<Photo>
|
334
367
|
name #=> :image
|
335
368
|
context #=> { ... }
|
@@ -345,7 +378,7 @@ forwarded to the processor:
|
|
345
378
|
attacher.process_derivatives(:my_processor, foo: "bar")
|
346
379
|
```
|
347
380
|
```rb
|
348
|
-
Attacher.
|
381
|
+
Attacher.derivatives :my_processor do |original, **options|
|
349
382
|
options #=> { :foo => "bar" }
|
350
383
|
# ...
|
351
384
|
end
|
@@ -357,7 +390,7 @@ By default, the `Attacher#process_derivatives` method will download the
|
|
357
390
|
attached file and pass it to the processor:
|
358
391
|
|
359
392
|
```rb
|
360
|
-
Attacher.
|
393
|
+
Attacher.derivatives :my_processor do |original|
|
361
394
|
original #=> #<File:...>
|
362
395
|
# ...
|
363
396
|
end
|
@@ -366,29 +399,54 @@ end
|
|
366
399
|
attacher.process_derivatives(:my_processor) # downloads attached file and passes it to the processor
|
367
400
|
```
|
368
401
|
|
369
|
-
If you want to use a different source file,
|
370
|
-
|
371
|
-
|
402
|
+
If you want to use a different source file, you can pass it in to the process
|
403
|
+
call. Typically you'd pass a local file on disk. If you pass a
|
404
|
+
`Shrine::UploadedFile` object or another IO-like object, it will be
|
405
|
+
automatically downloaded/copied to a local TempFile on disk.
|
406
|
+
|
407
|
+
```rb
|
408
|
+
# named processor:
|
409
|
+
attacher.process_derivatives(:my_processor, source_file)
|
410
|
+
|
411
|
+
# default processor:
|
412
|
+
attacher.process_derivatives(source_file)
|
413
|
+
```
|
414
|
+
|
415
|
+
If you want to call multiple processors in a row with the same source file, you
|
416
|
+
can use this to avoid re-downloading the same source file each time:
|
372
417
|
|
373
418
|
```rb
|
374
|
-
# this way the source file is downloaded only once
|
375
419
|
attacher.file.download do |original|
|
376
420
|
attacher.process_derivatives(:thumbnails, original)
|
377
421
|
attacher.process_derivatives(:colors, original)
|
378
422
|
end
|
379
423
|
```
|
380
424
|
|
425
|
+
If a processor might not always need a local source file, you avoid a
|
426
|
+
potentially expensive download/copy by registering the processor with
|
427
|
+
`download: false`, in which case the source file will be passed to the
|
428
|
+
processor as is.
|
429
|
+
|
430
|
+
```rb
|
431
|
+
Attacher.derivatives :my_processor, download: false do |source|
|
432
|
+
source #=> Could be File, Shrine::UploadedFile, or other IO-like object
|
433
|
+
shrine_class.with_file(source) do |file|
|
434
|
+
# can force download/copy if necessary with `with_file`,
|
435
|
+
end
|
436
|
+
end
|
437
|
+
```
|
438
|
+
|
381
439
|
## Adding derivatives
|
382
440
|
|
383
441
|
If you already have processed files that you want to save, you can do that with
|
384
442
|
`Attacher#add_derivatives`:
|
385
443
|
|
386
444
|
```rb
|
387
|
-
attacher.add_derivatives(
|
445
|
+
attacher.add_derivatives({
|
388
446
|
one: file_1,
|
389
447
|
two: file_2,
|
390
448
|
# ...
|
391
|
-
)
|
449
|
+
})
|
392
450
|
|
393
451
|
attacher.derivatives #=>
|
394
452
|
# {
|
@@ -402,7 +460,7 @@ New derivatives will be merged with existing ones:
|
|
402
460
|
|
403
461
|
```rb
|
404
462
|
attacher.derivatives #=> { one: #<Shrine::UploadedFile> }
|
405
|
-
attacher.add_derivatives(two: two_file)
|
463
|
+
attacher.add_derivatives({ two: two_file })
|
406
464
|
attacher.derivatives #=> { one: #<Shrine::UploadedFile>, two: #<Shrine::UploadedFile> }
|
407
465
|
```
|
408
466
|
|
@@ -410,7 +468,7 @@ The merging is deep, so the following will work as well:
|
|
410
468
|
|
411
469
|
```rb
|
412
470
|
attacher.derivatives #=> { nested: { one: #<Shrine::UploadedFile> } }
|
413
|
-
attacher.add_derivatives(nested: { two: two_file })
|
471
|
+
attacher.add_derivatives({ nested: { two: two_file } })
|
414
472
|
attacher.derivatives #=> { nested: { one: #<Shrine::UploadedFile>, two: #<Shrine::UploadedFile> } }
|
415
473
|
```
|
416
474
|
|
@@ -421,6 +479,11 @@ For adding a single derivative, you can also use the singular
|
|
421
479
|
attacher.add_derivative(:thumb, thumbnail_file)
|
422
480
|
```
|
423
481
|
|
482
|
+
> Note that new derivatives will replace any existing derivatives living under
|
483
|
+
the same key, but won't delete them. If this is your case, make sure to save a
|
484
|
+
reference to the old derivatives before assigning new ones, and then delete
|
485
|
+
them after persisting the change.
|
486
|
+
|
424
487
|
Any options passed to `Attacher#add_derivative(s)` will be forwarded to
|
425
488
|
[`Attacher#upload_derivatives`](#uploading-derivatives).
|
426
489
|
|
@@ -437,11 +500,11 @@ If you want to upload processed files without setting them, you can use
|
|
437
500
|
`Attacher#upload_derivatives`:
|
438
501
|
|
439
502
|
```rb
|
440
|
-
derivatives = attacher.upload_derivatives(
|
503
|
+
derivatives = attacher.upload_derivatives({
|
441
504
|
one: file_1,
|
442
505
|
two: file_2,
|
443
506
|
# ...
|
444
|
-
)
|
507
|
+
})
|
445
508
|
|
446
509
|
derivatives #=>
|
447
510
|
# {
|
@@ -479,9 +542,7 @@ attacher.upload_derivative :thumb, thumbnail_file,
|
|
479
542
|
location: "path/to/derivative"
|
480
543
|
```
|
481
544
|
|
482
|
-
|
483
|
-
name of the derivative, which you can use when extracting metadata, generating
|
484
|
-
location or generating upload options:
|
545
|
+
The `:derivative` name is automatically passed to the uploader:
|
485
546
|
|
486
547
|
```rb
|
487
548
|
class MyUploader < Shrine
|
@@ -510,8 +571,6 @@ If you want to disable this behaviour, pass `delete: false`:
|
|
510
571
|
|
511
572
|
```rb
|
512
573
|
attacher.upload_derivative(:thumb, thumbnail_file, delete: false)
|
513
|
-
|
514
|
-
File.exist?(thumbnail_file.path) #=> true
|
515
574
|
```
|
516
575
|
|
517
576
|
## Merging derivatives
|
@@ -521,7 +580,7 @@ If you want to save already uploaded derivatives, you can use
|
|
521
580
|
|
522
581
|
```rb
|
523
582
|
attacher.derivatives #=> { one: #<Shrine::UploadedFile> }
|
524
|
-
attacher.merge_derivatives attacher.upload_derivatives(two: two_file)
|
583
|
+
attacher.merge_derivatives attacher.upload_derivatives({ two: two_file })
|
525
584
|
attacher.derivatives #=> { one: #<Shrine::UploadedFile>, two: #<Shrine::UploadedFile> }
|
526
585
|
```
|
527
586
|
|
@@ -529,10 +588,15 @@ This does a deep merge, so the following will work as well:
|
|
529
588
|
|
530
589
|
```rb
|
531
590
|
attacher.derivatives #=> { nested: { one: #<Shrine::UploadedFile> } }
|
532
|
-
attacher.merge_derivatives attacher.upload_derivatives(nested: { two: two_file })
|
591
|
+
attacher.merge_derivatives attacher.upload_derivatives({ nested: { two: two_file } })
|
533
592
|
attacher.derivatives #=> { nested: { one: #<Shrine::UploadedFile>, two: #<Shrine::UploadedFile> } }
|
534
593
|
```
|
535
594
|
|
595
|
+
> Note that new derivatives will replace any existing derivatives living under
|
596
|
+
the same key, but won't delete them. If this is your case, make sure to save a
|
597
|
+
reference to the old derivatives before assigning new ones, and then delete
|
598
|
+
them after persisting the change.
|
599
|
+
|
536
600
|
The `Attacher#merge_derivatives` method is thread-safe.
|
537
601
|
|
538
602
|
### Setting derivatives
|
@@ -542,7 +606,7 @@ If instead of adding you want to *override* existing derivatives, you can use
|
|
542
606
|
|
543
607
|
```rb
|
544
608
|
attacher.derivatives #=> { one: #<Shrine::UploadedFile> }
|
545
|
-
attacher.set_derivatives attacher.upload_derivatives(two: two_file)
|
609
|
+
attacher.set_derivatives attacher.upload_derivatives({ two: two_file })
|
546
610
|
attacher.derivatives #=> { two: #<Shrine::UploadedFile> }
|
547
611
|
```
|
548
612
|
|
@@ -650,7 +714,7 @@ You can store derivatives even if there is no main attached file:
|
|
650
714
|
|
651
715
|
```rb
|
652
716
|
attacher.file #=> nil
|
653
|
-
attacher.add_derivatives(one: one_file, two: two_file)
|
717
|
+
attacher.add_derivatives({ one: one_file, two: two_file })
|
654
718
|
attacher.data #=>
|
655
719
|
# {
|
656
720
|
# "derivatives" => {
|
@@ -728,15 +792,17 @@ plugin :derivatives
|
|
728
792
|
Processing derivatives will trigger a `derivatives.shrine` event with the
|
729
793
|
following payload:
|
730
794
|
|
731
|
-
| Key | Description
|
732
|
-
| :-- | :----
|
733
|
-
| `:processor` | Name of the derivatives processor
|
734
|
-
| `:processor_options` | Any options passed to the processor
|
735
|
-
| `:
|
795
|
+
| Key | Description |
|
796
|
+
| :-- | :---- |
|
797
|
+
| `:processor` | Name of the derivatives processor |
|
798
|
+
| `:processor_options` | Any options passed to the processor |
|
799
|
+
| `:io` | The source file passed to the processor |
|
800
|
+
| `:attacher` | The attacher instance doing the processing |
|
801
|
+
| `:uploader` | The uploader class that sent the event |
|
736
802
|
|
737
803
|
A default log subscriber is added as well which logs these events:
|
738
804
|
|
739
|
-
```
|
805
|
+
```
|
740
806
|
Derivatives (2133ms) – {:processor=>:thumbnails, :processor_options=>{}, :uploader=>ImageUploader}
|
741
807
|
```
|
742
808
|
|
@@ -747,8 +813,8 @@ plugin :derivatives, log_subscriber: -> (event) {
|
|
747
813
|
Shrine.logger.info JSON.generate(name: event.name, duration: event.duration, **event.payload)
|
748
814
|
}
|
749
815
|
```
|
750
|
-
```
|
751
|
-
{"name":"derivatives","duration":2133,"processor":"thumbnails","processor_options":{},"uploader":"ImageUploader"}
|
816
|
+
```
|
817
|
+
{"name":"derivatives","duration":2133,"processor":"thumbnails","processor_options":{},"io":"#<File:...>","uploader":"ImageUploader"}
|
752
818
|
```
|
753
819
|
|
754
820
|
Or disable logging altogether:
|
@@ -757,6 +823,7 @@ Or disable logging altogether:
|
|
757
823
|
plugin :derivatives, log_subscriber: nil
|
758
824
|
```
|
759
825
|
|
826
|
+
[derivatives]: https://github.com/shrinerb/shrine/blob/master/lib/shrine/plugins/derivatives.rb
|
760
827
|
[default_url]: https://shrinerb.com/docs/plugins/default_url
|
761
828
|
[entity]: https://shrinerb.com/docs/plugins/entity
|
762
829
|
[model]: https://shrinerb.com/docs/plugins/model
|