想在Nginx上开发具有这样功能的一个转发模块,外部转发策略控制服务器将一些指定的URL发送给数据库,Nginx读取数据库中的URL列表,将列表指定的这些URL转发到特定的缓存代理服务器上,其他非数据库URL列表中的URL请求直接通过Nginx转发到出口网关上。实际上以上的功能就是很多网站利用Nginx做负载均衡时的实现的七层转发功能,不太一样的是,我想加一个外部的转发策略控制服务器将一些实时的URL列表发送给Nginx,这样如果这些URL信息是根据大数据处理结果统计出的最高热点访问URL,就可以在Nginx上实现基于内容热度的七层转发。

  以上是后续要完成的目标,看了看网上的资料,目前有这方面的类似模块,就是Nginx+HttpLuaModule+Redis的实现方式,Nginx不用多说是服务器,HttpLuaModule是由淘宝的工程师清无(王晓哲)和春来(章亦春)所开发的nginx第三方模块,它能将lua语言嵌入到nginx配置中,从而使用lua就极大增强了nginx的能力。Redis是一个开源的使用ANSI C语言编写、支持网络、可基于内存亦可持久化的日志型、Key-Value数据库,并提供多种语言的API,说简单了就是内存数据库,效率高。这三个模块组合在一起的基本流程是url请求nginx服务器,然后lua查询redis,返回数据,这种方式依然保持高并发。废话不多说了下面将配置方式,中间遇到了各种坑,不过幸好最后都成功解决。

  方式一:

  直接安装OpenResty,OpenResty是Nginx打包装好了各种模块,包括Lua等等。具体可以看官方主页:

  主页:http://openresty.org/

  但是我没有选择这种傻瓜,没有困难制造困难也要上,下面是手动配置Nginx+Lua+Redis步骤。

  方式二:

  1.下载Nginx源码

  http://nginx.org/en/download.html

  下载和解压都在目录 /home/zjf/ 下进行的

  cd /home/zjf

  tar -zxvf nginx-1.6.2 

  解压后出现以下目录

  /home/zjf/nginx-1.6.2

  先不编译Nginx,接着下载其他需要的模块

  2.配置HttpLuaModule模块

  2.1   下载并编译安装LuaJIT 2.0 

             HttpLuaModule模块需要LuaJIT 2.0(推荐LuaJIT-2.1)或者Lua 5.1(Lua 5.2不支持)

      下载链接:http://luajit.org/download.html

      下载文件:LuaJIT-2.0.3.tar.gz

     解压到:/home/zjf/luajit-2.0.3/

     编译并安装: cd /home/zjf/luajit-2.0.3/

            make 

            sudo make install

     2.2    下载ngx_devel_kit

     下载链接:https://github.com/simpl/ngx_devel_kit/tags

     下载文件:ngx_devel_kit-0.2.19.tar.gz

     解压到:/home/zjf/ngx_devel_kit-0.2.19

  2.3    下载ngx_lua

           下载链接:https://github.com/openresty/lua-nginx-module/tags

      下载文件: lua-nginx-module-0.9.13rc1.tar.gz

      解压到:/home/zjf/lua-nginx-module-0.9.13rc1

      2.4    将以上模块和Nginx一起编译 

      第一次编译的时候出现了error:

      error while loading shared libraries: libluajit-5.1.so.2: cannot open shared object file: No such file or directory  

        参考了的解决方法,我用的是在编译前导入环境变量的方法

            export LD_LIBRARY_PATH=/usr/local/lib/:$LD_LIBRARY_PATH 

      或者在编译时加入选项 --with-ld-opt="-Wl,-rpath,$LUAJIT_LIB" 如下

# tell nginx's build system where to find LuaJIT 2.0:
    export LUAJIT_LIB=/usr/local/include/luajit-2.0/
    export LUAJIT_INC=/usr/local/lib/

  #configure Nginx
   ./configure --with-ld-opt="-Wl,-rpath,$LUAJIT_LIB" --prefix=/usr/local/nginx/ --add-module=/home/zjf/ngx_devel_kit-0.2.19/ --add-module=/home/zjf/lua-nginx-module-0.9.13rc1/

    make -j2
    make install

   2.5  测试Nginx+HttpLuaModule模块

     经过以上的步骤Nginx的HttpLuaModule模块就配置完成了,在浏览器输入localhost地址会出现Nginx服务器的欢迎界面,表示Nginx正常工作,如果不能正常工作,考虑是不是有其他已安装的服务器占用了端口。

     编辑/usr/local/nginx/conf/nginx.conf 文件看到server{...}内会有如下默认配置,这个是欢迎网页的url配置。

location / {
               root   html;
               index  index.html index.htm;
          }

             我们再加入如下配置:

location /lua {  
              default_type 'text/plain';  
              content_by_lua 'ngx.say("hello, lua")';  
      }

     然后访问localhose/lua,返回字符hello,lua,则证明Lua模块配置成功,Nginx成功的使用Lua输出了字符串hello,lua。

    3.安装Redis数据库

     redis安装很简单,下载-解压-安装即可,下载地址 http://www.redis.io/download

     tar -zxvf redis-2.8.6.tar.gz

     cd /home/zjf/redis-2.8.6

     make 

              sudo make install

    4.安装lua-resty-redis-master

     lua-resty-redis-masters是Lua client driver for the ngx_lua based on the cosocket API,也就是Nginx通过Lua操作Redis数据库的驱动模块。

     下载地址:https://github.com/openresty/lua-resty-redis

     解压后的目录为:/home/zjf/lua-resty-redis-master

     配置nginx.conf文件

     在http{...}段内加入以下路径,让nginx可以根据刚才的解压目录找到redis.lua模块驱动文件,目录不同可以根据情况修改,只要找到redis.lua即可。

lua_package_path "/home/zjf/lua-resty-redis-master/lib/resty/redis.lua;;";

     然后加入一个新的location,用/test的URL来测试该模块是否配置成功。

location /test {
        default_type 'text/plain';
            content_by_lua '
                local redis = require "resty.redis"
                local red = redis:new()

                red:set_timeout(1000) -- 1 sec

                -- or connect to a unix domain socket file listened
                -- by a redis server:
                --     local ok, err = red:connect("unix:/path/to/redis.sock")

                local ok, err = red:connect("127.0.0.1", 6379)
                if not ok then
                    ngx.say("failed to connect: ", err)
                    return
                end

                ok, err = red:set("dog", "an animal")
                if not ok then
                    ngx.say("failed to set dog: ", err)
                    return
                end

                ngx.say("set result: ", ok)

                local res, err = red:get("dog")
                if not res then
                    ngx.say("failed to get dog: ", err)
                    return
                end

                if res == ngx.null then
                    ngx.say("dog not found.")
                    return
                end

                ngx.say("dog: ", res)

                red:init_pipeline()
                red:set("cat", "Marry")
                red:set("horse", "Bob")
                red:get("cat")
                red:get("horse")
                local results, err = red:commit_pipeline()
                if not results then
                    ngx.say("failed to commit the pipelined requests: ", err)
                    return
                end

                for i, res in ipairs(results) do
                    if type(res) == "table" then
                        if not res[1] then
                            ngx.say("failed to run command ", i, ": ", res[2])
                        else
                            -- process the table value
                        end
                    else
                        -- process the scalar value
                    end
                end

                -- put it into the connection pool of size 100,
                -- with 10 seconds max idle time
                local ok, err = red:set_keepalive(10000, 100)
                if not ok then
                    ngx.say("failed to set keepalive: ", err)
                    return
                end

                -- or just close the connection right away:
                -- local ok, err = red:close()
                -- if not ok then
                --     ngx.say("failed to close: ", err)
                --     return
                -- end
            ';
        }

      访问localhost/test若配置成功,则显示

set result: OK
dog: an animal

      注意,要启动Redis数据库服务,否则后显示failed to connect : connection refused

      5.经过以上步骤,Nginx+Lua+Redis的所有配置就完成了,实现了Nginx根据Url查询Redis返回相应内容的功能,可以进行相应的二次开发。