No commit activity in last 3 years
No release in over 3 years
Helps to maintain referential integrity for Rails polymorphic associations.
2005
2006
2007
2008
2009
2010
2011
2012
2013
2014
2015
2016
2017
2018
2019
2020
2021
2022
2023
2024
2025
 Dependencies

Development

>= 0
>= 0
~> 3.0.2

Runtime

>= 0
 Project Readme

Polymorphic Constraints

Gem Version Build Status Dependency Status Coverage Status Code Climate

Polymorphic Constraints gem introduces some methods to your migrations to help to maintain the referential integrity for your Rails polymorphic associations.

It uses triggers to enforce the constraints. It enforces constraints on insert, update and delete. update and delete constraints works like the :restrict option of foreign key.

Support

It supports the following adapters:

  • sqlite3
  • postgresql
  • mysql2

Supported platforms:

  • Rails versions - 3 and 4.

Installation

Add the following to your Gemfile:

gem 'polymorphic_constraints'

API Examples

This gem adds the following methods to your migrations:

  • add_polymorphic_constraints(relation, associated_table, options)
  • update_polymorphic_constraints(relation, associated_table, options)
  • remove_polymorphic_constraints(relation)

From Rails Guide take these examples:

class Picture < ActiveRecord::Base
  belongs_to :imageable, polymorphic: true
end
 
class Employee < ActiveRecord::Base
  has_many :pictures, as: :imageable, dependent: :destroy
end
 
class Product < ActiveRecord::Base
  has_many :pictures, as: :imageable
end

Add a new migration:

class AddPolymorphicConstraints < ActiveRecord::Migration
  def change
    add_polymorphic_constraints :imageable, :pictures
  end
end

Or you can add it to pictures migration:

class CreateComments < ActiveRecord::Migration
  create_table :pictures do |t|
    t.references :imageable, polymorphic: true
    t.timestamps
  end

  add_polymorphic_constraints :imageable, :pictures
end

For the second method to work properly, the polymorphic tables employees and products have to be in the database first i.e pictures migration should come after the migrations of employees and products.

run: rake db:migrate

This migration will create the necessary triggers to apply insert, update and delete constraints on imageable polymorphic relation.

# insert
>> picture = Picture.new
>> picture.imageable_id = 1
>> picture.imageable_type = 'Product'
>> picture.save # raises ActiveRecord::RecordNotFound exception. there's no product with id 1

>> product = Product.create

>> picture.imageable_id = product.id
>> picture.imageable_type = 'World'
>> picture.save # raises ActiveRecord::RecordNotFound exception. there's no imageable model named 'World'.

>> picture.imageable_type = product.class.to_s # 'Product'
>> picture.save # saves successfully

# update
>> picture.imageable_type = 'Hello'
>> picture.save # raises ActiveRecord::RecordNotFound exception. there's no imageable model named 'Hello'.

>> employee = Employee.create

>> picture.imageable_id = employee.id
>> picture.imageable_type = employee.class.to_s # 'Employee'
>> picture.save # update completes successfully

# delete/destroy
>> employee.delete # raises ActiveRecord::ReferenceViolation exeption. cannot delete because the picture still refers to the employee as the imageable.
>> employee.destroy # destroys successfully. unlike product, employee implements dependent destroy on imageable. so it destroys the picture first, then it destroys itself.
>> Employee.count # 0
>> Picture.count # 0

>> picture = Picture.new
>> picture.imageable_id = product.id
>> picture.imageable_type = product.class.to_s # 'Product'
>> picture.save

>> product.delete # raises ActiveRecord::ReferenceViolation exeption. cannot delete because the picture still refers to the product as the imageable.
>> product.destroy # raises ActiveRecord::ReferenceViolation exeption. works the same as delete because product model hasn't implemented dependent destroy on imageable.

>> another_product = Product.create
>> another_product.delete # deletes successfully as no picture refers to this product.

Exceptions

  • ActiveRecord::RecordNotFound - It is raised when insert/update is attempted on a table with a record that refers to a non-existent polymorphic record in another table.
  • ActiveRecord::ReferenceViolation - It is raised when a delete is attempted from a polymorphic table that is referred to by a record in the associated table.

naming convention: in the above example Product and Employee are polymorphic tables. Picture is an associated table.

Model Search Strategy:

When you add polymorphic constraints like this:

add_polymorphic_constraints :imageable, :pictures

the gem will search for models acting as imageable using ActiveRecord::Base.descendants. This will search all the models including your gems, models directory etc.

You can also explicitly specify the models with which you want to create polymorphic constraints.

add_polymorphic_constraints :imageable, :pictures, polymorphic_models: [:employee]

This will create polymorphic constraints only between pictures and employees. :polymorphic_models will supersede ActiveRecord::Base.descendants search_strategy.

Note: :polymorphic_models option requires an array. The models specified in the array should be in singular form. Make sure the models indeed have the polymorphic relationship (in this example, :employee acting as :imageable with :pictures).

Update Constraints

This gem creates triggers using the existing state of the application. If you add any model later or add new polymorphic relationships in the existing model, it wont have any polymorphic constraint applied to it. For example, if you add a member class later in the application life cycle:

class Member < ActiveRecord::Base
  has_many :pictures, as: :imageable, dependent: :destroy
end

There will be no polymorphic constraints between pictures and members. You have to renew the imageable constraints by adding another migration:

class AgainUpdatePolymorphicConstraints < ActiveRecord::Migration
  def change
    update_polymorphic_constraints :imageable, :pictures
  end
end

This will delete all the existing :imageable constraints and create new ones. You can also specify :polymorphic_models options with update_polymorphic_constraints method. See Model Search Strategy

Note: update_polymorphic_constraints is simply an alias to add_polymorphic_constraints.

Schema Dump

The gem doesn't support ruby schema dump yet. You have to dump sql instead of schema.rb. To do this, change the application config settings:

# app/config/application.rb
config.active_record.schema_format = :sql
rake db:structure:dump

Migration Rollback

add_polymorphic_constraints and update_polymorphic_constraints are both reversible. So you don't need to worry about rollback.

class AddPolymorphicConstraints < ActiveRecord::Migration
  def change
    add_polymorphic_constraints :imageable, :pictures
  end
end

You can also use up and down like this:

class AddPolymorphicConstraints < ActiveRecord::Migration
  def self.up
    add_polymorphic_constraints :imageable, :pictures
  end

  def self.down
    remove_polymorphic_constraints :imageable
  end
end

This remove_polymorphic_constraints will delete all the existing :imageable constraints during rollback.

Caution: After migration, always test if rollback works properly.

Tests

$ rake
$ rake test:unit
$ rake test:integration:all
$ rake test:integration:sqlite
$ rake test:integration:postgresql
$ rake test:integration:mysql

# test different active model versions
$ appraisal install
$ appraisal rake

Problems

Please use GitHub's issue tracker.

TODO

  1. Ruby schema dump
  2. Supporting on_delete, on_update options with :nullify, :restrict and :cascade.

Contributing

  1. Fork it
  2. Create your feature branch (git checkout -b my-new-feature)
  3. Commit your changes (git commit -am 'Added some feature')
  4. Push to the branch (git push origin my-new-feature)
  5. Create a new Pull Request

License

This project rocks and uses MIT-LICENSE.