我正在编写一个简单的基于PHP的MVC-ish框架。我希望这个框架能够安装在任何目录中。
我的PHP脚本抓取请求uri并将其分解为段。它使段1成为控制器,段2成为行动。当我这样做时,这一切都很好:
http://www.example.com/mvc/module/test/
它将转到特定的模块控制器和方法。现在我有一个默认控制器,即家庭控制器,它位于文件夹中。
现在直接访问此文件夹http://www.example.com/mvc/home/ 它会显示403被禁止,因为这个文件夹确实存在,相反它也应该回到http://www.example.com/mvc/index.php
如果我将框架安装在不同的文件夹中,让我们说文件夹框架必须重定向回http://www.example.com/framework/index.php
我想将每个文件夹和php文件重定向回index.php,保留其他所有内容。
我遇到的第一个问题是它永远不会重定向到正确的文件夹,总是重定向到域根文件夹。
这就是我的尝试:
RewriteEngine on
RewriteCond %{REQUEST_FILENAME} !-d
RewriteCond %{REQUEST_FILENAME} !-f
RewriteRule . index.php [L]
答案 0 :(得分:205)
你的重写规则看起来几乎没问题。
首先确保您的.htaccess
文件位于文档根目录(与index.php
相同的位置),否则它只会影响它所在的子文件夹(以及其中的任何子文件夹) - 递归)。
接下来对您的规则稍作修改,使其看起来像:
RewriteEngine on
RewriteCond %{REQUEST_FILENAME} !-f
RewriteCond %{REQUEST_FILENAME} !-d
RewriteRule ^(.*)$ /index.php?path=$1 [NC,L,QSA]
目前,您只是匹配.
一个任何字符的实例,您至少需要.*
才能匹配任意字符的任意数量的实例
$_GET['path']
变量将包含伪目录结构,例如/mvc/module/test
,然后您可以在index.php中使用它来确定 Controller 和您想要的操作表演。
如果您希望将整个shebang安装在子目录中,例如/mvc/
或/framework/
,则最简单的方法是稍微更改重写规则以将其考虑在内。
RewriteRule ^(.*)$ /mvc/index.php?path=$1 [NC,L,QSA]
确保index.php
文件位于文档根目录中时,.htaccess
位于该文件夹中。
替代$_GET['path']
(更新于2002年2月和19年1月)
将路径设置为$_GET
变量实际上并不是必需的(现在甚至不常见),许多框架将依赖$_SERVER['REQUEST_URI']
来检索相同的信息 - 通常确定使用哪个 Controller - 但原理完全相同。
这会稍微简化RewriteRule
,因为您不需要创建路径参数(这意味着OP的原始RewriteRule
将现在工作):
RewriteEngine on
RewriteCond %{REQUEST_FILENAME} !-f
RewriteCond %{REQUEST_FILENAME} !-d
RewriteRule ^.*$ /index.php [L,QSA]
但是,关于在子目录中安装的规则仍然适用,例如
RewriteRule ^.*$ /mvc/index.php [L,QSA]
旗帜:
NC
=无案例(不区分大小写,因为模式中没有字符,所以不是必需的)
L
=最后一次(在重写之后它会停止重写,所以确保它是你的重写列表中的最后一件事)
QSA
=查询字符串附加,以防你想要保留的?like=penguins
之类的东西,并传递给index.php。
答案 1 :(得分:24)
要将不存在的所有内容重定向到index.php
,您还可以使用 FallBackResource
指令
FallbackResource /index.php
它与ErrorDocument
的工作方式相同,当您在服务器上请求不存在的路径或文件时,该指令会默默地将请求转发给index.php
。
如果您想将所有内容(including existant files or folders
)重定向到index.php
,可以使用以下内容:
RewriteEngine on
RewriteRule ^((?!index\.php).+)$ /index.php [L]
注意模式^((?!index\.php).+)$
匹配除index.php
之外的任何uri,我们已排除目标路径以防止无限循环错误。
答案 2 :(得分:14)
您可以使用以下内容:
RewriteEngine on
RewriteRule ^.+$ /index.php [L]
这会将每个查询重定向到根目录的index.php。请注意,它还会重定向查询现有的文件,例如图像,javascript文件或样式表。
答案 3 :(得分:6)
这个问题有一个“技巧”适合所有场景,这是一个非常明显的解决方案,您必须尝试相信它实际上有效......:)
这是......
<IfModule mod_rewrite.c>
RewriteEngine On
RewriteCond %{REQUEST_FILENAME} -f [OR]
RewriteCond %{REQUEST_FILENAME} !-f
RewriteRule ^(.*)$ index.php [L,QSA]
</IfModule>
基本上,您要求 MOD_REWRITE 转发index.php
URI请求始终在文件存在时始终在请求的文件不存在时存在!
在调查 MOD-REWRITE 的源代码以了解其工作原理时,我意识到如果引用的文件存在与否,则所有检查都会在验证后发生。只有这样才能处理RegEx
。即使您的URI指向文件夹,Apache也会强制检查其配置文件中列出的索引文件。
基于这个简单的发现,显而易见的是,对于所有可能的呼叫,简单的文件验证就足够了,只要我们双击文件存在检查并将两个结果路由到同一端 - 这一点,涵盖了100%的可能性。
重要提示:请注意index.php
中没有“ / ”。默认情况下, MOD_REWRITE 将使用它设置为“基本文件夹”的文件夹进行转发。它的美妙之处在于它不一定需要是网站的“根文件夹”,允许此解决方案适用于localhost/
和/或您应用它的任何子文件夹。
最终,我之前测试的其他一些解决方案(看起来工作正常的解决方案)打破了PHP通过其相对路径“需要”文件的能力,这是一个无赖。小心。
有些人可能会说这是一个不优雅的解决方案。实际上,它可能是测试,在几种情况下,几个服务器,几个不同的Apache版本等,这个解决方案在所有情况下100%工作!
答案 4 :(得分:3)
愚蠢的回答,但如果你无法弄清楚为什么它没有重定向检查是否为web文件夹启用了以下内容..
AllowOverride All
这将使您能够运行必须运行的htaccess! (有替代方案但不会导致问题https://httpd.apache.org/docs/2.4/mod/core.html#allowoverride)
答案 5 :(得分:0)
完成操作后,请不要忘记更改锚标签, 家
示例:
import glob
for i, a_file in enumerate(glob.glob('./A*')):
A = cv2.imread(a_file,0)
B = cv2.imread(a_file.replace('A', 'B'),0)
image = cv2.add([A,B])
filename = ('Merged' + {i} + '.jpeg')
cv2.imwrite(filename, image)
答案 6 :(得分:0)
以防万一您仍然想知道如何(如果存在目录(对于核心框架文件夹和文件))将所有请求重定向到框架索引处理程序,只是在发现一些错误/成功尝试之后,我只需要在其中更改RewriteCond .htaccess文件
RewriteCond %{REQUEST_FILENAME} !-f
RewriteCond %{REQUEST_FILENAME} !-d
上述条件指出“未找到文件”和“未找到目录”,好吧,如果只删除“未找到”(!-d)行,并以如下所示结束,那该怎么办:
RewriteEngine on
RewriteBase /framework/
RewriteCond %{REQUEST_FILENAME} !-f
RewriteRule ^(.*)$ /framework/index.php [L,QSA]
对我来说就像魅力一样
答案 7 :(得分:0)
我的 Laravel 7 项目在 Debian 10 共享主机中遇到了同样的问题。我必须将 RewriteBase /
添加到 /public/ 目录中的 .htaccess 中。所以 .htaccess 看起来像
RewriteEngine On
RewriteBase /
RewriteCond %{REQUEST_FILENAME} !-f
RewriteCond %{REQUEST_FILENAME} !-d
RewriteRule ^.*$ /index.php [L,QSA]