Fixing Rails Active Storage with UUID Models
If you’re using UUIDs as primary keys in your Rails application, you might have encountered issues with Active Storage attachments. This is because Active Storage’s default migrations don’t respect the application-wide primary key type configuration. Let’s fix this!
The Problem
When you set up your Rails application to use UUIDs as primary keys (which is a best practice fullstop in my opinion), you typically configure it in migrations:
1
2
3
4
5
create_table "accounts", id: :uuid, default: -> { "gen_random_uuid()" }, force: :cascade do |t|
t.string "title"
t.boolean "active"
...
end
However, if you ran rails active_storage:install
before this change, the generated migrations will assume you use the default of bigint for primary keys. This can lead to foreign key mismatches and potential issues with your attachments.
The most common error you’ll encounter is when trying to remove an entry with Active Storage variants:
1
undefined method `attachment_reflections' for nil:NilClass
This error occurs because the polymorphic association between your model and the Active Storage attachment is broken due to the mismatched primary key types. The record_id
in the active_storage_attachments
table is trying to reference a UUID, but it’s stored as a bigint.
We see these lines in the Rails 8 Active Storage migrations:
1
2
3
4
5
6
7
8
private
def primary_and_foreign_key_types
config = Rails.configuration.generators
setting = config.options[config.orm][:primary_key_type]
primary_key_type = setting || :primary_key
foreign_key_type = setting || :bigint
[ primary_key_type, foreign_key_type ]
end
The Solution
There are two ways to fix this:
1. Configure Active Storage to use UUIDs
Add this configuration to your config/application.rb
:
1
2
3
config.generators do |g|
g.orm :active_record, primary_key_type: :uuid
end
This will ensure that all new Active Storage tables use UUIDs as their primary keys.
2. Either: modify existing migrations
If you’ve already run the Active Storage migrations, you’ll need to modify them. Here’s how to update the tables:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
class FixActiveStorageUuidSupport < ActiveRecord::Migration[7.1]
def change
# Drop existing tables
drop_table :active_storage_attachments
drop_table :active_storage_blobs
drop_table :active_storage_variant_records
# Recreate with UUID support
create_table :active_storage_blobs, id: :uuid do |t|
t.string :key, null: false
t.string :filename, null: false
t.string :content_type
t.text :metadata
t.string :service_name, null: false
t.bigint :byte_size, null: false
t.string :checksum
t.timestamps
t.index [ :key ], unique: true
end
create_table :active_storage_attachments, id: :uuid do |t|
t.string :name, null: false
t.references :record, null: false, polymorphic: true, index: false, type: :uuid
t.references :blob, null: false, type: :uuid
t.timestamps
t.index [ :record_type, :record_id, :name, :blob_id ], name: "index_active_storage_attachments_uniqueness", unique: true
t.foreign_key :active_storage_blobs, column: :blob_id, type: :uuid
end
create_table :active_storage_variant_records, id: :uuid do |t|
t.belongs_to :blob, null: false, index: false, type: :uuid
t.string :variation_digest, null: false
t.index %i[ blob_id variation_digest ], name: "index_active_storage_variant_records_uniqueness", unique: true
t.foreign_key :active_storage_blobs, column: :blob_id, type: :uuid
end
end
end
3. Or: Remake the whole schema from scratch if the app is new
1
2
3
4
bin/rails db:drop
rm db/schema.rb
bin/rails db:create
bin/rails db:migrate
You’ll see the schema changes to UUIDs for the active_storage tables
Notes
- Make sure you have the
pgcrypto
extension enabled in your PostgreSQL database:1 2 3 4 5
class EnableUuidExtension < ActiveRecord::Migration[7.1] def change enable_extension 'pgcrypto' end end
- This migration must be the first one to run in your application.
- If you have existing attachments, you’ll need to handle the data migration carefully and perhaps not just trust a random blogger with this, test it more.