我们有script/generate migration add_fieldname_to_tablename fieldname:datatype语法,用于向模型添加新列。

在同一行中,是否有用于更改列的数据类型的脚本/generate ?或者我应该直接将SQL写入我的常规迁移中?

我想将一列从datetime改为date。


当前回答

Rails 5

来自Rails指南:

如果你希望一个迁移做一些活动记录不知道如何反转的事情,你可以使用可逆:

class ChangeTablenameFieldname < ActiveRecord::Migration[5.1]
  def change
    reversible do |dir|
      change_table :tablename do |t|
        dir.up   { t.change :fieldname, :date }
        dir.down { t.change :fieldname, :datetime }
      end
    end
  end
end

其他回答

这都假设列的数据类型对任何现有数据都有隐式转换。我遇到过几种情况,现有的数据,比如String可以隐式转换为新的数据类型,比如Date。

在这种情况下,了解可以使用数据转换创建迁移是很有帮助的。就我个人而言,我喜欢把它们放在我的模型文件中,然后在所有数据库模式都已迁移并稳定之后删除它们。

/app/models/table.rb
  ...
  def string_to_date
    update(new_date_field: date_field.to_date)
  end

  def date_to_string
    update(old_date_field: date_field.to_s)
  end
  ...
    def up
        # Add column to store converted data
        add_column :table_name, :new_date_field, :date
        # Update the all resources
        Table.all.each(&:string_to_date)
        # Remove old column
        remove_column :table_name, :date_field
        # Rename new column
        rename_column :table_name, :new_date_field, :date_field
    end

    # Reversed steps does allow for migration rollback
    def down
        add_column :table_name, :old_date_field, :string
        Table.all.each(&:date_to_string)
        remove_column :table_name, :date_field
        rename_column :table_name, :old_date_field, :date_field
    end

使用# change_column。

change_column(:table_name, :column_name, :date)

# a few more examples:
change_column(:suppliers, :name, :string, limit: 80)
change_column(:accounts, :description, :text)

注意:即使在数据库迁移之外也可以得到相同的结果,这可能对测试/调试很方便,但这个方法需要非常谨慎地使用:

ActiveRecord::Base.connection.change_column(:table_name, :column_name, :date)

只需生成迁移:

rails g migration change_column_to_new_from_table_name

像这样更新迁移:

class ClassName < ActiveRecord::Migration
  change_table :table_name do |table|
    table.change :column_name, :data_type
  end
end

最后

rake db:migrate

你可以使用change_column:

def change
  change_column :table_name, :column_name, :new_data_type
end

Rails 5

来自Rails指南:

如果你希望一个迁移做一些活动记录不知道如何反转的事情,你可以使用可逆:

class ChangeTablenameFieldname < ActiveRecord::Migration[5.1]
  def change
    reversible do |dir|
      change_table :tablename do |t|
        dir.up   { t.change :fieldname, :date }
        dir.down { t.change :fieldname, :datetime }
      end
    end
  end
end