The Properties plugin is a useful plugin that changes the way Gradle loads properties from the various properties files.
By default, Gradle applies properties to your project in a particular order. The last thing to define a property wins. Gradle's order of processing is:
-
The gradle.properties file in the project directory
-
The gradle.properties file in the user's ${gradleUserHomeDir}/.gradle directory.
-
The command line properties set with -P arguments.
The properties plugin enhances this sequence by adding two additional steps. One for project properties that change from environment to environment, and one for properties that are common to a user (or client), but change from user to user (or client to client). The order of execution for the properties plugin is:
-
The gradle.properties file in the project directory
-
The gradle-${environmentName}.properties file in the project directory. if no environment is specified, the plugin will assume an environment name of "local". We strongly recommend adding gradle-local.properties to the .gitignore file of the project so that develpers' local configurations don't interfere with each other.
-
The gradle.properties file in the user's ${gradleUserHomeDir}/.gradle directory. Properties in this file are generally things that span projects, or shouldn't be checked into a repository, such as user credentials, etc.
-
If the
${gradleUserName} property is set, the properties plugin will load properties from $ {gradleUserHomeDir}/gradle-${gradleUserName}.properties. This file is useful for cases when you need to build for a different user or client and you have properties that span projects. For example you might have several projects that have pages with a customized banner. The contents of the banner change from client to client. The gradle-${gradleUserName}.properties file is a great way to put the client's custom text into a single file per client, and specify at build time which client's banners should be used. -
The command line properties set with -P arguments.
As with standard Gradle property processing, the last one in wins. The properties plugin also creates a "filterTokens" property that can be used to do token replacement in files, allowing Gradle to edit configuration files for you.
The Properties plugin is designed to make it easier to work with properties that change from environment to environment, or client to client. It makes life easier for developers who are new to a project to configure and build a because properties for multiple environments can be stored with the project itself, reducing the number of external magic that needs to happen for a project to run. It it makes life easier for experienced developers to create different configurations for different scenarios on their boxes.
One of the challenges to building a project is that it often contains things that should be changed from environment to environment. This includes things like the log file directory, or the JNDI data source name. The values of these properties are often complicated by platform differences between the OSX and Windows environments most developers use and the Unix environments used for deployments. It gets even messier when you consider that these values are often scattered amongst several files buried in the project.
This can be a real hindrance to new developers to a project, who don't know all the touch points of project configuration, or experienced developers who don't remember all the things that need to be changed to spin up a new environment.
The solution is to extract all the environment specific information to a single file for each environment, with everything the project needs in that single file. Files that need these values, like persistence.xml or log4j.properties, become template files with tokens that can be replaced by the build.
This provides an added benefit; new developers see some of the things that need to be setup external to the project. For example, if the project mentions a tomcat home, the developer knows they need to install tomcat. If they see a JNDI data source, they know they need to set one up inside their container.
The initial setup of a project is a little involved, but once done, this process greatly simplifies things for the developers who follow.
Step 1: Extract environment sensitive files to a template directorty
Make a directory to hold the template files and copy files that change, like log4j.properties, into it.
Edit the files and replace the environment specific values with tokens. For example, if the file had a property like "log.dir=/opt/tomcat/logs", replace it with "[email protected]@".
Step 2: Create property files
Create a file for each named environment you use for deployment. For example, if you have "dev", "qa", and "prod" environments. You would create gradle-dev.properties, gradle-qa.properties and gradle-prod.properties files. There is no limitation on environment names, they are just labels to tell the plugin which file to read at build time. If no environment is specified, the properties plugin uses the gradle-local.properties file, which should not be checked in to source control, because this is the file where developers setup the properties for their own machines.
In our example, the gradle-local.properties file might contain "applicationLogDir=c:/opt/tomcat" or "applicationLogDir=/Users/steve/projects/myproject/logs" to define the log directory needed by the log4j.properties template.
Step 3: Include and apply the plugin
Add the following to the build.gradle file:
// Pull the plugin from Maven Central
buildscript {
repositories {
mavenCentral()
}
dependencies {
classpath 'net.saliman:gradle-properties-plugin:1.1.0'
}
}
// invoke the plugin
apply plugin: 'properties'
When the properties plugin is applied, three things happen. First, the plugin processes the various property files as described above.
Next, the plugin creates a project property named "filterTokens". The filter tokens is a map of name-value pairs that can be used when doing a filtered file copy. There will be token for each property defined in the given properties file. The name of each token will be the name of the property from the properties file, after converting the camel case property name to dot notation. In our example, the "applicationLogDir=/opt/tomcat/logs" entry in the property file will create a token named "application.log.dir" with the value of /opt/tomcat/logs
Finally, the properties plugin adds some property closures to every task in the build (and causes them to be added to new tasks defined later). These properties can be used in the configuration of a task define which properties must (or should) be present for that task to work. Properties will be checked between the Gradle configuration and execution phases.
Step 4: Define prep tasks
Before the build can happen, tokenized files need to be copied to the right locations in the project. This can be done with a task like the following:
task prep() {
requiredProperties "applicationLogDir", "logfileName", "defaultLogLevel"
doFirst {
copy {
from templateDir
include "log4j.properties"
into srcResourceDir
filter(org.apache.tools.ant.filters.ReplaceTokens, tokens: project.ext.filterTokens)
}
}
}
compileJava.dependsOn << "prep"
The reason we copy into the source resource directory instead of the build directory is to keep IDEs happy. In general, you'll want to run the prep (or prepTest) task whenever properties or template files change, or to switch environments.
This plugin adds some closures to each task, which can be used to check for the presence of properties after configuration, but before any tasks are executed.
requiredProperty
requiredProperty "somePropertyName"
This closure throws a MissingPropertyException if the named property is not defined.
requiredProperties
requiredProperties "property1", "property2", ...
This closure throws a MissingPropertyException if any of the named properties are not defined
recommendedProperty
recommendedProperty "somePropertyName", "default File Text"
This closure is handy when there are properties that have defaults somewhere else. For example, the build file might define it, ir the application might be able to get it from a system file. It is most useful in alerting newer developers that something must be configured somewhere on their systems.
The closure checks to see if the given property is defined. If it is not, a warning message is displayed alerting the user that a default will be used, and if the defaultFile has been given, the message will include it so that the developer knows which file will be providing the default value.
recommendedProperties
recommendedProperties names: ["property1", "property2", ...], defaultFile: "default File Text"
This closure checks all the given property names, and prints a message if we're missing any.
If a property is set in the build.gradle file before the properties plugin is applied, and it happens to match a property in one of the 5 standard locations defined earlier, the build.gradle property's value will be overwritten. If you need to set a property in build.gradle, it is best to do it after the properties plugin is defined. Keep in mind that properties set in the build.gradle file will not be in the filterTokens.
There are a few ways to change the Gradle user home directory. The properties plugin uses a gradle property to use a different user home directory. The properties plugin uses the same property, but I haven't done much testing of that yet.
A special thank you to to Hans Dockter at Gradleware for showing me how to dynamically define the requiredProperty method and attach it to the right place in the Gradle build lifecycle.