說說 nginx 剛發(fā)布的 JavaScript 能力 nginScript
背景
2015年9月,nginx宣布支持類JavaScript語言。這意味著開發(fā)者可以更輕松、自由的控制全球***秀的HTTP及反向代理服務(wù)器,并在此之上可以衍生出更多有用、好玩的創(chuàng)意。Nginx也更開發(fā)的走向了動態(tài)配置化的下一個階段。大家可以點擊查看 官方介紹鏈接。
先簡單說說nginx
Nginx [engine x]是全球***,也是***秀的web服務(wù)器、反向代理服務(wù)器。通過第三方公司的統(tǒng)計,目前全球至少有23%的服務(wù)器采用了nginx,當(dāng)然這個數(shù)字還在不斷的擴(kuò)大。目前也是國內(nèi)BAT***,所以這也是為什么我們***時間關(guān)注到它的原因。 Nginx主要可以做以下幾點:
-
1、工作在TCP第七層,可以對HTTP協(xié)議的所有內(nèi)容進(jìn)行分析和處理。
-
2、支持lua,perl,JavaScript動態(tài)語言
-
3、支持第三方插件
再說說nginScript
1、nginScript是JavaScript/ECMAscript的子集
。它實現(xiàn)了大部分的JavaScript語言的能力,沒有完全遵從ECMAScript標(biāo)準(zhǔn),同時拋棄了JavaScript比較難懂的部分。
2、nginScript不是通過V8引擎實現(xiàn)的
。而是通過一個更小、能耗更低、更符合nginx應(yīng)用場景的小虛擬機(jī)(VM)來實現(xiàn)??梢岳斫鉃閚ginx為其實現(xiàn)了一套自己的詞法解析。
3、nginScript是跑在nginx的配置文件里
。 比如:nginx.conf文件里。所以nginScript可以完成傳統(tǒng)配置文件所能處理的所有事情,同時可以讓配置管理動態(tài)化。這也是nginScript出現(xiàn)的最重要的原因。
4、nginScript
是以nginx插件的方式存在。 插件名叫:njs 。和其他nginx插件一樣,我們需要重新編譯nginx來完成安裝。
5、nginScript目前是早期研發(fā)狀態(tài)
。大家可以通過郵件nginx-devel@nginx.org等方式和nginx團(tuán)隊進(jìn)行溝通和提出你的訴求。
如何安裝nginScript
這里直接按照官方給出的步驟來就好:
// 1、下載***nginx包,地址可見: http://nginx.org/en/download.html wget http://nginx.org/download/nginx-1.9.4.tar.gz //2、 解壓 tar -xzvf nginx-1.9.4.tar.gz //3、通過mercurial獲取nginScript模塊,這里如果沒有安裝mercurial,需要先運(yùn)行 yum install mercurial hg clone http://hg.nginx.org/njs
//4、編譯nginx,這里只具體了njs模塊,其他需要的模塊自己要記得一起裝哦。如果你沒編譯過nginx,有些依賴模塊需要yum安裝,請自行搜索。 cd nginx-1.9.4 ./configure --add-module=../njs/nginx --prefix=/usr/local make make install ok,這就安裝完了,我們可以開始玩啦。
具體如何使用nginScript
nginScript的使用主要是在nginx的配置體系里增加了2個指令。具體指令分別為:
-
js_set
,設(shè)置配置里的變量值 -
js_run
,直接執(zhí)行配置規(guī)則
1、先看看js_set
在nginx.conf里怎么運(yùn)行的。
http {
js_set $msg"
var str = 'hello,imweb';
// JavaScript str;
";
server {
...
location /{
return 200 $msg;
}
}
}
結(jié)果:
上面例子里,可以看出,我們可以通過JS隨意地給nginx設(shè)置變量值。而這些變量是可以用在nginx配置的各個地方。比如:proxy_pass,limit_req_zone, and sub_filter。這里相對之前配置已經(jīng)大大的提高了靈活性。
2、js_run
的運(yùn)行規(guī)則和場景
-
js_run
是運(yùn)行在location指令里,匹配指定location的路徑就會執(zhí)行對應(yīng)的JavaScript -
js_run
是直接通過JavaScript來產(chǎn)生HTTP返回的內(nèi)容
下面舉個具體的例子:
location /imwebteam {
js_run "
var res;
res = $r.response;
res.status = 200;
res.send('hello,imweb!');
res.finish();
";
}
這個結(jié)果和***個結(jié)果是一樣的。這里就不贅述。
3、處理兩個指令以外,還有個重要的變量$r
通過js_set
和js_run
可以對HTTP request請求有完整的控制權(quán),控制的方式就是變量$r
的使用。 $r
里有什么可以通過以下簡單例子看到。
http { js_set $summary " var a, s, h; s = 'JS summary\n\n'; s += 'Method: ' + $r.method + '\n'; s += 'HTTP version: ' + $r.httpVersion + '\n'; s += 'Host: ' + $r.headers.host + '\n'; s += 'Remote Address: ' + $r.remoteAddress + '\n'; s += 'URI: ' + $r.uri + '\n'; s += 'Headers:\n'; for (h in $r.headers) { s += ' header \"' + h + '\" is \"' + $r.headers[h] + '\"\n'; } s += 'Args:\n'; for (a in $r.args) { s += ' arg \"' + a + '\" is \"' + $r.args[a] + '\"\n'; } s; "; server { listen 8000; location /imwebteam{ return 200 $summary; } }
結(jié)果如圖:
nginScript目前還存在的問題
經(jīng)過上面的介紹,相信大家對nginScript已經(jīng)有了基本的認(rèn)識。那么我們在看看這個新生兒有哪些問題吧。
- 首先,調(diào)試方法弱。目前還是比較原始,通過log的方式來展示,且錯誤日志的詳細(xì)程度很不如人意。
- 其次,控制力度弱。目前nginScript的處理力度還僅限于http request的處理和返回response的層面,還無法做到動態(tài)處理nginx請求之外的一些內(nèi)容,比如動態(tài)用戶數(shù)據(jù)或轉(zhuǎn)發(fā)配置表動態(tài)更新等。
- ***,整體實現(xiàn)弱。整體結(jié)構(gòu)還是比較簡單,js_run和js_set的運(yùn)行環(huán)境是不太一致的,js_set執(zhí)行ok的代碼段在js_run上會出現(xiàn)一些異常。
綜合來說,nginScript還是一個愿望和前景很美好的新生兒。需要一定時間打磨和優(yōu)化。也希望大家多多的提供意見和反饋,甚至是提交自己的插件。從而使得它有更好的成長。
對于我們的實踐場景
這個之前和黎小騰君,donald討論過的2個主要場景,realLog系統(tǒng)和nohost2.0系統(tǒng)。 nginScript對這里兩個場景來講都無疑是很大的好消息,這里在規(guī)則響應(yīng)上,在現(xiàn)有的體系下就可以有很靈活的處理方法。 但在用戶配置動態(tài)加載方面,我們?nèi)孕枰ㄟ^其他方式來實現(xiàn),這個部分我們先提issue給到nginx開發(fā)團(tuán)隊,看下具體情況再和大家進(jìn)一步討論和同步。