# Controlling JAR Content Merging
Shadow allows for customizing the process by which the output JAR is generated through the
Transformer
(opens new window) interface.
This is a concept that has been carried over from the original Maven Shade implementation.
A Transformer
(opens new window) is invoked for each
entry in the JAR before being written to the final output JAR.
This allows a Transformer
(opens new window) to
determine if it should process a particular entry and apply any modifications before writing the stream to the output.
// Adding a Transformer
import com.github.jengelman.gradle.plugins.shadow.transformers.Transformer
import com.github.jengelman.gradle.plugins.shadow.transformers.TransformerContext
import javax.annotation.Nonnull
import org.apache.tools.zip.ZipOutputStream
import org.gradle.api.file.FileTreeElement
class MyTransformer implements Transformer {
@Override
boolean canTransformResource(@Nonnull FileTreeElement element) { return true }
@Override
void transform(@Nonnull TransformerContext context) {}
@Override
boolean hasTransformedResource() { return true }
@Override
void modifyOutputStream(@Nonnull ZipOutputStream os, boolean preserveFileTimestamps) {}
}
tasks.named('shadowJar', com.github.jengelman.gradle.plugins.shadow.tasks.ShadowJar) {
transform(MyTransformer.class)
}
Additionally, a Transformer
can accept a Closure
to configure the provided Transformer
.
// Configuring a Transformer
import com.github.jengelman.gradle.plugins.shadow.transformers.Transformer
import com.github.jengelman.gradle.plugins.shadow.transformers.TransformerContext
import javax.annotation.Nonnull
import org.apache.tools.zip.ZipOutputStream
import org.gradle.api.file.FileTreeElement
class MyTransformer implements Transformer {
boolean enabled
@Override
boolean canTransformResource(@Nonnull FileTreeElement element) { return true }
@Override
void transform(@Nonnull TransformerContext context) {}
@Override
boolean hasTransformedResource() { return true }
@Override
void modifyOutputStream(@Nonnull ZipOutputStream os, boolean preserveFileTimestamps) {}
}
tasks.named('shadowJar', com.github.jengelman.gradle.plugins.shadow.tasks.ShadowJar) {
transform(MyTransformer.class) {
enabled = true
}
}
An instantiated instance of a Transformer
can also be provided.
// Adding a Transformer Instance
import com.github.jengelman.gradle.plugins.shadow.transformers.Transformer
import com.github.jengelman.gradle.plugins.shadow.transformers.TransformerContext
import javax.annotation.Nonnull
import org.apache.tools.zip.ZipOutputStream
import org.gradle.api.file.FileTreeElement
class MyTransformer implements Transformer {
final boolean enabled
MyTransformer(boolean enabled) {
this.enabled = enabled
}
@Override
boolean canTransformResource(@Nonnull FileTreeElement element) { return true }
@Override
void transform(@Nonnull TransformerContext context) {}
@Override
boolean hasTransformedResource() { return true }
@Override
void modifyOutputStream(@Nonnull ZipOutputStream os, boolean preserveFileTimestamps) {}
}
tasks.named('shadowJar', com.github.jengelman.gradle.plugins.shadow.tasks.ShadowJar) {
transform(new MyTransformer(true))
}
# Merging Service Descriptor Files
Java libraries often contain service descriptors files in the META-INF/services
directory of the JAR.
A service descriptor typically contains a line delimited list of classes that are supported for a particular service.
At runtime, this file is read and used to configure library or application behavior.
Multiple dependencies may use the same service descriptor file name.
In this case, it is generally desired to merge the content of each instance of the file into a single output file.
The ServiceFileTransformer
(opens new window)
class is used to perform this merging. By default, it will merge each copy of a file under META-INF/services
into a
single file in the output JAR.
// Merging Service Files
tasks.named('shadowJar', com.github.jengelman.gradle.plugins.shadow.tasks.ShadowJar) {
mergeServiceFiles()
}
The above code snippet is a convenience syntax for calling
transform(ServiceFileTransformer.class)
(opens new window).
Groovy Extension Module descriptor files (located at
META-INF/services/org.codehaus.groovy.runtime.ExtensionModule
) are ignored by theServiceFileTransformer
(opens new window). This is due to these files having a different syntax than standard service descriptor files. Use themergeGroovyExtensionModules()
(opens new window) method to merge these files if your dependencies contain them.
# Configuring the Location of Service Descriptor Files
By default the ServiceFileTransformer
(opens new window)
is configured to merge files in META-INF/services
.
This directory can be overridden to merge descriptor files in a different location.
// Merging Service Files in a Specific Directory
tasks.named('shadowJar', com.github.jengelman.gradle.plugins.shadow.tasks.ShadowJar) {
mergeServiceFiles {
path = 'META-INF/custom'
}
}
# Excluding/Including Specific Service Descriptor Files From Merging
The ServiceFileTransformer
(opens new window)
class supports specifying specific files to include or exclude from merging.
// Excluding a Service Descriptor From Merging
tasks.named('shadowJar', com.github.jengelman.gradle.plugins.shadow.tasks.ShadowJar) {
mergeServiceFiles {
exclude 'META-INF/services/com.acme.*'
}
}
# Merging Groovy Extension Modules
Shadow provides a specific transformer for dealing with Groovy extension module files.
This is due to their special syntax and how they need to be merged together.
The GroovyExtensionModuleTransformer
(opens new window)
will handle these files.
The ShadowJar
(opens new window) task also provides a short syntax
method to add this transformer.
// Merging Groovy Extension Modules
tasks.named('shadowJar', com.github.jengelman.gradle.plugins.shadow.tasks.ShadowJar) {
mergeGroovyExtensionModules()
}
# Appending Text Files
Generic text files can be appended together using the
AppendingTransformer
(opens new window).
Each file is appended using new lines to separate content.
The ShadowJar
(opens new window) task provides a short syntax
method of
append(String)
(opens new window) to
configure this transformer.
// Appending a Property File
tasks.named('shadowJar', com.github.jengelman.gradle.plugins.shadow.tasks.ShadowJar) {
append 'test.properties'
}
# Appending XML Files
XML files require a special transformer for merging.
The XmlAppendingTransformer
(opens new window)
reads each XML document and merges each root element into a single document.
There is no short syntax method for the XmlAppendingTransformer
(opens new window).
It must be added using the transform
(opens new window) methods.
// Appending a XML File
tasks.named('shadowJar', com.github.jengelman.gradle.plugins.shadow.tasks.ShadowJar) {
transform(com.github.jengelman.gradle.plugins.shadow.transformers.XmlAppendingTransformer.class) {
resource = 'properties.xml'
}
}