5

我在新的 Rails 应用程序(3.2.3)中运行迁移时遇到了问题。我们正在使用 postrgres 9.1.3 和 - pg (0.13.2) -

当我运行 rake db:create,然后 rake db:migrate,我得到 ->

1.9.3-p194 (master) rake db:migrate --trace
** Invoke db:migrate (first_time)
** Invoke environment (first_time)
** Execute environment
rake aborted!
PG::Error: ERROR:  relation "roles" does not exist
LINE 4:              WHERE a.attrelid = '"roles"'::regclass
                                    ^
:             SELECT a.attname, format_type(a.atttypid, a.atttypmod), d.adsrc, a.attnotnull
          FROM pg_attribute a LEFT JOIN pg_attrdef d
            ON a.attrelid = d.adrelid AND a.attnum = d.adnum
         WHERE a.attrelid = '"roles"'::regclass
           AND a.attnum > 0 AND NOT a.attisdropped
         ORDER BY a.attnum

即使没有定义任何迁移,我也能得到这个,所以我不认为这是迁移本身的问题。当我查看堆栈跟踪时,我看到在我的用户模型中定义的范围正在运行 - 当我将它们注释掉时,迁移运行没有问题。

scope :team_leaders, where(role_id: Role.where(name: 'Team Leader').first.try(:id))
scope :area_leaders, where(role_id: Role.where(name: 'Area Leader').first.try(:id))
scope :nation_leaders, where(role_id: Role.where(name: 'Nation Leader').first.try(:id))
scope :employees, where(role_id: Role.where(name: 'Employee').first.try(:id))

这是rails中的错误,还是我做错了什么?我真的很感谢一些帮助 - 我们可以在整个应用程序中删除这些范围的使用,但这是我们想要避免的事情。

我是否应该将这些范围放在某种条件中,当 Rails 加载到控制台或作为服务器时调用,而不是在迁移期间调用?

非常感谢,

丹·索特

4

3 回答 3

17

我遇到了完全相同的问题。经过 2 个小时的调试并拔掉我的头发后,这个名叫Carl Zulauf的有福的人在评论中发布了答案。

问题是我们在运行迁移时正在评估范围,因此与尚未迁移的另一个表的任何依赖关系都会导致该错误。

只需用 lambda 包装所有作用域。例如:

scope :team_leaders, lambda { where(role_id: Role.where(name: 'Team Leader').first.try(:id)) }

对所有范围都这样做。

这应该够了吧。他们需要被惰性评估(就在被调用时),并且没有lambda他们被立即评估。

于 2012-05-26T16:52:12.087 回答
3

find_如果您的范围以like开头,find_by_foo那么它们将中断rake db:migrate。这是我的情况的错误。

于 2012-12-26T11:40:13.147 回答
0

我实际上对由默认范围引起的迁移有同样的问题,如下所示:

default_scope where(deleted: false)

错误是由这样的代码块引起的:

ModelName.all.each_with_index do |m, i|
...
end

这个问题通过取消作用域解决:

ModelName.unscoped.each_with_index do |m, i|
...
end
于 2013-10-10T13:19:53.307 回答