All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.wildfly.plugin.provision.PackageServerMojo Maven / Gradle / Ivy

/*
 * Copyright The WildFly Authors
 * SPDX-License-Identifier: Apache-2.0
 */
package org.wildfly.plugin.provision;

import static org.wildfly.plugin.core.Constants.CLI_ECHO_COMMAND_ARG;
import static org.wildfly.plugin.core.Constants.STANDALONE;
import static org.wildfly.plugin.core.Constants.STANDALONE_XML;

import java.io.File;
import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.nio.file.StandardCopyOption;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;

import javax.inject.Inject;

import org.apache.maven.plugin.MojoExecutionException;
import org.apache.maven.plugin.MojoFailureException;
import org.apache.maven.plugins.annotations.LifecyclePhase;
import org.apache.maven.plugins.annotations.Mojo;
import org.apache.maven.plugins.annotations.Parameter;
import org.apache.maven.plugins.annotations.ResolutionScope;
import org.jboss.galleon.ProvisioningException;
import org.jboss.galleon.api.GalleonBuilder;
import org.jboss.galleon.api.config.GalleonProvisioningConfig;
import org.jboss.galleon.maven.plugin.util.MvnMessageWriter;
import org.jboss.galleon.util.IoUtils;
import org.wildfly.glow.ScanResults;
import org.wildfly.plugin.cli.BaseCommandConfiguration;
import org.wildfly.plugin.cli.CliSession;
import org.wildfly.plugin.cli.OfflineCommandExecutor;
import org.wildfly.plugin.common.PropertyNames;
import org.wildfly.plugin.common.StandardOutput;
import org.wildfly.plugin.common.Utils;
import org.wildfly.plugin.deployment.MojoDeploymentException;
import org.wildfly.plugin.deployment.PackageType;
import org.wildfly.plugin.tools.bootablejar.BootableJarSupport;

/**
 * Provision a server, copy extra content and deploy primary artifact if it
 * exists
 *
 * @author jfdenise
 * @since 3.0
 */
@Mojo(name = "package", requiresDependencyResolution = ResolutionScope.COMPILE_PLUS_RUNTIME, defaultPhase = LifecyclePhase.PACKAGE)
public class PackageServerMojo extends AbstractProvisionServerMojo {

    public static final String JAR = "jar";
    public static final String BOOTABLE_JAR_NAME_RADICAL = "server-";

    /**
     * A list of directories to copy content to the provisioned server. If a
     * directory is not absolute, it has to be relative to the project base
     * directory.
     */
    @Parameter(alias = "extra-server-content-dirs")
    List extraServerContentDirs = Collections.emptyList();

    /**
     * List of execution of CLI scripts and commands. An embedded server is
     * started for each CLI execution. If a script file is not absolute, it has
     * to be relative to the project base directory. CLI executions are
     * configured in the following way:
     *
     * 
     *   <packaging-scripts>
     *     <packaging-script>
     *       <scripts>
     *         <script>../scripts/script1.cli</script>
     *       </scripts>
     *       <commands>
     *         <command>/system-property=foo:add(value=bar)</command>
     *       </commands>
     *       <properties-files>
     *         <property-file>my-properties.properties</property-file>
     *       </properties-files>
     *       <java-opts>
     *         <java-opt>-Xmx256m</java-opt>
     *       </java-opts>
     *       <!-- Expressions resolved during server execution -->
     *       <resolve-expressions>false</resolve-expressions>
     *     </packaging-script>
     *   </packaging-scripts>
     * 
*/ @Parameter(alias = "packaging-scripts") private List packagingScripts = new ArrayList<>(); /** * The file name of the application to be deployed. *

* The {@code filename} property does have a default of * ${project.build.finalName}.${project.packaging}. The default * value is not injected as it normally would be due to packaging types like * {@code ejb} that result in a file with a {@code .jar} extension rather * than an {@code .ejb} extension. *

*/ @Parameter(property = PropertyNames.DEPLOYMENT_FILENAME) private String filename; /** * The name of the server configuration to use when deploying the * deployment. Defaults to 'standalone.xml'. If {@code layers-configuration-file-name} has been set, * this property is ignored and the deployment is deployed inside the configuration referenced from * {@code layers-configuration-file-name}. *

* The value of this parameter is also ignored if any layers are defined. *

*/ @Parameter(property = PropertyNames.SERVER_CONFIG, alias = "server-config", defaultValue = STANDALONE_XML) String serverConfig; /** * Specifies the name used for the deployment. *

* When the deployment is copied to the server, it is renamed with this name. *

*/ @Parameter(property = PropertyNames.DEPLOYMENT_NAME) private String name; /** * The runtime name for the deployment. *

* When the deployment is copied to the server, it is renamed with the {@code runtime-name}. * If both {@code name} and {@code runtime-name} are specified, {@code runtime-name} is used. *

* * @deprecated use the {@code name} property instead to change the name of the deployment. */ @Deprecated(since = "4.1.O") @Parameter(property = PropertyNames.DEPLOYMENT_RUNTIME_NAME, alias = "runtime-name") protected String runtimeName; /** * Indicates how {@code stdout} and {@code stderr} should be handled for the * spawned CLI processes. Note that {@code stderr} will be redirected to * {@code stdout} if the value is defined unless the value is {@code none}. *
* By default {@code stdout} and {@code stderr} are inherited from the * current process. You can change the setting to one of the follow: *
    *
  • {@code none} indicates the {@code stdout} and {@code stderr} stream * should not be consumed
  • *
  • {@code System.out} or {@code System.err} to redirect to the current * processes (use this option if you see odd behavior from maven with * the default value)
  • *
  • Any other value is assumed to be the path to a file and the * {@code stdout} and {@code stderr} will be written there
  • *
*
*/ @Parameter(name = "stdout", defaultValue = "System.out", property = PropertyNames.STDOUT) private String stdout; /** * Set to {@code true} if you want the goal to be skipped, otherwise * {@code false}. */ @Parameter(defaultValue = "false", property = PropertyNames.SKIP_PACKAGE) private boolean skip; /** * Skip deploying the deployment after the server is provisioned ({@code false} by default). */ @Parameter(defaultValue = "false", property = PropertyNames.SKIP_PACKAGE_DEPLOYMENT) protected boolean skipDeployment; /** * Galleon provisioning info discovery. *

* By enabling this feature, the set of Galleon feature-packs * and layers are automatically discovered by scanning the deployed application. * You can configure the following items: *

*
*
    *
  • addOns: List of addOn to enable. An addOn brings extra galleon layers to the provisioning (eg: {@code wildfly-cli} to * include CLI.
  • *
  • context: {@code bare-metal} or (@code cloud}. Default to {@code bare-metal}
  • *
  • failsOnError: true|false. If errors are detected (missing datasource, missing messaging broker, ambiguous JNDI call, * provisioning is aborted. Default to {@code false}
  • *
  • layersForJndi: List of Galleon layers required by some JNDI calls located in your application.
  • *
  • preview: {@code true} | {@code false}. Use preview feature-packs. Default to {@code false}.
  • *
  • profile: {@code ha}. Default being non ha server configuration.
  • *
  • suggest: {@code true} | {@code false}. Display addOns that you can use to enhance discovered provisioning * configuration. Default to {@code false}.
  • *
  • excludedArchives: List of archives contained in the deployment to exclude when scanning. * Wildcards ({@code *}) are allowed. N.B. Just the name of the archive is matched, do not attempt * to specify a full path within the jar. The following examples would be valid exclusions: {@code my-jar.jar}, * {@code *-internal.rar}.
  • *
  • verbose: {@code true} | {@code false}. Display more information. The set of rules that selected Galleon layers are * printed. Default to {@code false}.
  • *
  • version: server version. Default being the latest released version.
  • * *
*
* * For example, cloud, ha profile with CLI and openapi addOns enabled. mail layer being explicitly included: * *
     *   <discover-provisioning-info>
     *     <context>cloud</context>
     *     <profile>ha</profile>
     *     <addOns>
     *       <addOn>wildfly-cli</addOn>
     *       <addOn>openapi</addOn>
     *     </addOns>
     *     <layersForJndi>
     *       <layer>mail</layer>
     *     </layersForJndi>
     *   </discover-provisioning-info>
     * 
* * @since 5.0 */ @Parameter(alias = "discover-provisioning-info") private GlowConfig discoverProvisioningInfo; /** * Package the provisioned server into a WildFly Bootable JAR. *

* Note that the produced fat JAR is ignored when running the {@code dev},{@code image},{@code start} or {@code run} goals. *

*/ @Parameter(alias = "bootable-jar", required = false, property = PropertyNames.BOOTABLE_JAR) private boolean bootableJar; /** * When {@code bootable-jar} is set to true, use this parameter to name the generated jar file. * The jar file is named by default {@code server-bootable.jar}. */ @Parameter(alias = "bootable-jar-name", required = false, property = PropertyNames.BOOTABLE_JAR_NAME) private String bootableJarName; /** * When {@code bootable-jar} is set to true, the bootable JAR artifact is attached to the project with the classifier * 'bootable'. Use this parameter to * configure the classifier. */ @Parameter(alias = "bootable-jar-install-artifact-classifier", property = PropertyNames.BOOTABLE_JAR_INSTALL_CLASSIFIER, defaultValue = BootableJarSupport.BOOTABLE_SUFFIX) private String bootableJarInstallArtifactClassifier; @Inject private OfflineCommandExecutor commandExecutor; private GalleonProvisioningConfig config; @Override protected GalleonProvisioningConfig getDefaultConfig() throws ProvisioningException { return null; } @Override protected GalleonProvisioningConfig buildGalleonConfig(GalleonBuilder pm) throws MojoExecutionException, ProvisioningException { if (discoverProvisioningInfo == null) { config = super.buildGalleonConfig(pm); return config; } try { try (ScanResults results = Utils.scanDeployment(discoverProvisioningInfo, layers, excludedLayers, featurePacks, dryRun, getLog(), getDeploymentContent(), artifactResolver, Paths.get(project.getBuild().getDirectory()), pm, galleonOptions, layersConfigurationFileName)) { config = results.getProvisioningConfig(); return config; } } catch (Exception ex) { throw new MojoExecutionException(ex.getLocalizedMessage(), ex); } } @Override protected String getGoal() { return "package"; } @Override public void execute() throws MojoExecutionException, MojoFailureException { if (skip) { getLog().debug(String.format("Skipping " + getGoal() + " of %s:%s", project.getGroupId(), project.getArtifactId())); return; } super.execute(); } @Override protected void serverProvisioned(Path jbossHome) throws MojoExecutionException, MojoFailureException { try { if (StandardOutput.isFile(stdout)) { // Delete it, we are appending to it. Files.deleteIfExists(Paths.get(stdout)); } if (!extraServerContentDirs.isEmpty()) { getLog().info("Copying extra content to server"); copyExtraContent(jbossHome); } } catch (IOException ex) { throw new MojoExecutionException(ex.getLocalizedMessage(), ex); } if (!skipDeployment) { final Path deploymentContent = getDeploymentContent(); if (Files.exists(deploymentContent)) { Path standaloneDeploymentDir = Path.of(provisioningDir, "standalone", "deployments"); if (!standaloneDeploymentDir.isAbsolute()) { standaloneDeploymentDir = Path.of(project.getBuild().getDirectory()).resolve(standaloneDeploymentDir); } try { Path deploymentTarget = standaloneDeploymentDir.resolve(getDeploymentTargetName()); getLog().info("Copy deployment " + deploymentContent + " to " + deploymentTarget); Files.copy(deploymentContent, deploymentTarget, StandardCopyOption.REPLACE_EXISTING); } catch (IOException e) { throw new MojoDeploymentException("Could not copy deployment in provisioned server", e); } } } // CLI execution try { if (!packagingScripts.isEmpty()) { getLog().info("Excuting CLI commands and scripts"); for (CliSession session : packagingScripts) { List wrappedScripts = wrapOfflineScripts(session.getScripts()); try { final BaseCommandConfiguration cmdConfig = new BaseCommandConfiguration.Builder() .addCommands(wrapOfflineCommands(session.getCommands())) .addScripts(wrappedScripts) .addCLIArguments(CLI_ECHO_COMMAND_ARG) .setJBossHome(jbossHome) .setAppend(true) .setStdout(stdout) .addPropertiesFiles(resolveFiles(session.getPropertiesFiles())) .addJvmOptions(session.getJavaOpts()) .setResolveExpression(session.getResolveExpression()) .build(); commandExecutor.execute(cmdConfig, artifactResolver); } finally { for (File f : wrappedScripts) { Files.delete(f.toPath()); } } } } cleanupServer(jbossHome); if (bootableJar) { packageBootableJar(jbossHome, config); } } catch (Exception ex) { throw new MojoExecutionException(ex.getLocalizedMessage(), ex); } } private void attachJar(Path jarFile) { if (getLog().isDebugEnabled()) { getLog().debug("Attaching bootable jar " + jarFile + " as a project artifact with classifier " + bootableJarInstallArtifactClassifier); } projectHelper.attachArtifact(project, JAR, bootableJarInstallArtifactClassifier, jarFile.toFile()); } private void packageBootableJar(Path jbossHome, GalleonProvisioningConfig activeConfig) throws Exception { String jarName = bootableJarName == null ? BOOTABLE_JAR_NAME_RADICAL + BootableJarSupport.BOOTABLE_SUFFIX + "." + JAR : bootableJarName; Path targetPath = Paths.get(project.getBuild().getDirectory()); Path targetJarFile = targetPath.toAbsolutePath() .resolve(jarName); Files.deleteIfExists(targetJarFile); BootableJarSupport.packageBootableJar(targetJarFile, targetPath, activeConfig, jbossHome, artifactResolver, new MvnMessageWriter(getLog())); attachJar(targetJarFile); getLog().info("Bootable JAR packaging DONE. To run the server: java -jar " + targetJarFile); } /** * Return the file name of the deployment to put in the server deployment directory * * @throws MojoExecutionException */ protected String getDeploymentTargetName() throws MojoExecutionException { String targetName; if (runtimeName != null) { targetName = runtimeName; } else { targetName = name != null ? name : getDeploymentContent().getFileName().toString(); } return targetName; } private List resolveFiles(List files) { if (files == null || files.isEmpty()) { return files; } List resolvedFiles = new ArrayList<>(); for (File f : files) { resolvedFiles.add(resolvePath(project, f.toPath()).toFile()); } return resolvedFiles; } private List wrapOfflineCommands(List commands) { if (commands == null || commands.isEmpty()) { return commands; } List offlineCommands = new ArrayList<>(); String serverConfigName = serverConfig; if (!layersConfigurationFileName.equals(STANDALONE_XML)) { serverConfigName = layersConfigurationFileName; } offlineCommands.add("embed-server --server-config=" + serverConfigName); offlineCommands.addAll(commands); offlineCommands.add("stop-embedded-server"); return offlineCommands; } private List wrapOfflineScripts(List scripts) throws IOException, MojoExecutionException { List wrappedScripts = new ArrayList<>(); for (File script : scripts) { if (script == null) { continue; } wrappedScripts.add(wrapScript(script).toFile()); } return wrappedScripts; } private Path wrapScript(File script) throws IOException, MojoExecutionException { final Path tempScript = Files.createTempFile("offline-cli-script", ".cli"); Path resolvedScript = resolvePath(project, script.toPath()); if (!Files.exists(resolvedScript)) { throw new MojoExecutionException("CLI script " + resolvedScript + " doesn't exist"); } List cmds = Files.readAllLines(resolvedScript, StandardCharsets.UTF_8); List wrappedCommands = wrapOfflineCommands(cmds); Files.write(tempScript, wrappedCommands, StandardCharsets.UTF_8); return tempScript; } public void copyExtraContent(Path target) throws MojoExecutionException, IOException { for (String path : extraServerContentDirs) { Path extraContent = Paths.get(path); extraContent = resolvePath(project, extraContent); if (Files.notExists(extraContent)) { throw new MojoExecutionException("Extra content dir " + extraContent + " doesn't exist"); } if (!Files.isDirectory(extraContent)) { throw new MojoExecutionException("Extra content dir " + extraContent + " is not a directory"); } // Check for the presence of a standalone.xml file warnExtraConfig(extraContent); IoUtils.copy(extraContent, target); } } private void warnExtraConfig(Path extraContentDir) { Path config = extraContentDir.resolve(STANDALONE).resolve("configurations").resolve(STANDALONE_XML); if (Files.exists(config)) { getLog().warn("The file " + config + " overrides the Galleon generated configuration, " + "un-expected behavior can occur when starting the server"); } } protected Path getDeploymentContent() throws MojoExecutionException { final PackageType packageType = PackageType.resolve(project); final String filename; if (this.filename == null) { filename = String.format("%s.%s", project.getBuild().getFinalName(), packageType.getFileExtension()); } else { filename = this.filename; } Path deployment = Paths.get(project.getBuild().getDirectory()).resolve(filename); if (Files.notExists(deployment)) { if (this.filename != null) { throw new MojoExecutionException("No deployment found with name " + this.filename); } if (this.runtimeName != null) { throw new MojoExecutionException("No deployment found with name " + filename + ". A runtime-name has been set that indicates that a deployment is expected. " + "A custom file name can be set with the parameter."); } getLog().warn("The project doesn't define a deployment artifact to deploy to the server."); } return deployment; } private static void cleanupServer(Path jbossHome) throws IOException { Path history = jbossHome.resolve("standalone").resolve("configuration").resolve("standalone_xml_history"); IoUtils.recursiveDelete(history); Path tmp = jbossHome.resolve("standalone").resolve("tmp"); IoUtils.recursiveDelete(tmp); Path log = jbossHome.resolve("standalone").resolve("log"); IoUtils.recursiveDelete(log); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy