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