Строгое перенаправление NGINX без параметров запроса

#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; строкой.