之前写过Nginx重定向了,http://homeway.me/2014/10/28/nginx-reverse-proxy-conf/,但当时比较模糊。
这里主要说两种常用的重定向,都是php中的重定向。
一种是typecho的带index.php例如http://homeway.me/index.php/arg1/arg2,另一种是隐藏index.php的http://homeway.me/arg1/arg2。
以下配置代码均亲测可用。
首先还是按常理,先脑补下nginx地rewrite规则,http://nginx.org/en/docs/http/ngx_http_rewrite_module.html
关于nginx重写的指令主要由这么一些:
break指令
停止执行当前虚拟主机的后续rewrite指令集
if指令
对给定的条件condition进行判断。如果为真,大括号内的rewrite指令将被执行。
有几个要记住的操作符:
使用=,!= 比较的一个变量和字符串
是用~,~*与正则表达式匹配的变量,如果这个正则表达式中包含},;则整个表达式需要用” 或’ 包围
使用-f,!-f 检查一个文件是否存在
使用-d,!-d 检查一个目录是否存在
使用-e,!-e 检查一个文件、目录、符号链接是否存在
使用-x,!-x 检查一个文件是否可执行
详细中文看这里好了: http://www.nginx.cn/216.html
首先要明白我们现在要做的事情是两类,/index.php/arg1/arg2和/arg1/arg2
/index.php/arg1/arg2跑的location是index.php文件,也就是说,我们要做一个location匹配.php的正则,并且要让它分辨出uri中的/arg1/arg2
这个正则有很多种写法,我用的是lnmp传统的写法~ [^/]\.php(/|$)。
完整匹配如下:
location ~ [^/]\.php(/|$) { #try_files $uri =404; 住址扫描目录用的,这里我们都是虚假目录,删除。 fastcgi_split_path_info ^(.+\.php)(/.+)$; fastcgi_pass unix:/tmp/php-cgi.sock; fastcgi_index index.php; include fastcgi.conf; #fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name; #include fastcgi_params; }
我测了下,这样其实就搞定了,typecho重定向就好了,不要向网上说的3个if语句。
只要访问/index.php/arg1/arg2就能访问到内容了。
这里有几个问题:
去查找官网文档吧, http://nginx.org/en/docs/http/ngx_http_fastcgi_module.html#fastcgi_split_path_info
Defines a regular expression that captures a value for the $fastcgi_path_info variable. The regular expression should have two captures: the first becomes a value of the $fastcgi_script_name variable, the second becomes a value of the $fastcgi_path_info variable.
也就是说,fastcgi_split_path_info的作用就是把参数分割成$fastcgi_script_name和$fastcgi_path_info,分割方式是后面的正则表达式。
我用echo模块输出了这里的参数,得到下面的结果,上面的是不加fastcgi_split_path_info,下面是加了fastcgi_split_path_info的结果:
这里用到的这两个配置文件是fastcgi的配置文件,我查看了下,发现fastcgi.conf和fastcgi_params的差别就在fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;这句话,也就是说,随意选一个。
fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name; fastcgi_param QUERY_STRING $query_string; fastcgi_param REQUEST_METHOD $request_method; fastcgi_param CONTENT_TYPE $content_type; fastcgi_param CONTENT_LENGTH $content_length; fastcgi_param SCRIPT_NAME $fastcgi_script_name; fastcgi_param REQUEST_URI $request_uri; fastcgi_param DOCUMENT_URI $document_uri; fastcgi_param DOCUMENT_ROOT $document_root; fastcgi_param SERVER_PROTOCOL $server_protocol; fastcgi_param HTTPS $https if_not_empty; fastcgi_param GATEWAY_INTERFACE CGI/1.1; fastcgi_param SERVER_SOFTWARE nginx/$nginx_version; fastcgi_param REMOTE_ADDR $remote_addr; fastcgi_param REMOTE_PORT $remote_port; fastcgi_param SERVER_ADDR $server_addr; fastcgi_param SERVER_PORT $server_port; fastcgi_param SERVER_NAME $server_name; fastcgi_param REDIRECT_STATUS 200;
fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;也就用到了之前的那个函数fastcgi_split_path_info,如果不做分割,就会回调: No input file specified,没有指定文件,就是因为$fastcgi_script_name的值找不到对应的文件。
其实,我在conf里面添加了echo输出,发现/index.php/arg1/arg2就只会访问.php那个location。
然后,/arg1/arg2就用到了网上广为流传的那个typecho配置了:
location / { index index.html index.php; if (-f $request_filename/index.html){ rewrite (.*) $1/index.html break; } if (-f $request_filename/index.php){ rewrite (.*) $1/index.php; } if (!-f $request_filename){ rewrite (.*) /index.php; } }
看看就明白,目录/arg1/arg2,不会有有index.php可以匹配.php后缀,所以只能匹配到/里面。
同样地,这里我用echo模块把参数输出了下:
测试配置文件如下:
location / { index index.html index.php; if (-f $request_filename/index.html){ rewrite (.*) $1/index.html break; } if (-f $request_filename/index.php){ echo "request_filename -f index.php = $request_filename"; echo "request_filename -f index.php= $request_filename"; echo "fastcgi_path_info -f index.php = $fastcgi_path_info"; #rewrite (.*) $1/index.php; } if (!-f $request_filename){ echo "request_filename !-f index.php = $request_filename"; echo "fastcgi_script_name !-f index.php = $fastcgi_script_name"; echo "fastcgi_path_info !-f index.php = $fastcgi_path_info"; #rewrite (.*) /index.php; } }
测试的结果如下:
也就是访问了第3个if语句,发现没有/ajax这个文件,就重定向到/index.php去了。
这里重定向到/index.php后,解析.php的fastcgi的规则用的又是上面用到的,原理也一样,把uri分割,然后匹配。
配置代码如下:
location ~ [^/]\.php(/|$) { #try_files $uri =404; 住址扫描目录用的,这里我们都是虚假目录,删除。 fastcgi_split_path_info ^(.+\.php)(/.+)$; fastcgi_pass unix:/tmp/php-cgi.sock; fastcgi_index index.php; include fastcgi.conf; #fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name; #include fastcgi_params; }
嗯…今天就算是,重新证明了下,网上流传的那种,3个if语句的typecho语句,其实并没有什么卵用。
最后进去的/index.php/2015/05/22/并不会去查询/的location匹配,而是.php的正则匹配。
nginx的模块测试都比较麻烦,都要重新编译安装,关于echo模块调试起来比较方便。
http://wiki.nginx.org/HttpEchoModule
下面是一些if判断用得到的nginx参数,可以通过echo输出测试查看:
代码我保存了一份,可以点击链接下载:http://homeway.me/code/nginx-rewrite-conf.zip