fail plugins on version mismatch

This commit is contained in:
Robert Muir 2015-07-13 21:01:39 -04:00
parent 2628ecfacf
commit 5d48f93557
2 changed files with 78 additions and 26 deletions

View File

@ -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) {

View File

@ -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"));
}
}
} }