本文已授权code小生独家发布,转载请注明出处http://www.jianshu.com/p/0ec4c4e132e7
概述
在开发的过程中,很多时候完成了一个功能的开发,往往需要打包给测试进行测试,之前就是打个包,要么是通过USB进行安装,要么就是打个包通过QQ给测试发送过去,后来接触到Jenkins,发现可以进行持续集成,但是很多时候往往只是改了一个很小的功能,比如说字体,或者颜色之类的,Jenkins就有点大材小用了,这个时候,总想着要是能够通过脚本进行自动打包上传至服务器并且生成一个下载的二维码就好了,最近学习了Python并且也接触了fir这个第三方托管工具,发现,梦想还是要有的,万一实现了呢?下面介绍一下如果利用Python跟fir这两个工具来实现梦想的。
关于Python
Python是一门高级编程语言,而且是一门动态语言,可以用来编写各种脚本来帮助人们从一些重复性的操作中解放出来,当然也可以用来开发网站,不过实现Python的自动打包上传只需要准备:
- 了解基本的Python语法
- 熟悉Requests这个网络库
关于fir
fir是一个第三方的托管网站, fir.im 为开发者提供测试应用极速发布,应用崩溃实时分析、用户反馈收集等一系列开发测试效率工具服务,所以需要准备的是
- 注册一个fir账号
- 了解fir对外提供的API
正文
注册fir账号
fir的注册地址是fir注册地址,不过免费的应用每天提供的免费下载次数是100次
配置Python运行环境
Python现在大致分为两个大的版本:2.X以及3.X,不过Python的3.X版本有些语法是不向下兼容的,由于对Python不是很熟悉,所以还是选择了2.7版本来配置环境,官网下载地址是Python官网,我是Windows系统,所以下载的安装包,然后下载了一个编辑Python的IDE名字是PyCharm,之所以选择IDE没有用SublimeText等文本编辑器是因为windows下的环境配置比较麻烦,而且刚开始有IDE的提示不至于在一些小问题上卡壳,当然如果你愿意去配置文本编辑器,我推荐SublimeText,提供了很多插件。
编写Python脚本
获取上传地址
名称 | 类型 | 标题 | 说明 |
---|---|---|---|
type | String | 是 | ios 或者 android(发布新应用时必填) |
bundle_id | String | 是 | App 的 bundleId(发布新应用时必填 |
api_token | String | 是 | 长度为 32, 用户在 fir 的 api_token |
服务器地址:http://api.fir.im/apps()
参数列表
名称 | 类型 | 标题 | 说明 |
---|---|---|---|
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
服务器地址:upload_url
binary字段对应的binary
名称 | 类型 | 标题 | 说明 |
---|---|---|---|
key | String | 是 | binary字段对应的key |
token | String | 是 | binary字段对应的token |
file | File | 是 | 安装包文件 |
x:name | String | 是 | 应用名称 |
x:version | String | 是 | 版本号 |
x:build | String | 是 | Build 号 |
x:changelog | String | 是 | 长度为 32, 用户在 fir 的 api_token |
参数列表
名称 | 类型 | 标题 | 说明 |
---|---|---|---|
key | String | 是 | binary字段对应的key |
token | String | 是 | binary字段对应的token |
file | File | 是 | 安装包文件 |
x:name | String | 是 | 应用名称 |
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界面查看结果
界面显示已经上传成功,但是发现没有Logo,我开始以为他会自动提取apk中的logo,实际上并没有,但是它提供了上传logo的接口,现在来继续上传logo
上传应用图标
服务器地址:upload_url
icon字段对应的upload_url
参数列表
名称 | 类型 | 标题 | 说明 |
---|---|---|---|
key | String | 是 | binary字段对应的 key |
token | String | 是 | binary字段对应的 token |
file | File | 是 | icon |
Postman测试
fir查看上传结果
这里用了一张微信朋友圈的logo上传,已经成功替换。
编写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。