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

com.seleniumtests.core.utils.TestNGResultUtils Maven / Gradle / Ivy

/**
 * Orignal work: Copyright 2015 www.seleniumtests.com
 * Modified work: Copyright 2016 www.infotel.com
 * 				Copyright 2017-2019 B.Hecquet
 *
 * 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 com.seleniumtests.core.utils;

import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;

import org.apache.commons.lang.ObjectUtils;
import org.apache.logging.log4j.Logger;
import org.testng.ITestNGMethod;
import org.testng.ITestResult;
import org.testng.Reporter;
import org.testng.annotations.CustomAttribute;
import org.testng.annotations.Test;
import org.testng.internal.BaseTestMethod;
import org.testng.internal.TestResult;

import com.seleniumtests.connectors.selenium.SeleniumRobotSnapshotServerConnector;
import com.seleniumtests.connectors.selenium.SeleniumRobotSnapshotServerConnector.SnapshotComparisonResult;
import com.seleniumtests.core.SeleniumTestsContext;
import com.seleniumtests.core.SeleniumTestsContextManager;
import com.seleniumtests.core.TestVariable;
import com.seleniumtests.core.runner.CucumberScenarioWrapper;
import com.seleniumtests.core.testanalysis.ErrorCause;
import com.seleniumtests.customexception.ScenarioException;
import com.seleniumtests.customexception.SeleniumRobotServerException;
import com.seleniumtests.driver.screenshots.SnapshotComparisonBehaviour;
import com.seleniumtests.reporter.info.Info;
import com.seleniumtests.reporter.logger.Snapshot;
import com.seleniumtests.reporter.logger.TestStep;
import com.seleniumtests.reporter.reporters.CommonReporter;
import com.seleniumtests.util.ExceptionUtility;
import com.seleniumtests.util.StringUtility;
import com.seleniumtests.util.logging.SeleniumRobotLogger;

public class TestNGResultUtils {

	protected static Logger logger = SeleniumRobotLogger.getLogger(TestNGResultUtils.class);

	private static final String UNIQUE_METHOD_NAME = "uniqueMethodName"; // unique name of the test (in case several tests have the same name)
	private static final String VISUAL_TEST_NAME = "visualTestName"; 	// Name that will be displayed in reports for user (HTML and JUnit report)
	private static final String TEST_CONTEXT = "testContext";
	private static final String LINKED_TEST_METHOD = "linkedTestMethod"; // the test method associated to this configuration method. Obviously, this is null for all test methods 
	private static final String RETRY = "retry";						// index of the retry
	private static final String NO_MORE_RETRY = "noMoreRetry";			// set to true when is not going to be retried
	private static final String TEST_INFO = "testInfo";
	private static final String SELENIUM_SERVER_REPORT = "seleniumServerReport";// true if the result has already been recorded to the seleniumRobot server
	private static final String SELENIUM_SERVER_REPORT_TEST_CASE_SESSION_ID = "seleniumServerReportTcsId"; // ID of the TestCaseInSession when snapshot comparison has been done
	private static final String HTML_REPORT = "htmlReport";				// true if the HTML result has already been generated
	private static final String TEST_MANAGER_REPORT = "testManagerReport";// true if the result has already been recorded on test manager
	private static final String BUGTRACKER_REPORT = "bugtrackerReport";// true if the failure has already been recorded on bugtracker
	private static final String CUSTOM_REPORT = "customReport";			// true if the custom result has already been generated
	private static final String METHOD_NAME = "methodName";				// name of the test method (or the cucumber scenario)
	private static final String SNAPSHOT_COMPARISON_RESULT = "snapshotComparisonResult";	// the result of snapshot comparison, when enabled
	private static final String DESCRIPTION = "description";			// description of the test method, if any
	private static final String ERROR_CAUSES = "errorCauses"; 			// list of causes of the test error
	private static final String ERROR_CAUSE_IN_LAST_STEP = "errorCauseInLastStep"; // true when we have searched for error cause in the last step
	private static final String ERROR_CAUSE_IN_REFERENCE = "errorCauseInReference"; // true when we have searched for error cause by comparing reference picture of the failed step
	private static final String FINISHED = "finished"; // true when all after methods has been executed

	private TestNGResultUtils() {
		// nothing to do
	}
	
	/**
	 * Copy a TestResult into an other one
	 * @param toCopy
	 * @return
	 * @throws SecurityException 
	 * @throws NoSuchFieldException 
	 * @throws IllegalAccessException 
	 * @throws IllegalArgumentException 
	 */
	public static ITestResult copy(ITestResult toCopy, String name, String description) throws NoSuchFieldException, IllegalAccessException {
		
		
		Field methodNameField = BaseTestMethod.class.getDeclaredField("m_methodName");
		methodNameField.setAccessible(true);
		ITestNGMethod newMethod = toCopy.getMethod().clone();
		methodNameField.set(newMethod, name);	
		newMethod.setDescription(description);

		ITestResult newTestResult = TestResult.newTestResultFrom((TestResult) toCopy, newMethod, toCopy.getTestContext(), toCopy.getStartMillis());
		newTestResult.setEndMillis(toCopy.getEndMillis());
		newTestResult.setThrowable(toCopy.getThrowable());

		newTestResult.setParameters(toCopy.getParameters());
		for (String attributeName: toCopy.getAttributeNames()) {
			newTestResult.setAttribute(attributeName, toCopy.getAttribute(attributeName));
		}
		
		// reset flags of generated results
		setHtmlReportCreated(newTestResult, false);
		setCustomReportCreated(newTestResult, false);
		setSnapshotComparisonResult(newTestResult, ITestResult.SKIP);
		
		// create testResult own context
		SeleniumTestsContext newTestContext = new SeleniumTestsContext(getSeleniumRobotTestContext(toCopy), false);
		newTestContext.createTestSpecificOutputDirectory(newTestResult);
		setSeleniumRobotTestContext(newTestResult, newTestContext);
		
		// change name
		setTestMethodName(newTestResult, getTestName(newTestResult));
		setUniqueTestName(newTestResult, newTestContext.getRelativeOutputDir());
		
		return newTestResult;
	}
	
	/**
	 * Returns the test name (will be method name for SeleniumTestPlan test and scenario name for cucumber test)
	 * @param testNGResult
	 * @return
	 */
	public static String getTestName(ITestResult testNGResult) {
		if (testNGResult == null) {
			return null;
		}
		
    	if (testNGResult.getParameters().length > 0 
    			&& testNGResult.getParameters()[0] instanceof CucumberScenarioWrapper 
//    			&& "com.seleniumtests.core.runner.CucumberTestPlan".equals(testNGResult.getMethod().getTestClass().getName()) // prevents from doing unit tests
    			) {
			return testNGResult.getParameters()[0].toString();
    		
    	} else {
    		// issue #137: in case we are in a BeforeMethod, take class name and method name from TestMethod
    		if (testNGResult.getMethod().isBeforeMethodConfiguration()) {
    			Method testMethod = (Method)(testNGResult.getParameters()[0]);
    			return "before-" + testMethod.getName();
    		} else {
    			return testNGResult.getMethod().getMethodName();
    		}
    	}
	}
	
	/**
	 * Returns a "visual" name for this test.
	 * In case "testName" is not specified for @Test annotation, then, we return the "unique test name"
	 * If "testName" is specified, then, we return it, possibly, interpolating string
	 * @param testNGResult
	 * @return
	 */
	public static String getVisualTestName(ITestResult testNGResult) {
		if (testNGResult == null) {
			return null;
		}
		
		Test testAnnotation = testNGResult.getMethod().getConstructorOrMethod().getMethod().getAnnotation(Test.class);
		String testName = getUniqueTestName(testNGResult);
		if (testAnnotation != null && !testAnnotation.testName().isEmpty()) {
			int i = 0;
	    	for (Object parameter: testNGResult.getParameters()) {
	    		String key = String.format("arg%d", i++);
	    		getSeleniumRobotTestContext(testNGResult).getConfiguration().put(key, new TestVariable(key, parameter.toString()));
	    	}
	    	testName = StringUtility.interpolateString(testAnnotation.testName(), getSeleniumRobotTestContext(testNGResult));
		} 
		
		// store it for usage in reports (velocity access the attribute directly)
		testNGResult.setAttribute(VISUAL_TEST_NAME, testName);

		return testName;
	}
	
    /**
     * Generate a String which is unique for each combination of suite/testNG test/class/test method/parameters
     * @return
     */
    public static String getHashForTest(ITestResult testNGResult) {

    	String uniqueIdentifier;
    	if (testNGResult != null) {
    		
    		String suiteName = testNGResult.getTestContext().getSuite().getName();
    		String xmlTestName = testNGResult.getTestContext().getName();
			String className = testNGResult.getMethod().getRealClass().getName();
			String testMethodName = getTestName(testNGResult);
			Object[] testMethodParams = testNGResult.getParameters();
			
    		// issue #137: in case we are in a BeforeMethod, take class name and method name from TestMethod
    		if (testNGResult.getMethod().isBeforeMethodConfiguration()) {
    			Method testMethod = (Method)(testNGResult.getParameters()[0]);
    			className = testMethod.getDeclaringClass().getName();
    			testMethodName = "before-" + testMethod.getName();
    			testMethodParams = testMethod.getParameters();
    		}
        	
    		String testNameModified = StringUtility.replaceOddCharsFromFileName(testMethodName);
    		
			uniqueIdentifier = suiteName
	    			+ "-" + xmlTestName
	    			+ "-" + className
	    			+ "-" + testNameModified
	    			+ "-" + Arrays.hashCode(testMethodParams)
					+ "-" + ObjectUtils.identityToString(testNGResult.getMethod()); // #626: add ID of the test method itself.
			// During a test retry, method object remains the same so that we build the same output folder when test is retried
			// During a new invocation (through invocationCount for example), method object is different among executions


    	} else {
    		uniqueIdentifier = "null-null-null-null-0";
    	}
    	
    	return uniqueIdentifier;
    }
    
    /**
     * Returns the unique test name (it's the test name as returned by {@link TestNGResultUtils.getTestName()}) plus an index depending on the execution order of the method
     * (This index is calculated in {@link SeleniumTestsContext.hashTest()}. Resulting name is the folder name and becomes the unique test name, put into testResult attributes
     * @return
     */
    public static String getUniqueTestName(ITestResult testNGResult) {
    	return (String) testNGResult.getAttribute(UNIQUE_METHOD_NAME);
    }
    
    public static void setUniqueTestName(ITestResult testNGResult, String name) {
    	testNGResult.setAttribute(UNIQUE_METHOD_NAME, name);
    }
    
    public static String getTestMethodName(ITestResult testNGResult) {
    	return (String) testNGResult.getAttribute(METHOD_NAME);
    }
    
    public static void setTestMethodName(ITestResult testNGResult, String name) {
    	testNGResult.setAttribute(METHOD_NAME, name);
    }
    
    public static ITestNGMethod getLinkedTestMethod(ITestResult testNGResult) {
    	return (ITestNGMethod) testNGResult.getAttribute(LINKED_TEST_METHOD);
    }
    
    public static void setLinkedTestMethod(ITestResult testNGResult, ITestNGMethod testMethod) {
		if (testMethod == null) {
			return;
		}
    	testNGResult.setAttribute(LINKED_TEST_METHOD, testMethod);
    }
    
    public static SeleniumTestsContext getSeleniumRobotTestContext(ITestResult testNGResult) {
    	return (SeleniumTestsContext) testNGResult.getAttribute(TEST_CONTEXT);
    }
    
    public static void setSeleniumRobotTestContext(ITestResult testNGResult, SeleniumTestsContext context) {
    	testNGResult.setAttribute(TEST_CONTEXT, context);
    }
    
    // number of retry already done
    public static Integer getRetry(ITestResult testNGResult) {
    	return (Integer) testNGResult.getAttribute(RETRY);
    }
    
    public static void setRetry(ITestResult testNGResult, Integer retry) {
    	testNGResult.setAttribute(RETRY, retry);
    }
    
    // TestCaseInSession id as stored in snapshot server
    public static Integer getSnapshotTestCaseInSessionId(ITestResult testNGResult) {
    	return (Integer) testNGResult.getAttribute(SELENIUM_SERVER_REPORT_TEST_CASE_SESSION_ID);
    }
    
    public static void setSnapshotTestCaseInSessionId(ITestResult testNGResult, Integer sessionId) {
    	testNGResult.setAttribute(SELENIUM_SERVER_REPORT_TEST_CASE_SESSION_ID, sessionId);
    }
    
    // the snapshot comparison result
    public static Integer getSnapshotComparisonResult(ITestResult testNGResult) {
    	return (Integer) testNGResult.getAttribute(SNAPSHOT_COMPARISON_RESULT);
    }
    
    public static void setSnapshotComparisonResult(ITestResult testNGResult, int result) {
    	testNGResult.setAttribute(SNAPSHOT_COMPARISON_RESULT, result);
    }
    
    // did we already recorded this result to the server
    public static boolean isSeleniumServerReportCreated(ITestResult testNGResult) {
    	return isReportCreated(testNGResult, SELENIUM_SERVER_REPORT);
    }
    
    public static void setSeleniumServerReportCreated(ITestResult testNGResult, Boolean recordedToServer) {
    	testNGResult.setAttribute(SELENIUM_SERVER_REPORT, recordedToServer);
    }
    
    /**
     * 
     * @param testNGResult
     * @return true if the result has already been recorded to test manager
     */
    public static boolean isTestManagerReportCreated(ITestResult testNGResult) {
    	return isReportCreated(testNGResult, TEST_MANAGER_REPORT);
    }
   
    public static void setTestManagereportCreated(ITestResult testNGResult, Boolean recordedToServer) {
    	testNGResult.setAttribute(TEST_MANAGER_REPORT, recordedToServer);
    }
    
    public static boolean isBugtrackerReportCreated(ITestResult testNGResult) {
    	return isReportCreated(testNGResult, BUGTRACKER_REPORT);
    }

    public static void setBugtrackerReportCreated(ITestResult testNGResult, Boolean recordedToServer) {
    	testNGResult.setAttribute(BUGTRACKER_REPORT, recordedToServer);
    }
    
    /**
     * 
     * @param testNGResult
     * @return true if the HTML result has already been created for this result
     */
    public static boolean isHtmlReportCreated(ITestResult testNGResult) {
    	return isReportCreated(testNGResult, HTML_REPORT);
    }
    
    public static void setHtmlReportCreated(ITestResult testNGResult, Boolean recordedToServer) {
    	testNGResult.setAttribute(HTML_REPORT, recordedToServer);
    }
    
    /**
     * 
     * @param testNGResult
     * @return true if the Custom result has already been created for this result
     */
    public static boolean isCustomReportCreated(ITestResult testNGResult) {
    	return isReportCreated(testNGResult, CUSTOM_REPORT);
    }
    
    public static void setCustomReportCreated(ITestResult testNGResult, Boolean recordedToServer) {
    	testNGResult.setAttribute(CUSTOM_REPORT, recordedToServer);
    }
    
    private static boolean isReportCreated(ITestResult testNGResult, String attributeName) {
    	Boolean alreadyCreated = (Boolean) testNGResult.getAttribute(attributeName);
    	if (alreadyCreated == null) {
    		return false;
    	} else {
    		return alreadyCreated;
    	}
    }

    /**
     * whether retry will be allowed for this test
     * @param testNGResult
     * @return	true if no more retry is planned (test is finally failed) or when test is sucessful
     * 			false if test has failed and will be retried (current status is then skipped)
     * 			null if we never ask for a retry for this test
     */
    public static Boolean getNoMoreRetry(ITestResult testNGResult) {
    	return (Boolean) testNGResult.getAttribute(NO_MORE_RETRY);
    }
    
    public static void setNoMoreRetry(ITestResult testNGResult, Boolean noMoreRetry) {
    	testNGResult.setAttribute(NO_MORE_RETRY, noMoreRetry);
    }
    
    // information about test
    public static Map getTestInfo(ITestResult testNGResult) {
    	Map testInfo = (Map) testNGResult.getAttribute(TEST_INFO);
    	if (testInfo != null) {
    		return testInfo;
    	} else {
    		return new HashMap<>();
    	}
    }
    
    /**
     * returns test information encoded in the requested format
     * @param testNGResult
     * @param format		either "xml", "json", "html", "csv"
     * @return
     */
    public static Map getTestInfoEncoded(ITestResult testNGResult, String format) {

    	Map encodedTestInfos = new HashMap<>();
		for (Entry infoEntry: getTestInfo(testNGResult).entrySet()) {
			encodedTestInfos.put(StringUtility.encodeString(infoEntry.getKey(), format.toLowerCase()), 
									infoEntry.getValue().encode(format.toLowerCase()));
		}
		return encodedTestInfos;
    }
    
    public static void setTestInfo(ITestResult testNGResult, String key, Info value) {
    	Map testInfo = (Map) testNGResult.getAttribute(TEST_INFO);
    	if (testInfo == null) {
    		testInfo = new HashMap<>();
    	} 
    	testInfo.put(key, value);
    	testNGResult.setAttribute(TEST_INFO, testInfo);
    }
    
    /**
     * Returns the test description, interpolating variable
     * @param testNGResult
     * @return
     */
    public static String getTestDescription(ITestResult testNGResult) {
    	int i = 0;
    	for (Object parameter: testNGResult.getParameters()) {
    		String key = String.format("arg%d", i++);
    		getSeleniumRobotTestContext(testNGResult).getConfiguration().put(key, new TestVariable(key, parameter == null ? "null" : parameter.toString()));
    	}
    	String description = StringUtility.interpolateString(testNGResult.getMethod().getDescription(), getSeleniumRobotTestContext(testNGResult));
    	testNGResult.setAttribute(DESCRIPTION, description); // store it so that it's used in reports
    	return description;
    }

    /**
     * Change the test result when snapshot comparison fails
     * These comparison are done for every test execution (every retry). At this point, snapshot are not recorded on server. This will be recorded in SeleniumRobotServerTestRecorder
     * only with the last test execution.
     * @param testResult
     */
	public static void changeTestResultWithSnapshotComparison(final ITestResult testResult) {
		
		if (testResult.getStatus() == ITestResult.FAILURE  // test is already failed
				|| !Boolean.TRUE.equals(SeleniumTestsContextManager.getGlobalContext().seleniumServer().getSeleniumRobotServerActive())
				|| SeleniumTestsContextManager.getGlobalContext().seleniumServer().getSeleniumRobotServerCompareSnapshotBehaviour() == SnapshotComparisonBehaviour.DISPLAY_ONLY // as the comparison result is only displayed, do not retry
				|| SeleniumTestsContextManager.getGlobalContext().seleniumServer().getSeleniumRobotServerCompareSnapshotBehaviour() == SnapshotComparisonBehaviour.ADD_TEST_RESULT // complicated to set the test failed, and then success again
				|| !SeleniumTestsContextManager.getGlobalContext().seleniumServer().getSeleniumRobotServerCompareSnapshot()) {
			return;
		}
		
		SeleniumRobotSnapshotServerConnector serverConnector = SeleniumRobotSnapshotServerConnector.getInstance();
		
		List testSteps = getSeleniumRobotTestContext(testResult).getTestStepManager().getTestSteps();
		if (testSteps == null) {
			return;
		}
		
		for (TestStep testStep: testSteps) {
			for (Snapshot snapshot: new ArrayList<>(testStep.getSnapshots())) {
				if (snapshot.getCheckSnapshot().recordSnapshotOnServerForComparison()) {
					if (snapshot.getName() == null || snapshot.getName().isEmpty()) {
						logger.warn("Snapshot hasn't any name, it won't be sent to server");
						continue;
					} 
					
					try {
						SnapshotComparisonResult comparisonResult = serverConnector.checkSnapshotHasNoDifferences(snapshot, CommonReporter.getTestCaseName(testResult), testStep.getName());
						if (comparisonResult == SnapshotComparisonResult.KO) {
							testResult.setStatus(ITestResult.FAILURE);
							testResult.setThrowable(new ScenarioException("Snapshot comparison failed"));
							
							// move test from passedTests to failedTests if test is not already in failed tests
							if (testResult.getTestContext().getPassedTests().getAllMethods().contains(testResult.getMethod())) {
								testResult.getTestContext().getPassedTests().removeResult(testResult);
								testResult.getTestContext().getFailedTests().addResult(testResult);
							}
							return;
						}
					} catch (SeleniumRobotServerException e) {
						logger.error("Could not create snapshot on server", e);
					}
				}
			}
		}
		


	}	
	

	/**
	 * In case test result is SUCCESS but some softAssertions were raised, change test result to 
	 * FAILED
	 * 
	 * @param result
	 */
	public static void changeTestResultWithSoftAssertion(ITestResult result) {
		List verificationFailures = SeleniumTestsContextManager.getThreadContext().getVerificationFailures(Reporter.getCurrentTestResult());

		int size = verificationFailures.size();
		if (size == 0 || result.getStatus() == ITestResult.FAILURE) {
			return;
		}

		result.setStatus(ITestResult.FAILURE);

		if (size == 1) {
			result.setThrowable(verificationFailures.get(0));
		} else {
			
			StringBuilder stackString = new StringBuilder("!!! Many Test Failures (").append(size).append(")\n\n");
			
			for (int i = 0; i < size - 1; i++) {
				ExceptionUtility.generateTheStackTrace(verificationFailures.get(i), String.format("%n.%nFailure %d of %d%n", i + 1, size), stackString, "text");
			}
			
			Throwable last = verificationFailures.get(size - 1);
			stackString.append(String.format("%n.%nFailure %d of %d%n", size, size));
			stackString.append(last.toString());

			// set merged throwable
			Throwable merged = new AssertionError(stackString.toString());
			merged.setStackTrace(last.getStackTrace());

			result.setThrowable(merged);
		}

		// move test from passedTests to failedTests if test is not already in failed tests
		if (result.getTestContext().getPassedTests().getAllMethods().contains(result.getMethod())) {
			result.getTestContext().getPassedTests().removeResult(result);
			result.getTestContext().getFailedTests().addResult(result);
		}
	}	
	

    @SuppressWarnings("unchecked")
	public static List getErrorCauses(ITestResult testNGResult) {
    	if (testNGResult.getAttribute(ERROR_CAUSES) == null) {
    		return new ArrayList<>();
    	} else {
    		return (List) testNGResult.getAttribute(ERROR_CAUSES);
    	}
    }
    
    /**
     * Store the list of detected error causes
     * @param testNGResult
     * @param errorCauses
     */
    public static void setErrorCauses(ITestResult testNGResult, List errorCauses) {
    	testNGResult.setAttribute(ERROR_CAUSES, errorCauses);
    }
    
    public static boolean isErrorCauseSearchedInLastStep(ITestResult testNGResult) {
    	return isReportCreated(testNGResult, ERROR_CAUSE_IN_LAST_STEP);
    }
    
    public static void setErrorCauseSearchedInLastStep(ITestResult testNGResult, Boolean errorCauseInLastStep) {
    	testNGResult.setAttribute(ERROR_CAUSE_IN_LAST_STEP, errorCauseInLastStep);
    }
    
    public static boolean isErrorCauseSearchedInReferencePicture(ITestResult testNGResult) {
    	return isReportCreated(testNGResult, ERROR_CAUSE_IN_REFERENCE);
    }
    
    public static void setErrorCauseSearchedInReferencePicture(ITestResult testNGResult, Boolean errorCauseInReferencePicture) {
    	testNGResult.setAttribute(ERROR_CAUSE_IN_REFERENCE, errorCauseInReferencePicture);
    }

	public static void setFinished(ITestResult testNGResult, Boolean finished) {
		testNGResult.setAttribute(FINISHED, finished);
	}

	public static boolean isFinished(ITestResult testNGResult) {
		Boolean finished = (Boolean) testNGResult.getAttribute(FINISHED);
		return finished == null ? false: finished;
	}

	/**
	 * Returns the string representation of the status: SUCCESS, ERROR, SKIPPED, ...
	 * @param testNGResult
	 * @return
	 */
	public static String getTestStatusString(ITestResult testNGResult) {

		switch (testNGResult.getStatus()) {
			case -1:
				return "CREATED";
			case 1:
				return "SUCCESS";
			case 2:
				return "FAILURE";
			case 3:
				return "SKIP";
			case 4:
				return "SUCCESS_PERCENTAGE_FAILURE";
			case 16:
				return "STARTED";
			default:
				return "UNKNOWN";
		}
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy