4

我正在使用Phusion Passenger部署的 Rails 站点上进行维护工作。工作流程与标准的三层 Railsian 测试-开发-生产安排略有不同;相反,针对并行 Oracle 数据库运行相同代码库的两个单独安装;开发站点位于 qa.domain.com,实时站点位于 www.domain.com

我在以下两个环境之间的以下代码片段(来自使用 AuthenticatedSystem 的“vendors_controller.rb”)中遇到了不同的行为:

def create
  @user = current_user || User.new(params[:user])
  @registration = Registration.new(params[:registration])

  unless current_user

    @user.user_type = 'vendor'
    @user.active = 1

    if @user.save
      @user.activate!
      @user.has_role 'owner', @user
      @user.has_role 'vendor'
      self.current_user = user = @user

      @registration.active = 1
      @registration.email = @user.email
      @registration.user_id = @user.id
      if @registration.save
        send_confirmation(@user)
        send_solicitations_notifications(@registration) if @registration.notification_desired == true
        redirect_to thank_you_vendors_path
      else
        # THIS BEHAVIOR DIFFERS ACROSS PRODUCTION AND DEVELOPMENT
        @user.destroy
        self.current_user = user = nil
        # END DIFFERENCE
        respond_to do |format|
          format.html { render :action => 'new' }
          format.xml  { render :xml => @registration.errors, :status => :unprocessable_entity }
        end
      end

    else
      respond_to do |format|
        format.html  { render :action => 'new' }
        format.xml  { render :xml => @user.errors, :status => :unprocessable_entity }
      end
    end
...

如果系统无法创建相应的注册,注释之间的代码会破坏刚刚创建的用户对象。它在开发服务器上工作正常,但在生产服务器上却不行,即使保存注册失败,用户对象也会顽固地挂在数据库周围。将更改推送到生产环境很简单,只需上传控制器文件并touch tmp/restart.txt通过 shell 执行即可。这两个代码库在其他方面是相同的;什么可能导致这种差异?

感谢您的考虑!

贾斯汀

编辑:两个安装之间存在一些差异production.rb,可能有助于诊断问题。在生产上,

config.cache_classes = true

# Full error reports are disabled and caching is turned on
config.action_controller.consider_all_requests_local = false
config.action_controller.perform_caching             = true

在开发过程中,这三个标志设置为它们的反值。谢谢!

4

1 回答 1

1

您应该考虑更改代码的一些事项:

  1. 你没有使用交易
  2. 你在控制器中做的太多了

话虽如此,您出现问题的原因可能是由于生产和开发之间的环境差异,很可能是这样:

config.cache_classes = false

但是,我认为您不应该在生产中更改此设置,因为它会减慢您的所有操作。相反,我建议使用与您的生产环境非常匹配的暂存环境。

为了解决您的问题,我很可能会像这样重写操作:

# using before filters will keep your actions tight
before_filter :cannot_create_user, :if => :signed_in?

def create
  # setup all the objects
  @user = User.new(params[:user])

  @user.user_type = 'vendor'
  @user.active = 1

  @user.has_role 'owner', @user
  @user.has_role 'vendor'

  @registration = @user.registrations.build(params[:registration])
  @registration.active = 1
  @registration.email = @user.email

  # make sure everything is valid before trying to save and activate
  if @user.valid?
    @user.save!  # might not need this if activate calls save!
    @user.activate!

    # this should probably be a sign_in() method... 
    self.current_user = @user

    send_confirmation(@user)
    send_solicitations_notifications(@registration) if @registration.notification_desired?

    redirect_to thank_you_vendors_path
  else
    respond_to do |format|
      format.html { render :action => 'new' }
      format.xml  { render :xml => @registration.errors, :status => :unprocessable_entity }
    end
  end

...
end


protected 

def signed_in?
  !current_user.nil?
end

def cannot_create_user
  respond_to do |format|
    format.html  { render :action => 'new' }
    format.xml  { render :xml => @user.errors, :status => :unprocessable_entity }
  end
end

注意我没有测试这个,它可能不起作用,但你应该明白......如果你有单元测试(我希望你这样做......)你应该能够将它放入并查看它是否有效!

您的下一步是使用accept_nested_attribute_for作为您的注册对象,以便它可以作为用户参数的一部分提交。

我还会重构它,以便所有角色设置等......都在callbacks中完成。

此时,您的create操作很可能非常简单,您可以切换控制器以使用继承的资源

我希望这有帮助!

于 2010-03-16T15:02:55.610 回答