activestorage 6.0.3.1 → 6.1.0.rc2

Sign up to get free protection for your applications and to get access to all the features.

Potentially problematic release.


This version of activestorage might be problematic. Click here for more details.

Files changed (62) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +148 -145
  3. data/MIT-LICENSE +1 -1
  4. data/README.md +35 -3
  5. data/app/controllers/active_storage/base_controller.rb +11 -0
  6. data/app/controllers/active_storage/blobs/proxy_controller.rb +14 -0
  7. data/app/controllers/active_storage/{blobs_controller.rb → blobs/redirect_controller.rb} +2 -2
  8. data/app/controllers/active_storage/direct_uploads_controller.rb +1 -1
  9. data/app/controllers/active_storage/disk_controller.rb +8 -20
  10. data/app/controllers/active_storage/representations/proxy_controller.rb +19 -0
  11. data/app/controllers/active_storage/{representations_controller.rb → representations/redirect_controller.rb} +2 -2
  12. data/app/controllers/concerns/active_storage/file_server.rb +18 -0
  13. data/app/controllers/concerns/active_storage/set_blob.rb +1 -1
  14. data/app/controllers/concerns/active_storage/set_current.rb +2 -2
  15. data/app/controllers/concerns/active_storage/set_headers.rb +12 -0
  16. data/app/jobs/active_storage/mirror_job.rb +15 -0
  17. data/app/models/active_storage/attachment.rb +18 -10
  18. data/app/models/active_storage/blob.rb +114 -59
  19. data/app/models/active_storage/blob/analyzable.rb +6 -2
  20. data/app/models/active_storage/blob/identifiable.rb +7 -6
  21. data/app/models/active_storage/blob/representable.rb +34 -4
  22. data/app/models/active_storage/preview.rb +31 -10
  23. data/app/models/active_storage/record.rb +7 -0
  24. data/app/models/active_storage/variant.rb +28 -41
  25. data/app/models/active_storage/variant_record.rb +8 -0
  26. data/app/models/active_storage/variant_with_record.rb +54 -0
  27. data/app/models/active_storage/variation.rb +25 -20
  28. data/config/routes.rb +58 -8
  29. data/db/migrate/20170806125915_create_active_storage_tables.rb +14 -5
  30. data/db/update_migrate/20190112182829_add_service_name_to_active_storage_blobs.rb +17 -0
  31. data/db/update_migrate/20191206030411_create_active_storage_variant_records.rb +11 -0
  32. data/lib/active_storage.rb +5 -2
  33. data/lib/active_storage/analyzer.rb +6 -0
  34. data/lib/active_storage/analyzer/image_analyzer.rb +3 -0
  35. data/lib/active_storage/analyzer/null_analyzer.rb +4 -0
  36. data/lib/active_storage/analyzer/video_analyzer.rb +14 -3
  37. data/lib/active_storage/attached/changes/create_many.rb +1 -0
  38. data/lib/active_storage/attached/changes/create_one.rb +17 -4
  39. data/lib/active_storage/attached/many.rb +4 -3
  40. data/lib/active_storage/attached/model.rb +77 -16
  41. data/lib/active_storage/attached/one.rb +4 -3
  42. data/lib/active_storage/engine.rb +25 -27
  43. data/lib/active_storage/gem_version.rb +3 -3
  44. data/lib/active_storage/log_subscriber.rb +6 -0
  45. data/lib/active_storage/previewer.rb +3 -2
  46. data/lib/active_storage/previewer/mupdf_previewer.rb +3 -3
  47. data/lib/active_storage/previewer/poppler_pdf_previewer.rb +3 -3
  48. data/lib/active_storage/previewer/video_previewer.rb +2 -2
  49. data/lib/active_storage/service.rb +36 -7
  50. data/lib/active_storage/service/azure_storage_service.rb +40 -35
  51. data/lib/active_storage/service/configurator.rb +3 -1
  52. data/lib/active_storage/service/disk_service.rb +36 -31
  53. data/lib/active_storage/service/gcs_service.rb +18 -16
  54. data/lib/active_storage/service/mirror_service.rb +31 -7
  55. data/lib/active_storage/service/registry.rb +32 -0
  56. data/lib/active_storage/service/s3_service.rb +53 -23
  57. data/lib/active_storage/transformers/image_processing_transformer.rb +13 -7
  58. data/lib/active_storage/transformers/transformer.rb +0 -3
  59. metadata +55 -19
  60. data/db/update_migrate/20180723000244_add_foreign_key_constraint_to_active_storage_attachments_for_blob_id.rb +0 -9
  61. data/lib/active_storage/downloading.rb +0 -47
  62. data/lib/active_storage/transformers/mini_magick_transformer.rb +0 -38
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: fa09b18c03cccb24be6c16e35d4fd89a28debf4ce0adaf1b462c8b2eb84b0b3a
4
- data.tar.gz: 3d6669db0d4ceceb301acc1dec4fa8ce35854d6af0eef4bb88fe6e9f4172e5d2
3
+ metadata.gz: '0899559a3bc6de6d857ebb86597fab1354a02ae06697d643f689dcd763c688c1'
4
+ data.tar.gz: edbc9a4f7a25b9bd92f6bec1dca219b49df78c26fb774e0c4097dd59430f95f1
5
5
  SHA512:
6
- metadata.gz: 29404be9ea8d895ad20b89d5e23082852fac433b18495487b3b1a8590424b021d1033e2906be721394899e20deba6c8105329559cb4024ed8db2d377aa9e46dc
7
- data.tar.gz: 4cdca9240fc3f2e04a2267fe9cd0a2014a81a5b0f34f1d7bfc3ae87f57951fc11200f5aea3b1a929820f85198cfdd2b18811935a2d96eb11e30c303c222a4e29
6
+ metadata.gz: 257274e3e39b3af4dbab977180cc0357644171832a319d213fadefb24958b282a122001bb97890b2fffd8ec5124c03ca00e4fafd20f00bc33a2268595109db75
7
+ data.tar.gz: c4a122be03c4f7b513dc5d3dc5f229bbaef056af48e250133cfd250be2481ea22c864dded8f0668ef0da6a97e99c8061c837e21fafb0127840095d5bd7dfdd49
@@ -1,235 +1,238 @@
1
- ## Rails 6.0.3.1 (May 18, 2020) ##
1
+ ## Rails 6.1.0.rc2 (December 01, 2020) ##
2
2
 
3
- * [CVE-2020-8162] Include Content-Length in signature for ActiveStorage direct upload
3
+ * Implement `strict_loading` on ActiveStorage associations.
4
4
 
5
- ## Rails 6.0.3 (May 06, 2020) ##
5
+ *David Angulo*
6
6
 
7
- * No changes.
7
+ ## Rails 6.1.0.rc1 (November 02, 2020) ##
8
8
 
9
+ * Remove deprecated support to pass `:combine_options` operations to `ActiveStorage::Transformers::ImageProcessing`.
9
10
 
10
- ## Rails 6.0.2.2 (March 19, 2020) ##
11
+ *Rafael Mendonça França*
11
12
 
12
- * No changes.
13
+ * Remove deprecated `ActiveStorage::Transformers::MiniMagickTransformer`.
13
14
 
15
+ *Rafael Mendonça França*
14
16
 
15
- ## Rails 6.0.2.1 (December 18, 2019) ##
17
+ * Remove deprecated `config.active_storage.queue`.
16
18
 
17
- * No changes.
19
+ *Rafael Mendonça França*
18
20
 
21
+ * Remove deprecated `ActiveStorage::Downloading`.
19
22
 
20
- ## Rails 6.0.2 (December 13, 2019) ##
23
+ *Rafael Mendonça França*
21
24
 
22
- * No changes.
25
+ * Add per-environment configuration support
23
26
 
27
+ *Pietro Moro*
24
28
 
25
- ## Rails 6.0.1 (November 5, 2019) ##
29
+ * The Poppler PDF previewer renders a preview image using the original
30
+ document's crop box rather than its media box, hiding print margins. This
31
+ matches the behavior of the MuPDF previewer.
26
32
 
27
- * `ActiveStorage::AnalyzeJob`s are discarded on `ActiveRecord::RecordNotFound` errors.
33
+ *Vincent Robert*
28
34
 
29
- *George Claghorn*
30
-
31
- * Blobs are recorded in the database before being uploaded to the service.
32
- This fixes that generated blob keys could silently collide, leading to
33
- data loss.
34
-
35
- *Julik Tarkhanov*
35
+ * Touch parent model when an attachment is purged.
36
36
 
37
+ *Víctor Pérez Rodríguez*
37
38
 
38
- ## Rails 6.0.0 (August 16, 2019) ##
39
+ * Files can now be served by proxying them from the underlying storage service
40
+ instead of redirecting to a signed service URL. Use the
41
+ `rails_storage_proxy_path` and `_url` helpers to proxy an attached file:
39
42
 
40
- * No changes.
41
-
42
-
43
- ## Rails 6.0.0.rc2 (July 22, 2019) ##
44
-
45
- * No changes.
43
+ ```erb
44
+ <%= image_tag rails_storage_proxy_path(@user.avatar) %>
45
+ ```
46
46
 
47
+ To proxy by default, set `config.active_storage.resolve_model_to_route`:
47
48
 
48
- ## Rails 6.0.0.rc1 (April 24, 2019) ##
49
+ ```ruby
50
+ # Proxy attached files instead.
51
+ config.active_storage.resolve_model_to_route = :rails_storage_proxy
52
+ ```
49
53
 
50
- * Don't raise when analyzing an image whose type is unsupported by ImageMagick.
54
+ ```erb
55
+ <%= image_tag @user.avatar %>
56
+ ```
51
57
 
52
- Fixes #36065.
58
+ To redirect to a signed service URL when the default file serving strategy
59
+ is set to proxying, use the `rails_storage_redirect_path` and `_url` helpers:
53
60
 
54
- *Guilherme Mansur*
61
+ ```erb
62
+ <%= image_tag rails_storage_redirect_path(@user.avatar) %>
63
+ ```
55
64
 
56
- * Permit generating variants of BMP images.
65
+ *Jonathan Fleckenstein*
57
66
 
58
- *Younes Serraj*
67
+ * Add `config.active_storage.web_image_content_types` to allow applications
68
+ to add content types (like `image/webp`) in which variants can be processed,
69
+ instead of letting those images be converted to the fallback PNG format.
59
70
 
71
+ *Jeroen van Haperen*
60
72
 
61
- ## Rails 6.0.0.beta3 (March 11, 2019) ##
73
+ * Add support for creating variants of `WebP` images out of the box.
62
74
 
63
- * No changes.
75
+ *Dino Maric*
64
76
 
77
+ * Only enqueue analysis jobs for blobs with non-null analyzer classes.
65
78
 
66
- ## Rails 6.0.0.beta2 (February 25, 2019) ##
79
+ *Gannon McGibbon*
67
80
 
68
- * No changes.
81
+ * Previews are created on the same service as the original blob.
69
82
 
83
+ *Peter Zhu*
70
84
 
71
- ## Rails 6.0.0.beta1 (January 18, 2019) ##
85
+ * Remove unused `disposition` and `content_type` query parameters for `DiskService`.
72
86
 
73
- * [Rename npm package](https://github.com/rails/rails/pull/34905) from
74
- [`activestorage`](https://www.npmjs.com/package/activestorage) to
75
- [`@rails/activestorage`](https://www.npmjs.com/package/@rails/activestorage).
87
+ *Peter Zhu*
76
88
 
77
- *Javan Makhmali*
89
+ * Use `DiskController` for both public and private files.
78
90
 
79
- * Replace `config.active_storage.queue` with two options that indicate which
80
- queues analysis and purge jobs should use, respectively:
91
+ `DiskController` is able to handle multiple services by adding a
92
+ `service_name` field in the generated URL in `DiskService`.
81
93
 
82
- * `config.active_storage.queues.analysis`
83
- * `config.active_storage.queues.purge`
94
+ *Peter Zhu*
84
95
 
85
- `config.active_storage.queue` is preferred over the new options when it's
86
- set, but it is deprecated and will be removed in Rails 6.1.
96
+ * Variants are tracked in the database to avoid existence checks in the storage service.
87
97
 
88
98
  *George Claghorn*
89
99
 
90
- * Permit generating variants of TIFF images.
100
+ * Deprecate `service_url` methods in favour of `url`.
91
101
 
92
- *Luciano Sousa*
102
+ Deprecate `Variant#service_url` and `Preview#service_url` to instead use
103
+ `#url` method to be consistent with `Blob`.
93
104
 
94
- * Use base36 (all lowercase) for all new Blob keys to prevent
95
- collisions and undefined behavior with case-insensitive filesystems and
96
- database indices.
105
+ *Peter Zhu*
97
106
 
98
- *Julik Tarkhanov*
99
-
100
- * It doesn’t include an `X-CSRF-Token` header if a meta tag is not found on
101
- the page. It previously included one with a value of `undefined`.
107
+ * Permanent URLs for public storage blobs.
102
108
 
103
- *Cameron Bothner*
109
+ Services can be configured in `config/storage.yml` with a new key
110
+ `public: true | false` to indicate whether a service holds public
111
+ blobs or private blobs. Public services will always return a permanent URL.
104
112
 
105
- * Fix `ArgumentError` when uploading to amazon s3
113
+ Deprecates `Blob#service_url` in favor of `Blob#url`.
106
114
 
107
- *Hiroki Sanpei*
115
+ *Peter Zhu*
108
116
 
109
- * Add progressive JPG to default list of variable content types
117
+ * Make services aware of configuration names.
110
118
 
111
- *Maurice Kühlborn*
119
+ *Gannon McGibbon*
112
120
 
113
- * Add `ActiveStorage.routes_prefix` for configuring generated routes.
121
+ * The `Content-Type` header is set on image variants when they're uploaded to third-party storage services.
114
122
 
115
- *Chris Bisnett*
123
+ *Kyle Ribordy*
116
124
 
117
- * `ActiveStorage::Service::AzureStorageService` only handles specifically
118
- relevant types of `Azure::Core::Http::HTTPError`. It previously obscured
119
- other types of `HTTPError`, which is the azure-storage gem’s catch-all
120
- exception class.
125
+ * Allow storage services to be configured per attachment.
121
126
 
122
- *Cameron Bothner*
127
+ ```ruby
128
+ class User < ActiveRecord::Base
129
+ has_one_attached :avatar, service: :s3
130
+ end
123
131
 
124
- * `ActiveStorage::DiskController#show` generates a 404 Not Found response when
125
- the requested file is missing from the disk service. It previously raised
126
- `Errno::ENOENT`.
132
+ class Gallery < ActiveRecord::Base
133
+ has_many_attached :photos, service: :s3
134
+ end
135
+ ```
127
136
 
128
- *Cameron Bothner*
137
+ *Dmitry Tsepelev*
129
138
 
130
- * `ActiveStorage::Blob#download` and `ActiveStorage::Blob#open` raise
131
- `ActiveStorage::FileNotFoundError` when the corresponding file is missing
132
- from the storage service. Services translate service-specific missing object
133
- exceptions (e.g. `Google::Cloud::NotFoundError` for the GCS service and
134
- `Errno::ENOENT` for the disk service) into
135
- `ActiveStorage::FileNotFoundError`.
139
+ * You can optionally provide a custom blob key when attaching a new file:
136
140
 
137
- *Cameron Bothner*
141
+ ```ruby
142
+ user.avatar.attach key: "avatars/#{user.id}.jpg",
143
+ io: io, content_type: "image/jpeg", filename: "avatar.jpg"
144
+ ```
138
145
 
139
- * Added the `ActiveStorage::SetCurrent` concern for custom Active Storage
140
- controllers that can't inherit from `ActiveStorage::BaseController`.
146
+ Active Storage will store the blob's data on the configured service at the provided key.
141
147
 
142
148
  *George Claghorn*
143
149
 
144
- * Active Storage error classes like `ActiveStorage::IntegrityError` and
145
- `ActiveStorage::UnrepresentableError` now inherit from `ActiveStorage::Error`
146
- instead of `StandardError`. This permits rescuing `ActiveStorage::Error` to
147
- handle all Active Storage errors.
150
+ * Replace `Blob.create_after_upload!` with `Blob.create_and_upload!` and deprecate the former.
148
151
 
149
- *Andrei Makarov*, *George Claghorn*
152
+ `create_after_upload!` has been removed since it could lead to data
153
+ corruption by uploading to a key on the storage service which happened to
154
+ be already taken. Creating the record would then correctly raise a
155
+ database uniqueness exception but the stored object would already have
156
+ overwritten another. `create_and_upload!` swaps the order of operations
157
+ so that the key gets reserved up-front or the uniqueness error gets raised,
158
+ before the upload to a key takes place.
150
159
 
151
- * Uploaded files assigned to a record are persisted to storage when the record
152
- is saved instead of immediately.
160
+ *Julik Tarkhanov*
153
161
 
154
- In Rails 5.2, the following causes an uploaded file in `params[:avatar]` to
155
- be stored:
162
+ * Set content disposition in direct upload using `filename` and `disposition` parameters to `ActiveStorage::Service#headers_for_direct_upload`.
156
163
 
157
- ```ruby
158
- @user.avatar = params[:avatar]
159
- ```
164
+ *Peter Zhu*
160
165
 
161
- In Rails 6, the uploaded file is stored when `@user` is successfully saved.
166
+ * Allow record to be optionally passed to blob finders to make sharding
167
+ easier.
162
168
 
163
- *George Claghorn*
169
+ *Gannon McGibbon*
164
170
 
165
- * Add the ability to reflect on defined attachments using the existing
166
- ActiveRecord reflection mechanism.
171
+ * Switch from `azure-storage` gem to `azure-storage-blob` gem for Azure service.
167
172
 
168
- *Kevin Deisz*
173
+ *Peter Zhu*
169
174
 
170
- * Variant arguments of `false` or `nil` will no longer be passed to the
171
- processor. For example, the following will not have the monochrome
172
- variation applied:
175
+ * Add `config.active_storage.draw_routes` to disable Active Storage routes.
173
176
 
174
- ```ruby
175
- avatar.variant(monochrome: false)
176
- ```
177
+ *Gannon McGibbon*
177
178
 
178
- *Jacob Smith*
179
+ * Image analysis is skipped if ImageMagick returns an error.
179
180
 
180
- * Generated attachment getter and setter methods are created
181
- within the model's `GeneratedAssociationMethods` module to
182
- allow overriding and composition using `super`.
181
+ `ActiveStorage::Analyzer::ImageAnalyzer#metadata` would previously raise a
182
+ `MiniMagick::Error`, which caused persistent `ActiveStorage::AnalyzeJob`
183
+ failures. It now logs the error and returns `{}`, resulting in no metadata
184
+ being added to the offending image blob.
183
185
 
184
- *Josh Susser*, *Jamon Douglas*
185
-
186
- * Add `ActiveStorage::Blob#open`, which downloads a blob to a tempfile on disk
187
- and yields the tempfile. Deprecate `ActiveStorage::Downloading`.
186
+ *George Claghorn*
188
187
 
189
- *David Robertson*, *George Claghorn*
188
+ * Method calls on singular attachments return `nil` when no file is attached.
190
189
 
191
- * Pass in `identify: false` as an argument when providing a `content_type` for
192
- `ActiveStorage::Attached::{One,Many}#attach` to bypass automatic content
193
- type inference. For example:
190
+ Previously, assuming the following User model, `user.avatar.filename` would
191
+ raise a `Module::DelegationError` if no avatar was attached:
194
192
 
195
193
  ```ruby
196
- @message.image.attach(
197
- io: File.open('/path/to/file'),
198
- filename: 'file.pdf',
199
- content_type: 'application/pdf',
200
- identify: false
201
- )
194
+ class User < ApplicationRecord
195
+ has_one_attached :avatar
196
+ end
202
197
  ```
203
198
 
204
- *Ryan Davidson*
199
+ They now return `nil`.
205
200
 
206
- * The Google Cloud Storage service properly supports streaming downloads.
207
- It now requires version 1.11 or newer of the google-cloud-storage gem.
208
-
209
- *George Claghorn*
201
+ *Matthew Tanous*
210
202
 
211
- * Use the [ImageProcessing](https://github.com/janko-m/image_processing) gem
212
- for Active Storage variants, and deprecate the MiniMagick backend.
203
+ * The mirror service supports direct uploads.
213
204
 
214
- This means that variants are now automatically oriented if the original
215
- image was rotated. Also, in addition to the existing ImageMagick
216
- operations, variants can now use `:resize_to_fit`, `:resize_to_fill`, and
217
- other ImageProcessing macros. These are now recommended over raw `:resize`,
218
- as they also sharpen the thumbnail after resizing.
205
+ New files are directly uploaded to the primary service. When a
206
+ directly-uploaded file is attached to a record, a background job is enqueued
207
+ to copy it to each secondary service.
219
208
 
220
- The ImageProcessing gem also comes with a backend implemented on
221
- [libvips](http://jcupitt.github.io/libvips/), an alternative to
222
- ImageMagick which has significantly better performance than
223
- ImageMagick in most cases, both in terms of speed and memory usage. In
224
- Active Storage it's now possible to switch to the libvips backend by
225
- changing `Rails.application.config.active_storage.variant_processor` to
226
- `:vips`.
209
+ Configure the queue used to process mirroring jobs by setting
210
+ `config.active_storage.queues.mirror`. The default is `:active_storage_mirror`.
227
211
 
228
- *Janko Marohnić*
212
+ *George Claghorn*
229
213
 
230
- * Rails 6 requires Ruby 2.5.0 or newer.
214
+ * The S3 service now permits uploading files larger than 5 gigabytes.
215
+
216
+ When uploading a file greater than 100 megabytes in size, the service
217
+ transparently switches to [multipart uploads](https://docs.aws.amazon.com/AmazonS3/latest/dev/mpuoverview.html)
218
+ using a part size computed from the file's total size and S3's part count limit.
219
+
220
+ No application changes are necessary to take advantage of this feature. You
221
+ can customize the default 100 MB multipart upload threshold in your S3
222
+ service's configuration:
223
+
224
+ ```yaml
225
+ production:
226
+ service: s3
227
+ access_key_id: <%= Rails.application.credentials.dig(:aws, :access_key_id) %>
228
+ secret_access_key: <%= Rails.application.credentials.dig(:aws, :secret_access_key) %>
229
+ region: us-east-1
230
+ bucket: my-bucket
231
+ upload:
232
+ multipart_threshold: <%= 250.megabytes %>
233
+ ```
231
234
 
232
- *Jeremy Daer*, *Kasper Timm Hansen*
235
+ *George Claghorn*
233
236
 
234
237
 
235
- Please check [5-2-stable](https://github.com/rails/rails/blob/5-2-stable/activestorage/CHANGELOG.md) for previous changes.
238
+ Please check [6-0-stable](https://github.com/rails/rails/blob/6-0-stable/activestorage/CHANGELOG.md) for previous changes.
@@ -1,4 +1,4 @@
1
- Copyright (c) 2017-2019 David Heinemeier Hansson, Basecamp
1
+ Copyright (c) 2017-2020 David Heinemeier Hansson, Basecamp
2
2
 
3
3
  Permission is hereby granted, free of charge, to any person obtaining
4
4
  a copy of this software and associated documentation files (the
data/README.md CHANGED
@@ -16,7 +16,7 @@ A key difference to how Active Storage works compared to other attachment soluti
16
16
 
17
17
  ## Installation
18
18
 
19
- Run `rails active_storage:install` to copy over active_storage migrations.
19
+ Run `bin/rails active_storage:install` to copy over active_storage migrations.
20
20
 
21
21
  NOTE: If the task cannot be found, verify that `require "active_storage/engine"` is present in `config/application.rb`.
22
22
 
@@ -55,7 +55,7 @@ url_for(user.avatar)
55
55
 
56
56
  class AvatarsController < ApplicationController
57
57
  def update
58
- # params[:avatar] contains a ActionDispatch::Http::UploadedFile object
58
+ # params[:avatar] contains an ActionDispatch::Http::UploadedFile object
59
59
  Current.user.avatar.attach(params.require(:avatar))
60
60
  redirect_to Current.user
61
61
  end
@@ -106,6 +106,37 @@ Variation of image attachment:
106
106
  <%= image_tag user.avatar.variant(resize_to_limit: [100, 100]) %>
107
107
  ```
108
108
 
109
+ ## File serving strategies
110
+
111
+ Active Storage supports two ways to serve files: redirecting and proxying.
112
+
113
+ ### Redirecting
114
+
115
+ Active Storage generates stable application URLs for files which, when accessed, redirect to signed, short-lived service URLs. This relieves application servers of the burden of serving file data. It is the default file serving strategy.
116
+
117
+ When the application is configured to proxy files by default, use the `rails_storage_redirect_path` and `_url` route helpers to redirect instead:
118
+
119
+ ```erb
120
+ <%= image_tag rails_storage_redirect_path(@user.avatar) %>
121
+ ```
122
+
123
+ ### Proxying
124
+
125
+ Optionally, files can be proxied instead. This means that your application servers will download file data from the storage service in response to requests. This can be useful for serving files from a CDN.
126
+
127
+ Explicitly proxy attachments using the `rails_storage_proxy_path` and `_url` route helpers:
128
+
129
+ ```erb
130
+ <%= image_tag rails_storage_proxy_path(@user.avatar) %>
131
+ ```
132
+
133
+ Or configure Active Storage to use proxying by default:
134
+
135
+ ```ruby
136
+ # config/initializers/active_storage.rb
137
+ Rails.application.config.active_storage.resolve_model_to_route = :rails_storage_proxy
138
+ ```
139
+
109
140
  ## Direct uploads
110
141
 
111
142
  Active Storage, with its included JavaScript library, supports uploading directly from the client to the cloud.
@@ -120,7 +151,8 @@ Active Storage, with its included JavaScript library, supports uploading directl
120
151
  ```
121
152
  Using the npm package:
122
153
  ```js
123
- require("@rails/activestorage").start()
154
+ import * as ActiveStorage from "@rails/activestorage"
155
+ ActiveStorage.start()
124
156
  ```
125
157
  2. Annotate file inputs with the direct upload URL.
126
158