GitHub Action
Maven-Lockfile
This plugin is a state-of-the-art solution that validates the integrity of a maven build.
It does this by generating a lock file that contains the checksums of all the artifacts in the repository.
The lock file can then be used to validate the integrity prior to building.
This guards the supply chain against malicious actors that might tamper with the artifacts in the repository.
We also allow you to rebuild your old versions with the pinned versions from the lockfile with freeze
.
This plugin is available on maven central. See https://search.maven.org/artifact/io.github.chains-project/maven-lockfile for the latest version.
To generate a lock file, run the following command:
mvn io.github.chains-project:maven-lockfile:generate
This generates a lockfile.json file in each module of the repository, in readable JSON. This file contains the checksums of all the artifacts in the repository. The complete dependency tree, with transitive dependencies, is stored in the lockfile (akin a sbom). For multi-module projects, there is one lockfile per module.
Run the following command to validate the repository:
mvn io.github.chains-project:maven-lockfile:validate
If this runs successfully, the repository is valid. All dependencies defined are still the same as when the lock file was generated. If the command fails, this means a dependency has changed.
First create pom.lockfile.xml
mvn io.github.chains-project:maven-lockfile:freeze
This creates a new pom file with the default name pom.lockfile.xml
. A custom name can be passed with the flag pomLockfileOutput
.
In the new pom file, every version of direct dependencies in the original pom will be replaced with the versions from the lockfile. Also, every transitive dependency is added to the pom inside the dependencyManagement
section with the version and scope from the lockfile.
Then, invoke maven with the -f flag
mvn -f pom.lockfile.xml
reduced
will reduce the lockfile only containing the dependencies after dependency resolution conflicts are resolved. This format is smaller, and easier to review and read. Only use this if you do not need the full dependency tree.includeMavenPlugins
will include the maven plugins in the lockfile. This is useful if you want to validate the Maven plugins as well.allowValidationFailure
(default=false) allow validation failures, printing a warning instead of an error. This is useful if you want to only validate the Maven lockfile, but do not need to fail the build in case the lockfile is not valid. Use with caution, you loose all guarantees.includeEnvironment
will include the environment metadata in the lockfile. This is useful if you want to have warnings when the environment changes.checksumAlgorithm
will set the checksum algorithm used to generate the lockfile. The default depends on your checksum mode.checksumMode
will set the checksum mode used to generate the lockfile. See Checksum Modes for more information.skip
will skip the execution of the plugin. This is useful if you would like to disable the plugin for a specific module.getConfigFromFile
will read the configuration of maven lockfile from the existing lockfile.
An example lockfile is shown below: For a full example, see the lockfile.json file in this repository.
{
"artifactID":"my-app",
"groupID":"com.mycompany.app",
"version":"1",
"lockFileVersion":1,
"dependencies":[
{
"groupId":"org.junit.platform",
"artifactId":"junit-platform-engine",
"version":"1.9.2",
"checksumAlgorithm":"SHA-256",
"checksum":"25f23dc535a091e9dc80c008faf29dcb92be902e6911f77a736fbaf019908367",
"id":"org.junit.platform:junit-platform-engine:1.9.2",
"parent":"org.junit.jupiter:junit-jupiter-engine:5.9.2",
"children":[
{
"groupId":"org.apiguardian",
"artifactId":"apiguardian-api",
"version":"1.1.2",
"checksumAlgorithm":"SHA-256",
"checksum":"b509448ac506d607319f182537f0b35d71007582ec741832a1f111e5b5b70b38",
"id":"org.apiguardian:apiguardian-api:1.1.2",
"parent":"org.junit.platform:junit-platform-engine:1.9.2",
"children":[
]
},
{
"groupId":"org.junit.platform",
"artifactId":"junit-platform-commons",
"version":"1.9.2",
"checksumAlgorithm":"SHA-256",
"checksum":"624a3d745ef1d28e955a6a67af8edba0fdfc5c9bad680a73f67a70bb950a683d",
"id":"org.junit.platform:junit-platform-commons:1.9.2",
"parent":"org.junit.platform:junit-platform-engine:1.9.2",
"children":[
{
"groupId":"org.apiguardian",
"artifactId":"apiguardian-api",
"version":"1.1.2",
"checksumAlgorithm":"SHA-256",
"checksum":"b509448ac506d607319f182537f0b35d71007582ec741832a1f111e5b5b70b38",
"id":"org.apiguardian:apiguardian-api:1.1.2",
"parent":"org.junit.platform:junit-platform-commons:1.9.2",
"children":[
]
}
]
},
{
"groupId":"org.opentest4j",
"artifactId":"opentest4j",
"version":"1.2.0",
"checksumAlgorithm":"SHA-256",
"checksum":"58812de60898d976fb81ef3b62da05c6604c18fd4a249f5044282479fc286af2",
"id":"org.opentest4j:opentest4j:1.2.0",
"parent":"org.junit.platform:junit-platform-engine:1.9.2",
"children":[
]
}
]
}
]
}
This is close to the format of the lock file in the npm package-lock.json file.
We made some java-specific changes to the format, e.g., we added the groupId
field.
For each artifact, we store the hashes of all transitive dependencies in the children
field.
This allows us to validate the integrity of the transitive dependencies as well.
We have created a GithubAction that can be used to validate the integrity of your maven
repository.
A sample workflow is shown below:
Usage:
name: Lockfile
on:
pull_request:
permissions:
contents: read
jobs:
check-lockfile:
permissions:
contents: write
runs-on: ubuntu-latest
steps:
- name: run maven-lockfile
uses: chains-project/maven-lockfile@526cd67327ab19c7bd95be6d2d16530d80bf3c9e # v5.0.0
with:
github-token: ${{ secrets.JRELEASER_GITHUB_TOKEN }}
include-maven-plugins: true
If a pom.xml or lockfile.json file is changed, this action will add a commit with the updated lockfile to the pull request. Otherwise, it will validate the lockfile and fail if the lockfile is incorrect. A lockfile is incorrect if any dependency has changed since the lockfile was generated. This includes versions and checksums.
Here we list some related work that we found while researching this topic.
- Maven: https://github.com/vandmo/dependency-lock-maven-plugin
- Gradle: For Gradle, there exists a built-in solution: https://docs.gradle.org/current/userguide/dependency_locking.html. This solution only works for Gradle builds and is deeply connected to the Gradle build system. The Gradle ecosystem is fast changing and so is its dependency resolution. Our lockfile is independent of the build system and can be used to validate the integrity of a maven repository.
- NPM: https://docs.npmjs.com/cli/v9/configuring-npm/package-lock-json