rake db:migrate没有检测到新的迁移?
有经验的Rails / ActiveRecord 2.1.1
- 您创建了第一个版本(例如)ruby脚本\生成脚手架产品标题:字符串描述:text image_url:string
- 这创建(例如)名为20080910122415_create_products.rb的迁移文件
- 您可以使用rake db:migrate应用迁移
- 现在,使用ruby脚本生成一个字段到产品表\ generate migration add_price_to_product price:decimal
- 这将创建一个名为20080910125745_add_price_to_product.rb的迁移文件
- 如果您尝试运行rake db:migrate,它实际上将恢复第一次迁移,而不是应用下一次迁移! 所以你的产品表会被破坏!
- 但是,如果你单独使用rake,它会告诉你一个迁移正在等待
请注意,应用rake db:migrate(一旦表被销毁)将按顺序应用所有迁移。
我找到的唯一解决方法是指定新迁移的版本,如:
rake db:migrate version=20080910125745
所以我想知道:这是一个预期的新行为吗?
你应该可以使用
rake db:migrate:up
强制它继续前进,但是你可能会失去团队中其他人的交错迁移
如果你跑
rake db:migrate
两次,它将重新应用所有迁移。
我在使用SQLite的Windows上遇到相同的行为,它可能是特定于此类环境的错误。
编辑 – 我找到了原因。 在railstie database.rake任务中,您有以下代码:
desc "Migrate the database through scripts in db/migrate. Target specific version with VERSION=x. Turn off output with VERBOSE=false." task :migrate => :environment do ActiveRecord::Migration.verbose = ENV["VERBOSE"] ? ENV["VERBOSE"] == "true" : true ActiveRecord::Migrator.migrate("db/migrate/", ENV["VERSION"] ? ENV["VERSION"].to_i : nil) Rake::Task["db:schema:dump"].invoke if ActiveRecord::Base.schema_format == :ruby end
然后在我的环境变量中
echo %Version% #=> V3.5.0f
在Ruby中
ENV["VERSION"] # => V3.5.0f ENV["VERSION"].to_i #=>0 not nil !
因此rake任务调用
ActiveRecord::Migrator.migrate("db/migrate/", 0)
在ActiveRecord :: Migrator中我们有:
class Migrator#:nodoc: class << self def migrate(migrations_path, target_version = nil) case when target_version.nil? then up(migrations_path, target_version) when current_version > target_version then down(migrations_path, target_version) else up(migrations_path, target_version) end end
是的, rake db:migrate VERSION=0
是rake db:migrate:down
的长版本
编辑 – 我会更新灯塔错误,但我超级公司代理禁止我连接那里
在此期间,您可能会在调用迁移之前尝试取消设置版本…
我恭敬地不同意汤姆! 这是一个错误!! V3.5.0f不是rake迁移的有效版本。 Rake不应该使用它来迁移:因为ruby选择考虑“V3.5.0f”.to_i为0 …
Rake应该大声抱怨VERSION无效,以便用户知道是什么(在你和我之间,通过转换为整数检查版本是YYYYMMDD格式的时间戳有点轻)
[该死的IE6不允许我发表评论! 并且我不能改变浏览器感谢公司]
这不是预期的行为。 我打算建议将此报告为灯塔上的错误,但我看到你已经这样做了 ! 如果您提供更多信息(包括OS /数据库/ ruby版本),我会看一下。
吉恩,
非常感谢您的调查。 你是对的,实际上我认为你已经发现了一个更严重的物种’设计虫’。
发生的事情是rake会抓取你传递给命令行的任何值并将它们存储为环境变量。 最终将被调用的rake任务只是从环境变量中提取这些值。 当db:migrate查询ENV [“VERSION”]时,它实际上会请求你设置的调用rake的version参数。 当您调用rake db:migrate时,您不会传递任何版本。
但是我们确实有一个名为VERSION的环境变量,它已被其他程序设置用于其他目的(我还没有其他程序)。 rake背后的人(或者在database.rake之后)并没有想到会发生这种情况。 这是一个设计错误。 至少,他们可以使用更多特定的变量名称,如“RAKE_VERSION”或“RAKE_PARAM_VERSION”,而不仅仅是“VERSION”。
汤姆,我绝对不会关闭,但编辑我关于灯塔的错误报告以反映这些新发现。
再次感谢Jean的帮助。 我已经在灯塔上发布了这个错误,就像5天agao一样,仍然没有答案!
罗洛