如何使用其他方法扩展ActiveRecord :: Migration?

时间:2010-11-21 20:38:24

标签: ruby-on-rails ruby activerecord

我正在创建一个Ruby gem,并希望使用我自己的帮助程序扩展ActiveRecord :: Migration,以创建必要的列。 (这类似于Devise在为各种身份验证策略创建迁移时所做的事情。)我意识到我添加的功能本身非常简单,可能有更好/更有效的方法 - 我正在尝试这样做学习经验而不是实际应用。我只是想了解如何在Rails中添加新的迁移功能这些具有侵入性的东西。

到目前为止,我已经成功构建了一个gem并安装,但是当我尝试运行如下的迁移时:

class CreatePosts < ActiveRecord::Migration
  def self.up
    create_table :posts do |t|
      t.string :name
      t.string :title
      t.text :content
      t.hideable
      t.tracks_hidden_at
      t.timestamps
    end
  end
end

...它没有说可隐藏的未定义。

我已经研究过Devise做到这一点的方式,我不得不承认我有点失落,但我试图摸索它。我做了以下事情:

使用我的新模型添加扩展ActiveRecord,并创建了一种基于我的新迁移方法应用模式更改的方法

require 'orm_adapter/adapters/active_record'

module HiddenRecord
  module Orm
    # This module contains some helpers and handle schema (migrations):
    #
    #   create_table :accounts do |t|
    #     t.hideable
    #     t.tracks_hidden_timestamp
    #   end
    #
    module ActiveRecord
      module Schema
        include HiddenRecord::Schema

        # Tell how to apply schema methods.
        def apply_hiddenrecord_schema(name, type, options={})
          column name, type.to_s.downcase.to_sym, options
        end
      end
    end
  end
end
ActiveRecord::Base.extend HiddenRecord::Models
ActiveRecord::ConnectionAdapters::Table.send :include, HiddenRecord::Orm::ActiveRecord::Schema
ActiveRecord::ConnectionAdapters::TableDefinition.send :include, HiddenRecord::Orm::ActiveRecord::Schema

创建了一个类似于Devise的schema.rb的Schema模块,它定义了我想在迁移中使用的方法,并调用了一个方法来应用模式

module HiddenRecord
  # Holds schema definition for hiddenrecord model options.
  module Schema
    # Sets the model as having hidable rows
    #
    # == Options
    # * :null - When true, allows the hidden row flag to be null
    # * :default - Used to set default hidden status to true. If not set, default is false (rows are not hidden)
    def hideable(options={})
      null = options[:null] || false
      default = options[:default] || false

      apply_hiddenrecord_schema :hiddenrecord_is_row_hidden, Boolean, :null => null, :default => default
    end

    # Sets the model to record the timestamp when a row was hidden
    def tracks_hidden_timestamp()
      apply_hiddenrecord_schema :hiddenrecord_hidden_at, DateTime
    end
  end
end

为模型添加了支持新字段的方法

module HiddenRecord
  module Models
    # This module implements the hideable API
    module Hideable
      def self.included(base)
        base.class_eval do
          extend ClassMethods
        end
      end

      scope :visible, where(:hiddenrecord_is_row_hidden => true)

      def hidden?
        return hiddenrecord_is_row_hidden || false
      end

      def hide
        hiddenrecord_is_row_hidden = true
      end

      def hide!
        hiddenrecord_is_row_hidden = true
        save!
      end

      def unhide
        hiddenrecord_is_row_hidden = false
      end

      def unhide!
        hiddenrecord_is_row_hidden = false
        save!
      end

    end
  end
end

加载架构和模型文件以及gem的主模块

module HiddenRecord
  autoload :Schema, 'hiddenrecord/schema'
  autoload :Models, 'hiddenrecord/models'
  ...
end
require 'hiddenrecord/models/hideable'
require 'hiddenrecord/models/tracks_hidden_timestamp'

再次,认识到这主要是一种学习经历,我希望有人可以指出我如何做到这一点的正确方向。我在Rails 3上尝试这个。

2 个答案:

答案 0 :(得分:6)

以下是我为以前的项目添加Rails 2和MySQL的自定义迁移字段的方法。效果很好。

我不知道这有多少适用于您的确切需求,所以请随时向我提问。

我把这段代码放在Rails.root / lib / dbd_migration_helper.rb

module Ddb

  module MigrationHelper

    def self.included(base) # :nodoc:
      base.send(:include, InstanceMethods)
    end

    module InstanceMethods
      def active    (column_name=:active)     column(column_name, :boolean, :default=>true) end
      def email     (column_name=:email)      column(column_name, :string)     end
      def latitude  (column_name=:latitude)   column(column_name, :float)      end
      def longitude (column_name=:longitude)  column(column_name, :float)      end
      def position  (column_name=:position)   column(column_name, :integer)    end
    end
  end
end

require 'activerecord'
if defined?(ActiveRecord::ConnectionAdapters::TableDefinition)
   ActiveRecord::ConnectionAdapters::TableDefinition.send(:include, Ddb::MigrationHelper)
end

答案 1 :(得分:2)

快速说明,这些行:

ActiveRecord::ConnectionAdapters::Table.send :include, HiddenRecord::Orm::ActiveRecord::Schema
ActiveRecord::ConnectionAdapters::TableDefinition.send :include, HiddenRecord::Orm::ActiveRecord::Schema

似乎没有包含正确的模块。我认为他们应该是:

ActiveRecord::ConnectionAdapters::Table.send :include, HiddenRecord::Schema
ActiveRecord::ConnectionAdapters::TableDefinition.send :include, HiddenRecord::Schema

但是,您似乎没有在任何地方定义#tracks_hidden_at