Rails 3 - ошибки разработки в производственном режиме - PullRequest
7 голосов
/ 16 марта 2011

Я использую Rails, Passenger (оба версии 3.0.5) и Nginx на моем производственном сервере. Как я слышал, Rails должен показывать public/404.html или public/500.html вместо ошибок разработки, таких как ActiveRecord::RecordNotFound или Unknown action, но этого не происходит. Я пытался удалить файл config.ru и установить rack_env или rails_env в nginx.conf, но ничего не помогло.

Вот мой nginx.conf:

worker_processes  1;

events {
    worker_connections  1024;
}

http {
    passenger_root /home/makk/.rvm/gems/ruby-1.9.2-p0/gems/passenger-3.0.5;
    passenger_ruby /home/makk/.rvm/bin/passenger_ruby;
    #passenger_ruby /home/makk/.rvm/wrappers/ruby-1.9.2-p0/ruby;

    include       mime.types;
    default_type  application/octet-stream;

    sendfile        on;
    keepalive_timeout  65;

    server {
        listen       80;
        server_name  localhost;

        location / {
            root   /home/makk/projects/1server/deploy/current/public;
            index  index.html index.htm;
            passenger_enabled on;
            rack_env production;

            recursive_error_pages on;

            if (-f /home/makk/projects/1server/maintenance.html) {
              return 503;
            }

            error_page 404 /404.html;
            error_page 500 502 504 /500.html;
            error_page 503 @503;
        }

        location @503 {
            error_page 405 = /maintenance.html;

            # Serve static assets if found.
              if (-f $request_filename) {
              break;
            }
            rewrite ^(.*)$ /maintenance.html break;
        }

        location ~ ^(\/phpmyadmin\/)(.*)$ {
        fastcgi_pass    127.0.0.1:9000;
        fastcgi_index   index.php;
        fastcgi_split_path_info         ^(\/phpmyadmin\/)(.*)$;
        fastcgi_param   SCRIPT_FILENAME /usr/share/phpmyadmin/$fastcgi_path_info;
        include         fastcgi_params;
        }
    }
}

Кажется, что этот вопрос дублирует этот , но никаких рабочих предложений нет.

UPD : У меня есть приложения для разработки и производства на одном ПК. В производстве Rails игнорирует config.consider_all_requests_local = false (в /config/environments/production.rb) из-за метода local_request?. Таким образом, одно из возможных решений приведено ниже (взято из здесь ):

# config/initializers/local_request_override.rb
module CustomRescue
  def local_request?
    return false if Rails.env.production? || Rails.env.staging?
    super
  end
end

ActionController::Base.class_eval do
  include CustomRescue
end

Или для Rails 3:

class ActionDispatch::Request
 def local?
   false
 end
end

Ответы [ 4 ]

5 голосов
/ 16 марта 2011

Чтобы это работало в Rails 3, вам нужно сделать следующее:

Сначала создайте страницы ошибок 404 и 500.Я помещаю мои в app/views/errors/404.html.erb и app/views/errors/500.html.erb.

Во-вторых, добавьте в application_controller.rb следующее:

unless Rails.application.config.consider_all_requests_local
  rescue_from Exception, :with => :render_error
  rescue_from ActiveRecord::RecordNotFound, :with => :render_not_found
  rescue_from AbstractController::ActionNotFound, :with => :render_not_found
  rescue_from ActionController::RoutingError, :with => :render_not_found
  rescue_from ActionController::UnknownController, :with => :render_not_found
  rescue_from ActionController::UnknownAction, :with => :render_not_found
end

def render_error exception
  Rails.logger.error(exception)
  render :template => "/errors/500.haml", :status => 500
end

def render_not_found exception
  Rails.logger.error(exception)
  render :template => "/errors/404.haml", :status => 404
end

Наконец, сделайте ваш production.rb не учитывающим все запросы локально:

config.consider_all_requests_local = false

PS: Имейте в виду, что, когда происходит полная ошибка маршрутизации - то есть, когда нет абсолютно никакого совпадения маршрута, а не только ошибка ActiveFecord NotFound, будет отображаться public / 404.html, поэтомухорошо иметь это на месте.Обычно я просто перенаправляю его на мой errors_controller, который гарантирует, что любые 404 ошибки, не обнаруженные последними упомянутыми исключениями, все еще правильно перенаправляются в ErrorsController.

<script type="text/javascript">
<!--
window.location = "<%= request.host_with_port %>/errors/404"
//-->
</script>
3 голосов
/ 27 июня 2011

Для получения ошибок только в определенных контроллерах

class AdminController < ApplicationController
  rescue_from Exception, :with => :render_simple_error if Rails.env.production?
  def render_simple_error(e)
    render :text => "#{e.message} -- #{e.class}<br/>#{e.backtrace.join("<br/>")}"
  end
end
2 голосов
/ 16 марта 2011

Переместите объявление root в блок «сервер», чтобы сервер мог использовать страницы обработки ошибок, определенные приложением Rails, при получении ошибок 5XX, 4XX.

Или добавьте директивы error_page в блок местоположения, где используется пассажир, чтобы обработчики ошибок могли при необходимости разрешать public/5xx.html в общедоступном каталоге приложения Rails.

Если ваше приложение не обслуживает страницу, а nginx не отображает статическую страницу, оно не может обслуживать нужную вам страницу.

1 голос
/ 23 марта 2012

На пассажире apache я использовал опцию конфигурации apache PassengerFriendlyErrorPages выкл.

...