Skip to content

jeremiahishere/denormalizer

Repository files navigation

denormalizer

Run and cache methods so they can be used later. Create chainable scopes using the cached outputs of the methods.

Installation

Add to your Gemfile

gem "denormalizer"

Run the installer

rails generate denormalizer:install
rake db:migrate

Denormalize a method in a model. This will run the method and save the result every time the model is updated.

class Widget < ActiveRecord::Base
  def active?
    do_something_complex
  end
  denormalize :active?
end

Denormalize associations. This will run the denormalization method on each of the objects in the association.

class Item < ActiveRecord::Base
  has_many :widgets
  also_denormalize :widgets
end

Use the denormalized data

Book.denormalized_actives
Book.denormalized_not_actives

Book.first.denormalized_active?

Thought Process

I am trying to avoid duplicating complex code in instance methods and scopes. Given this method,

def active?
  received_at >= Time.now && items.expired.empty?
end

You have two choices to get all active records,

def self.active
  all.select(&:active?)
end

scope :active, lambda { where(["widgets.received_at >= ?", Time.now]).where("(select count(*) from items where items.widget_id=widgets.id and (expired_at is null or expired_at >= ?)) = 0", Time.now]) }

The first minimizes code duplication but is not chainable and often slower than the pure sql version. The second solution is fast and chainable but requires two updates every time the active? criteria needs to be updated. Additionally, it requires an updated whenever the Item model’s expired scope is updated.

Todo

  • Add code to make sure association denormalization does loop infinitely

  • Improve testing on the Denormalizer::MethodOutput model

  • Refactor dome of the denormalize mixin code to the MethodOutput model

  • Figure out how to replace mysql with a different database/memory store system

Contributing to denormalizer

  • Check out the latest master to make sure the feature hasn’t been implemented or the bug hasn’t been fixed yet.

  • Check out the issue tracker to make sure someone already hasn’t requested it and/or contributed it.

  • Fork the project.

  • Start a feature/bugfix branch.

  • Commit and push until you are happy with your contribution.

  • Make sure to add tests for it. This is important so I don’t break it in a future version unintentionally.

  • Please try not to mess with the Rakefile, version, or history. If you want to have your own version, or is otherwise necessary, that is fine, but please isolate to its own commit so I can cherry-pick around it.

Copyright © 2012 Jeremiah Hemphill. See LICENSE.txt for further details.

About

Run and cache methods used as query conditionals

Resources

License

Stars

Watchers

Forks

Packages

No packages published