Version Catalogs
- Using a catalog
- Importing a published catalog
- Importing a catalog from a file
- Importing multiple catalogs
- Creating a catalog
- Programming catalogs
- Publishing a catalog
- Changing the catalog name
- Overwriting catalog versions
- Using a catalog in
buildSrc - Working with multiple version catalogs
- Using catalog entries in Gradle APIs
A version catalog is a selected list of dependencies that can be referenced in build scripts, simplifying dependency management.
Instead of specifying dependencies directly using string notation, you can pick them from a version catalog:
dependencies {
implementation(libs.groovy.core)
}
dependencies {
implementation(libs.groovy.core)
}
In this example, libs represents the catalog, and groovy is a dependency available in it.
Where the version catalog defining libs.groovy.core is a libs.versions.toml file in the gradle directory:
[libraries]
groovy-core = { group = "org.codehaus.groovy", name = "groovy", version = "3.0.5" }
Version catalogs offer several advantages:
-
Type-Safe Accessors: Gradle generates type-safe accessors for each catalog, enabling autocompletion in IDEs.
-
Centralized Version Management: Each catalog is visible to all projects in a build.
-
Dependency Bundles: Catalogs can group commonly used dependencies into bundles.
-
Version Separation: Catalogs can separate dependency coordinates from version information, allowing shared version declarations.
-
Conflict Resolution: Like regular dependency notation, version catalogs declare requested versions but do not enforce them during conflict resolution.
| Version catalogs declare requested versions but do not enforce them. Gradle may still select different versions due to dependency graph conflicts or constraints applied through platforms or other dependency management APIs. |
Using a catalog
To access items in a version catalog defined in the standard libs.versions.toml file located in the gradle directory, you use the libs object in your build scripts.
For example, to reference a library, you can use libs.<alias>, and for a plugin, you can use libs.plugins.<alias>.
Declaring dependencies using a version catalog:
dependencies {
implementation(libs.groovy.core)
implementation(libs.groovy.json)
implementation(libs.groovy.nio)
}
dependencies {
implementation libs.groovy.core
implementation libs.groovy.json
implementation libs.groovy.nio
}
Is the same as:
dependencies {
implementation("org.codehaus.groovy:groovy:3.0.5")
implementation("org.codehaus.groovy:groovy-json:3.0.5")
implementation("org.codehaus.groovy:groovy-nio:3.0.5")
}
dependencies {
implementation 'org.codehaus.groovy:groovy:3.0.5'
implementation 'org.codehaus.groovy:groovy-json:3.0.5'
implementation 'org.codehaus.groovy:groovy-nio:3.0.5'
}
If the catalog looks as follows:
[versions]
groovy = "3.0.5"
[libraries]
groovy-core = { module = "org.codehaus.groovy:groovy", version.ref = "groovy" }
groovy-json = { module = "org.codehaus.groovy:groovy-json", version.ref = "groovy" }
groovy-nio = { module = "org.codehaus.groovy:groovy-nio", version.ref = "groovy" }
Where the catalog can be imported from a local file or a repository.
Version catalogs use a TOML format in which accessors map directly to the aliases and versions defined in it, providing type-safe access to dependencies and plugins. This enables IDEs to provide autocompletion, highlight typos, and identify missing dependencies as errors.
Aliases and type-safe accessors
Aliases in a version catalog consist of identifiers separated by a dash (-) or underscore (_).
Type-safe accessors are generated for each alias, normalized to dot notation.
Consider the following catalog:
[libraries]
# Dashes create nested accessor groups
ktor-client-core = "io.ktor:ktor-client-core:3.1.0"
ktor-client-cio = "io.ktor:ktor-client-cio:3.1.0"
ktor-client-logging = "io.ktor:ktor-client-logging:3.1.0"
# A single-segment alias stays flat
junit = "junit:junit:4.13.2"
Each dash in the alias maps to a dot in the accessor, creating a nested group that IDEs can autocomplete:
| Alias | Generated accessor |
|---|---|
|
|
|
|
|
|
|
|
Classifiers, artifact types, and capabilities
Version catalogs intentionally capture only dependency coordinates: group, name, and version.
They do not support classifiers, artifact types (@aar, @zip), excludes, or capabilities directly in the TOML file.
[libraries]
my-lib = { module = "com.example:my-lib", version = "1.0.0" }
When you need a specific classifier for a dependency declared in a version catalog, use variantOf() at the dependency declaration site in your build script:
dependencies {
implementation(variantOf(libs.my.lib) { classifier("test-fixtures") })
}
dependencies {
implementation(variantOf(libs.my.lib) { classifier('test-fixtures') })
}
This keeps the catalog focused on coordinates while letting each consumer choose the appropriate classifier.
For artifact types (common in Android projects needing @aar), use the artifact block:
dependencies {
implementation(libs.my.lib) {
artifact {
name = "my-lib"
type = "aar"
}
}
}
dependencies {
implementation(libs.my.lib) {
artifact {
name = 'my-lib'
type = 'aar'
}
}
}
See TOML Limitations to learn more.
Importing a published catalog
A catalog published with the version-catalog plugin can be imported in a consumer’s settings.gradle(.kts) file using the Settings API:
dependencyResolutionManagement {
versionCatalogs {
create("libs") {
from("com.mycompany:catalog:1.0")
}
}
}
dependencyResolutionManagement {
versionCatalogs {
libs {
from("com.mycompany:catalog:1.0")
}
}
}
The from() method accepts standard dependency notation ("group:artifact:version").
Gradle downloads the TOML file from whatever repositories are configured in the dependencyResolutionManagement block and makes its entries available as type-safe accessors.
Once imported, all projects in the build can reference the catalog’s libraries, plugins, and bundles using the catalog name (e.g., libs.someLibrary).
Importing a catalog from a file
Gradle automatically imports a catalog in the gradle directory named libs.versions.toml. You do not need to import it programmatically.
|
The version catalog builder API allows importing a catalog from an external file, enabling reuse across different parts of a build, such as sharing the main build’s catalog with buildSrc.
For example, you can include a catalog in the buildSrc/settings.gradle(.kts) file as follows:
dependencyResolutionManagement {
versionCatalogs {
create("libs") {
from(files("../gradle/libs.versions.toml"))
}
}
}
dependencyResolutionManagement {
versionCatalogs {
libs {
from(files("../gradle/libs.versions.toml"))
}
}
}
The VersionCatalogBuilder.from(Object dependencyNotation) method accepts only a single file, meaning that notations like Project.files(java.lang.Object…) must refer to one file. Otherwise, the build will fail.
Remember that you don’t need to import the version catalog named libs.versions.toml if it resides in your gradle folder. It will be imported automatically.
|
Importing multiple catalogs
You can declare multiple catalogs using the Settings API.
For example, the default libs catalog is auto-imported from gradle/libs.versions.toml.
To add additional catalogs for build tools and test dependencies, create gradle/tools.versions.toml and gradle/test.versions.toml files and register them:
versionCatalogs {
create("tools") {
from(files("gradle/tools.versions.toml"))
}
create("testLibs") {
from(files("gradle/test.versions.toml"))
}
// HOWEVER THIS IS NOT ALLOWED - IT WILL NOT WORK
/* In version catalog libs, you can only call the 'from' method a single time:
create("libs") {
from(files("gradle/base.versions.toml"))
from(files("gradle/extras.versions.toml")) // Error!
}
*/
}
versionCatalogs {
tools {
from(files("gradle/tools.versions.toml"))
}
testLibs {
from(files("gradle/test.versions.toml"))
}
// HOWEVER THIS IS NOT ALLOWED - IT WILL NOT WORK
/* In version catalog libs, you can only call the 'from' method a single time:
libs {
from(files("gradle/base.versions.toml"))
from(files("gradle/extras.versions.toml")) // Error!
}
*/
}
Each catalog generates its own extension, so you reference them by name in build scripts:
dependencies {
implementation(libs.guava)
implementation(libs.commons.lang3)
implementation(tools.errorprone.annotations)
implementation(tools.jsr305)
testImplementation(testLibs.junit.api)
testImplementation(testLibs.mockito)
}
dependencies {
implementation libs.guava
implementation libs.commons.lang3
implementation tools.errorprone.annotations
implementation tools.jsr305
testImplementation testLibs.junit.api
testImplementation testLibs.mockito
}
|
To minimize the risk of naming conflicts, each catalog generates an extension applied to all projects, so it’s advisable to choose a unique name. One effective approach is to select a name that ends with |
Creating a catalog
Version catalogs are conventionally declared using a libs.versions.toml file located in the gradle subdirectory of the root build.
They use the TOML format:
[versions]
groovy = "3.0.5"
checkstyle = "8.37"
[libraries]
groovy-core = { module = "org.codehaus.groovy:groovy", version.ref = "groovy" }
groovy-json = { module = "org.codehaus.groovy:groovy-json", version.ref = "groovy" }
groovy-nio = { module = "org.codehaus.groovy:groovy-nio", version.ref = "groovy" }
commons-lang3 = { group = "org.apache.commons", name = "commons-lang3", version = { strictly = "[3.8, 4.0[", prefer = "3.9" } }
[bundles]
groovy = ["groovy-core", "groovy-json", "groovy-nio"]
[plugins]
versions = { id = "com.github.ben-manes.versions", version = "0.45.0" }
However, they can reside in other folders and use different names. See Importing a catalog from a file for importing a catalog from a custom location, or Importing multiple catalogs for registering additional catalogs.
The TOML format
The version catalog TOML file has four sections:
-
[versions]– Declares version identifiers. -
[libraries]– Maps aliases to GAV coordinates. -
[bundles]– Defines dependency bundles. -
[plugins]– Declares plugin versions.
The TOML file format is very lenient and allows you to write "dotted" properties as shortcuts for full object declarations.
Versions
Versions can be declared either as a single string, in which case they are interpreted as a required version, or as a rich version:
[versions]
other-lib = "5.5.0" # Required version
my-lib = { strictly = "[1.0, 2.0[", prefer = "1.2" } # Rich version
Supported members of a version declaration are:
-
require: the required version -
strictly: the strict version -
prefer: the preferred version -
reject: the list of rejected versions -
rejectAll: a boolean to reject all versions
Libraries
Each library is mapped to a GAV coordinate: group, artifact, version.
They can be declared as a simple string, in which case they are interpreted as coordinates, or a separate group and name:
[versions]
common = "1.4"
[libraries]
my-lib = "com.mycompany:mylib:1.4"
my-lib-no-version.module = "com.mycompany:mylib"
my-other-lib = { module = "com.mycompany:other", version = "1.4" }
my-other-lib2 = { group = "com.mycompany", name = "alternate", version = "1.4" }
mylib-full-format = { group = "com.mycompany", name = "alternate", version = { require = "1.4" } }
You can also define strict or preferred versions using strictly or prefer:
[libraries]
commons-lang3 = { group = "org.apache.commons", name = "commons-lang3", version = { strictly = "[3.8, 4.0[", prefer = "3.9" } }
In case you want to reference a version declared in the [versions] section, use the version.ref property:
[versions]
some = "1.4"
[libraries]
my-lib = { group = "com.mycompany", name="mylib", version.ref="some" }
Bundles
Bundles group multiple library aliases, so they can be referenced together in the build script.
[versions]
groovy = "3.0.9"
[libraries]
groovy-core = { group = "org.codehaus.groovy", name = "groovy", version.ref = "groovy" }
groovy-json = { group = "org.codehaus.groovy", name = "groovy-json", version.ref = "groovy" }
groovy-nio = { group = "org.codehaus.groovy", name = "groovy-nio", version.ref = "groovy" }
[bundles]
groovy = ["groovy-core", "groovy-json", "groovy-nio"]
This is useful for pulling in several related dependencies with a single alias:
dependencies {
implementation(libs.bundles.groovy)
}
dependencies {
implementation libs.bundles.groovy
}
Plugins
This section defines the plugins and their versions by mapping plugin IDs to version numbers.
Just like libraries, you can define plugin versions using aliases from the [versions] section or directly specify the version.
[plugins]
versions = { id = "com.github.ben-manes.versions", version = "0.45.0" }
Which can be accessed in any project of the build using the plugins {} block.
To refer to a plugin from the catalog, use the alias() function:
plugins {
`java-library`
checkstyle
alias(libs.plugins.versions)
}
plugins {
id 'java-library'
id 'checkstyle'
// Use the plugin `versions` as declared in the `libs` version catalog
alias(libs.plugins.versions)
}
| You cannot use a plugin declared in a version catalog in your settings file or settings plugin. |
Avoiding subgroup accessors
To avoid generating subgroup accessors, use camelCase notation:
| Aliases | Accessors |
|---|---|
|
|
|
|
Reserved keywords
Certain keywords, like extensions, class, and convention, are reserved and cannot be used as aliases.
Additionally, bundles, versions, and plugins cannot be the first subgroup in a dependency alias.
For example, the alias versions-dependency is not valid, but versionsDependency or dependency-versions are valid.
Limitations
These are explicit limitations the catalog TOML format does not support:
-
classifierattributes -
type = "aar"orext = "zip"artifact types -
excluderules -
capabilitiesrequirements
Programming catalogs
Version catalogs can be declared programmatically in the settings.gradle(.kts) file using the Settings API:
dependencyResolutionManagement {
versionCatalogs {
create("libs") {
version("groovy", "3.0.5")
version("checkstyle", "8.37")
library("groovy-core", "org.codehaus.groovy", "groovy").versionRef("groovy")
library("groovy-json", "org.codehaus.groovy", "groovy-json").versionRef("groovy")
library("groovy-nio", "org.codehaus.groovy", "groovy-nio").versionRef("groovy")
library("commons-lang3", "org.apache.commons", "commons-lang3").version {
strictly("[3.8, 4.0[")
prefer("3.9")
}
}
}
}
dependencyResolutionManagement {
versionCatalogs {
libs {
version('groovy', '3.0.5')
version('checkstyle', '8.37')
library('groovy-core', 'org.codehaus.groovy', 'groovy').versionRef('groovy')
library('groovy-json', 'org.codehaus.groovy', 'groovy-json').versionRef('groovy')
library('groovy-nio', 'org.codehaus.groovy', 'groovy-nio').versionRef('groovy')
library('commons-lang3', 'org.apache.commons', 'commons-lang3').version {
strictly '[3.8, 4.0['
prefer '3.9'
}
}
}
}
Don’t use libs for your programmatic version catalog name if you have the default libs.versions.toml in your project.
|
Publishing a catalog
There are three ways to share a version catalog across builds:
-
Check the TOML file into source control — The simplest approach. Place
gradle/libs.versions.tomlin a shared repository that all builds can access. This works well for monorepos or when teams can reference a common Git repository. -
Write a settings plugin — Create a plugin that programmatically declares catalog entries, publish it to the Gradle Plugin Portal or an internal repository, and have consumers apply it in their settings file. This offers the most flexibility but requires more effort.
-
Use the
version-catalogplugin — Publish the catalog as a Maven or Ivy artifact so it can be consumed like any other dependency. This is the best approach for organizations that need to distribute a catalog to many independent builds.
Using the version-catalog plugin
The version-catalog plugin lets you publish a catalog as a Maven or Ivy artifact.
It works with both an existing TOML file and programmatic catalog declarations.
If you already have a gradle/libs.versions.toml file, you can publish it directly by importing it into the catalog extension:
plugins {
`version-catalog`
`maven-publish`
}
catalog {
versionCatalog {
from(files("gradle/libs.versions.toml"))
}
}
group = "com.mycompany"
version = "1.0"
publishing {
publications {
create<MavenPublication>("maven") {
from(components["versionCatalog"])
}
}
}
plugins {
id 'version-catalog'
id 'maven-publish'
}
catalog {
versionCatalog {
from(files("gradle/libs.versions.toml"))
}
}
group = 'com.mycompany'
version = '1.0'
publishing {
publications {
maven(MavenPublication) {
from components.versionCatalog
}
}
}
This keeps the TOML file as the single source of truth — there is no need to redeclare entries in the build script.
Alternatively, you can declare the catalog entries programmatically in the catalog extension:
catalog {
// declare the aliases, bundles and versions in this block
versionCatalog {
library("my-lib", "com.mycompany:mylib:1.2")
}
}
catalog {
// declare the aliases, bundles and versions in this block
versionCatalog {
library('my-lib', 'com.mycompany:mylib:1.2')
}
}
In both cases, configure a publication using the versionCatalog component:
publishing {
publications {
create<MavenPublication>("maven") {
from(components["versionCatalog"])
}
}
}
publishing {
publications {
maven(MavenPublication) {
from components.versionCatalog
}
}
}
When you run ./gradlew publish, the catalog is uploaded as a Maven artifact (with the .toml extension).
This artifact can then be imported by other Gradle builds.
The published artifact is a standard TOML file attached to Maven coordinates (e.g., com.mycompany:catalog:1.0).
Consumers reference these coordinates to import the catalog — they do not need access to the original source code or build script.
|
For example, a consumer imports the published catalog in their settings.gradle(.kts) file:
dependencyResolutionManagement {
versionCatalogs {
create("libs") {
from("com.mycompany:catalog:1.0")
}
}
}
dependencyResolutionManagement {
versionCatalogs {
libs {
from("com.mycompany:catalog:1.0")
}
}
}
Changing the catalog name
By default, the libs.versions.toml file is used as input for the libs catalog.
However, you can rename the default catalog if an extension with the same name already exists:
dependencyResolutionManagement {
defaultLibrariesExtensionName = "projectLibs"
}
dependencyResolutionManagement {
defaultLibrariesExtensionName = 'projectLibs'
}
Overwriting catalog versions
You can overwrite versions when importing a catalog:
dependencyResolutionManagement {
versionCatalogs {
create("amendedLibs") {
from("com.mycompany:catalog:1.0")
// overwrite the "groovy" version declared in the imported catalog
version("groovy", "3.0.6")
}
}
}
dependencyResolutionManagement {
versionCatalogs {
amendedLibs {
from("com.mycompany:catalog:1.0")
// overwrite the "groovy" version declared in the imported catalog
version("groovy", "3.0.6")
}
}
}
In the examples above, any dependency referencing the groovy version will automatically be updated to use 3.0.6.
You can see how this is used in Working with multiple version catalogs.
| Overwriting a version only affects what is imported and used when declaring dependencies. The actual resolved dependency version may differ due to conflict resolution. |
Using a catalog in buildSrc
Version catalogs provide a centralized way to manage dependencies in a project.
However, buildSrc does not automatically inherit the version catalog from the main project, so additional configuration is required.
To access a version catalog inside buildSrc, you need to explicitly import it in buildSrc/settings.gradle(.kts):
// Add the version catalog to buildSrc using dependencyResolutionManagement
dependencyResolutionManagement {
versionCatalogs {
create("libs") {
from(files("../gradle/libs.versions.toml"))
}
}
}
// Add the version catalog to buildSrc using dependencyResolutionManagement
dependencyResolutionManagement {
versionCatalogs {
create("libs") {
from(files("../gradle/libs.versions.toml"))
}
}
}
Once the version catalog is imported, dependencies can be referenced using a few tricks:
plugins {
`kotlin-dsl`
alias(libs.plugins.versions) // Access version catalog in buildSrc build file for plugin
}
repositories {
gradlePluginPortal()
}
dependencies {
// Access version catalog in buildSrc build file for dependencies
implementation(plugin(libs.plugins.jacocolog)) // Plugin dependency
implementation(libs.groovy.core) // Regular library from version catalog
implementation("org.apache.commons:commons-lang3:3.9") // Direct dependency
}
// Helper function that transforms a Gradle Plugin alias from a
// Version Catalog into a valid dependency notation for buildSrc
fun DependencyHandlerScope.plugin(plugin: Provider<PluginDependency>) =
plugin.map { "${it.pluginId}:${it.pluginId}.gradle.plugin:${it.version}" }
plugins {
id 'groovy-gradle-plugin'
alias(libs.plugins.versions) // Access version catalog in buildSrc for plugin
}
repositories {
gradlePluginPortal()
}
def catalogs = project.extensions.getByType(VersionCatalogsExtension)
def libs = catalogs.named("libs")
dependencies {
// Access version catalog in buildSrc for dependencies
implementation plugin(libs.findPlugin("jacocolog").get()) // Plugin dependency
implementation libs.findLibrary("groovy-core").get() // Regular library from version catalog
implementation "org.apache.commons:commons-lang3:3.9" // Direct dependency
}
// Helper function that transforms a Gradle Plugin alias from a
// Version Catalog into a valid dependency notation for buildSrc
def plugin(Provider<PluginDependency> plugin) {
return plugin.map { it.pluginId + ":" + it.pluginId + ".gradle.plugin:" + it.version }
}
In precompiled script plugins inside buildSrc, the version catalog can be accessed using extensions.getByType(VersionCatalogsExtension) as demonstrated in the dependencies block of this convention plugin:
plugins {
id("java-library")
//alias(libs.plugins.jacocolog) // Unfortunately it is not possible the version catalog in buildSrc code for plugins
// Remember that unlike regular Gradle projects, convention plugins in buildSrc do not automatically resolve
// external plugins. We must declare them as dependencies in buildSrc/build.gradle.kts.
id("org.barfuin.gradle.jacocolog") // Apply the plugin manually as a workaround with the external plugin
// version from the version catalog specified in implementation dependency
// artifact in build file
}
repositories {
mavenCentral()
}
// Access the version catalog
val libs = extensions.getByType(VersionCatalogsExtension::class.java).named("libs")
//val libs = the<VersionCatalogsExtension>().named("libs")
dependencies {
// Access version catalog in buildSrc code for dependencies
implementation(libs.findLibrary("guava").get()) // Regular library from version catalog
testImplementation(platform("org.junit:junit-bom:5.9.1")) // Platform dependency
testImplementation("org.junit.jupiter:junit-jupiter") // Direct dependency
}
tasks.test {
useJUnitPlatform()
}
plugins {
id 'java-library'
// alias(libs.plugins.jacocolog) - Unfortunately, it is not possible to use the version catalog for plugins in buildSrc
// Unlike regular Gradle projects, convention plugins in buildSrc do not automatically resolve
// external plugins. We must declare them as dependencies in buildSrc/build.gradle.
id 'org.barfuin.gradle.jacocolog' // Apply the plugin manually as a workaround
// The external plugin version comes from the implementation dependency
// artifact in the build file
}
repositories {
mavenCentral()
}
// Access the version catalog
def libs = project.extensions.getByType(VersionCatalogsExtension).named("libs")
dependencies {
// Access version catalog in buildSrc for dependencies
implementation libs.findLibrary("guava").get() // Regular library from version catalog
testImplementation platform("org.junit:junit-bom:5.9.1") // Platform dependency
testImplementation "org.junit.jupiter:junit-jupiter" // Direct dependency
}
tasks.withType(Test).configureEach {
useJUnitPlatform()
}
However, the plugins block in the precompiled script plugin cannot access the version catalog.
Working with multiple version catalogs
Most builds should use a single libs.versions.toml file.
A single catalog centralizes all dependency coordinates, avoids version duplication, and lets every subproject share the same type-safe accessors.
Use naming conventions, such as alias prefixes, to logically separate groups of dependencies within that file:
[versions]
kotlin = "2.3.20"
junit = "5.12.0"
[libraries]
# ── Production ───────────────────────────────
guava = { module = "com.google.guava:guava", version = "33.4.0-jre" }
commons-lang3 = { module = "org.apache.commons:commons-lang3", version = "3.17.0" }
# ── Testing ──────────────────────────────────
junit-jupiter = { module = "org.junit.jupiter:junit-jupiter", version.ref = "junit" }
mockito-core = { module = "org.mockito:mockito-core", version = "5.14.0" }
Consider splitting into multiple catalogs only when you have a concrete need to enforce boundaries between dependency scopes. Typical reasons include:
-
Auditing shipped artifacts. When you must clearly distinguish which dependencies end up in a published distribution from those used only for testing or code generation, separate catalogs make accidental cross-contamination visible at the declaration site.
-
Independent build-logic dependencies. Plugins applied inside
build-logicorbuildSrcmay need their own catalog that is resolved in the settings classpath — before the main build’s catalog is available. -
Organizational sharing. An organization-wide "platform" catalog is published as a Maven artifact and imported by many repositories, while each repository maintains a local catalog for project-specific dependencies.
If none of these situations apply, a single catalog with clear naming conventions is simpler and avoids the version-sharing challenges described below.
Sharing Versions Across Multiple Catalogs
Each version catalog TOML file is self-contained.
There is no built-in syntax, such as an include directive or cross-file version.ref, that lets one TOML file reference a version declared in another.
If the same library version must appear in two catalogs, you must arrange for it to reach both catalogs yourself.
The from() method, which loads a catalog from a TOML file or a published artifact, may only be called once per catalog.
Attempting to call it twice produces an error:
In version catalog libs, you can only call the from method a single time.
|
This means you cannot merge two TOML files into a single catalog.
Below are three concrete workarounds, ordered from simplest to most flexible.
Workaround 1 — Duplicate and verify
Declare the shared version in every TOML file that needs it, and add a lightweight check that the values stay in sync:
[versions]
errorProne = "2.28.0"
[versions]
errorProne = "2.28.0"
Add a small verification task (or a CI script) that parses both files and fails the build if the values diverge:
tasks.register("verifySharedVersions") {
val distToml = file("gradle/distribution.versions.toml").readText()
val testToml = file("gradle/test.versions.toml").readText()
doLast {
// Extract the errorProne version from each file
val regex = """errorProne\s*=\s*"(.+?)"""".toRegex()
val distVersion = regex.find(distToml)?.groupValues?.get(1)
val testVersion = regex.find(testToml)?.groupValues?.get(1)
require(distVersion == testVersion) {
"errorProne version mismatch: distribution=$distVersion, test=$testVersion"
}
}
}
tasks.register('verifySharedVersions') {
def distToml = file('gradle/distribution.versions.toml').text
def testToml = file('gradle/test.versions.toml').text
doLast {
// Extract the errorProne version from each file
def regex = /errorProne\s*=\s*"(.+?)"/
def distMatcher = distToml =~ regex
def testMatcher = testToml =~ regex
def distVersion = distMatcher ? distMatcher[0][1] : null
def testVersion = testMatcher ? testMatcher[0][1] : null
assert distVersion == testVersion :
"errorProne version mismatch: distribution=$distVersion, test=$testVersion"
}
}
This approach is straightforward and keeps each TOML file independently readable, but it requires discipline and will not scale well beyond a handful of shared versions.
Workaround 2 — External properties file injected programmatically
Extract shared versions into a plain properties file and inject them into each catalog in settings.gradle(.kts).
errorProne=2.28.0
kotlin=2.3.20
val sharedVersions = java.util.Properties().apply {
file("gradle/shared-versions.properties").inputStream().use { load(it) }
}
dependencyResolutionManagement {
repositories {
mavenCentral()
}
versionCatalogs {
create("distributionLibs") {
from(files("gradle/distribution.versions.toml"))
version("errorProne", sharedVersions.getProperty("errorProne"))
version("kotlin", sharedVersions.getProperty("kotlin"))
}
create("testLibs") {
from(files("gradle/test.versions.toml"))
version("errorProne", sharedVersions.getProperty("errorProne"))
version("kotlin", sharedVersions.getProperty("kotlin"))
}
}
}
def sharedVersions = new Properties()
file('gradle/shared-versions.properties').withInputStream { sharedVersions.load(it) }
dependencyResolutionManagement {
repositories {
mavenCentral()
}
versionCatalogs {
distributionLibs {
from(files('gradle/distribution.versions.toml'))
version('errorProne', sharedVersions.getProperty('errorProne'))
version('kotlin', sharedVersions.getProperty('kotlin'))
}
testLibs {
from(files('gradle/test.versions.toml'))
version('errorProne', sharedVersions.getProperty('errorProne'))
version('kotlin', sharedVersions.getProperty('kotlin'))
}
}
}
The TOML files use version.ref to reference these versions as if they were declared locally:
# The "errorProne" and "kotlin" versions are injected
# programmatically in settings.gradle.kts — do not
# declare them here.
[libraries]
errorProne-core = { module = "com.google.errorprone:error_prone_core", version.ref = "errorProne" }
kotlin-stdlib = { module = "org.jetbrains.kotlin:kotlin-stdlib", version.ref = "kotlin" }
When you call version("errorProne", "2.28.0") on the catalog builder after from(files(…)), the programmatic call adds or overrides the version in the catalog that was loaded from the TOML file.
Libraries in the TOML file that reference version.ref = "errorProne" will pick up the injected value.
This is the pattern the Gradle Build Tool team uses in the gradle/gradle repository, where a shared-versions.properties file feeds versions into four separate catalogs (distribution, provided, test, build).
Workaround 3 — Published base catalog with local overrides
For organizations that share a common set of dependency versions across many repositories, publish a base catalog as a Maven artifact and let each repository layer project-specific versions on top.
Publishing project (build.gradle(.kts)):
plugins {
`version-catalog`
`maven-publish`
}
catalog {
versionCatalog {
version("kotlin", "2.3.20")
version("errorProne", "2.28.0")
library("kotlin-stdlib", "org.jetbrains.kotlin", "kotlin-stdlib").versionRef("kotlin")
library("errorProne-core", "com.google.errorprone", "error_prone_core").versionRef("errorProne")
}
}
group = "com.mycompany"
version = "1.0"
publishing {
publications {
create<MavenPublication>("catalog") {
from(components["versionCatalog"])
}
}
}
plugins {
id 'version-catalog'
id 'maven-publish'
}
catalog {
versionCatalog {
version('kotlin', '2.3.20')
version('errorProne', '2.28.0')
library('kotlin-stdlib', 'org.jetbrains.kotlin', 'kotlin-stdlib').versionRef('kotlin')
library('errorProne-core', 'com.google.errorprone', 'error_prone_core').versionRef('errorProne')
}
}
group = 'com.mycompany'
version = '1.0'
publishing {
publications {
maven(MavenPublication) {
from components.versionCatalog
}
}
}
Consuming project (settings.gradle(.kts)):
dependencyResolutionManagement {
versionCatalogs {
// Import the organization-wide catalog
create("libs") {
from("com.mycompany:catalog:1.0")
// Override a specific version for this project
version("kotlin", "2.3.21")
}
// A local catalog for project-specific dependencies
create("projectLibs") {
from(files("gradle/project.versions.toml"))
}
}
}
dependencyResolutionManagement {
versionCatalogs {
// Import the organization-wide catalog
libs {
from('com.mycompany:catalog:1.0')
// Override a specific version for this project
version('kotlin', '2.3.21')
}
// A local catalog for project-specific dependencies
projectLibs {
from(files('gradle/project.versions.toml'))
}
}
}
This pattern cleanly separates organizational standards from project-level concerns, but it introduces a publishing and versioning lifecycle for the catalog artifact itself.
Combining from() with Programmatic version() Calls
The version catalog builder API allows you to mix a TOML file import with programmatic declarations in the same catalog.
The from() call loads the base content, and subsequent version(), library(), bundle(), or plugin() calls add to or override that content.
dependencyResolutionManagement {
versionCatalogs {
create("libs") {
from(files("gradle/catalog.versions.toml"))
// Override an existing version
version("groovy", "3.0.6")
// Add a new version not present in the TOML file
version("errorProne", "2.28.0")
// Add a library that uses the injected version
library("errorProne-core", "com.google.errorprone", "error_prone_core").versionRef("errorProne")
}
}
}
dependencyResolutionManagement {
versionCatalogs {
libs {
from(files("gradle/catalog.versions.toml"))
// Override an existing version
version("groovy", "3.0.6")
// Add a new version not present in the TOML file
version("errorProne", "2.28.0")
// Add a library that uses the injected version
library("errorProne-core", "com.google.errorprone", "error_prone_core").versionRef("errorProne")
}
}
}
This works identically when importing from a published artifact:
create("libs") {
from("com.mycompany:catalog:1.0")
version("guava", "33.5.0-jre") // override the published version
}
libs {
from("com.mycompany:catalog:1.0")
version("guava", "33.5.0-jre") // override the published version
}
A common scenario is needing the same Kotlin version for both the kotlin("jvm") plugin and the kotlin-stdlib library dependency.
Plugins are resolved from the [plugins] section, while libraries come from [libraries] — and both need the same version.
[libraries]
kotlin-stdlib = { module = "org.jetbrains.kotlin:kotlin-stdlib", version.ref = "kotlin" }
kotlin-reflect = { module = "org.jetbrains.kotlin:kotlin-reflect", version.ref = "kotlin" }
[plugins]
kotlin-jvm = { id = "org.jetbrains.kotlin.jvm", version.ref = "kotlin" }
dependencyResolutionManagement {
versionCatalogs {
create("libs") {
from(files("gradle/catalog.versions.toml"))
// Single source of truth for the Kotlin version
version("kotlin", "2.3.20")
}
}
}
dependencyResolutionManagement {
versionCatalogs {
libs {
from(files("gradle/catalog.versions.toml"))
// Single source of truth for the Kotlin version
version("kotlin", "2.3.20")
}
}
}
Both the plugin alias and the library aliases resolve to 2.3.20, and updating requires changing only one line in settings.gradle(.kts).
Using catalog entries in Gradle APIs
A version catalog accessor such as libs.groovy.core returns a Provider<MinimalExternalModuleDependency>.
Calling .get() on it unwraps the provider and gives you the MinimalExternalModuleDependency directly.
In dependency declarations, both forms work identically:
dependencies {
// Provider<MinimalExternalModuleDependency>
implementation(libs.groovy.core)
// MinimalExternalModuleDependency
implementation(libs.groovy.core.get())
}
dependencies {
// Provider<MinimalExternalModuleDependency>
implementation(libs.groovy.core)
// MinimalExternalModuleDependency
implementation(libs.groovy.core.get())
}
However, in other Gradle APIs, such as dependency substitution rules, dependency constraints, or any API that accepts a dependency notation, try to pass the provider (libs.groovy.core) rather than the unwrapped value (libs.groovy.core.get()):
configurations.configureEach {
resolutionStrategy.dependencySubstitution {
all {
val componentSelector = requested
if (componentSelector is ModuleComponentSelector
&& componentSelector.group == "org.codehaus.groovy"
&& componentSelector.module == "groovy-all") {
useTarget(libs.groovy.core)
}
}
}
}
configurations.configureEach {
resolutionStrategy.dependencySubstitution {
all {
if (requested instanceof ModuleComponentSelector
&& requested.group == "org.codehaus.groovy"
&& requested.module == "groovy-all") {
useTarget(libs.groovy.core)
}
}
}
}
When you pass the provider, Gradle handles any necessary type conversions internally.
Unwrapping with .get() gives you a MinimalExternalModuleDependency, which implements ExternalDependency and ModuleVersionSelector — but some resolution APIs expect ModuleComponentSelector, a different type.
Passing the provider avoids mismatches between these types.