分享

Android 借助 Python 实现自动打包上传 fir

 codingSmart 2021-10-22

概述

在开发的过程中,很多时候完成了一个功能的开发,往往需要打包给测试进行测试,之前就是打个包,要么是通过 USB 进行安装,要么就是打个包通过 QQ 给测试发送过去,后来接触到 Jenkins,发现可以进行持续集成,但是很多时候往往只是改了一个很小的功能,比如说字体,或者颜色之类的,Jenkins 就有点大材小用了,这个时候,总想着要是能够通过脚本进行自动打包上传至服务器并且生成一个下载的二维码就好了,最近学习了 Python 并且也接触了 fir 这个第三方托管工具,发现,梦想还是要有的,万一实现了呢

关于 Python 和 fir

关于Python

Python 是一门高级编程语言,而且是一门动态语言,可以用来编写各种脚本来帮助人们从一些重复性的操作中解放出来,当然也可以用来开发网站,不过实现 Python 的自动打包上传只需要准备:

了解基本的 Python 语法

熟悉 Requests 这个网络库

关于fir

fir 是一个第三方的托管网站, 为开发者提供测试应用极速发布,应用崩溃实时分析、用户反馈收集等一系列开发测试效率工具服务,所以需要准备的是

注册一个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脚本

获取上传地址

名称类型标题说明

typeStringios 或者 android(发布新应用时必填)

bundle_idStringApp 的 bundleId(发布新应用时必填

api_tokenString长度为 32, 用户在 fir 的 api_token

服务器地址:http://api./apps()

参数列表

名称类型标题说明

typeStringios 或者 android(发布新应用时必填)

bundle_idStringApp 的 bundleId(发布新应用时必填

api_tokenString长度为 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./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

服务器地址:upload_url

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.', 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

参数列表

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.', 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./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, 0length)
            }
            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./apps
success_apk:{"is_completed":true}
success_icon:{"is_completed":true}
上传结束 with value 0

运行成功,到官网查看结果

完美,简单,以后简单的打包就用一行代码就可以搞定了,吼吼。

小结

其实 Python 的语法很简洁,作为一门动态语言,不需要像 Java 定义各种类型变量,gradle 的语法其实也一样,掌握这两种语言的基本用法,有助于更高效的开发 Android。

源码下载 https://github.com/wustor/PythoPackage

与之相关

6 天时间修改 1 行代码:现实中的软件开发流程

编程之旅,致新入行的朋友

如何阅读代码(八点要记牢)

    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多