繼上次搗鼓出了飛秋的群聊協議后,鑒于年底沒啥事情做,就用lua寫了個簡單的協議兼容的IM。本來開始讓
另一個同事在iptux的基礎上修改的,結果大概是因為iptux的代碼不是那么容易修改,就不了了之了。這個
剛發布的luafeiq功能非常簡單,僅支持與飛秋(包括大部分兼容IP messager的IM)進行單聊,群消息的
收發,簡易的消息盒子(暫存未讀消息)。因為選的庫都是跨平臺的,所以很容易的luafeiq也是跨平臺的,
最主要的是我想在linux下使用。
之所以選用lua,一方面是想練練lua,另一方面則是因為開發效率。前段時間在android下寫了些java代碼,
用java寫代碼覺得甚為爽快(當然算不了完美)。這幾天寫了千把行的lua(也許有3K行,未統計過),
感覺也不錯。綜合來說,這些高級語言的很多好用的語法特性,例如閉包(closure),垃圾回收,都提高
了不少寫代碼的速度。當然,lua于我而言也算不上完美的語言。例如我經常因為變量敲錯字母,而在運行時
才暴露nil錯誤。這也許可以通過諸如IDE之類的工具在寫代碼的時候就給予提示。lua 在遇到一個符號時,
默認地將其處理為全局的。關于這個語法特性早有人提出不爽,只能說大家設計的準則不一樣。(在我們
項目里,我直接改寫了全局變量的metatable,從而防止策劃隨意定義全局變量)
再來談談實現過程中的一些瑣事。因為飛秋也算是IP messager協議的兼容實現,很多通信除了可以使用
抓包軟件分析外,還可以直接通過IP messager的源碼來了解。所以,基礎通信協議的實現過程也比較
簡單。飛秋與飛秋之間發送私聊消息是經過加密的。其加密過程也不簡單,更重要的是,我并不想浪費太
多時間在這上面。后來發現其實可以通過上線消息里某個標志位表明自己不需要加密。這個標志就是消息頭
里的option。上線廣播出去的消息里一旦表明自己不加密,那么以后和飛秋通信也就不需要解密了。
發送私聊消息時,消息里會攜帶一個消息ID。這個ID可以通過任意算法生成,例如直接取time的值。接收到
對方的消息時,需要取出該ID,然后加入回應消息。對方收到回應消息后,就知道自己發送成功。這個過程
算是ip messager在UDP上做的消息可靠驗證,過程也比較簡單。
群聊消息在之前提到過,是通過UDP多播實現。我們可以接收所有群的消息。如果之前已經處于某個群里,
那么一旦你上線后(廣播上線消息),你就可以直接在這個群里發言。但如果你之前不在這個群里,則
可以通過多播一個加入群的消息,然后就可以不請自來地在這個群里發言。詳細的消息值和實現都可以從
luafeiq的代碼里讀到(message_sender.lua)。
在linux下接收windows上的飛秋消息,是需要做字符編碼轉換的。因為luafeiq使用IUP作為UI庫,IUP在
linux下使用GTK作為底層實現,默認全部是UTF8編碼。luafeiq里我自己寫了個lua庫,用于編碼轉換。
話說IUP作為一個UI庫,還是比較不錯的。正如其介紹文檔里所說,學習曲線低,基本上看一會文檔,就可以
直接使用了。luafeiq使用的IUP版本至少需要3.0以上。當初在linux下為了安裝IUP3.3,基本花了4個小時
時間,各種奇怪的沒多大意義的錯誤信息。后來換成3.2版本,居然一下子就和諧了,無限怨念。
luafeiq目前放在googlecode的版本,可以說是一個很不負責任的版本。早上我才剛把字符編碼轉換的代碼
調試好。今天已經請假,家里就一臺電腦,也就測試不了這個字符編碼轉換是否真的能正常工作。我在
windows下dump了些字符,看上去能正常功能。明天得回老家過春節,上不了網,索性就提前發布了。
luafeiq項目地址:http://code.google.com/p/luafeiq/