尝试创建或保存时,“PGError:没有连接到服务器”

时间:2011-06-27 17:08:23

标签: ruby activerecord

Ruby 1.9.1 + ActiveRecord 2.3.5 + Postgres 8.3.7

这是我的代码的草图。忽略遗漏的任何明显的语法细节。下面的模型继承自通过ActiveRecord 2.3.5连接到Postgres 8.3.7数据库的ActiveRecord :: Base。

class TableA
   has_many :tableB
end

class TableB
   belongs_to :tableA
   has_many :tableC
end

class TableC
   belongs_to :tableB
   has_many :tableD
end

class TableD
   belongs_to :tableC
   has_many :tableE
end

class TableE
   belongs_to :tableD
end

# Note that tableA has fids that are referenced in tableE but is not part of this model
#
# Later in the script, in the same global scope, I want to add entries to these tables if
# I cannot find what I need. Bear in mind that this part betrays much Ruby noobiness.

toAdd.each do |widget|
   add_tableA = TableA.find_by_sql().first # assumes I will get one back based on earlier sanity checks

   add_tableB = TableB.find_by_sql().first
   if (add_tableB == nil)
       new_tableB = TableB.new( # value assignments )
       new_tableB.save
       add_tableB = TableB.find_by_sql().first
   end

   add_tableC = TableC.find_by_sql().first
   if (add_tableC == nil)
       new_tableC = TableC.new( # value assignments )
       new_tableC.save
       add_tableC = TableC.find_by_sql().first
   end

   add_tableD = TableD.find_by_sql().first
   if (add_tableD == nil)
       new_tableD = TableD.new( # value assignments )
       new_tableD.save
       add_tableD = TableD.find_by_sql().first
   end

   # I step into TableA again because items in TableE are linked to items in TableA, but they are
   # distinct from the "high level" item I grabbed from TableA earlier.

   add_tableA = TableA.find_by_sql().first
   if (add_tableA == nil)
       new_tableA = TableA.new( # value assignments )
       new_tableA.save
       add_tableA = TableA.find_by_sql().first
   end

   # Now that I have a TableA id to put into TableE, just create TableE row because I know this
   # does not exist yet.

   new_tableE = TableE.new( # value assignments ) # again, this is assumed to be new based on earlier checks
   new_tableE.save

end

总是会发生以下堆栈跟踪:

/...gems/activerecord-2.3.5/lib/active_record/connection_adapters/abstract_adapter.rb:219:in `rescue in log': PGError: no connection to the server (ActiveRecord::StatementInvalid)
: ROLLBACK
    from .../gems/1.9.1/gems/activerecord-2.3.5/lib/active_record/connection_adapters/abstract_adapter.rb:202:in `log'
    from .../gems/1.9.1/gems/activerecord-2.3.5/lib/active_record/connection_adapters/postgresql_adapter.rb:550:in `execute'
    from .../gems/1.9.1/gems/activerecord-2.3.5/lib/active_record/connection_adapters/postgresql_adapter.rb:576:in `rollback_db_transaction'
    from .../gems/1.9.1/gems/activerecord-2.3.5/lib/active_record/connection_adapters/abstract/database_statements.rb:143:in `rescue in transaction'
    from .../gems/1.9.1/gems/activerecord-2.3.5/lib/active_record/connection_adapters/abstract/database_statements.rb:125:in `transaction'
    from .../gems/1.9.1/gems/activerecord-2.3.5/lib/active_record/transactions.rb:182:in `transaction'
    from .../gems/1.9.1/gems/activerecord-2.3.5/lib/active_record/transactions.rb:200:in `block in save_with_transactions!'
    from .../gems/1.9.1/gems/activerecord-2.3.5/lib/active_record/transactions.rb:208:in `rollback_active_record_state!'
    from .../gems/activerecord-2.3.5/lib/active_record/transactions.rb:200:in `save_with_transactions!'

....无论我是在调用save,save!还是创建一个而不是new并保存。

strace显示我只能获得一个BEGIN..INSERT..COMMIT事务来为每次运行工作。在同一个循环运行或下一个循环中,在事务中INSERT的任何后续尝试都会在发送COMMIT之前删除连接。很明显,我在这里做错了,我是如何进入ActiveRecord模型的。

我在第一次成功设置INSERT语句之前就看到了以下的strace。 ActiveRecord中有什么东西允许我在逐步执行表时保留它,或者我只是做错了吗?

rt_sigaction(SIGPIPE, {0x1, [], SA_RESTORER|SA_RESTART, 0x3876c0eb10}, {0x4b2ff0, [], SA_RESTORER|SA_RESTART, 0x3876c0eb10}, 8) = 0

sendto(3, "Q\0\0\2e          SELECT attr.attna"..., 614, 0, NULL, 0) = 614

rt_sigaction(SIGPIPE, {0x4b2ff0, [], SA_RESTORER|SA_RESTART, 0x3876c0eb10}, {0x1, [], SA_RESTORER|SA_RESTART, 0x3876c0eb10}, 8) = 0

poll([{fd=3, events=POLLIN|POLLERR}], 1, -1) = 1 ([{fd=3, revents=POLLIN}])

recvfrom(3, "T\0\0\0:\0\2attname\0\0\0\4\341\0\2\0\0\0\23\0@\377\377\377\377\0"..., 16384, 0, NULL, NULL) = 541

非常感谢任何帮助。

1 个答案:

答案 0 :(得分:1)

谢谢大家。我为任何人花时间试图解决这个问题而道歉。 postgres的这个实例依赖于第二个进程来运行以将触发事件推送到其他进程。该进程未运行,因此数据库服务器在第一次提交的INSERT之后启动。这是一种内部定制的东西。