python 游戲自動更新

本腳本用于游戲服務(wù)器端的自動更新,主要流程就是把更新包和腳本的配置文件放在一個中心端center,分支機(jī)房client去中心端center下載更新包,并驗(yàn)證MD5坑资,分支機(jī)房的游戲服務(wù)器根據(jù)自身需求通過rsync下載相應(yīng)的更新包,游戲服務(wù)器通過解壓朋譬、拷貝盐茎、驗(yàn)證部分重要文件的MD5來確保更新正確完成兴垦,目前此腳本已經(jīng)用于更新本公司多款游戲徙赢,兼容windows、linux系統(tǒng)(根據(jù)自身需要更改相關(guān)路徑)探越,可以用頁面去調(diào)用狡赐,已經(jīng)寫了post頁面返回值。

#!/usr/bin/env python  
# -*- coding: utf-8 -*-  
#Used update game path to server  
import os,re,sys,urllib,urllib2,hashlib,time,shutil,platform  
 
def post(status, type, info, err_info=""):  
    post_url = args["post_url"]  
    aid = args["aid"]  
    data = {"aid": aid, "status": status, "type": type, "info": info}  
    if err_info:  
        data = {"aid": aid, "status": status, "type": type, "info": info, "err_info": err_info}  
    print data  
    f = urllib2.urlopen(url=post_url, data=urllib.urlencode(data))  
 
def md5sum(file_name):  
    if os.path.isfile(file_name):  
        f = open(file_name,'rb')  
        py_ver = sys.version[:3]  
        if py_ver == "2.4":  
            import md5 as hashlib  
        else:  
            import hashlib  
            md5 = hashlib.md5(f.read()).hexdigest()  
            f.close()  
            return md5  
    else:  
        return 0 
 
def config(args,files):  
    try:  
        game = args["main_prefix"]  
        url = "http://208.asktao.com/autoupdate/%s/config.ini" % game  
        get = urllib.urlopen(url)  
        aa = get.readlines()  
        w = {}  
        for i in aa:  
            a = i.strip().split()[0]  
            if "[" in a:  
                x = a.strip("[]")  
                w[x] = {}  
                continue 
            w[x][i.strip().split()[1].strip()] = i.strip().split()[0].strip()  
        for key in w:  
            if key == files:  
                return w[key]  
    except Exception,e:  
        return 0 
 
class down_start():  
 
    def work(self,args):  
        aa = config(args,"path_md5")  
        game = args["main_prefix"]  
        if aa == 0:  
            post(2,"read update config","Not find %s config file" % game)  
            sys.exit()  
        local = '/data/autoupdate/' 
        url = "http://208.asktao.com/autoupdate/%s/" % game  
        for f in aa:  
            md5_r = f.strip().split()[0]  
            pkg_name = f.strip().split()[1]  
            get = urllib.urlopen(os.path.join(url,pkg_name))  
            status = get.getcode()  
            if status == 200:#驗(yàn)證MD5钦幔,MD5錯誤的話重新下載一次枕屉,再次錯誤就提示失敗,退出程序  
                urllib.urlretrieve(os.path.join(url,pkg_name),os.path.join(local,pkg_name),)  
                md5_l = md5sum(os.path.join(local,pkg_name))  
                if md5_l == md5_r:  
                    post(1,"down",pkg_name)  
                else:  
                    urllib.urlretrieve(os.path.join(url,pkg_name),os.path.join(local,pkg_name),)  
                    md5_l = md5sum(os.path.join(local,pkg_name))  
                    if md5_l == md5_r:  
                        post(1,"down",pkg_name)  
                    else:  
                        post(2,"down",pkg_name,"Download %s,MD5 not right" % pkg_name)  
                        sys.exit()  
            else:  
                post(2,"down",pkg_name,"Not find %s path file" % pkg_name)  
        # 檢查rsync服務(wù)是否啟動  
        while True:  
            pid = os.popen("ps auxww | grep 'rsync --daemon' | grep -v grep").read()  
            if not pid:  
                getso("/usr/bin/rsync --daemon")  
                continue 
            break 
        #全部完成鲤氢,更新中心狀態(tài)  
        post(1,"down","down_done")  
 
class sync_start():  
 
    def rsync(self,update_pkg,args):  
        node_ip = args["node_ip"]  
        game = args["main_prefix"]  
        os_type = platform.system()  
        if os_type == "Windows":  
            local = "C:\\update\\" 
        elif os_type == "Linux":  
            local = "/home/update/tmp/" 
        if not os.path.isdir(local):  
            os.mkdir(local)  
        for f in os.listdir(local):  
            if os.path.isfile(f):  
                os.remove(os.path.join(local,f))  
            elif os.path.isdir(f):  
                shutil.rmtree(os.path.join(local,f))  
        aa = config(args,"path")  
        if aa == 0:  
            post(2,"read update config","Not find %s config file" % game)  
            sys.exit()  
        for f in aa:  
            if update_pkg in f:  
                md5_r = aa[f]  
                pkg_name = f  
                if os_type == "Windows":  
                    sync = "rsync -avz %s::update/*%s* /cygdrive/c/update/" % (node_ip,update_pkg)  
                elif os_type == "Linux":  
                    sync = "rsync -avz %s::update/*%s* /home/update/tmp/" % (node_ip,update_pkg)  
                result = os.popen(sync).readlines()  
                md5_l = md5sum(os.path.join(local,pkg_name))  
                if md5_l == md5_r:  
                    post(1,"sync",pkg_name)  
                else:  
                    if os_type == "Windows":  
                        sync = "rsync -avz %s::update/*%s* /cygdrive/c/update/" % (node_ip,update_pkg)  
                    elif os_type == "Linux":  
                        sync = "rsync -avz %s::update/*%s* /home/update/tmp/" % (node_ip,update_pkg)  
                    result = os.popen(sync).readlines()  
                    md5_l = md5sum(os.path.join(local,pkg_name))  
                    if md5_l == md5_r:  
                        post(1,"sync",pkg_name)  
                    else:  
                        post(2,"rsync",pkg_name,"sync Error,%s not find or file's md5 wrong" % pkg_name)  
                        sys.exit()  
 
    def work(self,args):  
        update_pkg = args["update_pkg"].split(',')  
        for i in range(len(update_pkg)):  
            self.rsync(update_pkg[i],args)  
        post(1,"sync","sync_done")  
 
class update_start():  
 
    def copy(self,src, dst):  
        if os.path.isdir(src):  
            base = os.path.basename(src)  
            if os.path.exists(dst):  
                dst = os.path.join(dst, base)  
            if not os.path.exists(dst):  
                os.makedirs(dst)  
            names = os.listdir(src)  
            for name in names:  
                srcname = os.path.join(src, name)  
                self.copy(srcname, dst)  
        else:  
            shutil.copy2(src, dst)  
 
    def unrar(self,src,dst,args):  
        os_type = platform.system()  
        try:  
            if not os.path.exists(dst) or not os.path.exists(src):  
                raise Exception, "%s or %s not exist!" % (src, dst)  
            if os_type == "Windows":  
                os.system(r'C:\Progra~1\WinRAR\rar x -o+ -inul %s %s' % (src, dst))  
            elif os_type == "Linux":  
                if os.path.splitext(src)[1] == ".tgz":  
                    os.system("tar -zxf %s -C %s" % (src, dst))  
                elif os.path.splitext(src)[1] == ".zip":  
                    os.system("unzip -oq %s -d %s" % (src, dst))  
            return 0 
        except Exception,e:  
            return e  
 
    def work(self,args):  
        os_type = platform.system()  
        if os_type == "Windows":  
            update = "C:\\update\\" 
        elif os_type == "Linux":  
            update = "/home/update/tmp/" 
        server_dir = {"tmcs":"C:\\Server\\","wd":"/home/asktao/"}  
        server = server_dir[args["main_prefix"]]  
        update_pkg = args["update_pkg"].split(',')  
        game = args["main_prefix"]  
        for i in range(len(update_pkg)):   
#####################解壓更新包  
            for tgz in os.listdir(update):  
                if update_pkg[i] in tgz:  
                    src = os.path.join(update,tgz)  
                    r = self.unrar(src,update,args)  
                    if r == 0:  
                        post(1,"update","%s unzip success" % src)  
                    else:  
                        post(2,"update",src,"unzip fail:%s" % r)  
#####################拷貝更新文件到游戲目錄  
            ser_list = os.listdir(server)  
            up_list = os.listdir(update)  
            for dir in up_list:  
                for line in ser_list:  
                    filepath = os.path.join(update,dir+"\\")  
                    serv = os.path.join(server,line+"\\")  
                    if dir in line:  
                        self.copy(filepath,serv)  
                        post(1,"update","%s files copy success" % line)  
#####################驗(yàn)證重要文件MD5  
        aa = config(args,"files")  
        if aa == 0:  
            post(2,"read update config","Not find %s config file" % game)  
            sys.exit()  
        for f in aa:  
            md5_r = aa[f]  
            pkg_name = f  
            if os.path.exists(pkg_name):  
                md5_s = md5sum(pkg_name)  
                if md5_r == md5_s:  
                    post(1,"update","%s md5 Ok,update success" % pkg_name)  
                else:  
                    post(2,"update","%s md5 Error,update fail" % pkg_name)  
                    sys.exit()  
        shutil.rmtree(update)  
        os.mkdir(update)  
        post(1,"update","All_done")  
 
if __name__ == "__main__":  
    args = {"pack":"auto_update","func1":"down_start","url":"http://208.2222.com/manage/auto_update","post_url":"http://192.168.50.209/reg.php","update_pkg":"up,up","node_ip":"192.168.50.208","main_prefix":"tmcs","aid":"123"}  
    #down = down_start()  
    #down.work(args)  
    sync = sync_start()  
    sync.work(args)  
    update = update_start()  
    update.work(args)  
 
 
'''''config.ini內(nèi)容  
[files]#需要驗(yàn)證MD5的重要文件絕對路徑和MD5  
15a6605156e29f68fdfd637e73a889d4  C:\Server\Line1\SAFlashPlayer.exe  
15a6605156e29f68fdfd637e73a889d4  C:\Server\Line2\SAFlashPlayer.exe  
[path]#更新包名字和MD5  
32fcb8932799aa553db13b5b9b41e5e9  auto.rar  
32fcb8932799aa553db13b5b9b41e5e9  update.rar  
''' 
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末搀擂,一起剝皮案震驚了整個濱河市西潘,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌哨颂,老刑警劉巖喷市,帶你破解...
    沈念sama閱讀 218,941評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異威恼,居然都是意外死亡品姓,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,397評論 3 395
  • 文/潘曉璐 我一進(jìn)店門箫措,熙熙樓的掌柜王于貴愁眉苦臉地迎上來腹备,“玉大人,你說我怎么就攤上這事斤蔓≈菜郑” “怎么了?”我有些...
    開封第一講書人閱讀 165,345評論 0 356
  • 文/不壞的土叔 我叫張陵弦牡,是天一觀的道長惧互。 經(jīng)常有香客問我,道長喇伯,這世上最難降的妖魔是什么喊儡? 我笑而不...
    開封第一講書人閱讀 58,851評論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮稻据,結(jié)果婚禮上艾猜,老公的妹妹穿的比我還像新娘。我一直安慰自己捻悯,他們只是感情好匆赃,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,868評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著今缚,像睡著了一般算柳。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上姓言,一...
    開封第一講書人閱讀 51,688評論 1 305
  • 那天瞬项,我揣著相機(jī)與錄音,去河邊找鬼何荚。 笑死囱淋,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的餐塘。 我是一名探鬼主播妥衣,決...
    沈念sama閱讀 40,414評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了税手?” 一聲冷哼從身側(cè)響起蜂筹,我...
    開封第一講書人閱讀 39,319評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎芦倒,沒想到半個月后狂票,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,775評論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡熙暴,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,945評論 3 336
  • 正文 我和宋清朗相戀三年闺属,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片周霉。...
    茶點(diǎn)故事閱讀 40,096評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡掂器,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出俱箱,到底是詐尸還是另有隱情国瓮,我是刑警寧澤,帶...
    沈念sama閱讀 35,789評論 5 346
  • 正文 年R本政府宣布狞谱,位于F島的核電站乃摹,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏跟衅。R本人自食惡果不足惜孵睬,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,437評論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望伶跷。 院中可真熱鬧掰读,春花似錦、人聲如沸叭莫。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,993評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽雇初。三九已至拢肆,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間靖诗,已是汗流浹背郭怪。 一陣腳步聲響...
    開封第一講書人閱讀 33,107評論 1 271
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留呻畸,地道東北人移盆。 一個月前我還...
    沈念sama閱讀 48,308評論 3 372
  • 正文 我出身青樓,卻偏偏與公主長得像伤为,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,037評論 2 355

推薦閱讀更多精彩內(nèi)容

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理绞愚,服務(wù)發(fā)現(xiàn)叙甸,斷路器,智...
    卡卡羅2017閱讀 134,661評論 18 139
  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,167評論 25 707
  • 國家電網(wǎng)公司企業(yè)標(biāo)準(zhǔn)(Q/GDW)- 面向?qū)ο蟮挠秒娦畔?shù)據(jù)交換協(xié)議 - 報(bào)批稿:20170802 前言: 排版 ...
    庭說閱讀 10,986評論 6 13
  • 周末閑來無事位衩,偶翻相冊裆蒸。那些年的江南游又歷歷可見。 江南好糖驴,風(fēng)景舊曾諳僚祷;日出江花紅勝火,春來江水綠如藍(lán)贮缕。能不憶江南...
    jiangming888閱讀 514評論 24 20
  • 年輕時(shí)看《紅樓夢》辙谜,很討厭里面有個人物王熙鳳,感覺她陰險(xiǎn)感昼、狡詐和狠毒装哆,把一個無知的‘賈瑞’幾下就活活整死了,又利用...
    半畝方塘yx閱讀 592評論 5 5