activestorage 5.2.4.4 → 6.0.0.beta1

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 (59) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +103 -81
  3. data/MIT-LICENSE +1 -1
  4. data/README.md +6 -5
  5. data/app/assets/javascripts/activestorage.js +4 -1
  6. data/app/controllers/active_storage/base_controller.rb +3 -5
  7. data/app/controllers/active_storage/blobs_controller.rb +1 -1
  8. data/app/controllers/active_storage/disk_controller.rb +4 -1
  9. data/app/controllers/active_storage/representations_controller.rb +1 -1
  10. data/app/controllers/concerns/active_storage/set_current.rb +15 -0
  11. data/app/javascript/activestorage/blob_record.js +6 -1
  12. data/app/jobs/active_storage/analyze_job.rb +4 -0
  13. data/app/jobs/active_storage/base_job.rb +0 -1
  14. data/app/jobs/active_storage/purge_job.rb +3 -0
  15. data/app/models/active_storage/attachment.rb +18 -9
  16. data/app/models/active_storage/blob.rb +63 -22
  17. data/app/models/active_storage/blob/representable.rb +5 -5
  18. data/app/models/active_storage/filename.rb +0 -6
  19. data/app/models/active_storage/preview.rb +3 -3
  20. data/app/models/active_storage/variant.rb +51 -52
  21. data/app/models/active_storage/variation.rb +23 -32
  22. data/config/routes.rb +13 -12
  23. data/db/update_migrate/20180723000244_add_foreign_key_constraint_to_active_storage_attachments_for_blob_id.rb +7 -0
  24. data/lib/active_storage.rb +13 -2
  25. data/lib/active_storage/analyzer.rb +9 -4
  26. data/lib/active_storage/analyzer/video_analyzer.rb +2 -4
  27. data/lib/active_storage/attached.rb +7 -22
  28. data/lib/active_storage/attached/changes.rb +16 -0
  29. data/lib/active_storage/attached/changes/create_many.rb +46 -0
  30. data/lib/active_storage/attached/changes/create_one.rb +68 -0
  31. data/lib/active_storage/attached/changes/create_one_of_many.rb +10 -0
  32. data/lib/active_storage/attached/changes/delete_many.rb +23 -0
  33. data/lib/active_storage/attached/changes/delete_one.rb +19 -0
  34. data/lib/active_storage/attached/many.rb +16 -10
  35. data/lib/active_storage/attached/model.rb +140 -0
  36. data/lib/active_storage/attached/one.rb +16 -19
  37. data/lib/active_storage/downloader.rb +44 -0
  38. data/lib/active_storage/downloading.rb +8 -0
  39. data/lib/active_storage/engine.rb +35 -6
  40. data/lib/active_storage/errors.rb +22 -3
  41. data/lib/active_storage/gem_version.rb +4 -4
  42. data/lib/active_storage/previewer.rb +21 -11
  43. data/lib/active_storage/previewer/poppler_pdf_previewer.rb +1 -1
  44. data/lib/active_storage/previewer/video_previewer.rb +2 -3
  45. data/lib/active_storage/reflection.rb +64 -0
  46. data/lib/active_storage/service.rb +5 -6
  47. data/lib/active_storage/service/azure_storage_service.rb +28 -12
  48. data/lib/active_storage/service/configurator.rb +3 -1
  49. data/lib/active_storage/service/disk_service.rb +20 -16
  50. data/lib/active_storage/service/gcs_service.rb +48 -46
  51. data/lib/active_storage/service/mirror_service.rb +1 -1
  52. data/lib/active_storage/service/s3_service.rb +10 -7
  53. data/lib/active_storage/transformers/image_processing_transformer.rb +39 -0
  54. data/lib/active_storage/transformers/mini_magick_transformer.rb +38 -0
  55. data/lib/active_storage/transformers/transformer.rb +42 -0
  56. data/lib/tasks/activestorage.rake +7 -0
  57. metadata +26 -14
  58. data/app/models/active_storage/filename/parameters.rb +0 -36
  59. data/lib/active_storage/attached/macros.rb +0 -110
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 0bc512e307b589b0c1a583a886620e3e47820ee9d5fc72dcd9098f15c39ccd5b
4
- data.tar.gz: 5d45410f437c16c89a5fe1c082c95f13b6f606c266531815ca351c88e4a7edf5
3
+ metadata.gz: c76cc0190a59a7e19312ae68c2b6c3712f62eca50d9fa4d287cdc6feca94a5aa
4
+ data.tar.gz: 6d8f47c3d81c83b98b0dfc254a6e9fe6ae9b447501db6176cb96691ceb77c61c
5
5
  SHA512:
6
- metadata.gz: ae16a55d7b9c4457bc2b839c1eda407c73d77f474a78ddddd1abaaa2a1b443f670bb2d14699335c270b34aa441908c998436dd41c648f70028d31e3f10d3e866
7
- data.tar.gz: 9759a7cb1f7c951fe481de3bceafa4afc24a567ce94c8c64b01bd535893a58d307807ecdc51db4ae3ca515cbc0318d4ca5c8cd8e66edb3d438abe23ce326174d
6
+ metadata.gz: a86e9a4fd481f1aa56946f24ff7e02869fe6591952929021e5c46428b230a34f18f324847db51eb0ebfd17ed8f43fb61f215798997766d072a6dbf51de21369d
7
+ data.tar.gz: 0231fa7771d92de8b9a9012291d69e2c5b976af3fa3f932c5fd0331f9cccd76ad90887d1101f96d47a6c7949832dabff1f1011412ab27a4a4ed76091ba851e10
@@ -1,137 +1,159 @@
1
- ## Rails 5.2.4.4 (September 09, 2020) ##
1
+ ## Rails 6.0.0.beta1 (January 18, 2019) ##
2
2
 
3
- * No changes.
3
+ * Replace `config.active_storage.queue` with two options that indicate which
4
+ queues analysis and purge jobs should use, respectively:
4
5
 
6
+ * `config.active_storage.queues.analysis`
7
+ * `config.active_storage.queues.purge`
5
8
 
6
- ## Rails 5.2.4.3 (May 18, 2020) ##
7
-
8
- * [CVE-2020-8162] Include Content-Length in signature for ActiveStorage direct upload
9
-
10
-
11
- ## Rails 5.2.4.1 (December 18, 2019) ##
12
-
13
- * No changes.
14
-
15
-
16
- ## Rails 5.2.4 (November 27, 2019) ##
17
-
18
- * No changes.
19
-
20
-
21
- ## Rails 5.2.3 (March 27, 2019) ##
22
-
23
- * No changes.
9
+ `config.active_storage.queue` is preferred over the new options when it's
10
+ set, but it is deprecated and will be removed in Rails 6.1.
24
11
 
12
+ *George Claghorn*
25
13
 
26
- ## Rails 5.2.2.1 (March 11, 2019) ##
14
+ * Permit generating variants of TIFF images.
27
15
 
28
- * No changes.
16
+ *Luciano Sousa*
29
17
 
18
+ * Use base36 (all lowercase) for all new Blob keys to prevent
19
+ collisions and undefined behavior with case-insensitive filesystems and
20
+ database indices.
30
21
 
31
- ## Rails 5.2.2 (December 04, 2018) ##
22
+ *Julik Tarkhanov*
32
23
 
33
- * Support multiple submit buttons in Active Storage forms.
24
+ * It doesn’t include an `X-CSRF-Token` header if a meta tag is not found on
25
+ the page. It previously included one with a value of `undefined`.
34
26
 
35
- *Chrıs Seelus*
27
+ *Cameron Bothner*
36
28
 
37
29
  * Fix `ArgumentError` when uploading to amazon s3
38
30
 
39
31
  *Hiroki Sanpei*
40
32
 
41
- * Add a foreign-key constraint to the `active_storage_attachments` table for blobs.
42
-
43
- *George Claghorn*
44
-
45
- * Discard `ActiveStorage::PurgeJobs` for missing blobs.
46
-
47
- *George Claghorn*
33
+ * Add progressive JPG to default list of variable content types
48
34
 
49
- * Fix uploading Tempfiles to Azure Storage.
35
+ *Maurice Kühlborn*
50
36
 
51
- *George Claghorn*
37
+ * Add `ActiveStorage.routes_prefix` for configuring generated routes.
52
38
 
39
+ *Chris Bisnett*
53
40
 
54
- ## Rails 5.2.1.1 (November 27, 2018) ##
41
+ * `ActiveStorage::Service::AzureStorageService` only handles specifically
42
+ relevant types of `Azure::Core::Http::HTTPError`. It previously obscured
43
+ other types of `HTTPError`, which is the azure-storage gem’s catch-all
44
+ exception class.
55
45
 
56
- * Prevent content type and disposition bypass in storage service URLs.
46
+ *Cameron Bothner*
57
47
 
58
- Fix CVE-2018-16477.
48
+ * `ActiveStorage::DiskController#show` generates a 404 Not Found response when
49
+ the requested file is missing from the disk service. It previously raised
50
+ `Errno::ENOENT`.
59
51
 
60
- *Rosa Gutierrez*
52
+ *Cameron Bothner*
61
53
 
54
+ * `ActiveStorage::Blob#download` and `ActiveStorage::Blob#open` raise
55
+ `ActiveStorage::FileNotFoundError` when the corresponding file is missing
56
+ from the storage service. Services translate service-specific missing object
57
+ exceptions (e.g. `Google::Cloud::NotFoundError` for the GCS service and
58
+ `Errno::ENOENT` for the disk service) into
59
+ `ActiveStorage::FileNotFoundError`.
62
60
 
63
- ## Rails 5.2.1 (August 07, 2018) ##
61
+ *Cameron Bothner*
64
62
 
65
- * Fix direct upload with zero-byte files.
63
+ * Added the `ActiveStorage::SetCurrent` concern for custom Active Storage
64
+ controllers that can't inherit from `ActiveStorage::BaseController`.
66
65
 
67
66
  *George Claghorn*
68
67
 
69
- * Exclude JSON root from `active_storage/direct_uploads#create` response.
68
+ * Active Storage error classes like `ActiveStorage::IntegrityError` and
69
+ `ActiveStorage::UnrepresentableError` now inherit from `ActiveStorage::Error`
70
+ instead of `StandardError`. This permits rescuing `ActiveStorage::Error` to
71
+ handle all Active Storage errors.
70
72
 
71
- *Javan Makhmali*
73
+ *Andrei Makarov*, *George Claghorn*
72
74
 
75
+ * Uploaded files assigned to a record are persisted to storage when the record
76
+ is saved instead of immediately.
73
77
 
74
- ## Rails 5.2.0 (April 09, 2018) ##
78
+ In Rails 5.2, the following causes an uploaded file in `params[:avatar]` to
79
+ be stored:
75
80
 
76
- * Allow full use of the AWS S3 SDK options for authentication. If an
77
- explicit AWS key pair and/or region is not provided in `storage.yml`,
78
- attempt to use environment variables, shared credentials, or IAM
79
- (instance or task) role credentials. Order of precedence is determined
80
- by the [AWS SDK](https://docs.aws.amazon.com/sdk-for-ruby/v3/developer-guide/setup-config.html).
81
+ ```ruby
82
+ @user.avatar = params[:avatar]
83
+ ```
81
84
 
82
- *Brian Knight*
85
+ In Rails 6, the uploaded file is stored when `@user` is successfully saved.
83
86
 
84
- * Remove path config option from Azure service.
87
+ *George Claghorn*
85
88
 
86
- The Active Storage service for Azure Storage has an option called `path`
87
- that is ambiguous in meaning. It needs to be set to the primary blob
88
- storage endpoint but that can be determined from the blobs client anyway.
89
+ * Add the ability to reflect on defined attachments using the existing
90
+ ActiveRecord reflection mechanism.
89
91
 
90
- To simplify the configuration, we've removed the `path` option and
91
- now get the endpoint from the blobs client instead.
92
+ *Kevin Deisz*
92
93
 
93
- Closes #32225.
94
+ * Variant arguments of `false` or `nil` will no longer be passed to the
95
+ processor. For example, the following will not have the monochrome
96
+ variation applied:
94
97
 
95
- *Andrew White*
98
+ ```ruby
99
+ avatar.variant(monochrome: false)
100
+ ```
96
101
 
97
- * Generate root-relative paths in disk service URL methods.
102
+ *Jacob Smith*
98
103
 
99
- Obviate the disk service's `:host` configuration option.
104
+ * Generated attachment getter and setter methods are created
105
+ within the model's `GeneratedAssociationMethods` module to
106
+ allow overriding and composition using `super`.
100
107
 
101
- *George Claghorn*
108
+ *Josh Susser*, *Jamon Douglas*
109
+
110
+ * Add `ActiveStorage::Blob#open`, which downloads a blob to a tempfile on disk
111
+ and yields the tempfile. Deprecate `ActiveStorage::Downloading`.
102
112
 
103
- * Add source code to published npm package.
113
+ *David Robertson*, *George Claghorn*
104
114
 
105
- This allows activestorage users to depend on the javascript source code
106
- rather than the compiled code, which can produce smaller javascript bundles.
115
+ * Pass in `identify: false` as an argument when providing a `content_type` for
116
+ `ActiveStorage::Attached::{One,Many}#attach` to bypass automatic content
117
+ type inference. For example:
107
118
 
108
- *Richard Macklin*
119
+ ```ruby
120
+ @message.image.attach(
121
+ io: File.open('/path/to/file'),
122
+ filename: 'file.pdf',
123
+ content_type: 'application/pdf',
124
+ identify: false
125
+ )
126
+ ```
109
127
 
110
- * Preserve display aspect ratio when extracting width and height from videos
111
- with rectangular samples in `ActiveStorage::Analyzer::VideoAnalyzer`.
128
+ *Ryan Davidson*
112
129
 
113
- When a video contains a display aspect ratio, emit it in metadata as
114
- `:display_aspect_ratio` rather than the ambiguous `:aspect_ratio`. Compute
115
- its height by scaling its encoded frame width according to the DAR.
130
+ * The Google Cloud Storage service properly supports streaming downloads.
131
+ It now requires version 1.11 or newer of the google-cloud-storage gem.
116
132
 
117
133
  *George Claghorn*
118
134
 
119
- * Use `after_destroy_commit` instead of `before_destroy` for purging
120
- attachments when a record is destroyed.
135
+ * Use the [ImageProcessing](https://github.com/janko-m/image_processing) gem
136
+ for Active Storage variants, and deprecate the MiniMagick backend.
121
137
 
122
- *Hiroki Zenigami*
138
+ This means that variants are now automatically oriented if the original
139
+ image was rotated. Also, in addition to the existing ImageMagick
140
+ operations, variants can now use `:resize_to_fit`, `:resize_to_fill`, and
141
+ other ImageProcessing macros. These are now recommended over raw `:resize`,
142
+ as they also sharpen the thumbnail after resizing.
123
143
 
124
- * Force `:attachment` disposition for specific, configurable content types.
125
- This mitigates possible security issues such as XSS or phishing when
126
- serving them inline. A list of such content types is included by default,
127
- and can be configured via `content_types_to_serve_as_binary`.
144
+ The ImageProcessing gem also comes with a backend implemented on
145
+ [libvips](http://jcupitt.github.io/libvips/), an alternative to
146
+ ImageMagick which has significantly better performance than
147
+ ImageMagick in most cases, both in terms of speed and memory usage. In
148
+ Active Storage it's now possible to switch to the libvips backend by
149
+ changing `Rails.application.config.active_storage.variant_processor` to
150
+ `:vips`.
128
151
 
129
- *Rosa Gutierrez*
152
+ *Janko Marohnić*
130
153
 
131
- * Fix the gem adding the migrations files to the package.
154
+ * Rails 6 requires Ruby 2.5.0 or newer.
132
155
 
133
- *Yuji Yaginuma*
156
+ *Jeremy Daer*, *Kasper Timm Hansen*
134
157
 
135
- * Added to Rails.
136
158
 
137
- *DHH*
159
+ Please check [5-2-stable](https://github.com/rails/rails/blob/5-2-stable/activestorage/CHANGELOG.md) for previous changes.
@@ -1,4 +1,4 @@
1
- Copyright (c) 2017-2018 David Heinemeier Hansson, Basecamp
1
+ Copyright (c) 2017-2019 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
@@ -4,11 +4,11 @@ Active Storage makes it simple to upload and reference files in cloud services l
4
4
 
5
5
  Files can be uploaded from the server to the cloud or directly from the client to the cloud.
6
6
 
7
- Image files can furthermore be transformed using on-demand variants for quality, aspect ratio, size, or any other [MiniMagick](https://github.com/minimagick/minimagick) supported transformation.
7
+ Image files can furthermore be transformed using on-demand variants for quality, aspect ratio, size, or any other [MiniMagick](https://github.com/minimagick/minimagick) or [Vips](http://www.rubydoc.info/gems/ruby-vips/Vips/Image) supported transformation.
8
8
 
9
9
  ## Compared to other storage solutions
10
10
 
11
- A key difference to how Active Storage works compared to other attachment solutions in Rails is through the use of built-in [Blob](https://github.com/rails/rails/blob/5-2-stable/activestorage/app/models/active_storage/blob.rb) and [Attachment](https://github.com/rails/rails/blob/5-2-stable/activestorage/app/models/active_storage/attachment.rb) models (backed by Active Record). This means existing application models do not need to be modified with additional columns to associate with files. Active Storage uses polymorphic associations via the `Attachment` join model, which then connects to the actual `Blob`.
11
+ A key difference to how Active Storage works compared to other attachment solutions in Rails is through the use of built-in [Blob](https://github.com/rails/rails/blob/master/activestorage/app/models/active_storage/blob.rb) and [Attachment](https://github.com/rails/rails/blob/master/activestorage/app/models/active_storage/attachment.rb) models (backed by Active Record). This means existing application models do not need to be modified with additional columns to associate with files. Active Storage uses polymorphic associations via the `Attachment` join model, which then connects to the actual `Blob`.
12
12
 
13
13
  `Blob` models store attachment metadata (filename, content-type, etc.), and their identifier key in the storage service. Blob models do not store the actual binary data. They are intended to be immutable in spirit. One file, one blob. You can associate the same blob with multiple application models as well. And if you want to do transformations of a given `Blob`, the idea is that you'll simply create a new one, rather than attempt to mutate the existing one (though of course you can delete the previous version later if you don't need it).
14
14
 
@@ -16,6 +16,8 @@ A key difference to how Active Storage works compared to other attachment soluti
16
16
 
17
17
  Run `rails active_storage:install` to copy over active_storage migrations.
18
18
 
19
+ NOTE: If the task cannot be found, verify that `require "active_storage/engine"` is present in `config/application.rb`.
20
+
19
21
  ## Examples
20
22
 
21
23
  One attachment:
@@ -99,7 +101,7 @@ Variation of image attachment:
99
101
 
100
102
  ```erb
101
103
  <%# Hitting the variant URL will lazy transform the original blob and then redirect to its new service location %>
102
- <%= image_tag user.avatar.variant(resize: "100x100") %>
104
+ <%= image_tag user.avatar.variant(resize_to_fit: [100, 100]) %>
103
105
  ```
104
106
 
105
107
  ## Direct uploads
@@ -116,8 +118,7 @@ Active Storage, with its included JavaScript library, supports uploading directl
116
118
  ```
117
119
  Using the npm package:
118
120
  ```js
119
- import * as ActiveStorage from "activestorage"
120
- ActiveStorage.start()
121
+ require("@rails/activestorage").start()
121
122
  ```
122
123
  2. Annotate file inputs with the direct upload URL.
123
124
 
@@ -560,7 +560,10 @@
560
560
  this.xhr.setRequestHeader("Content-Type", "application/json");
561
561
  this.xhr.setRequestHeader("Accept", "application/json");
562
562
  this.xhr.setRequestHeader("X-Requested-With", "XMLHttpRequest");
563
- this.xhr.setRequestHeader("X-CSRF-Token", getMetaValue("csrf-token"));
563
+ var csrfToken = getMetaValue("csrf-token");
564
+ if (csrfToken != undefined) {
565
+ this.xhr.setRequestHeader("X-CSRF-Token", csrfToken);
566
+ }
564
567
  this.xhr.addEventListener("load", function(event) {
565
568
  return _this.requestDidLoad(event);
566
569
  });
@@ -1,10 +1,8 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- # The base controller for all ActiveStorage controllers.
3
+ # The base class for all Active Storage controllers.
4
4
  class ActiveStorage::BaseController < ActionController::Base
5
- protect_from_forgery with: :exception
5
+ include ActiveStorage::SetCurrent
6
6
 
7
- before_action do
8
- ActiveStorage::Current.host = request.base_url
9
- end
7
+ protect_from_forgery with: :exception
10
8
  end
@@ -8,7 +8,7 @@ class ActiveStorage::BlobsController < ActiveStorage::BaseController
8
8
  include ActiveStorage::SetBlob
9
9
 
10
10
  def show
11
- expires_in ActiveStorage::Blob.service.url_expires_in
11
+ expires_in ActiveStorage.service_urls_expire_in
12
12
  redirect_to @blob.service_url(disposition: params[:disposition])
13
13
  end
14
14
  end
@@ -13,16 +13,19 @@ class ActiveStorage::DiskController < ActiveStorage::BaseController
13
13
  else
14
14
  head :not_found
15
15
  end
16
+ rescue Errno::ENOENT
17
+ head :not_found
16
18
  end
17
19
 
18
20
  def update
19
21
  if token = decode_verified_token
20
22
  if acceptable_content?(token)
21
23
  disk_service.upload token[:key], request.body, checksum: token[:checksum]
22
- head :no_content
23
24
  else
24
25
  head :unprocessable_entity
25
26
  end
27
+ else
28
+ head :not_found
26
29
  end
27
30
  rescue ActiveStorage::IntegrityError
28
31
  head :unprocessable_entity
@@ -8,7 +8,7 @@ class ActiveStorage::RepresentationsController < ActiveStorage::BaseController
8
8
  include ActiveStorage::SetBlob
9
9
 
10
10
  def show
11
- expires_in ActiveStorage::Blob.service.url_expires_in
11
+ expires_in ActiveStorage.service_urls_expire_in
12
12
  redirect_to @blob.representation(params[:variation_key]).processed.service_url(disposition: params[:disposition])
13
13
  end
14
14
  end
@@ -0,0 +1,15 @@
1
+ # frozen_string_literal: true
2
+
3
+ # Sets the <tt>ActiveStorage::Current.host</tt> attribute, which the disk service uses to generate URLs.
4
+ # Include this concern in custom controllers that call ActiveStorage::Blob#service_url,
5
+ # ActiveStorage::Variant#service_url, or ActiveStorage::Preview#service_url so the disk service can
6
+ # generate URLs using the same host, protocol, and base path as the current request.
7
+ module ActiveStorage::SetCurrent
8
+ extend ActiveSupport::Concern
9
+
10
+ included do
11
+ before_action do
12
+ ActiveStorage::Current.host = request.base_url
13
+ end
14
+ end
15
+ end
@@ -17,7 +17,12 @@ export class BlobRecord {
17
17
  this.xhr.setRequestHeader("Content-Type", "application/json")
18
18
  this.xhr.setRequestHeader("Accept", "application/json")
19
19
  this.xhr.setRequestHeader("X-Requested-With", "XMLHttpRequest")
20
- this.xhr.setRequestHeader("X-CSRF-Token", getMetaValue("csrf-token"))
20
+
21
+ const csrfToken = getMetaValue("csrf-token")
22
+ if (csrfToken != undefined) {
23
+ this.xhr.setRequestHeader("X-CSRF-Token", csrfToken)
24
+ }
25
+
21
26
  this.xhr.addEventListener("load", event => this.requestDidLoad(event))
22
27
  this.xhr.addEventListener("error", event => this.requestDidError(event))
23
28
  }
@@ -2,6 +2,10 @@
2
2
 
3
3
  # Provides asynchronous analysis of ActiveStorage::Blob records via ActiveStorage::Blob#analyze_later.
4
4
  class ActiveStorage::AnalyzeJob < ActiveStorage::BaseJob
5
+ queue_as { ActiveStorage.queues[:analysis] }
6
+
7
+ retry_on ActiveStorage::IntegrityError, attempts: 10, wait: :exponentially_longer
8
+
5
9
  def perform(blob)
6
10
  blob.analyze
7
11
  end
@@ -1,5 +1,4 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  class ActiveStorage::BaseJob < ActiveJob::Base
4
- queue_as { ActiveStorage.queue }
5
4
  end
@@ -2,7 +2,10 @@
2
2
 
3
3
  # Provides asynchronous purging of ActiveStorage::Blob records via ActiveStorage::Blob#purge_later.
4
4
  class ActiveStorage::PurgeJob < ActiveStorage::BaseJob
5
+ queue_as { ActiveStorage.queues[:purge] }
6
+
5
7
  discard_on ActiveRecord::RecordNotFound
8
+ retry_on ActiveRecord::Deadlocked, attempts: 10, wait: :exponentially_longer
6
9
 
7
10
  def perform(blob)
8
11
  blob.purge
@@ -3,9 +3,8 @@
3
3
  require "active_support/core_ext/module/delegation"
4
4
 
5
5
  # Attachments associate records with blobs. Usually that's a one record-many blobs relationship,
6
- # but it is possible to associate many different records with the same blob. If you're doing that,
7
- # you'll want to declare with <tt>has_one/many_attached :thingy, dependent: false</tt>, so that destroying
8
- # any one record won't destroy the blob as well. (Then you'll need to do your own garbage collecting, though).
6
+ # but it is possible to associate many different records with the same blob. A foreign-key constraint
7
+ # on the attachments table prevents blobs from being purged if they’re still attached to any records.
9
8
  class ActiveStorage::Attachment < ActiveRecord::Base
10
9
  self.table_name = "active_storage_attachments"
11
10
 
@@ -15,17 +14,18 @@ class ActiveStorage::Attachment < ActiveRecord::Base
15
14
  delegate_missing_to :blob
16
15
 
17
16
  after_create_commit :analyze_blob_later, :identify_blob
17
+ after_destroy_commit :purge_dependent_blob_later
18
18
 
19
- # Synchronously purges the blob (deletes it from the configured service) and destroys the attachment.
19
+ # Synchronously deletes the attachment and {purges the blob}[rdoc-ref:ActiveStorage::Blob#purge].
20
20
  def purge
21
- destroy
22
- blob.purge
21
+ delete
22
+ blob&.purge
23
23
  end
24
24
 
25
- # Destroys the attachment and asynchronously purges the blob (deletes it from the configured service).
25
+ # Deletes the attachment and {enqueues a background job}[rdoc-ref:ActiveStorage::Blob#purge_later] to purge the blob.
26
26
  def purge_later
27
- destroy
28
- blob.purge_later
27
+ delete
28
+ blob&.purge_later
29
29
  end
30
30
 
31
31
  private
@@ -36,4 +36,13 @@ class ActiveStorage::Attachment < ActiveRecord::Base
36
36
  def analyze_blob_later
37
37
  blob.analyze_later unless blob.analyzed?
38
38
  end
39
+
40
+ def purge_dependent_blob_later
41
+ blob&.purge_later if dependent == :purge_later
42
+ end
43
+
44
+
45
+ def dependent
46
+ record.attachment_reflections[name]&.options[:dependent]
47
+ end
39
48
  end