/* * Copyright (c) 2012, 2018, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. Oracle designates this * particular file as subject to the "Classpath" exception as provided * by Oracle in the LICENSE file that accompanied this code. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. */ package jdk.jpackager.internal.windows; import jdk.jpackager.internal.AbstractImageBundler; import jdk.jpackager.internal.BundlerParamInfo; import jdk.jpackager.internal.ConfigException; import jdk.jpackager.internal.IOUtils; import jdk.jpackager.internal.Log; import jdk.jpackager.internal.Platform; import jdk.jpackager.internal.StandardBundlerParam; import jdk.jpackager.internal.UnsupportedPlatformException; import jdk.jpackager.internal.builders.windows.WindowsAppImageBuilder; import jdk.jpackager.internal.resources.windows.WinResources; import jdk.jpackager.internal.JLinkBundlerHelper; import jdk.jpackager.internal.Arguments; import jdk.jpackager.internal.builders.AbstractAppImageBuilder; import java.io.ByteArrayOutputStream; import java.io.File; import java.io.IOException; import java.io.PrintStream; import java.text.MessageFormat; import java.util.Arrays; import java.util.Collection; import java.util.Map; import java.util.ResourceBundle; import static jdk.jpackager.internal.windows.WindowsBundlerParam.*; import static jdk.jpackager.internal.windows.WinMsiBundler.WIN_APP_IMAGE; public class WinAppBundler extends AbstractImageBundler { private static final ResourceBundle I18N = ResourceBundle.getBundle( "jdk.jpackager.internal.resources.windows.WinAppBundler"); public static final BundlerParamInfo ICON_ICO = new StandardBundlerParam<>( I18N.getString("param.icon-ico.name"), I18N.getString("param.icon-ico.description"), "icon.ico", File.class, params -> { File f = ICON.fetchFrom(params); if (f != null && !f.getName().toLowerCase().endsWith(".ico")) { Log.error(MessageFormat.format( I18N.getString("message.icon-not-ico"), f)); return null; } return f; }, (s, p) -> new File(s)); public WinAppBundler() { super(); baseResourceLoader = WinResources.class; } public final static String WIN_BUNDLER_PREFIX = BUNDLER_PREFIX + "windows/"; @Override public boolean validate(Map params) throws UnsupportedPlatformException, ConfigException { try { if (params == null) throw new ConfigException( I18N.getString("error.parameters-null"), I18N.getString("error.parameters-null.advice")); return doValidate(params); } catch (RuntimeException re) { if (re.getCause() instanceof ConfigException) { throw (ConfigException) re.getCause(); } else { throw new ConfigException(re); } } } // to be used by chained bundlers, e.g. by EXE bundler to avoid // skipping validation if p.type does not include "image" boolean doValidate(Map p) throws UnsupportedPlatformException, ConfigException { if (Platform.getPlatform() != Platform.WINDOWS) { throw new UnsupportedPlatformException(); } imageBundleValidation(p); if (StandardBundlerParam.getPredefinedAppImage(p) != null) { return true; } // Make sure that jpackager.exe exists. File tool = new File( System.getProperty("java.home") + "\\bin\\jpackager.exe"); if (!tool.exists()) { throw new ConfigException( I18N.getString("error.no-windows-resources"), I18N.getString("error.no-windows-resources.advice")); } // validate runtime bit-architectire testRuntimeBitArchitecture(p); return true; } private static void testRuntimeBitArchitecture( Map params) throws ConfigException { if ("true".equalsIgnoreCase(System.getProperty( "fxpackager.disableBitArchitectureMismatchCheck"))) { Log.debug(I18N.getString("message.disable-bit-architecture-check")); return; } if ((BIT_ARCH_64.fetchFrom(params) != BIT_ARCH_64_RUNTIME.fetchFrom(params))) { throw new ConfigException( I18N.getString("error.bit-architecture-mismatch"), I18N.getString("error.bit-architecture-mismatch.advice")); } } private static boolean usePredefineAppName(Map p) { return (PREDEFINED_APP_IMAGE.fetchFrom(p) != null); } private static String appName; synchronized static String getAppName( Map p) { // If we building from predefined app image, then we should use names // from image and not from CLI. if (usePredefineAppName(p)) { if (appName == null) { // Use WIN_APP_IMAGE here, since we already copy pre-defined // image to WIN_APP_IMAGE File appImageDir = new File( WIN_APP_IMAGE.fetchFrom(p).toString() + "\\app"); File [] files = appImageDir.listFiles( (File dir, String name) -> name.endsWith(".cfg")); if (files == null || files.length == 0) { throw new RuntimeException(MessageFormat.format( I18N.getString("error.cannot-find-launcher"), appImageDir)); } else { appName = files[0].getName(); int index = appName.indexOf("."); if (index != -1) { appName = appName.substring(0, index); } if (files.length > 1) { Log.error(MessageFormat.format(I18N.getString( "message.multiple-launchers"), appName)); } } return appName; } else { return appName; } } return APP_NAME.fetchFrom(p); } public static String getLauncherName(Map p) { return getAppName(p) + ".exe"; } public static String getLauncherCfgName(Map p) { return "app\\" + getAppName(p) +".cfg"; } public boolean bundle(Map p, File outputDirectory) { return doBundle(p, outputDirectory, false) != null; } File doBundle(Map p, File outputDirectory, boolean dependentTask) { if (Arguments.CREATE_JRE_INSTALLER.fetchFrom(p)) { return doJreBundle(p, outputDirectory, dependentTask); } else { return doAppBundle(p, outputDirectory, dependentTask); } } File doJreBundle(Map p, File outputDirectory, boolean dependentTask) { try { File rootDirectory = createRoot(p, outputDirectory, dependentTask, APP_NAME.fetchFrom(p), "windowsapp-image-builder"); AbstractAppImageBuilder appBuilder = new WindowsAppImageBuilder( APP_NAME.fetchFrom(p), outputDirectory.toPath()); File predefined = PREDEFINED_RUNTIME_IMAGE.fetchFrom(p); if (predefined == null ) { JLinkBundlerHelper.generateServerJre(p, appBuilder); } else { return predefined; } return rootDirectory; } catch (Exception ex) { Log.error("Exception: "+ex); Log.verbose(ex); return null; } } File doAppBundle(Map p, File outputDirectory, boolean dependentTask) { try { File rootDirectory = createRoot(p, outputDirectory, dependentTask, APP_NAME.fetchFrom(p), "windowsapp-image-builder"); AbstractAppImageBuilder appBuilder = new WindowsAppImageBuilder(p, outputDirectory.toPath()); if (PREDEFINED_RUNTIME_IMAGE.fetchFrom(p) == null ) { JLinkBundlerHelper.execute(p, appBuilder); } else { StandardBundlerParam.copyPredefinedRuntimeImage(p, appBuilder); } if (!dependentTask) { Log.verbose(MessageFormat.format( I18N.getString("message.result-dir"), outputDirectory.getAbsolutePath())); } return rootDirectory; } catch (Exception ex) { Log.error("Exception: "+ex); Log.verbose(ex); return null; } } private static final String RUNTIME_AUTO_DETECT = ".runtime.autodetect"; public static void extractFlagsFromRuntime( Map params) { if (params.containsKey(".runtime.autodetect")) return; params.put(RUNTIME_AUTO_DETECT, "attempted"); String commandline; File runtimePath = JLinkBundlerHelper.getJDKHome(params).toFile(); File launcherPath = new File(runtimePath, "bin\\java.exe"); ProcessBuilder pb = new ProcessBuilder(launcherPath.getAbsolutePath(), "-version"); try (ByteArrayOutputStream baos = new ByteArrayOutputStream()) { try (PrintStream pout = new PrintStream(baos)) { IOUtils.exec(pb, Log.isDebug(), true, pout); } commandline = baos.toString(); } catch (IOException e) { e.printStackTrace(); params.put(RUNTIME_AUTO_DETECT, "failed"); return; } AbstractImageBundler.extractFlagsFromVersion(params, commandline); params.put(RUNTIME_AUTO_DETECT, "succeeded"); } @Override public String getName() { return I18N.getString("bundler.name"); } @Override public String getDescription() { return I18N.getString("bundler.description"); } @Override public String getID() { return "windows.app"; } @Override public String getBundleType() { return "IMAGE"; } @Override public Collection> getBundleParameters() { return getAppBundleParameters(); } public static Collection> getAppBundleParameters() { return Arrays.asList( APP_NAME, APP_RESOURCES, ARGUMENTS, CLASSPATH, ICON_ICO, JVM_OPTIONS, JVM_PROPERTIES, MAIN_CLASS, MAIN_JAR, PREFERENCES_ID, VERSION, VERBOSE ); } @Override public File execute( Map params, File outputParentDir) { return doBundle(params, outputParentDir, false); } @Override public boolean supported() { return (Platform.getPlatform() == Platform.WINDOWS); } }