Python(一)Android藉助Python實現打包自動上傳fir

wustor發表於2017-11-12

本文已授權code小生獨家釋出,轉載請註明出處https://juejin.im/post/5a0675b851882531d016ded1

概述

在開發的過程中,很多時候完成了一個功能的開發,往往需要打包給測試進行測試,之前就是打個包,要麼是通過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指令碼

獲取上傳地址

伺服器地址: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

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介面檢視結果

fir介面截圖
介面顯示已經上傳成功,但是發現沒有Logo,我開始以為他會自動提取apk中的logo,實際上並沒有,但是它提供了上傳logo的介面,現在來繼續上傳logo

上傳應用圖示

伺服器地址:upload_url

icon欄位對應的upload_url

引數列表

名稱 型別 標題 說明
key String binary欄位對應的 key
token String binary欄位對應的 token
file File icon

Postman測試

測試logo上傳

fir檢視上傳結果

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。

原始碼下載

相關文章