Active Record Doctor helps to keep the database in a good shape. Currently, it can:
- index unindexed foreign keys
- detect extraneous indexes
- detect unindexed
deleted_at
columns - detect missing foreign key constraints
- detect models referencing undefined tables
- detect uniqueness validations not backed by an unique index
- detect missing non-
NULL
constraints - detect missing presence validations
- detect incorrect presence validations on boolean columns
More features coming soon!
Want to suggest a feature? Just shoot me an email.
The preferred installation method is adding active_record_doctor
to your
Gemfile
:
gem 'active_record_doctor', group: :development
Then run:
bundle install
Foreign keys should be indexed unless it's proven ineffective. However, Rails makes it easy to create an unindexed foreign key. Active Record Doctor can automatically generate database migrations that add the missing indexes. It's a three-step process:
- Generate a list of unindexed foreign keys by running
bundle exec rake active_record_doctor:unindexed_foreign_keys > unindexed_foreign_keys.txt
-
Remove columns that should not be indexed from
unindexed_foreign_keys.txt
as a column can look like a foreign key (i.e. end with_id
) without being one. -
Generate the migrations
rails generate active_record_doctor:add_indexes unindexed_foreign_keys.txt
- Run the migrations
bundle exec rake db:migrate
Let me illustrate with an example. Consider a users
table with columns
first_name
and last_name
. If there are two indexes:
- A two-column index on
last_name, first_name
. - A single-column index on
last_name
.
Then the latter index can be dropped as the former can play its role. In
general, a multi-column index on column_1, column_2, ..., column_n
can replace
indexes on:
column_1
column_1, column_2
- ...
column_1, column_2, ..., column_(n - 1)
To discover such indexes automatically just follow these steps:
- List extraneous indexes by running:
bundle exec rake active_record_doctor:extraneous_indexes
-
Confirm that each of the indexes can be indeed dropped.
-
Create a migration to drop the indexes.
The indexes aren't dropped automatically because there's usually just a few of them and it's a good idea to double-check that you won't drop something necessary.
Also, extra indexes on primary keys are considered extraneous too and will be reported.
Note that a unique index can never be replaced by a non-unique one. For
example, if there's a unique index on users.login
and a non-unique index on
users.login, users.domain
then the tool will not suggest dropping
users.login
as it could violate the uniqueness assumption.
If you soft-delete some models (e.g. with paranoia
) then you need to modify
your indexes to include only non-deleted rows. Otherwise they will include
logically non-existent rows. This will make them larger and slower to use. Most
of the time they should only cover columns satisfying deleted_at IS NULL
.
active_record_doctor
can automatically detect indexes on tables with a
deleted_at
column. Just run:
bundle exec rake active_record_doctor:unindexed_deleted_at
This will print a list of indexes that don't have the deleted_at IS NULL
clause. Currently, active_record_doctor
cannot automatically generate
appropriate migrations. You need to do that manually.
If users.profile_id
references a row in profiles
then this can be expressed
at the database level with a foreign key constraint. It forces
users.profile_id
to point to an existing row in profiles
. The problem is
that in many legacy Rails apps the constraint isn't enforced at the database
level.
active_record_doctor
can automatically detect foreign keys that could benefit
from a foreign key constraint (a future version will generate a migrations that
add the constraint; for now, it's your job). You can obtain the list of foreign
keys with the following command:
bundle exec rake active_record_doctor:missing_foreign_keys
The output will look like:
users profile_id
comments user_id article_id
Tables are listed one per line. Each line starts with a table name followed by
column names that should have a foreign key constraint. In the example above,
users.profile_id
, comments.user_id
, and comments.article_id
lack a foreign
key constraint.
In order to add a foreign key constraint to users.profile_id
use the following
migration:
class AddForeignKeyConstraintToUsersProfileId < ActiveRecord::Migration
def change
add_foreign_key :users, :profiles
end
end
Active Record guesses the table name based on the class name. There are a few cases where the name can be wrong (e.g. you forgot to commit a migration or changed the table name). Active Record Doctor can help you identify these cases before they hit production.
IMPORTANT. Models backed by views are supported only in:
- Rails 5+ and any database or
- Rails 4.2 with PostgreSQL.
The only think you need to do is run:
bundle exec rake active_record_doctor:undefined_table_references
If there a model references an undefined table then you'll see a message like this:
The following models reference undefined tables:
Contract (the table contract_records is undefined)
On top of that rake
will exit with status code of 1. This allows you to use
this check as part of your Continuous Integration pipeline.
A model-level uniqueness validations should be backed by a database index in order to be robust. Otherwise you risk inserting duplicate values under heavy load.
In order to detect such validations run:
bundle exec rake active_record_doctor:missing_unique_indexes
If there are such indexes then the command will print:
The following indexes should be created to back model-level uniqueness validations:
users: email
This means that you should create a unique index on users.email
.
If there's an unconditional presence validation on a column then it should be
marked as non-NULL
-able at the database level.
In order to detect columns whose presence is required but that are marked
null: true
in the database run the following command:
bundle exec rake active_record_doctor:missing_non_null_constraint
The output of the command is similar to:
The following columns should be marked as `null: false`:
users: name
You can mark the columns mentioned in the output as null: false
by creating a
migration and calling change_column_null
.
If a column is marked as null: false
then it's likely it should have the
corresponding presence validator.
In order to detect models lacking these validations run:
bundle exec rake active_record_doctor:missing_presence_validation
The output of the command looks like this:
The following models and columns should have presence validations:
User: email, name
This means User
should have a presence validator on email
and name
.
A boolean column's presence should be validated using inclusion or exclusion validators instead of the usual presence validator.
In order to detect boolean columns whose presence is validated incorrectly run:
bundle exec rake active_record_doctor:incorrect_boolean_presence_validation
The output of the command looks like this:
The presence of the following boolean columns is validated incorrectly:
User: active
This means active
is validated with presence: true
instead of
inclusion: { in: [true, false] }
or exclusion: { in: [nil] }
.
This gem is developed and maintained by Greg Navis.