Gradle build null console object

Solution 1:

For some reason, running gradle in daemon mode causes a null console object. If you specify the appropriate command line flag,

./gradlew assembleRelease --no-daemon

it'll work.

Solution 2:

I found a solution here at https://www.timroes.de/2014/01/19/using-password-prompts-with-gradle-build-files and slightly modified it. Nevertheless, all credits go to Tim Roes!

gradle.taskGraph.whenReady { taskGraph ->
if(taskGraph.hasTask(':app:assembleRelease')) {
    def storePass = ''
    def keyPass = ''
    if(System.console() == null) {
        new SwingBuilder().edt {
            dialog(modal: true, title: 'Enter password', alwaysOnTop: true, resizable: false, locationRelativeTo: null, pack: true, show: true) {
                vbox { // Put everything below each other
                    label(text: "Please enter store passphrase:")
                    def input1 = passwordField()
                    label(text: "Please enter key passphrase:")
                    def input2 = passwordField()
                    button(defaultButton: true, text: 'OK', actionPerformed: {
                        storePass = input1.password;
                        keyPass = input2.password;
                        dispose();
                    })
                }
            }
        }
    } else {
        storePass = System.console().readPassword("\nPlease enter store passphrase: ")
        keyPass = System.console().readPassword("\nPlease enter key passphrase: ")
    }

    if(storePass.size() <= 0 || keyPass.size() <= 0) {
        throw new InvalidUserDataException("You must enter the passwords to proceed.")
    }

    storePass = new String(storePass)
    keyPass = new String(keyPass)

    android.signingConfigs.release.storePassword = storePass
    android.signingConfigs.release.keyPassword = keyPass
    }
}

Somewhere in the some gradle file, you have the configuration for the release signing defined.

android {
...
signingConfigs {
    ...
    release {
        storeFile file(System.getProperty("user.home")+"\\android-key")
        storePassword ''
        keyAlias "standard"
        keyPassword ''
    }
}

...
}

(Don't forget to import groovy.swing.SwingBuilder.)

Regarding the second part, you may also have a look at How to create a release signed apk file using Gradle?

Solution 3:

Ok, the reason why this didn't work was silly, but just in case anyone else comes across it I thought I'd post.

I was running the task through android studio and didn't realise that the console object would always be null. When running from the command line the "command" object isn't null and it works ok.

Solution 4:

Executing System.getConsole() from Gradle when org.gradle.daemon property is true, or when it's executed from an IDE like IntelliJ or Android Studio it returns null. So for example do System.console().readLine() becomes not possible.

Furthermore starting from Gradle 3.0 gradle.daemon is turned on by default.

Then instead of a workaround to use System.getConsole() I purpose an alternative, use ant.input like so:

task avoidNullOnConsole << {
    ant.input(message: 'Enter keystore password:', addproperty: 'userInputPassword', defaultValue : '1234')
    def password = ant.properties.userInputPassword
}

In this case ant.input shows the message and adds the user input in ant.properties using as a property name the value defined in addProperty. If there is no user input then the value defined in default attribute is used.

Once executed you can get the user input using ant.properties.yourProperty or ant.properties['yourProperty'].

You can check the rest of the ant.input attributes here.

Note: If you want to use ant.input multiple times take in account that you cannot override and existing property so addProperty attribute must be different for each one.

Solution 5:

Take a look at this blog post (https://www.timroes.de/2013/09/22/handling-signing-configs-with-gradle/).

It describes multiple ways to handle signing configs and one of them is exactly your question regarding console input for the password.