Tech Dev

Using ActiveRecord and Metaprogramming to Define Constants for Enumerated Types

A Code Monkey Writes Enumerated Types for Fruit

Have you ever stored your application's enumerated types in a database? If you have, you might have also noticed that code will often times "duplicate" this data by defining constants or enums that reference what is in the database. If you're anything like me, this duplication does not feel right.

With the metaprogramming capabilities of Ruby, we can address the maintenance costs of duplication by generating these constants at runtime.

Take for example the following data, which represents available fruits in our system:

id   name
1    orange
2    apple
3    banana

In our system, we want to be able to refer to these fruit types in the code as FruitType::ORANGE, etc. Let's start with a unit test to document the behavior which we want.

class FruitTypeTest < ActiveSupport::TestCase
  def test_constants_defined
    assert_equal FruitType.find_by_name('orange'), FruitType::ORANGE
    assert_equal FruitType.find_by_name('apple'), FruitType::APPLE
    assert_equal FruitType.find_by_name('banana'), FruitType::BANANA
  end
end

This test fails with undefined constant, which is exactly what we'd expect. Now that we have a failing unit test, we can write some code to make the test pass.

class FruitType < ActiveRecord::Base
  find(:all).each do |type|
    eval "#{type.name.upcase} = type"
  end
end

With only those couple lines of code, the test passes! But how exactly does this code work?

First, we leverage active record to find all the fruit types defined in the database. This is pretty basic active record stuff, and isn't that interesting.

The next bit is a little more interesting. We enumerate over each result, and use metaprogramming to define a constant in the class for every fruit type found in the database. This little bit of code relies on the eval method, which is a Ruby Kernel method that evaluates the passed string as an expression. Since no binding is passed to eval, it evaluates in the current context (which happens to be the FruitType class). The end result is the same as if you had typed ORANGE = find_by_name('orange') in the source code.

While this example is pretty straightforward and simple, much more can be accomplished through this approach. However, discretion should be used when employing this tactic. Just because you can change the behavior of Ruby core classes based on some data you have in your database doesn't mean you should...

Topics: ,

Comments: 1 so far

  1. very clever

    Comment by Oscar Del Ben, Tuesday, May 27, 2008 @ 4:45 pm

Leave a comment

Powered by WP Hashcash

About Pathfinder

  • We design and build extraordinary applications for companies looking to make the next great idea a reality.
  • learn more

Topics

WordPress

Comments about this site: info@pathf.com