#nginx #redirect
#nginx #перенаправление
Вопрос:
У меня есть URL:
https://www.example.com/test.html
Этот URL-адрес может иметь параметры запроса, такие как:
https://www.example.com/test.html?cat=11amp;rating=5
Я хочу перенаправить ТОЛЬКО:
https://www.example.com/test.html
Для:
https://www.example.com/test2
Когда я делаю это через:
location = /test.html {
return 301 https://example.com/test2;
}
URL-адрес с параметрами также перенаправляет:
https://www.example.com/test.html?cat=11amp;rating=5
перенаправляет на: https://example.com/test2
Возможно ли перенаправить URL строго без влияющих параметров?
ОБНОВЛЕНИЕ: я все еще хочу https://www.example.com/test.html?cat=11amp;rating=5 работать в обычном режиме, но https://www.example.com/test.html следует сделать перенаправление на /test2
ОБНОВЛЕНИЕ: раздел моего локального сервера docker в NGINX
server {
listen 80;
listen [::]:80;
server_name magento.test;
set $MAGE_ROOT /var/www/php;
set $maintenance off;
if (-f $MAGE_ROOT/maintenance.enable){
set $maintenance on;
}
include /var/www/php/nginx.conf;
}
Обновить:
root $MAGE_ROOT;
index index.php index.html index.htm;
autoindex off;
charset UTF-8;
error_page 404 403 = /errors/404.php;
#add_header "X-UA-Compatible" "IE=Edge";
add_header 'X-Content-Type-Options' 'nosniff';
# Deny access to sensitive files
location /.user.ini {
deny all;
}
# PHP entry point for setup application
location ~* ^/setup($|/) {
root $MAGE_ROOT;
location ~ ^/setup/index.php {
fastcgi_pass php:9000;
fastcgi_param PHP_FLAG "session.auto_start=off n suhosin.session.cryptua=off";
fastcgi_param PHP_VALUE "memory_limit=4000M n max_execution_time=600";
fastcgi_read_timeout 600s;
fastcgi_connect_timeout 600s;
fastcgi_index index.php;
fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
include fastcgi_params;
}
location ~ ^/setup/(?!pub/). {
deny all;
}
location ~ ^/setup/pub/ {
add_header X-Frame-Options "SAMEORIGIN";
}
}
# PHP entry point for update application
location ~* ^/update($|/) {
root $MAGE_ROOT;
location ~ ^/update/index.php {
fastcgi_split_path_info ^(/update/index.php)(/. )$;
fastcgi_pass php:9000;
fastcgi_index index.php;
fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
fastcgi_param PATH_INFO $fastcgi_path_info;
include fastcgi_params;
}
# Deny everything but index.php
location ~ ^/update/(?!pub/). {
deny all;
}
location ~ ^/update/pub/ {
add_header X-Frame-Options "SAMEORIGIN";
}
}
location / {
try_files $uri $uri/ /index.php$is_args$args;
}
location /pub/ {
location ~ ^/pub/media/(downloadable|customer|import|custom_options|theme_customization/.*.xml) {
deny all;
}
alias $MAGE_ROOT/pub/;
add_header X-Frame-Options "SAMEORIGIN";
}
location /static/ {
# Uncomment the following line in production mode
# expires max;
# Remove signature of the static files that is used to overcome the browser cache
location ~ ^/static/version {
rewrite ^/static/(versiond*/)?(.*)$ /static/$2 last;
}
location ~* .(ico|jpg|jpeg|png|gif|svg|js|css|swf|eot|ttf|otf|woff|woff2|html|json)$ {
add_header Cache-Control "public";
add_header X-Frame-Options "SAMEORIGIN";
expires 1y;
if (!-f $request_filename) {
rewrite ^/static/(versiond*/)?(.*)$ /static.php?resource=$2 last;
}
}
location ~* .(zip|gz|gzip|bz2|csv|xml)$ {
add_header Cache-Control "no-store";
add_header X-Frame-Options "SAMEORIGIN";
expires off;
if (!-f $request_filename) {
rewrite ^/static/(versiond*/)?(.*)$ /static.php?resource=$2 last;
}
}
if (!-f $request_filename) {
rewrite ^/static/(versiond*/)?(.*)$ /static.php?resource=$2 last;
}
add_header X-Frame-Options "SAMEORIGIN";
}
location /media/ {
try_files $uri $uri/ /get.php$is_args$args;
location ~ ^/media/theme_customization/.*.xml {
deny all;
}
location ~* .(ico|jpg|jpeg|png|gif|svg|js|css|swf|eot|ttf|otf|woff|woff2)$ {
add_header Cache-Control "public";
add_header X-Frame-Options "SAMEORIGIN";
expires 1y;
try_files $uri $uri/ /get.php$is_args$args;
}
location ~* .(zip|gz|gzip|bz2|csv|xml)$ {
add_header Cache-Control "no-store";
add_header X-Frame-Options "SAMEORIGIN";
expires off;
try_files $uri $uri/ /get.php$is_args$args;
}
add_header X-Frame-Options "SAMEORIGIN";
}
location /media/customer/ {
deny all;
}
location /media/downloadable/ {
deny all;
}
location /media/import/ {
deny all;
}
location /media/custom_options/ {
deny all;
}
location /errors/ {
location ~* .xml$ {
deny all;
}
}
# PHP entry point for main application
location ~ ^/(index|get|static|errors/report|errors/404|errors/503|health_check).php$ {
try_files $uri =404;
fastcgi_pass php:9000;
fastcgi_buffers 1024 4k;
fastcgi_param PHP_FLAG "session.auto_start=off n suhosin.session.cryptua=off";
fastcgi_param PHP_VALUE "memory_limit=4000M n max_execution_time=18000";
fastcgi_read_timeout 600s;
fastcgi_connect_timeout 600s;
fastcgi_index index.php;
fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
include fastcgi_params;
}
gzip on;
gzip_disable "msie6";
gzip_comp_level 6;
gzip_min_length 1100;
gzip_buffers 16 8k;
gzip_proxied any;
gzip_types
text/plain
text/css
text/js
text/xml
text/javascript
application/javascript
application/x-javascript
application/json
application/xml
application/xml rss
image/svg xml;
gzip_vary on;
# Banned locations (only reached if the earlier PHP entry point regexes don't match)
location ~* (.php$|.phtml$|.htaccess$|.git) {
deny all;
}
Ответ №1:
Аргументы запроса не подлежат проверке в location
rewrite
директивах or, они работают только с нормализованным URI HTTP-запроса. Единственный способ сделать то, что вы хотите, это проверить $args
(или $is_args
) переменную:
location = /test.html {
if ($is_args = '') {
return 301 /test2;
}
... # default request processing
}
Если вы покажете свой полный server
блок, возможно, я мог бы предложить более оптимизированное решение. Например, если у вас есть только один location
блок, должно работать что-то вроде этого:
location / {
if ($is_args = '') {
rewrite ^/test.html$ /test2 permanent;
}
... # default request processing
}
Однако в этой конфигурации может быть предостережение. Я не знаю, учитывают ли различные браузеры строку запроса при кэшировании перенаправлений, и это может привести к безусловному перенаправлению с любым запросом /test.html
независимо от наличия строки запроса. Чтобы избежать этого, я предлагаю использовать временное перенаправление 302:
return 302 /test2;
или
rewrite ^/test.html$ /test2 redirect;
Обновить
Вы можете либо добавить эту проверку в server
контекст в главном файле конфигурации:
server {
listen 80;
listen [::]:80;
server_name magento.test;
set $MAGE_ROOT /var/www/php;
set $maintenance off;
if (-f $MAGE_ROOT/maintenance.enable){
set $maintenance on;
}
if ($is_args = '') {
rewrite ^/test.html$ /test2 permanent;
}
include /var/www/php/nginx.conf;
}
или на location / { ... }
блокировку файла конфигурации magento:
location / {
if ($is_args = '') {
rewrite ^/test.html$ /test2 permanent;
}
try_files $uri $uri/ /index.php$is_args$args;
}
Комментарии:
1. @KalvinKlien 1.
$args
значение переменной не содержит знака вопроса, есть правильная директиваreturn 301 /hardware.html?$args;
. 2. Вы уверены, что это не кэшированный результат? Можете ли вы проверить фактический ответ сервера с помощью чего-то вродеcurl
из командной строки? В любом случае я собираюсь проверить это в своей песочнице, я расскажу вам, что я получил после тестирования этого.2. @KalvinKlien О, конечно, у вас бесконечный цикл с этой конфигурацией, поскольку вы перенаправляете
/hardware.html?some_query_string
на себя. И я имею в виду кэширование внутри браузера (они кэшируют 301 перенаправление, как и любой другой статический контент), а не кэширование nginx. Вам нужно обработать этот/hardware.html
запрос, как и любые другие запросы, но мне нужно увидеть весь вашserver
блок, чтобы предложить, как это сделать.3. @KalvinKlien Еще раз — обе
location
rewrite
директивы and работают с нормализованным URI HTTP-запроса, а не со всей строкой запроса. Вы не можете проверить строку запроса сlocation
rewrite
помощью директив or, нормализованный URI не содержит этой строки, он содержит только/hardware.html
в вашем случае, независимо от того, присутствует строка запроса или нет. Вам не нужно экранировать/
символ в регулярных выражениях nginx (хотя вы можете сделать это, если хотите), но если вы не экранируете символ точки, он будет соответствовать любому символу.4. @KalvinKlien Мне нужно увидеть ваши
location
блоки, этот мне ничего не дал. Возможно, они находятся внутри/var/www/php/nginx.conf
? Посмотрите на мой второй пример, который я используюlocation / { ... }
, а не/location = test.html { ... }
. Это потому, что я хочу обработать любой другой запрос по умолчанию, в том числе/hardware.html
с аргументами запроса.5. @KalvinKlien Вы можете попробовать
if ($is_args = '') { rewrite ^/hardware.html$ /test-page permanent; }
напрямую вserver
контексте вне каких-либоlocation
блоков, добавьте это передinclude /var/www/php/nginx.conf;
строкой.