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

com.lazerycode.jmeter.testrunner.TestManager Maven / Gradle / Ivy

package com.lazerycode.jmeter.testrunner;

import com.lazerycode.jmeter.configuration.JMeterArgumentsArray;
import com.lazerycode.jmeter.configuration.JMeterProcessJVMSettings;
import com.lazerycode.jmeter.configuration.RemoteArgumentsArrayBuilder;
import com.lazerycode.jmeter.configuration.RemoteConfiguration;
import com.lazerycode.jmeter.perfana.PerfanaClient;
import com.lazerycode.jmeter.utility.UtilityFunctions;
import org.apache.commons.io.FilenameUtils;
import org.apache.maven.plugin.MojoExecutionException;
import org.apache.tools.ant.DirectoryScanner;
import org.joda.time.DateTime;
import org.joda.time.format.DateTimeFormatter;
import org.joda.time.format.DateTimeFormatterBuilder;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.BufferedReader;
import java.io.File;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;
import java.util.concurrent.TimeUnit;

/**
 * TestManager encapsulates functions that gather JMeter Test files and execute the tests
 */
public class TestManager {

	private static final Logger LOGGER = LoggerFactory.getLogger(JMeterProcessBuilder.class);
	
	private static final String REPORT_DIR_DATE_FORMAT = "yyyyMMdd_HHmmss";
	private final JMeterArgumentsArray baseTestArgs;
	private final File binDir;
	private final File testFilesDirectory;
	private final String[] testFilesIncluded;
	private final String[] testFilesExcluded;
	private final boolean suppressJMeterOutput;
	private final RemoteConfiguration remoteServerConfiguration;
	private final JMeterProcessJVMSettings jMeterProcessJVMSettings;
	private long postTestPauseInSeconds;
	private final String runtimeJarName;
    private File reportDirectory;
    private boolean generateReports;
    private PerfanaClient perfanaClient;

	public TestManager(JMeterArgumentsArray baseTestArgs, File testFilesDirectory, List testFilesIncluded, List testFilesExcluded, RemoteConfiguration remoteServerConfiguration, boolean suppressJMeterOutput, File binDir, JMeterProcessJVMSettings jMeterProcessJVMSettings, String runtimeJarName,
                       File reportDirectory, boolean generateReports, PerfanaClient perfanaClient) {
		this.binDir = binDir;
		this.baseTestArgs = baseTestArgs;
		this.testFilesDirectory = testFilesDirectory;
		this.remoteServerConfiguration = remoteServerConfiguration;
		this.suppressJMeterOutput = suppressJMeterOutput;
		this.jMeterProcessJVMSettings = jMeterProcessJVMSettings;
		this.runtimeJarName = runtimeJarName;
		this.testFilesExcluded = testFilesExcluded.toArray(new String[0]);
		this.reportDirectory = reportDirectory;
		this.generateReports = generateReports;
		if (!testFilesIncluded.isEmpty()) {
			this.testFilesIncluded = testFilesIncluded.toArray(new String[0]);
		} else {
			this.testFilesIncluded = new String[]{"**/*.jmx"};
		}
		this.perfanaClient = perfanaClient;
	}

    /**
     * Sets a pause after each test has been executed.
     *
     * @param postTestPauseInSeconds Number of seconds to pause after a test has completed
     */
    public void setPostTestPauseInSeconds(String postTestPauseInSeconds) {
        Long testPause = null;
        try {
            testPause = Long.parseLong(postTestPauseInSeconds);
        } catch(NumberFormatException ex) {
            LOGGER.error("Error parsing value {} of , will default to 0", 
                    postTestPauseInSeconds, ex);
        }
        if (null == testPause) {
            testPause = 0l;
        }

        this.postTestPauseInSeconds = testPause;
    }

	/**
	 * Executes all tests and returns the resultFile names
	 *
	 * @return the list of resultFile names
	 * @throws MojoExecutionException
	 */
	public List executeTests() throws MojoExecutionException {

		if (perfanaClient != null) {
		    perfanaClient.startSession();
        }

        JMeterArgumentsArray thisTestArgs = baseTestArgs;
		List tests = generateTestList();
		List results = new ArrayList<>();
		DateTimeFormatter sdf = new DateTimeFormatterBuilder().appendPattern(REPORT_DIR_DATE_FORMAT).toFormatter();
		for (String file : tests) {
		    if(generateReports) {
		        thisTestArgs.setReportsDirectory(
		                reportDirectory+File.separator+ 
		            FilenameUtils.getBaseName(file)+"_"+
		            sdf.print(new DateTime()));
		    }
			if (remoteServerConfiguration != null) {
				if ((remoteServerConfiguration.isStartServersBeforeTests() && tests.get(0).equals(file)) || remoteServerConfiguration.isStartAndStopServersForEachTest()) {
					thisTestArgs.setRemoteStart();
					thisTestArgs.setRemoteStartServerList(remoteServerConfiguration.getServerList());
				}
				if ((remoteServerConfiguration.isStopServersAfterTests() && tests.get(tests.size() - 1).equals(file)) || remoteServerConfiguration.isStartAndStopServersForEachTest()) {
					thisTestArgs.setRemoteStop();
				}
			}
			results.add(executeSingleTest(new File(testFilesDirectory, file), thisTestArgs));
			try {
				TimeUnit.SECONDS.sleep(postTestPauseInSeconds);
			} catch (InterruptedException ignored) {
			    Thread.currentThread().interrupt();
			}
		}

		if (perfanaClient != null) {
            try {
                perfanaClient.stopSession();
            } catch (PerfanaClient.PerfanaClientException e) {
                throw new MojoExecutionException("Perfana assertions check failed.", e);
            }
        }

		return results;
	}

	//=============================================================================================

	/**
	 * Executes a single JMeter test by building up a list of command line
	 * parameters to pass to JMeter.start().
	 *
	 * @param test JMeter test XML
	 * @return the report file names.
	 * @throws org.apache.maven.plugin.MojoExecutionException Exception
	 */
	@SuppressWarnings("ResultOfMethodCallIgnored")
	private String executeSingleTest(File test, JMeterArgumentsArray testArgs) throws MojoExecutionException {
		LOGGER.info(" ");
		testArgs.setTestFile(test, testFilesDirectory);
		//Delete results file if it already exists
		File currentResultsFile = new File(testArgs.getResultsLogFileName());
		if(currentResultsFile.exists() && !currentResultsFile.delete()) {
		    throw new MojoExecutionException("Failed to delete existing results file:"+currentResultsFile.getAbsolutePath());
		}
		List argumentsArray = testArgs.buildArgumentsArray();
		argumentsArray.addAll(buildRemoteArgs(remoteServerConfiguration));
		if(LOGGER.isDebugEnabled()) {
		    LOGGER.debug("JMeter is called with the following command line arguments: {}",
		        UtilityFunctions.humanReadableCommandLineOutput(argumentsArray));
		}
		LOGGER.info("Executing test: {}", test.getName());
		//Start the test.
		JMeterProcessBuilder jmeterProcessBuilder = 
		        new JMeterProcessBuilder(jMeterProcessJVMSettings, runtimeJarName);
		jmeterProcessBuilder.setWorkingDirectory(binDir);
		jmeterProcessBuilder.addArguments(argumentsArray);
		try {
			final Process process = jmeterProcessBuilder.startProcess();

			Runtime.getRuntime().addShutdownHook(new Thread() {
				@Override
				public void run() {
					LOGGER.info("Shutdown detected, destroying JMeter process...");
					process.destroy();
				}
			});

			try (InputStreamReader isr = new InputStreamReader(process.getInputStream());
			        BufferedReader br = new BufferedReader(isr)) {
    			String line;
    			while ((line = br.readLine()) != null) {
    				if (suppressJMeterOutput) {
    					LOGGER.debug(line);
    				} else {
    					LOGGER.info(line);
    				}
    			}
    			int jMeterExitCode = process.waitFor();
    			if (jMeterExitCode != 0) {
    				throw new MojoExecutionException("Test failed with exit code:"+jMeterExitCode);
    			}
    			LOGGER.info("Completed Test: {}", test.getAbsolutePath());
			}
		} catch (InterruptedException ex) {
			LOGGER.info(" ");
			LOGGER.info("System Exit Detected!  Stopping Test...");
			LOGGER.info(" ");
            Thread.currentThread().interrupt();
		} catch (IOException e) {
			LOGGER.error(e.getMessage());
		}
		return testArgs.getResultsLogFileName();
	}

	private List buildRemoteArgs(RemoteConfiguration remoteConfig) {
		if (remoteConfig == null) {
			return Collections.emptyList();
		}
		return new RemoteArgumentsArrayBuilder().buildRemoteArgumentsArray(remoteConfig.getPropertiesMap());
	}

	/**
	 * Scan Project directories for JMeter Test Files according to includes and excludes
	 *
	 * @return found JMeter tests
	 */
	private List generateTestList() {
		DirectoryScanner scanner = new DirectoryScanner();
		scanner.setBasedir(this.testFilesDirectory);
		scanner.setIncludes(this.testFilesIncluded);
		scanner.setExcludes(this.testFilesExcluded);
		scanner.scan();

		return Arrays.asList(scanner.getIncludedFiles());
	}
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy