fail plugins on version mismatch
This commit is contained in:
parent
2628ecfacf
commit
5d48f93557
|
@ -19,6 +19,7 @@
|
||||||
|
|
||||||
package org.elasticsearch.bootstrap;
|
package org.elasticsearch.bootstrap;
|
||||||
|
|
||||||
|
import org.elasticsearch.Version;
|
||||||
import org.elasticsearch.common.SuppressForbidden;
|
import org.elasticsearch.common.SuppressForbidden;
|
||||||
import org.elasticsearch.common.io.PathUtils;
|
import org.elasticsearch.common.io.PathUtils;
|
||||||
import org.elasticsearch.common.logging.ESLogger;
|
import org.elasticsearch.common.logging.ESLogger;
|
||||||
|
@ -27,7 +28,6 @@ import org.elasticsearch.common.logging.Loggers;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.net.URL;
|
import java.net.URL;
|
||||||
import java.net.URLClassLoader;
|
import java.net.URLClassLoader;
|
||||||
import java.net.URLDecoder;
|
|
||||||
import java.nio.file.FileVisitResult;
|
import java.nio.file.FileVisitResult;
|
||||||
import java.nio.file.Files;
|
import java.nio.file.Files;
|
||||||
import java.nio.file.Path;
|
import java.nio.file.Path;
|
||||||
|
@ -99,23 +99,7 @@ public class JarHell {
|
||||||
try (JarFile file = new JarFile(path.toString())) {
|
try (JarFile file = new JarFile(path.toString())) {
|
||||||
Manifest manifest = file.getManifest();
|
Manifest manifest = file.getManifest();
|
||||||
if (manifest != null) {
|
if (manifest != null) {
|
||||||
// inspect Manifest: give a nice error if jar requires a newer java version
|
checkManifest(manifest, path);
|
||||||
String systemVersion = System.getProperty("java.specification.version");
|
|
||||||
String targetVersion = manifest.getMainAttributes().getValue("X-Compile-Target-JDK");
|
|
||||||
if (targetVersion != null) {
|
|
||||||
float current = Float.POSITIVE_INFINITY;
|
|
||||||
float target = Float.NEGATIVE_INFINITY;
|
|
||||||
try {
|
|
||||||
current = Float.parseFloat(systemVersion);
|
|
||||||
target = Float.parseFloat(targetVersion);
|
|
||||||
} catch (NumberFormatException e) {
|
|
||||||
// some spec changed, time for a more complex parser
|
|
||||||
}
|
|
||||||
if (current < target) {
|
|
||||||
throw new IllegalStateException(path + " requires Java " + targetVersion
|
|
||||||
+ ", your system: " + systemVersion);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
// inspect entries
|
// inspect entries
|
||||||
Enumeration<JarEntry> elements = file.entries();
|
Enumeration<JarEntry> elements = file.entries();
|
||||||
|
@ -149,6 +133,35 @@ public class JarHell {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** inspect manifest for sure incompatibilities */
|
||||||
|
static void checkManifest(Manifest manifest, Path jar) {
|
||||||
|
// give a nice error if jar requires a newer java version
|
||||||
|
String systemVersion = System.getProperty("java.specification.version");
|
||||||
|
String targetVersion = manifest.getMainAttributes().getValue("X-Compile-Target-JDK");
|
||||||
|
if (targetVersion != null) {
|
||||||
|
float current = Float.POSITIVE_INFINITY;
|
||||||
|
float target = Float.NEGATIVE_INFINITY;
|
||||||
|
try {
|
||||||
|
current = Float.parseFloat(systemVersion);
|
||||||
|
target = Float.parseFloat(targetVersion);
|
||||||
|
} catch (NumberFormatException e) {
|
||||||
|
// some spec changed, time for a more complex parser
|
||||||
|
}
|
||||||
|
if (current < target) {
|
||||||
|
throw new IllegalStateException(jar + " requires Java " + targetVersion
|
||||||
|
+ ", your system: " + systemVersion);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// give a nice error if jar is compiled against different es version
|
||||||
|
String systemESVersion = Version.CURRENT.toString();
|
||||||
|
String targetESVersion = manifest.getMainAttributes().getValue("X-Compile-Elasticsearch-Version");
|
||||||
|
if (targetESVersion != null && targetESVersion.equals(systemESVersion) == false) {
|
||||||
|
throw new IllegalStateException(jar + " requires Elasticsearch " + targetESVersion
|
||||||
|
+ ", your system: " + systemESVersion);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static void checkClass(Map<String,Path> clazzes, String clazz, Path jarpath) {
|
static void checkClass(Map<String,Path> clazzes, String clazz, Path jarpath) {
|
||||||
Path previous = clazzes.put(clazz, jarpath);
|
Path previous = clazzes.put(clazz, jarpath);
|
||||||
if (previous != null) {
|
if (previous != null) {
|
||||||
|
|
|
@ -19,10 +19,9 @@
|
||||||
|
|
||||||
package org.elasticsearch.bootstrap;
|
package org.elasticsearch.bootstrap;
|
||||||
|
|
||||||
|
import org.elasticsearch.Version;
|
||||||
import org.elasticsearch.test.ElasticsearchTestCase;
|
import org.elasticsearch.test.ElasticsearchTestCase;
|
||||||
|
|
||||||
import java.io.ByteArrayOutputStream;
|
|
||||||
import java.io.FileOutputStream;
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.net.URL;
|
import java.net.URL;
|
||||||
import java.nio.file.Files;
|
import java.nio.file.Files;
|
||||||
|
@ -54,7 +53,7 @@ public class JarHellTests extends ElasticsearchTestCase {
|
||||||
URL makeFile(Path dir, String name) throws IOException {
|
URL makeFile(Path dir, String name) throws IOException {
|
||||||
Path filepath = dir.resolve(name);
|
Path filepath = dir.resolve(name);
|
||||||
Files.newOutputStream(filepath, StandardOpenOption.CREATE).close();
|
Files.newOutputStream(filepath, StandardOpenOption.CREATE).close();
|
||||||
return filepath.toUri().toURL();
|
return dir.toUri().toURL();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void testDifferentJars() throws Exception {
|
public void testDifferentJars() throws Exception {
|
||||||
|
@ -62,6 +61,7 @@ public class JarHellTests extends ElasticsearchTestCase {
|
||||||
URL[] jars = {makeJar(dir, "foo.jar", null, "DuplicateClass.class"), makeJar(dir, "bar.jar", null, "DuplicateClass.class")};
|
URL[] jars = {makeJar(dir, "foo.jar", null, "DuplicateClass.class"), makeJar(dir, "bar.jar", null, "DuplicateClass.class")};
|
||||||
try {
|
try {
|
||||||
JarHell.checkJarHell(jars);
|
JarHell.checkJarHell(jars);
|
||||||
|
fail("did not get expected exception");
|
||||||
} catch (IllegalStateException e) {
|
} catch (IllegalStateException e) {
|
||||||
assertTrue(e.getMessage().contains("jar hell!"));
|
assertTrue(e.getMessage().contains("jar hell!"));
|
||||||
assertTrue(e.getMessage().contains("DuplicateClass"));
|
assertTrue(e.getMessage().contains("DuplicateClass"));
|
||||||
|
@ -95,6 +95,7 @@ public class JarHellTests extends ElasticsearchTestCase {
|
||||||
URL[] dirs = {makeFile(dir1, "DuplicateClass.class"), makeFile(dir2, "DuplicateClass.class")};
|
URL[] dirs = {makeFile(dir1, "DuplicateClass.class"), makeFile(dir2, "DuplicateClass.class")};
|
||||||
try {
|
try {
|
||||||
JarHell.checkJarHell(dirs);
|
JarHell.checkJarHell(dirs);
|
||||||
|
fail("did not get expected exception");
|
||||||
} catch (IllegalStateException e) {
|
} catch (IllegalStateException e) {
|
||||||
assertTrue(e.getMessage().contains("jar hell!"));
|
assertTrue(e.getMessage().contains("jar hell!"));
|
||||||
assertTrue(e.getMessage().contains("DuplicateClass"));
|
assertTrue(e.getMessage().contains("DuplicateClass"));
|
||||||
|
@ -109,6 +110,7 @@ public class JarHellTests extends ElasticsearchTestCase {
|
||||||
URL[] dirs = {makeJar(dir1, "foo.jar", null, "DuplicateClass.class"), makeFile(dir2, "DuplicateClass.class")};
|
URL[] dirs = {makeJar(dir1, "foo.jar", null, "DuplicateClass.class"), makeFile(dir2, "DuplicateClass.class")};
|
||||||
try {
|
try {
|
||||||
JarHell.checkJarHell(dirs);
|
JarHell.checkJarHell(dirs);
|
||||||
|
fail("did not get expected exception");
|
||||||
} catch (IllegalStateException e) {
|
} catch (IllegalStateException e) {
|
||||||
assertTrue(e.getMessage().contains("jar hell!"));
|
assertTrue(e.getMessage().contains("jar hell!"));
|
||||||
assertTrue(e.getMessage().contains("DuplicateClass"));
|
assertTrue(e.getMessage().contains("DuplicateClass"));
|
||||||
|
@ -135,6 +137,7 @@ public class JarHellTests extends ElasticsearchTestCase {
|
||||||
URL[] jars = {JarHellTests.class.getResource("duplicate-classes.jar")};
|
URL[] jars = {JarHellTests.class.getResource("duplicate-classes.jar")};
|
||||||
try {
|
try {
|
||||||
JarHell.checkJarHell(jars);
|
JarHell.checkJarHell(jars);
|
||||||
|
fail("did not get expected exception");
|
||||||
} catch (IllegalStateException e) {
|
} catch (IllegalStateException e) {
|
||||||
assertTrue(e.getMessage().contains("jar hell!"));
|
assertTrue(e.getMessage().contains("jar hell!"));
|
||||||
assertTrue(e.getMessage().contains("DuplicateClass"));
|
assertTrue(e.getMessage().contains("DuplicateClass"));
|
||||||
|
@ -154,12 +157,15 @@ public class JarHellTests extends ElasticsearchTestCase {
|
||||||
System.setProperty("java.specification.version", "1.7");
|
System.setProperty("java.specification.version", "1.7");
|
||||||
|
|
||||||
Manifest manifest = new Manifest();
|
Manifest manifest = new Manifest();
|
||||||
manifest.getMainAttributes().put(new Attributes.Name("X-Compile-Target-JDK"), "1.8");
|
Attributes attributes = manifest.getMainAttributes();
|
||||||
|
attributes.put(Attributes.Name.MANIFEST_VERSION, "1.0.0");
|
||||||
|
attributes.put(new Attributes.Name("X-Compile-Target-JDK"), "1.8");
|
||||||
URL[] jars = {makeJar(dir, "foo.jar", manifest, "Foo.class")};
|
URL[] jars = {makeJar(dir, "foo.jar", manifest, "Foo.class")};
|
||||||
try {
|
try {
|
||||||
JarHell.checkJarHell(jars);
|
JarHell.checkJarHell(jars);
|
||||||
|
fail("did not get expected exception");
|
||||||
} catch (IllegalStateException e) {
|
} catch (IllegalStateException e) {
|
||||||
assertTrue(e.getMessage().contains("requires java 1.8"));
|
assertTrue(e.getMessage().contains("requires Java 1.8"));
|
||||||
assertTrue(e.getMessage().contains("your system: 1.7"));
|
assertTrue(e.getMessage().contains("your system: 1.7"));
|
||||||
} finally {
|
} finally {
|
||||||
System.setProperty("java.specification.version", previousJavaVersion);
|
System.setProperty("java.specification.version", previousJavaVersion);
|
||||||
|
@ -172,7 +178,9 @@ public class JarHellTests extends ElasticsearchTestCase {
|
||||||
System.setProperty("java.specification.version", "1.7");
|
System.setProperty("java.specification.version", "1.7");
|
||||||
|
|
||||||
Manifest manifest = new Manifest();
|
Manifest manifest = new Manifest();
|
||||||
manifest.getMainAttributes().put(new Attributes.Name("X-Compile-Target-JDK"), "1.7");
|
Attributes attributes = manifest.getMainAttributes();
|
||||||
|
attributes.put(Attributes.Name.MANIFEST_VERSION, "1.0.0");
|
||||||
|
attributes.put(new Attributes.Name("X-Compile-Target-JDK"), "1.7");
|
||||||
URL[] jars = {makeJar(dir, "foo.jar", manifest, "Foo.class")};
|
URL[] jars = {makeJar(dir, "foo.jar", manifest, "Foo.class")};
|
||||||
try {
|
try {
|
||||||
JarHell.checkJarHell(jars);
|
JarHell.checkJarHell(jars);
|
||||||
|
@ -187,7 +195,9 @@ public class JarHellTests extends ElasticsearchTestCase {
|
||||||
System.setProperty("java.specification.version", "bogus");
|
System.setProperty("java.specification.version", "bogus");
|
||||||
|
|
||||||
Manifest manifest = new Manifest();
|
Manifest manifest = new Manifest();
|
||||||
manifest.getMainAttributes().put(new Attributes.Name("X-Compile-Target-JDK"), "1.7");
|
Attributes attributes = manifest.getMainAttributes();
|
||||||
|
attributes.put(Attributes.Name.MANIFEST_VERSION, "1.0.0");
|
||||||
|
attributes.put(new Attributes.Name("X-Compile-Target-JDK"), "1.7");
|
||||||
URL[] jars = {makeJar(dir, "foo.jar", manifest, "Foo.class")};
|
URL[] jars = {makeJar(dir, "foo.jar", manifest, "Foo.class")};
|
||||||
try {
|
try {
|
||||||
JarHell.checkJarHell(jars);
|
JarHell.checkJarHell(jars);
|
||||||
|
@ -199,8 +209,37 @@ public class JarHellTests extends ElasticsearchTestCase {
|
||||||
public void testBadJDKVersionInJar() throws Exception {
|
public void testBadJDKVersionInJar() throws Exception {
|
||||||
Path dir = createTempDir();
|
Path dir = createTempDir();
|
||||||
Manifest manifest = new Manifest();
|
Manifest manifest = new Manifest();
|
||||||
manifest.getMainAttributes().put(new Attributes.Name("X-Compile-Target-JDK"), "bogus");
|
Attributes attributes = manifest.getMainAttributes();
|
||||||
|
attributes.put(Attributes.Name.MANIFEST_VERSION, "1.0.0");
|
||||||
|
attributes.put(new Attributes.Name("X-Compile-Target-JDK"), "bogus");
|
||||||
URL[] jars = {makeJar(dir, "foo.jar", manifest, "Foo.class")};
|
URL[] jars = {makeJar(dir, "foo.jar", manifest, "Foo.class")};
|
||||||
JarHell.checkJarHell(jars);
|
JarHell.checkJarHell(jars);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** make sure if a plugin is compiled against the same ES version, it works */
|
||||||
|
public void testGoodESVersionInJar() throws Exception {
|
||||||
|
Path dir = createTempDir();
|
||||||
|
Manifest manifest = new Manifest();
|
||||||
|
Attributes attributes = manifest.getMainAttributes();
|
||||||
|
attributes.put(Attributes.Name.MANIFEST_VERSION, "1.0.0");
|
||||||
|
attributes.put(new Attributes.Name("X-Compile-Elasticsearch-Version"), Version.CURRENT.toString());
|
||||||
|
URL[] jars = {makeJar(dir, "foo.jar", manifest, "Foo.class")};
|
||||||
|
JarHell.checkJarHell(jars);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** make sure if a plugin is compiled against a different ES version, it fails */
|
||||||
|
public void testBadESVersionInJar() throws Exception {
|
||||||
|
Path dir = createTempDir();
|
||||||
|
Manifest manifest = new Manifest();
|
||||||
|
Attributes attributes = manifest.getMainAttributes();
|
||||||
|
attributes.put(Attributes.Name.MANIFEST_VERSION, "1.0.0");
|
||||||
|
attributes.put(new Attributes.Name("X-Compile-Elasticsearch-Version"), "1.0-bogus");
|
||||||
|
URL[] jars = {makeJar(dir, "foo.jar", manifest, "Foo.class")};
|
||||||
|
try {
|
||||||
|
JarHell.checkJarHell(jars);
|
||||||
|
fail("did not get expected exception");
|
||||||
|
} catch (IllegalStateException e) {
|
||||||
|
assertTrue(e.getMessage().contains("requires Elasticsearch 1.0-bogus"));
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue