在開發(fā)的過程中,很多時候完成了一個功能的開發(fā),往往需要打包給測試進(jìn)行測試,之前就是打個包,要么是通過 USB 進(jìn)行安裝,要么就是打個包通過 QQ 給測試發(fā)送過去,后來接觸到 Jenkins,發(fā)現(xiàn)可以進(jìn)行持續(xù)集成,但是很多時候往往只是改了一個很小的功能,比如說字體,或者顏色之類的,Jenkins 就有點大材小用了,這個時候,總想著要是能夠通過腳本進(jìn)行自動打包上傳至服務(wù)器并且生成一個下載的二維碼就好了,最近學(xué)習(xí)了 Python 并且也接觸了 fir 這個第三方托管工具,發(fā)現(xiàn),夢想還是要有的,萬一實現(xiàn)了呢 關(guān)于Python Python 是一門高級編程語言,而且是一門動態(tài)語言,可以用來編寫各種腳本來幫助人們從一些重復(fù)性的操作中解放出來,當(dāng)然也可以用來開發(fā)網(wǎng)站,不過實現(xiàn) Python 的自動打包上傳只需要準(zhǔn)備: 了解基本的 Python 語法 熟悉 Requests 這個網(wǎng)絡(luò)庫 關(guān)于fir fir 是一個第三方的托管網(wǎng)站, 為開發(fā)者提供測試應(yīng)用極速發(fā)布,應(yīng)用崩潰實時分析、用戶反饋收集等一系列開發(fā)測試效率工具服務(wù),所以需要準(zhǔn)備的是 注冊一個fir賬號 了解fir對外提供的API 注冊fir賬號 fir的注冊地址是fir注冊地址,不過免費的應(yīng)用每天提供的免費下載次數(shù)是100次 配置Python運行環(huán)境 Python 現(xiàn)在大致分為兩個大的版本:2.X以及3.X,不過 Python 的3.X版本有些語法是不向下兼容的,由于對 Python 不是很熟悉,所以還是選擇了2.7版本來配置環(huán)境,官網(wǎng)下載地址是 Python 官網(wǎng),我是Windows 系統(tǒng),所以下載的安裝包,然后下載了一個編輯 Python 的IDE 名字是 PyCharm,之所以選擇 IDE 沒有用 SublimeText 等文本編輯器是因為 windows 下的環(huán)境配置比較麻煩,而且剛開始有 IDE 的提示不至于在一些小問題上卡殼,當(dāng)然如果你愿意去配置文本編輯器,我推薦 SublimeText,提供了很多插件。 編寫Python腳本 獲取上傳地址 名稱類型標(biāo)題說明 typeString是ios 或者 android(發(fā)布新應(yīng)用時必填) bundle_idString是App 的 bundleId(發(fā)布新應(yīng)用時必填 api_tokenString是長度為 32, 用戶在 fir 的 api_token
服務(wù)器地址:http://api./apps() 參數(shù)列表 名稱類型標(biāo)題說明 typeString是ios 或者 android(發(fā)布新應(yīng)用時必填) bundle_idString是App 的 bundleId(發(fā)布新應(yīng)用時必填 api_tokenString是長度為 32, 用戶在 fir 的 api_token
Postman調(diào)試 
Python腳本編寫 import requests
data = {'type': 'android', 'bundle_id': 'com.wustor.pythopackage', 'api_token': '9812fa28e4dac156673a5e45e7119631'} req = requests.post(url='http://api./apps', data=data) print req.content
運行測試 { "id": "5a059de3959d6961bb000257", "type": "android", "short": "asxn", "cert": { "icon": { "key": "5cc5942ccb1b7b86bd39c0f3ad84ea0c3e93a5e7", "token": "太長,以文字代替", "upload_url": "https://upload." }, "binary": { "key": "63b159e5456d6151ace59ed7322d6942b05a4c6e.apk", "token": "太長,以文字代替", "upload_url": "https://upload." }, "mqc": { "total": 5, "used": 0, "is_mqc_availabled": true }, "support": "qiniu", "prefix": "x:" } }
上傳apk服務(wù)器地址:upload_url binary 字段對應(yīng)的 binary

參數(shù)列表 
Postman進(jìn)行測試 
Python腳本編寫 # coding=utf-8 import requests try: print("上傳apk") apk_path = 'F:/PythonDemo/Demo/app-release.apk' file = {'file': open(apk_path, 'rb')} param = {"key": '61a53809c7b58d8b68e537c3d4831b01325b1f0b.apk', "token": '你自己的token', "x:name": '測試', "x:version": '1.0', "x:build": '1', "x:changelog": '暫無更新'} req = requests.post('https://upload.', files=file, data=param, verify=False) print 'success:' + req.content except Exception as e: print'error:' + e
運行測試
{"is_completed":true}
在fir界面查看結(jié)果 
界面顯示已經(jīng)上傳成功,但是發(fā)現(xiàn)沒有Logo,我開始以為他會自動提取apk中的logo,實際上并沒有,但是它提供了上傳logo的接口,現(xiàn)在來繼續(xù)上傳logo 上傳應(yīng)用圖標(biāo) 服務(wù)器地址:upload_url icon字段對應(yīng)的upload_url
參數(shù)列表 
Postman測試 
fir查看上傳結(jié)果 
這里用了一張微信朋友圈的logo上傳,已經(jīng)成功替換。
編寫Python腳本 # coding=utf-8 import requests
try: print("上傳icon") icon_path = 'F:/PythonDemo/Demo/demo.png' file = {'file': open(icon_path, 'rb')} param = {"key": 'd1bca0636623f17782d9f851aa9e08c77f875a62', 'token': '替換成你自己的token' } req = requests.post('https://upload.', files=file, data=param, verify=False) print 'success:' + req.content except Exception as e: print'error:' + e
運行結(jié)果 {"is_completed":true} 編寫gradle腳本 task debugToFir { dependsOn 'assembleDebug' doLast { def upUrl = "http://api./apps" def appName = "Python2" def bundleId = project.android.defaultConfig.applicationId def verName = project.android.defaultConfig.versionName def apiToken = "9812fa28e4dac156673a5e45e7119631" def iconPath = "F:/PythoPackage/app/src/main/res/mipmap-xxhdpi/ic_launcher.png" def apkPath = "F:/PythoPackage/app/build/outputs/apk/debug/app-debug.apk" def buildNumber = project.android.defaultConfig.versionCode def changeLog = "版本更新日志" //執(zhí)行Python腳本 def process = "python upToFir.py ${upUrl} ${appName} ${bundleId} ${verName} ${apiToken} ${iconPath} ${apkPath} ${buildNumber} ${changeLog}".execute() println("開始上傳至fir") //獲取Python腳本日志,便于出錯調(diào)試 ByteArrayOutputStream result = new ByteArrayOutputStream() def inputStream = process.getInputStream() byte[] buffer = new byte[1024] int length while ((length = inputStream.read(buffer)) != -1) { result.write(buffer, 0, length) } println(result.toString("UTF-8")) println "上傳結(jié)束 " } }
該腳本放在 app/build.gradle 中的 android 目錄下
統(tǒng)一Python腳本 # coding=utf-8 # encoding = utf-8 import requests import sys def upToFir(): # 打印傳遞過來的參數(shù)數(shù)組長度,便于校驗 print 'the argLength--->:' + len(sys.argv) upUrl = sys.argv[1] appName = sys.argv[2] bundleId = sys.argv[3] verName = sys.argv[4] apiToken = sys.argv[5] iconPath = sys.argv[6] apkPath = sys.argv[7] buildNumber = sys.argv[8] changeLog = sys.argv[9] queryData = {'type': 'android', 'bundle_id': bundleId, 'api_token': apiToken} iconDict = {} binaryDict = {} # 獲取上傳信息 try: response = requests.post(url=upUrl, data=queryData) json = response.json() iconDict = (json["cert"]["icon"]) binaryDict = (json["cert"]["binary"]) except Exception as e: print('query:' + e)
# 上傳apk try: file = {'file': open(apkPath, 'rb')} param = {"key": binaryDict['key'], 'token': binaryDict['token'], "x:name": appName, "x:version": verName, "x:build": buildNumber, "x:changelog": changeLog} req = requests.post(url=binaryDict['upload_url'], files=file, data=param, verify=False) print 'success_apk:' + req.content except Exception as e: print'error_apk:' + e
# 上傳logo try: file = {'file': open(iconPath, 'rb')} param = {"key": iconDict['key'], 'token': iconDict['token']} req = requests.post(url=iconDict['upload_url'], files=file, data=param, verify=False) print 'success_icon:' + req.content except Exception as e: print'error_icon:' + e
if __name__ == '__main__': upToFir()
前面的三個 python 腳本的參數(shù)都是寫死的,所以需要改變成動態(tài)從 gradle 中獲取,獲取的時候先判斷一下數(shù)組長度,看看是不是跟之前約定的一樣
整體進(jìn)行測試 這個時候修改一下apk的一些參數(shù),跟logo versionCode 3 versionName "1.2" iconPath=ic_launcher.png appName="python"
執(zhí)行g(shù)radle命令 gradlew debugToFir,運行結(jié)果 開始上傳至fir http://api./apps success_apk:{"is_completed":true} success_icon:{"is_completed":true} 上傳結(jié)束 with value 0
運行成功,到官網(wǎng)查看結(jié)果

完美,簡單,以后簡單的打包就用一行代碼就可以搞定了,吼吼。 其實 Python 的語法很簡潔,作為一門動態(tài)語言,不需要像 Java 定義各種類型變量,gradle 的語法其實也一樣,掌握這兩種語言的基本用法,有助于更高效的開發(fā) Android。 源碼下載 https://github.com/wustor/PythoPackage
|