org.wildfly.plugin.provision.PackageServerMojo Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of wildfly-maven-plugin Show documentation
Show all versions of wildfly-maven-plugin Show documentation
A maven plugin that allows various management operations to be executed on WildFly Application
Server.
/*
* Copyright 2020 Red Hat, Inc. and/or its affiliates
* and other contributors as indicated by the @author tags.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
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.ProvisioningDescriptionException;
import org.jboss.galleon.config.ProvisioningConfig;
import org.jboss.galleon.util.IoUtils;
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.deployment.MojoDeploymentException;
import org.wildfly.plugin.deployment.PackageType;
/**
* 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 {
/**
* 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}.
*/
@Parameter(property = PropertyNames.SERVER_CONFIG, alias = "server-config", defaultValue = STANDALONE_XML)
private 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;
@Inject
private OfflineCommandExecutor commandExecutor;
@Override
protected ProvisioningConfig getDefaultConfig() throws ProvisioningDescriptionException {
return null;
}
@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 = Paths
.get(project.getBuild().getDirectory(), provisioningDir, "standalone", "deployments").normalize();
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);
} catch (IOException ex) {
throw new MojoExecutionException(ex.getLocalizedMessage(), ex);
}
}
/**
* 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);
}
}