Rails2.3开始支持Rack,今天尝试了一下将原先基于Controller/Action做ajax校验的部分代码迁移到Rack,改动还是蛮简单的,假设原先的请求如下:
/ajax_validator/check_login?value=quake
考虑到兼容,改写成通过PATH_INFO和QUERY_STRING拿数据,代码如下:
在本机上测试一下,因为少去routes和controller/filter等开销,性能有提高。兴冲冲地放到了服务器上去,发现出奇怪问题了,一直报404错误,经过debug,原因是PATH_INFO和QUERY_STRING都为空,服务器和本机区别是部署不一样,服务器部署在Lighttpd+FastCgi,google一下,发现Lighttpd在使用FastCgi将应用映射到根(/)应用的时候,有这个bug。不过Lighttpd有另外一个参数REQUEST_URI完整地记录了请求,作为零时解决方案,可以写个Rack小应用修复一下:
然后在部署环境的配置文件中,让它加入到Failsafe之前,统一处理一下:
/ajax_validator/check_login?value=quake
考虑到兼容,改写成通过PATH_INFO和QUERY_STRING拿数据,代码如下:
class AjaxValidator
def self.call(env)
if env["PATH_INFO"] =~ /^\/ajax_validator/
value = Rack::Utils.unescape(env["QUERY_STRING"].split("=").last || '')
if value != ''
type = env["PATH_INFO"].split("/").last
case type
when 'check_login'
User.find_by_login(value)
#...
end
end
end
end
end
在本机上测试一下,因为少去routes和controller/filter等开销,性能有提高。兴冲冲地放到了服务器上去,发现出奇怪问题了,一直报404错误,经过debug,原因是PATH_INFO和QUERY_STRING都为空,服务器和本机区别是部署不一样,服务器部署在Lighttpd+FastCgi,google一下,发现Lighttpd在使用FastCgi将应用映射到根(/)应用的时候,有这个bug。不过Lighttpd有另外一个参数REQUEST_URI完整地记录了请求,作为零时解决方案,可以写个Rack小应用修复一下:
class LighttpdFcgiFix
def initialize(app)
@app = app
end
def call(env)
env["PATH_INFO"], env["QUERY_STRING"] = env["REQUEST_URI"].split(/\?/, 2)
@app.call(env)
end
end
然后在部署环境的配置文件中,让它加入到Failsafe之前,统一处理一下:
require 'lighttpd_fcgi_fix'
config.middleware.insert_before ActionController::Failsafe, LighttpdFcgiFix