Skip to content

Commit 9587e59

Browse files
committed
wip: append upgrader
1 parent 4b3c0ce commit 9587e59

File tree

9 files changed

+170
-114
lines changed

9 files changed

+170
-114
lines changed

alchemy_cms.gemspec

-2
Original file line numberDiff line numberDiff line change
@@ -43,8 +43,6 @@ Gem::Specification.new do |gem|
4343
gem.add_runtime_dependency "cancancan", [">= 2.1", "< 4.0"]
4444
gem.add_runtime_dependency "coffee-rails", [">= 4.0", "< 6.0"]
4545
gem.add_runtime_dependency "csv", ["~> 3.3"]
46-
gem.add_runtime_dependency "dragonfly", ["~> 1.4"]
47-
gem.add_runtime_dependency "dragonfly_svg", ["~> 0.0.4"]
4846
gem.add_runtime_dependency "gutentag", ["~> 2.2", ">= 2.2.1"]
4947
gem.add_runtime_dependency "image_processing", [">= 1.2"]
5048
gem.add_runtime_dependency "importmap-rails", ["~> 1.2", ">= 1.2.1"]

app/models/alchemy/attachment.rb

-13
Original file line numberDiff line numberDiff line change
@@ -24,19 +24,6 @@ class Attachment < BaseRecord
2424
include Alchemy::Taggable
2525
include Alchemy::TouchElements
2626

27-
# Legacy Dragonfly file attachments
28-
extend Dragonfly::Model
29-
dragonfly_accessor :legacy_file, app: :alchemy_attachments
30-
DEPRECATED_COLUMNS = %i[
31-
legacy_file
32-
legacy_file_name
33-
legacy_file_size
34-
legacy_file_uid
35-
].each do |column|
36-
deprecate column, deprecator: Alchemy::Deprecation
37-
deprecate :"#{column}=", deprecator: Alchemy::Deprecation
38-
end
39-
4027
# Use ActiveStorage file attachments
4128
has_one_attached :file, service: :alchemy_cms
4229

app/models/alchemy/picture.rb

-16
Original file line numberDiff line numberDiff line change
@@ -80,22 +80,6 @@ def self.preprocessor_class=(klass)
8080
@_preprocessor_class = klass
8181
end
8282

83-
# Legacy Dragonfly image attachments
84-
extend Dragonfly::Model
85-
dragonfly_accessor :legacy_image_file, app: :alchemy_pictures
86-
DEPRECATED_COLUMNS = %i[
87-
legacy_image_file
88-
legacy_image_file_format
89-
legacy_image_file_height
90-
legacy_image_file_name
91-
legacy_image_file_size
92-
legacy_image_file_uid
93-
legacy_image_file_width
94-
].each do |column|
95-
deprecate column, deprecator: Alchemy::Deprecation
96-
deprecate :"#{column}=", deprecator: Alchemy::Deprecation
97-
end
98-
9983
# Use ActiveStorage image processing
10084
has_one_attached :image_file, service: :alchemy_cms do |attachable|
10185
# Only works in Rails 7.1

lib/alchemy/upgrader/eight_zero.rb

+28-81
Original file line numberDiff line numberDiff line change
@@ -1,53 +1,42 @@
11
require "alchemy/shell"
2+
require "alchemy/upgrader/tasks/active_storage_migration"
23
require "benchmark"
3-
require "active_storage/service/disk_service"
4+
require "fileutils"
5+
require "thor"
46

57
module Alchemy
68
class Upgrader::EightZero < Upgrader
7-
extend Alchemy::Shell
8-
DEFAULT_CONTENT_TYPE = "application/octet-stream"
9-
DISK_SERVICE = ActiveStorage::Service::DiskService
10-
SERVICE_NAME = :alchemy_cms
11-
12-
# Prevents (down)loading the original file
13-
METADATA = {
14-
identified: true, # Skip identifying file type
15-
analyzed: true, # Skip analyze job
16-
composed: true # Skip checksum check
17-
}
9+
include Thor::Base
10+
include Thor::Actions
1811

1912
class << self
13+
def install_active_storage
14+
Rake::Task["active_storage:install"].invoke
15+
Rake::Task["db:migrate"].invoke
16+
17+
text = <<-YAML.strip_heredoc
18+
19+
alchemy_cms:
20+
service: Disk
21+
root: <%= Rails.root.join("storage") %>
22+
YAML
23+
24+
storage_yml = Rails.application.root.join("config/storage.yml")
25+
if File.exist?(storage_yml)
26+
task.insert_into_file(storage_yml, text)
27+
else
28+
task.create_file(storage_yml, text)
29+
end
30+
end
31+
2032
def migrate_pictures_to_active_storage
2133
pictures_without_as_attachment = Alchemy::Picture.where.missing(:image_file_attachment)
2234
count = pictures_without_as_attachment.count
2335
if count > 0
2436
log "Migrating #{count} Dragonfly image file(s) to ActiveStorage."
2537
realtime = Benchmark.realtime do
2638
pictures_without_as_attachment.find_each do |picture|
27-
Alchemy::Deprecation.silence do
28-
uid = picture.legacy_image_file_uid
29-
key = key_for_uid(uid)
30-
content_type = Mime::Type.lookup_by_extension(picture.legacy_image_file_format) || DEFAULT_CONTENT_TYPE
31-
Alchemy::Picture.transaction do
32-
blob = ActiveStorage::Blob.create!(
33-
key: key,
34-
filename: picture.legacy_image_file_name,
35-
byte_size: picture.legacy_image_file_size,
36-
content_type: content_type,
37-
metadata: METADATA.merge(
38-
width: picture.legacy_image_file_width,
39-
height: picture.legacy_image_file_height
40-
),
41-
service_name: SERVICE_NAME
42-
)
43-
picture.create_image_file_attachment!(
44-
name: :image_file,
45-
record: picture,
46-
blob: blob
47-
)
48-
end
49-
move_file(Rails.root.join("uploads/pictures", uid), key)
50-
end
39+
Alchemy::Upgrader::Tasks::ActiveStorageMigration.migrate_picture(picture)
5140
print "."
5241
end
5342
end
@@ -64,26 +53,7 @@ def migrate_attachments_to_active_storage
6453
log "Migrating #{count} Dragonfly attachment file(s) to ActiveStorage."
6554
realtime = Benchmark.realtime do
6655
attachments_without_as_attachment.find_each do |attachment|
67-
Alchemy::Deprecation.silence do
68-
uid = attachment.legacy_file_uid
69-
key = key_for_uid(uid)
70-
Alchemy::Attachment.transaction do
71-
blob = ActiveStorage::Blob.create!(
72-
key: key,
73-
filename: attachment.legacy_file_name,
74-
byte_size: attachment.legacy_file_size,
75-
content_type: attachment.file_mime_type.presence || DEFAULT_CONTENT_TYPE,
76-
metadata: METADATA,
77-
service_name: SERVICE_NAME
78-
)
79-
attachment.create_file_attachment!(
80-
record: attachment,
81-
name: :file,
82-
blob: blob
83-
)
84-
end
85-
move_file(Rails.root.join("uploads/attachments", uid), key)
86-
end
56+
Alchemy::Upgrader::Tasks::ActiveStorageMigration.migrate_attachment(attachment)
8757
print "."
8858
end
8959
end
@@ -95,31 +65,8 @@ def migrate_attachments_to_active_storage
9565

9666
private
9767

98-
# ActiveStorage::Service::DiskService stores files in a folder structure
99-
# based on the first two characters of the file uid.
100-
def key_for_uid(uid)
101-
case service
102-
when DISK_SERVICE
103-
uid.split("/").last
104-
else
105-
uid
106-
end
107-
end
108-
109-
# ActiveStorage::Service::DiskService stores files in a folder structure
110-
# based on the first two characters of the file uid.
111-
def move_file(uid, key)
112-
case service
113-
when DISK_SERVICE
114-
if File.exist?(uid)
115-
service.send(:make_path_for, key)
116-
FileUtils.mv uid, service.send(:path_for, key)
117-
end
118-
end
119-
end
120-
121-
def service
122-
ActiveStorage::Blob.services.fetch(SERVICE_NAME)
68+
def task
69+
@_task || new
12370
end
12471
end
12572
end

lib/alchemy/upgrader/tasks/.keep

Whitespace-only changes.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,108 @@
1+
require "active_storage/service"
2+
require "active_storage/service/disk_service"
3+
4+
module Alchemy
5+
class Upgrader
6+
module Tasks
7+
class ActiveStorageMigration
8+
DEFAULT_CONTENT_TYPE = "application/octet-stream"
9+
DISK_SERVICE = ActiveStorage::Service::DiskService
10+
SERVICE_NAME = :alchemy_cms
11+
12+
METADATA = {
13+
identified: true, # Skip identifying file type
14+
analyzed: true, # Skip analyze job
15+
composed: true # Skip checksum check
16+
}
17+
18+
class << self
19+
def migrate_picture(picture)
20+
Alchemy::Picture.extend Dragonfly::Model
21+
Alchemy::Picture.dragonfly_accessor :legacy_image_file, app: :alchemy_pictures
22+
23+
Alchemy::Deprecation.silence do
24+
uid = picture.legacy_image_file_uid
25+
key = key_for_uid(uid)
26+
content_type = Mime::Type.lookup_by_extension(picture.legacy_image_file_format) || DEFAULT_CONTENT_TYPE
27+
Alchemy::Picture.transaction do
28+
blob = ActiveStorage::Blob.create!(
29+
key: key,
30+
filename: picture.legacy_image_file_name,
31+
byte_size: picture.legacy_image_file_size,
32+
content_type: content_type,
33+
# Prevents (down)loading the original file
34+
metadata: METADATA.merge(
35+
width: picture.legacy_image_file_width,
36+
height: picture.legacy_image_file_height
37+
),
38+
service_name: SERVICE_NAME
39+
)
40+
picture.create_image_file_attachment!(
41+
name: :image_file,
42+
record: picture,
43+
blob: blob
44+
)
45+
end
46+
move_file(Rails.root.join("uploads/pictures", uid), key)
47+
end
48+
end
49+
50+
def migrate_attachment(attachment)
51+
Alchemy::Attachment.extend Dragonfly::Model
52+
Alchemy::Attachment.dragonfly_accessor :legacy_file, app: :alchemy_attachments
53+
54+
Alchemy::Deprecation.silence do
55+
uid = attachment.legacy_file_uid
56+
key = key_for_uid(uid)
57+
Alchemy::Attachment.transaction do
58+
blob = ActiveStorage::Blob.create!(
59+
key: key,
60+
filename: attachment.legacy_file_name,
61+
byte_size: attachment.legacy_file_size,
62+
content_type: attachment.file_mime_type.presence || DEFAULT_CONTENT_TYPE,
63+
metadata: METADATA,
64+
service_name: SERVICE_NAME
65+
)
66+
attachment.create_file_attachment!(
67+
record: attachment,
68+
name: :file,
69+
blob: blob
70+
)
71+
end
72+
move_file(Rails.root.join("uploads/attachments", uid), key)
73+
end
74+
end
75+
76+
private
77+
78+
# ActiveStorage::Service::DiskService stores files in a folder structure
79+
# based on the first two characters of the file uid.
80+
def key_for_uid(uid)
81+
case service
82+
when DISK_SERVICE
83+
uid.split("/").last
84+
else
85+
uid
86+
end
87+
end
88+
89+
# ActiveStorage::Service::DiskService stores files in a folder structure
90+
# based on the first two characters of the file uid.
91+
def move_file(uid, key)
92+
case service
93+
when DISK_SERVICE
94+
if File.exist?(uid)
95+
service.send(:make_path_for, key)
96+
FileUtils.mv uid, service.send(:path_for, key)
97+
end
98+
end
99+
end
100+
101+
def service
102+
ActiveStorage::Blob.services.fetch(SERVICE_NAME)
103+
end
104+
end
105+
end
106+
end
107+
end
108+
end

lib/alchemy_cms.rb

+1-2
Original file line numberDiff line numberDiff line change
@@ -7,10 +7,9 @@
77
require "acts_as_list"
88
require "action_view/dependency_tracker"
99
require "active_model_serializers"
10+
require "active_storage/engine"
1011
require "awesome_nested_set"
1112
require "cancan"
12-
require "dragonfly"
13-
require "dragonfly_svg"
1413
require "gutentag"
1514
require "importmap-rails"
1615
require "kaminari"

lib/generators/alchemy/install/install_generator.rb

+4
Original file line numberDiff line numberDiff line change
@@ -69,6 +69,10 @@ def install_assets
6969
append_to_file Rails.root.join("app/assets/config/manifest.js"), "//= link alchemy/admin/custom.css\n"
7070
end
7171

72+
def install_active_storage
73+
rake "active_storage:install:migrations"
74+
end
75+
7276
def set_active_storage_service
7377
insert_into_file app_config_path.join("storage.yml"), <<-YAML.strip_heredoc
7478

lib/tasks/alchemy/upgrade.rake

+29
Original file line numberDiff line numberDiff line change
@@ -15,10 +15,33 @@ namespace :alchemy do
1515
namespace :upgrade do
1616
desc "Alchemy Upgrader: Prepares the database and updates Alchemys configuration file."
1717
task prepare: [
18+
"alchemy:upgrade:ensure_dragonfly_gems",
1819
"alchemy:upgrade:database",
1920
"alchemy:upgrade:config"
2021
]
2122

23+
task :ensure_dragonfly_gems do
24+
require "dragonfly"
25+
require "dragonfly_svg"
26+
rescue LoadError
27+
abort <<~WARN
28+
29+
== Alchemy Upgrader ==
30+
31+
Please make sure you have `dragonfly` and `dragonfly_svg` gems installed in order to migrate Alchemy.
32+
33+
Add these to your Gemfile:
34+
35+
gem "dragonfly", "~> 1.4", require: false
36+
gem "dragonfly_svg", "~> 0.0.4", require: false
37+
38+
and run `bundle install`.
39+
40+
Then try again, please!
41+
42+
WARN
43+
end
44+
2245
desc "Alchemy Upgrader: Prepares the database."
2346
task database: [
2447
"alchemy:install:migrations",
@@ -32,10 +55,16 @@ namespace :alchemy do
3255

3356
namespace "8.0" do
3457
task "run" => [
58+
"alchemy:upgrade:8.0:install_active_storage",
3559
"alchemy:upgrade:8.0:migrate_pictures_to_active_storage",
3660
"alchemy:upgrade:8.0:migrate_attachments_to_active_storage"
3761
]
3862

63+
desc "Install active_storage"
64+
task install_active_storage: [:environment] do
65+
Alchemy::Upgrader::EightZero.install_active_storage
66+
end
67+
3968
desc "Migrate pictures to active_storage"
4069
task migrate_pictures_to_active_storage: [:environment] do
4170
Alchemy::Upgrader::EightZero.migrate_pictures_to_active_storage

0 commit comments

Comments
 (0)