Organizing Gradle Projects
- Separate language-specific source files
- Separate source files per test type
- Use standard conventions as much as possible
- Always define a settings file
- Use
buildSrc
to abstract imperative logic - Declare properties in
gradle.properties
file - Avoid overlapping task outputs
- Standardizing builds with a custom Gradle distribution
Source code and build logic of every software project should be organized in a meaningful way. This page lays out the best practices that lead to readable, maintainable projects. The following sections also touch on common problems and how to avoid them.
Separate language-specific source files
Gradle’s language plugins establish conventions for discovering and compiling source code.
For example, a project applying the Java plugin will automatically compile the code in the directory src/main/java
.
Other language plugins follow the same pattern.
The last portion of the directory path usually indicates the expected language of the source files.
Some compilers are capable of cross-compiling multiple languages in the same source directory.
The Groovy compiler can handle the scenario of mixing Java and Groovy source files located in src/main/groovy
.
Gradle recommends that you place sources in directories according to their language, because builds are more performant and both the user and build can make stronger assumptions.
The following source tree contains Java and Kotlin source files. Java source files live in src/main/java
, whereas Kotlin source files live in src/main/kotlin
.
Groovy
Kotlin
.
├── build.gradle
├── settings.gradle
└── src
└── main
├── java
│ └── HelloWorld.java
└── kotlin
└── Utils.kt
Separate source files per test type
It’s very common that a project defines and executes different types of tests e.g. unit tests, integration tests, functional tests or smoke tests. Optimally, the test source code for each test type should be stored in dedicated source directories. Separated test source code has a positive impact on maintainability and separation of concerns as you can run test types independent from each other.
The following source tree demonstrates how to separate unit from integration tests in a Java-based project.
Groovy
Kotlin
.
├── build.gradle
├── gradle
│ └── integration-test.gradle
├── settings.gradle
└── src
├── integTest
│ └── java
│ └── DefaultFileReaderIntegrationTest.java
├── main
│ └── java
│ ├── DefaultFileReader.java
│ ├── FileReader.java
│ └── StringUtils.java
└── test
└── java
└── StringUtilsTest.java
Gradle models source code directories with the help of the source set concept. By pointing an instance of a source set to one or many source code directories, Gradle will automatically create a corresponding compilation task out-of-the-box.
Groovy
Kotlin
sourceSets {
integTest {
java.srcDir file('src/integTest/java')
resources.srcDir file('src/integTest/resources')
compileClasspath += sourceSets.main.output + configurations.testRuntimeClasspath
runtimeClasspath += output + compileClasspath
}
}
Source sets are only responsible for compiling source code, but do not deal with executing the byte code. For the purpose of test execution, a corresponding task of type Test needs to be established.
Groovy
Kotlin
task integTest(type: Test) {
description = 'Runs the integration tests.'
group = 'verification'
testClassesDirs = sourceSets.integTest.output.classesDirs
classpath = sourceSets.integTest.runtimeClasspath
mustRunAfter test
}
check.dependsOn integTest
Use standard conventions as much as possible
All Gradle core plugins follow the software engineering paradigm convention over configuration. The plugin logic provides users with sensible defaults and standards, the conventions, in a certain context. Let’s take the Java plugin as an example.
-
It defines the directory
src/main/java
as the default source directory for compilation. -
The output directory for compiled source code and other artifacts (like the JAR file) is
build
.
By sticking to the default conventions, new developers to the project immediately know how to find their way around. While those conventions can be reconfigured, it makes it harder to build script users and authors to manage the build logic and its outcome. Try to stick to the default conventions as much as possible except if you need to adapt to the layout of a legacy project. Refer to the reference page of the relevant plugin to learn about its default conventions.
Always define a settings file
Gradle tries to locate a settings.gradle
(Groovy DSL) or a settings.gradle.kts
(Kotlin DSL) file with every invocation of the build.
For that purpose, the runtime walks the hierarchy of the directory tree up to the root directory.
The algorithm stops searching as soon as it finds the settings file.
Always add a settings.gradle
to the root directory of your build to avoid the initial performance impact.
This recommendation applies to single project builds as well as multi-project builds.
The file can either be empty or define the desired name of the project.
A typical Gradle project with a settings file look as such:
Groovy
Kotlin
.
├── build.gradle
└── settings.gradle
Use buildSrc
to abstract imperative logic
Complex build logic is usually a good candidate for being encapsulated either as custom task or binary plugin.
Custom task and plugin implementations should not live in the build script.
It is very convenient to use buildSrc
for that purpose as long as the code does not need to be shared among multiple, independent projects.
The directory buildSrc
is treated as an included build. Upon discovery of the directory, Gradle automatically compiles and tests this code and puts it in the classpath of your build script.
For multi-project builds there can be only one buildSrc
directory, which has to sit in the root project directory.
buildSrc
should be preferred over script plugins as it is easier to maintain, refactor and test the code.
buildSrc
uses the same source code conventions applicable to Java and Groovy projects.
It also provides direct access to the Gradle API. Additional dependencies can be declared in a dedicated build.gradle
under buildSrc
.
Groovy
Kotlin
repositories {
mavenCentral()
}
dependencies {
testImplementation 'junit:junit:4.12'
}
A typical project including buildSrc
has the following layout.
Any code under buildSrc
should use a package similar to application code.
Optionally, the buildSrc
directory can host a build script if additional configuration is needed (e.g. to apply plugins or to declare dependencies).
Groovy
Kotlin
.
├── build.gradle
├── buildSrc
│ ├── build.gradle
│ └── src
│ ├── main
│ │ └── java
│ │ └── com
│ │ └── enterprise
│ │ ├── Deploy.java
│ │ └── DeploymentPlugin.java
│ └── test
│ └── java
│ └── com
│ └── enterprise
│ └── DeploymentPluginTest.java
└── settings.gradle
✨
|
A change in buildSrc causes the whole project to become out-of-date.
Thus, when making small incremental changes, the --no-rebuild command-line option is often helpful to get faster feedback.
Remember to run a full build regularly or at least when you’re done, though.
|
Declare properties in gradle.properties
file
In Gradle, properties can be defined in the build script, in a gradle.properties
file or as parameters on the command line.
It’s common to declare properties on the command line for ad-hoc scenarios.
For example you may want to pass in a specific property value to control runtime behavior just for this one invocation of the build.
Properties in a build script can easily become a maintenance headache and convolute the build script logic.
The gradle.properties
helps with keeping properties separate from the build script and should be explored as viable option.
It’s a good location for placing properties that control the build environment.
A typical project setup places the gradle.properties
file in the root directory of the build.
Alternatively, the file can also live in the GRADLE_USER_HOME
directory if you want to it apply to all builds on your machine.
Groovy
Kotlin
.
├── build.gradle
├── gradle.properties
└── settings.gradle
Avoid overlapping task outputs
Tasks should define inputs and outputs to get the performance benefits of incremental build functionality. When declaring the outputs of a task, make sure that the directory for writing outputs is unique among all the tasks in your project.
Intermingling or overwriting output files produced by different tasks compromises up-to-date checking causing slower builds. In turn, these filesystem changes may prevent Gradle’s build cache from properly identifying and caching what would otherwise be cacheable tasks.
Standardizing builds with a custom Gradle distribution
Often enterprises want to standardize the build platform for all projects in the organization by defining common conventions or rules. You can achieve that with the help of initialization scripts. Initialization scripts make it extremely easy to apply build logic across all projects on a single machine. For example, to declare a in-house repository and its credentials.
There are some drawbacks to the approach. First of all, you will have to communicate the setup process across all developers in the company. Furthermore, updating the initialization script logic uniformly can prove challenging.
Custom Gradle distributions are a practical solution to this very problem. A custom Gradle distribution is comprised of the standard Gradle distribution plus one or many custom initialization scripts. The initialization scripts come bundled with the distribution and are applied every time the build is run. Developers only need to point their checked-in Wrapper files to the URL of the custom Gradle distribution.
The following steps are typical for creating a custom Gradle distribution:
-
Implement logic for downloading and repackaging a Gradle distribution.
-
Define one or many initialization scripts with the desired logic.
-
Bundle the initialization scripts with the Gradle distribution.
-
Upload the Gradle distribution archive to a HTTP server.
-
Change the Wrapper files of all projects to point to the URL of the custom Gradle distribution.
You can find a sample project that covers steps one to three in the samples
directory of the standard -all
Gradle distribution.