本文已授權(quán)code小生獨家發(fā)布,轉(zhuǎn)載請注明出處http://m.itdecent.cn/p/0ec4c4e132e7
概述
在開發(fā)的過程中,很多時候完成了一個功能的開發(fā),往往需要打包給測試進行測試,之前就是打個包,要么是通過USB進行安裝,要么就是打個包通過QQ給測試發(fā)送過去,后來接觸到Jenkins,發(fā)現(xiàn)可以進行持續(xù)集成,但是很多時候往往只是改了一個很小的功能,比如說字體,或者顏色之類的,Jenkins就有點大材小用了,這個時候,總想著要是能夠通過腳本進行自動打包上傳至服務(wù)器并且生成一個下載的二維碼就好了,最近學習了Python并且也接觸了fir這個第三方托管工具,發(fā)現(xiàn),夢想還是要有的,萬一實現(xiàn)了呢?下面介紹一下如果利用Python跟fir這兩個工具來實現(xiàn)夢想的。
關(guān)于Python
Python是一門高級編程語言,而且是一門動態(tài)語言,可以用來編寫各種腳本來幫助人們從一些重復性的操作中解放出來,當然也可以用來開發(fā)網(wǎng)站,不過實現(xiàn)Python的自動打包上傳只需要準備:
- 了解基本的Python語法
- 熟悉Requests這個網(wǎng)絡(luò)庫
關(guān)于fir
fir是一個第三方的托管網(wǎng)站, fir.im 為開發(fā)者提供測試應(yīng)用極速發(fā)布,應(yīng)用崩潰實時分析、用戶反饋收集等一系列開發(fā)測試效率工具服務(wù),所以需要準備的是
- 注冊一個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的提示不至于在一些小問題上卡殼,當然如果你愿意去配置文本編輯器,我推薦SublimeText,提供了很多插件。
編寫Python腳本
獲取上傳地址
| 名稱 | 類型 | 標題 | 說明 |
|---|---|---|---|
| type | String | 是 | ios 或者 android(發(fā)布新應(yīng)用時必填) |
| bundle_id | String | 是 | App 的 bundleId(發(fā)布新應(yīng)用時必填 |
| api_token | String | 是 | 長度為 32, 用戶在 fir 的 api_token |
服務(wù)器地址:http://api.fir.im/apps()
參數(shù)列表
| 名稱 | 類型 | 標題 | 說明 |
|---|---|---|---|
| type | String | 是 | ios 或者 android(發(fā)布新應(yīng)用時必填) |
| bundle_id | String | 是 | App 的 bundleId(發(fā)布新應(yīng)用時必填 |
| api_token | String | 是 | 長度為 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.fir.im/apps', data=data)
print req.content
運行測試
{
"id": "5a059de3959d6961bb000257",
"type": "android",
"short": "asxn",
"cert": {
"icon": {
"key": "5cc5942ccb1b7b86bd39c0f3ad84ea0c3e93a5e7",
"token": "太長,以文字代替",
"upload_url": "https://upload.qbox.me"
},
"binary": {
"key": "63b159e5456d6151ace59ed7322d6942b05a4c6e.apk",
"token": "太長,以文字代替",
"upload_url": "https://upload.qbox.me"
},
"mqc": {
"total": 5,
"used": 0,
"is_mqc_availabled": true
},
"support": "qiniu",
"prefix": "x:"
}
}
上傳apk
服務(wù)器地址:upload_url
binary字段對應(yīng)的binary
| 名稱 | 類型 | 標題 | 說明 |
|---|---|---|---|
| key | String | 是 | binary字段對應(yīng)的key |
| token | String | 是 | binary字段對應(yīng)的token |
| file | File | 是 | 安裝包文件 |
| x:name | String | 是 | 應(yīng)用名稱 |
| x:version | String | 是 | 版本號 |
| x:build | String | 是 | Build 號 |
| x:changelog | String | 是 | 長度為 32, 用戶在 fir 的 api_token |
參數(shù)列表
| 名稱 | 類型 | 標題 | 說明 |
|---|---|---|---|
| key | String | 是 | binary字段對應(yīng)的key |
| token | String | 是 | binary字段對應(yīng)的token |
| file | File | 是 | 安裝包文件 |
| x:name | String | 是 | 應(yīng)用名稱 |
| x:version | String | 是 | 版本號 |
| x:build | String | 是 | Build 號 |
| x:changelog | String | 是 | 長度為 32, 用戶在 fir 的 api_token |
Postman進行測試

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.qbox.me', 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)用圖標
服務(wù)器地址:upload_url
icon字段對應(yīng)的upload_url
參數(shù)列表
| 名稱 | 類型 | 標題 | 說明 |
|---|---|---|---|
| key | String | 是 | binary字段對應(yīng)的 key |
| token | String | 是 | binary字段對應(yīng)的 token |
| file | File | 是 | icon |
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.qbox.me', 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.fir.im/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ù)組長度,看看是不是跟之前約定的一樣
整體進行測試
這個時候修改一下apk的一些參數(shù),跟logo
versionCode 3
versionName "1.2"
iconPath=ic_launcher.png
appName="python"
執(zhí)行g(shù)radle命令 gradlew debugToFir,運行結(jié)果
開始上傳至fir
http://api.fir.im/apps
success_apk:{"is_completed":true}
success_icon:{"is_completed":true}
上傳結(jié)束 with value 0
運行成功,到官網(wǎng)查看結(jié)果

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