forked from elastic/elasticsearch
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Validate that stable plugins do not break compatibility (elastic#92776)
We need to verify, for each release, that our stable plugin APIs are not breaking. This commit adds some Gradle support for basic backwards compatibility testing. On the Gradle side, we add a new qa project to test the current commit against downloads of released versions, and against fresh builds of snapshot versions. As for the actual comparison, we break up the output of javap (the decompiler) by line and create maps of classes to public class, field, and method declarations within those class files. We then check that the signature map from the new jar is not missing any elements present in the old jar. This method has known limitations, which are documented in the JarApiComparisonTask class. Co-authored-by: Mark Vieira <[email protected]>
- Loading branch information
1 parent
e22e024
commit c9e8101
Showing
9 changed files
with
334 additions
and
2 deletions.
There are no files selected for viewing
50 changes: 50 additions & 0 deletions
50
build-tools-internal/src/main/groovy/elasticsearch.stable-api.gradle
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,50 @@ | ||
import org.apache.tools.ant.taskdefs.condition.Os | ||
import org.elasticsearch.gradle.Version | ||
import org.elasticsearch.gradle.VersionProperties | ||
import org.elasticsearch.gradle.internal.BwcVersions | ||
import org.elasticsearch.gradle.internal.JarApiComparisonTask | ||
import org.elasticsearch.gradle.internal.info.BuildParams | ||
|
||
import static org.elasticsearch.gradle.internal.InternalDistributionBwcSetupPlugin.buildBwcTaskName | ||
|
||
configurations { | ||
newJar | ||
} | ||
|
||
dependencies { | ||
newJar project(":libs:${project.name}") | ||
} | ||
|
||
BuildParams.bwcVersions.withIndexCompatible({ it.onOrAfter(Version.fromString(ext.stableApiSince)) | ||
&& it != VersionProperties.elasticsearchVersion | ||
}) { bwcVersion, baseName -> | ||
|
||
BwcVersions.UnreleasedVersionInfo unreleasedVersion = BuildParams.bwcVersions.unreleasedInfo(bwcVersion) | ||
|
||
configurations { | ||
"oldJar${baseName}" { | ||
transitive = false | ||
} | ||
} | ||
|
||
dependencies { | ||
if (unreleasedVersion) { | ||
// For unreleased snapshot versions, build them from source | ||
"oldJar${baseName}"(files(project(unreleasedVersion.gradleProjectPath).tasks.named(buildBwcTaskName(project.name)))) | ||
} else { | ||
// For released versions, download it | ||
"oldJar${baseName}"("org.elasticsearch:${project.name}:${bwcVersion}") | ||
} | ||
} | ||
|
||
def jarApiComparisonTask = tasks.register(bwcTaskName(bwcVersion), JarApiComparisonTask) { | ||
oldJar = configurations."oldJar${baseName}" | ||
newJar = configurations.newJar | ||
} | ||
|
||
jarApiComparisonTask.configure { | ||
onlyIf { | ||
!Os.isFamily(Os.FAMILY_WINDOWS) | ||
} | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
224 changes: 224 additions & 0 deletions
224
...-tools-internal/src/main/java/org/elasticsearch/gradle/internal/JarApiComparisonTask.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,224 @@ | ||
/* | ||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one | ||
* or more contributor license agreements. Licensed under the Elastic License | ||
* 2.0 and the Server Side Public License, v 1; you may not use this file except | ||
* in compliance with, at your election, the Elastic License 2.0 or the Server | ||
* Side Public License, v 1. | ||
*/ | ||
|
||
package org.elasticsearch.gradle.internal; | ||
|
||
import org.elasticsearch.gradle.internal.conventions.precommit.PrecommitTask; | ||
import org.gradle.api.file.FileCollection; | ||
import org.gradle.api.provider.Property; | ||
import org.gradle.api.tasks.CacheableTask; | ||
import org.gradle.api.tasks.CompileClasspath; | ||
import org.gradle.api.tasks.TaskAction; | ||
|
||
import java.io.BufferedReader; | ||
import java.io.File; | ||
import java.io.IOException; | ||
import java.io.InputStream; | ||
import java.io.InputStreamReader; | ||
import java.util.ArrayList; | ||
import java.util.HashMap; | ||
import java.util.HashSet; | ||
import java.util.List; | ||
import java.util.Map; | ||
import java.util.Set; | ||
import java.util.jar.JarFile; | ||
import java.util.regex.Pattern; | ||
import java.util.stream.Collectors; | ||
import java.util.zip.ZipEntry; | ||
|
||
/** | ||
* This implementation of a jar API comparison uses the "javap" tool to compare | ||
* the "signatures" of two different jars. We assume that calling out to javap | ||
* is not too expensive at this stage of the stable API project. We also assume | ||
* that for every public class, method, and field, javap will print a consistent | ||
* single line. This should let us make string comparisons, rather than having | ||
* to parse the output of javap. | ||
* <p> | ||
* While the above assumptions appear to hold, they are not guaranteed, and hence | ||
* brittle. We could overcome these problems with an ASM implementation of the | ||
* Jar Scanner. | ||
* <p> | ||
* We also assume that we will not be comparing multi-version JARs. | ||
* <p> | ||
* This "javap" approach has a few further drawbacks: | ||
* <ol> | ||
* <li>We don't account for class visibility when examining fields and methods.</li> | ||
* <li>We don't consider what is exported from the module. Is a public method from | ||
* a non-exported package considered part of the stable api?</li> | ||
* <li>Changing method types to their superclass or return types to an implementation | ||
* class will be considered a change by this approach, even though that doesn't break | ||
* an API.</li> | ||
* <li>Finally, moving a method up the class hierarchy is not really a breaking change, | ||
* but it will trip this test.</li> | ||
* </ol> | ||
*/ | ||
@CacheableTask | ||
public abstract class JarApiComparisonTask extends PrecommitTask { | ||
|
||
@TaskAction | ||
public void compare() { | ||
FileCollection fileCollection = getOldJar().get(); | ||
File newJarFile = getNewJar().get().getSingleFile(); | ||
|
||
Set<String> oldJarNames = fileCollection.getFiles().stream().map(File::getName).collect(Collectors.toSet()); | ||
if (oldJarNames.size() > 1) { | ||
throw new IllegalStateException("Expected a single original jar, but found: " + oldJarNames); | ||
} | ||
if (oldJarNames.contains(newJarFile.getName())) { | ||
throw new IllegalStateException( | ||
"We should be comparing different jars, but original and new jars were both: " + newJarFile.getAbsolutePath() | ||
); | ||
} | ||
|
||
JarScanner oldJS = new JarScanner(getOldJar().get().getSingleFile().getPath()); | ||
JarScanner newJS = new JarScanner(newJarFile.getPath()); | ||
try { | ||
JarScanner.compareSignatures(oldJS.jarSignature(), newJS.jarSignature()); | ||
} catch (IOException e) { | ||
throw new RuntimeException(e); | ||
} | ||
} | ||
|
||
@CompileClasspath | ||
public abstract Property<FileCollection> getOldJar(); | ||
|
||
@CompileClasspath | ||
public abstract Property<FileCollection> getNewJar(); | ||
|
||
public static class JarScanner { | ||
|
||
private final String path; | ||
|
||
public JarScanner(String path) { | ||
this.path = path; | ||
} | ||
|
||
private String getPath() { | ||
return path; | ||
} | ||
|
||
/** | ||
* Get a list of class names contained in this jar by looking for file names | ||
* that end in ".class" | ||
*/ | ||
List<String> classNames() throws IOException { | ||
Pattern classEnding = Pattern.compile(".*\\.class$"); | ||
try (JarFile jf = new JarFile(this.path)) { | ||
return jf.stream().map(ZipEntry::getName).filter(classEnding.asMatchPredicate()).collect(Collectors.toList()); | ||
} | ||
} | ||
|
||
/** | ||
* Given a path to a file in the jar, get the output of javap as a list of strings. | ||
*/ | ||
public List<String> disassembleFromJar(String fileInJarPath, String classpath) { | ||
String location = "jar:file://" + getPath() + "!/" + fileInJarPath; | ||
return disassemble(location, getPath(), classpath); | ||
} | ||
|
||
/** | ||
* Invoke javap on a class file, optionally providing a module path or class path | ||
*/ | ||
static List<String> disassemble(String location, String modulePath, String classpath) { | ||
ProcessBuilder pb = new ProcessBuilder(); | ||
List<String> command = new ArrayList<>(); | ||
command.add("javap"); | ||
if (modulePath != null) { | ||
command.add("--module-path"); | ||
command.add(modulePath); | ||
} | ||
if (classpath != null) { | ||
command.add("--class-path"); | ||
command.add(classpath); | ||
} | ||
command.add(location); | ||
pb.command(command.toArray(new String[] {})); | ||
Process p; | ||
try { | ||
p = pb.start(); | ||
p.onExit().get(); | ||
} catch (Exception e) { | ||
throw new RuntimeException(e); | ||
} | ||
|
||
InputStream streamToRead = p.exitValue() == 0 ? p.getInputStream() : p.getErrorStream(); | ||
|
||
try (BufferedReader br = new BufferedReader(new InputStreamReader(streamToRead))) { | ||
return br.lines().toList(); | ||
} catch (IOException e) { | ||
throw new RuntimeException(e); | ||
} | ||
} | ||
|
||
/** | ||
* Given the output of the javap command, that is, the disassembled class file, | ||
* return a set of signatures for all public classes, methods, and fields. | ||
*/ | ||
public static Set<String> signaturesSet(List<String> javapOutput) { | ||
return javapOutput.stream().filter(s -> s.matches("^\\s*public.*")).collect(Collectors.toSet()); | ||
} | ||
|
||
/** | ||
* Given a disassembled module-info.class, return all unqualified exports. | ||
*/ | ||
public static Set<String> moduleInfoSignaturesSet(List<String> javapOutput) { | ||
return javapOutput.stream() | ||
.filter(s -> s.matches("^\\s*exports.*")) | ||
.filter(s -> s.matches(".* to$") == false) | ||
.collect(Collectors.toSet()); | ||
} | ||
|
||
/** | ||
* Iterate over classes and gather signatures. | ||
*/ | ||
public Map<String, Set<String>> jarSignature() throws IOException { | ||
return this.classNames().stream().collect(Collectors.toMap(s -> s, s -> { | ||
List<String> disassembled = disassembleFromJar(s, null); | ||
if ("module-info.class".equals(s)) { | ||
return moduleInfoSignaturesSet(disassembled); | ||
} | ||
return signaturesSet(disassembled); | ||
})); | ||
} | ||
|
||
/** | ||
* Comparison: The signatures are maps of class names to public class, field, or method | ||
* declarations. | ||
* </p> | ||
* First, we check that the new jar signature contains all the same classes | ||
* as the old jar signature. If not, we return an error. | ||
* </p> | ||
* Second, we iterate over the signature for each class. If a signature from the old | ||
* jar is absent in the new jar, we add it to our list of errors. | ||
* </p> | ||
* Note that it is fine for the new jar to have additional elements, as this | ||
* is backwards compatible. | ||
*/ | ||
public static void compareSignatures(Map<String, Set<String>> oldSignature, Map<String, Set<String>> newSignature) { | ||
Set<String> deletedClasses = new HashSet<>(oldSignature.keySet()); | ||
deletedClasses.removeAll(newSignature.keySet()); | ||
if (deletedClasses.size() > 0) { | ||
throw new IllegalStateException("Classes from a previous version not found: " + deletedClasses); | ||
} | ||
|
||
Map<String, Set<String>> deletedMembersMap = new HashMap<>(); | ||
for (Map.Entry<String, Set<String>> entry : oldSignature.entrySet()) { | ||
Set<String> deletedMembers = new HashSet<>(entry.getValue()); | ||
deletedMembers.removeAll(newSignature.get(entry.getKey())); | ||
if (deletedMembers.size() > 0) { | ||
deletedMembersMap.put(entry.getKey(), Set.copyOf(deletedMembers)); | ||
} | ||
} | ||
if (deletedMembersMap.size() > 0) { | ||
throw new IllegalStateException( | ||
"Classes from a previous version have been modified, violating backwards compatibility: " + deletedMembersMap | ||
); | ||
} | ||
} | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,6 @@ | ||
subprojects { | ||
apply plugin: 'elasticsearch.java' | ||
apply plugin: 'elasticsearch.bwc-test' | ||
|
||
group = 'org.elasticsearch.qa.stable-api' | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,4 @@ | ||
ext.stableApiSince = "8.7.0" | ||
|
||
apply plugin: 'elasticsearch.stable-api' | ||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,3 @@ | ||
ext.stableApiSince = "8.7.0" | ||
|
||
apply plugin: 'elasticsearch.stable-api' |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,3 @@ | ||
ext.stableApiSince = "8.7.0" | ||
|
||
apply plugin: 'elasticsearch.stable-api' |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters