Я использую Nginx 0.7.64, Passenger 2.2.9, Rails 2.3.5. Мой каталог кеширования страниц установлен в / public / cache, и я хотел бы иметь возможность обслуживать кешированные страницы при запросе через HTTP, но всегда попадать в приложение Rails при запросе через HTTPS.
Основная часть моей конфигурации выглядит так:
server {
listen 80;
server_name website.com www.website.com;
proxy_set_header X-Forwarded-Proto http;
root /home/deploy/website/current/public;
passenger_enabled on;
if (-f $document_root/cache/$request_filename.html) {
rewrite (.*) $document_root/cache/$1.html break;
}
}
server {
listen 443;
server_name website.com www.website.com;
root /home/deploy/website/current/public;
passenger_enabled on;
proxy_set_header X-Forwarded-Proto https;
ssl on;
ssl_certificate /home/deploy/website/shared/ssl/www.website.com.combined.crt;
ssl_certificate_key /home/deploy/website/shared/ssl/www.website.com.key;
}
Я ожидаю, что когда я запрашиваю website.com/about, меня обслужат /public/cache/about.html, но вместо этого я попадаю на сервер Rails (это видно из журнала).
Думая, что у меня может быть несоответствующая косая черта (и не вижу $document_root
в большинстве примеров), я также пробовал все следующие варианты, ни один из которых не работает:
if (-f cache$request_filename.html) {
rewrite (.*) cache$1.html break;
}
if (-f /cache$request_filename.html) {
rewrite (.*) /cache$1.html break;
}
if (-f cache/$request_filename.html) {
rewrite (.*) cache/$1.html break;
}
if (-f /cache/$request_filename.html) {
rewrite (.*) /cache/$1.html break;
}
Я также бросил root
, passenger_enabled
, и переписать правила в отдельный location /
блокировать, но это тоже не работает. Я также изменил порядок операторов, чтобы passenger_enabled
придет в конце. Я также пробовал использовать $uri
. Ясно, что я чего-то не понимаю!
Это немного упрощено, так как у меня также есть XML api, который местами кешируется (предположительно, правило перезаписи будет таким же, за исключением .html
части), а также мне нужно будет обслуживать public/cache/index.html
когда корень website.com
запрашивается. Я просто хочу, чтобы хоть что-то работало. :)
Любая помощь приветствуется!
Обновить
Условный
if (-f $document_root/cache$request_uri.html)
Кажется, работает! Однако то, что я подумал бы, переписать не получится! Пытаюсь
if (-f $document_root/cache$request_uri.html) {
rewrite (.*) /cache$1.html break;
break;
}
Перезаписывает URL как /cache/cache/about.html.html
и отправляет его в Rails, который тут же отключает. Выглядит вдвое, да! Но если я напишу просто /cache$1
он посылает /cache/cache/about
в Rails и $1.html
отправляет /about.html.html
в Rails и просто $1
отправляет просто /about
который переходит в Rails и не попадает в кеш. Очевидно, это неправильное поведение. Nginx переписывает его, а затем Passenger тоже переписывает?
try_files
версия:
server {
listen 80;
server_name website.com www.website.com;
location / {
root /home/deploy/website/current/public;
try_files $uri /cache/$uri/index.html /cache/$uri.html /cache/$uri @passenger;
}
location @passenger {
root /home/deploy/website/current/public;
proxy_set_header X-Forwarded-Proto http;
passenger_enabled on;
}
}
Обратите внимание, что это может работать или не работать с пассажиром. Я уверен, что это работает с единорогами, дворнягами и т. Д.
Ответ был найден здесь: https://stackoverflow.com/questions/1177979/nginx-rewrite-rules-with-passenger
Конфигурация оказалась такой:
# root
if (-f $document_root/cache/$uri/index.html) {
rewrite (.*) /cache/$1/index.html break;
}
# pages like /about, cached with .html but accessed without
if (-f $document_root/cache/$uri.html) {
rewrite (.*) /cache/$1.html break;
}
# pages like /api/v1/something.xml, cached as xml
if (-f $document_root/cache/$uri) {
rewrite (.*) /cache/$1 break;
}