diff --git a/Launcher/build.gradle b/Launcher/build.gradle index 0990074d..79da931f 100644 --- a/Launcher/build.gradle +++ b/Launcher/build.gradle @@ -1,6 +1,9 @@ -String mainClassName = "ru.gravit.launcher.LauncherEngine" +String realMainClassName = "ru.gravit.launcher.LauncherEngine" String mainAgentName = "ru.gravit.launcher.LauncherAgent" +String mainClassName = "ru.gravit.launcher.relauncher.VerRelauncher" +String errMessage = "Please, download Java 8 or higher." +String minVer = "52" repositories { maven { @@ -14,6 +17,10 @@ jar { from { configurations.runtime.collect { it.isDirectory() ? it : zipTree(it) } } manifest.attributes("Main-Class": mainClassName, + "MainRun-Class": realMainClassName, + "Graphic-Enabled": "true", + "ErrorMessage-String": errMessage, + "MinVesion-Integer": minVer, "Premain-Class": mainAgentName, "Can-Redefine-Classes": "true", "Can-Retransform-Classes": "true", @@ -22,6 +29,7 @@ dependencies { compile project(':LauncherAPI') + compile project(':javaVerRelauncher') compile 'org.javassist:javassist:3.23.1-GA' } diff --git a/javaVerRelauncher/build.gradle b/javaVerRelauncher/build.gradle new file mode 100644 index 00000000..ba761cda --- /dev/null +++ b/javaVerRelauncher/build.gradle @@ -0,0 +1,8 @@ +String mainClassName = "ru.gravit.launcher.relauncher.VerRelauncher" + +sourceCompatibility = '1.6' +targetCompatibility = '1.6' + +jar { + manifest.attributes("Main-Class": mainClassName) +} diff --git a/javaVerRelauncher/src/main/java/ru/gravit/launcher/relauncher/Helper.java b/javaVerRelauncher/src/main/java/ru/gravit/launcher/relauncher/Helper.java new file mode 100644 index 00000000..a69deac1 --- /dev/null +++ b/javaVerRelauncher/src/main/java/ru/gravit/launcher/relauncher/Helper.java @@ -0,0 +1,127 @@ +package ru.gravit.launcher.relauncher; + +import java.io.File; +import java.io.InputStream; +import java.lang.management.ManagementFactory; +import java.net.URL; +import java.util.Locale; +import java.util.jar.Manifest; + +public final class Helper { + public static enum OS { + LINUX("linux"), MACOSX("macosx"), MUSTDIE("mustdie"), OTHER("other"); + public static OS byName(final String name) { + if (name.startsWith("Windows")) + return MUSTDIE; + if (name.startsWith("Linux")) + return LINUX; + if (name.startsWith("Mac OS X")) + return MACOSX; + return OTHER; + } + + public final String name; + + private OS(final String name) { + this.name = name; + } + } + + private static final String DEFERR = "Invalid java version."; + + public static final ClassLoader LOADER = ClassLoader.getSystemClassLoader(); + public static final OS os = OS.byName(ManagementFactory.getOperatingSystemMXBean().getName()); + public static Manifest mf = null; + + public static final String getErrMessage() { + try { + return getErrMessage(getManifest()); + } catch (final Throwable t) { + return DEFERR; + } + } + + public static final String getErrMessage(final Manifest mf) { + String mess = DEFERR; + try { + mess = mf.getMainAttributes().getValue("ErrorMessage-String"); + } catch (final Throwable t) { + } + return mess; + } + + public static final String getMainClass() { + try { + return getMainClass(getManifest()); + } catch (final Throwable t) { + return null; + } + } + + public static final String getMainClass(final Manifest mf) { + String main = null; + try { + main = mf.getMainAttributes().getValue("MainRun-Class").trim(); + } catch (final Throwable t) { + } + return main; + } + + public static final Manifest getManifest() { + if (mf != null) return mf; + try { + InputStream in = VerRelauncher.class.getResourceAsStream("/META-INF/MANIFEST.MF"); + Manifest mf = new Manifest(in); + in.close(); + Helper.mf = mf; + return mf; + } catch (final Throwable t) { + return null; + } + } + + public static final int getMinVer() { + try { + return getMinVer(getManifest()); + } catch (final Throwable t) { + return JavaVersionInfo.JAVA_6; + } + } + + public static final int getMinVer(final Manifest mf) { + int ver = JavaVersionInfo.JAVA_6; + try { + ver = Integer.parseInt(mf.getMainAttributes().getValue("MinVesion-Integer").trim()); + } catch (final Throwable t) { + } + return ver; + } + + public static final OS getOs() { + return os; + } + + public static final boolean isGraphic() { + try { + return isGraphic(getManifest()); + } catch (final Throwable t) { + return false; + } + } + + public static final boolean isGraphic(final Manifest mf) { + boolean graph = false; + try { + graph = "TRUE".equalsIgnoreCase(mf.getMainAttributes().getValue("Graphic-Enabled").trim()); + } catch (final Throwable t) { + } + return graph; + } + + public static void verifySystemProperties(final Class mainClass, final boolean requireSystem) { + Locale.setDefault(Locale.US); + // Verify ClassLoader + if (requireSystem && !mainClass.getClassLoader().equals(LOADER)) + throw new SecurityException("ClassLoader should be system"); + } +} diff --git a/javaVerRelauncher/src/main/java/ru/gravit/launcher/relauncher/JavaVersionInfo.java b/javaVerRelauncher/src/main/java/ru/gravit/launcher/relauncher/JavaVersionInfo.java new file mode 100644 index 00000000..60d5d2a1 --- /dev/null +++ b/javaVerRelauncher/src/main/java/ru/gravit/launcher/relauncher/JavaVersionInfo.java @@ -0,0 +1,96 @@ +package ru.gravit.launcher.relauncher; + +public final class JavaVersionInfo { + /** + * The major version number of class files for JDK 1.1. + */ + public static final int JAVA_1 = 45; + + /** + * The major version number of class files for JDK 10. + */ + public static final int JAVA_10 = 54; + + /** + * The major version number of class files for JDK 11. + */ + public static final int JAVA_11 = 55; + + /** + * The major version number of class files for JDK 1.2. + */ + public static final int JAVA_2 = 46; + + /** + * The major version number of class files for JDK 1.3. + */ + public static final int JAVA_3 = 47; + + /** + * The major version number of class files for JDK 1.4. + */ + public static final int JAVA_4 = 48; + + /** + * The major version number of class files for JDK 1.5. + */ + public static final int JAVA_5 = 49; + + /** + * The major version number of class files for JDK 1.6. + */ + public static final int JAVA_6 = 50; + + /** + * The major version number of class files for JDK 1.7. + */ + public static final int JAVA_7 = 51; + + /** + * The major version number of class files for JDK 1.8. + */ + public static final int JAVA_8 = 52; + + /** + * The major version number of class files for JDK 1.9. + */ + public static final int JAVA_9 = 53; + + /** + * The major version number of class files created from scratch. The default + * value is 47 (JDK 1.3). It is 49 (JDK 1.5) if the JVM supports + * java.lang.StringBuilder. It is 50 (JDK 1.6) if the JVM supports + * java.util.zip.DeflaterInputStream. It is 51 (JDK 1.7) if the JVM + * supports java.lang.invoke.CallSite. It is 52 (JDK 1.8) if the + * JVM supports java.util.function.Function. It is 53 (JDK 1.9) if + * the JVM supports java.lang.reflect.Module. It is 54 (JDK 10) if + * the JVM supports java.util.List.copyOf(Collection). It is 55 + * (JDK 11) if the JVM supports java.util.Optional.isEmpty(). + */ + public static final int MAJOR_VERSION; + static { + int ver = JAVA_3; + try { + Class.forName("java.lang.StringBuilder"); + ver = JAVA_5; + Class.forName("java.util.zip.DeflaterInputStream"); + ver = JAVA_6; + Class.forName("java.lang.invoke.CallSite", false, ClassLoader.getSystemClassLoader()); + ver = JAVA_7; + Class.forName("java.util.function.Function"); + ver = JAVA_8; + Class.forName("java.lang.Module"); + ver = JAVA_9; + Class.forName("java.util.List").getMethod("copyOf", Class.forName("java.util.Collection")); + ver = JAVA_10; + Class.forName("java.util.Optional").getMethod("isEmpty"); + ver = JAVA_11; + } catch (final Throwable t) { + } + MAJOR_VERSION = ver; + } + + public static final int getVersion() { + return MAJOR_VERSION; + } +} diff --git a/javaVerRelauncher/src/main/java/ru/gravit/launcher/relauncher/VerRelauncher.java b/javaVerRelauncher/src/main/java/ru/gravit/launcher/relauncher/VerRelauncher.java new file mode 100644 index 00000000..9e41527a --- /dev/null +++ b/javaVerRelauncher/src/main/java/ru/gravit/launcher/relauncher/VerRelauncher.java @@ -0,0 +1,43 @@ +package ru.gravit.launcher.relauncher; + +import java.lang.reflect.Method; + +import javax.swing.JOptionPane; + +public final class VerRelauncher { + private static final void checkCompat() { + if (JavaVersionInfo.MAJOR_VERSION < Helper.getMinVer()) { + if (Helper.isGraphic()) + runGraph(Helper.getErrMessage()); + throw new AssertionError(Helper.getErrMessage()); + } + } + + public static void main(final String[] args) { + verifySystemProperties(); + try { + checkCompat(); + final Class main = Class.forName(Helper.getMainClass(), true, ClassLoader.getSystemClassLoader()); + Helper.verifySystemProperties(main, true); + final Method mainMethod = main.getMethod("main", String[].class); + mainMethod.setAccessible(true); + mainMethod.invoke(null, new Object[] { args }); + } catch (final Throwable t) { + if (t instanceof AssertionError) + throw (AssertionError) t; + if (t instanceof InternalError) + throw (InternalError) t; + throw new InternalError(t); + } + } + + private static void runGraph(final String errMessage) { + JOptionPane.showMessageDialog(null, errMessage); + } + + private static void verifySystemProperties() { + Helper.verifySystemProperties(Helper.class, true); + Helper.verifySystemProperties(VerRelauncher.class, true); + Helper.verifySystemProperties(JavaVersionInfo.class, true); + } +} diff --git a/settings.gradle b/settings.gradle index 5b0fea68..1e34055c 100644 --- a/settings.gradle +++ b/settings.gradle @@ -1,9 +1,10 @@ rootProject.name = 'Launcher' -include 'Launcher' include 'libLauncher' include 'LauncherAPI' +include 'javaVerRelauncher' include 'ServerWrapper' +include 'Launcher' include 'LaunchServer' include 'modules' file('modules').eachDir { sub ->