org.wildfly.plugin.cli.ExecuteCommandsMojo Maven / Gradle / Ivy
/*
* Copyright The WildFly Authors
* SPDX-License-Identifier: Apache-2.0
*/
package org.wildfly.plugin.cli;
import java.io.File;
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;
import javax.inject.Inject;
import org.apache.maven.execution.MavenSession;
import org.apache.maven.plugin.MojoExecutionException;
import org.apache.maven.plugin.MojoFailureException;
import org.apache.maven.plugin.logging.Log;
import org.apache.maven.plugins.annotations.Component;
import org.apache.maven.plugins.annotations.Mojo;
import org.apache.maven.plugins.annotations.Parameter;
import org.apache.maven.project.MavenProject;
import org.eclipse.aether.RepositorySystem;
import org.eclipse.aether.RepositorySystemSession;
import org.eclipse.aether.repository.RemoteRepository;
import org.jboss.as.controller.client.ModelControllerClient;
import org.jboss.as.controller.client.helpers.ClientConstants;
import org.jboss.galleon.maven.plugin.util.MavenArtifactRepositoryManager;
import org.jboss.galleon.universe.maven.repo.MavenRepoManager;
import org.wildfly.plugin.common.AbstractServerConnection;
import org.wildfly.plugin.common.PropertyNames;
import org.wildfly.plugin.common.Utils;
import org.wildfly.plugin.core.MavenRepositoriesEnricher;
import org.wildfly.plugin.tools.server.ServerManager;
/**
* Execute commands to the running WildFly Application Server.
*
* Commands should be formatted in the same manner CLI commands are formatted.
*
* Executing commands in a batch will rollback all changes if one command fails.
*
*
* <batch>true</batch>
* <fail-on-error>false</fail-on-error>
* <commands>
* <command>/subsystem=logging/console=CONSOLE:write-attribute(name=level,value=DEBUG)</command>
* </commands>
*
*
* @author James R. Perkins
*/
@Mojo(name = "execute-commands", threadSafe = true)
public class ExecuteCommandsMojo extends AbstractServerConnection {
/**
* {@code true} if commands execution should be skipped.
*/
@Parameter(defaultValue = "false", property = PropertyNames.SKIP)
private boolean skip;
/**
* {@code true} if commands should be executed in a batch or {@code false} if they should be executed one at a
* time.
*/
@Parameter(defaultValue = "false", property = PropertyNames.BATCH)
private boolean batch;
/**
* The WildFly Application Server's home directory.
*
* This parameter is required when {@code offline} is set to {@code true}. Otherwise this is not required, but
* should be used for commands such as {@code module add} as they are executed on the local file system.
*
*/
@Parameter(alias = "jboss-home", property = PropertyNames.JBOSS_HOME)
private String jbossHome;
/**
* The system properties to be set when executing CLI commands.
*/
@Parameter(alias = "system-properties")
private Map systemProperties;
/**
* The properties files to use when executing CLI scripts or commands.
*/
@Parameter
private List propertiesFiles = new ArrayList<>();
/**
* The CLI commands to execute.
*/
@Parameter(property = PropertyNames.COMMANDS)
private List commands = new ArrayList<>();
/**
* The CLI script files to execute.
*/
@Parameter(property = PropertyNames.SCRIPTS)
private List scripts = new ArrayList<>();
/**
* Indicates whether or not subsequent commands should be executed if an error occurs executing a command. A value of
* {@code false} will continue processing commands even if a previous command execution results in a failure.
*
* Note that this value is ignored if {@code offline} is set to {@code true}.
*
*/
@Parameter(alias = "fail-on-error", defaultValue = "true", property = PropertyNames.FAIL_ON_ERROR)
private boolean failOnError = true;
/**
* Indicates the commands should be run in a new process. If the {@code jboss-home} property is not set an attempt
* will be made to download a version of WildFly to execute commands on. However it's generally considered best
* practice to set the {@code jboss-home} property if setting this value to {@code true}.
*
* Note that if {@code offline} is set to {@code true} this setting really has no effect.
*
*
* WARNING: In 3.0.0 you'll be required to set the {@code jboss-home}. An error will occur if
* this option is {@code true} and the {@code jboss-home} is not set.
*
*
* @since 2.0.0
*/
@Parameter(defaultValue = "false", property = "wildfly.fork")
private boolean fork;
/**
* Indicates whether or not CLI scrips or commands should be executed in an offline mode. This is useful for using
* an embedded server or host controller.
*
*
* This does not start an embedded server it instead skips checking if a server is running.
*
*/
@Parameter(name = "offline", defaultValue = "false", property = PropertyNames.OFFLINE)
private boolean offline = false;
/**
* Indicates how {@code stdout} and {@code stderr} should be handled for the spawned CLI process. Currently a new
* process is only spawned if {@code offline} is set to {@code true} or {@code fork} is set to {@code true}. 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;
/**
* The JVM options to pass to the offline process if the {@code offline} configuration parameter is set to
* {@code true}.
*/
@Parameter(alias = "java-opts", property = PropertyNames.JAVA_OPTS)
private String[] javaOpts;
/**
* Automatically reloads the server if the commands leave the server in the "reload-required" state. Note a reload
* will not be done if {@code offline} is set to {@code true}.
*
* @since 4.2.1
*/
@Parameter(alias = "auto-reload", defaultValue = "true", property = PropertyNames.AUTO_RELOAD)
private boolean autoReload;
@Component
RepositorySystem repoSystem;
@Parameter(defaultValue = "${repositorySystemSession}", readonly = true, required = true)
private RepositorySystemSession session;
@Parameter(defaultValue = "${project.remoteProjectRepositories}", readonly = true, required = true)
private List repositories;
@Parameter(defaultValue = "${project}", readonly = true, required = true)
MavenProject project;
@Parameter(defaultValue = "${session}", readonly = true, required = true)
MavenSession mavenSession;
@Parameter(defaultValue = "${project.build.directory}", readonly = true, required = true)
private File buildDir;
/**
* Resolve expressions prior to send the commands to the server.
*
* @since 3.0
*/
@Parameter(alias = "resolve-expressions", defaultValue = "false", property = PropertyNames.RESOLVE_EXPRESSIONS)
private boolean resolveExpressions;
@Inject
private CommandExecutor commandExecutor;
private MavenRepoManager mavenRepoManager;
@Override
public String goal() {
return "execute-commands";
}
@Override
public void execute() throws MojoExecutionException, MojoFailureException {
if (skip) {
getLog().debug("Skipping commands execution");
return;
}
MavenRepositoriesEnricher.enrich(mavenSession, project, repositories);
mavenRepoManager = new MavenArtifactRepositoryManager(repoSystem, session, repositories);
final CommandConfiguration.Builder cmdConfigBuilder = CommandConfiguration
.of(this::createClient, this::getClientConfiguration)
.addCommands(commands)
.addJvmOptions(javaOpts)
.addPropertiesFiles(propertiesFiles)
.addScripts(scripts)
.addSystemProperties(systemProperties)
.setBatch(batch)
.setFailOnError(failOnError)
.setFork(fork)
.setJBossHome(jbossHome)
.setOffline(offline)
.setAutoReload(autoReload)
.setStdout(stdout)
.setTimeout(timeout)
.setResolveExpression(resolveExpressions);
// Why is that? fork implies a jboss-home?
if (fork) {
cmdConfigBuilder.setJBossHome(getInstallation(buildDir.toPath().resolve(Utils.WILDFLY_DEFAULT_DIR)));
}
commandExecutor.execute(cmdConfigBuilder.build(), mavenRepoManager);
// Check the server state if we're not in offline mode
if (!offline) {
try (ModelControllerClient client = createClient()) {
final String serverState = ServerManager.builder().client(client).build().get(timeout, TimeUnit.SECONDS)
.serverState();
if (!ClientConstants.CONTROLLER_PROCESS_STATE_RUNNING.equals(serverState)) {
getLog().warn(String.format(
"The server may be in an unexpected state for further interaction. The current state is %s",
serverState));
}
} catch (IOException | InterruptedException | ExecutionException | TimeoutException e) {
final Log log = getLog();
log.warn(String.format(
"Failed to determine the server-state. The server may be in an unexpected state. Failure: %s",
e.getMessage()));
if (log.isDebugEnabled()) {
log.debug(e);
}
}
}
}
/**
* Allows the {@link #javaOpts} to be set as a string. The string is assumed to be space delimited.
*
* @param value a spaced delimited value of JVM options
*/
@SuppressWarnings("unused")
public void setJavaOpts(final String value) {
if (value != null) {
javaOpts = value.split("\\s+");
}
}
private Path getInstallation(Path installDir) throws MojoFailureException {
if (jbossHome != null) {
return Paths.get(jbossHome);
}
if (!Files.exists(installDir)) {
throw new MojoFailureException("No server installed.");
}
return installDir;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy