Skip to content

Commit

Permalink
Merge pull request ESQL-612 from elastic/main
Browse files Browse the repository at this point in the history
🤖 ESQL: Merge upstream
  • Loading branch information
elasticsearchmachine authored Jan 18, 2023
2 parents 89749bb + 394d4f0 commit 8be14da
Show file tree
Hide file tree
Showing 39 changed files with 2,247 additions and 191 deletions.
6 changes: 0 additions & 6 deletions .ci/packer_cache.sh
Original file line number Diff line number Diff line change
Expand Up @@ -44,9 +44,3 @@ fi
## therefore we run main _AFTER_ we run 6.8 which uses an earlier gradle version
export JAVA_HOME="${HOME}"/.java/${ES_BUILD_JAVA}
./gradlew --parallel clean -s resolveAllDependencies -Dorg.gradle.warning.mode=none -Drecurse.bwc=true

## Copy all dependencies into a "read-only" location to be used by nested Gradle builds
mkdir -p ${HOME}/gradle_ro_cache
rsync -r ${HOME}/.gradle/caches/modules-2 ${HOME}/gradle_ro_cache
rm ${HOME}/gradle_ro_cache/modules-2/gc.properties
rm ${HOME}/gradle_ro_cache/modules-2/*.lock
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)
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,7 @@
import java.util.ArrayList;
import java.util.List;
import java.util.Locale;
import java.util.Set;
import java.util.stream.Collectors;

import javax.inject.Inject;
Expand Down Expand Up @@ -120,6 +121,35 @@ private void configureBwcProject(Project project, BwcVersions.UnreleasedVersionI
buildBwcTaskProvider,
"assemble"
);

// for versions before 8.7.0, we do not need to set up stable API bwc
if (bwcVersion.get().before(Version.fromString("8.7.0"))) {
return;
}

for (Project stableApiProject : resolveStableProjects(project)) {

String relativeDir = project.getRootProject().relativePath(stableApiProject.getProjectDir());

DistributionProjectArtifact stableAnalysisPluginProjectArtifact = new DistributionProjectArtifact(
new File(
checkoutDir.get(),
relativeDir + "/build/distributions/" + stableApiProject.getName() + "-" + bwcVersion.get() + "-SNAPSHOT.jar"
),
null
);

createBuildBwcTask(
bwcSetupExtension,
project,
bwcVersion,
stableApiProject.getName(),
"libs/" + stableApiProject.getName(),
stableAnalysisPluginProjectArtifact,
buildBwcTaskProvider,
"assemble"
);
}
}

private void registerBwcDistributionArtifacts(Project bwcProject, DistributionProject distributionProject) {
Expand Down Expand Up @@ -209,7 +239,16 @@ private static List<DistributionProject> resolveArchiveProjects(File checkoutDir
}).collect(Collectors.toList());
}

private static String buildBwcTaskName(String projectName) {
private static List<Project> resolveStableProjects(Project project) {
Set<String> stableProjectNames = Set.of("elasticsearch-logging", "elasticsearch-plugin-api", "elasticsearch-plugin-analysis-api");
return project.findProject(":libs")
.getSubprojects()
.stream()
.filter(subproject -> stableProjectNames.contains(subproject.getName()))
.toList();
}

public static String buildBwcTaskName(String projectName) {
return "buildBwc"
+ stream(projectName.split("-")).map(i -> i.substring(0, 1).toUpperCase(Locale.ROOT) + i.substring(1))
.collect(Collectors.joining());
Expand Down
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
);
}
}
}
}
3 changes: 3 additions & 0 deletions docs/reference/docs/reindex.asciidoc
Original file line number Diff line number Diff line change
Expand Up @@ -575,6 +575,9 @@ Valid values: `index`, `create`. Defaults to `index`.
IMPORTANT: To reindex to a data stream destination, this argument must be
`create`.

`pipeline`:::
(Optional, string) the name of the <<reindex-with-an-ingest-pipeline,pipeline>> to use.

`script`::
`source`:::
(Optional, string) The script to run to update the document source or metadata when reindexing.
Expand Down
2 changes: 1 addition & 1 deletion docs/reference/modules/cluster/shards_allocation.asciidoc
Original file line number Diff line number Diff line change
Expand Up @@ -35,7 +35,7 @@ one of the active allocation ids in the cluster state.
`cluster.routing.allocation.node_concurrent_recoveries`::
(<<dynamic-cluster-setting,Dynamic>>)
A shortcut to set both `cluster.routing.allocation.node_concurrent_incoming_recoveries` and
`cluster.routing.allocation.node_concurrent_outgoing_recoveries`.
`cluster.routing.allocation.node_concurrent_outgoing_recoveries`. Defaults to 2.


`cluster.routing.allocation.node_initial_primaries_recoveries`::
Expand Down
1 change: 0 additions & 1 deletion libs/logging/build.gradle
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,6 @@
apply plugin: 'elasticsearch.publish'
apply plugin: 'elasticsearch.build'


tasks.named("loggerUsageCheck").configure {enabled = false }

dependencies {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -45,8 +45,8 @@ public static List<ClassReader> ofDirWithJars(String path) {
return Collections.emptyList();
}
Path dir = Paths.get(path);
try {
return ofPaths(Files.list(dir));
try (var stream = Files.list(dir)) {
return ofPaths(stream);
} catch (IOException e) {
throw new UncheckedIOException(e);
}
Expand Down
Loading

0 comments on commit 8be14da

Please sign in to comment.