Welcome to thirty-first week of 52-technologies-in-2016 series. This week I decided to write down Gradle tips that I have learnt in last year or so. Over last year or so I have started using Gradle as my primary build tool for JVM based projects. Before using Gradle I was an Apache Maven user. Gradle takes best from both Apache Maven and Apache Ant providing you best of both worlds. Gradle borrows flexibility from Ant and convention over configuration, dependency management and plugins from Maven. Gradle treats task as first class citizen just like Ant.
A Gradle build has three distinct phases - initialization, configuration, and execution. The initialization phase determine which all projects will take part in the build process and create a Project instance for each of the project. During configuration phase, it execute build scripts of all the project that are taking part in build process. Finally, during the execution phase all the tasks configured during the configuration phase are executed.
In this document, I will list down tips that I have learnt over last year or so.
The actual gradle tips is maintained at github:shekhargulati/gradle-tips. If you wish to add any Gradle tip please send send pull request to github:shekhargulati/gradle-tips project.
One of the Gradle features that impressed me a lot when I was starting with Gradle was support for wrapper scripts. Gradle wrapper makes your project self contained and independent of build tool installation. It lets you run Gradle builds without a previously installed Gradle distribution in a zero configuration manner. This will ensure everyone use same version of the build tool.
To create Gradle wrapper scripts for your Grade project you can run following command.
$ gradle wrapper --gradle-version 2.14.1
To run the above command you will need Gradle installed on your machine. If you are on Mac, then you can use
brew install gradle
.
This will generate few files in your project -- gradlew
, gradlew.bat
, gradle/wrapper/gradle-wrapper.jar
, and gradle/wrapper/gradle-wrapper.properties
. Now instead of running gradle
you should run gradlew
.
Make sure to unignore
gradle-wrapper.jar
in your version control ignone file. By default, version control ignore files ignore jar files.
At any point in time, if you wish to upgrade the Gradle version just regenerate the Gradle wrapper scripts passing it the Gradle version you want to use. Let's suppose we want to upgrade to Gradle 3.0-milestone-2
run the command again as shown below.
$ gradle wrapper --gradle-version 3.0-milestone-2
Also, it is a good idea to set an alias for ./gradlew
.
alias gradle="./gradlew"
To view a dependency graph for your project you can run the following command.
$ gradle dependencies
Gradle supports both single and multi-project builds. Let's suppose our multi-project structure looks like as shown below.
app
api
model
rest
core
web
itests
To build the rest
project we will run the following command.
$ gradle api:rest:build
To execute a task you use -x
option. Let's suppose we want to skip tests then we can use following command
$ gradle clean build -x test
Gradle has in-built support for profiling. If you are facing performance issues, you should use the --profile
option to generate profile report. The report displays time taken by different tasks. Let's suppose we want to profile the build task then we can run the following command.
$ gradle --profile build
This will generate report in the build/reports/profile
directory.
There are times when you wish to see all the tasks that will be executed during the build but don't want to execute them. For this scenario Gradle provides --dry-run
option.
$ gradle build --dry-run
$ gradle install
$ gradle tasks
The above does not list all the tasks. To view all the tasks you have to pass --all
flag.
$ gradle tasks --all
One of the easiest way to speed your Gradle build is to use Gradle daemon to run your build. Gradle daemon is a long-lived background process that performs bootstrapping only once during its lifetime. Gradle daemon is not enabled by default. To use Gradle daemon, you can use --daemon
flag to your build command.
$ gradle build --daemon
It will enabled by default in 3.0.
Passing the --daemon
flag each time is cumbersome so you can enable it on your developer machine by adding this flag in the ~/.gradle/gradle.properties
file.
org.gradle.daemon=true
Open your ~/.gradle/gradle.properties
and add the following line.
org.gradle.parallel=true
You can customize any Gradle tasks by overriding its doFirst
and doLast
life cycle methods. Let's suppose we want add print statements before and after executing tests we can do that by following.
apply plugin:'java'
test.doFirst {
println("running tests...")
}
test.doLast {
println("done executing tests...")
}
You can specify JVM arguments to Gradle daemon by entering a line in ~/.gradle/gradle.properties
as shown below.
org.gradle.jvmargs=-Xmx2048m -XX:MaxPermSize=512m -XX:+HeapDumpOnOutOfMemoryError -Dfile.encoding=UTF-8
$ gradle build --offline
Configuration on demand is an incubation feature of Gradle, so it’s not enabled by default yet.
$ gradle clean build --configure-on-demand
If you want to make this a default option, then you can provide this option globally by adding a line to ~/.gradle/gradle.properties
org.gradle.configureondemand=true
$ gradle clean build --refresh-dependencies
You can also delete the cached files under ~/.gradle/caches
. Next time, when you will run the build it will download all the dependencies and populate your cache.
Let's suppose you have a lib
directory that contains jar file that you need to use in your Gradle project.
dependencies {
compile files('libs/myjar.jar')
}
This can also be done by following.
repositories {
flatDir {
dirs 'libs'
}
}
dependencies {
compile name: 'myjar'
}
If you need to add all the libraries in a directory then you can do the following:
dependencies {
compile fileTree(dir: 'libs', include: ['*.jar'])
}
$ gradle api:model:buildNeeded
bash
$ gradle api:rest:buildDependents
It is a good practice to define default tasks for your project so that a first time user of your project can easily get started. In your Gradle script, define defaultTasks
variable passing it the tasks it should execute.
defaultTasks "clean","build"
Now, if a user will run gradle
command default tasks will be executed.
apply plugin: 'java'
archivesBaseName = 'checksum-sample'
jar.doLast { task ->
ant.checksum file: task.archivePath
}
By default build file has build.gradle
as its name. You can use a different name by doing following in the settings.gradle
file.
rootProject.buildFileName = "gradle-tips.gradle"
Now, rename your build file build.gradle
to gradle-tips.gradle
By convention, we use build.gradle
as the name of the Gradle build script. When you are working with a multi-project Gradle project then it make sense to use different names for build scripts. Let's suppose our multi module project looks like this:
app
api
core
web
itests
By default, all of these sub projects will have build.gradle
as their Gradle build file. We can change that by overriding that in settings.gradle
rootProject.children.each {
it.buildFileName = it.name + '.gradle'
}
Now, you can use build.gradle for root project. Sub projects will have api.gradle
, core.gradle
, web.gradle
, and itests.gradle
as their build definition files.
You can use Gradle gui by launching it via command-line as shown below.
$ gradle --gui
This will open the Gradle gui as shown below.
task untar( type : Copy) {
from tarTree(‘dist.tar.gz’)
into ‘destFolder’
}
In your build script, define a configuration block as shown below.
configurations {
compile.resolutionStrategy.failOnVersionConflict()
}
You can use maven like provided
scope in gradle 2.12+ by using ‘compileOnly’ scope.
dependencies {
compileOnly 'javax.servlet:servlet-api:3.0-alpha-1'
}
In your build.gradle
add the following line.
compileJava.options.encoding = 'UTF-8'
Turn transitive dependencies off for a whole configuration:
configurations {
compile.transitive = false
}
You can view Gradle version
$ gradle -v
------------------------------------------------------------
Gradle 2.14.1
------------------------------------------------------------
Build time: 2016-07-18 06:38:37 UTC
Revision: d9e2113d9fb05a5caabba61798bdb8dfdca83719
Groovy: 2.4.4
Ant: Apache Ant(TM) version 1.9.6 compiled on June 29 2015
JVM: 1.8.0_60 (Oracle Corporation 25.60-b23)
OS: Mac OS X 10.10.5 x86_64
You can view the Gradle version that your current build is running by using GradleVersion.current()
. You can create a task that does the work.
task gradleVersion {
group = "help"
description = "Prints Gradle version"
doLast {
logger.quiet("You are using [${GradleVersion.current()}]")
}
}
When you will run it you will see the following:
$ gradle gradleVersion
:gradleVersion
You are using [Gradle 2.14.1]
BUILD SUCCESSFUL
Total time: 0.667 secs
taskName.enabled = false
If you want to disable test task then you can disable it as shown below.
test.enabled = false
To create Java Gradle project that uses testng testing framework you can use the following command.
$ gradle init --type java-library --test-framework testng
If you want to use JUnit, then don't specify --test-framework
$ gradle init --type java-library
You can also create groovy and scala projects as well.
$ gradle init --type scala-library
$ gradle init --type groovy-library
apply plugin: 'signing'
signing {
sign configurations.archives
}
If you only want to sign releases not snapshots then you can do following
apply plugin: 'signing'
signing {
required { !version.endsWith("SNAPSHOT”) }
}
test {
maxParallelForks = 2
}
test {
minHeapSize = ‘512m'
maxHeapSize = ‘1024m'
}
If you have a task buildServerDistribution
then you can call it as shown below.
$ gradle bSD
You have to make sure it is unique among all tasks. If there is another task buildSafeDistribution
then you have specify
$ gradle bSeD
$ gradle help --task <task name>
$ gradle help --task dependencies
$ gradle clean build --debug
$ gradle clean build --continue
Go to your Maven project and run the following command.
$ gradle init --type pom
$ gradle build --rerun-tasks
When you are declaring dependencies don't use + in your dependencies, rather use exact version numbers. This will make your build faster and reproducible.
If you wish to continuously run your build then you can use --continuous
flag. It will look for file changes and whenever it detects one it will rerun the command. To enable continuous tests,
$ gradle test --continuous
There are times when we only when to run a single test case rather than running the full test suite. This can be accomplished by following command.
$ gradle test --tests tips.CalculatorTest
To run only a single test case of tips.CalculatorTest
you can do following.
$ gradle test --tests tips.CalculatorTest.shouldAddTwoNumbers
You can also use regex to specify multiple tests
$ gradle test --tests "tips.Calculator*Test"
You can also use --tests
flag multiple times.
$ gradle test --tests tips.CalculatorTest --tests tips.Calculator1Test
To run a single test in a submodule you can do following
$ gradle api:test --tests app.api.PingResourceTest
task sourcesJar(type: Jar, dependsOn: classes) {
classifier = 'sources'
from sourceSets.main.allSource
}
task javadocJar(type: Jar, dependsOn: javadoc) {
classifier = 'javadoc'
from javadoc.destinationDir
}
artifacts {
archives sourcesJar, javadocJar
}
You can access environment variables in couple of ways
println(System.getenv("HOME"))
println("$System.env.HOME")
By default Gradle will only log test failures on console. This at times limits the visibility of which all tests have ran. Gradle allows you to configure it using the testLogging property. To log all the events, you can do following. For more information read this.
test {
testLogging {
events "passed", "skipped", "failed"
}
}
Now, when you will run the ./gradlew clean build
you will see passed tests as well.
$ gradle clean test
:clean
:compileJava
:processResources UP-TO-DATE
:classes
:compileTestJava
:processTestResources UP-TO-DATE
:testClasses
:test
tips.CalculatorTest > shouldSubtractTwoNumbers PASSED
tips.CalculatorTest > shouldAddTwoNumbers PASSED
tips.CalculatorTest > shouldSubtractTwoNumbers1 PASSED
One thing to keep in mind is that gradle test
command executes the test only when they change. So if you run it second time without any change then no output will be produced. You will see :test UP-TO-DATE
which means no changes were detected. You can force Gradle to run tests each time by using ./gradlew cleanTest test
.
test {
testLogging {
events "passed", "skipped", "failed"
showStandardStreams = true
}
}
You should not hardcode credentials in your build.gradle
instead you should rely on your user home ~/.gradle/gradle.properties
to store credentials. Let's suppose you want to use a Maven repository protected by credentials. One way to specify credentials is to hard code them in your build.gradle as shown below.
repositories {
maven {
credentials {
username "admin"
password "admin123"
}
url "http://nexus.mycompany.com/"
}
}
The better way is to change your personal ~/.gradle/gradle.properties
nexusUsername = admin
nexusPassword = admin123
Now, refer this in the build.gradle
repositories {
maven {
credentials {
username "$nexusUsername"
password "$nexusPassword"
}
url "http://nexus.mycompany.com/"
}
}
If your application is packaged as an executable jar that you can run via Gradle then you can debug it by passing --debug-jvm
option. Spring Boot applications are run as an executable jar. You can use gradle bootRun
to run the application. To debug the app at port 5005 you can start app in debug mode.
$ gdw <taskname> --debug-jvm
$ gradle bootRun --debug-jvm
That's all for this week. Please provide your valuable feedback by adding a comment to shekhargulati#44.
You can follow me on twitter at https://twitter.com/shekhargulati or email me at [email protected]. Also, you can read my blogs at http://shekhargulati.com/