• Nginx自定义模块编写:根据post参数路由到不同服务器


    Nginx自定义模块编写:根据post参数路由到不同服务器

    2014-05-05 15:27 blogread IT技术博客 字号:T | T

    Nginx可以轻松实现根据不同的url 或者 get参数来转发到不同的服务器,然而当我们需要根据http包体来进行请求路由时,Nginx默认的配置规则就捉襟见肘了,但是没关系,Nginx提供了强大的自定义模块功能,我们只要进行需要的扩展就行了。

    我们来理一下思路,我们的需求是:

    Nginx根据http包体的参数,来选择合适的路由

    在这之前,我们先来考虑另一个问题:

    在Nginx默认配置的支持下,能否实现服务器间的跳转呢?即类似于状态机,从一个服务器执行OK后,跳转到另一台服务器,按照规则依次传递下去。

    答案是可以的,这也是我之前写bayonet之后,在nginx上特意尝试的功能。

    一个示例的配置如下:

    1. server { 
    2.     listen       8080; 
    3.     server_name  localhost; 
    4.     location / { 
    5.         proxy_pass http://localhost:8888; 
    6.         error_page 433 = @433; 
    7.         error_page 434 = @434; 
    8.     } 
    9.     location @433 { 
    10.         proxy_pass http://localhost:6788; 
    11.     } 
    12.     location @434 { 
    13.         proxy_pass http://localhost:6789; 
    14.     } 
    15.     error_page   500 502 503 504  /50x.html; 
    16.     location = /50x.html { 
    17.         root   html; 
    18.     } 

    看明白了吧?我们使用了 433和434 这两个非标准http协议的返回码,所有请求进入时都默认进入 http://localhost:8888;,然后再根据返回码是 433 还是 434 来选择进入 http://localhost:6788 还是 http://localhost:6789。

    OK,也许你已经猜到我将这个例子的用意了,是的,我们只要在我们的自定义模块中,根据http的包体返回不同的返回码,进而 proxy_pass 到不同的后端服务器即可。

    好吧,接下来,我们正式进入nginx自定义模块的编写中来。

    一. nginx 自定义模块编写 由于这也是我第一次写nginx模块,所以也是参考了非常多文档,我一一列在这里,所以详细的入门就不说了,只说比较不太一样的地方。 参考链接:

    1. nginx的helloworld模块的helloworld
    2. nginx 一个例子模块,简单的将http请求的内容返输出
    3. nginx 自定义协议 扩展模块开发
    4. Emiller的Nginx模块开发指南

    而我们这个模块一个最大的特点就是,需要等包体整个接收完才能进行处理,所以有如下代码:

    1. void ngx_http_foo_post_handler(ngx_http_request_t *r){ 
    2.     // 请求全部读完后从这里入口, 可以产生响应 
    3.     ngx_http_request_body_t* rrb = r->request_body; 
    4.   
    5.     char* body = NULL
    6.     int body_size = 0
    7.   
    8.     if (rb && rb->buf) 
    9.     { 
    10.         body = (char*)rb->buf->pos; 
    11.         body_size = rb->buf->last - rb->buf->pos; 
    12.     } 
    13.   
    14.     int result = get_route_id(r->connection->log,  
    15.                               (int)r->method, 
    16.                               (char*)r->uri.data, 
    17.                               (char*)r->args.data, 
    18.                               body, 
    19.                               body_size 
    20.                               ); 
    21.     if (result < 0
    22.     { 
    23.         ngx_log_error(NGX_LOG_ERR, r->connection->log, 0, "get_route_id fail, result:%d", result); 
    24.         result = DFT_ROUTE_ID
    25.     } 
    26.     ngx_http_finalize_request(r, result); 
    27.   
    28. static ngx_int_t ngx_http_req_route_handler(ngx_http_request_t *r) 
    29.     ngx_http_read_client_request_body(r, ngx_http_foo_post_handler); 
    30.     return NGX_DONE; // 主handler结束 

    我们注册了一个回调函数 ngx_http_foo_post_handler,当包体全部接受完成时就会调用。之后我们调用了get_route_id来获取返回码,然后通过 ngx_http_finalize_request(r, result); 来告诉nginx处理的结果。

    这里有个小插曲,即get_route_id。我们来看一下它定义的原型:

    1. extern int get_route_id(ngx_log_t *log, int method, char* uri, char* args, char* body, int body_size) 

    第一个参数是 ngx_log_t *log,是为了方便在报错的时候打印日志。然而在最开始的时候,get_route_id 的原型是这样:

    1. extern int get_route_id(ngx_http_request_t *r, int method, char* uri, char* args, char* body, int body_size); 

    结果在 get_route_id 函数内部,调用:

    1. r->connection->log 

    的结果总是null,至今也不知道为什么。

    OK,接下来我们只要在get_route_id中增加逻辑代码,读几行配置,判断一下就可以了~ 但是,我想要的远不止如此。

    二、lua解析器的加入

    老博友应该都看过我之前写的一篇博客: 代码即数据,数据即代码(1)-把难以变更的代码变成易于变更的数据,而这一次的需求也非常符合使用脚本的原则:

    只需要告诉我返回nginx哪个返回码,具体怎么算出来的,再复杂,再多变,都放到脚本里面去。

    所以接下来我又写了c调用lua的代码:

    1. int get_route_id(ngx_log_t *log, int method, char* uri, char* args, char* body, int body_size) 
    2.     const char lua_funcname[] = "get_route_id"; 
    3.     lua_State *L = luaL_newstate(); 
    4.     luaL_openlibs(L); 
    5.     if (luaL_loadfile(L, LUA_FILENAME) || lua_pcall(L, 0, 0, 0)) 
    6.     { 
    7.         ngx_log_error(NGX_LOG_ERR, log, 0, "cannot run configuration file: %s", lua_tostring(L, -1)); 
    8.         lua_close(L); 
    9.         return -1; 
    10.     }  
    11.     lua_getglobal(L, lua_funcname); /* function to be called */ 
    12.     lua_pushnumber(L, method); 
    13.     lua_pushstring(L, uri); 
    14.     lua_pushstring(L, args); 
    15.     lua_pushlstring(L, body, body_size); 
    16.     /* do the call (1 arguments, 1 result) */ 
    17.     if (lua_pcall(L, 4, 1, 0) != 0) 
    18.     { 
    19.         ngx_log_error(NGX_LOG_ERR, log, 0, "error running function %s: %s", lua_funcname, lua_tostring(L, -1)); 
    20.         lua_close(L); 
    21.         return -2; 
    22.     } 
    23.     /* retrieve result */ 
    24.     if (!lua_isnumber(L, -1)) 
    25.     { 
    26.         ngx_log_error(NGX_LOG_ERR, log, 0, "function %s must return a number", lua_funcname); 
    27.         lua_close(L); 
    28.         return -3; 
    29.     } 
    30.     int result = (int)lua_tonumber(L, -1); 
    31.   
    32.     lua_pop(L, 1); /* pop returned value */ 
    33.   
    34.     lua_close(L); 
    35.     return result; 

    比较郁闷的是,lua 5.2的很多函数都变了,比如lua_open废弃,变成luaL_newstate等,不过总体来说还算没浪费太多时间。

    接下来是req_route.lua的内容,我只截取入口函数如下:

    1. function get_route_id(method, uri, args, body) 
    2.     loc, pf ,appid = get_need_vals(method, uri, args, body) 
    3.     if loc == nil or pf == nil or appid == nil then 
    4.         return OUT_CODE 
    5.     end 
    6.     --到这里位置,就把所有的数据都拿到了 
    7.     --print (loc, pf, appid) 
    8.     -- 找是否在对应的url, loc中 
    9.     if not is_match_pf_and_loc(pf, loc) then 
    10.         return OUT_CODE 
    11.     end 
    12.     -- 找是否在对应的appid中 
    13.     if not is_match_appid(appid) then 
    14.         return OUT_CODE 
    15.     end 
    16.     return IN_CODE 
    17. end 

    OK,结合了lua解析器之后,无论多复杂的调整,我们都基本可以做到只修改lua脚本而不需要重新修改、编译nginx模块代码了。

    接下来,就该是体验我们的成果了。

    三、Nginx配置

    1. server { 
    2.     listen       8080; 
    3.     server_name  localhost; 
    4.   
    5.     location /req_route { 
    6.         req_route; 
    7.         error_page 433 = @433; 
    8.         error_page 434 = @434; 
    9.     } 
    10.     location @433 { 
    11.         proxy_pass http://localhost:6788; 
    12.     } 
    13.     location @434 { 
    14.         proxy_pass http://localhost:6789; 
    15.     } 
    16.     error_page   500 502 503 504  /50x.html; 
    17.     location = /50x.html { 
    18.         root   html; 
    19.     } 

    OK,enjoy it!

    最后,放出代码如下:

    https://vimercode.googlecode.com/svn/trunk/nginx_req_route


     




  • 相关阅读:
    ADF_Tutorials系列17_ADF Faces_使用布局组件
    ADF_Tutorials系列17_ADF Faces_开发下拉和删除
    ADF_Tutorials系列17_ADF Faces_ADF预定义组件的创建和使用
    ADF_ADF Faces系列6_ADF数据可视化组件简介之建立Thematic Map Component
    ADF_ADF Faces系列5_ADF数据可视化组件简介之建立GeographicMap/Pivot Table/Gantt Chart
    ADF_ADF Faces系列4_ADF数据可视化组件简介之建立BarChart/Gauge/ExportExcel
    ADF_ADF Faces系列3_ADF数据可视化组件简介之建立Master-Detail
    ADF_ADF Faces系列2_使用JSF开发基于Ajax的用户界面:ADF Faces富客户端组件简介(Part2)
    ADF_ADF Faces系列1_使用JSF开发基于Ajax的用户界面:ADF Faces 富客户端组件简介(Part1)
    Form_Form Builder国际化多语言开发(案例)
  • 原文地址:https://www.cnblogs.com/aserlinux/p/3709714.html
Copyright © 2020-2023  润新知