Essentially, this is just a wrapper around the usual ActiveRecord::Assocations::Classmethods#belongs_to that inlines delegation of all active_record attribute setters/getters/helpers (with prefix support) and any additional methods, thus serving as a proxy for another table and model.
- Transparently write to multiple tables from one active record model by proxying attributes and their methods through a second model.
- Easily extract a new table from an existing table, but keep the existing API intact without breaking any consumers.
- When using STI to maintain easy joins, this is an easy way to proxy unique child attributes through to another table for that child
- Can help you enforce the Law of Demeter
This is designed for any belongs_to relationship where lifecycles are tightly coupled and proxying of attribute helpers from belongs_to -> has_one is desired.
(see example below and /spec for more detail)
Add this line to your application's Gemfile:
gem 'acts_as'
And then:
$ bundle
# This pattern encourages foreign keys to be stored on the STI's root table for easy reads.
#
# table :users
# name :string
# clan_id :integer
# profile_id :integer
#
class User
include ActsAs
end
class Rebel < User
acts_as :profile, class_name: 'RebelProfile'
acts_as :clan, prefix: %w( name ), with: %w( delegate_at_will )
end
# table :clans
# name :string
# strength :integer
# cool :boolean
#
class Clan < ActiveRecord::Base
has_many :rebels
def delegate_at_will
'10'
end
end
# table :rebel_profiles
# serial_data :string
#
class RebelProfile < ActiveRecord::Base
has_one :rebel
end
Now a whole slew of methods related to ActiveRecord attributes are available for the fields being delegated to another table
# Fully Proxied Setters/Getters
rebel.strength = 10
rebel.clan.strength = 20
rebel.strength #=> 20
# ActiveModel::Dirty helpers
rebel.strength_was #=> 10
# Shorthand for prefix-delegating specific fields
rebel.clan_name #=> rebel.clan.name
# Automagic boolean helpers
rebel.cool? #=> rebel.clan.cool?
# Any method you want
rebel.delegate_at_will #=> '10'
Just get to what you want
Rebel.where(strength: 10) #=> the same as Rebel.joins(:clan).where(clan: {strength: 10)
One major caveat, it only works for first-level attributes at the moment. (see pending spec)
If your actor/acted association is 1:1 (see rebel -> profile association in spec), just specify foreign_key: :id and acts_as will automatically ensure the creation of the acted model with a primary key that matches the actor after_create.
acts_as :profile, class_name: 'RebelProfile', foreign_key: :id
# later in RebelProfile
has_one :rebel, foreign_key: :id
Now any Rebel.create! will automatically trigger a creation of a RebelProfile by the same ID. Because the primaries keys are matched between tables, joins against either of these tables should be easier.
If your actor/acted association is X:1 (see rebel -> clan association in spec), feel free to add your own foreign key to the belongs_to table and manage the association lifecycle yourself.
- Fork it
- Create your feature branch (
git checkout -b my-new-feature
) - Commit your changes (
git commit -am 'Add some feature'
) - Push to the branch (
git push origin my-new-feature
) - Create new Pull Request