diff --git a/src/jdk.incubator.jpackage/linux/classes/jdk/incubator/jpackage/internal/LinuxDebBundler.java b/src/jdk.incubator.jpackage/linux/classes/jdk/incubator/jpackage/internal/LinuxDebBundler.java index 11d67a2477b..e137ee6ff33 100644 --- a/src/jdk.incubator.jpackage/linux/classes/jdk/incubator/jpackage/internal/LinuxDebBundler.java +++ b/src/jdk.incubator.jpackage/linux/classes/jdk/incubator/jpackage/internal/LinuxDebBundler.java @@ -432,10 +432,18 @@ public class LinuxDebBundler extends LinuxPackageBundler { } private File getConfig_CopyrightFile(Map params) { - PlatformPackage thePackage = createMetaPackage(params); - return thePackage.sourceRoot().resolve(Path.of(".", - LINUX_INSTALL_DIR.fetchFrom(params), PACKAGE_NAME.fetchFrom( - params), "share/doc/copyright")).toFile(); + final String installDir = LINUX_INSTALL_DIR.fetchFrom(params); + final String packageName = PACKAGE_NAME.fetchFrom(params); + + final Path installPath; + if (isInstallDirInUsrTree(installDir) || installDir.startsWith("/usr/")) { + installPath = Path.of("/usr/share/doc/", packageName, "copyright"); + } else { + installPath = Path.of(installDir, packageName, "share/doc/copyright"); + } + + return createMetaPackage(params).sourceRoot().resolve( + Path.of("/").relativize(installPath)).toFile(); } private File buildDeb(Map params, diff --git a/src/jdk.incubator.jpackage/linux/classes/jdk/incubator/jpackage/internal/LinuxPackageBundler.java b/src/jdk.incubator.jpackage/linux/classes/jdk/incubator/jpackage/internal/LinuxPackageBundler.java index a4688a64a4c..b97ea19cdb5 100644 --- a/src/jdk.incubator.jpackage/linux/classes/jdk/incubator/jpackage/internal/LinuxPackageBundler.java +++ b/src/jdk.incubator.jpackage/linux/classes/jdk/incubator/jpackage/internal/LinuxPackageBundler.java @@ -31,6 +31,7 @@ import java.text.MessageFormat; import java.util.*; import java.util.function.Function; import java.util.function.Predicate; +import java.util.function.Supplier; import java.util.stream.Collectors; import java.util.stream.Stream; import static jdk.incubator.jpackage.internal.DesktopIntegration.*; @@ -114,8 +115,8 @@ abstract class LinuxPackageBundler extends AbstractBundler { initAppImageLayout.apply(appImage).copy( thePackage.sourceApplicationLayout()); } else { - appImage = appImageBundler.execute(params, - thePackage.sourceRoot().toFile()); + final Path srcAppImageRoot = thePackage.sourceRoot().resolve("src"); + appImage = appImageBundler.execute(params, srcAppImageRoot.toFile()); ApplicationLayout srcAppLayout = initAppImageLayout.apply( appImage); if (appImage.equals(PREDEFINED_RUNTIME_IMAGE.fetchFrom(params))) { @@ -126,11 +127,7 @@ abstract class LinuxPackageBundler extends AbstractBundler { // Application image is a newly created directory tree. // Move it. srcAppLayout.move(thePackage.sourceApplicationLayout()); - if (appImage.exists()) { - // Empty app image directory might remain after all application - // directories have been moved. - appImage.delete(); - } + IOUtils.deleteRecursive(srcAppImageRoot.toFile()); } } @@ -240,6 +237,17 @@ abstract class LinuxPackageBundler extends AbstractBundler { final protected PlatformPackage createMetaPackage( Map params) { + + Supplier packageLayout = () -> { + String installDir = LINUX_INSTALL_DIR.fetchFrom(params); + if (isInstallDirInUsrTree(installDir)) { + return ApplicationLayout.linuxUsrTreePackageImage( + Path.of("/").relativize(Path.of(installDir)), + packageName.fetchFrom(params)); + } + return appImageLayout(params); + }; + return new PlatformPackage() { @Override public String name() { @@ -253,19 +261,23 @@ abstract class LinuxPackageBundler extends AbstractBundler { @Override public ApplicationLayout sourceApplicationLayout() { - return appImageLayout(params).resolveAt( + return packageLayout.get().resolveAt( applicationInstallDir(sourceRoot())); } @Override public ApplicationLayout installedApplicationLayout() { - return appImageLayout(params).resolveAt( + return packageLayout.get().resolveAt( applicationInstallDir(Path.of("/"))); } private Path applicationInstallDir(Path root) { - Path installDir = Path.of(LINUX_INSTALL_DIR.fetchFrom(params), - name()); + String installRoot = LINUX_INSTALL_DIR.fetchFrom(params); + if (isInstallDirInUsrTree(installRoot)) { + return root; + } + + Path installDir = Path.of(installRoot, name()); if (installDir.isAbsolute()) { installDir = Path.of("." + installDir.toString()).normalize(); } @@ -284,10 +296,6 @@ abstract class LinuxPackageBundler extends AbstractBundler { private static void validateInstallDir(String installDir) throws ConfigException { - if (installDir.startsWith("/usr/") || installDir.equals("/usr")) { - throw new ConfigException(MessageFormat.format(I18N.getString( - "error.unsupported-install-dir"), installDir), null); - } if (installDir.isEmpty()) { throw new ConfigException(MessageFormat.format(I18N.getString( @@ -312,6 +320,10 @@ abstract class LinuxPackageBundler extends AbstractBundler { } } + protected static boolean isInstallDirInUsrTree(String installDir) { + return Set.of("/usr/local", "/usr").contains(installDir); + } + private final BundlerParamInfo packageName; private final Bundler appImageBundler; private boolean withFindNeededPackages; diff --git a/src/jdk.incubator.jpackage/linux/classes/jdk/incubator/jpackage/internal/LinuxRpmBundler.java b/src/jdk.incubator.jpackage/linux/classes/jdk/incubator/jpackage/internal/LinuxRpmBundler.java index 816ccc7410a..d0ffdb3d030 100644 --- a/src/jdk.incubator.jpackage/linux/classes/jdk/incubator/jpackage/internal/LinuxRpmBundler.java +++ b/src/jdk.incubator.jpackage/linux/classes/jdk/incubator/jpackage/internal/LinuxRpmBundler.java @@ -159,8 +159,15 @@ public class LinuxRpmBundler extends LinuxPackageBundler { Map params) throws IOException { Map data = new HashMap<>(); - data.put("APPLICATION_DIRECTORY", Path.of(LINUX_INSTALL_DIR.fetchFrom( - params), PACKAGE_NAME.fetchFrom(params)).toString()); + final Path prefix = Path.of(LINUX_INSTALL_DIR.fetchFrom(params)); + + Path appDirectory = prefix; + if (!isInstallDirInUsrTree(prefix.toString())) { + appDirectory = appDirectory.resolve(PACKAGE_NAME.fetchFrom(params)); + } + + data.put("APPLICATION_PREFIX", prefix.toString()); + data.put("APPLICATION_DIRECTORY", appDirectory.toString()); data.put("APPLICATION_SUMMARY", APP_NAME.fetchFrom(params)); data.put("APPLICATION_LICENSE_TYPE", LICENSE_TYPE.fetchFrom(params)); diff --git a/src/jdk.incubator.jpackage/linux/classes/jdk/incubator/jpackage/internal/resources/LinuxResources.properties b/src/jdk.incubator.jpackage/linux/classes/jdk/incubator/jpackage/internal/resources/LinuxResources.properties index f054cf12346..9ad1b2d6a4b 100644 --- a/src/jdk.incubator.jpackage/linux/classes/jdk/incubator/jpackage/internal/resources/LinuxResources.properties +++ b/src/jdk.incubator.jpackage/linux/classes/jdk/incubator/jpackage/internal/resources/LinuxResources.properties @@ -44,7 +44,6 @@ error.tool-not-found.advice=Please install required packages error.tool-old-version.advice=Please install required packages error.invalid-install-dir=Invalid installation directory "{0}" -error.unsupported-install-dir=Installing to system directory "{0}" is currently unsupported error.invalid-value-for-package-name=Invalid value "{0}" for the bundle name. error.invalid-value-for-package-name.advice=Set the "linux-bundle-name" option to a valid Debian package name. Note that the package names must consist only of lower case letters (a-z), digits (0-9), plus (+) and minus (-) signs, and periods (.). They must be at least two characters long and must start with an alphanumeric character. diff --git a/src/jdk.incubator.jpackage/linux/classes/jdk/incubator/jpackage/internal/resources/LinuxResources_ja.properties b/src/jdk.incubator.jpackage/linux/classes/jdk/incubator/jpackage/internal/resources/LinuxResources_ja.properties index 005c42ab0ce..5c751db4468 100644 --- a/src/jdk.incubator.jpackage/linux/classes/jdk/incubator/jpackage/internal/resources/LinuxResources_ja.properties +++ b/src/jdk.incubator.jpackage/linux/classes/jdk/incubator/jpackage/internal/resources/LinuxResources_ja.properties @@ -44,7 +44,6 @@ error.tool-not-found.advice=\u5FC5\u8981\u306A\u30D1\u30C3\u30B1\u30FC\u30B8\u30 error.tool-old-version.advice=\u5FC5\u8981\u306A\u30D1\u30C3\u30B1\u30FC\u30B8\u3092\u30A4\u30F3\u30B9\u30C8\u30FC\u30EB\u3057\u3066\u304F\u3060\u3055\u3044 error.invalid-install-dir=\u7121\u52B9\u306A\u30A4\u30F3\u30B9\u30C8\u30FC\u30EB\u30FB\u30C7\u30A3\u30EC\u30AF\u30C8\u30EA"{0}" -error.unsupported-install-dir=\u30B7\u30B9\u30C6\u30E0\u30FB\u30C7\u30A3\u30EC\u30AF\u30C8\u30EA"{0}"\u3078\u306E\u30A4\u30F3\u30B9\u30C8\u30FC\u30EB\u306F\u73FE\u5728\u30B5\u30DD\u30FC\u30C8\u3055\u308C\u3066\u3044\u307E\u305B\u3093 error.invalid-value-for-package-name=\u30D0\u30F3\u30C9\u30EB\u540D\u306E\u5024"{0}"\u304C\u7121\u52B9\u3067\u3059\u3002 error.invalid-value-for-package-name.advice="linux-bundle-name"\u30AA\u30D7\u30B7\u30E7\u30F3\u3092\u6709\u52B9\u306ADebian\u30D1\u30C3\u30B1\u30FC\u30B8\u540D\u306B\u8A2D\u5B9A\u3057\u3066\u304F\u3060\u3055\u3044\u3002\u30D1\u30C3\u30B1\u30FC\u30B8\u540D\u306B\u306F\u3001\u5C0F\u6587\u5B57(a-z)\u3001\u6570\u5B57(0-9)\u3001\u30D7\u30E9\u30B9(+)\u3068\u30DE\u30A4\u30CA\u30B9(-)\u306E\u8A18\u53F7\u304A\u3088\u3073\u30D4\u30EA\u30AA\u30C9(.)\u306E\u307F\u3092\u542B\u3081\u308B\u3088\u3046\u306B\u3057\u3066\u304F\u3060\u3055\u3044\u3002\u9577\u3055\u306F2\u6587\u5B57\u4EE5\u4E0A\u3068\u3057\u3001\u82F1\u6570\u5B57\u3067\u59CB\u3081\u308B\u5FC5\u8981\u304C\u3042\u308A\u307E\u3059\u3002 diff --git a/src/jdk.incubator.jpackage/linux/classes/jdk/incubator/jpackage/internal/resources/LinuxResources_zh_CN.properties b/src/jdk.incubator.jpackage/linux/classes/jdk/incubator/jpackage/internal/resources/LinuxResources_zh_CN.properties index 1e8c83e1768..c636d584a15 100644 --- a/src/jdk.incubator.jpackage/linux/classes/jdk/incubator/jpackage/internal/resources/LinuxResources_zh_CN.properties +++ b/src/jdk.incubator.jpackage/linux/classes/jdk/incubator/jpackage/internal/resources/LinuxResources_zh_CN.properties @@ -44,7 +44,6 @@ error.tool-not-found.advice=\u8BF7\u5B89\u88C5\u6240\u9700\u7684\u7A0B\u5E8F\u53 error.tool-old-version.advice=\u8BF7\u5B89\u88C5\u6240\u9700\u7684\u7A0B\u5E8F\u5305 error.invalid-install-dir=\u5B89\u88C5\u76EE\u5F55 "{0}" \u65E0\u6548 -error.unsupported-install-dir=\u5F53\u524D\u4E0D\u652F\u6301\u5B89\u88C5\u5230\u7CFB\u7EDF\u76EE\u5F55 "{0}" error.invalid-value-for-package-name=\u5305\u540D\u7684\u503C "{0}" \u65E0\u6548\u3002 error.invalid-value-for-package-name.advice=\u5C06 "linux-bundle-name" \u9009\u9879\u8BBE\u7F6E\u4E3A\u6709\u6548\u7684 Debian \u7A0B\u5E8F\u5305\u540D\u79F0\u3002\u8BF7\u6CE8\u610F\uFF0C\u7A0B\u5E8F\u5305\u540D\u79F0\u53EA\u80FD\u5305\u542B\u5C0F\u5199\u5B57\u6BCD (a-z)\u3001\u6570\u5B57 (0-9)\u3001\u52A0\u53F7 (+) \u548C\u51CF\u53F7 (-) \u4EE5\u53CA\u53E5\u70B9 (.)\u3002\u540D\u79F0\u957F\u5EA6\u5FC5\u987B\u81F3\u5C11\u4E3A\u4E24\u4E2A\u5B57\u7B26\u5E76\u4E14\u5FC5\u987B\u4EE5\u5B57\u6BCD\u6570\u5B57\u5B57\u7B26\u5F00\u5934\u3002 diff --git a/src/jdk.incubator.jpackage/linux/classes/jdk/incubator/jpackage/internal/resources/template.spec b/src/jdk.incubator.jpackage/linux/classes/jdk/incubator/jpackage/internal/resources/template.spec index a2e1ad8aa12..a1885ff949a 100644 --- a/src/jdk.incubator.jpackage/linux/classes/jdk/incubator/jpackage/internal/resources/template.spec +++ b/src/jdk.incubator.jpackage/linux/classes/jdk/incubator/jpackage/internal/resources/template.spec @@ -4,8 +4,13 @@ Version: APPLICATION_VERSION Release: APPLICATION_RELEASE License: APPLICATION_LICENSE_TYPE Vendor: APPLICATION_VENDOR -Prefix: %{dirname:APPLICATION_DIRECTORY} + +%if "xAPPLICATION_PREFIX" != x +Prefix: APPLICATION_PREFIX +%endif + Provides: APPLICATION_PACKAGE + %if "xAPPLICATION_GROUP" != x Group: APPLICATION_GROUP %endif @@ -21,6 +26,12 @@ Requires: PACKAGE_DEFAULT_DEPENDENCIES PACKAGE_CUSTOM_DEPENDENCIES #build time will substantially increase and it may require unpack200/system java to install %define __jar_repack %{nil} +%define package_filelist %{_tmppath}/%{name}.files +%define app_filelist %{_tmppath}/%{name}.app.files +%define filesystem_filelist %{_tmppath}/%{name}.filesystem.files + +%define default_filesystem / /opt /usr /usr/bin /usr/lib /usr/local /usr/local/bin /usr/local/lib + %description APPLICATION_DESCRIPTION @@ -34,19 +45,22 @@ install -d -m 755 %{buildroot}APPLICATION_DIRECTORY cp -r %{_sourcedir}APPLICATION_DIRECTORY/* %{buildroot}APPLICATION_DIRECTORY %if "xAPPLICATION_LICENSE_FILE" != x %define license_install_file %{_defaultlicensedir}/%{name}-%{version}/%{basename:APPLICATION_LICENSE_FILE} - install -d -m 755 %{buildroot}%{dirname:%{license_install_file}} - install -m 644 APPLICATION_LICENSE_FILE %{buildroot}%{license_install_file} + install -d -m 755 "%{buildroot}%{dirname:%{license_install_file}}" + install -m 644 "APPLICATION_LICENSE_FILE" "%{buildroot}%{license_install_file}" +%endif +(cd %{buildroot} && find . -type d) | sed -e 's/^\.//' -e '/^$/d' | sort > %{app_filelist} +{ rpm -ql filesystem || echo %{default_filesystem}; } | sort > %{filesystem_filelist} +comm -23 %{app_filelist} %{filesystem_filelist} > %{package_filelist} +sed -i -e 's/.*/%dir "&"/' %{package_filelist} +(cd %{buildroot} && find . -not -type d) | sed -e 's/^\.//' -e 's/.*/"&"/' >> %{package_filelist} +%if "xAPPLICATION_LICENSE_FILE" != x + sed -i -e 's|"%{license_install_file}"||' -e '/^$/d' %{package_filelist} %endif -%files +%files -f %{package_filelist} %if "xAPPLICATION_LICENSE_FILE" != x - %license %{license_install_file} - %{dirname:%{license_install_file}} + %license "%{license_install_file}" %endif -# If installation directory for the application is /a/b/c, we want only root -# component of the path (/a) in the spec file to make sure all subdirectories -# are owned by the package. -%(echo APPLICATION_DIRECTORY | sed -e "s|\(^/[^/]\{1,\}\).*$|\1|") %post DESKTOP_COMMANDS_INSTALL diff --git a/src/jdk.incubator.jpackage/linux/native/applauncher/Executor.cpp b/src/jdk.incubator.jpackage/linux/native/applauncher/Executor.cpp new file mode 100644 index 00000000000..d8a1755ac88 --- /dev/null +++ b/src/jdk.incubator.jpackage/linux/native/applauncher/Executor.cpp @@ -0,0 +1,71 @@ +/* + * Copyright (c) 2020, 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. + */ + +#include +#include +#include "Executor.h" +#include "Log.h" +#include "ErrorHandling.h" + + +int executeCommandLineAndReadStdout(const std::string& cmd, + CommandOutputConsumer& consumer) { + FILE * stream = popen(cmd.c_str(), "r"); + if (!stream) { + JP_THROW(tstrings::any() << "popen(" << cmd + << ") failed. Error: " << lastCRTError()); + } + + LOG_TRACE(tstrings::any() << "Reading output of [" << cmd << "] command"); + + try { + bool useConsumer = true; + std::string buf; + for (;;) { + const int c = fgetc(stream); + if(c == EOF) { + if (useConsumer && !buf.empty()) { + LOG_TRACE(tstrings::any() << "Next line: [" << buf << "]"); + consumer.accept(buf); + } + break; + } + + if (c == '\n' && useConsumer) { + LOG_TRACE(tstrings::any() << "Next line: [" << buf << "]"); + useConsumer = !consumer.accept(buf); + buf.clear(); + } else { + buf.push_back(static_cast(c)); + } + } + return pclose(stream); + } catch (...) { + if (stream) { + pclose(stream); + } + throw; + } +} diff --git a/src/jdk.incubator.jpackage/linux/native/applauncher/Executor.h b/src/jdk.incubator.jpackage/linux/native/applauncher/Executor.h new file mode 100644 index 00000000000..44a5f0c7e26 --- /dev/null +++ b/src/jdk.incubator.jpackage/linux/native/applauncher/Executor.h @@ -0,0 +1,44 @@ +/* + * Copyright (c) 2020, 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. + */ + +#ifndef EXECUTOR_H +#define EXECUTOR_H + +#include "tstrings.h" + + +class CommandOutputConsumer { +public: + virtual ~CommandOutputConsumer() {} + + virtual bool accept(const std::string& line) { + return true; + }; +}; + +int executeCommandLineAndReadStdout(const std::string& cmd, + CommandOutputConsumer& consumer); + +#endif // #ifndef EXECUTOR_H diff --git a/src/jdk.incubator.jpackage/linux/native/applauncher/LinuxLauncher.cpp b/src/jdk.incubator.jpackage/linux/native/applauncher/LinuxLauncher.cpp index 4debabc2fe2..88a7ad48355 100644 --- a/src/jdk.incubator.jpackage/linux/native/applauncher/LinuxLauncher.cpp +++ b/src/jdk.incubator.jpackage/linux/native/applauncher/LinuxLauncher.cpp @@ -23,29 +23,41 @@ * questions. */ +#include #include "AppLauncher.h" #include "FileUtils.h" #include "UnixSysInfo.h" +#include "Package.h" namespace { + void launchApp() { setlocale(LC_ALL, "en_US.utf8"); const tstring launcherPath = SysInfo::getProcessModulePath(); - // Launcher should be in "bin" subdirectory of app image. - const tstring appImageRoot = FileUtils::dirname( - FileUtils::dirname(launcherPath)); + const Package ownerPackage = Package::findOwnerOfFile(launcherPath); - AppLauncher() - .setImageRoot(appImageRoot) - .addJvmLibName(_T("lib/libjli.so")) - .setAppDir(FileUtils::mkpath() << appImageRoot << _T("lib/app")) - .setDefaultRuntimePath(FileUtils::mkpath() << appImageRoot - << _T("lib/runtime")) - .launch(); + AppLauncher appLauncher; + appLauncher.addJvmLibName(_T("lib/libjli.so")); + + if (ownerPackage.name().empty()) { + // Launcher should be in "bin" subdirectory of app image. + const tstring appImageRoot = FileUtils::dirname( + FileUtils::dirname(launcherPath)); + + appLauncher + .setImageRoot(appImageRoot) + .setAppDir(FileUtils::mkpath() << appImageRoot << _T("lib/app")) + .setDefaultRuntimePath(FileUtils::mkpath() << appImageRoot + << _T("lib/runtime")); + } else { + ownerPackage.initAppLauncher(appLauncher); + } + + appLauncher.launch(); } } // namespace diff --git a/src/jdk.incubator.jpackage/linux/native/applauncher/Package.cpp b/src/jdk.incubator.jpackage/linux/native/applauncher/Package.cpp new file mode 100644 index 00000000000..3665d7e14a0 --- /dev/null +++ b/src/jdk.incubator.jpackage/linux/native/applauncher/Package.cpp @@ -0,0 +1,148 @@ +/* + * Copyright (c) 2020, 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. + */ + +#include "Package.h" +#include "Executor.h" +#include "AppLauncher.h" +#include "ErrorHandling.h" + + +Package::Package(): type(Unknown) { +} + + +namespace { +class FirstLineConsumer : public CommandOutputConsumer { +public: + FirstLineConsumer(): processed(false) { + } + + virtual bool accept(const std::string& line) { + if (!processed) { + value = line; + processed = true; + } + return processed; + }; + + std::string getValue() const { + if (!processed) { + JP_THROW("No output captured"); + } + return value; + } + +private: + bool processed; + std::string value; +}; + + +std::string findOwnerOfFile(const std::nothrow_t&, const std::string& cmdline, + const std::string& path) { + try { + FirstLineConsumer consumer; + int exitCode = executeCommandLineAndReadStdout( + cmdline + " \'" + path + "\' 2>/dev/null", consumer); + if (exitCode == 0) { + return consumer.getValue(); + } + } catch (...) { + } + return ""; +} + +} // namespace + +Package Package::findOwnerOfFile(const std::string& path) { + Package result; + result.theName = ::findOwnerOfFile(std::nothrow, + "rpm --queryformat '%{NAME}' -qf", path); + if (!result.theName.empty()) { + result.type = RPM; + } else { + tstring_array components = tstrings::split(::findOwnerOfFile( + std::nothrow, "dpkg -S", path), ":"); + if (!components.empty()) { + result.theName = components.front(); + if (!result.theName.empty()) { + result.type = DEB; + } + } + } + + return result; +} + + +namespace { +class AppLauncherInitializer : public CommandOutputConsumer { +public: + AppLauncherInitializer() { + } + + virtual bool accept(const std::string& line) { + if (appDir.empty()) { + if (tstrings::endsWith(line, "/app")) { + appDir = line; + } + } + + if (runtimeDir.empty()) { + if (tstrings::endsWith(line, "/runtime")) { + runtimeDir = line; + } + } + + return !appDir.empty() && !runtimeDir.empty(); + }; + + void apply(AppLauncher& launcher) { + launcher.setDefaultRuntimePath(runtimeDir); + launcher.setAppDir(appDir); + } + +private: + std::string appDir; + std::string runtimeDir; +}; + +} // namespace + +void Package::initAppLauncher(AppLauncher& appLauncher) const { + AppLauncherInitializer consumer; + int exitCode = -1; + if (type == RPM) { + exitCode = executeCommandLineAndReadStdout( + "rpm -ql \'" + theName + "\'", consumer); + } else if (type == DEB) { + exitCode = executeCommandLineAndReadStdout( + "dpkg -L \'" + theName + "\'", consumer); + } + + if (exitCode == 0) { + consumer.apply(appLauncher); + } +} diff --git a/src/jdk.incubator.jpackage/linux/native/applauncher/Package.h b/src/jdk.incubator.jpackage/linux/native/applauncher/Package.h new file mode 100644 index 00000000000..4c25d2c7bc1 --- /dev/null +++ b/src/jdk.incubator.jpackage/linux/native/applauncher/Package.h @@ -0,0 +1,55 @@ +/* + * Copyright (c) 2020, 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. + */ + +#ifndef PACKAGE_H +#define PACKAGE_H + + +#include "tstrings.h" + + +class AppLauncher; + + +class Package { +public: + Package(); + + std::string name() const { + return theName; + } + + void initAppLauncher(AppLauncher& appLauncher) const; + + static Package findOwnerOfFile(const std::string& path); + +private: + enum Type { Unknown, RPM, DEB }; + + Type type; + std::string theName; +}; + +#endif // #ifndef PACKAGE_H diff --git a/src/jdk.incubator.jpackage/share/classes/jdk/incubator/jpackage/internal/ApplicationLayout.java b/src/jdk.incubator.jpackage/share/classes/jdk/incubator/jpackage/internal/ApplicationLayout.java index 0b651b115cd..ef2e1eef136 100644 --- a/src/jdk.incubator.jpackage/share/classes/jdk/incubator/jpackage/internal/ApplicationLayout.java +++ b/src/jdk.incubator.jpackage/share/classes/jdk/incubator/jpackage/internal/ApplicationLayout.java @@ -177,5 +177,18 @@ public final class ApplicationLayout implements PathGroup.Facade { * corresponding layout. */ public ApplicationLayout appLayout() { - final ApplicationLayout layout; - if (isRuntime()) { + ApplicationLayout layout = onLinuxPackageInstallDir(null, + installDir -> { + String packageName = LinuxHelper.getPackageName(this); + // Convert '/usr' to 'usr'. It will be set to proper root in + // subsequent ApplicationLayout.resolveAt() call. + return ApplicationLayout.linuxUsrTreePackageImage(Path.of( + "/").relativize(installDir), packageName); + }); + + if (layout != null) { + } else if (isRuntime()) { layout = ApplicationLayout.javaRuntime(); } else { layout = ApplicationLayout.platformAppImage(); @@ -364,7 +373,25 @@ public final class JPackageCommand extends CommandArguments { return layout.resolveAt(outputBundle()); } - return layout.resolveAt(appInstallationDirectory()); + return layout.resolveAt(pathToUnpackedPackageFile( + appInstallationDirectory())); + } + + /** + * Returns path to package file in unpacked package directory or the given + * path if the package is not unpacked. + */ + public Path pathToUnpackedPackageFile(Path path) { + Path unpackDir = unpackedPackageDirectory(); + if (unpackDir == null) { + return path; + } + return unpackDir.resolve(TKit.removeRootFromAbsolutePath(path)); + } + + Path unpackedPackageDirectory() { + verifyIsOfType(PackageType.NATIVE); + return getArgumentValue(UNPACKED_PATH_ARGNAME, () -> null, Path::of); } /** @@ -372,28 +399,19 @@ public final class JPackageCommand extends CommandArguments { * this is build image command. * * E.g. on Linux for app named Foo default the function will return - * `/opt/foo` + * `/opt/foo`. + * On Linux for install directory in `/usr` tree the function returns `/`. + * */ public Path appInstallationDirectory() { - Path unpackedDir = getArgumentValue(UNPACKED_PATH_ARGNAME, () -> null, - Path::of); - if (unpackedDir != null) { - return unpackedDir; - } - if (isImagePackageType()) { return null; } if (TKit.isLinux()) { - if (isRuntime()) { - // Not fancy, but OK. - return Path.of(getArgumentValue("--install-dir", () -> "/opt"), - LinuxHelper.getPackageName(this)); - } - - // Launcher is in "bin" subfolder of the installation directory. - return appLauncherPath().getParent().getParent(); + return onLinuxPackageInstallDir(installDir -> installDir.resolve( + LinuxHelper.getPackageName(this)), + installDir -> Path.of("/")); } if (TKit.isWindows()) { @@ -444,14 +462,6 @@ public final class JPackageCommand extends CommandArguments { launcherName = launcherName + ".exe"; } - if (isImagePackageType() || isPackageUnpacked()) { - return appLayout().launchersDirectory().resolve(launcherName); - } - - if (TKit.isLinux()) { - return LinuxHelper.getLauncherPath(this).getParent().resolve(launcherName); - } - return appLayout().launchersDirectory().resolve(launcherName); } @@ -514,6 +524,23 @@ public final class JPackageCommand extends CommandArguments { return false; } + public boolean canRunLauncher(String msg) { + if (isFakeRuntime(msg)) { + return false; + } + + if (isPackageUnpacked()) { + return Boolean.FALSE != onLinuxPackageInstallDir(null, installDir -> { + TKit.trace(String.format( + "%s because the package in [%s] directory is not installed ", + msg, installDir)); + return Boolean.FALSE; + }); + } + + return true; + } + public boolean isPackageUnpacked(String msg) { if (isPackageUnpacked()) { TKit.trace(String.format( @@ -523,7 +550,7 @@ public final class JPackageCommand extends CommandArguments { return false; } - boolean isPackageUnpacked() { + public boolean isPackageUnpacked() { return hasArgument(UNPACKED_PATH_ARGNAME); } @@ -780,6 +807,22 @@ public final class JPackageCommand extends CommandArguments { return !immutable; } + private T onLinuxPackageInstallDir(Function anyInstallDirConsumer, + Function usrInstallDirConsumer) { + if (TKit.isLinux()) { + Path installDir = Path.of(getArgumentValue("--install-dir", + () -> "/opt")); + if (Set.of("/usr", "/usr/local").contains(installDir.toString())) { + if (usrInstallDirConsumer != null) { + return usrInstallDirConsumer.apply(installDir); + } + } else if (anyInstallDirConsumer != null) { + return anyInstallDirConsumer.apply(installDir); + } + } + return null; + } + private final class Actions implements Runnable { Actions() { actions = new ArrayList<>(); diff --git a/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/LinuxHelper.java b/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/LinuxHelper.java index 78b7a171ddc..72ab4cef4e7 100644 --- a/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/LinuxHelper.java +++ b/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/LinuxHelper.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2019, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2019, 2020, 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 @@ -165,8 +165,7 @@ public class LinuxHelper { .addArgument(cmd.outputBundle()) .addArgument(destinationDir) .execute(); - return destinationDir.resolve(String.format(".%s", - cmd.appInstallationDirectory())).normalize(); + return destinationDir; }; return deb; } @@ -191,8 +190,7 @@ public class LinuxHelper { cmd.outputBundle().toAbsolutePath().toString()))) .setDirectory(destinationDir) .execute(); - return destinationDir.resolve(String.format(".%s", - cmd.appInstallationDirectory())).normalize(); + return destinationDir; }; return rpm; diff --git a/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/MacHelper.java b/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/MacHelper.java index 2f012904d7b..0aecd6443cd 100644 --- a/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/MacHelper.java +++ b/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/MacHelper.java @@ -102,20 +102,27 @@ public class MacHelper { withExplodedDmg(cmd, dmgImage -> { Executor.of("sudo", "cp", "-r") .addArgument(dmgImage) - .addArgument("/Applications") + .addArgument(getInstallationDirectory(cmd).getParent()) .execute(); }); }; dmg.unpackHandler = (cmd, destinationDir) -> { - Path[] unpackedFolder = new Path[1]; + Path unpackDir = destinationDir.resolve( + TKit.removeRootFromAbsolutePath( + getInstallationDirectory(cmd)).getParent()); + try { + Files.createDirectories(unpackDir); + } catch (IOException ex) { + throw new RuntimeException(ex); + } + withExplodedDmg(cmd, dmgImage -> { Executor.of("cp", "-r") .addArgument(dmgImage) - .addArgument(destinationDir) + .addArgument(unpackDir) .execute(); - unpackedFolder[0] = destinationDir.resolve(dmgImage.getFileName()); }); - return unpackedFolder[0]; + return destinationDir; }; dmg.uninstallHandler = cmd -> { cmd.verifyIsOfType(PackageType.MAC_DMG); @@ -143,13 +150,25 @@ public class MacHelper { .addArgument(cmd.outputBundle()) .addArgument(destinationDir.resolve("data")) // We need non-existing folder .execute(); + + final Path unpackRoot = destinationDir.resolve("unpacked"); + + Path installDir = TKit.removeRootFromAbsolutePath( + getInstallationDirectory(cmd)).getParent(); + final Path unpackDir = unpackRoot.resolve(installDir); + try { + Files.createDirectories(unpackDir); + } catch (IOException ex) { + throw new RuntimeException(ex); + } + Executor.of("tar", "-C") - .addArgument(destinationDir) + .addArgument(unpackDir) .addArgument("-xvf") .addArgument(Path.of(destinationDir.toString(), "data", cmd.name() + "-app.pkg", "Payload")) .execute(); - return destinationDir.resolve(cmd.name() + ".app"); + return unpackRoot; }; pkg.uninstallHandler = cmd -> { cmd.verifyIsOfType(PackageType.MAC_PKG); diff --git a/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/PackageTest.java b/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/PackageTest.java index 46413a07bfa..215b782448a 100644 --- a/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/PackageTest.java +++ b/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/PackageTest.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2019, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2019, 2020, 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 @@ -27,15 +27,30 @@ import java.awt.GraphicsEnvironment; import java.io.File; import java.nio.file.Files; import java.nio.file.Path; -import java.util.*; -import java.util.function.*; +import java.util.ArrayList; +import java.util.Collection; +import java.util.Collections; +import java.util.HashMap; +import java.util.HashSet; +import java.util.List; +import java.util.ListIterator; +import java.util.Map; +import java.util.Objects; +import java.util.Set; +import java.util.function.BiConsumer; +import java.util.function.BiFunction; +import java.util.function.Consumer; +import java.util.function.Predicate; import java.util.stream.Collectors; import java.util.stream.Stream; -import jdk.jpackage.test.Functional.ThrowingConsumer; import jdk.incubator.jpackage.internal.AppImageFile; +import jdk.incubator.jpackage.internal.ApplicationLayout; import jdk.jpackage.test.Functional.ThrowingBiConsumer; +import jdk.jpackage.test.Functional.ThrowingConsumer; import jdk.jpackage.test.Functional.ThrowingRunnable; -import static jdk.jpackage.test.PackageType.*; +import jdk.jpackage.test.Functional.ThrowingSupplier; + + /** * Instance of PackageTest is for configuring and running a single jpackage @@ -170,7 +185,7 @@ public final class PackageTest extends RunnablePackageTest { } public PackageTest addBundleDesktopIntegrationVerifier(boolean integrated) { - forTypes(LINUX, () -> { + forTypes(PackageType.LINUX, () -> { LinuxHelper.addBundleDesktopIntegrationVerifier(this, integrated); }); return this; @@ -549,8 +564,19 @@ public final class PackageTest extends RunnablePackageTest { } } - TKit.assertPathExists(AppImageFile.getPathInAppImage( - cmd.appInstallationDirectory()), false); + if (cmd.isPackageUnpacked()) { + final Path appImageFile = AppImageFile.getPathInAppImage( + Path.of("")); + try (Stream walk = ThrowingSupplier.toSupplier( + () -> Files.walk(cmd.unpackedPackageDirectory())).get()) { + walk.filter(path -> path.getFileName().equals(appImageFile)) + .findFirst() + .ifPresent(path -> TKit.assertPathExists(path, false)); + } + } else { + TKit.assertPathExists(AppImageFile.getPathInAppImage( + cmd.appInstallationDirectory()), false); + } installVerifiers.forEach(v -> v.accept(cmd)); } @@ -566,7 +592,13 @@ public final class PackageTest extends RunnablePackageTest { } } - TKit.assertPathExists(cmd.appInstallationDirectory(), false); + Path appInstallDir = cmd.appInstallationDirectory(); + if (TKit.isLinux() && Path.of("/").equals(appInstallDir)) { + ApplicationLayout appLayout = cmd.appLayout(); + TKit.assertPathExists(appLayout.runtimeDirectory(), false); + } else { + TKit.assertPathExists(appInstallDir, false); + } uninstallVerifiers.forEach(v -> v.accept(cmd)); } diff --git a/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/TKit.java b/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/TKit.java index dd474e0be33..799286188f4 100644 --- a/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/TKit.java +++ b/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/TKit.java @@ -209,6 +209,17 @@ final public class TKit { } } + static Path removeRootFromAbsolutePath(Path v) { + if (!v.isAbsolute()) { + throw new IllegalArgumentException(); + } + + if (v.getNameCount() == 0) { + return Path.of(""); + } + return v.subpath(0, v.getNameCount()); + } + public static void createTextFile(Path propsFilename, Collection lines) { createTextFile(propsFilename, lines.stream()); } diff --git a/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/WindowsHelper.java b/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/WindowsHelper.java index aebecc67e9e..9ca4800a471 100644 --- a/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/WindowsHelper.java +++ b/test/jdk/tools/jpackage/helpers/jdk/jpackage/test/WindowsHelper.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2019, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2019, 2020, 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 @@ -41,14 +41,18 @@ public class WindowsHelper { } static Path getInstallationDirectory(JPackageCommand cmd) { - Path installSubDir = getInstallationSubDirectory(cmd); - if (isUserLocalInstall(cmd)) { - return USER_LOCAL.resolve(installSubDir); - } - return PROGRAM_FILES.resolve(installSubDir); + return getInstallationRootDirectory(cmd).resolve( + getInstallationSubDirectory(cmd)); } - static Path getInstallationSubDirectory(JPackageCommand cmd) { + private static Path getInstallationRootDirectory(JPackageCommand cmd) { + if (isUserLocalInstall(cmd)) { + return USER_LOCAL; + } + return PROGRAM_FILES; + } + + private static Path getInstallationSubDirectory(JPackageCommand cmd) { cmd.verifyIsOfType(PackageType.WINDOWS); return Path.of(cmd.getArgumentValue("--install-dir", () -> cmd.name())); } @@ -81,11 +85,21 @@ public class WindowsHelper { msi.uninstallHandler = cmd -> installMsi.accept(cmd, false); msi.unpackHandler = (cmd, destinationDir) -> { cmd.verifyIsOfType(PackageType.WIN_MSI); - runMsiexecWithRetries(Executor.of("msiexec", "/a") - .addArgument(cmd.outputBundle().normalize()) - .addArguments("/qn", String.format("TARGETDIR=%s", - destinationDir.toAbsolutePath().normalize()))); - return destinationDir.resolve(getInstallationSubDirectory(cmd)); + final Path unpackBat = destinationDir.resolve("unpack.bat"); + final Path unpackDir = destinationDir.resolve( + TKit.removeRootFromAbsolutePath( + getInstallationRootDirectory(cmd))); + // Put msiexec in .bat file because can't pass value of TARGETDIR + // property containing spaces through ProcessBuilder properly. + TKit.createTextFile(unpackBat, List.of(String.join(" ", List.of( + "msiexec", + "/a", + String.format("\"%s\"", cmd.outputBundle().normalize()), + "/qn", + String.format("TARGETDIR=\"%s\"", + unpackDir.toAbsolutePath().normalize()))))); + runMsiexecWithRetries(Executor.of("cmd", "/c", unpackBat.toString())); + return destinationDir; }; return msi; } diff --git a/test/jdk/tools/jpackage/linux/jdk/jpackage/tests/UsrTreeTest.java b/test/jdk/tools/jpackage/linux/jdk/jpackage/tests/UsrTreeTest.java new file mode 100644 index 00000000000..7a349175f0e --- /dev/null +++ b/test/jdk/tools/jpackage/linux/jdk/jpackage/tests/UsrTreeTest.java @@ -0,0 +1,127 @@ +/* + * Copyright (c) 2020, 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. + * + * 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. + */ + +import java.nio.file.Path; +import java.util.List; +import java.util.function.Consumer; +import java.util.stream.Collectors; +import jdk.jpackage.test.TKit; +import jdk.jpackage.test.PackageTest; +import jdk.jpackage.test.PackageType; +import jdk.jpackage.test.LinuxHelper; +import jdk.jpackage.test.Annotations.Test; + + +/** + * Simple Linux specific packaging test. Resulting package should be installed + * in /usr directory tree. + */ + +/* + * @test + * @summary jpackage command run installing app in /usr directory tree + * @library ../../../../helpers + * @key jpackagePlatformPackage + * @requires jpackage.test.SQETest == null + * @requires (os.family == "linux") + * @build jdk.jpackage.test.* + * @modules jdk.incubator.jpackage/jdk.incubator.jpackage.internal + * @compile UsrTreeTest.java + * @run main/othervm/timeout=720 -Xmx512m jdk.jpackage.test.Main + * --jpt-run=UsrTreeTest + */ +public class UsrTreeTest { + + @Test + public static void testUsr() { + test("/usr", true); + } + + @Test + public static void testUsrLocal() { + test("/usr/local", true); + } + + @Test + public static void testUsrCustom() { + test("/usr/foo", false); + } + + @Test + public static void testUsrCustom2() { + test("/usrbuz", false); + } + + private static void test(String installDir, boolean expectedImageSplit) { + new PackageTest() + .forTypes(PackageType.LINUX) + .configureHelloApp() + .addInitializer(cmd -> cmd.addArguments("--install-dir", installDir)) + .addBundleDesktopIntegrationVerifier(false) + .addBundleVerifier(cmd -> { + final String packageName = LinuxHelper.getPackageName(cmd); + final Path launcherPath = cmd.appLauncherPath(); + final Path launcherCfgPath = cmd.appLauncherCfgPath(null); + final Path commonPath = commonPath(launcherPath, launcherCfgPath); + + final boolean actualImageSplit = !commonPath.getFileName().equals( + Path.of(packageName)); + TKit.assertTrue(expectedImageSplit == actualImageSplit, + String.format( + "Check there is%spackage name [%s] in common path [%s] between [%s] and [%s]", + expectedImageSplit ? " no " : " ", packageName, + commonPath, launcherPath, launcherCfgPath)); + + List packageFiles = LinuxHelper.getPackageFiles(cmd).collect( + Collectors.toList()); + + Consumer packageFileVerifier = file -> { + TKit.assertTrue(packageFiles.stream().filter( + path -> path.equals(file)).findFirst().orElse( + null) != null, String.format( + "Check file [%s] is in [%s] package", file, + packageName)); + }; + + packageFileVerifier.accept(launcherPath); + packageFileVerifier.accept(launcherCfgPath); + }) + .run(); + } + + private static Path commonPath(Path a, Path b) { + if (a.equals(b)) { + return a; + } + + final int minCount = Math.min(a.getNameCount(), b.getNameCount()); + for (int i = minCount; i > 0; i--) { + Path sp = a.subpath(0, i); + if (sp.equals(b.subpath(0, i))) { + return a.getRoot().resolve(sp); + } + } + + return a.getRoot(); + } +} diff --git a/test/jdk/tools/jpackage/share/InstallDirTest.java b/test/jdk/tools/jpackage/share/InstallDirTest.java index 81c3e1e3c07..41ab9c28721 100644 --- a/test/jdk/tools/jpackage/share/InstallDirTest.java +++ b/test/jdk/tools/jpackage/share/InstallDirTest.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2018, 2019, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2018, 2020, 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 @@ -76,7 +76,7 @@ import jdk.jpackage.test.Annotations.Parameter; * @requires (os.family == "linux") * @requires (jpackage.test.SQETest == null) * @run main/othervm/timeout=360 -Xmx512m jdk.jpackage.test.Main - * --jpt-run=InstallDirTest.testLinuxInvalid,testLinuxUnsupported + * --jpt-run=InstallDirTest.testLinuxInvalid */ public class InstallDirTest { @@ -90,11 +90,12 @@ public class InstallDirTest { reply.put(PackageType.LINUX_RPM, reply.get(PackageType.LINUX_DEB)); reply.put(PackageType.MAC_PKG, Path.of("/Applications/jpackage")); + reply.put(PackageType.MAC_DMG, reply.get(PackageType.MAC_PKG)); return reply; }).get(); - new PackageTest().excludeTypes(PackageType.MAC_DMG).configureHelloApp() + new PackageTest().configureHelloApp() .addInitializer(cmd -> { cmd.addArguments("--install-dir", INSTALL_DIRS.get( cmd.packageType())); @@ -109,13 +110,6 @@ public class InstallDirTest { testLinuxBad(installDir, "Invalid installation directory"); } - @Parameter("/usr") - @Parameter("/usr/local") - @Parameter("/usr/foo") - public static void testLinuxUnsupported(String installDir) { - testLinuxBad(installDir, "currently unsupported"); - } - private static void testLinuxBad(String installDir, String errorMessageSubstring) { new PackageTest().configureHelloApp() diff --git a/test/jdk/tools/jpackage/share/LicenseTest.java b/test/jdk/tools/jpackage/share/LicenseTest.java index 52d6aa8db2d..fadfe59ad1e 100644 --- a/test/jdk/tools/jpackage/share/LicenseTest.java +++ b/test/jdk/tools/jpackage/share/LicenseTest.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2018, 2019, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2018, 2020, 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 @@ -78,47 +78,47 @@ import jdk.jpackage.test.TKit; * @summary jpackage with --license-file * @library ../helpers * @key jpackagePlatformPackage + * @build jdk.jpackage.test.* * @compile LicenseTest.java * @requires (os.family == "linux") * @requires (jpackage.test.SQETest == null) * @modules jdk.incubator.jpackage/jdk.incubator.jpackage.internal - * @run main/othervm/timeout=360 -Xmx512m jdk.jpackage.test.Main + * @run main/othervm/timeout=1440 -Xmx512m jdk.jpackage.test.Main * --jpt-run=LicenseTest.testCustomDebianCopyright * --jpt-run=LicenseTest.testCustomDebianCopyrightSubst + * --jpt-run=LicenseTest.testLinuxLicenseInUsrTree + * --jpt-run=LicenseTest.testLinuxLicenseInUsrTree2 + * --jpt-run=LicenseTest.testLinuxLicenseInUsrTree3 + * --jpt-run=LicenseTest.testLinuxLicenseInUsrTree4 */ public class LicenseTest { public static void testCommon() { - new PackageTest().configureHelloApp() + PackageTest test = new PackageTest().configureHelloApp() .addInitializer(cmd -> { cmd.addArguments("--license-file", TKit.createRelativePathCopy( LICENSE_FILE)); - }) - .forTypes(PackageType.LINUX) - .addBundleVerifier(cmd -> { - verifyLicenseFileInLinuxPackage(cmd, linuxLicenseFile(cmd)); - }) - .addInstallVerifier(cmd -> { - Path path = linuxLicenseFile(cmd); - if (path != null) { - TKit.assertReadableFileExists(path); - } - }) - .addUninstallVerifier(cmd -> { - verifyLicenseFileNotInstalledLinux(linuxLicenseFile(cmd)); - }) - .forTypes(PackageType.LINUX_DEB) - .addInstallVerifier(cmd -> { - verifyLicenseFileInstalledDebian(debLicenseFile(cmd)); - }) - .forTypes(PackageType.LINUX_RPM) - .addInstallVerifier(cmd -> { - Path path = rpmLicenseFile(cmd); - if (path != null) { - verifyLicenseFileInstalledRpm(path); - } - }) - .run(); + }); + + initLinuxLicenseVerifier(test.forTypes(PackageType.LINUX)); + + test.run(); + } + + public static void testLinuxLicenseInUsrTree() { + testLinuxLicenseInUsrTree("/usr"); + } + + public static void testLinuxLicenseInUsrTree2() { + testLinuxLicenseInUsrTree("/usr/local"); + } + + public static void testLinuxLicenseInUsrTree3() { + testLinuxLicenseInUsrTree("/usr/foo"); + } + + public static void testLinuxLicenseInUsrTree4() { + testLinuxLicenseInUsrTree("/usrbuz"); } public static void testCustomDebianCopyright() { @@ -129,38 +129,78 @@ public class LicenseTest { new CustomDebianCopyrightTest().withSubstitution(true).run(); } - private static Path rpmLicenseFile(JPackageCommand cmd) { - if (cmd.isPackageUnpacked("Not checking for rpm license file")) { - return null; - } + private static PackageTest initLinuxLicenseVerifier(PackageTest test) { + return test + .addBundleVerifier(cmd -> { + verifyLicenseFileInLinuxPackage(cmd, linuxLicenseFile(cmd)); + }) + .addInstallVerifier(cmd -> { + verifyLicenseFileInstalledLinux(cmd); + }) + .addUninstallVerifier(cmd -> { + verifyLicenseFileNotInstalledLinux(linuxLicenseFile(cmd)); + }); + } + private static void testLinuxLicenseInUsrTree(String installDir) { + PackageTest test = new PackageTest() + .forTypes(PackageType.LINUX) + .configureHelloApp() + .addInitializer(cmd -> { + cmd.setFakeRuntime(); + cmd.addArguments("--license-file", TKit.createRelativePathCopy( + LICENSE_FILE)); + cmd.addArguments("--install-dir", installDir); + }); + + initLinuxLicenseVerifier(test); + + test.run(); + } + + private static Path rpmLicenseFile(JPackageCommand cmd) { final Path licenseRoot = Path.of( new Executor() .setExecutable("rpm") .addArguments("--eval", "%{_defaultlicensedir}") .executeAndGetFirstLineOfOutput()); + final Path licensePath = licenseRoot.resolve(String.format("%s-%s", LinuxHelper.getPackageName(cmd), cmd.version())).resolve( LICENSE_FILE.getFileName()); + return licensePath; } + private static Path debLicenseFile(JPackageCommand cmd) { + Path installDir = cmd.appInstallationDirectory(); + + if (installDir.equals(Path.of("/")) || installDir.startsWith("/usr")) { + // Package is in '/usr' tree + return Path.of("/usr/share/doc/", LinuxHelper.getPackageName(cmd), + "copyright"); + } + + return installDir.resolve("share/doc/copyright"); + } + private static Path linuxLicenseFile(JPackageCommand cmd) { cmd.verifyIsOfType(PackageType.LINUX); + final Path licenseFile; switch (cmd.packageType()) { case LINUX_DEB: - return debLicenseFile(cmd); + licenseFile = debLicenseFile(cmd); + break; case LINUX_RPM: - return rpmLicenseFile(cmd); + licenseFile = rpmLicenseFile(cmd); + break; default: - return null; + throw new IllegalArgumentException(); } - } - private static Path debLicenseFile(JPackageCommand cmd) { - return cmd.appInstallationDirectory().resolve("share/doc/copyright"); + return cmd.pathToUnpackedPackageFile(licenseFile); } private static void verifyLicenseFileInLinuxPackage(JPackageCommand cmd, @@ -199,6 +239,26 @@ public class LicenseTest { licenseFile, LICENSE_FILE)); } + private static void verifyLicenseFileInstalledLinux(JPackageCommand cmd) + throws IOException { + + final Path licenseFile = linuxLicenseFile(cmd); + TKit.assertReadableFileExists(licenseFile); + + switch (cmd.packageType()) { + case LINUX_DEB: + verifyLicenseFileInstalledDebian(licenseFile); + break; + + case LINUX_RPM: + verifyLicenseFileInstalledRpm(licenseFile); + break; + + default: + throw new IllegalArgumentException(); + } + } + private static void verifyLicenseFileNotInstalledLinux(Path licenseFile) { TKit.assertPathExists(licenseFile.getParent(), false); } @@ -266,7 +326,7 @@ public class LicenseTest { licenseFileText()); }) .addInstallVerifier(cmd -> { - Path installedLicenseFile = debLicenseFile(cmd); + Path installedLicenseFile = linuxLicenseFile(cmd); TKit.assertStringListEquals(expetedLicenseFileText(), DEBIAN_COPYRIGT_FILE_STRIPPER.apply(Files.readAllLines( installedLicenseFile)), String.format( diff --git a/test/jdk/tools/jpackage/share/RuntimePackageTest.java b/test/jdk/tools/jpackage/share/RuntimePackageTest.java index 42a53db1663..058cb0cccc2 100644 --- a/test/jdk/tools/jpackage/share/RuntimePackageTest.java +++ b/test/jdk/tools/jpackage/share/RuntimePackageTest.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2018, 2019, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2018, 2020, 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 @@ -50,18 +50,53 @@ import jdk.jpackage.test.Annotations.Test; * @library ../helpers * @key jpackagePlatformPackage * @build jdk.jpackage.test.* - * @comment Temporary disable for Linux and OSX until functionality implemented + * @comment Temporary disable for OSX until functionality implemented * @requires (os.family != "mac") + * @requires (jpackage.test.SQETest == null) + * @modules jdk.incubator.jpackage/jdk.incubator.jpackage.internal + * @compile RuntimePackageTest.java + * @run main/othervm/timeout=1400 -Xmx512m jdk.jpackage.test.Main + * --jpt-run=RuntimePackageTest + */ + +/* + * @test + * @summary jpackage with --runtime-image + * @library ../helpers + * @key jpackagePlatformPackage + * @build jdk.jpackage.test.* + * @comment Temporary disable for OSX until functionality implemented + * @requires (os.family != "mac") + * @requires (jpackage.test.SQETest != null) * @modules jdk.incubator.jpackage/jdk.incubator.jpackage.internal * @compile RuntimePackageTest.java * @run main/othervm/timeout=720 -Xmx512m jdk.jpackage.test.Main - * --jpt-run=RuntimePackageTest + * --jpt-run=RuntimePackageTest.test */ public class RuntimePackageTest { @Test public static void test() { - new PackageTest() + init(PackageType.NATIVE).run(); + } + + @Test + public static void testUsrInstallDir() { + init(PackageType.LINUX) + .addInitializer(cmd -> cmd.addArguments("--install-dir", "/usr")) + .run(); + } + + @Test + public static void testUsrInstallDir2() { + init(PackageType.LINUX) + .addInitializer(cmd -> cmd.addArguments("--install-dir", "/usr/lib/Java")) + .run(); + } + + private static PackageTest init(Set types) { + return new PackageTest() + .forTypes(types) .addInitializer(cmd -> { cmd.addArguments("--runtime-image", Optional.ofNullable( JPackageCommand.DEFAULT_RUNTIME_IMAGE).orElse(Path.of( @@ -83,8 +118,7 @@ public class RuntimePackageTest { assertFileListEmpty(srcRuntime, "Missing"); assertFileListEmpty(dstRuntime, "Unexpected"); - }) - .run(); + }); } private static Set listFiles(Path root) throws IOException {