Daniel Doubrovkine bio photo

Daniel Doubrovkine

aka dB., @awscloud, former CTO @artsy, +@vestris, NYC

Email Twitter LinkedIn Github Strava
Creative Commons License

I recently had to deal with reading a schema in a MySQL database in Ruby and migrating data to MongoDB. It has been made real easy with DataMapper, an awesome library that doesn’t force me into creating a complete and rigid model for the MySQL database.

Today’s problem looks as follows. I have a table called fruits. Each fruit _has a field called _color _amongst two dozen other fields. Colors are referred by name, for example _yellow or green. What I would like to do is to promote the colors into a first class object, therefore creating a collection of colors based on the data in the fruits table. Effectively I’d like to SELECT DISTINCT color from fruits, but I’d like the results to be of type Color.

Turns out to be very straightforward with DataMapper.

class Color
  include DataMapper::Resource
  storage_names[:default] = "fruits"
  property :name, String, :field => "color"

  class << self
    alias_method :_all, :all

    def all
      coll = []
      repository(:default).adapter.select("SELECT DISTINCT color FROM fruits").each do |c|
        coll << Color.new({name: c})
      end
      coll
    end
  end
end

This creates a class called Color with a name field that comes from the color column of the fruits table. We replace the Color.all method with our own implementation that makes a DISTINCT query at the adapter level. As a bonus you can see how you can rename the all method (we don’t really call the old _all implementation here, but it’s often useful). This lets you do Color.all.