Як автоматично збільшити код версії в Android Gradle


78

Я експериментую з новою системою збірки Android, заснованою на Gradle, і думаю, який найкращий спосіб автоматично збільшити versionCode з нею. Я думаю про два варіанти

  1. створити файл versionCode, прочитати з нього номер, збільшити його та записати назад у файл
  2. проаналізуйте AndroidManifest.xml, прочитайте з нього codeCode, збільште його та запишіть назад до AndroidManifest.xml

Чи є ще якесь просте чи підходяще рішення?

Хтось використовував один із варіантів згаданих і міг поділитися ним зі мною?


Читання з version.propertiesфайлу stackoverflow.com/a/21405744
Дорі

Відповіді:


58

Я вирішив для другого варіанту - проаналізувати AndroidManifest.xml. Ось робочий фрагмент.

task('increaseVersionCode') << {
    def manifestFile = file("AndroidManifest.xml")
    def pattern = Pattern.compile("versionCode=\"(\\d+)\"")
    def manifestText = manifestFile.getText()
    def matcher = pattern.matcher(manifestText)
    matcher.find()
    def versionCode = Integer.parseInt(matcher.group(1))
    def manifestContent = matcher.replaceAll("versionCode=\"" + ++versionCode + "\"")
    manifestFile.write(manifestContent)
}

tasks.whenTaskAdded { task ->
    if (task.name == 'generateReleaseBuildConfig') {
        task.dependsOn 'increaseVersionCode'
    }
}

versionCodeу цьому випадку випущено для збірки випусків. Щоб збільшити його для налагоджувальних збірок, змініть рівняння task.name у task.whenTaskAddedзворотному виклику.


5
одне питання, як інтегрувати цей код у систему збірки Cradle?
LiangWang

Крім того, дещо чіткіша відповідь тут не змінює XML на місці, але зберігає номер версії у файлі реквізиту: stackoverflow.com/questions/21405457/…
Пол Кантрелл,

1
Просто змініть файл ("AndroidManifest.xml") на файл ("src / main / AndroidManifest.xml"), якщо ви використовуєте нову структуру проекту.
Elhanan Mishraky,

3
Що робити, якщо я не встановлю versionCode у своєму AndroidManifest.xml?
Ігор Ганапольський

@Jacky Якщо ви , хлопці , до сих пір не отримав обхідний шлях, побачити моя відповідь: stackoverflow.com/a/39619297/1150251
Істар

39

Я використовую цей код для оновлення і versionCode, і versionName, використовуючи схему "major.minor.patch.build".

import java.util.regex.Pattern

task('increaseVersionCode') << {
    def manifestFile = file("src/main/AndroidManifest.xml")
    def pattern = Pattern.compile("versionCode=\"(\\d+)\"")
    def manifestText = manifestFile.getText()
    def matcher = pattern.matcher(manifestText)
    matcher.find()
    def versionCode = Integer.parseInt(matcher.group(1))
    def manifestContent = matcher.replaceAll("versionCode=\"" + ++versionCode + "\"")
    manifestFile.write(manifestContent)
}

task('incrementVersionName') << {
    def manifestFile = file("src/main/AndroidManifest.xml")
    def patternVersionNumber = Pattern.compile("versionName=\"(\\d+)\\.(\\d+)\\.(\\d+)\\.(\\d+)\"")
    def manifestText = manifestFile.getText()
    def matcherVersionNumber = patternVersionNumber.matcher(manifestText)
    matcherVersionNumber.find()
    def majorVersion = Integer.parseInt(matcherVersionNumber.group(1))
    def minorVersion = Integer.parseInt(matcherVersionNumber.group(2))
    def pointVersion = Integer.parseInt(matcherVersionNumber.group(3))
    def buildVersion = Integer.parseInt(matcherVersionNumber.group(4))
    def mNextVersionName = majorVersion + "." + minorVersion + "." + pointVersion + "." + (buildVersion + 1)
    def manifestContent = matcherVersionNumber.replaceAll("versionName=\"" + mNextVersionName + "\"")
    manifestFile.write(manifestContent)
}

tasks.whenTaskAdded { task ->
    if (task.name == 'generateReleaseBuildConfig' || task.name == 'generateDebugBuildConfig') {
        task.dependsOn 'increaseVersionCode'
        task.dependsOn 'incrementVersionName'
    }
}

1
Ви забули імпортувати шаблон вгорі: import java.util.regex.Pattern
Mic92

1
Я знаю, що це щось старе, але привіт, не боляче запитувати, але я спробував скористатися наведеним вище фрагментом, я отримую цю помилку Error:(48) Execution failed for task ':app:incrementVersionName'. > No match foundбудь-які ідеї?
acrichm

1
@acrichm: переконайтеся, що ви встановили android:versionName="0.0.0.1"в маніфесті, інакше не очікується кількість збігів
Jay Wick

Він працює ідеально, але в BuildConfig.java versionName порожній, загальнодоступний статичний фінальний рядок VERSION_NAME = "";
Аль-Мотафар

Остерігайтеся, що "Миттєвий запуск" в Android Studio буде повільнішим, тому щоFull recompilation is required because 'BuildConfig.java' was changed.
Іван Чау

36

здається, це не точне налаштування, яке ви використовуєте, але в моєму випадку збірками керують jenkins, і я хотів використати його $ BUILD_NUMBER як код версії програми. наступне зробило для мене фокус.

defaultConfig {
    ...
    versionCode System.getenv("BUILD_NUMBER") as Integer ?: 9999
    ...
}

Що таке System.genenv?
Ігор Ганапольський

Це насправді мені дуже допомогло при використанні Дженкінса. Ви можете використовувати цей плагін для створення змінної середовища з номером збірки та читання з допомогою System.getenv()у gradle.
Ламорак

19

Я використовую позначку часу для коду версії:

def date = new Date()
def formattedDate = date.format('yyMMddHHmm')
def code = formattedDate.toInteger()

defaultConfig {
    minSdkVersion 10
    targetSdkVersion 21
    versionCode code
}

1
це просто і геніально. великі пальці вгору
Євген Гарастович

7

Якщо ви тримаєте код версії у файлі build.gradle, використовуйте наступний фрагмент:

import java.util.regex.Pattern    
task('increaseVersionCode') << {
    def buildFile = file("build.gradle")
    def pattern = Pattern.compile("versionCode\\s+(\\d+)")
    def manifestText = buildFile.getText()
    def matcher = pattern.matcher(manifestText)
    matcher.find()
    def versionCode = Integer.parseInt(matcher.group(1))
    def manifestContent = matcher.replaceAll("versionCode " + ++versionCode)
    buildFile.write(manifestContent)
}

Цей шаблон не відповідає коду versionCode.
Лагос,

Чи можете ви опублікувати свій build.gradle?
Елханан Мішракі,

Мій build.gradle - це звичайний файл збірки. Рядок VersionCode виглядає так versionCode 18 . Навіть якщо я запустив ваш регулярний вираз на тесті ( "versionCode \d+"), він не буде збігатися.
Лагос,

Помилка полягала в тому, що ще один пробіл. Android Studio використовує два пробіли versionCodeта цифру, оскільки вона буде вирівняна разом із текстом у versionName.
Лагос,

5

Gradle Advanced Build Version - це плагін для Android, який дозволяє автоматично генерувати versionCode та versionName . є багато налаштувань. тут ви можете знайти більше інформації про це https://github.com/moallemi/gradle-advanced-build-version


Примітка: це ще не підтримує автоматичне збільшення назви версії
hooby3dfx

4

Щоб взяти до уваги обидва типи продуктів та типи збірки та використовуючи логіку @ sealskej для аналізу маніфесту:

android.applicationVariants.all { variant ->
    /* Generate task to increment version code for release */
    if (variant.name.contains("Release")) {
        def incrementVersionCodeTaskName = "increment${variant.name}VersionCode"
        task(incrementVersionCodeTaskName) << {
            if (android.defaultConfig.versionCode == -1) {
                def manifestFile = file(android.sourceSets.main.manifest.srcFile)
                def pattern = Pattern.compile("versionCode=\"(\\d+)\"")
                def manifestText = manifestFile.getText()
                def matcher = pattern.matcher(manifestText)
                matcher.find()
                def versionCode = Integer.parseInt(matcher.group(1))
                android.defaultConfig.versionCode = versionCode + 1
                def manifestContent = matcher.replaceAll("versionCode=\"" + android.defaultConfig.versionCode + "\"")
                manifestFile.write(manifestContent)
            }
        }
        def hookTask = variant.generateBuildConfig
        hookTask.dependsOn(incrementVersionCodeTaskName)
    }
}

1
Де нам це потрібно написати ... Я маю на увазі, в який клас / файл потрібно додати цей код
ChArAnJiT

це частина build.gradle, оскільки це частина процесу збірки
Max Ch

3

Завдання Increment VersionCode (Integer):

Це працює шляхом збільшення коду версії 1, наприклад:

  android:versionCode="1"
1 + 1 = 2
import java.util.regex.Pattern

task incrementVersionCode << {
    def manifestFile = file('AndroidManifest.xml')
    def matcher = Pattern.compile('versionCode=\"(\\d+)\"')
    .matcher(manifestFile.getText())
    matcher.find()
    def manifestContent = matcher.replaceAll('versionCode=\"' +
        ++Integer.parseInt(matcher.group(1)) + '\"')
    manifestFile.write(manifestContent)
}

Завдання Increment VersionName (рядок):

Попередження: Обов’язково повинен містити 1крапку для регулярного виразу

Це працює, збільшуючи назву версії 0.01, наприклад, наприклад: Ви можете легко змінити та змінити свій приріст або додати більше цифр.

android:versionName="1.0"
1.00 + 0.01 -> 1.01
1.01 + 0.01 -> 1.02
1.10 + 0.01 -> 1.11
1.99 + 0.01 -> 2.0
1.90 + 0.01 -> 1.91
import java.util.regex.Pattern

task incrementVersionName << {
    def manifestFile = file('AndroidManifest.xml')
    def matcher = Pattern.compile('versionName=\"(\\d+)\\.(\\d+)\"')
    .matcher(manifestFile.getText())
    matcher.find()
    def versionName = String.format("%.2f", Integer
        .parseInt(matcher.group(1)) + Double.parseDouble("." + matcher
            .group(2)) + 0.01)
    def manifestContent = matcher.replaceAll('versionName=\"' +
        versionName + '\"')
    manifestFile.write(manifestContent)
}

До:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.exmaple.test"
    android:installLocation="auto"
    android:versionCode="1"
    android:versionName="1.0" >

Після:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.exmaple.test"
    android:installLocation="auto"
    android:versionCode="2"
    android:versionName="1.01" >

3

Якщо ви пишете versionCodeв gradle.buildфайл (самий випадок в даний час), тут обхідний шлях. Трохи дурний (оновіть "себе"), але це працює!

import java.util.regex.Pattern

task('increaseVersionCode') << {
    def buildFile = file("build.gradle")
    def pattern = Pattern.compile("versionCode(\\s+\\d+)")
    def buildText = buildFile.getText()
    def matcher = pattern.matcher(buildText)
    matcher.find()
    def versionCode = android.defaultConfig.versionCode
    def buildContent = matcher.replaceAll("versionCode " + ++versionCode)
    buildFile.write(buildContent)

    System.out.println("Incrementing Version Code ===> " + versionCode)
}

tasks.whenTaskAdded { task ->
    if (task.name == 'generateReleaseBuildConfig') {
        task.dependsOn 'increaseVersionCode'
    }
}

2

Щоб додати до допису @ sealskej, ось як ви можете оновити як код вашої версії, так і назву версії (тут я припускаю, що і ваша основна, і другорядна версії 0):

task('increaseVersion') << {
    def manifestFile = file("AndroidManifest.xml")
    def patternVersionCode = Pattern.compile("versionCode=\"(\\d+)\"")
    def manifestText = manifestFile.getText()
    def matcherVersionCode = patternVersionCode.matcher(manifestText)
    matcherVersionCode.find()
    def versionCode = Integer.parseInt(matcherVersionCode.group(1))
    def manifestContent = matcherVersionCode.replaceAll("versionCode=\"" + ++versionCode + "\"")

    manifestFile.write(manifestContent)

    def patternVersionNumber = Pattern.compile("versionName=\"0.0.(\\d+)\"")
    manifestText = manifestFile.getText()
    def matcherVersionNumber = patternVersionNumber.matcher(manifestText)
    matcherVersionNumber.find()
    def versionNumber = Integer.parseInt(matcherVersionNumber.group(1))
    manifestContent = matcherVersionNumber.replaceAll("versionName=\"0.0." + ++versionNumber + "\"")
    manifestFile.write(manifestContent)
}

1

як що до цього ? додати до build.gradle (модуль програми)

def getBuildVersionCode() {
    def date = new Date()
    def formattedDate = date.format('yyyyMMdd')
    def formattedSeconds = date.format('HHmmssSSS')
    def formatInt = formattedDate as int;
    def SecondsInt = formattedSeconds as int;
    return (formatInt + SecondsInt) as int
}

   defaultConfig {
    applicationId "com.app"
    minSdkVersion 17
    targetSdkVersion 22
    versionCode getBuildVersionCode()
    versionName "1.0"
}

2
Чудова думка, але(20160129 + 000000000) < (20160128 + 125959999)
C0D3LIC1OU5

0

Отож, як я розглядав більшість рішень, вони були приємними, але недостатніми, тому я написав це, один крок на кожне розгортання:

Це збільшить збірку під час компіляції версій налагодження, а код точки та версії - під час розгортання.

import java.util.regex.Pattern

def incrementVersionName(int length, int index) {
    def gradleFile = file("build.gradle")
    def versionNamePattern = Pattern.compile("versionName\\s*\"(.*?)\"")
    def gradleText = gradleFile.getText()
    def matcher = versionNamePattern.matcher(gradleText)
    matcher.find()

    def originalVersion = matcher.group(1)
    def originalVersionArray = originalVersion.split("\\.")
    def versionKeys = [0, 0, 0, 0]
    for (int i = 0; i < originalVersionArray.length; i++) {
        versionKeys[i] = Integer.parseInt(originalVersionArray[i])
    }
    def finalVersion = ""
    versionKeys[index]++;
    for (int i = 0; i < length; i++) {
        finalVersion += "" + versionKeys[i]
        if (i < length - 1)
            finalVersion += "."
    }

    System.out.println("Incrementing Version Name: " + originalVersion + " ==> " + finalVersion)

    def newGradleContent = gradleText.replaceAll("versionName\\s*\"(.*?)\"", "versionName \"" + finalVersion + "\"")
    gradleFile.write(newGradleContent)
}

def incrementVersionCode() {
    def gradleFile = file("build.gradle")
    def versionCodePattern = Pattern.compile("versionCode\\s*(\\d+)")
    def gradleText = gradleFile.getText()
    def matcher = versionCodePattern.matcher(gradleText)
    matcher.find()

    def originalVersionCode = Integer.parseInt(matcher.group(1) + "")
    def finalVersionCode = originalVersionCode + 1;
    System.out.println("Incrementing Version Code: " + originalVersionCode + " ==> " + finalVersionCode)

    def newGradleContent = gradleText.replaceAll("versionCode\\s*(\\d+)", "versionCode " + finalVersionCode)
    gradleFile.write(newGradleContent)
}

task('incrementVersionNameBuild') << {
    incrementVersionName(4, 3)
}

task('incrementVersionNamePoint') << {
    incrementVersionName(3, 2)
}

task('incrementVersionCode') << {
    incrementVersionCode()
}


def incrementedBuild = false
def incrementedRelease = false

tasks.whenTaskAdded { task ->
    System.out.println("incrementedRelease: " + incrementedRelease)
    System.out.println("incrementedBuild: " + incrementedBuild)
    System.out.println("task.name: " + task.name)

    if (!incrementedBuild && task.name.matches('generate.*?DebugBuildConfig')) {
        task.dependsOn 'incrementVersionNameBuild'
        incrementedBuild = true
        return
    }

    if (!incrementedRelease && task.name.matches('generate.*?ReleaseBuildConfig')) {
        task.dependsOn 'incrementVersionCode'
        task.dependsOn 'incrementVersionNamePoint'
        incrementedRelease = true
        return
    }
}

-1

Мій підхід полягає в тому, щоб прочитати файл маніфесту з папки збірки та витягти звідти buildVersion, ніж видалити папку. Коли завдання створює новий маніфест, моя збільшена змінна buildVersion вже є там.

def versionPattern = "Implementation-Version=(\\d+.\\d+.\\d+.\\d+\\w+)"

task generateVersion (dependsOn : 'start') {
    // read build version from previous manifest
    def file = file("build/libs/MANIFEST.MF")
    if (file.exists()) {
        def pattern = Pattern.compile(versionPattern)
        def text = file.getText()
        def matcher = pattern.matcher(text)
        matcher.find()
        buildNumber = Integer.parseInt(matcher.group(1))
        // increment build version
        version = "${majorVer}.${minorVer}.${patchVer}.${++buildNumber}${classifier}_${access}"
    } 
    else 
        version = "${majorVer}.${minorVer}.${patchVer}.1${classifier}_${access}"
}

task specifyOutputDir (dependsOn : 'generateVersion', type : JavaCompile) {
    // create a folder for new build
    destinationDir = file("build/${version}/")
}

task clean (dependsOn : 'generateVersion', type : Delete) {
    doLast {
        delete "build/${version}"
        println 'Build directory is deleted'
    }
}

task configureJar (dependsOn : 'generateVersion', type : Jar) {
    baseName = applicationName
    version = project.version
    archiveName = "${applicationName}_ver${version}.${extension}"
    manifest {[
            "Main-Class" : mainClassName,
            "Implementation-Title" : name,
            "Implementation-Version" : version,
            "Access" : access,
            "Developer" : developer
    ]}
}

Це не дає відповіді на запитання. Отримавши достатню репутацію, ви зможете коментувати будь-яку публікацію ; натомість надайте відповіді, які не вимагають роз’яснень від запитувача . - З огляду
MLavoie

@MLavoie перевірте зараз мій фрагмент WIP
Katoteshi Fuku
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.