No meu servidor web, tenho algumas regras de reescrita adicionadas ao meu arquivo .htaccess localizado na raiz do documento (no meu caso, /var/www/main/public_html/.htaccess
). O objetivo dessas regras de reescrita é eliminar a necessidade de extensões de arquivo .php e .html no URL.
Aqui está o problema: funciona perfeitamente para a maioria dos URLs. No entanto, ao tentar acessar subdiretórios que na verdade são arquivos, a regra de reescrita continua adicionando .php ao final até que o servidor desista e gere um 500
erro.
Para explicar melhor o problema... Digamos que minha estrutura de arquivo seja assim:
/index.php
/file.php
/folder/index.php
/folder/file.php
Esses URLs exibem as páginas corretamente:
example.com
example.com/index
example.com/file
example.com/folder
example.com/folder/index
example.com/folder/file
No entanto, esse loop estranho acontece em uma URL como esta:
example.com/index/file
example.com/file/index
No caso de /index/file
, os logs mostram: O
[core:debug] [pid 253721] core.c(3849): [client #####] AH00121: r->uri = /index/file.php.php.php.php.php.php.php.php.php.php
resultado esperado é que uma URL como essa example.com/index/file
resultaria em um 404
erro
E todo o resto exibe um 404 conforme esperado:
example.com/badfile
example.com/folder/badfile
example.com/badfolder/file
Não tenho muita experiência com regras de reescrita e realmente não sei o que estou fazendo: D Tenho quase certeza de que cometi um erro estúpido em algum lugar. Qualquer ajuda seria apreciada!
O servidor é auto-hospedado rodando em Ubuntu 20.04 com Apache2.
Isto é o que meu arquivo .htaccess contém:
Options +FollowSymLinks -MultiViews
RewriteEngine On
RewriteBase /
# External Redirect
RewriteCond %{THE_REQUEST} ^[A-Z]{3,}\s([^.]+)\.html [NC]
RewriteRule ^ %1 [R=302,L,NE]
RewriteCond %{THE_REQUEST} ^[A-Z]{3,}\s([^.]+)\.php [NC]
RewriteRule ^ %1 [R=302,L,NE]
# Internal Redirect
RewriteCond %{REQUEST_FILENAME}.html -f
RewriteRule ^ %{REQUEST_URI}.html [L]
RewriteCond %{REQUEST_FILENAME}.php -f
RewriteRule ^ %{REQUEST_URI}.php [L]
# Error Documents
ErrorDocument 403 /errors/403.php
ErrorDocument 404 /errors/404.php
ErrorDocument 500 /errors/500.php
O problema é o seu "redirecionamento/reescrita interna". A condição (por exemplo,
%{REQUEST_FILENAME}.html
) não está necessariamente testando a mesma coisa que você está reescrevendo na string de substituição%{REQUEST_URI}.html
(por exemplo, ), o que resulta em um loop de reescrita sem fim em determinados cenários. (Sim, usar oEND
sinalizador - conforme mencionado nos comentários - impediria o loop de reescrita, no entanto, isso não resolve o problema subjacente e ainda resultaria em uma reescrita incorreta e no "URL errado" sendo registrado como 404.)Ao solicitar um URL no formato
example.com/index/file
(ondeindex
também mapeia para um arquivo.php
ou.html
e não é um diretório físico), aREQUEST_FILENAME
variável do servidor está no formato/path/to/index
(observe o/file
), masREQUEST_URI
é/index/file
(o URL solicitado). Portanto, a condição (RewriteCond
diretiva) foi bem-sucedida (uma vez que/path/to/index.php
existe), mas a solicitação foi reescrita incorretamente para/index/file.php
- o que não existe. O mecanismo de reescrita é reiniciado e o mesmo problema ocorre. (OEND
sinalizador faz com que o mecanismo de reescrita pare após a primeira reescrita "incorreta".)Essas regras devem ser escritas assim:
A condição anterior agora está testando o mesmo caminho de arquivo que estamos reescrevendo na string de substituição .