Android 借助 Python 实现自动打包上传 fir
Posted code小生
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Android 借助 Python 实现自动打包上传 fir相关的知识,希望对你有一定的参考价值。
作者 | wustor
声明 | 本文是 wustor 原创,已获授权发布,未经原作者允许请勿转载
概述
在开发的过程中,很多时候完成了一个功能的开发,往往需要打包给测试进行测试,之前就是打个包,要么是通过 USB 进行安装,要么就是打个包通过 QQ 给测试发送过去,后来接触到 Jenkins,发现可以进行持续集成,但是很多时候往往只是改了一个很小的功能,比如说字体,或者颜色之类的,Jenkins 就有点大材小用了,这个时候,总想着要是能够通过脚本进行自动打包上传至服务器并且生成一个下载的二维码就好了,最近学习了 Python 并且也接触了 fir 这个第三方托管工具,发现,梦想还是要有的,万一实现了呢
关于 Python 和 fir
关于Python
Python 是一门高级编程语言,而且是一门动态语言,可以用来编写各种脚本来帮助人们从一些重复性的操作中解放出来,当然也可以用来开发网站,不过实现 Python 的自动打包上传只需要准备:
了解基本的 Python 语法
熟悉 Requests 这个网络库
关于fir
fir 是一个第三方的托管网站, fir.im 为开发者提供测试应用极速发布,应用崩溃实时分析、用户反馈收集等一系列开发测试效率工具服务,所以需要准备的是
注册一个fir账号
了解fir对外提供的API
流程
注册fir账号
配置Python运行环境
编写Python脚本
名称 类型 标题 说明
type String 是 ios 或者 android(发布新应用时必填)
bundle_id String 是 App 的 bundleId(发布新应用时必填
api_token String 是 长度为 32, 用户在 fir 的 api_token
参数列表
名称 类型 标题 说明
type String 是 ios 或者 android(发布新应用时必填)
bundle_id String 是 App 的 bundleId(发布新应用时必填
api_token String 是 长度为 32, 用户在 fir 的 api_token
Postman调试
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
binary 字段对应的 binary
参数列表
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界面查看结果
界面显示已经上传成功,但是发现没有Logo,我开始以为他会自动提取apk中的logo,实际上并没有,但是它提供了上传logo的接口,现在来继续上传logo
上传应用图标
icon字段对应的upload_url
参数列表
Postman测试
fir查看上传结果
编写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
运行结果
{"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 = "版本更新日志"
//执行Python脚本
def process = "python upToFir.py ${upUrl} ${appName} ${bundleId} ${verName} ${apiToken} ${iconPath} ${apkPath} ${buildNumber} ${changeLog}".execute()
println("开始上传至fir")
//获取Python脚本日志,便于出错调试
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 "上传结束 "
}
}
该脚本放在 app/build.gradle 中的 android 目录下
统一Python脚本
# coding=utf-8
# encoding = utf-8
import requests
import sys
def upToFir():
# 打印传递过来的参数数组长度,便于校验
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 脚本的参数都是写死的,所以需要改变成动态从 gradle 中获取,获取的时候先判断一下数组长度,看看是不是跟之前约定的一样
整体进行测试
这个时候修改一下apk的一些参数,跟logo
versionCode 3
versionName "1.2"
iconPath=ic_launcher.png
appName="python"
执行gradle命令 gradlew debugToFir,运行结果
开始上传至fir
http://api.fir.im/apps
success_apk:{"is_completed":true}
success_icon:{"is_completed":true}
上传结束 with value 0
运行成功,到官网查看结果
完美,简单,以后简单的打包就用一行代码就可以搞定了,吼吼。
小结
其实 Python 的语法很简洁,作为一门动态语言,不需要像 Java 定义各种类型变量,gradle 的语法其实也一样,掌握这两种语言的基本用法,有助于更高效的开发 Android。
源码下载 https://github.com/wustor/PythoPackage
与之相关
1
2
3
以上是关于Android 借助 Python 实现自动打包上传 fir的主要内容,如果未能解决你的问题,请参考以下文章
Android QA专用,Python实现不一样的多渠道打包工具