Как обслуживать все существующие статические файлы напрямую с NGINX, а остальные проксировать на бэкэнд-сервер.


87
location / {
    proxy_set_header X-Real-IP  $remote_addr;
    proxy_set_header Host $host;
    proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;

    if (-f $request_filename) {
        access_log off;
        expires 30d;
        break;
        }

    if (!-f $request_filename) {
        proxy_pass http://127.0.0.1:8080; # backend server listening
        break;
        }
    }

Вышеупомянутое будет обслуживать все существующие файлы напрямую с помощью Nginx (например, Nginx просто отображает исходный код PHP), в противном случае перенаправляет запрос в Apache. Мне нужно исключить файлы * .php из правила, чтобы запросы на * .php также передавались в Apache и обрабатывались.

Я хочу, чтобы Nginx обрабатывал все статические файлы, а Apache - все динамические файлы.

РЕДАКТИРОВАТЬ: есть подход с использованием белого списка, но он не очень элегантен. Посмотрите все эти расширения, я не хочу этого.

location ~* ^.+.(jpg|jpeg|gif|png|ico|css|zip|tgz|gz|rar|bz2|doc|xls|exe|pdf|ppt|txt|tar|mid|midi|wav|bmp|rtf|js)$ {
    access_log off;
    expires 30d;
    }
location / {
    proxy_set_header X-Real-IP  $remote_addr;
    proxy_set_header Host $host;
    proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
    proxy_pass http://127.0.0.1:8080;
    }

РЕДАКТИРОВАТЬ 2: В более новых версиях Nginx try_filesвместо этого используйте http://wiki.nginx.org/HttpCoreModule#try_files


Чтобы прояснить ситуацию: приведенный выше код будет обслуживать статический файл, если он существует на диске, если файл не существует, запрос передается в Apache. Это работает большую часть времени, поскольку все URL-адреса в моих приложениях используют mod_rewrite (или маршрутизацию) и на самом деле не существуют на диске. Только прямой доступ к файлу * .php является исключением и должен быть проанализирован Apache.
fmalina

Ответы:


152

Используйте try_files и именованный блок местоположения ('@apachesite'). Это удалит ненужное совпадение регулярного выражения и блокировку if. Более эффективным.

location / {
    root /path/to/root/of/static/files;
    try_files $uri $uri/ @apachesite;

    expires max;
    access_log off;
}

location @apachesite {
    proxy_set_header X-Real-IP  $remote_addr;
    proxy_set_header Host $host;
    proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
    proxy_pass http://127.0.0.1:8080;
}

Обновление: предполагается, что в этой конфигурации не существует скрипта php под /path/to/root/of/static/files. Это распространено в большинстве современных фреймворков php. Если ваши устаревшие проекты php содержат как сценарии php, так и статические файлы, смешанные в одной папке, вам может потребоваться внести в белый список все типы файлов, которые должен обслуживать nginx.


Я использую wordpress, мне нужно разместить папку загрузки на сервере nginx?
Chameron

7
Моя проблема с этим решением: nginx не использует другое место (прокси) и пытается показать список каталогов. Таким образом, он выдает ошибку 403, потому что список каталогов не активирован. Изменить: хорошо, если вы удалите, $uri/он работает.
ChristophLSA,

1
Это будет прокси / example для apache, но позволит загрузить /example.php.
Теренс Джонсон,

1
@TerenceJohnson Помещать ваши php-файлы в папку со статическими файлами небезопасно. Обратите внимание, что я установил root /path/to/root/of/*static*/files;. php файлы должны быть помещены в другую папку, недоступную напрямую из Интернета.
Чуан Ма

2
Я знаю, что так и должно быть, но если кто-то ставит Nginx перед любым из многих распространенных готовых приложений PHP, у которых все находится в корневом веб-каталоге и которые полагаются на файлы .htaccess, они не должны копировать и вставьте свою конфигурацию.
Теренс Джонсон

18

Попробуй это:

location / {
    root /path/to/root;
    expires 30d;
    access_log off;
}

location ~* ^.*\.php$ {
    if (!-f $request_filename) {
        return 404;
    }
    proxy_set_header X-Real-IP  $remote_addr;
    proxy_set_header Host $host;
    proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
    proxy_pass http://127.0.0.1:8080;
}

Надеюсь, это сработает. Регулярные выражения имеют более высокий приоритет, чем простые строки, поэтому все запросы, заканчивающиеся на, .phpдолжны перенаправляться в Apache, если существует только соответствующий .phpфайл. Остальные будут обрабатываться как статические файлы. Фактический алгоритм оценки местоположения находится здесь .


6

Если вы используете mod_rewrite, чтобы скрыть расширение ваших скриптов, или если вам просто нравятся красивые URL-адреса, заканчивающиеся на /, тогда вы можете подойти к этому с другой стороны. Скажите nginx, чтобы все, что имеет нестатическое расширение, могло пройти через apache. Например:

location ~* ^.+\.(jpg|jpeg|gif|png|ico|css|zip|tgz|gz|rar|bz2|pdf|txt|tar|wav|bmp|rtf|js|flv|swf|html|htm)$
{
    root   /path/to/static-content;
}

location ~* ^!.+\.(jpg|jpeg|gif|png|ico|css|zip|tgz|gz|rar|bz2|pdf|txt|tar|wav|bmp|rtf|js|flv|swf|html|htm)$ {
    if (!-f $request_filename) {
        return 404;
    }
    proxy_set_header X-Real-IP  $remote_addr;
    proxy_set_header Host $host;
    proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
    proxy_pass http://127.0.0.1:8080;
}

Я нашел первую часть этого фрагмента по адресу: http://code.google.com/p/scalr/wiki/NginxStatic


Это отлично сработало, хотя я все еще получаю ошибки для минифицированных файлов (например, site.min.css, main.min.js и т. Д.). Как мне поймать их в регулярном выражении местоположения? @lo_fye
Garth
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.