軟件的增量更新

軟件的增量更新

因爲轉戰C#了,之前很多東西都丟了。現在從頭開始弄基礎服務,首先第一個就是客戶端的自動更新。之前簡單搜了一下相關功能的實現。有一個文章我沒有看懂,另一片文章裏邊說的應該是提交本地數據,然後計算差異化包,讓服務器返回差異化數據包。當然這樣不是不行。肯定是可行的,但是對於服務器來說這部分工作可能就有點麻煩了。因爲你得讓服務器有這個計算能力。參考Cocos2dx 3.9的Lua增量更新模塊,簡單做了一個基礎框架模型出來。

原來怎麼做

覆蓋安裝

這個其實很簡單,就是重新下載一個完整的安裝包,然後重新安裝一遍,不管原來存不存在內容,如果原來存在內容那麼久替換掉,如果原來不存在內容那麼就添加上新內容就是了。其實這個說起來很簡單,但是可能會存在一些問題。

- 流量問題 可能現在看來這個問題並不是多麼大的問題,因爲現在帶寬已經非常寬了。100M的內容按照10Mpbs的帶寬來算,也就一分多鐘就能下載完了。

- 渣子問題 這種覆蓋安裝一般會存在一個渣子的問題。比如說,我在安裝目錄裏邊生成了一個不在後續安裝包的文件,那麼這個文件就沒有辦法被清理掉。這就可能很尷尬了,比如說你的項目依賴系統提供的一個Dll,如果你的目錄中直接存在這個Dll那麼就會優先使用你對應目錄中的Dll(如果我沒記錯應該是這樣),如果是我作爲攻擊者的話,我很有可能會給你放一個我種下病毒的Dll。這就很尷尬了

打補丁

這個原理也比較簡單,其實就是我們都覺得完全安裝太費勁了,那麼我的軟件又需要比較頻繁的更新,比如說某些桌遊可能過個節日要上節日相關的功能,這樣就可以添加新的Dll然後又不能出一個一個的完整安裝包。那麼我可以在完整安裝包的基礎上打補丁嘛。比如說,我出了版本1.0,過了十天半個月過端午了,我出個龍舟皮膚一類的,那我就可以直接在1.0的基礎上打個龍舟補丁,這樣他就變成了最新的客戶端1.1。如果將來要上別的功能了我就在1.1的基礎上打個補丁,讓客戶端變成1.2。不過這樣也會有他的問題。

- 順序安裝 在安裝的過程中只能以此遞增式安裝,我只能1.0 => 1.1 => 1.2;不能1.0 =》 1.2。因爲中間是沒有對應的補丁的。

- 流量問題 其實這種解決方案可能會帶來一些問題,比如說,現在端午節,我需要把房子裝飾成龍舟的樣式;然後五一勞動節,我又需要把房子裝修成五一勞動節的樣子。那麼都是關於房子的皮膚,我是沒有辦法都保留的,因爲來年的時候肯定就不能這麼裝修了,因爲過時了太Low了。那麼關於這部分的內容,如果你想一點一點的升級上來對於最後的版本來說是沒用的,你佔用的流量一點用都沒有。太尷尬了。

- 維護的複雜度 因爲你不能直接出了一個1.0之後全都是使用補丁,如果當你的版本號遞增到一定程度以後,補丁的大小可能遠遠超過了你重新去下載一個最新的客戶端的大小。所以只能通過時間也好(比如半年或者一個季度)通過意義(比如說大版本號 2.0 3.0)來生成一個完整的客戶端。這樣用戶在下載的時候就可以找一個最近的完整的客戶端版本號。然後再打補丁的方式來獲得最新的客戶端,不過這種維護的複雜度應該也不小。

理理我們最核心的需求吧

其實我們的需求很簡單,獲取最新的客戶端。然後附加要求就是要省流量、下載方便、服務端發佈方便。

省流量

其實說到省流量,就是能用本地的就直接使用本地。本地實在是沒有的文件,那麼就從網絡上下載。這樣基本上就做到了省流量的效果。

下載方便

不需要做太多的操作,當然這個很多軟件都做到了這一點。上文中提到的其實也可以做到自動化,比如說,完整安裝的那麼我就直接下載最新的完整安裝包就好了,如果是打補丁的這種,那麼就下載最新的完整安裝包以及後邊的補丁就好了。其實這個真的要做,對用戶來說應該是沒有感覺得。都一樣,不過對於程序員來說。可能面臨的開發就不太一樣了。

服務端發佈方便

其實這個完全是針對於程序員的了,一般來說,如果這個事情可以程序來自動完成那麼就肯定交給程序了。比如說完整晚裝包的這種,肯定能夠做到自動打包。打補丁的這種,無非也就是根據上一個版本生成一個補丁。或者再生成一個完整安裝包。上傳到合適的文件服務器就好了。其實打補丁也好,完整安裝包也好,都有一個顯著的優勢就是可以很方便的放到多個服務器上來進行文件的負載均衡。

那麼我們該怎麼做

在考慮這個問題的時候,我想到了之前接觸的Cocos2dx 3.9 Lua 自動更新模塊,他是這麼做,通過一個配置文件,來說明最新的客戶端中都包含了那些文件,這些文件的MD5值是什麼,然後網絡路徑是什麼。這樣客戶端拿到這個配置清單的時候,就可以輕鬆的判斷本地的那些文件是可以繼續用的。那些文件是過時了的,這樣客戶端通過配置心中的網絡路徑位置獲取最新的對應文件就好了嘛。不過那也是很久之前的事情了,不然,我就不需要自己重新規劃了。直接抄一份代碼就好了嘛。還是自己整理一套吧。這樣來的更徹底一些,想改什麼就改什麼。

文件的格式

{
    "VersionsCheckCode": "XC09VU4QCRD43LRF01BYOD26D45DWEEKX5KECUKIA7Q4160FKAWQBHXTKE63Z148",
    "TimeStamp": 1496649771,
    "ServerUrl": "http://or2dwwrsz.bkt.clouddn.com",
    "FileInfos": [
        {
            "FilePath": "JumpKick.HttpLib\\packages\\Moq.4.2.1409.1722\\lib\\net40\\Moq.xml",
            "FileMD5": "c7e9c70a19b84f31e51eb65f4ee38803",
            "FileUrl": "LV4ZBB_c7e9c70a19b84f31e51eb65f4ee38803"
        },
        {
            "FilePath": "JumpKick.HttpLib\\packages\\Moq.4.2.1409.1722\\lib\\sl4\\Moq.Silverlight.dll",
            "FileMD5": "0ee20e7ccba7d6667c48efebe41503ff",
            "FileUrl": "X057QT_0ee20e7ccba7d6667c48efebe41503ff"
        },
        {
            "FilePath": "JumpKick.HttpLib\\packages\\Moq.4.2.1409.1722\\lib\\sl4\\Moq.Silverlight.xml",
            "FileMD5": "c25417228db2dd820f45e93112e8596c",
            "FileUrl": "S0LO6G_c25417228db2dd820f45e93112e8596c"
        }
    ]
}

簡單的說明

  • VersionsCheckCode:當前的版本文件校驗信息。
  • TimeStamp:做這個文件的的時間戳
  • ServerUrl:服務器的地址,主要是用來跟後續的文件進行拼接來用的
  • FileInfos:對應的文件信息列表
  • FileInfos[?]:FilePath:本地文件路徑,從網絡下載之後對應的本地地址
  • FileInfos[?]:FileMD5:這個文件的MD5值,用來判斷原始的對應位置的文件是否與網絡中的文件相同
  • FileInfos[?]:FileUrl:這個文件在網絡中存在的位置,當然這個是沒有前邊的URL路徑的是ServerUrl後邊的內容

項目地址

其實嘛整個項目最複雜的地方時這個更新的想法與這個文件的制定。剩下的內容其實就比較簡單了,就是具體的代碼的實現了。代碼方便我就懶得講了,直接把項目的地址扔上來了事。

客戶端

http://git.oschina.net/anxin1225/autoupdateclient

服務端

http://git.oschina.net/anxin1225/autoupdateserver

這樣做的好處

省流量、跟其他軟件結合方便、服務器發佈方便。省流量這個上邊提到了我就說了。

跟其他軟件結合方便

其實很容易理解,就是這個軟件跟被更新的軟件一毛錢關係沒有。所以我可以直接跑起來就行了,不需要關係具體被更新的軟件是怎麼搞得。最多采用這個的項目。重新改一下我們這邊的UI就行了。

服務器發佈方便

其實最麻煩的事情就是服務器這邊。需要生成這個配置文件,我這邊服務器端其實並沒有在運行指的就是生成這個文件的工具。我可以指定一個目錄。然後生成這個文件,將對應目錄的所有文件導出到一個輸出目錄。不過對於很多CDN不支持多級目錄(比如七牛),所以我將所有的文件都換掉了名字,讓他們儘量的不重複,程序可讀就行了。

怎麼用

首先使用我寫好的服務端生成對應的配置文件和改名文件。

Markdown

生成的目錄結構是這樣色的,配置文件放到一個固定的目錄裏邊去。UpLoad文件夾上傳到某一個文件服務器上,這裏我是用的七牛雲

Markdown

然後把UpLoad目錄中的文件全都上傳上來

Markdown

上傳完了就是這個樣子的。

Markdown

致辭服務器就部署好了,等有了新版本重複一遍這個操作就行。其實上傳服務器的這部分工作可以集成到服務端中。上傳內容就好了嘛,其實很簡單的。當然了,這個我懶。之前也沒有研究七牛的SDK這個可以作爲一個功能上的擴展,反正項目我已經開源了,感興趣的人可以自己擴展這部分功能。好吧我們繼續來說客戶端怎麼弄吧。

怎麼用客戶端呢

            AutoUpdateHelper helper = new AutoUpdateHelper();
            helper.WebXmlUrl = "http://7xs9hw.com1.z0.glb.clouddn.com/VersionInfo.json";
            helper.ConfigXmlPath = "SynchronizeVersions.xml";
            helper.TempXmlPath = "SynchronizeVersions_Temp.xml";
            helper.FilePath = "Client";

            helper.CallBack = obj =>
            {
                if (obj is Dictionary<UpdateDataType, object>)
                {
                    var dic = obj as Dictionary<UpdateDataType, object>;
                    foreach (var item in dic)
                    {
                        if (Name2Action.ContainsKey(item.Key))
                            Name2Action[item.Key](item.Value);
                    }
                }
            };

            try
            {
                helper.Start();
            }
            catch (Exception ex)
            {
                MessageBox.Show(ex.Message);
            }

其實就是一個簡單的設置網址跟配置文件。其實呢這個地方應該吧配置也放到配置文件裏邊去,爲什麼沒放呢?因爲我懶,哈哈哈。

客戶端的運行圖

Markdown

客戶端跑完了就關了,其實應該是跑完了運行某一個具體的文件,然後自動更新的邏輯就完成了,這部分我會之後繼續完善。

不妥之處

這麼樣處理其實下載會變得很靈活。但是也會帶來其他的問題。比如之前提到的打包的問題。因爲服務器只是一個文件服務器,所以服務器並沒有計算出差異包的能力,所以所有的文件都是一個一個的下載的,這樣就會出現很多小文件的下載。這樣的下載其實是比較蛋疼的。這是設計上的坑。爲了靈活只能妥協了。

實現的不妥之處

其實到目前爲止我只是實現了最基礎的功能。甚至還不全,比如之後的文件啓動,不過大體的框架已經搭建起來了。至於後邊有很多實現不是很合理的地方,我先簡單列一列,方便之後維護。

  • 下載失敗重試不存在
  • 下載數量的限制不存在(現在是有多少下載多少,很多同時下載可能會存在超時的問題。)
  • 完成了之後沒有啓動對應的文件
  • 現在沒有快速啓動的功能(現在每次啓動都需要重新校驗所有文件,其實可以避免這個問題的。)

後話

雖然這個項目只是一個並不完善的框架。但是這種更新方式應該會讓更新變得更有意思。讓我們一塊來完善這個框架吧。比較希望做一個成功的開源項目,不忘初心。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章