2019-12-11 12:41:27 -05:00
|
|
|
// This adds validation of project dependencies:
|
|
|
|
// 1) license file
|
|
|
|
// 2) notice file
|
|
|
|
// 3) checksum validation/ generation.
|
|
|
|
|
|
|
|
import org.apache.commons.codec.digest.DigestUtils
|
|
|
|
import org.apache.commons.codec.digest.MessageDigestAlgorithms
|
|
|
|
|
2019-12-13 06:12:29 -05:00
|
|
|
// This should be false only for debugging.
|
|
|
|
def failOnError = true
|
2019-12-13 06:01:26 -05:00
|
|
|
|
2019-12-17 08:27:25 -05:00
|
|
|
// We're using commons-codec for computing checksums.
|
2019-12-11 12:41:27 -05:00
|
|
|
buildscript {
|
|
|
|
repositories {
|
|
|
|
mavenCentral()
|
|
|
|
}
|
|
|
|
|
|
|
|
dependencies {
|
|
|
|
classpath 'commons-codec:commons-codec:1.13'
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Configure license checksum folder for top-level projects.
|
|
|
|
// (The file("licenses") inside the configure scope resolves
|
|
|
|
// relative to the current project so they're not the same).
|
|
|
|
configure(project(":lucene")) {
|
|
|
|
ext.licensesDir = file("licenses")
|
|
|
|
}
|
|
|
|
configure(project(":solr")) {
|
2019-12-30 08:05:08 -05:00
|
|
|
ext.licensesDir = file("licenses")
|
2019-12-11 12:41:27 -05:00
|
|
|
}
|
|
|
|
|
2019-12-12 13:25:46 -05:00
|
|
|
// All known license types. If 'noticeOptional' is true then
|
|
|
|
// the notice file must accompany the license.
|
|
|
|
def licenseTypes = [
|
2019-12-17 08:27:25 -05:00
|
|
|
"ASL" : [name: "Apache Software License 2.0"],
|
|
|
|
"BSD" : [name: "Berkeley Software Distribution"],
|
2019-12-12 13:25:46 -05:00
|
|
|
//BSD like just means someone has taken the BSD license and put in their name, copyright, or it's a very similar license.
|
|
|
|
"BSD_LIKE": [name: "BSD like license"],
|
2019-12-17 08:27:25 -05:00
|
|
|
"CDDL" : [name: "Common Development and Distribution License", noticeOptional: true],
|
|
|
|
"CPL" : [name: "Common Public License"],
|
|
|
|
"EPL" : [name: "Eclipse Public License Version 1.0", noticeOptional: true],
|
|
|
|
"MIT" : [name: "Massachusetts Institute of Tech. License", noticeOptional: true],
|
|
|
|
"MPL" : [name: "Mozilla Public License", noticeOptional: true /* NOT SURE on the required notice */],
|
|
|
|
"PD" : [name: "Public Domain", noticeOptional: true],
|
|
|
|
"SUN" : [name: "Sun Open Source License", noticeOptional: true],
|
2019-12-12 13:25:46 -05:00
|
|
|
"COMPOUND": [name: "Compound license (details in NOTICE file)."],
|
|
|
|
]
|
|
|
|
|
2019-12-13 11:09:25 -05:00
|
|
|
allprojects {
|
|
|
|
task licenses() {
|
|
|
|
group = 'Dependency validation'
|
|
|
|
description = "Apply all dependency/ license checks."
|
|
|
|
}
|
|
|
|
check.dependsOn(licenses)
|
|
|
|
}
|
|
|
|
|
2019-12-11 12:41:27 -05:00
|
|
|
subprojects {
|
|
|
|
// Configure jarValidation configuration for all projects. Any dependency
|
|
|
|
// declared on this configuration (or any configuration it extends from) will
|
|
|
|
// be verified.
|
|
|
|
configurations {
|
|
|
|
jarValidation
|
|
|
|
}
|
|
|
|
|
2019-12-13 07:31:23 -05:00
|
|
|
// For Java projects, add all dependencies from the following configurations
|
|
|
|
// to jar validation
|
2019-12-11 12:41:27 -05:00
|
|
|
plugins.withType(JavaPlugin) {
|
|
|
|
configurations {
|
|
|
|
jarValidation {
|
|
|
|
extendsFrom runtimeClasspath
|
|
|
|
extendsFrom compileClasspath
|
2019-12-13 07:31:23 -05:00
|
|
|
extendsFrom testRuntimeClasspath
|
|
|
|
extendsFrom testCompileClasspath
|
2019-12-11 12:41:27 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-12 13:25:46 -05:00
|
|
|
// Collects dependency JAR information for a project and saves it in
|
|
|
|
// project.ext.jarInfos. Each dependency has a map of attributes
|
|
|
|
// which make it easier to process it later on (name, hash, origin module,
|
|
|
|
// see the code below for details).
|
2019-12-11 12:41:27 -05:00
|
|
|
task collectJarInfos() {
|
|
|
|
dependsOn configurations.jarValidation
|
|
|
|
|
|
|
|
doFirst {
|
2019-12-17 09:02:08 -05:00
|
|
|
def isSolr = project.path.startsWith(":solr")
|
|
|
|
|
2019-12-30 08:05:08 -05:00
|
|
|
// When gradle resolves a configuration it applies exclude rules from inherited configurations
|
|
|
|
// globally (this seems like a bug to me). So we process each inherited configuration independently
|
|
|
|
// but make sure there are no other dependencies on jarValidation itself.
|
|
|
|
if (!configurations.jarValidation.dependencies.isEmpty()) {
|
|
|
|
throw new GradleException("jarValidation must only inherit from other configurations (can't have its own dependencies).")
|
|
|
|
}
|
|
|
|
|
|
|
|
def excludeRules = configurations.jarValidation.excludeRules
|
2019-12-11 12:41:27 -05:00
|
|
|
|
2019-12-30 08:05:08 -05:00
|
|
|
ArrayDeque<ResolvedDependency> queue = new ArrayDeque<>()
|
|
|
|
configurations.jarValidation.extendsFrom.each { conf ->
|
|
|
|
if (excludeRules) {
|
|
|
|
conf = configurations.detachedConfiguration().extendsFrom(conf)
|
|
|
|
conf.excludeRules = excludeRules
|
|
|
|
}
|
|
|
|
if (conf.canBeResolved) {
|
|
|
|
queue.addAll(conf.resolvedConfiguration.firstLevelModuleDependencies)
|
2019-12-17 09:02:08 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-30 08:05:08 -05:00
|
|
|
def visited = new HashSet<>()
|
|
|
|
def infos = []
|
|
|
|
|
|
|
|
while (!queue.isEmpty()) {
|
|
|
|
def dep = queue.removeFirst()
|
|
|
|
|
|
|
|
// Skip any artifacts from other Solr modules (they will be resolved there).
|
|
|
|
if (dep.moduleGroup == "org.apache.solr") {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
// Skip any artifacts from Lucene modules.
|
|
|
|
if (dep.moduleGroup.startsWith("org.apache.lucene")) {
|
|
|
|
// ... but process their transitive dependencies for Solr compatibility.
|
|
|
|
if (isSolr) {
|
|
|
|
queue.addAll(dep.children)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
queue.addAll(dep.children)
|
|
|
|
dep.moduleArtifacts.each { resolvedArtifact ->
|
|
|
|
def file = resolvedArtifact.file
|
|
|
|
if (visited.add(file)) {
|
|
|
|
infos.add([
|
|
|
|
name : resolvedArtifact.name,
|
|
|
|
jarName : file.toPath().getFileName().toString(),
|
|
|
|
path : file,
|
|
|
|
module : resolvedArtifact.moduleVersion,
|
|
|
|
checksum : provider { new DigestUtils(MessageDigestAlgorithms.SHA_1).digestAsHex(file).trim() },
|
|
|
|
// We keep track of the files referenced by this dependency (sha, license, notice, etc.)
|
|
|
|
// so that we can determine unused dangling files later on.
|
|
|
|
referencedFiles: []
|
|
|
|
])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-12-11 12:41:27 -05:00
|
|
|
}
|
2019-12-30 08:05:08 -05:00
|
|
|
|
|
|
|
project.ext.jarInfos = infos.sort {a, b -> "${a.module}".compareTo("${b.module}")}
|
|
|
|
// jarInfos.each { info -> println "${info.module}" }
|
2019-12-11 12:41:27 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-12 13:25:46 -05:00
|
|
|
// Verifies that each JAR has a corresponding checksum and that it matches actual JAR available for this dependency.
|
2019-12-11 12:41:27 -05:00
|
|
|
task validateJarChecksums() {
|
|
|
|
group = 'Dependency validation'
|
2019-12-12 13:25:46 -05:00
|
|
|
description = "Validate checksums of dependencies"
|
2019-12-11 12:41:27 -05:00
|
|
|
dependsOn collectJarInfos
|
|
|
|
|
|
|
|
doLast {
|
|
|
|
def errors = []
|
|
|
|
jarInfos.each { dep ->
|
|
|
|
def expectedChecksumFile = file("${licensesDir}/${dep.jarName}.sha1")
|
|
|
|
if (!expectedChecksumFile.exists()) {
|
|
|
|
errors << "Dependency checksum missing ('${dep.module}'), expected it at: ${expectedChecksumFile}"
|
|
|
|
} else {
|
2019-12-13 07:31:23 -05:00
|
|
|
dep.referencedFiles += expectedChecksumFile
|
2019-12-11 12:41:27 -05:00
|
|
|
def expected = expectedChecksumFile.getText("UTF-8").trim()
|
2019-12-18 08:54:13 -05:00
|
|
|
def actual = dep.checksum.get()
|
2019-12-11 12:41:27 -05:00
|
|
|
if (expected.compareToIgnoreCase(actual) != 0) {
|
|
|
|
errors << "Dependency checksum mismatch ('${dep.module}'), expected it to be: ${expected}, but was: ${actual}"
|
2019-12-12 13:25:46 -05:00
|
|
|
} else {
|
|
|
|
logger.log(LogLevel.INFO, "Dependency checksum OK ('${dep.module}')")
|
2019-12-11 12:41:27 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (errors) {
|
|
|
|
def msg = "Dependency checksum validation failed:\n - " + errors.join("\n - ")
|
2019-12-12 13:25:46 -05:00
|
|
|
if (failOnError) {
|
|
|
|
throw new GradleException(msg)
|
|
|
|
} else {
|
|
|
|
logger.log(LogLevel.WARN, "WARNING: ${msg}")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Locate the set of license file candidates for this dependency. We
|
|
|
|
// search for [jar-or-prefix]-LICENSE-[type].txt
|
|
|
|
// where 'jar-or-prefix' can be any '-'-delimited prefix of the dependency JAR's name.
|
|
|
|
// So for 'commons-io' it can be 'commons-io-LICENSE-foo.txt' or
|
|
|
|
// 'commons-LICENSE.txt'
|
|
|
|
task validateJarLicenses() {
|
|
|
|
group = 'Dependency validation'
|
|
|
|
description = "Validate license and notice files of dependencies"
|
|
|
|
dependsOn collectJarInfos
|
|
|
|
|
|
|
|
doLast {
|
|
|
|
def errors = []
|
|
|
|
jarInfos.each { dep ->
|
|
|
|
def baseName = dep.name
|
|
|
|
def found = []
|
|
|
|
def candidates = []
|
|
|
|
while (true) {
|
|
|
|
candidates += file("${licensesDir}/${baseName}-LICENSE-[type].txt")
|
|
|
|
found += fileTree(dir: licensesDir, include: "${baseName}-LICENSE-*.txt").files
|
|
|
|
def prefix = baseName.replaceAll(/[\-][^-]+$/, "")
|
|
|
|
if (found || prefix == baseName) {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
baseName = prefix
|
|
|
|
}
|
|
|
|
|
|
|
|
if (found.size() == 0) {
|
|
|
|
errors << "License file missing ('${dep.module}'), expected it at: ${candidates.join(" or ")}," +
|
|
|
|
" where [type] can be any of ${licenseTypes.keySet()}."
|
|
|
|
} else if (found.size() > 1) {
|
|
|
|
errors << "Multiple license files matching for ('${dep.module}'): ${found.join(", ")}"
|
|
|
|
} else {
|
|
|
|
def licenseFile = found.get(0)
|
2019-12-13 07:31:23 -05:00
|
|
|
dep.referencedFiles += licenseFile
|
2019-12-12 13:25:46 -05:00
|
|
|
def m = (licenseFile.name =~ /LICENSE-(.+)\.txt$/)
|
|
|
|
if (!m) throw new GradleException("License file name doesn't contain license type?: ${licenseFile.name}")
|
|
|
|
|
|
|
|
def licenseName = m[0][1]
|
|
|
|
def licenseType = licenseTypes[licenseName]
|
|
|
|
if (!licenseType) {
|
|
|
|
errors << "Unknown license type suffix for ('${dep.module}'): ${licenseFile} (must be one of ${licenseTypes.keySet()})"
|
|
|
|
} else {
|
|
|
|
logger.log(LogLevel.INFO, "Dependency license file OK ('${dep.module}'): " + licenseName)
|
|
|
|
|
|
|
|
// Look for sibling NOTICE file.
|
|
|
|
def noticeFile = file(licenseFile.path.replaceAll(/\-LICENSE-.+/, "-NOTICE.txt"))
|
|
|
|
if (noticeFile.exists()) {
|
2019-12-13 07:31:23 -05:00
|
|
|
dep.referencedFiles += noticeFile
|
2019-12-12 13:25:46 -05:00
|
|
|
logger.log(LogLevel.INFO, "Dependency notice file OK ('${dep.module}'): " + noticeFile)
|
|
|
|
} else if (!licenseType.noticeOptional) {
|
|
|
|
errors << "Notice file missing for ('${dep.module}'), expected it at: ${noticeFile}"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (errors) {
|
|
|
|
def msg = "Certain license/ notice files are missing:\n - " + errors.join("\n - ")
|
|
|
|
if (failOnError) {
|
2019-12-11 12:41:27 -05:00
|
|
|
throw new GradleException(msg)
|
|
|
|
} else {
|
|
|
|
logger.log(LogLevel.WARN, "WARNING: ${msg}")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-12-12 13:25:46 -05:00
|
|
|
|
2019-12-13 11:09:25 -05:00
|
|
|
licenses.dependsOn validateJarChecksums, validateJarLicenses
|
2019-12-11 12:41:27 -05:00
|
|
|
}
|
|
|
|
|
2019-12-18 08:54:13 -05:00
|
|
|
// Add top-project level tasks validating dangling files
|
|
|
|
// and regenerating dependency checksums.
|
2019-12-17 08:27:25 -05:00
|
|
|
configure([project(":solr"), project(":lucene"),]) {
|
2019-12-18 08:54:13 -05:00
|
|
|
def validationTasks = subprojects.collectMany { it.tasks.matching { it.name == "licenses" } }
|
|
|
|
def jarInfoTasks = subprojects.collectMany { it.tasks.matching { it.name == "collectJarInfos" } }
|
2019-12-17 08:27:25 -05:00
|
|
|
|
|
|
|
// Update dependency checksums.
|
2019-12-18 08:14:39 -05:00
|
|
|
task updateLicenses() {
|
2019-12-17 08:27:25 -05:00
|
|
|
group = 'Dependency validation'
|
|
|
|
description = "Write or update checksums of dependencies"
|
|
|
|
dependsOn jarInfoTasks
|
|
|
|
|
|
|
|
doLast {
|
|
|
|
licensesDir.mkdirs()
|
|
|
|
|
|
|
|
// Clean any previous checksums. In theory we wouldn't have to do it --
|
|
|
|
// dangling files from any previous JARs would be reported;
|
|
|
|
// it automates the process of updating versions and makes it easier though so
|
|
|
|
// why not.
|
|
|
|
project.delete fileTree(licensesDir, {
|
|
|
|
include "*.sha1"
|
|
|
|
exclude checkDanglingLicenseFiles.ext.exclude
|
|
|
|
})
|
|
|
|
|
|
|
|
def updated = []
|
|
|
|
jarInfoTasks.collectMany { task -> task.project.jarInfos }.each { dep ->
|
|
|
|
def expectedChecksumFile = file("${licensesDir}/${dep.jarName}.sha1")
|
2019-12-18 08:54:13 -05:00
|
|
|
def actual = dep.checksum.get()
|
2019-12-17 08:27:25 -05:00
|
|
|
if (expectedChecksumFile.exists()) {
|
|
|
|
def expected = expectedChecksumFile.getText("UTF-8").trim()
|
|
|
|
if (expected.compareToIgnoreCase(actual) == 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
updated += "Updated checksum ('${dep.module}'): ${expectedChecksumFile}"
|
2019-12-18 08:54:13 -05:00
|
|
|
expectedChecksumFile.write(actual + "\n", "UTF-8")
|
2019-12-17 08:27:25 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
updated.sort().each { line -> logger.log(LogLevel.LIFECYCLE, line) }
|
|
|
|
}
|
|
|
|
}
|
2019-12-18 08:54:13 -05:00
|
|
|
|
|
|
|
// Any validation task must run after all updates have been applied.
|
|
|
|
// We add an ordering constraint that any validation task (or its dependency subgraph)
|
|
|
|
// must run after updateLicenses
|
|
|
|
validationTasks
|
|
|
|
.collectMany { task -> [task, task.dependsOn]}
|
|
|
|
.flatten()
|
|
|
|
.each { task ->
|
|
|
|
task.mustRunAfter updateLicenses
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check for dangling files in the licenses folder.
|
|
|
|
task checkDanglingLicenseFiles() {
|
|
|
|
dependsOn validationTasks
|
|
|
|
|
|
|
|
ext {
|
|
|
|
exclude = []
|
|
|
|
}
|
|
|
|
|
|
|
|
doFirst {
|
|
|
|
def allReferenced = validationTasks.collectMany { task ->
|
|
|
|
task.project.jarInfos.collectMany { it.referencedFiles }
|
|
|
|
}.collect { it.toString() }
|
|
|
|
|
|
|
|
def patterns = ext.exclude
|
|
|
|
def allExisting = fileTree(licensesDir, {
|
|
|
|
exclude patterns
|
|
|
|
}).files.collect { it.toString() }
|
|
|
|
|
|
|
|
def dangling = (allExisting - allReferenced).sort()
|
|
|
|
|
|
|
|
if (dangling) {
|
|
|
|
gradle.buildFinished {
|
|
|
|
logger.warn("WARNING: there were unreferenced files under license folder:\n - ${dangling.join("\n - ")}")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
licenses.dependsOn checkDanglingLicenseFiles
|
2019-12-13 09:07:59 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// Exclude files that are not a result of direct dependencies but have to be there.
|
|
|
|
// It would be probably better to move non-dependency licenses into the actual project
|
|
|
|
// where they're used and only assemble them for the distribution package.
|
|
|
|
configure(project(":lucene")) {
|
|
|
|
checkDanglingLicenseFiles {
|
|
|
|
exclude += [
|
|
|
|
"elegant-icon-font-*",
|
|
|
|
"ant-*",
|
|
|
|
"ivy-*",
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
configure(project(":solr")) {
|
|
|
|
checkDanglingLicenseFiles {
|
|
|
|
exclude += [
|
|
|
|
"README.committers.txt",
|
2019-12-30 08:05:08 -05:00
|
|
|
|
|
|
|
// solr-ref-guide compilation-only dependencies.
|
|
|
|
"android-json-*",
|
|
|
|
"ant-*",
|
|
|
|
"asciidoctor-ant-*",
|
|
|
|
"jsoup-*",
|
|
|
|
"junit4-ant-*",
|
|
|
|
"slf4j-simple-*",
|
|
|
|
"start.jar.sha1"
|
2019-12-13 09:07:59 -05:00
|
|
|
]
|
|
|
|
}
|
2019-12-13 07:31:23 -05:00
|
|
|
}
|
|
|
|
|
2019-12-17 08:27:25 -05:00
|
|
|
// solr-ref-guide doesn't contribute any JARs to dependency checks.
|
2019-12-11 12:41:27 -05:00
|
|
|
configure(project(":solr:solr-ref-guide")) {
|
2019-12-17 08:27:25 -05:00
|
|
|
configurations {
|
|
|
|
jarValidation {
|
|
|
|
exclude group: "*"
|
|
|
|
}
|
2019-12-12 13:25:46 -05:00
|
|
|
}
|
2019-12-30 08:05:08 -05:00
|
|
|
}
|