基于 Jenkins 的 Android 持续集成

项目背景

有多个 Android 项目,希望通过放在 Git 上的通用 pipeline 脚本实线出包
出包成功之后,上传到蒲公英并推送到企业微信。

前提

Java SDK、环境变量,Android 相应版本SDK、ANDROID_SDK_ROOT环境变量,gradle已经配置完成

Jenkins配置

1、创建项目

2、配置参数

脚本依赖参数:PRODUCT_NAME(项目名称)、CODE_NAME(代码Git名称)、API_ENV(环境)、BUILD_TYPE(类型)

3、pipeline脚本

// 蒲公英 API Key
def PGYER_API_KEY = "ad85*****************e1c4"
// 蒲公英 User Key
def PGYER_U_KEY = "a172*******************eacf"
// 蒲公英 URL
def PGYER_URL = "https://www.pgyer.com/"
// 微信机器人URL
def WEIXIN_SEND_URL = "https://qyapi.weixin.qq.com/cgi-bin/webhook/send?key=508***************26f"

/**
 * 描述:Android出包公共pipeline
 * 使用说明
 *  必选字段:PRODUCT_NAME       ->  项目名称
 *             CODE_NAME          ->  代码Git名称
 *           API_ENV            ->  构建环境
 *           BUILD_TYPE         ->  构建类型
 */
pipeline {
    agent any

    stages {
        stage('检查环境') {
            steps {
                script {
                    opsTaskId = "${BUILD_NUMBER}_" + getTime()
                }
                echo "生成本次OPS TASKID => ${opsTaskId}"
            }
        }

        stage('Pulling code') {
            steps {
                echo 'Pull..'
                sh "rm -rf /home/jenkins/doc/workspace/$JOB_NAME/* && rm -rf /home/jenkins/doc/workspace/$JOB_NAME/.git"
                git branch: 'release', credentialsId: 'ff5******acac', url: 'http://git.gtech.asia/application/${CODE_NAME}.git'
            }
        }

        // 编译Android App
        stage('编译 App') {
            steps {
                echo "Build APK.."

                script {
                    sh "chmod +x gradlew"
                    sh "./gradlew clean assemble${API_ENV}${BUILD_TYPE} -PisJenkins=true -PjenkinsOpsTaskId=$opsTaskId"
                }
            }
        }

        // 发布app
        stage('发布 APP') {
            steps {
                script {

                    def apkPath = sh(returnStdout: true, script: "find ${env.WORKSPACE}/app/build/outputs/apk -iname \"*${opsTaskId}.apk\"").trim()
                    echo "生成APK路径:$apkPath"

                    // sh "curl -F 'file=@$apkPath' -F 'uKey=$pgyerApiKey' -F '_api_key=$pgyerUKey' https://qiniu-storage.pgyer.com/apiv1/app/upload"

                    def uploadJson = sh(returnStdout: true, script: "curl -F 'file=@$apkPath' -F 'uKey=$PGYER_U_KEY' -F '_api_key=$PGYER_API_KEY' https://qiniu-storage.pgyer.com/apiv1/app/upload").trim()
                    echo "上传蒲公英返回:$uploadJson"

                    def jsonObj = readJSON text: uploadJson

                    if (jsonObj.code == 0) {
                        // 上传成功
                        def appShortcutUrl = "$PGYER_URL" + jsonObj.data.appShortcutUrl
                        echo "蒲公英短连接:$appShortcutUrl"

                        // 发送企业群
                        sh "curl '$WEIXIN_SEND_URL' \\\n" +
                                "   -H 'Content-Type: application/json' \\\n" +
                                "    -d '\n" +
                                "   {\n" +
                                "        \"msgtype\": \"markdown\",\n" +
                                "        \"markdown\": {\n" +
                                "               \"content\": \"###### 【${PRODUCT_NAME} Android】 ${API_ENV}环境出包完成。\n" +
                                " > 构建ID:$opsTaskId\n" +
                                " > 签名类型:${BUILD_TYPE}\n\n" +
                                "[点击下载安装]($appShortcutUrl) \"}\n" +
                                " }'"
                    } else {
                        // 上传失败
                        currentBuild.result = "FAILURE"
                        throw new Exception("发布APP失败:上传蒲公英失败!")
                    }
                }
            }
        }
    }
}

def getTime() {
    return new Date().format("yyyyMMddHHmmss", TimeZone.getTimeZone("UTC"))
}

4、配置流水线

配置到 Jenkins

配置到 Git

android 项目配置
修改 build.gradle 配置,实现与 pipeline 脚本的联动(只截取联动代码)

def isWindows() {
    return System.properties['os.name'].contains('windows');
}

// Jenkins
def mIsJenkins = ''
def mJenkinsOpsTaskId = ""

android {

    mIsJenkins = project.hasProperty('isJenkins') ? isJenkins : 'false'
    mJenkinsOpsTaskId = project.hasProperty('jenkinsOpsTaskId') ? jenkinsOpsTaskId : ''

    // 与API_ENV相对应
    productFlavors {
        dev {

        }

        test {

        }

        test_new {

        }

        uat {

        }
    }

    // 设置apk包名
    applicationVariants.all { variant ->
        def buildType = variant.buildType.name
        def productFlavor = variant.productFlavors[0].name

        variant.outputs.all {
            def appFileName = ''
            if (mIsJenkins.toBoolean()) {
                appFileName = productFlavor + buildType + "_v" + rootProject.ext.android["versionName"] + "_" + mJenkinsOpsTaskId + '.apk'
            } else {
                appFileName = productFlavor + buildType + "_v" + rootProject.ext.android["versionName"] + '.apk'
            }
            outputFileName = appFileName
        }
    }
}

其他

1、CentOS 配置 android 环境

// 下载SDK
wget https://dl.google.com/android/android-sdk_r24.4.1-linux.tgz
tar xvf android-sdk_r24.4.1-linux.tgz

// 配置Android环境变量
cd android-sdk-linux/

export ANDROID_SDK_ROOT=/root/android/android-sdk-linux
export PATH=$ANDROID_SDK_ROOT/tools:$PATH
export PATH=$ANDROID_SDK_ROOT/platform-tools:$PATH
source /etc/profile

// 下载Android各个版本SDK
android update sdk -u -a -t 1,2,3,5,6,51,53

// 下载和配置gradle-6.5
wget https://services.gradle.org/distributions/gradle-6.5-bin.zip

unzip gradle-6.5-bin.zip

export PATH=$PATH:/root/android/gradle-6.5/bin
source /etc/profile

2、企业微信机器人配置说明

https://work.weixin.qq.com/api/doc/90000/90136/91770


https://mp.weixin.qq.com/s/DGQDljk7gUp-7VMHLWusOw

Android动态传感器的介绍及其应用

发布于:1月以前  |  195次阅读  |  详细内容 »

Android 实现小红书登陆页面背景图无限滚动效果

发布于:1月以前  |  243次阅读  |  详细内容 »

关于 Android MVVM 一些理解与实践

发布于:2月以前  |  637次阅读  |  详细内容 »

原生 Android 集成 React Native

发布于:2月以前  |  485次阅读  |  详细内容 »

基于 Jenkins 的 Android 持续集成

发布于:3月以前  |  688次阅读  |  详细内容 »

有赞 Android 编译优化方案 Savitar 2.0

发布于:3月以前  |  900次阅读  |  详细内容 »

闲鱼是如何实践一套完整的埋点自动化验证方案的?

搜索推荐算法的精准和埋点数据的准确性息息相关。一旦埋点数据出现问题,用户侧就会出现推荐商品不准确、过度推荐等问题,同时宏观的交易大盘数据的统计也会有偏差,进而影响整个商品运营策略,因此采取有效的手段来保障埋点质量就成为了闲鱼客户端质量保障的关键的一环。

发布于:4月以前  |  830次阅读  |  详细内容 »

Android 样式系统 | 主题背景覆盖

在 Android 样式系统系列的前几篇文章中,我们探讨了样式和主题背景之间的区别,讨论了使用主题背景和主题背景属性的好处,并重点介绍了一些常用的主题背景属性。 今天,我们聚焦于主题背景的实际使用,如何将它们应用到我们的应用中,以及如何构建主题背景。

发布于:6月以前  |  1106次阅读  |  详细内容 »

Android 深色模式适配原理分析

从Android10(API 29)开始,在原有的主题适配的基础上,Google开始提供了Force Dark机制,在系统底层直接对颜色和图片进行转换处理,原生支持深色模式。深色模式可以节省电量、改善弱势及强光敏感用户的可视性,并能在环境亮度较暗的时候保护视力,更是夜间活跃用户的强烈需求。对深色模式的适配有利于提升用户口碑。

发布于:7月以前  |  1673次阅读  |  详细内容 »

百度APP-Android H5首屏优化实践

百度App自2016年上半年尝试Feed流业务形态,至2017年下半年,历经10个版本的迭代,基本完成了产品形态的初步探索。在整个Feed流形态的闭环中,新闻详情页(文中称为落地页)作为重要的组成部分,如果打开页面后,loading时间过长,会严重影响用户体验。因此我们针对落地页这种H5的首屏展现速度进行了长期优化,本文会详细阐述整个优化思路和技术细节

发布于:7月以前  |  1344次阅读  |  详细内容 »

Android 10分区存储介绍及百度APP适配实践

Google于 2019年9月3日发布了Android10 release版本,为了更好的保护用户数据并限制设备冗余文件增加,Android 10版本变更了设备外部存储访问方式,外部存储新特性称为分区存储(Scoped Storage), 分区存储遵循以下三个原则对外部存储文件访问方式重新设计,便于用户更好的管理外部存储文件

发布于:7月以前  |  1586次阅读  |  详细内容 »

深入探究Android应用启动起点

开发者文档中提到,Android应用有三种启动状态,每种状态都会影响应用向用户显示所需的时间:冷启动、温启动或热启动。三种启动状态中,冷启动耗时最久,系统和App有较多初始化的工作。如果启动时间过长,可能会导致用户在应用商店打低分,甚至完全弃用app,所以冷启动速度是各个app非常重要的性能指标之一。

发布于:7月以前  |  1313次阅读  |  详细内容 »

一文搞懂Android JetPack组件原理之Lifecycle、LiveData、ViewModel与源码分析技巧

Lifecycle、LiveData和ViewModel作为AAC架构的核心,常常被用在Android业务架构中。在京东商城Android应用中,为了事件传递等个性化需求,比如ViewModel间通信、ViewModel访问Activity等等,以及为了架构的扩展性,我们封装了BaseLiveData和BaseViewModel等基础组件,也对Activity、Fragement和ViewHolder进行了封装,以JDLifecycleBaseActivity、LifecycleBaseFragment和LifecycleBaseViewHolder等组件强化了View层功能,构建出了各业务线统一规范架构的基石。

发布于:7月以前  |  1656次阅读  |  详细内容 »

Android 记一次解决问题的过程

之前我写过一篇文章,介绍我在GitHub开源的滑动控件 ConsecutiveScroller 是如何实现布局吸顶功能的。有兴趣的朋友可以去看一下:Android滑动布局ConsecutiveScrollerLayout实现布局吸顶功能。

发布于:7月以前  |  1243次阅读  |  详细内容 »

Android内存异常机制(用户空间)_NE

常见的Android稳定性异常,有内核异常和Android层异常。内核异常也就是常说的“kernel panic”,简称KE异常;Android层异常又分为java层crash和Native层crash,简称JE、NE异常。 上篇文章介绍了JE异常的抓取机制和处理方式,本文再讲一下NE异常。

发布于:9月以前  |  1837次阅读  |  详细内容 »

Android-模块化-面向接口编程

随着业务的发展,工程的逐渐增大与开发人员增多,很多工程都走向了模块化、组件化、插件化道路,来方便大家的合作开发与降低业务之间的耦合度。现在就和大家谈谈模块化的交互问题,首先看下模块化的几个优势。

发布于:9月以前  |  2214次阅读  |  详细内容 »

Android 机型适配终极篇

发布于:10月以前  |  1977次阅读  |  详细内容 »

Android 内存缓存 LruCache 原理与实现

okhttp和glide都使用的lru缓存,那什么是lru缓存呢?android 又是如何实现lru缓存 的呢?

发布于:10月以前  |  2024次阅读  |  详细内容 »

ijkPlayer编译支持https的so文件-Android

发布于:10月以前  |  2043次阅读  |  详细内容 »

Android SurfaceView 播放gif

Android SurfaceView 是Android系统中的高级组件,它有自己的绘制界面,可以在一个独立的线程进行UI的绘制, 因此不会阻塞主线程,这也是我们使用SuefaceView播放gif图片的原因。

发布于:10月以前  |  1865次阅读  |  详细内容 »

最多阅读

简化Android的UI开发 1年以前  |  463533次阅读
30分钟搭建一个android的私有Maven仓库 2年以前  |  3394次阅读
Android设计与开发工作流 1年以前  |  3293次阅读
Google Enjarify:可代替dex2jar的dex反编译 2年以前  |  3184次阅读
Android Studio 生成so文件 及调用 10月以前  |  2802次阅读
Android多渠道打包工具:apptools 2年以前  |  2749次阅读
Google Java编程风格规范(中文版) 2年以前  |  2710次阅读
Android UI基本技术点 2年以前  |  2682次阅读
Android权限 - 第一篇 2年以前  |  2558次阅读
Android死锁初探 10月以前  |  2509次阅读
Stetho 2年以前  |  2504次阅读
2015 Google IO带来的新 Android 开发工具 2年以前  |  2434次阅读
听FackBook工程师讲*Custom ViewGroups* 2年以前  |  2342次阅读