Подтвердить что ты не робот

Страница обслуживания nginx с проблемой содержимого

Чтобы показать страницу обслуживания во время развертывания, я всегда использовал следующую конфигурацию в nginx:

if (-f /home/shared/system/maintenance.html) {
    return 503;
}

error_page 503 @maintenance;

location @maintenance {
    root /home/shared/errors;
    rewrite ^(.*)$ /maintenance.html break;
}

И все было нормально, пока мне не понадобилось добавлять статический контент на страницу обслуживания (изображения, таблицы стилей и т.д.).

Нет статического содержимого, работающего с такими журналами в файле error.log:

2011/05/05 02:47:20 [notice] 13760#0: *6 "^(.*)$" matches "/some.jpg", client: x.x.x.x, server: server.com, request: "GET /some.jpg HTTP/1.1", host: "server.com"
2011/05/05 02:47:20 [notice] 13760#0: *6 rewritten data: "/maintenance.html", args: "", client: x.x.x.x, server: server.com, request: "GET /some.jpg 2 HTTP/1.1", host: "server.com"

Что логично - если я переписываю все на maintenance.html, это означает, что статический контент не является исключением.

Но я не могу найти правильное решение для перенаправления на каждый файл, кроме тех, которые физически существуют в папке root /home/shared/errors.

PS. /home/shared/errors не имеет общих ресурсов с общей папкой проекта - это полностью отдельная папка (даже без символических ссылок на /current для проекта.

4b9b3361

Ответ 1

Прости, Фрэнк Фармер, но это не сработает.

Что работает, но не так аккуратно:

И как работают запросы →

  • Я использую правило # 1

    if (-f /home/shared/system/maintenance.html) {
        return 503;
    }
    

    это правило имеет поддержку именованного местоположения location @maintenance, а для общего перенаправления на /maintenance.html в root /home/shared/errors все работает.

  • Эта страница содержит и изображение some.jpg - для получения этого изображения браузер запускает новый запрос, и этот новый запрос снова попадает в правило # 1

    все это было описано в начальном вопросе BUT

    Если я использую магию if, как в ответах Фрэнка Фармера, я могу указать сервер на запрашиваемый файл, но HTTP-ответ будет 503, а браузеры (все, кроме Safari, в моем тесте) вызывают ошибки в консоли отладки и отображают изображение но не обрабатывать CSS файлы в той же ситуации.

    И это важно.

  • Я попытался разрешить это с помощью магии местоположения для моих новых запросов контента - это означает, что я должен:

    • НЕ return 503 для запросов контента и вообще пропустить именованное местоположение.

    • изменить root /home/shared/errors, поскольку содержание обслуживания все еще существует.

  • Наконец, у меня есть следующее решение:

    • создать maintenance-static папку для всего статического содержимого и изменить пути в моем файле maintenance.html и таблицах стилей обслуживания и статических

    • Далее используйте эти правила (я считаю, что они самоописательны), которые заменяют одиночный if (-f /home/shared/system/maintenance.html) в начальном вопросе:

      set $can503 0;
      if (-f /home/shared/system/maintenance.html) {
          set $can503 1;
      }
      if ($uri ~* /maintenance-static/) {
          set $can503 0;
      }
      location /maintenance-static/ {
          root /home/shared/errors;
      }
      if ($can503 = 1) {
         return 503;
      }
      

Это решение работает без ошибок во всех браузерах и для множества страниц в папке shared/errors, например. maintenance.html, error.html, overload.html и т.д.

Это решение не совсем понятно - возможно, вы можете намекнуть мне, как я могу сделать его более аккуратным, но помня, что мы имеем дело с отдельными запросами (и отдельными процессами nginx для каждого файла/запроса в случае высокой нагрузки и т.д.) для initial html и его содержимое мы не можем использовать одно и то же правило с перенаправлением 503 для всех файлов.

Ответ 2

Я потратил два часа на поиски ответа на эту проблему и, наконец, нашел этот пост. Похоже, это должно быть более распространенным явлением. Мое решение упало где-то между Фрэнком и Виллом. Как указано в Wile, некоторые браузеры (например, Chrome) не будут отображать CSS/JS из любых файлов, которые возвращаются 503, даже если они полностью и правильно выбрали их.

Но есть исправление, которое менее хаки, чем то, что сделал Уайл. Просто верните 200!

Мое полное решение было следующим:

error_page 503 @maintenance;

location @maintenance {
    root /path_to_static_root;
    if (!-f $request_filename) {
        rewrite ^(.*)$ /rest_of_path/maintenance.html break;
    }
    return 200;
}

Работал как шарм.:)

Ответ 3

location @maintenance {
    root /home/shared/errors;
    rewrite  (some\.jpg|some2\.gif)$ /$1 break;
    rewrite ^(.*)$ /maintenance.html break;
}

Это может работать без перечисления файлов в белом списке:

location @maintenance {
    root /home/shared/errors;
    if (!-f $request_filename) {
        rewrite ^(.*)$ /maintenance.html break;
    }
}

Ответ 4

это решение работает только с одним оператором if, а также более читаемо и понятно:

upstream unicorn {
        server unix:/path/to/unicorn.sock;
}

server {
        listen 3000 default deferred;

        proxy_read_timeout 3600;
        client_max_body_size 4G;
        set_real_ip_from 0.0.0.0/0;

        root /path/to/current/public;
        try_files $uri/index.html $uri.html $uri @unicorn;

        error_page 404 /404.html;
        error_page 500 502 504 /500.html;
        error_page 503 /system/maintenance.html;

        location /404.html {
                internal;
        }

        location /500.html {
                internal;
        }

        location @unicorn {
                if (-f $document_root/system/maintenance.html) {
                        return 503;
                }

                proxy_set_header Host $http_host;
                proxy_set_header X-Real-IP $remote_addr;
                proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
                proxy_set_header X-Forwarded-Proto $http_x_forwarded_proto;
                proxy_redirect off;
                proxy_pass http://unicorn;
        }
}

Ответ 5

server {
    listen 80;
    server_name myserv.trunk;
    autoindex off;
    access_log /var/log/nginx/sitename-access.log;

    location ^~ /static/ {
       root /home/dev/myserv-site/;
    }

    location ~ /(?P<language>en)?/? {
        uwsgi_pass unix:///tmp/uwsgi.sock;
        include uwsgi_params;
        error_page 502 @fallback;
        error_page 503 @maintenance;
        if (-f /home/dev/myserv-site/maintenance.active) {
          return 503;
        }
    }

    location @fallback {
      root /home/dev/myserv-site/;
      if ($language) {
          rewrite ^(?!\/static\/)(.*)$ /static/html/502.$language.html break;
      }
      rewrite ^(?!\/static\/)(.*)$ /static/html/502.zh.html break;
    }

    location @maintenance {
      root /home/dev/myserv-site/;
      if ($language) {
          rewrite ^(?!\/static\/)(.*)$ /static/html/503.$language.html;
      }
      rewrite ^(?!\/static\/)(.*)$ /static/html/503.zh.html break;
   }
}

Вот пример экрана страницы с переводимым обслуживанием (503) или 502. Если пользователь перейдет на http://site.com/en/, будет отображаться страница (если поддерживается файл maintenance.active). static dir находится в директории myserv-site.

Ответ 6

Я попробовал все вышеперечисленные предложения и, к сожалению, они не работают должным образом. Единственное решение, которое работает только для меня, следующее:

location / {
    if (-f /path/to/file/indicating/maintenance/mode) {
       rewrite ^(.+)$ /maintenance/$1;
    }

    #... the rest of the "normal" logic
}

location /maintenance {
    root /path/where/your/maintenance/root/is;
    rewrite ^/maintenance/(.*)$ /$1 break;
    try_files /$uri /index.html =404;
    return 200;
}

Я знаю, он создает ненужное/техническое обслуживание в обычных условиях, но вы можете представить себе другой путь к местоположению, который ваши пользователи никогда не догадываются:).