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

gwen.web.WebEngine.scala Maven / Gradle / Ivy

/*
 * Copyright 2014-2020 Branko Juric, Brady Wood
 * 
 * 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 gwen.web

import java.util.concurrent.TimeUnit

import com.applitools.eyes.{MatchLevel, RectangleSize}

import scala.concurrent.duration.Duration
import gwen.Predefs.Formatting.DurationFormatter
import gwen.Predefs.Kestrel
import gwen.Predefs.RegexContext
import gwen.{GwenSettings, Settings}
import gwen.dsl._
import gwen.errors.StepFailure
import gwen.errors.undefinedStepError
import gwen.errors.disabledStepError
import gwen.eval.{GwenOptions, ScopedDataStack}
import gwen.eval.support.DefaultEngineSupport
import gwen.web.errors.LocatorBindingException
import org.apache.commons.text.StringEscapeUtils

import scala.util.{Failure, Success, Try}

/**
  * A web engine that uses the Selenium web driver
  * API to automate various web operations.
  * 
  * @author Branko Juric, Brady Wood
  */
trait WebEngine extends DefaultEngineSupport[WebEnvContext] {
  
  /**
    * Initialises and returns a new web environment context.
    * 
    * @param options command line options
    */
  override def init(options: GwenOptions) = {
    if (WebSettings.`gwen.web.capture.screenshots.highlighting`) {
      val fps = GwenSettings.`gwen.report.slideshow.framespersecond`
      Settings.setLocal("gwen.report.slideshow.framespersecond", (fps.toDouble * 1.8d).toInt.toString)
    }
    new WebEnvContext(options)
  }

  /**
    * Evaluates priority steps supported by this engine. For example, a step that calls another step needs to execute
    * with priority to ensure that there is no match conflict between the two (which can occur if the step being
    * called by a step is a StepDef or another step that matches the entire calling step)
    *
    * @param step the step to evaluate
    * @param env the environment context
    */
  override def evaluatePriority(step: Step, env: WebEnvContext): Option[Step] = {

    val webContext = env.webContext

    Option {

      step.expression match {

        case r"""(.+?)$doStep for each (.+?)$element located by (id|name|tag name|css selector|xpath|class name|link text|partial link text|javascript)$locator "(.+?)"$expression in (.+?)$container with no (?:timeout|wait)""" =>
          env.getLocatorBinding(container)
          val binding = LocatorBinding(s"${element}/list", locator, expression, Some(container), Some(Duration.Zero), None)
          env.evaluate(foreach(() => List("$[dryRun:webElements]"), element, step, doStep, env)) {
            foreach(() => webContext.locateAll(binding), element, step, doStep, env)
          }

        case r"""(.+?)$doStep for each (.+?)$element located by (id|name|tag name|css selector|xpath|class name|link text|partial link text|javascript)$locator "(.+?)"$expression in (.+?)$container with (\d+)$timeout second (?:timeout|wait)""" =>
          env.getLocatorBinding(container)
          val binding = LocatorBinding(s"${element}/list", locator, expression, Some(container), Some(Duration.create(timeout.toLong, TimeUnit.SECONDS)), None)
          env.evaluate(foreach(() => List("$[dryRun:webElements]"), element, step, doStep, env)) {
            foreach(() => webContext.locateAll(binding), element, step, doStep, env)
          }

        case r"""(.+?)$doStep for each (.+?)$element located by (id|name|tag name|css selector|xpath|class name|link text|partial link text|javascript)$locator "(.+?)"$expression in (.+?)$container""" =>
          env.getLocatorBinding(container)
          val binding = LocatorBinding(s"${element}/list", locator, expression, Some(container), None, None)
          env.evaluate(foreach(() => List("$[dryRun:webElements]"), element, step, doStep, env)) {
            foreach(() => webContext.locateAll(binding), element, step, doStep, env)
          }

        case r"""(.+?)$doStep for each (.+?)$element located by (id|name|tag name|css selector|xpath|class name|link text|partial link text|javascript)$locator "(.+?)"$expression with no (?:timeout|wait)""" =>
          val binding = LocatorBinding(s"${element}/list", locator, expression, None, Some(Duration.Zero), None)
          env.evaluate(foreach(() => List("$[dryRun:webElements]"), element, step, doStep, env)) {
            foreach(() => webContext.locateAll(binding), element, step, doStep, env)
          }

        case r"""(.+?)$doStep for each (.+?)$element located by (id|name|tag name|css selector|xpath|class name|link text|partial link text|javascript)$locator "(.+?)"$expression with (\d+)$timeout second (?:timeout|wait)""" =>
          val binding = LocatorBinding(s"${element}/list", locator, step.orDocString(expression), None, Some(Duration.create(timeout.toLong, TimeUnit.SECONDS)), None)
          env.evaluate(foreach(() => List("$[dryRun:webElements]"), element, step, doStep, env)) {
            foreach(() => webContext.locateAll(binding), element, step, doStep, env)
          }

        case r"""(.+?)$doStep for each (.+?)$element located by (id|name|tag name|css selector|xpath|class name|link text|partial link text|javascript)$locator "(.+?)"$expression""" =>
          val binding = LocatorBinding(s"${element}/list", locator, step.orDocString(expression), None, None, None)
          env.evaluate(foreach(() => List("$[dryRun:webElements]"), element, step, doStep, env)) {
            foreach(() => webContext.locateAll(binding), element, step, doStep, env)
          }

        case r"""(.+?)$doStep for each (.+?)$element in (.+?)$$$iteration""" if !iteration.contains("delimited by") =>
          val binding = env.getLocatorBinding(iteration)
          env.evaluate(foreach(() => List("$[dryRun:webElements]"), element, step, doStep, env)) {
            foreach(() => webContext.locateAll(binding), element, step, doStep, env)
          }

        case r"""(.+?)$doStep (until|while)$operation (.+?)$condition using no delay and (.+?)$timeoutPeriod (minute|second|millisecond)$timeoutUnit (?:timeout|wait)""" =>
          repeat(operation, step, doStep, condition, Duration.Zero, Duration(timeoutPeriod.toLong, timeoutUnit), env)

        case r"""(.+?)$doStep (until|while)$operation (.+?)$condition using no delay""" =>
          repeat(operation, step, doStep, condition, Duration.Zero, defaultRepeatTimeout(DefaultRepeatDelay), env)

        case r"""(.+?)$doStep (until|while)$operation (.+?)$condition using (.+?)$delayPeriod (second|millisecond)$delayUnit delay and (.+?)$timeoutPeriod (minute|second|millisecond)$timeoutUnit (?:timeout|wait)""" =>
          repeat(operation, step, doStep, condition, Duration(delayPeriod.toLong, delayUnit), Duration(timeoutPeriod.toLong, timeoutUnit), env)

        case r"""(.+?)$doStep (until|while)$operation (.+?)$condition using (.+?)$delayPeriod (second|millisecond)$delayUnit delay""" =>
          val delayDuration = Duration(delayPeriod.toLong, delayUnit)
          repeat(operation, step, doStep, condition, delayDuration, defaultRepeatTimeout(delayDuration), env)

        case r"""(.+?)$doStep (until|while)$operation (.+?)$condition using (.+?)$timeoutPeriod (minute|second|millisecond)$timeoutUnit (?:timeout|wait)""" =>
          repeat(operation, step, doStep, condition, DefaultRepeatDelay, Duration(timeoutPeriod.toLong, timeoutUnit), env)

        case r"""(.+?)$doStep (until|while)$operation (.+?)$$$condition""" if doStep != "I wait" =>
          repeat(operation, step, doStep, condition, DefaultRepeatDelay, defaultRepeatTimeout(DefaultRepeatDelay), env)

        case _ =>
          super.evaluatePriority(step, env).orNull

      }
    }
  }

  /**
    * Evaluates a given step.  This method matches the incoming step against a
    * set of supported steps and evaluates only those that are successfully
    * matched.
    *
    * @param step the step to evaluate
    * @param env the web environment context
    */
  override def evaluate(step: Step, env: WebEnvContext): Unit = {

    val webContext = env.webContext

    step.expression match {

      case r"""I wait for (.+?)$element text for (.+?)$seconds second(?:s?)""" =>
        checkStepRules(step, BehaviorType.Action, env)
        val elementBinding = env.getLocatorBinding(element)
        webContext.waitUntil(seconds.toInt) {
          webContext.waitForText(elementBinding)
        }

      case r"""I wait for (.+?)$element text""" =>
        checkStepRules(step, BehaviorType.Action, env)
        val elementBinding = env.getLocatorBinding(element)
        webContext.waitUntil {
          webContext.waitForText(elementBinding)
        }

      case r"""I wait for (.+?)$element for (.+?)$seconds second(?:s?)""" =>
        checkStepRules(step, BehaviorType.Action, env)
        val elementBinding = env.getLocatorBinding(element)
        webContext.waitUntil(seconds.toInt) {
          Try(webContext.locateAndHighlight(elementBinding)).isSuccess
        }

      case r"""I wait for (.+?)$$$element""" =>
        checkStepRules(step, BehaviorType.Action, env)
        val elementBinding = env.getLocatorBinding(element)
        webContext.waitUntil {
          Try(webContext.locateAndHighlight(elementBinding)).isSuccess
        }

      case r"""I wait ([0-9]+?)$duration second(?:s?) when (.+?)$element is (clicked|right clicked|double clicked|submitted|checked|ticked|unchecked|unticked|selected|deselected|typed|entered|tabbed|cleared|moved to)$$$event""" =>
        checkStepRules(step, BehaviorType.Action, env)
        env.getLocatorBinding(element)
        env.scopes.set(s"$element/${WebEvents.EventToAction(event)}/wait", duration)

      case r"""I wait until (.+?)$condition when (.+?)$element is (clicked|right clicked|double clicked||submitted|checked|ticked|unchecked|unticked|selected|deselected|typed|entered|tabbed|cleared|moved to)$$$event""" =>
        checkStepRules(step, BehaviorType.Action, env)
        env.scopes.get(s"$condition/javascript")
        env.getLocatorBinding(element)
        env.scopes.set(s"$element/${WebEvents.EventToAction(event)}/condition", condition)

      case r"""I wait until "(.+?)$javascript"""" => step.orDocString(javascript) tap { javascript =>
        checkStepRules(step, BehaviorType.Action, env)
        webContext.waitUntil {
          env.evaluateJSPredicate(javascript)
        }
      }

      case r"""I wait until (.+?)$element is( not)?$negation (displayed|hidden|checked|ticked|unchecked|unticked|enabled|disabled)$$$state""" if env.scopes.getOpt(s"$element is${if(Option(negation).isDefined) " not" else ""} $state/javascript").isEmpty =>
        checkStepRules(step, BehaviorType.Action, env)
        val elementBinding = env.getLocatorBinding(element).jsEquivalent
        val negate = Option(negation).isDefined
        webContext.waitForElementState(elementBinding, state, negate)

      case r"""I wait until (.+?)$$$condition""" =>
        checkStepRules(step, BehaviorType.Action, env)
        val javascript = env.scopes.get(s"$condition/javascript")
        webContext.waitUntil {
          env.evaluateJSPredicate(javascript)
        }

      case r"""I am on the (.+?)$$$page""" =>
        checkStepRules(step, BehaviorType.Context, env)
        env.scopes.addScope(page)

      case r"""I navigate to the (.+?)$$$page""" =>
        checkStepRules(step, BehaviorType.Action, env)
        env.scopes.addScope(page)
        val url = env.getAttribute("url")
        webContext.navigateTo(url)

      case r"""I navigate to "(.+?)"$$$url""" => step.orDocString(url) tap { url =>
        checkStepRules(step, BehaviorType.Action, env)
        webContext.navigateTo(url)
      }

      case r"""I scroll to the (top|bottom)$position of (.+?)$$$element""" =>
        checkStepRules(step, BehaviorType.Action, env)
        val elementBinding = env.getLocatorBinding(element)
        webContext.scrollIntoView(elementBinding, ScrollTo.withName(position))

      case r"""the url will be defined by (?:property|setting) "(.+?)"$$$name""" => step.orDocString(name) tap { name =>
        checkStepRules(step, BehaviorType.Context, env)
        env.scopes.set("url", Settings.get(name))
      }

      case r"""the url will be "(.+?)"$$$url""" => step.orDocString(url) tap { url =>
        checkStepRules(step, BehaviorType.Context, env)
        env.scopes.set("url", url)
      }

      case r"""the (.+?)$page url is "(.+?)"$$$url""" => step.orDocString(url) tap { url =>
        checkStepRules(step, BehaviorType.Context, env)
        env.scopes.addScope(page)
        env.scopes.set("url", url)
      }

      case r"""(.+?)$element can be located by (id|name|tag name|css selector|xpath|class name|link text|partial link text|javascript)$locator "(.+?)"$expression at index (\d+)$index in (.+?)$container with no (?:timeout|wait)""" =>
        checkStepRules(step, BehaviorType.Context, env)
        env.getLocatorBinding(container)
        env.scopes.set(s"$element/locator", locator)
        env.scopes.set(s"$element/locator/$locator", expression)
        env.scopes.set(s"$element/locator/$locator/container", container)
        env.scopes.set(s"$element/locator/$locator/timeoutSecs", "0")
        env.scopes.set(s"$element/locator/$locator/index", index)

      case r"""(.+?)$element can be located by (id|name|tag name|css selector|xpath|class name|link text|partial link text|javascript)$locator "(.+?)"$expression in (.+?)$container with no (?:timeout|wait)""" =>
        checkStepRules(step, BehaviorType.Context, env)
        env.getLocatorBinding(container)
        env.scopes.set(s"$element/locator", locator)
        env.scopes.set(s"$element/locator/$locator", expression)
        env.scopes.set(s"$element/locator/$locator/container", container)
        env.scopes.set(s"$element/locator/$locator/timeoutSecs", "0")
        env.scopes.getOpt(s"$element/locator/$locator/index") foreach { _ =>
          env.scopes.set(s"$element/locator/$locator/index", null)
        }

      case r"""(.+?)$element can be located by (id|name|tag name|css selector|xpath|class name|link text|partial link text|javascript)$locator "(.+?)"$expression at index (\d+)$index in (.+?)$container with (\d+)$timeout second (?:timeout|wait)""" =>
        checkStepRules(step, BehaviorType.Context, env)
        env.getLocatorBinding(container)
        env.scopes.set(s"$element/locator", locator)
        env.scopes.set(s"$element/locator/$locator", expression)
        env.scopes.set(s"$element/locator/$locator/container", container)
        env.scopes.set(s"$element/locator/$locator/timeoutSecs", timeout)
        env.scopes.set(s"$element/locator/$locator/index", index)

      case r"""(.+?)$element can be located by (id|name|tag name|css selector|xpath|class name|link text|partial link text|javascript)$locator "(.+?)"$expression in (.+?)$container with (\d+)$timeout second (?:timeout|wait)""" =>
        checkStepRules(step, BehaviorType.Context, env)
        env.getLocatorBinding(container)
        env.scopes.set(s"$element/locator", locator)
        env.scopes.set(s"$element/locator/$locator", expression)
        env.scopes.set(s"$element/locator/$locator/container", container)
        env.scopes.set(s"$element/locator/$locator/timeoutSecs", timeout)
        env.scopes.getOpt(s"$element/locator/$locator/index") foreach { _ =>
          env.scopes.set(s"$element/locator/$locator/index", null)
        }

      case r"""(.+?)$element can be located by (id|name|tag name|css selector|xpath|class name|link text|partial link text|javascript)$locator "(.+?)"$expression at index (\d+)$index in (.+?)$container""" =>
        checkStepRules(step, BehaviorType.Context, env)
        env.getLocatorBinding(container)
        env.scopes.set(s"$element/locator", locator)
        env.scopes.set(s"$element/locator/$locator", expression)
        env.scopes.set(s"$element/locator/$locator/container", container)
        env.scopes.getOpt(s"$element/locator/$locator/timeoutSecs") foreach { _ =>
          env.scopes.set(s"$element/locator/$locator/timeoutSecs", null)
        }
        env.scopes.set(s"$element/locator/$locator/index", index)

      case r"""(.+?)$element can be located by (id|name|tag name|css selector|xpath|class name|link text|partial link text|javascript)$locator "(.+?)"$expression in (.+?)$container""" =>
        checkStepRules(step, BehaviorType.Context, env)
        env.getLocatorBinding(container)
        env.scopes.set(s"$element/locator", locator)
        env.scopes.set(s"$element/locator/$locator", expression)
        env.scopes.set(s"$element/locator/$locator/container", container)
        env.scopes.getOpt(s"$element/locator/$locator/timeoutSecs") foreach { _ =>
          env.scopes.set(s"$element/locator/$locator/timeoutSecs", null)
        }
        env.scopes.getOpt(s"$element/locator/$locator/index") foreach { _ =>
          env.scopes.set(s"$element/locator/$locator/index", null)
        }

      case r"""(.+?)$element can be located by (id|name|tag name|css selector|xpath|class name|link text|partial link text|javascript)$locator "(.+?)"$expression at index (\d+)$index with no (?:timeout|wait)""" =>
        checkStepRules(step, BehaviorType.Context, env)
        env.scopes.set(s"$element/locator", locator)
        env.scopes.set(s"$element/locator/$locator", expression)
        env.scopes.getOpt(s"$element/locator/$locator/container") foreach { _ =>
          env.scopes.set(s"$element/locator/$locator/container", null)
        }
        env.scopes.set(s"$element/locator/$locator/timeoutSecs", "0")
        env.scopes.set(s"$element/locator/$locator/index", index)

      case r"""(.+?)$element can be located by (id|name|tag name|css selector|xpath|class name|link text|partial link text|javascript)$locator "(.+?)"$expression with no (?:timeout|wait)""" =>
        checkStepRules(step, BehaviorType.Context, env)
        env.scopes.set(s"$element/locator", locator)
        env.scopes.set(s"$element/locator/$locator", expression)
        env.scopes.getOpt(s"$element/locator/$locator/container") foreach { _ =>
          env.scopes.set(s"$element/locator/$locator/container", null)
        }
        env.scopes.set(s"$element/locator/$locator/timeoutSecs", "0")
        env.scopes.getOpt(s"$element/locator/$locator/index") foreach { _ =>
          env.scopes.set(s"$element/locator/$locator/index", null)
        }

      case r"""(.+?)$element can be located by (id|name|tag name|css selector|xpath|class name|link text|partial link text|javascript)$locator "(.+?)"$expression at index (\d+)$index with (\d+)$timeout second (?:timeout|wait)""" =>
        checkStepRules(step, BehaviorType.Context, env)
        env.scopes.set(s"$element/locator", locator)
        env.scopes.set(s"$element/locator/$locator", expression)
        env.scopes.getOpt(s"$element/locator/$locator/container") foreach { _ =>
          env.scopes.set(s"$element/locator/$locator/container", null)
        }
        env.scopes.set(s"$element/locator/$locator/timeoutSecs", timeout)
        env.scopes.set(s"$element/locator/$locator/index", index)

      case r"""(.+?)$element can be located by (id|name|tag name|css selector|xpath|class name|link text|partial link text|javascript)$locator "(.+?)"$expression with (\d+)$timeout second (?:timeout|wait)""" =>
        checkStepRules(step, BehaviorType.Context, env)
        env.scopes.set(s"$element/locator", locator)
        env.scopes.set(s"$element/locator/$locator", expression)
        env.scopes.getOpt(s"$element/locator/$locator/container") foreach { _ =>
          env.scopes.set(s"$element/locator/$locator/container", null)
        }
        env.scopes.set(s"$element/locator/$locator/timeoutSecs", timeout)
        env.scopes.getOpt(s"$element/locator/$locator/index") foreach { _ =>
          env.scopes.set(s"$element/locator/$locator/index", null)
        }

      case r"""(.+?)$element can be located by (id|name|tag name|css selector|xpath|class name|link text|partial link text|javascript)$locator "(.+?)"$expression at index (\d+)$index""" =>
        checkStepRules(step, BehaviorType.Context, env)
        env.scopes.set(s"$element/locator", locator)
        env.scopes.set(s"$element/locator/$locator", expression)
        env.scopes.getOpt(s"$element/locator/$locator/container") foreach { _ =>
          env.scopes.set(s"$element/locator/$locator/container", null)
        }
        env.scopes.getOpt(s"$element/locator/$locator/timeoutSecs") foreach { _ =>
          env.scopes.set(s"$element/locator/$locator/timeoutSecs", null)
        }
        env.scopes.set(s"$element/locator/$locator/index", index)

      case r"""(.+?)$element can be located at index (\d+)$index by (id|name|tag name|css selector|xpath|class name|link text|partial link text|javascript)$locator "(.+?)"$expression""" => step.orDocString(expression) tap { expression =>
        checkStepRules(step, BehaviorType.Context, env)
        env.scopes.set(s"$element/locator", locator)
        env.scopes.set(s"$element/locator/$locator", expression)
        env.scopes.getOpt(s"$element/locator/$locator/container") foreach { _ =>
          env.scopes.set(s"$element/locator/$locator/container", null)
        }
        env.scopes.getOpt(s"$element/locator/$locator/timeoutSecs") foreach { _ =>
          env.scopes.set(s"$element/locator/$locator/timeoutSecs", null)
        }
        env.scopes.set(s"$element/locator/$locator/index", index)
      }

      case r"""(.+?)$element can be located by (id|name|tag name|css selector|xpath|class name|link text|partial link text|javascript)$locator "(.+?)"$expression""" => step.orDocString(expression) tap { expression =>
        checkStepRules(step, BehaviorType.Context, env)
        env.scopes.set(s"$element/locator", locator)
        env.scopes.set(s"$element/locator/$locator", expression)
        env.scopes.getOpt(s"$element/locator/$locator/container") foreach { _ =>
          env.scopes.set(s"$element/locator/$locator/container", null)
        }
        env.scopes.getOpt(s"$element/locator/$locator/timeoutSecs") foreach { _ =>
          env.scopes.set(s"$element/locator/$locator/timeoutSecs", null)
        }
        env.scopes.getOpt(s"$element/locator/$locator/index") foreach { _ =>
          env.scopes.set(s"$element/locator/$locator/index", null)
        }
      }

      case r"""(.+?)$element can be located at index (\d+)$index in (.+?)$container by""" if step.table.nonEmpty && step.table.head._2.size == 2 =>
        checkStepRules(step, BehaviorType.Context, env)
        env.getLocatorBinding(container)
        env.scopes.set(s"$element/locator", step.table.map(_._2.head).mkString(","))
        step.table foreach { case (_, row ) =>
          val locator = row.head
          val expression = row(1)
          env.scopes.set(s"$element/locator/$locator", expression)
          env.scopes.set(s"$element/locator/$locator/container", container)
          env.scopes.set(s"$element/locator/$locator/index", index)
        }

      case r"""(.+?)$element can be located in (.+?)$container by""" if step.table.nonEmpty && step.table.head._2.size == 2 =>
        checkStepRules(step, BehaviorType.Context, env)
        env.getLocatorBinding(container)
        env.scopes.set(s"$element/locator", step.table.map(_._2.head).mkString(","))
        step.table foreach { case (_, row ) =>
          val locator = row.head
          val expression = row(1)
          env.scopes.set(s"$element/locator/$locator", expression)
          env.scopes.set(s"$element/locator/$locator/container", container)
          env.scopes.getOpt(s"$element/locator/$locator/index") foreach { _ =>
            env.scopes.set(s"$element/locator/$locator/index", null)
          }
        }

      case r"""(.+?)$element can be located at index (\d+)$index with no (?:timeout|wait) by""" if step.table.nonEmpty && step.table.head._2.size == 2 => {
        checkStepRules(step, BehaviorType.Context, env)
        env.scopes.set(s"$element/locator", step.table.map(_._2.head).mkString(","))
        step.table foreach { case (_, row ) =>
          val locator = row.head
          val expression = row(1)
          env.scopes.set(s"$element/locator/$locator", expression)
          env.scopes.getOpt(s"$element/locator/$locator/container") foreach { _ =>
            env.scopes.set(s"$element/locator/$locator/container", null)
          }
          env.scopes.set(s"$element/locator/$locator/timeoutSecs", "0")
          env.scopes.set(s"$element/locator/$locator/index", index)
        }
      }

      case r"""(.+?)$element can be located with no (?:timeout|wait) by""" if step.table.nonEmpty && step.table.head._2.size == 2 => {
        checkStepRules(step, BehaviorType.Context, env)
        env.scopes.set(s"$element/locator", step.table.map(_._2.head).mkString(","))
        step.table foreach { case (_, row ) =>
          val locator = row.head
          val expression = row(1)
          env.scopes.set(s"$element/locator/$locator", expression)
          env.scopes.getOpt(s"$element/locator/$locator/container") foreach { _ =>
            env.scopes.set(s"$element/locator/$locator/container", null)
          }
          env.scopes.set(s"$element/locator/$locator/timeoutSecs", "0")
          env.scopes.getOpt(s"$element/locator/$locator/index") foreach { _ =>
            env.scopes.set(s"$element/locator/$locator/index", null)
          }
        }
      }

      case r"""(.+?)$element can be located at index (\d+)$index with (\d+)$timeout second (?:timeout|wait) by""" if step.table.nonEmpty && step.table.head._2.size == 2 => {
        checkStepRules(step, BehaviorType.Context, env)
        env.scopes.set(s"$element/locator", step.table.map(_._2.head).mkString(","))
        step.table foreach { case (_, row ) =>
          val locator = row.head
          val expression = row(1)
          env.scopes.set(s"$element/locator/$locator", expression)
          env.scopes.getOpt(s"$element/locator/$locator/container") foreach { _ =>
            env.scopes.set(s"$element/locator/$locator/container", null)
          }
          env.scopes.set(s"$element/locator/$locator/timeoutSecs", timeout)
          env.scopes.set(s"$element/locator/$locator/index", index)
        }
      }

      case r"""(.+?)$element can be located with (\d+)$timeout second (?:timeout|wait) by""" if step.table.nonEmpty && step.table.head._2.size == 2 => {
        checkStepRules(step, BehaviorType.Context, env)
        env.scopes.set(s"$element/locator", step.table.map(_._2.head).mkString(","))
        step.table foreach { case (_, row ) =>
          val locator = row.head
          val expression = row(1)
          env.scopes.set(s"$element/locator/$locator", expression)
          env.scopes.getOpt(s"$element/locator/$locator/container") foreach { _ =>
            env.scopes.set(s"$element/locator/$locator/container", null)
          }
          env.scopes.set(s"$element/locator/$locator/timeoutSecs", timeout)
          env.scopes.getOpt(s"$element/locator/$locator/index") foreach { _ =>
            env.scopes.set(s"$element/locator/$locator/index", null)
          }
        }
      }

      case r"""(.+?)$element can be located at index (\d+)$index by""" if step.table.nonEmpty && step.table.head._2.size == 2 => {
        checkStepRules(step, BehaviorType.Context, env)
        env.scopes.set(s"$element/locator", step.table.map(_._2.head).mkString(","))
        step.table foreach { case (_, row ) =>
          val locator = row.head
          val expression = row(1)
          env.scopes.set(s"$element/locator/$locator", expression)
          env.scopes.getOpt(s"$element/locator/$locator/container") foreach { _ =>
            env.scopes.set(s"$element/locator/$locator/container", null)
          }
          env.scopes.getOpt(s"$element/locator/$locator/timeoutSecs") foreach { _ =>
            env.scopes.set(s"$element/locator/$locator/timeoutSecs", null)
          }
          env.scopes.set(s"$element/locator/$locator/index", index)
        }
      }

      case r"""(.+?)$element can be located by""" if step.table.nonEmpty && step.table.head._2.size == 2 => {
        checkStepRules(step, BehaviorType.Context, env)
        env.scopes.set(s"$element/locator", step.table.map(_._2.head).mkString(","))
        step.table foreach { case (_, row ) =>
          val locator = row.head
          val expression = row(1)
          env.scopes.set(s"$element/locator/$locator", expression)
          env.scopes.getOpt(s"$element/locator/$locator/container") foreach { _ =>
            env.scopes.set(s"$element/locator/$locator/container", null)
          }
          env.scopes.getOpt(s"$element/locator/$locator/timeoutSecs") foreach { _ =>
            env.scopes.set(s"$element/locator/$locator/timeoutSecs", null)
          }
          env.scopes.getOpt(s"$element/locator/$locator/index") foreach { _ =>
            env.scopes.set(s"$element/locator/$locator/index", null)
          }
        }
      }

      case r"""(.+?)$element can be (clicked|right clicked|double clicked|submitted|checked|ticked|unchecked|unticked|selected|deselected|typed|entered|tabbed|cleared|moved to)$event by javascript "(.+?)"$$$expression""" => step.orDocString(expression) tap { expression =>
        checkStepRules(step, BehaviorType.Context, env)
        env.getLocatorBinding(element)
        env.scopes.set(s"$element/action/${WebEvents.EventToAction(event)}/javascript", expression)
      }

      case r"""the page title should( not)?$negation (be|contain|start with|end with|match regex|match xpath|match json path|match template|match template file)$operator "(.*?)"$$$expression""" => step.orDocString(expression) tap { expression =>
        checkStepRules(step, BehaviorType.Assertion, env)
        val expected = env.parseExpression(operator, expression)
        env.perform {
          env.compare("title", expected, () => webContext.getTitle, operator, Option(negation).isDefined)
        }
      }

      case r"""the page title should( not)?$negation (be|contain|start with|end with|match regex|match xpath|match json path)$operator (.+?)$$$attribute""" =>
      checkStepRules(step, BehaviorType.Assertion, env)
        val expected = env.getBoundReferenceValue(attribute)
        env.perform {
          env.compare("title", expected, () => webContext.getTitle, operator, Option(negation).isDefined)
        }

      case r"""the (alert|confirmation)$name popup message should( not)?$negation (be|contain|start with|end with|match regex|match xpath|match json path|match template|match template file)$operator "(.*?)"$$$expression""" => step.orDocString(expression) tap { expression =>
        checkStepRules(step, BehaviorType.Assertion, env)
        val expected = env.parseExpression(operator, expression)
        env.perform {
          env.compare(name, expected, () => webContext.getPopupMessage, operator, Option(negation).isDefined)
        }
      }

      case r"""the (alert|confirmation)$name popup message should( not)?$negation (be|contain|start with|end with|match regex|match xpath|match json path)$operator (.+?)$$$attribute""" =>
        checkStepRules(step, BehaviorType.Assertion, env)
        val expected = env.getBoundReferenceValue(attribute)
        env.perform {
          env.compare(name, expected, () => webContext.getPopupMessage, operator, Option(negation).isDefined)
        }

      case r"""(.+?)$element should( not)?$negation be (displayed|hidden|checked|ticked|unchecked|unticked|enabled|disabled)$$$state""" =>
        checkStepRules(step, BehaviorType.Assertion, env)
        val elementBinding = env.getLocatorBinding(element).jsEquivalent
        webContext.checkElementState(elementBinding, state, Option(negation).nonEmpty)

      case r"""(.+?)$element( text| value)?$selection should( not)?$negation (be|contain|start with|end with|match regex|match xpath|match json path|match template|match template file)$operator "(.*?)"$$$expression""" if !element.matches(".+at (json path|xpath).+") => step.orDocString(expression) tap { expression =>
        checkStepRules(step, BehaviorType.Assertion, env)
        if (element == "I") undefinedStepError(step)
        if (element == "the current URL") webContext.captureCurrentUrl(None)
        val negate = Option(negation).isDefined
        val expected = env.parseExpression(operator, expression)
        val actual = env.boundAttributeOrSelection(element, Option(selection))
        env.perform {
          if (env.scopes.findEntry { case (n, _) => n.startsWith(element) } forall { case (n, _) => n != element }) {
            env.compare(element + Option(selection).getOrElse(""), expected, actual, operator, negate)
          } else {
            val actualValue = env.scopes.getOpt(element).getOrElse(actual())
            val result = env.compare(element, expected, actualValue, operator, negate)
            result match {
              case Success(assertion) =>
                assert(assertion, s"Expected $element to ${if(negate) "not " else ""}$operator '$expected' but got '$actualValue'")
              case Failure(error) =>
                assert(assertion = false, error.getMessage)
            }
          }
        } getOrElse  {
          actual()
        }
      }

      case r"""(.+?)$element( value| text)?$selection should( not)?$negation (be|contain|start with|end with|match regex|match xpath|match json path)$operator (.+?)$$$attribute""" if attribute != "absent" && !element.matches(".+at (json path|xpath).+") =>
        checkStepRules(step, BehaviorType.Assertion, env)
        if (element == "I") undefinedStepError(step)
        if (element == "the current URL") webContext.captureCurrentUrl(None)
        val expected = env.getBoundReferenceValue(attribute)
        val actual = env.boundAttributeOrSelection(element, Option(selection))
        env.perform {
          env.compare(element + Option(selection).getOrElse(""), expected, actual, operator, Option(negation).isDefined)
        } getOrElse  {
          actual()
        }

      case r"""I capture (.+?)$attribute (?:of|on|in) (.+?)$element by javascript "(.+?)"$$$expression""" => step.orDocString(expression) tap { expression =>
        checkStepRules(step, BehaviorType.Action, env)
        val elementBinding = env.getLocatorBinding(element)
        env.scopes.set(s"$attribute/javascript", expression)
        try {
          env.perform {
            env.topScope.pushObject(s"$attribute/javascript/param/webElement", webContext.locate(elementBinding))
          }
          val value = env.getBoundReferenceValue(attribute)
          env.topScope.set(attribute, value tap { content =>
            env.addAttachment(attribute, "txt", content)
          })
        } finally {
          env.perform {
            env.topScope.popObject(s"$attribute/javascript/param/webElement")
          }
        }
      }

      case r"""I capture the current URL""" =>
        checkStepRules(step, BehaviorType.Action, env)
        webContext.captureCurrentUrl(None)

      case r"""I capture the current URL as (.+?)$name""" =>
        checkStepRules(step, BehaviorType.Action, env)
        webContext.captureCurrentUrl(Some(name))

      case r"""I capture the current screenshot""" =>
        checkStepRules(step, BehaviorType.Action, env)
        webContext.captureScreenshot(true)

      case r"""I capture (.+?)$element( value| text)$selection as (.+?)$attribute""" =>
        checkStepRules(step, BehaviorType.Action, env)
        try {
          val value = env.boundAttributeOrSelection(element, Option(selection))
          env.topScope.set(attribute, value() tap { content =>
            env.addAttachment(attribute, "txt", content)
          })
        } catch {
          case _: LocatorBindingException =>
            super.evaluate(step, env)
        }

      case r"""I capture (.+?)$element( value| text)$$$selection""" =>
        checkStepRules(step, BehaviorType.Action, env)
        try {
          val value = env.boundAttributeOrSelection(element, Option(selection))
          env.topScope.set(element, value() tap { content =>
            env.addAttachment(element, "txt", content)
          })
        } catch {
          case _: LocatorBindingException =>
            super.evaluate(step, env)
        }

      case r"I capture the (alert|confirmation)$name popup message" =>
        checkStepRules(step, BehaviorType.Action, env)
        env.topScope.set(s"the $name popup message", webContext.getPopupMessage tap { content =>
          env.addAttachment(s"the $name popup message", "txt", content)
        })

      case r"I capture the (?:alert|confirmation) popup message as (.+?)$attribute" =>
        checkStepRules(step, BehaviorType.Action, env)
        env.topScope.set(attribute, webContext.getPopupMessage tap { content =>
          env.addAttachment(attribute, "txt", content)
        })

      case r"""I start visual test as "(.+?)"$testName in (\d+?)$width x (\d+?)$height viewport""" =>
        checkStepRules(step, BehaviorType.Action, env)
        if (EyesSettings.`gwen.applitools.eyes.enabled`) {
          webContext.startVisualTest(testName, Some(new RectangleSize(width.toInt, height.toInt)))
        } else {
          disabledStepError(step)
        }

      case r"""I start visual test as "(.+?)"$testName""" =>
        checkStepRules(step, BehaviorType.Action, env)
        if (EyesSettings.`gwen.applitools.eyes.enabled`) {
          webContext.startVisualTest(testName, None)
        } else {
          disabledStepError(step)
        }

      case r"""I check (viewport|full page)?$mode visual as "(.+?)"$name using (.+?)$matchLevel match""" =>
        checkStepRules(step, BehaviorType.Action, env)
        if (EyesSettings.`gwen.applitools.eyes.enabled`) {
          webContext.checkVisual(name, mode == "full page", Some(MatchLevel.valueOf(matchLevel)))
        } else {
          disabledStepError(step)
        }

      case r"""I check (viewport|full page)?$mode visual as "(.+?)"$name""" =>
        checkStepRules(step, BehaviorType.Action, env)
        if (EyesSettings.`gwen.applitools.eyes.enabled`) {
          webContext.checkVisual(name, mode == "full page", None)
        } else {
          disabledStepError(step)
        }

      case "the visual test should pass" =>
        checkStepRules(step, BehaviorType.Assertion, env)
        if (EyesSettings.`gwen.applitools.eyes.enabled`) {
          webContext.asertVisuals()
        } else {
          disabledStepError(step)
        }

      case r"""I drag and drop (.+?)$source to (.+?)$$$target""" =>
        checkStepRules(step, BehaviorType.Action, env)
        val sourceBinding = env.getLocatorBinding(source)
        val targetBinding = env.getLocatorBinding(target)
        webContext.dragAndDrop(sourceBinding, targetBinding)

      case r"""I clear (.+?)$$$element""" =>
        checkStepRules(step, BehaviorType.Action, env)
        val elementBinding = env.getLocatorBinding(element)
        webContext.performAction("clear", elementBinding)

      case r"""I press (enter|tab)$key in (.+?)$$$element""" =>
        checkStepRules(step, BehaviorType.Action, env)
        val elementBinding = env.getLocatorBinding(element)
        webContext.sendKeys(elementBinding, Array[String](key))
        env.bindAndWait(element, key, "true")

      case r"""I send "(.+?)"$keys to (.+?)$$$element""" =>
        checkStepRules(step, BehaviorType.Action, env)
        val elementBinding = env.getLocatorBinding(element)
        webContext.sendKeys(elementBinding, keys.split(","))

      case r"""I (enter|type)$action "(.*?)"$value in (.+?)$$$element""" =>
        checkStepRules(step, BehaviorType.Action, env)
        val elementBinding = env.getLocatorBinding(element)
        webContext.sendValue(elementBinding, value, clearFirst = WebSettings.`gwen.web.sendKeys.clearFirst`, sendEnterKey = action == "enter")

      case r"""I (enter|type)$action (.+?)$attribute in (.+?)$$$element""" =>
        checkStepRules(step, BehaviorType.Action, env)
        val elementBinding = env.getLocatorBinding(element)
        val value = env.getBoundReferenceValue(attribute)
        webContext.sendValue(elementBinding, value, clearFirst = WebSettings.`gwen.web.sendKeys.clearFirst`, sendEnterKey = action == "enter")

      case r"""I (select|deselect)$action the (\d+?)$position(?:st|nd|rd|th) option in (.+?)$$$element""" =>
        checkStepRules(step, BehaviorType.Action, env)
        val elementBinding = env.getLocatorBinding(element)
        if (action == "select") {
          webContext.selectByIndex(elementBinding, position.toInt - 1)
        } else {
          webContext.deselectByIndex(elementBinding, position.toInt - 1)
        }

      case r"""I (select|deselect)$action "(.*?)"$value in (.+?)$element by value""" =>
        checkStepRules(step, BehaviorType.Action, env)
        val elementBinding = env.getLocatorBinding(element)
        if (action == "select") {
          webContext.selectByValue(elementBinding, value)
        } else {
          webContext.deselectByValue(elementBinding, value)
        }

      case r"""I (select|deselect)$action "(.*?)"$value in (.+?)$$$element""" =>
        checkStepRules(step, BehaviorType.Action, env)
        val elementBinding = env.getLocatorBinding(element)
        if (action == "select") {
          webContext.selectByVisibleText(elementBinding, value)
        } else {
          webContext.deselectByVisibleText(elementBinding, value)
        }

      case r"""I (select|deselect)$action (.+?)$attribute in (.+?)$element by value""" =>
        checkStepRules(step, BehaviorType.Action, env)
        val value = env.getBoundReferenceValue(attribute)
        val elementBinding = env.getLocatorBinding(element)
        if (action == "select") {
          webContext.selectByValue(elementBinding, value)
        } else {
          webContext.deselectByValue(elementBinding, value)
        }

      case r"""I (select|deselect)$action (.+?)$attribute in (.+?)$$$element""" =>
        checkStepRules(step, BehaviorType.Action, env)
        val value = env.getBoundReferenceValue(attribute)
        val elementBinding = env.getLocatorBinding(element)
        if (action == "select") {
          webContext.selectByVisibleText(elementBinding, value)
        } else {
          webContext.deselectByVisibleText(elementBinding, value)
        }

      case r"""I (click|right click|double click|check|tick|uncheck|untick|move to)$action (.+?)$element of (.+?)$$$context""" =>
        checkStepRules(step, BehaviorType.Action, env)
        webContext.performActionInContext(action, element, context)

      case r"""I (click|right click|double click|submit|check|tick|uncheck|untick|move to)$action (.+?)$$$element""" =>
        checkStepRules(step, BehaviorType.Action, env)
        val elementBinding = env.getLocatorBinding(element)
        webContext.performAction(action, elementBinding)

      case r"""I (.+?)$modifiers (click|right click|double click)$clickAction (.+?)$$$element""" =>
        checkStepRules(step, BehaviorType.Action, env)
        val elementBinding = env.getLocatorBinding(element)
        webContext.holdAndClick(modifiers.split("\\+"), clickAction, elementBinding)

      case r"""I (?:highlight|locate) (.+?)$$$element""" =>
        checkStepRules(step, BehaviorType.Action, env)
        val elementBinding = env.getLocatorBinding(element)
        env.perform {
          webContext.locateAndHighlight(elementBinding)
        }

      case "I refresh the current page" =>
        checkStepRules(step, BehaviorType.Action, env)
        webContext.refreshPage()

      case r"I start a new browser" =>
        checkStepRules(step, BehaviorType.Action, env)
        webContext.close("primary")
        webContext.switchToSession("primary")

      case r"""I start a browser for (.+?)$$$session""" =>
        checkStepRules(step, BehaviorType.Action, env)
        webContext.close(session)
        webContext.switchToSession(session)

      case r"""I should have (\d+?)$count open browser(?:s?)""" =>
        checkStepRules(step, BehaviorType.Assertion, env)
        env.perform {
          env.compare("open browser sessions", count, () => webContext.noOfSessions.toString, "be", false)
        }

      case r"I have (no|an)$open open browser" =>
        checkStepRules(step, BehaviorType.Context, env)
        if (open == "no") {
          webContext.close()
        } else {
          webContext.newOrCurrentSession()
        }

      case r"I close the(?: current)? browser" =>
        checkStepRules(step, BehaviorType.Action, env)
        webContext.close()

      case r"""I close the browser for (.+?)$session""" =>
        checkStepRules(step, BehaviorType.Action, env)
        webContext.close(session)

      case r"""I switch to the child (?:window|tab)""" =>
        checkStepRules(step, BehaviorType.Action, env)
        webContext.switchToChild()

      case r"""I close the child (?:window|tab)""" =>
        checkStepRules(step, BehaviorType.Action, env)
        webContext.closeChild()

      case r"""I switch to the parent (?:window|tab)""" =>
        checkStepRules(step, BehaviorType.Action, env)
        webContext.switchToParent(false)

      case """I switch to the default content""" =>
        checkStepRules(step, BehaviorType.Action, env)
        webContext.switchToDefaultContent()

      case r"""I switch to (.+?)$session""" =>
        checkStepRules(step, BehaviorType.Action, env)
        webContext.switchToSession(session)

      case r"I (accept|dismiss)$action the (?:alert|confirmation) popup" =>
        checkStepRules(step, BehaviorType.Action, env)
        webContext.handleAlert(action == "accept")

      case r"""I resize the window to width (\d+?)$width and height (\d+?)$$$height""" =>
        checkStepRules(step, BehaviorType.Action, env)
        webContext.resizeWindow(width.toInt, height.toInt)

      case r"""I maximi(?:z|s)e the window""" =>
        checkStepRules(step, BehaviorType.Action, env)
        webContext.maximizeWindow()

      case r"""I append "(.+?)"$text to (.+?)$$$element""" =>
        checkStepRules(step, BehaviorType.Action, env)
        val elementBinding = env.getLocatorBinding(element)
        webContext.sendValue(elementBinding, text, clearFirst = false, sendEnterKey = false)

      case r"""I append (.+?)$attribute to (.+?)$$$element""" =>
        checkStepRules(step, BehaviorType.Action, env)
        val elementBinding = env.getLocatorBinding(element)
        val text = env.getBoundReferenceValue(attribute)
        webContext.sendValue(elementBinding, text, clearFirst = false, sendEnterKey = false)

      case r"I insert a new line in (.+?)$$$element" =>
        checkStepRules(step, BehaviorType.Action, env)
        val elementBinding = env.getLocatorBinding(element)
        webContext.sendValue(elementBinding, StringEscapeUtils.unescapeJava("""\n"""), clearFirst = false, sendEnterKey = false)

      case _ => super.evaluate(step, env)

    }
  }
  
  /**
    * Performs a repeat until or while operation 
    */
  private def repeat(operation: String, step: Step, doStep: String, condition: String, delay: Duration, timeout: Duration, env: WebEnvContext): Step = {
    assert(delay.gteq(Duration.Zero), "delay cannot be less than zero")
    assert(timeout.gt(Duration.Zero), "timeout must be greater than zero")
    assert(timeout.gteq(delay), "timeout cannot be less than or equal to delay")
    var evaluatedStep = step
    env.perform {
      var iteration = 0L
      val start = System.nanoTime
      try {
        env.webContext.waitUntil(timeout.toSeconds) {
          iteration = iteration + 1
          env.topScope.set("iteration number", iteration.toString)
          operation match {
            case "until" =>
              logger.info(s"repeat-until $condition: iteration $iteration")
              evaluateStep(Step(step.keyword, doStep), env).evalStatus match {
                case Failed(_, e) => throw e
                case _ =>
                  val javascript = env.interpolate(env.scopes.get(s"$condition/javascript"))(env.getBoundReferenceValue)
                  env.evaluateJSPredicate(javascript) tap { result =>
                    if (!result) {
                      logger.info(s"repeat-until $condition: not complete, will repeat ${if (delay.gt(Duration.Zero)) s"in ${DurationFormatter.format(delay)}" else "now"}")
                      if (delay.gt(Duration.Zero)) Thread.sleep(delay.toMillis)
                    } else {
                      logger.info(s"repeat-until $condition: completed")
                    }
                  }
              }
            case "while" =>
              val javascript = env.interpolate(env.scopes.get(s"$condition/javascript"))(env.getBoundReferenceValue)
              val result = env.evaluateJSPredicate(javascript)
              if (result) {
                logger.info(s"repeat-while $condition: iteration $iteration")
                evaluateStep(Step(step.keyword, doStep), env).evalStatus match {
                  case Failed(_, e) => throw e
                  case _ =>
                    logger.info(s"repeat-while $condition: not complete, will repeat ${if (delay.gt(Duration.Zero)) s"in ${DurationFormatter.format(delay)}" else "now"}")
                    if (delay.gt(Duration.Zero)) Thread.sleep(delay.toMillis)
                }
              } else {
                logger.info(s"repeat-while $condition: completed")
              }
              !result
          }
        }
      } catch {
        case e: Throwable =>
          evaluatedStep = Step(step, Failed(System.nanoTime - start, new StepFailure(step, e)))
      } finally {
        env.topScope.set("iteration number", null)
      }
    } getOrElse {
      try {
        operation match {
          case "until" =>
            evaluatedStep = this.evaluateStep(Step(step.pos, step.keyword, doStep), env)
            env.scopes.get(s"$condition/javascript")
          case _ =>
            env.scopes.get(s"$condition/javascript")
            evaluatedStep = this.evaluateStep(Step(step.pos, step.keyword, doStep), env)
        }
      } catch {
        case _: Throwable => 
          // ignore in dry run mode
      }
    }
    evaluatedStep
  }
  
  lazy val DefaultRepeatDelay: Duration = {
    val waitSecs = WebSettings.`gwen.web.wait.seconds` 
    if (waitSecs > 9 && waitSecs % 10 == 0) Duration(waitSecs / 10, "second") else Duration(waitSecs * 100, "millisecond")
  }
  
  private def defaultRepeatTimeout(delay: Duration): Duration = delay * 30
  
}





© 2015 - 2025 Weber Informatics LLC | Privacy Policy