我正在创建一个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上尝试这个。
答案 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
。