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

com.seleniumtests.uipage.aspects.ReplayAction Maven / Gradle / Ivy

The newest version!
/**
 * 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.uipage.aspects;

import java.lang.reflect.Field;
import java.time.Clock;
import java.time.Instant;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;

import com.seleniumtests.customexception.ImageSearchException;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.DeclarePrecedence;
import org.openqa.selenium.InvalidElementStateException;
import org.openqa.selenium.NoSuchElementException;
import org.openqa.selenium.NotFoundException;
import org.openqa.selenium.TimeoutException;
import org.openqa.selenium.UnhandledAlertException;
import org.openqa.selenium.WebDriverException;
import org.openqa.selenium.WebElement;
import org.openqa.selenium.interactions.InputSource;
import org.openqa.selenium.interactions.Interaction;
import org.openqa.selenium.interactions.MoveTargetOutOfBoundsException;
import org.openqa.selenium.interactions.PointerInput;
import org.openqa.selenium.interactions.Sequence;
import org.openqa.selenium.remote.RemoteWebElement;
import org.openqa.selenium.remote.UnreachableBrowserException;

import com.seleniumtests.core.SeleniumTestsContextManager;
import com.seleniumtests.core.TestStepManager;
import com.seleniumtests.core.aspects.LogAction;
import com.seleniumtests.customexception.ConfigurationException;
import com.seleniumtests.customexception.DatasetException;
import com.seleniumtests.customexception.ScenarioException;
import com.seleniumtests.driver.BrowserType;
import com.seleniumtests.driver.CustomEventFiringWebDriver;
import com.seleniumtests.driver.WebUIDriver;
import com.seleniumtests.reporter.logger.TestAction;
import com.seleniumtests.uipage.ReplayOnError;
import com.seleniumtests.uipage.htmlelements.GenericPictureElement;
import com.seleniumtests.uipage.htmlelements.HtmlElement;
import com.seleniumtests.uipage.htmlelements.SelectList;
import com.seleniumtests.util.helper.WaitHelper;
import com.seleniumtests.util.logging.ScenarioLogger;

/**
 * Aspect to intercept calls to methods of HtmlElement. It allows to retry discovery and action 
 * when something goes wrong with the driver
 * 
 * @author behe
 *
 */
@Aspect
@DeclarePrecedence("*LogAction*, *ReplayAction*")
public class ReplayAction {

	private static Clock systemClock = Clock.systemUTC();
	private static final ScenarioLogger scenarioLogger = ScenarioLogger.getScenarioLogger(ReplayAction.class);

	private Object replayNonHtmlElement(ProceedingJoinPoint joinPoint, ReplayOnError replay) throws Throwable {
		
		int replayDelayMs = replay != null ? replay.replayDelayMs(): 100;
		
		Instant end = systemClock.instant().plusSeconds(SeleniumTestsContextManager.getThreadContext().getReplayTimeout());
		Object reply = null;

		while (end.isAfter(systemClock.instant())) {

			// chrome automatically scrolls to element before interacting but it may scroll behind fixed header and no error is
			// raised if action cannot be performed
			if (((CustomEventFiringWebDriver)WebUIDriver.getWebDriver(false)).getBrowserInfo().getBrowser() == BrowserType.CHROME
					|| ((CustomEventFiringWebDriver)WebUIDriver.getWebDriver(false)).getBrowserInfo().getBrowser() == BrowserType.EDGE) {
				updateScrollFlagForElement(joinPoint, true, null);
			}

			try {
				reply = joinPoint.proceed(joinPoint.getArgs());
				WaitHelper.waitForMilliSeconds(200);
				break;

			// do not replay if error comes from scenario
			} catch (ScenarioException | ConfigurationException | DatasetException e) {
				throw e;
			} catch (MoveTargetOutOfBoundsException | InvalidElementStateException e) {
				updateScrollFlagForElement(joinPoint, null, e);
			} catch (Throwable e) {

				if (end.minusMillis(replayDelayMs  + 200).isAfter(systemClock.instant())) {
					WaitHelper.waitForMilliSeconds(replayDelayMs);
					continue;
				} else {
					throw e;
				}
			}
		}
		return reply;

	}
	
	/**
	 * Replays the composite action in case any error occurs
	 * When the composite action is played inside an HtmlElement, and the calling method is annotated with {@code @ReplayOnError}, we have 2 replays
	 * 
	 *  method replay => ReplayOnError annotation
	 *  	composite action replay
	 *  
	 *  This does not seem to be a problem because if 'composite action replay' takes to much time, then 'method replay' will not effectively replay
	 *  
	 * @param joinPoint
	 */
	@Around("execution(public void org.openqa.selenium.interactions.Actions.BuiltAction.perform ())")
	public Object replayCompositeAction(ProceedingJoinPoint joinPoint) throws Throwable {
		return replayNonHtmlElement(joinPoint, null);
	}
	
	/**
	 * Updates the scrollToelementBeforeAction flag of HtmlElement for CompositeActions
	 * Therefore, it looks at origin field of PointerInput$Move CompositeAction and update the flag
	 * @throws SecurityException 
	 * @throws NoSuchFieldException 
	 * @throws IllegalAccessException 
	 * @throws IllegalArgumentException 
	 */
	private void updateScrollFlagForElement(ProceedingJoinPoint joinPoint, Boolean forcedValue, WebDriverException parentException) throws NoSuchFieldException, SecurityException, IllegalArgumentException, IllegalAccessException {
		Object actions = joinPoint.getTarget();
		
		// the calling method 'replay(ProceedingJoinPoint joinPoint, ReplayOnError replay)' may be called from GenericPictureElement.class or from Selenium 'Composite Actions'
		// Only the later case is covered here
		if (!actions.getClass().toString().contains("BuiltAction")) {
			return;
		}
		
		Field sequencesField = actions.getClass().getDeclaredField("sequences");
		sequencesField.setAccessible(true);
		Map sequences = (Map) sequencesField.get(actions);
		
		for (Sequence sequence: sequences.values()) {
			Field actionsField = Sequence.class.getDeclaredField("actions");
			actionsField.setAccessible(true);
			
			LinkedList actionsList = (LinkedList)actionsField.get(sequence);
			
			for (Interaction action: actionsList) {
				if (action.getClass().getName().contains("PointerInput$Move")) {
					Field originField = action.getClass().getDeclaredField("origin");
					originField.setAccessible(true);
					try {
						PointerInput.Origin origin = (PointerInput.Origin) originField.get(action);
						
						// we can change 'scrollToelementBeforeAction' flag only for HtmlElement objects. For RemoteWebElement, this cannot be done so we rethrow the exception
						// so that it can be treated elsewhere (mainly inside replayHtmlElement())
						if (origin.asArg() instanceof HtmlElement) {
							HtmlElement element = (HtmlElement) origin.asArg();
							if (forcedValue == null) {
								if (element.isScrollToElementBeforeAction()) {
					    			element.setScrollToElementBeforeAction(false);
					    		} else {
					    			element.setScrollToElementBeforeAction(true);
					    		}
							} else {
								element.setScrollToElementBeforeAction(forcedValue);
							}
						} else if (origin.asArg() instanceof RemoteWebElement && parentException != null) {
							throw parentException;
						}
					} catch (ClassCastException e1) {
						// nothing
					}
				}
			}
		}
	}

	private Integer getActionDelay() {
		return SeleniumTestsContextManager.getThreadContext().getActionDelay();
	}

	/**
	 * Replay all HtmlElement actions annotated by ReplayOnError.
	 * Classes which are not subclass of HtmlElement won't go there
	 * See javadoc of the annotation for details
	 * @param joinPoint
	 * @throws Throwable
	 */
	@Around("execution(public * com.seleniumtests.uipage.htmlelements.HtmlElement+.* (..))"
			+ "&& execution(@com.seleniumtests.uipage.ReplayOnError public * * (..)) && @annotation(replay)")
	public Object replayHtmlElement(ProceedingJoinPoint joinPoint, ReplayOnError replay) throws Throwable {

		Object reply = null;


		// update driver reference of the element
		// corrects bug of waitElementPresent which threw a SessionNotFoundError because driver reference were not
		// updated before searching element (it used the driver reference of an old test session)
		HtmlElement element = (HtmlElement)joinPoint.getTarget();
		element.setDriver(WebUIDriver.getWebDriver(false));
		String targetName = joinPoint.getTarget().toString();

		Instant end = systemClock.instant().plusSeconds(element.getReplayTimeout());

		TestAction currentAction = null;
		String methodName = joinPoint.getSignature().getName();
		if (!methodName.equals("getCoordinates")) {
			List pwdToReplace = new ArrayList<>();
			String actionName = String.format("%s on %s %s", methodName, targetName, LogAction.buildArgString(joinPoint, pwdToReplace, new HashMap<>()));
			currentAction = new TestAction(actionName, false, pwdToReplace, methodName, element);
		}

		// log action before its started. By default, it's OK. Then result may be overwritten if step fails
		// order of steps is the right one (first called is first displayed)
		if (currentAction != null && TestStepManager.getParentTestStep() != null) {
			TestStepManager.getParentTestStep().addAction(currentAction);
		}

		boolean actionFailed = false;
		boolean ignoreFailure = false;
		Throwable currentException = null;

		try {
			while (end.isAfter(systemClock.instant())) {

				// in case we have switched to an iframe for using previous webElement, go to default content
				if (element.getDriver() != null && SeleniumTestsContextManager.isWebTest()) {
					element.getDriver().switchTo().defaultContent(); // TODO: error when click is done, closing current window
				}

				try {
					reply = joinPoint.proceed(joinPoint.getArgs());

					// wait will be done only if action annotation request it
					if (replay.waitAfterAction()) {
						WaitHelper.waitForMilliSeconds(getActionDelay());
					}
					break;
				} catch (UnhandledAlertException e) {
					throw e;
				} catch (MoveTargetOutOfBoundsException | InvalidElementStateException e) {

					// if click has been intercepted, it means element could not be interacted, so allow auto scrolling for further retries
					// to avoid trying always the same method, we try without scrolling, then with scrolling, then without, ...
					element.setScrollToElementBeforeAction(!element.isScrollToElementBeforeAction());

				} catch (WebDriverException e) {
					// don't prevent TimeoutException to be thrown when coming from waitForPresent
					// only check that cause is the not found element and not an other error (NoSucheSessionError for example)
					if ((e instanceof TimeoutException
							&& joinPoint.getSignature().getName().equals("waitForPresent")
							&& e.getCause() instanceof NoSuchElementException) // issue #104: do not log error when waitForPresent raises TimeoutException
							|| (e instanceof NotFoundException
							&& isFromExpectedConditions(Thread.currentThread().getStackTrace())) // issue #194: return immediately if the action has been performed from ExpectedConditions class
						//   This way, we let the FluentWait process to retry or re-raise the exception
					)
					{
						ignoreFailure = true;
						throw e;
					}

					handleWebDriverException(replay, element, end, e);
				}

			}
			return reply;
		} catch (Throwable e) {
			if (e instanceof NoSuchElementException
					&& joinPoint.getTarget() instanceof HtmlElement
					&& (joinPoint.getSignature().getName().equals("findElements")
					|| joinPoint.getSignature().getName().equals("findHtmlElements"))) {
				return new ArrayList();
			} else {
				if (!ignoreFailure) {
					actionFailed = true;
					currentException = e;
				}
				throw e;
			}
		} finally {
			if (currentAction != null && TestStepManager.getParentTestStep() != null) {
				currentAction.setFailed(actionFailed);
				scenarioLogger.logActionError(currentException);
			}

			// restore element scrolling flag for further uses
			element.setScrollToElementBeforeAction(false);
		}
	}

	/**
	 * @param replay
	 * @param element
	 * @param end
	 * @param e
	 */
	private void handleWebDriverException(ReplayOnError replay, HtmlElement element, Instant end,
										  WebDriverException e) {
		if (end.minusMillis(replay.replayDelayMs() + 100L).isAfter(systemClock.instant())) {
			WaitHelper.waitForMilliSeconds(replay.replayDelayMs());
		} else {
			if (e instanceof NoSuchElementException) {
				if (element instanceof SelectList && e.getMessage().contains("option")) {
					throw new NoSuchElementException(String.format("'%s' from page '%s': %s", element, element.getOrigin(), e.getMessage()));
				} else {
					throw new NoSuchElementException(String.format("Searched element [%s] from page '%s' could not be found", element, element.getOrigin()));
				}
			} else if (e instanceof UnreachableBrowserException) {
				throw new WebDriverException("Browser did not reply, it may have frozen");
			}
			throw e;
		}
	}

	/**
	 * Replay all actions annotated by ReplayOnError if the class is not a subclass of
	 * HtmlElement (e.g: ScreenZone)
	 * @param joinPoint
	 * @throws Throwable
	 */
	@Around("execution(public * com.seleniumtests.uipage.htmlelements.GenericPictureElement+.* (..))"
			+ "&& execution(@com.seleniumtests.uipage.ReplayOnError public * * (..)) && @annotation(replay)")
	public Object replayGenericPicture(ProceedingJoinPoint joinPoint, ReplayOnError replay) throws Throwable {

		String methodName = joinPoint.getSignature().getName();
		String targetName = joinPoint.getTarget().toString();
		List pwdToReplace = new ArrayList<>();
		String actionName = String.format("%s on %s %s", methodName, targetName, LogAction.buildArgString(joinPoint, pwdToReplace, new HashMap<>()));
		TestAction currentAction = new TestAction(actionName, false, pwdToReplace, methodName, (GenericPictureElement)joinPoint.getTarget());

		// log action before its started. By default, it's OK. Then result may be overwritten if step fails
		// order of steps is the right one (first called is first displayed)
		if (TestStepManager.getParentTestStep() != null) {
			TestStepManager.getParentTestStep().addAction(currentAction);
		}

		boolean actionFailed = false;
		Throwable currentException = null;

		try {
			return replayNonHtmlElement(joinPoint, replay);
		} catch (Throwable e) {
			actionFailed = true;
			currentException = e;

			// log searched image and scene image in case image cannot be found.
			if (e instanceof ImageSearchException && joinPoint.getThis() instanceof GenericPictureElement) {
				scenarioLogger.logFile(((GenericPictureElement) joinPoint.getThis()).getObjectPictureFile(), "searched picture");
				scenarioLogger.logFile(((GenericPictureElement) joinPoint.getThis()).getScenePictureFile(), "scene to search in");
			}

			throw e;
		} finally {
			if (currentAction != null && TestStepManager.getParentTestStep() != null) {
				currentAction.setFailed(actionFailed);
				scenarioLogger.logActionError(currentException);

				if (joinPoint.getTarget() instanceof GenericPictureElement) {
					currentAction.setDurationToExclude(((GenericPictureElement)joinPoint.getTarget()).getActionDuration());
				}
			}
		}
	}


	/**
	 * issu #194: Returns true if the call to element action has been done from the org.openqa.selenium.support.ui.ExpectedConditions selenium class
	 *
	 * @param stack
	 * @return
	 */
	private boolean isFromExpectedConditions(StackTraceElement[] stack) {

		for(int i=0; i < stack.length; i++) {

			// when using aspects, class name may contain a "$", remove everything after that symbol
			String stackClass = stack[i].getClassName().split("\\$")[0];
			if (stackClass.equals("org.openqa.selenium.support.ui.ExpectedConditions")) {
				return true;
			}

		}
		return false;

	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy