- About
- Getting started
- Configuration
- Plugin setup
- Common plugin
- Tooling plugin
- Package plugin
- Bundle plugin
- Instance plugin
- Instance file structure
- Task instanceSetup
- Task instanceResetup
- Task instanceCreate
- Task instanceBackup
- Task instanceDestroy
- Task instanceUp
- Task instanceDown
- Task instanceRestart
- Task instanceReload
- Task instanceResolve
- Task instanceSatisfy
- Task instanceProvision
- Task instanceAwait
- Task instanceTail
- Environment plugin
- How to's
- Set AEM configuration properly for all / concrete project(s)
- Use lifecycle tasks
- Implement custom AEM tasks
- Instance services
- Defining CRX package via code then downloading and sharing it using external HTTP endpoint
- Calling AEM endpoints / making any HTTP requests
- Downloading CRX package from external HTTP endpoint and deploying it on desired AEM instances
- Working with content repository (JCR)
- Executing code on AEM runtime
- Controlling OSGi bundles, components and configurations
- Controlling workflows
- Running Docker image based tools
- Understand why there are one or two plugins to be applied in build script
- Work effectively on start and daily basis
- Filter instances to work with
- Know how properties are being expanded in instance or package files
- Customize convention for CRX package and OSGi bundle names and paths
- Known issues
- Compatibility
- Building
- Contributing
- License
Swiss army knife for AEM related automation. Incremental build which takes seconds, not minutes. Developer who does not loose focus between build time gaps. Extend freely your build system directly in project.
AEM developer - it's time to meet Gradle! You liked or used plugin? Don't forget to star this project on GitHub :)
Be inspired by watching live demo presented on official Sling adaptTo() 2018 conference.
Looking for dedicated version of plugin for Apache Sling? Check out Gradle Sling Plugin!
What is being done above by simply running super easy command sh gradlew
?
:aem:instanceSatisfy
-> checking for new dependent CRX packages to be deployed (in a lazy & fail-safe manner) that could arrive to our AEM instances automatically if somebody else add it to build configuration in the meantime.:aem:assembly:full:packageDeploy
-> building & deploying all-in-one CRX package to AEM instances in parallel, then awaiting for stable condition of AEM instances and built application.:aem:environmentClean
-> cleaning AEM dispatcher cache and restarting HTTPD service / Apache Web Server.:aem:environmentAwait
-> running health checks ensuring that all AEM instances / websites are responding correctly.
Build is incremental which guarantees optimized time every time regardless of build command used. Only changed parts of application are processed again:
- Dependent CRX packages are installed only when they are not already installed on particular AEM instances.
- CRX package is rebuild only when JCR content / files under jcr_root are changed.
- Java code is recompiled only when code in *.java files is changed.
- Front-end / Webpack build is run again only when code in *.scss and *.js etc files is changed.
Want to see it in action? Follow here!
- Automated complete AEM environment setup with virtualized AEM dispatcher and native AEM instances optimized for best development experience.
- Powerful AEM DSL scripting capabilities for performing JCR content migrations, managing AEM instances, integrating with Docker based tools.
- Advanced AEM instance(s) stability & health checking after CRX package deployment.
- Continuous AEM incident monitoring and interactive reporting (centralized log tailing of any AEM instances with no SSH).
- Easy parallel CRX package deployment to many remote group of instances.
- Hot reloading for HTTPD / AEM dispatcher configuration files with health checking (web browser usage not needed).
- Fail-safe dependent CRX packages installation from local and remote sources using various protocols (SMB / SSH / HTTP / custom).
- Fast JCR content synchronization from running AEM instances with advanced content normalization.
- Composing CRX package from multiple separate JCR content roots, bundles.
- Validating CRX package using seamless integration with OakPAL tool.
- All-in-one CRX packages generation (assemblies), vault filters merging etc.
- Easy OSGi bundle customization with BND tool embedded.
Gradle AEM Plugin is following strategy convention over configuration. When following built-in conventions about project structure & naming, then only minimal configuration is required. Still all features are fully configurable.
Major releases of plugin are introducing breaking changes. Build functionality is mostly covered, however build scripts need corrections.
Documentation for previous series:
- 6.2.0 (last in 6.x serie / with aem prefixes in task / property names)
- 5.1.4 (last supporting Groovy DSL)
Most effective way to experience Gradle AEM Plugin is to use Quickstart located in:
- AEM Single-Project Example - recommended for application/library development,
- AEM Multi-Project Example - recommended for long-term project development,
- AEM Boot - only booting local AEM instances and AEM dispatcher automatically. Useful when building CRX packages is covered separately, e.g by Maven & Content Package Maven Plugin.
The only software needed on your machine to start using plugin is Java 8 or newer (also to setup local native AEM instances). Optionally, Docker is needed (when using automatic AEM dispatcher setup).
As a build command, it is recommended to use Gradle Wrapper (gradlew
) instead of locally installed Gradle (gradle
) to easily have same version of build tool installed on all environments. Only at first build time, wrapper will be automatically downloaded and installed, then reused.
Released versions of plugin are available on Bintray, so that this repository needs to be included in buildscript section.
Configuration below assumes building and deploying CRX packages to AEM instance(s) via command: gradlew packageDeploy
.
File buildSrc/build.gradle.kts:
repositories {
jcenter()
}
dependencies {
implementation("com.cognifide.gradle:aem-plugin:9.0.9")
}
File build.gradle.kts:
plugins {
id("com.cognifide.aem.bundle") // or 'package' for JCR content only
}
group = "com.company.aem"
Configuration below assumes building and deploying on AEM instance(s) via command: gradlew
(default tasks will be used).
plugins {
id("com.cognifide.aem.bundle")
id("com.cognifide.aem.instance")
id("com.cognifide.aem.environment")
id("org.jetbrains.kotlin.jvm") // or any other like 'java' to compile OSGi bundle
}
group = "com.company.aem"
version = "1.0.0"
defaultTasks(":instanceSatisfy", ":instanceProvision", ":packageDeploy")
aem {
`package` { // built CRX package options
contentDir = project.file("src/main/content")
installPath = when {
project == project.rootProject -> "/apps/${project.rootProject.name}/install"
else -> "/apps/${project.rootProject.name}/${projectName}/install"
}
nodeTypesSync("PRESERVE_AUTO")
validator {
enabled = props.boolean("package.validator.enabled") ?: true
verbose = props.boolean("package.validator.verbose") ?: true
planName = props.string("package.validator.plan") ?: "default-plan.json"
severity("MAJOR")
}
// ...
}
instance { // AEM instances to work with
local("http://localhost:4502") // local-author
local("http://localhost:4503") // local-publish
remote("http://192.168.100.101:4502", "int-author")
remote("http://192.168.100.101:4503", "int-publish")
// etc
http { // allows to customize HTTP connection to AEM instances
connectionTimeout = props.int("instance.http.connectionTimeout") ?: 30000
connectionRetries = props.boolean("instance.http.connectionRetries") ?: true
connectionIgnoreSsl = props.boolean("instance.http.connectionIgnoreSsl") ?: true
proxyHost = props.string("instance.http.proxyHost")
proxyPort = props.int("instance.http.proxyPort")
proxyScheme = props.string("instance.http.proxyScheme")
}
}
localInstance { // config for AEM instances to be created on local file system
quickstart {
jarUrl = props.string("localInstance.quickstart.jarUrl")
licenseUrl = props.string("localInstance.quickstart.licenseUrl")
}
backup {
uploadUrl = props.string("localInstance.backup.uploadUrl")
downloadUrl = props.string("localInstance.backup.downloadUrl")
}
install {
files { // CRX packages and OSGi bundles to be pre-installed on created AEM instances
download("http://.../package.zip") // CRX package downloaded over HTTP
resolve("group:name:version") // OSGi bundle from Maven repository
}
}
init { // hook called once in scope of instance just created and up first time
logger.info("Initializing instance '$name'")
sync {
// ...
}
}
rootDir = props.string("localInstance.root")
// ...
}
environment { // config for AEM environment running on Docker
rootDir = props.string("environment.rootDir")?.let { project.file(it) } ?: projectMain.file(".aem/environment")
hosts { // domains to be appended to hosts file automatically
author("http://author.example.com")
publish("http://demo.example.com") { tag("test") }
publish("http://example.com") { tag("live") }
other("http://dispatcher.example.com")
}
docker { // Docker specific configuration
containers {
"httpd" { // control container described in 'docker-compose.yml.peb'
resolve {
resolveFiles {
download("http://download.macromedia.com/dispatcher/download/dispatcher-apache2.4-linux-x86_64-4.3.2.tar.gz").use {
copyArchiveFile(it, "**/dispatcher-apache*.so", file("modules/mod_dispatcher.so"))
}
}
ensureDir("cache", "logs")
}
up {
ensureDir("/usr/local/apache2/logs", "/opt/aem/dispatcher/cache/content/example/demo", "/opt/aem/dispatcher/cache/content/example/live")
execShell("Starting HTTPD server", "/usr/local/apache2/bin/httpd -k start")
}
reload {
cleanDir("/opt/aem/dispatcher/cache/content/example/demo", "/opt/aem/dispatcher/cache/content/example/live")
execShell("Restarting HTTPD server", "/usr/local/apache2/bin/httpd -k restart")
}
dev {
watchConfigDir("conf")
}
}
}
}
healthChecks { // checks (e.g GET requests) verifying running Docker containers like HTTPD
url("Live site", "http://example.com/en-us.html") { containsText("English US") }
url("Demo site", "http://demo.example.com/en-us.html") { containsText("English US") }
url("Author module 'Sites'", "http://author.example.com/sites.html") {
options { basicCredentials = authorInstance.credentials }
containsText("Sites")
}
}
}
fileTransfer { // config for resolving CRX packages, AEM Quickstart files and backups using HTTP/SFTP/SMB
sftp {
user = props.string("fileTransfer.sftp.user")
password = props.string("fileTransfer.sftp.password")
}
smb {
user = props.string("fileTransfer.smb.user")
password = props.string("fileTransfer.smb.password")
domain = props.string("fileTransfer.smb.domain")
}
http {
client {
basicUser = props.string("fileTransfer.http.user")
basicPassword = props.string("fileTransfer.http.password")
}
}
credentials("foo", "bar", "company-net") // shorthand to set all user / password pairs and domain above
}
tasks {
bundleCompose { // customizing OSGi bundle, manifest etc
javaPackage = "com.company.example.aem"
// ...
}
packageCompose { // customizing built CRX package
fromProject(":core")
fromProject(":config")
baseName = 'example-for-changing-zip-name'
vaultDefinition { // place for overriding CRX Package / Vault properties, defining hooks
// ...
}
}
instanceProvision {
step("enable-crxde") {
description = "Enables CRX DE"
condition { once() && instance.environment != "prod" }
action {
sync {
osgiFramework.configure("org.apache.sling.jcr.davex.impl.servlets.SlingDavExServlet", mapOf(
"alias" to "/crx/server"
))
}
}
}
// ...
}
instanceSatisfy { // customizing CRX packages to be deployed as dependencies before built AEM application
packages {
"my-package" { download("http://.../my-package-1.0.0.zip") }
"my-bundle" { resolve("group:name:version") } // will be wrapped on-the-fly to CRX package
}
}
// ... and all other tasks
}
}
To see all available options and actual documentation, please follow to:
aem
- AemExtensionpackage
- PackageOptionsinstance
- InstanceOptionslocalInstance
- LocalInstanceManagerenvironment
- EnvironmentfileTransfer
- FileTransferManagerbundleCompose
- BundleComposepackageCompose
- PackageComposeinstanceProvision
- InstanceProvisioninstanceSatisfy
- InstanceSatisfy...
- other tasks in similar way.
plugins {
id("com.cognifide.aem.common")
}
Applied transparently by other plugins. Provides AEM extension to build script / AEM Gradle DSL which consists of instance definitions, common configuration, methods for controlling local instances and virtualized environment.
It does not provide any tasks. Apply other plugins to have tasks or implement own.
- Instance name is a combination of ${environment}-${id} e.g local-author, integration-publish etc.
- Instance type indicates physical type of instance and could be only: local and remote. Local means that instance could be created by plugin automatically under local file system.
- Instance id is an instance purpose identifier and must start with prefix author or publish. Sample valid names: author, author1, author2, author-master and publish, publish1 publish2 etc.
- Only instances defined as local are considered in command
instanceSetup
,instanceCreate
,instanceUp
etc (that comes fromcom.cognifide.aem.instance
plugin). - All instances defined as local or remote are considered in commands CRX package deployment related like
instanceSatisfy
,packageDeploy
,packageUpload
,packageInstall
etc.
Instances could be defined in two ways, via:
- file
gradle.properties
- recommended approach, by properties convention. - build script - dynamic & more customizable approach.
The configuration could be specified through gradle.properties file using dedicated syntax.
instance.$ENVIRONMENT-$ID.$PROP_NAME=$PROP_VALUE
Part | Possible values | Description |
---|---|---|
$ENVIRONMENT |
local , int , stg etc |
Environment name. |
$ID |
author , publish , publish2 , etc |
Combination of AEM instance type and semantic suffix useful when more than one of instance of same type is being configured. |
$PROP_NAME=$PROP_VALUE |
Local instances: httpUrl=http://admin:admin@localhost:4502 type=local (or remote)password=foo runModes=nosamplecontent jvmOpts=-server -Xmx2048m -XX:MaxPermSize=512M -Djava.awt.headless=true , startOpts=... debugPort=24502 .Remote instances: httpUrl , type , user , password . |
Run modes, JVM opts and start opts should be comma delimited. |
Default remote instances defined via properties (below lines are optional):
instance.local-author.httpUrl=http://localhost:4502
instance.local-publish.httpUrl=http://localhost:4503
Example for defining multiple remote instances (that could be filtered):
instance.int-author.httpUrl=http://author.aem-integration.company.com
instance.int-publish.httpUrl=http://aem-integration.company.com
instance.stg-author.httpUrl=http://author.aem-staging.company.com
instance.stg-publish.httpUrl=http://aem-staging.company.com
Example for defining remote instance with credentials separated:
instance.test-author.httpUrl=http://author.aem-integration.company.com
instance.test-author.user=foo
instance.test-author.password=bar
Example for defining remote instance with credentials details included in URL:
instance.test-author.httpUrl=http://foo:[email protected]
Example for defining local instances (created on local file system):
instance.local-author.httpUrl=http://localhost:4502
instance.local-author.type=local
instance.local-author.runModes=nosamplecontent
instance.local-author.jvmOpts=-server -Xmx1024m -XX:MaxPermSize=256M -Djava.awt.headless=true
instance.local-publish.httpUrl=http://localhost:4503
instance.local-publish.type=local
instance.local-publish.runModes=nosamplecontent
instance.local-publish.jvmOpts=-server -Xmx1024m -XX:MaxPermSize=256M -Djava.awt.headless=true
Notice! Remember to define also AEM source files.
Example usage below. The commented value is an effective instance name.
aem {
instance {
local("http://localhost:4502") // local-author
local("http://localhost:4502") { // local-author
password = "admin"
id = "author"
debugPort = 14502
}
local("http://localhost:4503") // local-publish
local("http://localhost:4503") { // local-publish
password = "admin"
id = "publish"
debugPort = 14503
}
remote("http://192.168.10.1:4502") { // integration-author1
user = "user1"
password = "password2"
environment = "integration"
id = "author1"
}
remote("http://192.168.10.1:8080") { // integration-author2
user = "user1"
password = "password2"
environment = "integration"
id = "author2"
}
remote("http://192.168.10.2:4503") { // integration-publish1
user = "user2"
password = "password2"
environment = "integration"
id = "publish1"
}
remote("http://192.168.10.2:8080") { // integration-publish2
user = "user2"
password = "password2"
environment = "integration"
id = "publish2"
}
}
}
Remember to encode instance credentials (user & password) when passing it via properties: instance.$INSTANCE_NAME.httpUrl
or instance.list=[$INSTANCE_HTTP_URL1,$INSTANCE_HTTP_URL2,...]
.
For example, let's assume that instance is created by instance plugin using following properties:
instance.local-author.httpUrl=http://localhost:4502
instance.local-author.type=local
instance.local-author.password=gxJMge@6F5ZV#s9j
Password is generated by e.g strong password generator so it has special URL characters.
Then to be able to use this created instance but remotely and via instance.list
property on CI, then it should look as below:
sh gradlew :aem:assembly:full:packageDeploy -Pinstance.list=[http://admin:gxJMge%406F5ZV%[email protected]:4502]
Notice that when password is specified as separate instance.$INSTANCE_NAME.password
property it does not need to be encoded.
Otherwise, password value must be encoded by e.g online URL encoder.
plugins {
id("com.cognifide.aem.tooling")
}
Applied transparently by package and bundle plugins. Provides tooling related tasks like sync
, rcp
, debug
etc.
Check out then clean JCR content.
JCR content to be synchronized from AEM instance repository is determined by workspace filter file located at path src/main/content/META-INF/vault/sync.xml. However, when this file does not exist, plugin as a fallback will use same file used when composing CRX package which is src/main/content/META-INF/vault/filter.xml.
To exclude some files from synchronization, simply copy filter.xml file and create file named sync.xml. Then add exclusion in a following way:
<filter root="/content/dam/example">
<exclude pattern=".*\.gif"/>
</filter>
Cleaning assumes advanced JCR content normalization to minimize changes visible in VCS after each synchronization.
- unwanted JCR properties removal (with path based inclusion / exclusion rules),
- unwanted JCR mixin types removal,
- unwanted files removal,
- unused XML namespaces removal,
- flattening files (renaming e.g _cq_dialog/.content.xml to _cq_dialog.xml),
- preserving state of parent files for each Vault filter root (by backup mechanism),
- hooks for custom cleaning rules / processing .content.xml files.
aem {
tasks {
sync {
cleaner {
filesDotContent = {
include("**/.content.xml")
}
filesDeleted = {
include(
"**/.vlt",
"**/.vlt*.tmp",
"**/install/*.jar"
)
}
filesFlattened = {
include(
"**/_cq_design_dialog/.content.xml",
"**/_cq_dialog/.content.xml",
"**/_cq_htmlTag/.content.xml",
"**/_cq_template/.content.xml"
)
}
propertiesSkipped = listOf(
pathRule("jcr:uuid", listOf("**/home/users/*", "**/home/groups/*")),
pathRule("cq:lastModified*", listOf("**/content/experience-fragments/*")),
"jcr:lastModified*",
"jcr:created*",
"jcr:isCheckedOut",
"cq:lastReplicat*",
"dam:extracted",
"dam:assetState",
"dc:modified",
"*_x0040_*"
)
mixinTypesSkipped = listOf(
"cq:ReplicationStatus",
"mix:versionable"
)
namespacesSkipped = true
parentsBackupEnabled = true
parentsBackupSuffix = ".bak"
lineProcess = { file, line -> normalizeLine(file, line) }
contentProcess = { file, lines -> normalizeContent(file, lines) }
}
}
}
}
Cleaning could also ensure that AEM renditions will be never saved in VCS. Also any additional properties could be cleaned. For such cases, see configuration below:
aem {
tasks {
sync {
cleaner {
propertiesSkipped += listOf(
pathRule("dam:sha1", listOf(), listOf("**/content/dam/*.svg/*")),
pathRule("dam:size", listOf(), listOf("**/content/dam/*.svg/*")),
"cq:name",
"cq:parentPath",
"dam:copiedAt",
"dam:parentAssetID",
"dam:relativePath"
)
mixinTypesSkipped += listOf(
pathRule("dam:Thumbnails", listOf(), listOf("**/content/dam/*"))
)
filesDeleted = {
include(
"**/.vlt",
"**/.vlt*.tmp",
"**/install/*.jar",
"**/_jcr_content/folderThumbnail.dir/*",
"**/_jcr_content/folderThumbnail/*",
"**/_jcr_content/folderThumbnail",
"**/_jcr_content/renditions/**"
)
exclude(
"**/_jcr_content/renditions/original.dir/.content.xml",
"**/_jcr_content/renditions/original"
)
}
}
}
}
}
Available transfer types: package_download (default) and vlt_checkout.
gradlew :site.demo:sync -Psync.type=vlt_checkout
When transfer type is set to package_download then it is also possible to...
Download package only without extracting:
gradlew :site.demo:sync -PpackageDownload.extract=false
Download, delete all previous JCR root contents then extract fresh content:
gradlew :site.demo:sync -Pforce
Available mode types: copy_and_clean (default), clean_only and copy_only.
gradlew :site.demo:sync -Psync.mode=clean_only
gradlew :site.demo:sync -Pfilter.path=custom-filter.xml
gradlew :site.demo:sync -Pfilter.path=src/main/content/META-INF/vault/custom-filter.xml
gradlew :site.demo:sync -Pfilter.path=C:/aem/custom-filter.xml
gradlew :site.demo:sync -Pfilter.roots=[/etc/tags/example,/content/dam/example]
Copy JCR content from one instance to another. Sample usages below.
-
Using predefined instances with multiple different source and target nodes:
gradlew :rcp -Prcp.source.instance=int-author -Prcp.target.instance=local-author -Prcp.paths=[/content/example-demo=/content/example,/content/dam/example-demo=/content/dam/example]
-
Using predefined instances with multiple same source and target nodes:
gradlew :rcp -Prcp.source.instance=stg-author -Prcp.target.instance=int-author -Prcp.paths=[/content/example,/content/example2]
Right side of assignment could skipped if equals to left (same path on both source & target instance).
-
Using predefined instances with source and target nodes specified in file:
gradlew :rcp -Prcp.source.instance=int-author -Prcp.target.instance=local-author -Prcp.pathsFile=paths.txt
File format:
sourcePath1=targetPath1 sameSourceAndTargetPath1 sourcePath2=targetPath2 sameSourceAndTargetPath2
-
Using dynamically defined instances:
gradlew :rcp -Prcp.source.instance=http://user:[email protected]:4502 -Prcp.target.instance=http://user:[email protected]:4502 -Prcp.paths=[/content/example-demo=/content/example]
Keep in mind, that copying JCR content between instances, could be a trigger for running AEM workflows like DAM Update Asset which could cause heavy load on instance. Consider disabling AEM workflow launchers before running this task and re-enabling after.
RCP task is internally using Vault Remote Copy which requires bundle Apache Sling Simple WebDAV Access to repositories (org.apache.sling.jcr.webdav) present in active state on instance.
Execute any JCR File Vault command.
For instance, to reflect rcp
functionality, command below could be executed:
gradlew :site.demo:sync -Pvlt.command='rcp -b 100 -r -u -n http://admin:admin@localhost:4502/crx/-/jcr:root/content/dam/example http://admin:admin@localhost:4503/crx/-/jcr:root/content/dam/example'
For more details about available parameters, please visit VLT Tool documentation.
While using task sync
be aware that Gradle requires to have working directory with file build.gradle.kts in it, but Vault tool can work at any directory under jcr_root. To change working directory for Vault, use property aem.vlt.path
which is relative path to be appended to jcr_root for project task being currently executed.
Dumps effective AEM build configuration of concrete project to JSON file.
When command below is being run (for root project :
):
gradlew :debug
Then file at path build/aem/debug/debug.json with content below is being generated:
{
"buildInfo": {
"plugin": {
"pluginVersion": "7.0.0",
"gradleVersion": "5.4"
},
"gradle": {
"version": "5.4.1",
"homeDir": ".../gradle-5.4.1"
},
"java": {
"version": "1.8",
"homeDir": ".../jdk1.8.0_121/jre"
}
},
"projectInfo": {
"displayName": "project ':aem:assembly:full'",
"path": ":aem:assembly:full",
"name": "full",
"dir": ".../gradle-aem-multi/aem/assembly/full"
},
"packageDeployed": {
"local-author": {
"group": "example",
"name": "example.full",
"version": "1.0.0-SNAPSHOT",
"path": "/etc/packages/example/example.full-1.0.0-SNAPSHOT.zip",
"downloadName": "example.full-1.0.0-SNAPSHOT.zip",
"lastUnpacked": 1558622734756,
"installed": true
},
// ...
},
"aem": {
"projectPrefixes": [
"aem.",
"aem-",
"aem_"
],
"env": "local",
"lineSeparator": "LF",
"fileTransfer": {
"http": {
"enabled": true,
"client": {
"baseUrl": "",
"basicUser": null,
"basicPassword": null,
"authorizationPreemptive": false,
"connectionTimeout": 30000,
"connectionIgnoreSsl": true,
"connectionRetries": true,
"responseChecks": true
},
"name": "httpd",
"protocols": [
"http://*",
"https://*"
]
},
"sftp": {
"enabled": true,
"user": null,
"password": null,
"timeout": 60000,
"name": "sftp",
"protocols": [
"sftp://*"
]
},
// ...
},
"packageOptions": {
"rootDir": ".../gradle-aem-multi/aem/assembly/full/src/main/content",
"snapshots": [],
"metaCommonRootDir": ".../gradle-aem-multi/aem/gradle/package/META-INF",
"installPath": "/apps/example/full/install",
"installRepository": true,
"errors": [
"javax.jcr.nodetype.*Exception",
"org.apache.jackrabbit.oak.api.*Exception",
"org.apache.jackrabbit.vault.packaging.*Exception",
"org.xml.sax.*Exception"
],
"responseBuffer": 4096
},
"instanceOptions": {
"defined": {
"local-author": {
"properties": {},
"httpUrl": "http://localhost:4502",
"user": "admin",
"password": "*****",
"id": "author",
"environment": "local",
"debugPort": 14502,
"name": "local-author",
"type": "AUTHOR",
"httpPort": 4502,
"runModes": "author,local",
"startOpts": "",
"jvmOpts": "-Xdebug -Xrunjdwp:transport=dt_socket,address=14502,server=y,suspend=n -server -Xmx2048m -XX:MaxPermSize=512M -Djava.awt.headless=true"
}
// ...
}
},
"localInstanceManager" : {
"rootDir" : "*/minimal/.aem/instance",
"source" : "AUTO",
"overridesDir" : ".../gradle-aem-multi/gradle/instance",
"expandFiles" : [
"**/start.bat",
"**/stop.bat",
"**/start",
"**/stop"
],
"expandProperties" : { },
"quickstart": {
"jarUrl": null,
"licenseUrl": null
},
"backup": {
"uploadUrl": null,
"downloadUrl": null,
"downloadDir": ".../gradle-aem-multi/build/aem/instanceBackup/remote"
}
},
"environment": {
"rootDir": ".../gradle-aem-multi/aem/.aem/environment",
"hosts": {
"defined": []
},
"dispatcherModuleFile": ".../gradle-aem-multi/aem/.aem/environment/distributions/mod_dispatcher.so",
"configDir": ".../gradle-aem-multi/aem/gradle/environment",
"created": true,
},
"notifier": {
"enabled": true
},
"tasks": {
"bundles": []
},
"projectName": "full",
"baseName": "example.full",
"configCommonDir": ".../gradle-aem-multi/aem/gradle",
"configDir": ".../gradle-aem-multi/aem/assembly/full/gradle",
"javaPackages": [
"com.company.example.aem.common",
"com.company.example.aem.sites"
]
}
}
plugins {
id("com.cognifide.aem.package")
}
Should be applied to all projects that are composing CRX packages from JCR content only.
Provides CRX package related tasks: packageCompose
, packageDeploy
, packageActivate
, packagePurge
etc.
Inherits from Common Plugin.
Compose CRX package from JCR content and bundles.
Inherits from task ZIP.
aem {
tasks {
packageCompose {
duplicatesStrategy = DuplicatesStrategy.WARN
baseName = baseName
contentDir = packageOptions.rootDir
bundlePath = packageOptions.installPath
bundleRunMode = null
metaDefaults = true
vaultDefinition {
properties = mapOf(
"acHandling" to "merge_preserve",
"requiresRoot" to false
)
name = baseName
group = if (project == project.rootProject) {
project.group.toString()
} else {
project.rootProject.name
}
version = project.version.toString()
}
validator {
enabled = true
verbose = true
severity("MAJOR")
planName = "default-plan.json"
}
fileFilter {
expanding = true
expandFiles = listOf(
"**/META-INF/*.xml",
"**/META-INF/*.MF",
"**/META-INF/*.cnd"
)
excluding = true
excludeFiles = listOf(
"**/.gradle",
"**/.git",
"**/.git/**",
"**/.gitattributes",
"**/.gitignore",
"**/.gitmodules",
"**/.vlt",
"**/.vlt*.tmp",
"**/node_modules/**",
"jcr_root/.vlt-sync-config.properties"
)
bundleChecking = true
}
merging {
vaultFilters = true
}
fromConvention = true
}
}
}
Built package is validated using OakPAL tool. It helps preventing situation when CRX Package Manager reports message like Package installed with errors. Also gives immediate feedback for mistakes taken when using copy-pasting technique, forgetting about using entities ('&' instead of '&), missing XML namespaces and much more.
By default, nothing need to be configured so that default plan will be in use. However, more detailed checks could be provided by configuring base artifact called Opear File containing OakPAL checks.
It could be done via following snippet (ACS AEM Commons OakPAL Checks as example):
aem {
`package` {
validator {
base("com.adobe.acs:acs-aem-commons-oakpal-checks:4.3.4")
}
}
}
To use custom checks, only left thing is to choose them in custom plan. Simply create file at path [aem/]gradle/package/OAKPAL_OPEAR/default-plan.json
{
"checklists": [
"net.adamcin.oakpal.core/basic",
"acs-commons-integrators",
"content-class-aem65"
],
"installHookPolicy": "SKIP"
}
See plans documentationand examples provided by OakPAL tool.
There could be many plan files created. To validate CRX package using different plan than default one, specify property -package.validator.planName=<file>
.
Notice that running OakPAL requires to have included in CRX package up-to-date node type definitions coming from AEM instance. Such definitions could be manually downloaded using CRXDE Lite interface (Tools / Export Node Type) and put inside CRX package. After installing some dependent CRX packages, the list of exported node types may change.
To keep it up-to-date, plugin is synchronizing node types from one of available AEM instances automatically. Synchronized file containing node types, later used when building CRX packages is placed at path [aem/]gradle/package/nodetypes.export.cnd. Remember to save this file in VCS, so that CRX package validation will not fail on e.g CI server where AEM instance could be not available.
To configure node types synchronization behavior, use property package.nodeTypesSync=<option>
. Available options: when_missing (default, sync only if file nodetypes.export.cnd does not exist), always, never.
To disable using fallback node types, simply set property package.nodeTypesFallback=false
(might be useful to explicitly show error when node types are not yet synchronized or provided manually).
Package validation report is saved at path relative to project building CRX package: build/aem/packageCompose/OAKPAL_OPEAR/report.json.
Use dedicated task method named fromJar
, for example:
aem {
tasks {
packageCompose {
fromJar("com.github.mickleroy:aem-sass-compiler:1.0.1")
}
}
}
For reference, see usage above in AEM Multi-Project Example.
Use dedicated task method named fromZip
, For example:
aem {
tasks {
packageCompose {
fromZip("com.adobe.cq:core.wcm.components.all:2.4.0")
fromZip("com.adobe.cq:core.wcm.components.examples:2.4.0")
}
}
}
Let's assume following project structure:
- aem/build.gradle.kts (project
:aem
, no source files at all) - aem/sites/build.gradle.kts (project
:aem:sites
, JCR content and OSGi bundle) - aem/common/build.gradle.kts (project
:aem:common
, JCR content and OSGi bundle) - aem/site.live/build.gradle.kts (project
:aem:site.live
, JCR content only) - aem/site.demo/build.gradle.kts (project
:aem:site.demo
, JCR content only)
File content of aem/build.gradle.kts:
plugins {
id("com.cognifide.aem.package")
}
aem {
tasks {
packageCompose {
fromProject(":aem:sites")
fromProject(":aem:common")
fromProjects(":aem:site.*")
}
}
}
When building via command gradlew :aem:build
, then the effect will be a CRX package with assembled JCR content and OSGi bundles from projects: :aem:sites
, :aem:common
, :aem:site.live
, :aem:site.demo
.
Gradle AEM Plugin is configured in a way that project can have:
- JCR content,
- source code to compile OSGi bundle,
- both.
By distinguishing fromProject
, fromBundle
or fromCompose
there is ability to create any assembly CRX package with content of any type without restructuring the project.
When using fromProject
there is an ability to pass lambda to customize options like bundlePath
, bundleRunMode
, decide to include only JCR contents, only bundles and more.
However, one rule must be kept while developing a multi-module project: all Vault filter roots of all projects must be exclusive. In general, they are most often exclusive, to avoid strange JCR installer behaviors, but sometimes exceptional workspace filter rules are being applied like mode="merge"
etc.
In exactly the same way as it works for instance files, properties can be expanded inside metadata files of package being composed.
Related configuration:
aem {
tasks {
packageCompose {
fileFilter {
expandProperties = mapOf(
"organization" to "Company"
)
expandFiles = listOf(
"**/META-INF/*.xml",
"**/META-INF/*.MF",
"**/META-INF/*.cnd"
)
}
}
}
}
Predefined expandable properties:
aem
- AemExtension object,definition
- VltDefinition object,rootProject
- project with directory in which settings.gradle is located,project
- current project.
This feature is especially useful to generate valid META-INF/properties.xml file, below template is used by plugin by default:
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<!DOCTYPE properties SYSTEM "http://java.sun.com/dtd/properties.dtd">
<properties>
{% if definition.description is not empty %}
<comment>{{definition.description}}</comment>
<entry key="description">{{definition.description}}</entry>
{% endif %}
<entry key="group">{{definition.group}}</entry>
<entry key="name">{{definition.name}}</entry>
<entry key="version">{{definition.version}}</entry>
{% if definition.createdBy is not empty %}
<entry key="createdBy">{{definition.createdBy}}</entry>
{% endif %}
{% for e in definition.properties %}
<entry key="{{e.key}}">{{e.value | raw}}</entry>
{% endfor %}
</properties>
Also file nodetypes.cnd is dynamically expanded from template to generate file containing all node types from all sub packages being merged into assembly package.
Each JAR file in separate hooks directory will be combined into single directory when creating assembly package.
Processes CRX package metadata - combines default files provided by plugin itself with overridden ones. Also responsible for synchronizing Vault node types consumed later by CRX package validation in the end of compose task. Covers extracted logic being initially a part of compose task. Reason of separation is effectively better Gradle caching.
Upload & install CRX package into AEM instance(s). Primary, recommended form of deployment. Optimized version of packageUpload packageInstall
.
Simply use generic approach for filtering instances to work with.
Add any of below command line parameters to customize CRX package deployment behavior:
-Ppackage.deploy.awaited=false
- disable stability & health checks after deploying CRX package.-Ppackage.deploy.distributed=true
- use alternative form of deployment. At first, deploys CRX package to author instances, then triggers replication of CRX package so that it will be installed also on publish instances.-Ppackage.deploy.uploadForce=false
- disable force installation (by default even unchanged CRX package is forced to be reinstalled)-Ppackage.deploy.installRecursive=false
- disable automatic installation of subpackages located inside CRX package being deployed.-Ppackage.deploy.uploadRetry=n
- customize number of retries being performed after failed CRX package upload.-Ppackage.deploy.installRetry=n
- customize number of retries being performed after failed CRX package install.-Ppackage.deploy.workflowToggle=[id1=true,id2=false,...]
- temporarily enable or disable AEM workflows during deployment e.g when CRX package contains generated DAM asset renditions so that regeneration could be avoided and deploy time reduced. For example:-Ppackage.deploy.workflowToggle=[dam_asset=false]
. Workflow ID dam_asset is a shorthand alias for all workflows related with DAM asset processing.
Upload composed CRX package into AEM instance(s).
Delete uploaded CRX package from AEM instance(s).
Install uploaded CRX package on AEM instance(s).
Uninstall uploaded CRX package on AEM instance(s).
To prevent data loss, this unsafe task execution must be confirmed by parameter -Pforce
.
Fail-safe combination of packageUninstall
and packageDelete
.
To prevent data loss, this unsafe task execution must be confirmed by parameter -Pforce
.
Replicate installed CRX package to other AEM instance(s).
Should be applied to all projects that are composing CRX packages from both OSGi bundle being built and optionally JCR content.
Inherits from Package Plugin.
OSGi bundle jar base name and CRX package base name is computed from:
- for subproject of multi project build -
${project.rootProject.name}.${project.name}
, - for single project build -
${project.name}
(just root project name).
Value of bundle javaPackage
is computed from ${project.group}.${project.name}
.
settings.gradle.kts
includeProject(":aem:app.core")
aem/build.gradle.kts
allprojects {
group = "com.company.example.aem"
}
Then below section is absolutely redundant:
aem/sites/build.gradle.kts
aem {
tasks {
bundleCompose {
javaPackage = "${project.group}.${project.name}" // "com.company.example.aem.sites"
}
}
}
Use one of dedicated methods bundleExportEmbed
or bundlePrivateEmbed
.
These methods are above bundleCompose
section as of dependency management is task agnostic and these methods
are configuring compileOnly
dependency and setting correct OSGi manifest entry at once (simplification).
aem {
tasks {
bundleExportEmbed("group:name:version", "com.group.name")
// or
bundlePrivateEmbed("group:name:version", "com.group.name")
}
}
For the reference, see usage in AEM Multi-Project Example.
Plugin by default covers generation of few attributes by convention:
Bundle-Name
will grab value fromproject.description
Bundle-SymbolicName
will grab value fromjavaPackage
(from sectionaem.tasks.bundle
)Bundle-Activator
will grab value fromjavaPackage.activator
assuming that activator is an existing file named Activator or BundleActivator under main source set.Sling-Model-Packages
will grab value fromjavaPackage
Export-Package
will grab value fromjavaPackage
.
This values population behavior could be optionally disabled by bundle parameter attributesConvention = false
.
Regardless if this behavior is enabled or disabled, all of values are overiddable e.g:
aem {
tasks {
bundle {
displayName = 'My Bundle"
symbolicName = "com.company.aem.example.common"
slingModelPackages = "com.company.aem.example.common.models"
exportPackage("com.company.aem.example.common")
}
}
}
Sometimes BND tool could generate Import-Package directive that will import too many OSGi classes to be available on bundle class path. Especially when we are migrating non-OSGi dependency to OSGi bundle (because of transitive class dependencies).
To prevent that we could generate own manifest entry that will prevent importing optional classes.
For instance:
aem {
tasks {
bundle {
excludePackages("org.junit", "org.mockito")
importPackages("!org.junit", "!org.mockito", "*") // alternatively
}
}
}
Installs OSGi bundle on AEM instance(s).
Available options:
-Pbundle.deploy.awaited=false
- disable stability & health checks after deploying OSGi bundle.-Pbundle.deploy.retry=n
- customize number of retries being performed after failed OSGi bundle installation.
Uninstalls OSGi bundle on AEM instance(s).
plugins {
id("com.cognifide.aem.instance")
}
Provides instance related tasks: instanceUp
, instanceDown
, instanceSetup
, instanceBackup
, instanceAwait
, instanceSetup
, instanceCreate
etc.
Allows to create & customize AEM instances on local file system and control them. Also provides support for automated backups and restoring.
Should be applied only at root project / only once within whole build.
Inherits from Common Plugin.
By default, instance file are stored directly in project, under so called main AEM module usually named aem. Ensure having directory aem/.instance ignored in VCS and excluded from indexing by IDE.
Performs initial setup of local AEM instance(s). Automated version of instanceCreate instanceUp instanceSatisfy packageDeploy
.
Combination of instanceDown instanceDestroy instanceSetup
. Allows to quickly back to initial state of local AEM instance(s).
To prevent data loss, this unsafe task execution must be confirmed by parameter -Pforce
.
Create AEM instance(s) at local file system. Extracts crx-quickstart from downloaded JAR and applies configuration according to instance definitions.
To create instances from scratch, specify:
localInstance.quickstart.jarUrl=[protocol]://[user]:[password]@[host]/[path]/cq-quickstart.jar
localInstance.quickstart.licenseUrl=[protocol]://[user]:[password]@[host]/[path]/license.properties
To create instances from local backups, firstly create instances from scratch, then run backup task. After creating backup, at any time, instances could be restored to previous state by running resetup task. Nothing need to be configured by default.
To create instances from remote backups see section work with remote instance backups.
By default plugin tries to automatically find most recent backup from all available sources.
However to e.g avoid creating instances from the scratch accidentally, source mode can be adjusted by specifying property localInstance.source
.
Available values:
auto
- Create instances from most recent backup (remote or local) or fallback to creating from the scratch if there is no backup available. Default mode.scratch
- Force creating instances from the scratch.backup_any
- Force using any backup available at local or remote source.backup_remote
- Force using backup available at remote source (specified aslocalInstance.backup.downloadUrl
orlocalInstance.backup.uploadUrl
).backup_local
- Force using local backup (created by taskinstanceBackup
).
When mode is different than scratch
, then backup ZIP file selection rule could be adjusted:
aem {
localInstance {
backup {
selector = { // default implementation below
val name = props.string("localInstance.backup.name") ?: ""
when {
name.isNotBlank() -> firstOrNull { it.fileEntry.name == name }
else -> firstOrNull()
}
}
}
}
}
Notice that, default selector assumes that most recent backup will be selected.
Ordering by file name including timestamp then local backups precedence when backup is available on both local & remote source.
Still, backup selector could select exact backup by name when property localInstance.backup.name
is specified.
Use dedicated section:
aem {
localInstance {
install {
files {
download("http://my-company.com/aem/packages/my-package.zip")
}
}
}
}
Files section works in a same way as in instance satisfy task. For more details see AEM File Install Provider documentation.
Plugin allows to override or provide extra files to local AEM instance installations. This behavior is controlled by:
aem {
localInstance {
rootDir = props.string("localInstance.root")?.let { project.file(it) } ?: projectMain.file(".aem/instance")
overridesDir = File(configCommonDir, LocalInstance.FILES_PATH)
expandProperties = mapOf()
expandFiles = listOf(
"**/*.properties",
"**/*.sh",
"**/*.bat",
"**/*.xml",
"**/start",
"**/stop"
)
}
}
Properties:
- rootDir determines where AEM instance files will be extracted on local file system.
- overridesDir determines project location that holds extra instance files that will override plugin defaults (start / stop scripts) and / or extracted AEM files.
- expandProperties is a place for defining custom properties that can be expanded in AEM instance files.
- expandFiles specifies which AEM instance files have an ability to use expandable properties inside.
Predefined expandable properties:
instance
- LocalInstance object.
Archives local AEM instance(s) into ZIP file. Provides automated way to perform offline backup. Requires having instance(s) down.
To perform online backup consider to implement custom AEM task which will reflect cURL functionality from section Automating AEM Online Backup.
The most recent file created by this task will be reused automatically while running task instanceCreate
.
Backup files are stored at path relative to project that is applying plugin com.cognifide.aem.instance
.
Most often it will be path: build/aem/instanceBackup/local/xxx.backup.zip. It could be overridden by writing:
aem {
localInstance {
backup {
localDir = file("any/other/directory")
}
}
}
Instance backups created by this task could be used later by create task.
Backups can be automatically downloaded and uploaded from remote server.
Minimal requirement to have it working is only to specify localInstance.backup.uploadUrl
.
By having only this upload property specified, plugin will automatically download most recent backup found in directory determined by upload URL.
It is also possible to specify second property localInstance.backup.downloadUrl
which will cause that concrete backup will be always in use.
By having only this download property specified, plugin will not automatically upload any backups.
Backup files created, by default, have suffix .backup.zip. This matters in case of resolving backups from remote sources to distinguish AEM backups from other files. Most often it is not needed to update it. These few lines in gradle.properties files are required to have automatic two-way backups working:
localInstance.backup.uploadUrl=sftp://example.com/aem/packages
fileTransfer.sftp.user=foo
fileTransfer.sftp.password=pass
To use a custom suffix instead of the default one, localInstance.backup.suffix
property has to be set in gradle.properties file:
localInstance.backup.suffix=.backup.custom.zip
Protocols SFTP & SMB are supported by default. However if there is a need to upload backups to cloud storage like Amazon S3, Google Cloud Storage it is possible by implementing custom file transfer.
aem {
fileTransfer {
custom("s3") {
download { dirUrl: String, fileName: String, target: File ->
// ...
}
upload { dirUrl: String, fileName: String, source: File ->
// ...
}
}
}
}
Custom file transfers could be used more widely than in only backup file resolution. It is also possible to download packages to be satisfied on instances via custom file transfer or use it in task scripting:
aem {
tasks {
instanceSatisfy {
packages {
download("s3://packages/my-package.zip")
}
}
tasks {
register("doThings") {
fileTransfer.download("s3://packages/my-package.zip")
fileTransfer.upload("s3://packages", `package`)
// etc
}
}
}
}
Destroy local AEM instance(s).
To prevent data loss, this unsafe task execution must be confirmed by parameter -Pforce
.
Turn on local AEM instance(s).
Turn off local AEM instance(s).
Turn off and then turn on local AEM instance(s).
Reload OSGi Framework (Apache Felix) on local and remote AEM instance(s).
Resolve instance files from remote sources before running other tasks.
Files considered:
- CRX packages configured in satisfy task
- local instance source files (backup ZIP and AEM quickstart JAR & license file used by create task)
This task might be also useful to check amended configuration to verify HTTP urls, SMB / SSH credentials etc and fail fast when they are wrong.
Upload & install dependent CRX package(s) before deployment. Available methods:
group(name: String, options: Resolver<PackageGroup>.() -> Unit)
, useful for declaring group of packages (or just optionally naming single package) to be installed only on demand. For instance:group("tools") { download('http://example.com/package.zip'); download('smb://internal-nt/package2.zip') }
. Then to install only packages in grouptools
, use command:gradlew instanceSatisfy -Pinstance.satisfy.group=tools
.useLocal(path: String)
, use CRX package from local file system.useLocal(file: File)
, same as above, but file can be even located outside the project.useLocalRecent(dir: File, filePattern: String = "**/*.zip")
, useful to find and use most recent file in specified directory.resolve(notation: String)
, use OSGi bundle that will be resolved from defined Gradle repositories (for example from Maven) then wrapped to CRX package.download(url: String)
, use CRX package that will be downloaded from specified URL to local temporary directory.downloadHttp(url: String, options: HttpFileTransfer.() -> Unit)
, download package using HTTP with.downloadSftp(url: String, options: SftpFileTransfer.() -> Unit)
, download package using SFTP protocol.downloadSmb(url: String, options: SmbFileTransfer.() -> Unit)
, download package using SMB protocol.
Example configuration:
aem {
tasks {
instanceSatisfy {
packages {
group("default") {
useLocal("pkg/vanityurls-components-1.0.2.zip")
download("smb://company-share/aem/packages/my-lib.zip")
download("sftp://company-share/aem/packages/other-lib.zip")
download("file:///C:/Libraries/aem/package/extra-lib.zip")
}
group("tools") {
resolve("com.neva.felix:search-webconsole-plugin:1.2.0")
download("https://github.com/Cognifide/APM/releases/download/cqsm-3.0.0/apm-3.0.0.zip")
download("https://github.com/Adobe-Consulting-Services/acs-aem-tools/releases/download/acs-aem-tools-1.0.0/acs-aem-tools-content-1.0.0-min.zip")
}
}
}
}
}
By default, all packages will be deployed when running task instanceSatisfy
.
Although, by grouping packages, there are available new options:
- group name could be used to filter out packages that will be deployed (
-Pinstance.satisfy.group=tools
, wildcards supported, comma delimited). - after satisfying particular group, there are being run instance stability checks automatically (this behavior could be customized).
Task supports extra configuration related with particular CRX package deployment and hooks for preparing (and finalizing) instance before (after) deploying packages in group on each instance. Also there is a hook called when satisfying each package group on all instances completed (for instance for awaiting stable instances which is a default behavior). In other words, for instance, there is ability to run groovy console script before/after deploying some CRX package and then restarting instance(s) if it is exceptionally required.
aem {
tasks {
satisfy {
packages {
group("tool.groovy-console") {
download("https://github.com/OlsonDigital/aem-groovy-console/releases/download/11.0.0/aem-groovy-console-11.0.0.zip")
config {
instanceName = "*-author" // additional filter intersecting 'instance.name' property
workflowToggle("dam_asset", false)
initializer {
logger.info("Installing Groovy Console on $instance")
}
finalizer {
logger.info("Installed Groovy Console on $instance")
}
completer {
logger.info("Reloading instance(s) after installing Groovy Console")
instanceActions.reloadAndAwaitUp()
}
}
}
}
}
}
}
It is also possible to specify packages to be deployed only once via command line parameter, without a need to specify them in build script. Also for local files at any file system paths.
gradlew instanceSatisfy -Pinstance.satisfy.urls=[url1,url2]
For instance:
gradlew instanceSatisfy -Pinstance.satisfy.urls=[https://github.com/OlsonDigital/aem-groovy-console/releases/download/11.0.0/aem-groovy-console-11.0.0.zip,https://github.com/neva-dev/felix-search-webconsole-plugin/releases/download/search-webconsole-plugin-1.2.0/search-webconsole-plugin-1.2.0.jar]
As of task inherits from task packageDeploy
it is also possible to temporary enable or disable workflows during CRX package deployment:
gradlew :instanceSatisfy -Ppackage.deploy.workflowToggle=[dam_asset=false]
Performs configuration actions for AEM instances in customizable conditions (specific circumstances). Feature dedicated for pre-configuring AEM instances as of not all things like turning off OSGi bundles is easy realizable via CRX packages. For instance, provisioning could help to avoid using OSGi Bundle Disabler and OSGi Component Disabler etc and is a more powerful and general approach. Could be used for AEM related troubleshooting like periodic cache cleaning, restarting OSGi bundle or components before or after CRX package deployment etc.
Sample configuration:
aem {
tasks {
instanceProvision {
step("enable-crxde") {
description = "Enables CRX DE"
condition { once() && instance.environment != "prod" }
action {
sync {
osgiFramework.configure("org.apache.sling.jcr.davex.impl.servlets.SlingDavExServlet", mapOf(
"alias" to "/crx/server"
))
}
}
}
step("setup-replication-author") {
condition { once() && instance.author }
action {
sync {
repository {
node("/etc/replication/agents.author/publish/jcr:content", mapOf(
"enabled" to true,
"userId" to instance.user,
"transportUri" to "http://localhost:4503/bin/receive?sling:authRequestLogin=1",
"transportUser" to instance.user,
"transportPassword" to instance.password
))
}
}
}
}
step("disable-unsecure-bundles") {
condition { once() && instance.environment == "prod" }
action {
sync {
osgiFramework.stopBundle("org.apache.sling.jcr.webdav")
osgiFramework.stopBundle("com.adobe.granite.crxde-lite")
instanceActions.awaitUp() // include above in property: 'instance.awaitUp.bundles.symbolicNamesIgnored'
}
}
}
}
}
}
By running task instanceProvision
, provisioner will perform all steps for which conditions are met.
Specifying condition could be even omitted, then by default each step will be performed only once()
which means that configured action {}
will be executed only once on each AEM instance.
Conditions could be more complex and use helpful methods basing on:
- guaranteed execution:
once()
<=>failSafeOnce()
, - forced execution:
always()
,never()
, - time:
repeatAfterDays(n)
,repeatAfterHours(n)
,repeatAfterMinutes(n)
,repeatAfterMillis(n)
, - counter:
repeatEvery(times)
,repeatEvery { counter: Long -> Boolean }
, - probability:
repeatProbably(probability)
.
There are also options for making provisioning more fail-safe, especially when error will be triggered when performing step action. Then each step may be additionally configured with:
continueOnFail = true
- logging error to console instead of breaking build with exception so that next step might be performed,rerunOnFail = false
- disabling performing step again when previously failed. Considered only when using conditiononce()
orfailSafeOnce()
and other conditions based on time,retry { afterSquaredSecond(3) }
- redo step action on exception after delay time with distribution likeafterSquaredSecond(n)
,afterSecond(n)
or customafter(n, delayFunction)
.
To perform some step(s) selectively, use step name property (values comma delimited, wildcards supported):
gradlew instanceProvision -Pinstance.provision.stepName=enable-crxde,...
To perform step(s) regardless conditions, use greedy property (may be combined with previous one):
gradlew instanceProvision -Pinstance.provision.greedy
Check health condition of AEM instance(s) of any type (local & remote).
Customize behavior of each particular health check using following lambdas:
aem {
tasks {
instanceAwait {
awaitUp {
timeout {
stateTime = props.long("instance.awaitUp.timeout.stateTime") ?: TimeUnit.MINUTES.toMillis(2)
constantTime = props.long("instance.awaitUp.timeout.constantTime") ?: TimeUnit.MINUTES.toMillis(10)
}
bundles {
symbolicNamesIgnored = props.list("instance.awaitUp.bundles.symbolicNamesIgnored") ?: listOf()
}
components {
platformComponents = props.list("instance.awaitUp.components.platform") ?: listOf(
"com.day.crx.packaging.*",
"org.apache.sling.installer.*"
)
specificComponents = props.list("instance.awaitUp.components.specific") ?: javaPackages.map { "$it.*" }
}
events {
unstableTopics = props.list("instance.awaitUp.event.unstableTopics") ?: listOf(
"org/osgi/framework/ServiceEvent/*",
"org/osgi/framework/FrameworkEvent/*",
"org/osgi/framework/BundleEvent/*"
)
unstableAgeMillis = props.long("instance.awaitUp.event.unstableAgeMillis") ?: TimeUnit.SECONDS.toMillis(5)
}
}
}
}
}
By default, packageDeploy
task is also awaiting up instances (this could be optionally disabled by property package.deploy.awaited=false
).
So it is also possible to configure each health check there:
aem {
tasks {
packageDeploy {
awaitUp {
// ...
}
}
}
}
Continuosly downloads logs from any local or remote AEM instances. Detects and interactively notifies about unknown errors as incident reports.
Tailer eliminates a need for connecting to remote environments using SSH protocol to be able to run tail
command on that servers.
Instead, tailer is continuously polling log files using HTTP endpoint provided by Sling Framework.
New log entries are being dynamically appended to log files stored on local file system in a separate file for each environment.
By having all log files in one place, AEM developer or QA engineer has an opportunity to comportably analyze logs, verify incidents occuring on AEM instances.
To customize tailer behavior, see InstanceTailer.
aem {
tasks {
instanceTail {
tailer {
logFilePath = props.string("instance.tail.logFilePath") ?: "/logs/error.log"
logListener = { instance -> /* ... */ }
incidentFilter = props.string("instance.tail.incidentFilter")?.let { project.file(it) } ?: File(configCommonDir, "instanceTail/incidentFilter.txt")
incidentDelay = props.long("instance.tail.incidentDelay") ?: 5000L
}
}
}
}
Log files are stored under directory: build/aem/instanceTail/${instance.name}/error.log.
By default, tailer is buffering cannonade of log entries of level ERROR in 5 seconds time window then interactively shows notification. Clicking on that notification will browse to incident log file created containing only desired exceptions. These incident files are stored under directory: build/aem/instanceTail/${instance.name}/incidents/${timestamp}-error.log.
Which type of log entries are treated as a part of incident is determined by:
- property
-Pinstance.tail.incidentLevels=[ERROR,WARN]
- wildcard exclusion rules defined in file which location is controlled by property
-Pinstance.tail.incidentFilter=aem/gradle/instanceTail/incidentFilter.txt
Sample content of incidentFilter.txt file, which holds a fragments of log entries that will be treated as known issues (notifications will be no longer shown):
# On Unix OS, it is required to have execution rights on some scripts:
Error while executing script *diskusage.sh
Error while executing script *cpu.sh
Common use case could be to tail many remote AEM instances at once that comes from multiple environments. To cover such case, it is possible to run tailer using predefined instances and defined dynamically. Number of specified instance URLs is unlimited.
Simply use command:
gradlew instanceTail -Pinstance.list=[http://admin:[email protected]:4502,http://admin:[email protected]]
Instance tailer could be used as standalone tool beside of e.g Maven based AEM application builds using Content Package Maven Plugin. Just download it from here (< 100 KB), extract anywhere on disk and run.
plugins {
id("com.cognifide.aem.environment")
}
Controls virtualized AEM environment consisting of Apache Web Server (HTTPD) with AEM Dispatcher module installed. Web Server is deployed as Docker Container orchestrated by Swarm. This solution is cross-platform just like Docker.
Provides environment related tasks: environmentUp
, environmentDev
, environmentHosts
etc.
Should be applied only at root project / only once within whole build.
Inherits from Common Plugin.
Most of the configuration steps are automated. However, there are three manual steps to make this setup fully operating:
-
Setup hosts on local machine (admin rights are required to access
/etc/hosts
orC:\Windows\System32\drivers\etc\hosts
file)- In project based on Gradle AEM Multi, just run script
aem/hosts
oraem/hosts.bat
- Otherwise
- Unix: execute
sudo gradlew environmentHosts --no-daemon
- Windows: start PowerShell with "Run as administrator", then execute
./gradlew.bat environmentHosts --no-daemon
- Unix: execute
- In project based on Gradle AEM Multi, just run script
Ensure having shared folders enabled via GUI:
Please note that after each Windows credentials change (e.g forced by company policy), it is required to click button "Reset credentials" to ensure having shared drives working correctly.
When running task environmentUp
first time, Windows User Account Control (UAC) may require user confirmation to allow virtualized container to access local configuration files stored on shared drives.
While using Docker Toolbox, beware that, by default, there is only one shared folder in VirtualBox configured which mounts C:/Users to /c/users on Docker Machine / Ubuntu. As a consequence, if AEM project is located outside of C:/Users directory, there is a need to manually add corresponding shared folder and mount it on Docker Machine using VirtualBox GUI.
Environment plugin is a little more depending on convention in case of directory structure. Screenshot below presents recommended file structure which does not need any additional configuration.
Ensure having directory aem/.environment ignored in VCS and excluded from indexing by IDE.
Environment as a code paradigm is a main reason of locating environment configuration and runtime inside a project. Nowadays, advanced web applications (like AEM applications) are multi-layered and closely related with environment. By having AEM dispatcher configuration and HTTDP configuration near application code base, the changes could be applied very easily and quickly. By having all source code opened in single IDE window, e.g finding and amending content repository path occurrences at same time both in AEM application and HTTPD configuration could reduce risk of regression and they could be versioned once. All code base - application and environment - could be treated as a whole, fully operating unit.
In case of the dispatcher it takes few seconds to start. Service health checks could be described by following configuration. For example, to wait for all domains to be available and having particular text in response, write:
aem {
environment {
// ...
healthChecks {
url("Live site", "http://example.com/en-us.html", text = "English")
url("Demo site", "http://demo.example.com/en-us.html", text = "English")
url("Author login", "http://author.example.com/libs/granite/core/content/login.html" +
"?resource=%2F&\$\$login\$\$=%24%24login%24%24&j_reason=unknown&j_reason_code=unknown", text = "AEM Sign In")
}
}
}
These checks are performed automatically after each file change applied when running task environmentDev
or on demand when running task environmentAwait
.
Turns on local AEM environment.
Turns off local AEM environment.
Allows to listen for Apache Web Server / Dispatcher configuration files changed and then automatically reload HTTP service.
Workflow:
- Run command
gradlew environmentDev
, - Edit files located in aem/gradle/environment/httpd/conf ,
- Notice that HTTPD service should be restarted automatically after file changes,
- Check results of environment service health checks,
- Optionally, check:
- HTTPD and AEM Dispatcher logs located at path aem/.environment/httpd/logs
- AEM Dispatcher cache located at path aem/.environment/httpd/cache
Performs environment service health checks on demand.
Reloads virtualized AEM environment by reloading all Docker containers (e.g. removing Dispatcher cache files then restarting HTTPD server).
Appends AEM environment specific hosts entries to OS specific hosts file.
Common configuration like root of content for JCR package, should be defined in allprojects
section like below / e.g in root build.gradle.kts file:
allprojects {
plugins.withId("com.cognifide.aem.common") {
configure<AemExtension> {
`package` {
contentDir = project.file("src/main/aem") // overrides default dir named 'content'
}
}
}
plugins.withId("com.cognifide.aem.bundle") {
configure<AemExtension> {
tasks {
bundleCompose {
category = "example"
vendor = "Company"
}
}
}
dependencies {
"compileOnly"("com.adobe.aem:uber-jar:${Build.AEM_VERSION}:apis") // and more
}
}
}
For instance, subproject :aem:core
specific configuration like OSGi bundle or CRX package options should be defined in aem/core/build.gradle.kts
:
aem {
tasks {
bundleCompose {
javaPackage = "com.company.example.aem.core"
}
packageCompose {
fromProjects(':content:*')
baseName = "example-core"
duplicatesStrategy = DuplicatesStrategy.EXCLUDE
}
}
}
Warning! Very often plugin users mistake is to configure instanceSatisfy
task in allprojects
closure.
As an effect there will be same dependent CRX package defined multiple times.
To control lifecycle of both:
- AEM instances (author & publish) (provided from instance plugin)
- AEM environment (HTTPD server with AEM dispatcher) (provided by environment plugin)
use dynamically registered tasks (only if at least one of plugins above are applied):
up
<=>instanceUp
+environmentUp
down
<=>instanceDown
+environmentDown
destroy
<=>instanceDestroy
+environmentDestroy
setup
<=>instanceSetup
+environmentSetup
resetup
<=>instanceResetup
+environmentResetup
resolve
<=>instanceResolve
+environmentResolve
Most of built-in tasks logic is based onaem
object of type AemExtension.
It provides concise AEM related API for accessing AEM configuration, synchronizing with AEM instances via specialized instance services of aem.sync
to make tasks implementation a breeze. The options for automating things around AEM are almost unlimited.
While implementing custom AEM tasks, mix usages of following instance services:
packageManager
PackageManager - Allows to communicate with CRX Package Manager.osgiFramework
OsgiFramework - Controls OSGi framework using Apache Felix Web Console endpoints.repository
Repository - Allows to communicate with JCR Content Repository.http
InstanceHttpClient - Provides extremely easy to use HTTP client designed especially to be used with AEM (covers basic authentication, allows to use only relative paths instead of full URLs etc)status
Status - Allows to read statuses available at Apache Felix Web Console.groovyConsole
GroovyConsole - Allows to execute Groovy code / scripts on AEM instance having Groovy Console CRX package installed.
Below snippet could be used to automatize creation of production content backups.
aem {
tasks {
register("backupProductionAuthor") {
doLast {
val pkg = namedInstance("prod-author").sync {
downloadPackage {
group = "example"
name = "backup"
description = "Backup of content, tags and DAM"
archiveName = "backup-author.zip"
filters(
"/content/cq:tags/example",
"/content/example",
"/content/dam/example"
)
}
}
http {
basicUser = "foo"
basicPassword = "bar"
postMultipart("http://my-aem-backup-service.com/package/upload", mapOf("file" to pkg))
}
}
}
}
}
To make an HTTP request to some AEM endpoint (servlet) simply write:
aem {
tasks {
register("runHealthCheck") {
doLast {
syncInstances {
http {
get("/bin/example/healthCheck") { checkStatus(it, 200) }
}
}
}
}
}
}
There are unspecified AEM instances as parameter for method syncInstances
, so that instances matching default filtering will be used.
The fragment { checkStatus(it, 200) }
could be even ommitted because, by default sync API checks status code that it belongs to range [200,300).
To parse endpoint response as JSON (using JsonPath), simply write:
aem {
tasks {
register("runHealthCheck") {
doLast {
syncInstances {
http {
val json = get("/bin/example/healthCheck") { asJson(it) }
val status = json.read("status") as String
if (status != "OK") {
throw GradleException("Health check failed on: $instance because status '$status' detected.")
}
}
}
}
}
}
}
There are also available convenient methods asStream
, asString
to be able to process endpoint responses.
Below snippet could be used to automatize recovery from content backups (e.g for production or to replicate production content to test environment).
aem {
tasks {
register("deployProductionContent") {
doLast {
val instances = listOf(
instance("http://user:[email protected]") // URL specified directly, could be parametrized by some gradle command line property
// namedInstance("local-publish") // reused AEM instance defined in 'gradle.properties'
)
val pkg = httpFile { download("https://company.com/aem/backups/example-1.0.0-201901300932.backup.zip") }
sync(instances) {
packageManager.deploy(pkg)
}
}
}
}
}
To make changes in AEM content repository, use Repository instance service which is a part of instance sync tool.
For example, to migrate pages even without using Groovy Console deployed on instance, simply write:
aem {
tasks {
register("migratePages") {
description = "Migrates pages to new component"
doLast {
syncInstances {
repository {
node("/content/example")
.traverse()
.filter { it.type == "cq:PageContent" && it.properties["sling:resourceType"] == "example/components/basicPage" }
.forEach { page ->
logger.info("Migrating page: ${page.path}")
page.saveProperty("sling:resourceType", "example/components/advancedPage")
}
}
}
}
}
}
}
To create new / update existing nodes to configure e.g replication agents write:
aem {
tasks {
register("setupReplicationAgents") {
description = "Corrects publish replication agent transport URI"
doLast {
syncInstances {
repository {
node("/etc/replication/agents.publish/flush/jcr:content", mapOf( // shorthand for 'node(path).save(props)'
"transportUri" to "http://dispatcher.example.com/dispatcher/invalidate.cache"
))
}
}
}
}
}
}
Under the hood, repository service is using only AEM built-in Sling Post Servlet.
It is also possible to easily execute any code on AEM runtime using Groovy Console.
Assuming that on AEM instances there is already installed Groovy Console e.g via instanceSatisfy
task, then it is possible to use GroovyConsole instance service.
aem {
tasks {
satisfy {
group("tool.groovyconsole") { download("https://github.com/icfnext/aem-groovy-console/releases/download/12.0.0/aem-groovy-console-12.0.0.zip") }
}
register("generatePosts") {
doLast {
syncInstances {
groovyConsole.evalCode("""
def postsService = getService("com.company.example.aem.sites.services.posts.PostsService")
println postsService.randomPosts(5)
""")
// groovyConsole.evalScript("posts.groovy") // if script above moved to 'aem/gradle/groovyScript/posts.groovy'
}
}
}
}
}
Simply use OsgiFramework instance service.
To restart some bundle after deploying a CRX package, write:
aem {
tasks {
packageDeploy {
doLast {
syncInstances {
osgiFramework.restartBundle("com.adobe.cq.dam.cq-scene7-imaging")
}
}
}
}
}
To disable specific OSGi component by its PID value and only on publish instances, write:
aem {
tasks {
register("instanceSecure") {
doLast {
sync(publishInstances) {
osgiFramework.disableComponent("org.apache.sling.jcr.davex.impl.servlets.SlingDavExServlet")
// osgiFramework.stopBundle("org.apache.sling.jcr.webdav")
}
}
}
}
}
To configure specific OSGi service by its PID:
aem {
tasks {
register("enableCrx") {
doLast {
sync(publishInstances) {
osgiFramework.configure("org.apache.sling.jcr.davex.impl.servlets.SlingDavExServlet", mapOf(
"alias" to "/crx/server",
"dav.create-absolute-uri" to true,
"dav.protectedhandlers" to "org.apache.jackrabbit.server.remoting.davex.AclRemoveHandler"
))
}
}
}
}
}
All CRUD methods for manipulating OSGi configurations are available. Also for configuration factories.
Simply use Workflow Manager instance service.
Workflows can be either enabled or disabled by:
aem {
tasks {
register("setupWorkflows") {
doLast {
syncInstances {
workflowManager.workflow("update_asset_create").disable()
workflowManager.workflow("update_asset_mod").disable()
workflowManager.workflows("dam_asset").forEach { it.enable() } // reverts above using shorthand alias
}
}
}
}
}
Also it is possible to enable or disable workflows only for particular action to be performed:
workflowManager.toggleTemporarily("dam_asset", false) {
packageManager.deploy(file("my-package.zip"))
}
It is also possible to mix enabling and disabling workflows:
workflowManager.toggleTemporarily(mapOf(
"update_asset_create" to false,
"update_asset_mod" to false
"update_asset_custom" to true
)) {
// ...
}
Regardless type of Docker runtime used, running any Docker images is super-easy via AEM DSL method aem.runDocker()
.
Under the hood, GAP ensures that volume paths passed to Docker are correct. Also correctly handles streaming process output,
allows to set expected exit codes and other useful options.
Consider following task registration:
aem {
tasks {
register("runTool")
doLast {
runDocker {
operation("Runnning Docker based tool'")
image = "any-vendor/any-image"
volume(file("resources"), "/resources")
port(8080, 80)
command = "<any command>"
}
}
}
}
}
Then running any Docker image based tool could be simply achieved via running CLI command:
sh gradlew runTool
.
Gradle AEM Plugin assumes separation of 5 plugins to properly fit into Gradle tasks structure correctly.
Most often, Gradle commands are being launched from project root and tasks are being run by their name e.g instanceSatisfy
(which is not fully qualified, better if it will be :instanceSatisfy
of root project).
Let's imagine if task instanceSatisfy
will come from package plugin, then Gradle will execute more than one instanceSatisfy
(for all projects that have plugin applied), so that this is unintended behavior.
Currently used plugin architecture solves that problem.
Initially, to create fully configured local AEM instances simply run command gradlew instanceSetup
.
Later during development process, building and deploying to AEM should be done using the simplest command: gradlew
.
Above configuration uses default tasks, so that alternatively it is possible to do the same using explicitly specified command gradlew instanceSatisfy packageDeploy
.
- Firstly dependent packages (like AEM hotfixes, Vanity URL Components etc) will be installed lazily (only when they are not installed yet).
- In next step application is being built and deployed to all configured AEM instances.
- Finally build awaits till all AEM instances are stable.
When there are defined named AEM instances: local-author
, local-publish
, integration-author
and integration-publish
,
then it is possible to:
- deploy (or satisfy) CRX package(s)
- tail logs
- checkout JCR content
with taking into account:
- type of environment (local, integration, staging, etc)
- type of AEM instance (author / publish)
by filtering instances by names, e.g:
gradlew packageDeploy -Pinstance.name=integration-*
gradlew packageDeploy -Pinstance.name=*-author
gradlew packageDeploy -Pinstance.name=local-author,integration-author
Default value of that instance name filter is ${environment}-*
, so that typically local-*
.
Environment value comes from system environment variable ENV
or property env
.
To deploy only to author or publish instances:
gradlew packageDeploy -Pinstance.author
gradlew packageDeploy -Pinstance.publish
To deploy only to instances specified explicitly:
gradlew packageDeploy -Pinstance.list=[http://admin:admin@localhost:4502,http://admin:admin@localhost:4503]
Instance urls must be delimited by colon. Remember to encode instance user & password properly.
The properties syntax comes from Pebble Template Engine which means that all its features (if statements, for loops, filters etc) can be used inside files being expanded.
Expanding properties could be used separately on any string or file source in any custom task by using method aem.props.expand()
.
Because of bug related with regresion introduced in Gradle 5.1, there are some difficulties with setting archives base names. AEM Plugin is overriding default Gradle convention for not only having project name in archive base name, but also to having prefix - root project name when project is one of subprojects (multi-project build case as in Gradle AEM Multi). However overriding this convention might not be trivial and is not recommended as of AEM Plugin in most cases proposes good enough battle-tested convention.
Still, if it is really needed to be done - setting customized name for CRX packages and OSGi bundles built, use snippet:
subprojects {
afterEvaluate {
tasks {
withType<AbstractArchiveTask>().configureEach {
archiveBaseName.set("acme-${project.name}")
}
}
}
}
Then, also common case is to customize paths in which OSGi bundles should be placed in built CRX package. As practice shows up, mostly desired snippet to be used is:
subprojects {
plugins.withId("com.cognifide.aem.package") {
configure<AemExtension> {
`package` {
installPath = "/apps/acme/${project.name}/install"
}
}
}
}
Noticed when using Docker Desktop for Windows, after hibernation task environmentUp
may fail, because of some Docker related issue.
To mitigate it, just RPM on Docker icon in the tray and choose Restart. After Docker ends restarting (tray icon will be no longer animated),
run task environmentRestart
.
After correcting bad Java package case from camelCase to lowercase according to Oracle recommendations, BND tool may report error:
> Task :aem:sites:jar FAILED
...
error : Classes found in the wrong directory:
To fix above error simply run once following command to ensure building using fresh daemon and resources:
sh gradle clean :aem:sites:jar --no-daemon --rerun-tasks
Since AEM 6.2 it is recommended to use new OSGi service component annotations to register OSGi components instead SCR annotations (still supported, but not by Gradle AEM Plugin).
For the reference, please read post on official Adobe Blog.
Basically, Gradle AEM Plugin is designed to be used while implementing new projects on AEM in version greater than 6.2. Because, of that fact, there is no direct possibility to reuse code written for older AEM's which is using SCR annotations. However it is very easy to migrate these annotations to new ones and generally speaking it is not much expensive task to do.
import org.apache.felix.scr.annotations.Component;
->
import org.osgi.service.component.annotations.Component;
New API fully covers functionality of old one, so nothing to worry about while migrating.
Expandable properties with dynamically calculated value (unique per build) like created
and buildCount
are not used by default generated properties file intentionally,
because such usages will effectively forbid caching packageCompose
task and it will be never UP-TO-DATE
.
Vault tool current working directory cannot be easily configured, because of its API. AEM plugin is temporarily changing current working directory for Vault, then returning it back to original value. In case of that workaround, Vault tasks should not be run in parallel (by separated daemon processed / JVM synchronization bypassed), because of potential unpredictable behavior.
Gradle AEM Plugin | Gradle Build Tool | Adobe Experience Manager | Java |
---|---|---|---|
4.x -> 5.x | 4.x -> 4.8 | 6.x and up | 8 |
6.0.0 -> 6.2.1 | 4.9 -> 5.0 | 6.x and up | 8 |
6.3.0 -> 6.x | 5.1 -> 5.6 | 6.x and up | 8 |
7.2.0 -> 8.1.1 | 5.1 -> 5.6 | 6.x and up | 8,11 |
8.1.2 and up | 6.0 and up | 6.x and up | 8,11 |
- Clone this project using command
git clone https://github.com/Cognifide/gradle-aem-plugin.git
- To build plugin, simply enter cloned directory run command:
gradlew
- To debug built plugin:
- Append to build command parameters
--no-daemon -Dorg.gradle.debug=true
- Run build, it will suspend, then connect remote at port 5005 by using IDE
- Build will proceed and stop at previously set up breakpoint.
- Append to build command parameters
Issues reported or pull requests created will be very appreciated.
- Fork plugin source code using a dedicated GitHub button.
- Do code changes on a feature branch created from develop branch.
- Create a pull request with a base of develop branch.
Gradle AEM Plugin is licensed under the Apache License, Version 2.0 (the "License")