View Javadoc
1   /**
2    * Copyright 2005-2014 The Kuali Foundation
3    *
4    * Licensed under the Educational Community License, Version 2.0 (the "License");
5    * you may not use this file except in compliance with the License.
6    * You may obtain a copy of the License at
7    *
8    * http://www.opensource.org/licenses/ecl2.php
9    *
10   * Unless required by applicable law or agreed to in writing, software
11   * distributed under the License is distributed on an "AS IS" BASIS,
12   * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13   * See the License for the specific language governing permissions and
14   * limitations under the License.
15   */
16  package org.kuali.rice.testtools.selenium;
17  
18  import com.thoughtworks.selenium.SeleneseTestBase;
19  import org.apache.commons.lang.StringUtils;
20  import org.apache.commons.lang3.exception.ExceptionUtils;
21  import org.openqa.selenium.Alert;
22  import org.openqa.selenium.By;
23  import org.openqa.selenium.Dimension;
24  import org.openqa.selenium.JavascriptExecutor;
25  import org.openqa.selenium.NoSuchFrameException;
26  import org.openqa.selenium.Proxy;
27  import org.openqa.selenium.WebDriver;
28  import org.openqa.selenium.WebElement;
29  import org.openqa.selenium.chrome.ChromeDriver;
30  import org.openqa.selenium.chrome.ChromeDriverService;
31  import org.openqa.selenium.firefox.FirefoxDriver;
32  import org.openqa.selenium.firefox.FirefoxProfile;
33  import org.openqa.selenium.remote.CapabilityType;
34  import org.openqa.selenium.remote.DesiredCapabilities;
35  import org.openqa.selenium.remote.RemoteWebDriver;
36  import org.openqa.selenium.safari.SafariDriver;
37  
38  import java.io.BufferedReader;
39  import java.io.File;
40  import java.io.InputStream;
41  import java.io.InputStreamReader;
42  import java.net.MalformedURLException;
43  import java.net.URL;
44  import java.util.LinkedList;
45  import java.util.List;
46  import java.util.concurrent.TimeUnit;
47  
48  /**
49   * <p>
50   * The goal of the WebDriverUtils class is to invert the dependencies on WebDriver from {@see WebDriverLegacyITBase} for reuse
51   * without having to extend WebDriverLegacyITBase.
52   * </p><p>
53   * For compatibility with {@see JiraAwareFailureUtils}, external test framework asserts and fails should not be called from
54   * WebDriverUtils, instead use {@see JiraAwareAftBase}.
55   * </p><p>
56   * For the first example see waitFor
57   * </p>
58   * @see WebDriverLegacyITBase
59   * @author Kuali Rice Team (rice.collab@kuali.org)
60   */
61  public class WebDriverUtils {
62  
63      protected static SauceLabsWebDriverHelper saucelabs;
64  
65      public static boolean jGrowlEnabled = false;
66  
67      public static boolean jsHighlightEnabled = false;
68  
69      /**
70       * http://localhost:8080/kr-dev
71       */
72      public static final String DEFAULT_BASE_URL = "http://localhost:8080/kr-dev";
73  
74      /**
75       * http://localhost:8080/krad-dev
76       */
77      public static final String DEFAULT_BASE_URL_KRAD = "http://localhost:8080/krad-dev";
78  
79      /**
80       * <p>
81       * Set to true to not close the browser after the test has run.
82       * </p><p>
83       * -Dremote.driver.dontTearDown=true
84       * </p>
85       */
86      public static final String DONT_TEAR_DOWN_PROPERTY = "remote.driver.dontTearDown";
87  
88      /**
89       * <p>
90       * Set to true to not close the browser after the test has if the test failed.
91       * </p><p>
92       * -Dremote.driver.dontTearDownOnFailure=true
93       * </p>
94       */
95      public static final String DONT_TEAR_DOWN_ON_FAILURE_PROPERTY = "remote.driver.dontTearDownOnFailure";
96  
97      /**
98       * remote.public.driver
99       */
100     public static final String HUB_DRIVER_PROPERTY = "remote.public.driver";
101 
102     /**
103      * For use when running Selenium tests through a Selenium Hub.
104      * -Dremote.public.hub=
105      */
106     public static final String HUB_PROPERTY = "remote.public.hub";
107 
108     /**
109      * http://localhost:4444/wd/hub
110      */
111     public static final String HUB_URL_PROPERTY = "http://localhost:4444/wd/hub";
112 
113     /**
114      * wait Methods inter loop sleep period, default of 1000 Milliseconds.
115      * TODO parametrize for JVM Arg
116      */
117     public static int IMPLICIT_WAIT_TIME_LOOP_MS = 1000;
118 
119     /**
120      * <p>
121      * {@see IMPLICIT_WAIT_TIME_SECONDS_DEFAULT} to configure, default 30 seconds.
122      * </p><p>
123      * In code don't use this variable but call {@see configuredImplicityWait} to get the configured value.
124      * </p>
125      */
126     public static int IMPLICIT_WAIT_TIME_SECONDS_DEFAULT = 30;
127 
128     /**
129      * If true tests will fail on jGrowl errors, default of false
130      * TODO upgrade to config via JVM param.
131      */
132     public static final boolean JGROWL_ERROR_FAILURE = false;
133 
134     /**
135      * green (#66FF33)
136      */
137     public static final String JS_HIGHLIGHT_BACKGROUND = "#66FF33";
138 
139     /**
140      * green (#66FF33)
141      */
142     public static final String JS_HIGHLIGHT_BOARDER = "#66FF33";
143 
144     /**
145      * 400 milliseconds.
146      */
147     public static final int JS_HIGHLIGHT_MS = 400;
148 
149     /**
150      * <p>
151      * {@see JS_HIGHLIGHT_MS} as default.
152      * </p><p>
153      * -Dremote.driver.highlight.ms=
154      * </p>
155      */
156     public static final String JS_HIGHLIGHT_MS_PROPERTY = "remote.driver.highlight.ms";
157 
158     /**
159      * <p>
160      * Highlighting of elements as selenium runs.
161      * </p><p>
162      * -Dremote.driver.highlight=true
163      * </p>
164      */
165     public static final String JS_HIGHLIGHT_PROPERTY = "remote.driver.highlight";
166 
167     /**
168      * TODO: playback for javascript highlighting.
169      *
170      * -Dremote.driver.highlight.input=
171      */
172     public static final String JS_HIGHLIGHT_INPUT_PROPERTY = "remote.driver.highlight.input";
173 
174     /**
175      * <p>
176      * Local proxy used for running tests thru jmeter.
177      * </p><p>
178      * Include host name and port number. Example: localhost:7777
179      * </p><p>
180      * -Dremote.public.proxy=
181      * </p>
182      */
183     public static final String PROXY_HOST_PROPERTY = "remote.public.proxy";
184 
185     /**
186      * <p>
187      * Skip automatice login if set to anything other than true/
188      * </p><p>
189      * -Dremote.autologin=false
190      * </p>
191      */
192     public static final String REMOTE_AUTOLOGIN_PROPERTY = "remote.autologin";
193 
194     /**
195      * <p>
196      * Set to true to enable jGrowl test messages.
197      * </p><p>
198      * When enabled, jGrowl messages will be sent when clicking on buttons and links identified by their text.
199      * </p><p>
200      * -Dremote.jgrowl.enabled=true
201      * </p>
202      */
203     public static final String REMOTE_JGROWL_ENABLED = "remote.jgrowl.enabled";
204 
205     /**
206      * Set -Dremote.login.uif=KNS to use old login screen.  Default value = KRAD
207      */
208     public static final String REMOTE_LOGIN_UIF = "remote.login.uif";
209 
210     /**
211      * Set -Dremote.property.file= to load proprties from file
212      */
213     public static final String REMOTE_PROPERTIES_PROPERTY = "remote.property.file";
214 
215     /**
216      * Set -Dremote.public.chrome= or WEBDRIVER_CHROME_DRIVER
217      */
218     public static final String REMOTE_PUBLIC_CHROME = "remote.public.chrome";
219 
220     /**
221      * -Dremote.public.url=
222      */
223     public static final String REMOTE_PUBLIC_URL_PROPERTY = "remote.public.url";
224 
225     /**
226      * <p>
227      * Set -Dremote.public.wait.seconds to override DEFAULT_WAIT_SEC.
228      * </p><p>
229      * {@see IMPLICIT_WAIT_TIME_SECONDS_DEFAULT}
230      * </p>
231      */
232     public static final String REMOTE_PUBLIC_WAIT_SECONDS_PROPERTY = "remote.public.wait.seconds";
233 
234     /**
235      * Set -Dremote.public.user= to the username to login as
236      */
237     public static final String REMOTE_PUBLIC_USER_PROPERTY = "remote.public.user";
238 
239     /**
240      * You probably don't want to really be using a userpool, set -Dremote.public.userpool= to base url if you must.
241      */
242     public static final String REMOTE_PUBLIC_USERPOOL_PROPERTY = "remote.public.userpool";
243 
244     /**
245      * <p>
246      * Time to wait for the URL used in setup to load, 120 seconds by default.
247      * </p><p>
248      * Sometimes this is the first hit on the app and it needs a bit longer than any other.
249      * </p><p>
250      * TODO parametrize for JVM Arg
251      * <p>
252      */
253     public static final int SETUP_URL_LOAD_WAIT_SECONDS = 120;
254 
255     /**
256      * Selenium's webdriver.chrome.driver parameter, you can set -Dwebdriver.chrome.driver= or Rice's REMOTE_PUBLIC_CHROME.
257      */
258     public static final String WEBDRIVER_CHROME_DRIVER = "webdriver.chrome.driver";
259 
260     /**
261      * Setup the WebDriver test, login, and load the given web page
262      *
263      * @param username
264      * @param url
265      * @return driver
266      * @throws Exception
267      */
268     public static WebDriver setUp(String username, String url) throws Exception {
269         return setUp(username, url, null, null);
270     }
271 
272     /**
273      * Setup the WebDriver test, login, and load the given web page
274      *
275      * @param username
276      * @param url
277      * @param className
278      * @param testName
279      * @return driver
280      * @throws Exception
281      */
282     public static WebDriver setUp(String username, String url, String className, String testName) throws Exception {
283         if ("true".equals(System.getProperty(REMOTE_JGROWL_ENABLED, "false"))) {
284             jGrowlEnabled = true;
285         }
286 
287         if ("true".equals(System.getProperty(JS_HIGHLIGHT_PROPERTY, "false"))) {
288             jsHighlightEnabled = true;
289             if (System.getProperty(JS_HIGHLIGHT_INPUT_PROPERTY) != null) {
290                 InputStream in = WebDriverUtils.class.getResourceAsStream(System.getProperty(JS_HIGHLIGHT_INPUT_PROPERTY));
291                 BufferedReader reader = new BufferedReader(new InputStreamReader(in));
292                 String line = null;
293                 List<String> lines = new LinkedList<String>();
294                 while ((line = reader.readLine()) != null) {
295                     lines.add(line);
296                 }
297             }
298         }
299 
300         WebDriver driver = null;
301         if (System.getProperty(SauceLabsWebDriverHelper.REMOTE_DRIVER_SAUCELABS_PROPERTY) == null) {
302             driver = getWebDriver();
303         } else {
304             saucelabs = new SauceLabsWebDriverHelper();
305             saucelabs.setUp(className, testName);
306             driver = saucelabs.getDriver();
307         }
308 
309         driver.manage().timeouts().implicitlyWait(SETUP_URL_LOAD_WAIT_SECONDS, TimeUnit.SECONDS);
310 
311         if (!System.getProperty(SauceLabsWebDriverHelper.SAUCE_BROWSER_PROPERTY,"ff").equals("opera")) {
312             driver.manage().window().maximize();
313 //            driver.manage().window().setSize(new Dimension(800,600));
314         }
315 
316         // TODO Got into the situation where the first url doesn't expect server, but all others do.  Readdress once
317         // the NavIT WDIT conversion has been completed.
318         if (!url.startsWith("http")) {
319             url = getBaseUrlString() + url;
320         }
321 
322         driver.get(url);
323         driver.manage().timeouts().implicitlyWait(configuredImplicityWait(), TimeUnit.SECONDS);
324         return driver;
325     }
326 
327     /**
328      *<p>
329      * Calls {@see SauceLabsWebDriverHelper#tearDown} if {@see #REMOTE_PUBLIC_USERPOOL_PROPERTY} is enabled, calls a user pool
330      * url with the given poolParamTest and poolParamUser.
331      *</p>
332      *
333      * @param passed used by {@see SauceLabsWebDriverHelper#tearDown} to record Saucelabs test status can be null if Saucelabs
334      * is not being used
335      * @param sessionId used by {@see SauceLabsWebDriverHelper#tearDown} to record Saucelabs sessionId status can be null if Saucelabs
336      * is not being used
337      * @param poolParamTest can be null unless a user pool is being used
338      * @param poolParamUser can be null unless a user pool is being used
339      * @throws Exception
340      */
341     public static void tearDown(boolean passed, String sessionId, String poolParamTest, String poolParamUser, String className, String testName) throws Exception {
342 
343         if (passed) {
344             System.out.println("Registering session passed " + sessionId);
345         } else {
346             System.out.println("Registering session failed " + sessionId);
347         }
348 
349         if (System.getProperty(SauceLabsWebDriverHelper.REMOTE_DRIVER_SAUCELABS_PROPERTY) != null) {
350             saucelabs.tearDown(passed, sessionId, className, testName);
351         }
352 
353         if (System.getProperty(REMOTE_PUBLIC_USERPOOL_PROPERTY) != null) {
354             AutomatedFunctionalTestUtils.getHTML(AutomatedFunctionalTestUtils.prettyHttp(System.getProperty(
355                     REMOTE_PUBLIC_USERPOOL_PROPERTY) + "?test=" + poolParamTest + "&user=" + poolParamUser));
356         }
357     }
358 
359     /**
360      * <p>
361      * If an alert is present accept it print the alert text to System.out
362      * </p>
363      *
364      * @param driver to accept alert on
365      */
366     public static void acceptAlertIfPresent(WebDriver driver) {
367         if (WebDriverUtils.isAlertPresent(driver)) {
368             System.out.println("Alert present " + WebDriverUtils.alertText(driver));
369             alertAccept(driver);
370         }
371     }
372 
373     /**
374      * <p>
375      * Accept the javascript alert (clicking OK).
376      * </p>
377      *
378      * @param driver WebDriver to accept alert on
379      */
380     public static void alertAccept(WebDriver driver) {
381         Alert alert = driver.switchTo().alert();
382         jGrowl(driver, "AFT Step", false, "AFT Step: Accept Alert " + WebDriverUtils.alertText(driver));
383         alert.accept();
384     }
385 
386     /**
387      * <p>
388      * Dismiss the javascript alert (clicking Cancel).
389      * </p>
390      *
391      * @param driver WebDriver to dismiss alert on
392      */
393     public static void alertDismiss(WebDriver driver) {
394         Alert alert = driver.switchTo().alert();
395         jGrowl(driver, "AFT Step", false, "AFT Step: Dismiss Alert " + WebDriverUtils.alertText(driver));
396         alert.dismiss();
397     }
398 
399     /**
400      * <p>
401      * Return alert text.
402      * </p>
403      *
404      * @param driver to get alert text from
405      * @return alert text
406      */
407     public static String alertText(WebDriver driver) {
408         return driver.switchTo().alert().getText();
409     }
410 
411     /**
412      * <p>
413      * <a href="http://code.google.com/p/chromedriver/downloads/list">ChromeDriver downloads</a>, {@see #REMOTE_PUBLIC_CHROME},
414      * {@see #WEBDRIVER_CHROME_DRIVER}, and {@see #HUB_DRIVER_PROPERTY}
415      * </p>
416      *
417      * @return chromeDriverService
418      */
419     public static ChromeDriverService chromeDriverCreateCheck() {
420         String driverParam = System.getProperty(HUB_DRIVER_PROPERTY);
421         // TODO can the saucelabs driver stuff be leveraged here?
422         if (driverParam != null && "chrome".equals(driverParam.toLowerCase())) {
423             if (System.getProperty(WEBDRIVER_CHROME_DRIVER) == null) {
424                 if (System.getProperty(REMOTE_PUBLIC_CHROME) != null) {
425                     System.setProperty(WEBDRIVER_CHROME_DRIVER, System.getProperty(REMOTE_PUBLIC_CHROME));
426                 }
427             }
428             try {
429                 ChromeDriverService chromeDriverService = new ChromeDriverService.Builder()
430                         .usingDriverExecutable(new File(System.getProperty(WEBDRIVER_CHROME_DRIVER)))
431                         .usingAnyFreePort()
432                         .build();
433                 return chromeDriverService;
434             } catch (Throwable t) {
435                 throw new RuntimeException("Exception starting chrome driver service, is chromedriver ( http://code.google.com/p/chromedriver/downloads/list ) installed? You can include the path to it using -Dremote.public.chrome", t)   ;
436             }
437         }
438         return null;
439     }
440 
441     /**
442      * <p>
443      * Return the configured implicity wait seconds, {@see #REMOTE_PUBLIC_WAIT_SECONDS_PROPERTY} and {@see #IMPLICIT_WAIT_TIME_SECONDS_DEFAULT}.
444      * </p>
445      *
446      * @return seconds for implicity wait
447      */
448     public static int configuredImplicityWait() {
449         return Integer.parseInt(System.getProperty(REMOTE_PUBLIC_WAIT_SECONDS_PROPERTY, IMPLICIT_WAIT_TIME_SECONDS_DEFAULT + ""));
450     }
451 
452     /**
453      * <p>
454      * Remove double line spacing.
455      * </p>
456      *
457      * @param contents String to remove double line spacing from
458      * @return String with double line spacing removed.
459      */
460     public static String deLinespace(String contents) {
461         while (contents.contains("\n\n")) {
462             contents = contents.replaceAll("\n\n", "\n");
463         }
464 
465         return contents;
466     }
467 
468     /**
469      * <p>
470      * If {@see #REMOTE_PUBLIC_USER_PROPERTY} property is set, return its value, else if {@see #REMOTE_PUBLIC_USERPOOL_PROPERTY}
471      * is set use it to query the userpool service passing the testParam.
472      * </p>
473      * *
474      * @param testParam to use if using a user pool
475      * @return user
476      */
477     public static String determineUser(String testParam) {
478         String user = null;
479 
480         if (System.getProperty(REMOTE_PUBLIC_USER_PROPERTY) != null) {
481             return System.getProperty(REMOTE_PUBLIC_USER_PROPERTY);
482         } else if (System.getProperty(REMOTE_PUBLIC_USERPOOL_PROPERTY) != null) { // deprecated
483             String userResponse = AutomatedFunctionalTestUtils.getHTML(AutomatedFunctionalTestUtils.prettyHttp(
484                     System.getProperty(REMOTE_PUBLIC_USERPOOL_PROPERTY) + "?test=" + testParam.trim()));
485             return userResponse.substring(userResponse.lastIndexOf(":") + 2, userResponse.lastIndexOf("\""));
486         }
487 
488         return user;
489     }
490 
491     /**
492      * <p>
493      * Setting the JVM arg remote.driver.dontTearDown to y or t leaves the browser window open when the test has completed.
494      * <p></p>
495      * Valuable when debugging, updating, or creating new tests.  When implementing your own tearDown method rather than an
496      * inherited one, it is a common courtesy to include this check and not stop and shutdown the browser window to make it
497      * easy debug or update your test.
498      * </p>
499      *
500      * @return true if the dontTearDownProperty is not set.
501      */
502     public static boolean dontTearDownPropertyNotSet() {
503         return System.getProperty(DONT_TEAR_DOWN_PROPERTY) == null ||
504                 "f".startsWith(System.getProperty(DONT_TEAR_DOWN_PROPERTY).toLowerCase()) ||
505                 "n".startsWith(System.getProperty(DONT_TEAR_DOWN_PROPERTY).toLowerCase());
506     }
507 
508     /**
509      * Given the boolean parameter and depending on if {@see #DONT_TEAR_DOWN_ON_FAILURE_PROPERTY} is set to something other
510      * than n, don't tear down the browser window on a test failure.
511 
512      * @param passed
513      * @return
514      */
515     public static boolean dontTearDownOnFailure(boolean passed) {
516         if (!"n".equalsIgnoreCase(System.getProperty(DONT_TEAR_DOWN_ON_FAILURE_PROPERTY, "n"))) {
517             return passed;
518         }
519         return true;
520     }
521 
522     /**
523      * <p>
524      * Find Button by text.
525      * </p>
526      *
527      * @param driver to find button on
528      * @param buttonText text to find button by
529      * @return WebElement of button with button text
530      */
531     public static WebElement findButtonByText(WebDriver driver, String buttonText) {
532         return findElement(driver, By.xpath("//button[contains(text(), '" + buttonText + "')]"));
533     }
534 
535     /**
536      * <p>
537      * Find and highlight the WebElement using the given WebDriver and By.
538      * </p>
539      *
540      * @param driver driver to find on
541      * @param by selector to find
542      * @return
543      */
544     public static WebElement findElement(WebDriver driver, By by) {
545         WebElement found = driver.findElement(by);
546         WebDriverUtils.highlightElement(driver, found);
547         return found;
548     }
549 
550     /**
551      * <p>
552      * In order to run as a smoke test the ability to set the baseUrl via the JVM arg remote.public.url is required.
553      * </p><p>
554      * Trailing slashes are trimmed.  If the remote.public.url does not start with http:// it will be added.
555      * </p>
556      *
557      * @return http://localhost:8080/kr-dev by default else the value of remote.public.url
558      */
559     public static String getBaseUrlString() {
560         String baseUrl = System.getProperty(REMOTE_PUBLIC_URL_PROPERTY);
561         if (baseUrl == null) {
562             baseUrl = DEFAULT_BASE_URL;
563         }
564         baseUrl = AutomatedFunctionalTestUtils.prettyHttp(baseUrl);
565         return baseUrl;
566     }
567 
568     /**
569      * <p>
570      * Return the WebElement that has the attribute name with the given value.
571      * </p>
572      *
573      * @param driver to get element from
574      * @param attributeName attribute name to find element by
575      * @param value for the attribute name to find element by
576      * @return WebElement
577      */
578     public static WebElement getElementByAttributeValue(WebDriver driver, String attributeName, String value){
579         return findElement(driver, By.cssSelector("[" + attributeName + "='" + value +"']"));
580     }
581 
582     /**
583      * <p>
584      * In order to run as a smoke test under selenium grid the ability to set the hubUrl via the JVM arg remote.public.hub is required.
585      * </p><p>
586      * Trailing slashes are trimmed.  If the remote.public.hub does not start with http:// it will be added.
587      * </p>
588      *
589      * @return http://localhost:4444/wd/hub by default else the value of remote.public.hub
590      */
591     public static String getHubUrlString() {
592         String hubUrl = System.getProperty(HUB_PROPERTY);
593         if (hubUrl == null) {
594             hubUrl = HUB_URL_PROPERTY;
595         }
596         hubUrl = AutomatedFunctionalTestUtils.prettyHttp(hubUrl);
597         if (!hubUrl.endsWith("/wd/hub")) {
598             hubUrl = hubUrl + "/wd/hub";
599         }
600         return hubUrl;
601     }
602 
603     /**
604      * <p>
605      * remote.public.driver set to chrome or firefox (null assumes firefox).
606      * </p><p>
607      * if remote.public.hub is set a RemoteWebDriver is created (Selenium Grid)
608      * if proxy.host is set, the web driver is setup to use a proxy
609      * </p>
610      *
611      * @return WebDriver or null if unable to create
612      */
613     public static WebDriver getWebDriver() {
614         String driverParam = System.getProperty(HUB_DRIVER_PROPERTY);
615         String hubParam = System.getProperty(HUB_PROPERTY);
616         String proxyParam = System.getProperty(PROXY_HOST_PROPERTY);
617 
618         // setup proxy if specified as VM Arg
619         DesiredCapabilities capabilities = new DesiredCapabilities();
620         WebDriver webDriver = null;
621         if (StringUtils.isNotEmpty(proxyParam)) {
622             capabilities.setCapability(CapabilityType.PROXY, new Proxy().setHttpProxy(proxyParam));
623         }
624 
625         if (hubParam == null) {
626             if (driverParam == null || "firefox".equalsIgnoreCase(driverParam)) {
627                 FirefoxProfile profile = new FirefoxProfile();
628                 profile.setEnableNativeEvents(false);
629                 capabilities.setCapability(FirefoxDriver.PROFILE, profile);
630                 return new FirefoxDriver(capabilities);
631             } else if ("chrome".equalsIgnoreCase(driverParam)) {
632                 return new ChromeDriver(capabilities);
633             } else if ("safari".equals(driverParam)) {
634                 System.out.println("SafariDriver probably won't work, if it does please contact Erik M.");
635                 return new SafariDriver(capabilities);
636             }
637         } else {
638             try {
639                 if (driverParam == null || "firefox".equalsIgnoreCase(driverParam)) {
640                     return new RemoteWebDriver(new URL(getHubUrlString()), DesiredCapabilities.firefox());
641                 } else if ("chrome".equalsIgnoreCase(driverParam)) {
642                     return new RemoteWebDriver(new URL(getHubUrlString()), DesiredCapabilities.chrome());
643                 }
644             } catch (MalformedURLException mue) {
645                 System.out.println(getHubUrlString() + " " + mue.getMessage());
646                 mue.printStackTrace();
647             }
648         }
649         return null;
650     }
651 
652     public static void highlightElement(WebDriver webDriver, By by) {
653         List<WebElement> elements = webDriver.findElements(by);
654         for (WebElement element : elements) {
655             WebDriverUtils.highlightElement(webDriver, element);
656         }
657     }
658 
659 
660     public static void highlightElements(WebDriver webDriver, List<WebElement> webElements) {
661         for (WebElement webElement: webElements) {
662             highlightElement(webDriver, webElement);
663         }
664     }
665 
666     /**
667      * <p>
668      * Highlight given WebElement.
669      * </p>
670      *
671      * @param webDriver to execute highlight on
672      * @param webElement to highlight
673      */
674     public static void highlightElement(WebDriver webDriver, WebElement webElement) {
675         if (jsHighlightEnabled && webElement != null) {
676             try {
677                 //                System.out.println("highlighting " + webElement.toString() + " on url " + webDriver.getCurrentUrl());
678                 JavascriptExecutor js = (JavascriptExecutor) webDriver;
679                 String jsHighlight = "element = arguments[0];\n"
680                         + "originalStyle = element.getAttribute('style');\n"
681                         + "element.setAttribute('style', originalStyle + \"; background: "
682                         + JS_HIGHLIGHT_BACKGROUND + "; border: 2px solid " + JS_HIGHLIGHT_BOARDER + ";\");\n"
683                         + "setTimeout(function(){\n"
684                         + "    element.setAttribute('style', originalStyle);\n"
685                         + "}, " + System.getProperty(JS_HIGHLIGHT_MS_PROPERTY, JS_HIGHLIGHT_MS + "") + ");";
686                 js.executeScript(jsHighlight, webElement);
687             } catch (Throwable t) {
688                 System.out.println("Throwable during javascript highlight element");
689                 t.printStackTrace();
690             }
691         }
692     }
693 
694     /**
695      * <p>
696      * Return true if an alert is present, false if not.
697      * </p>
698      *
699      * @param driver to check for presents of alert on
700      * @return true if there is an alert present, false if not
701      */
702     public static boolean isAlertPresent(WebDriver driver) {
703         try {
704             driver.switchTo().alert();
705             return true;
706         } catch (Exception e) {
707             return false;
708         }
709     }
710 
711     public static Boolean isTextPresent(WebDriver driver, String pageText, String text) {
712         boolean textPresent = Boolean.FALSE;
713         if (pageText.contains(text)) {
714             WebDriverUtils.highlightElement(driver, By.xpath("//*[contains(text(), '" + text + "')]"));
715             textPresent = Boolean.TRUE;
716         }
717         WebDriverUtils.jGrowl(driver, "Is Text Present?", false, "Is text '" + text + " present?" + " " + textPresent);
718         return textPresent;
719     }
720 
721     /**
722      * <p>
723      * Display jGrowl.
724      * </p>
725      *
726      * @param driver WebDriver to execute jGrowl on
727      * @param jGrowlHeader header text for jGrowl
728      * @param sticky true to set the jGrowl to sticky, false for not sticky
729      * @param message message to display in the jGrowl
730      * @param throwable message and stacktrace to included in jGrowl
731      */
732     public static void jGrowl(WebDriver driver, String jGrowlHeader, boolean sticky, String message, Throwable throwable) {
733         if (jGrowlEnabled) { // check if jGrowl is enabled to skip over the stack trace extraction if it is not.
734             jGrowl(driver, jGrowlHeader, sticky, message + " " + throwable.getMessage() + "\n" + ExceptionUtils.getStackTrace(throwable));
735         }
736     }
737 
738     /**
739      * <p>
740      * Display jGrowl.
741      * </p>
742      *
743      * @param driver WebDriver to execute jGrowl on
744      * @param jGrowlHeader header text for jGrowl
745      * @param sticky true to set the jGrowl to sticky, false for not sticky
746      * @param message message to display in the jGrowl
747      */
748     public static void jGrowl(WebDriver driver, String jGrowlHeader, boolean sticky, String message) {
749         stepMessage(message);
750         if (jGrowlEnabled) {
751             try {
752                 String javascript="jQuery.jGrowl('" + message + "' , {sticky: " + sticky + ", header : '" + jGrowlHeader + "'});";
753                 ((JavascriptExecutor) driver).executeScript(javascript);
754             } catch (Throwable t) {
755                 jGrowlException(t);
756             }
757         }
758     }
759 
760     /**
761      * <p>
762      * Print jGrowl Excetion to System.out, if {@see #JGROWL_ERROR_FAILURE} is set to true, fail.
763      * </p>
764      *
765      * @param throwable message and stack trace to print and if configured fail with
766      */
767     public static void jGrowlException(Throwable throwable) {
768         String failMessage = throwable.getMessage() + "\n" + ExceptionUtils.getStackTrace(throwable);
769         System.out.println("jGrowl failure " + failMessage);
770         if (JGROWL_ERROR_FAILURE) {
771             SeleneseTestBase.fail(failMessage); // SeleneseTestBase fail okay here as jGrowl failures are not Jira worthy yet
772         }
773     }
774 
775     /**
776      * <p>
777      * Select frame defined by locator without throwing an Exception if it doesn't exist.
778      * </p>
779      *
780      * @param driver to select frame on
781      * @param locator to identify frame to select
782      */
783     public static void selectFrameSafe(WebDriver driver, String locator) {
784         try {
785             driver.switchTo().frame(locator);
786         } catch (NoSuchFrameException nsfe) {
787             // don't fail
788         }
789     }
790 
791     public static void stepMessage(String message) {
792         System.out.println("AFT Step: " + message);
793     }
794 
795     /**
796      * <p>
797      * Return the WebElement that has the attribute name with the given value within the given seconds to wait.
798      * </p>
799      *
800      * @param driver to get element from
801      * @param attribute name to find element by
802      * @param attributeValue for the attribute name to find element by
803      * @param waitSeconds number of seconds to wait
804      * @return WebElement
805      * @throws InterruptedException
806      */
807     public static WebElement waitAndGetElementByAttributeValue(WebDriver driver, String attribute, String attributeValue, int waitSeconds) throws InterruptedException {
808         // jenkins implies that implicitlyWait is worse than sleep loop for finding elements by 100+ test failures on the old sampleapp
809         //        driver.manage().timeouts().implicitlyWait(waitSeconds, TimeUnit.SECONDS);
810         //        driver.manage().timeouts().implicitlyWait(1, TimeUnit.SECONDS);
811 
812         driver.manage().timeouts().implicitlyWait(1, TimeUnit.SECONDS);
813 
814         boolean failed = false;
815 
816         for (int second = 0;; second++) {
817             Thread.sleep(1000);
818             if (second >= waitSeconds) {
819                 failed = true;
820             }
821             try {
822                 if (failed || (getElementByAttributeValue(driver, attribute, attributeValue) != null)) {
823                     break;
824                 }
825             } catch (Exception e) {}
826         }
827 
828         WebElement element = getElementByAttributeValue(driver, attribute, attributeValue);
829         driver.manage().timeouts().implicitlyWait(WebDriverUtils.configuredImplicityWait(), TimeUnit.SECONDS);
830         return element;
831     }
832 
833     public static void waitToAcceptAlert(WebDriver driver, int waitSeconds, String message) throws InterruptedException {
834         driver.manage().timeouts().implicitlyWait(IMPLICIT_WAIT_TIME_LOOP_MS, TimeUnit.MILLISECONDS);
835 
836         boolean failed = false;
837 
838         for (int second = 0;; second++) {
839             Thread.sleep(1000);
840             if (second >= waitSeconds) {
841                 failed = true;
842             }
843             try {
844                 if (failed) {
845                     break;
846                 } else if (isAlertPresent(driver)) {
847                     acceptAlertIfPresent(driver);
848                     break;
849                 }
850             } catch (Exception e) {}
851         }
852 
853         driver.manage().timeouts().implicitlyWait(configuredImplicityWait(), TimeUnit.SECONDS);
854     }
855 
856     /**
857      * <p>
858      * Wait for the given amount of seconds, for the given by, using the given driver.  The message is displayed if the
859      * by cannot be found.  No action is performed on the by, so it is possible that the by found is not visible or enabled.
860      * </p>
861      *
862      * @param driver WebDriver to wait on
863      * @param waitSeconds seconds to wait
864      * @param by By to wait for
865      * @param message to display if by is not found in waitSeconds
866      * @throws InterruptedException
867      */
868     public static WebElement waitFor(WebDriver driver, int waitSeconds, By by, String message) throws InterruptedException {
869         // jenkins implies that implicitlyWait is worse than sleep loop for finding elements by 100+ test failures on the old sampleapp
870         //        driver.manage().timeouts().implicitlyWait(waitSeconds, TimeUnit.SECONDS);
871         //        driver.manage().timeouts().implicitlyWait(1, TimeUnit.SECONDS);
872 
873         driver.manage().timeouts().implicitlyWait(IMPLICIT_WAIT_TIME_LOOP_MS, TimeUnit.MILLISECONDS);
874 
875         boolean failed = false;
876         WebElement element = null;
877 
878         for (int second = 0;; second++) {
879             Thread.sleep(1000);
880             if (second >= waitSeconds) {
881                 failed = true;
882             }
883             try {
884                 if (failed) {
885                     break;
886                 } else if ((driver.findElements(by)).size() > 0) {
887                     element = findElement(driver, by);  // NOTICE just the find, no action, so by is found, but might not be visible or enabled.
888                     highlightElement(driver, element);
889                     break;
890                 }
891             } catch (Exception e) {}
892         }
893 
894         driver.manage().timeouts().implicitlyWait(configuredImplicityWait(), TimeUnit.SECONDS);
895         return element;
896     }
897 
898     /**
899      * <p>
900      * Wait for WebElements.
901      * </p>
902      *
903      * @param driver WebDriver to wait on
904      * @param by By to wait for
905      * @return
906      * @throws InterruptedException
907      */
908     public static List<WebElement> waitFors(WebDriver driver, By by) throws InterruptedException {
909         return waitFors(driver, configuredImplicityWait(), by, "");
910     }
911 
912     /**
913      * <p>
914      * Wait for WebElements.
915      * </p>
916      *
917      * @param driver WebDriver to wait on
918      * @param by By to wait for
919      * @param message to display if by is not found in waitSeconds
920      * @return List of WebElements found
921      * @throws InterruptedException
922      */
923     public static List<WebElement> waitFors(WebDriver driver, By by, String message) throws InterruptedException {
924         return waitFors(driver, configuredImplicityWait(), by, message);
925     }
926 
927    /**
928     * <p>
929     * Wait for the given amount of seconds, for the given by, using the given driver.  The message is displayed if the
930     * by cannot be found.  No action is performed on the by, so it is possible that the by found is not visible or enabled.
931     * </p>
932     *
933     * @param driver WebDriver to wait on
934     * @param waitSeconds seconds to wait
935     * @param by By to wait for
936     * @param message to display if by is not found in waitSeconds
937     * @return List of WebElements found
938     * @throws InterruptedException
939     */
940     public static List<WebElement> waitFors(WebDriver driver, int waitSeconds, By by, String message) throws InterruptedException {
941         // jenkins implies that implicitlyWait is worse than sleep loop for finding elements by 100+ test failures on the old sampleapp
942         //        driver.manage().timeouts().implicitlyWait(waitSeconds, TimeUnit.SECONDS);
943         //        driver.manage().timeouts().implicitlyWait(1, TimeUnit.SECONDS);
944 
945         driver.manage().timeouts().implicitlyWait(IMPLICIT_WAIT_TIME_LOOP_MS, TimeUnit.MILLISECONDS);
946 
947         boolean failed = false;
948 
949         for (int second = 0;; second++) {
950             Thread.sleep(1000);
951             if (second >= waitSeconds) {
952                 failed = true;
953             }
954             try {
955                 if (failed || (driver.findElements(by)).size() > 0) {
956                     break;
957                 }
958             } catch (Exception e) {}
959         }
960 
961         driver.manage().timeouts().implicitlyWait(configuredImplicityWait(), TimeUnit.SECONDS);
962         return driver.findElements(by);  // NOTICE just the find, no action, so by is found, but might not be visible or enabled.
963     }
964 }