001    /*
002     * Copyright 2005-2013 The Kuali Foundation
003     *
004     * Licensed under the Educational Community License, Version 2.0 (the "License");
005     * you may not use this file except in compliance with the License.
006     * You may obtain a copy of the License at
007     *
008     * http://www.opensource.org/licenses/ecl2.php
009     *
010     * Unless required by applicable law or agreed to in writing, software
011     * distributed under the License is distributed on an "AS IS" BASIS,
012     * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
013     * See the License for the specific language governing permissions and
014     * limitations under the License.
015     */
016    package edu.samplu.common;
017    
018    import org.apache.commons.lang.StringUtils;
019    import org.apache.commons.lang3.exception.ExceptionUtils;
020    import org.openqa.selenium.By;
021    import org.openqa.selenium.JavascriptExecutor;
022    import org.openqa.selenium.NoSuchFrameException;
023    import org.openqa.selenium.Proxy;
024    import org.openqa.selenium.WebDriver;
025    import org.openqa.selenium.WebElement;
026    import org.openqa.selenium.chrome.ChromeDriver;
027    import org.openqa.selenium.chrome.ChromeDriverService;
028    import org.openqa.selenium.firefox.FirefoxDriver;
029    import org.openqa.selenium.firefox.FirefoxProfile;
030    import org.openqa.selenium.remote.CapabilityType;
031    import org.openqa.selenium.remote.DesiredCapabilities;
032    import org.openqa.selenium.remote.RemoteWebDriver;
033    import org.openqa.selenium.safari.SafariDriver;
034    
035    import com.thoughtworks.selenium.SeleneseTestBase;
036    
037    import java.io.File;
038    import java.net.MalformedURLException;
039    import java.net.URL;
040    import java.util.concurrent.TimeUnit;
041    
042    /**
043     * The goal of the WebDriverUtil class is to invert the dependencies on WebDriver from WebDriverLegacyITBase for reuse
044     * without having to extend WebDriverLegacyITBase.  For the first example see waitFor
045     *
046     * @see WebDriverLegacyITBase
047     * @author Kuali Rice Team (rice.collab@kuali.org)
048     */
049    public class WebDriverUtil {
050    
051        public static boolean jGrowlEnabled = false;
052    
053        public static boolean jsHighlightEnabled = false;
054    
055        /**
056         * TODO apparent dup WebDriverITBase.DEFAULT_WAIT_SEC
057         * TODO parametrize for JVM Arg
058         * 30 Seconds
059         */
060        public static int DEFAULT_IMPLICIT_WAIT_TIME = 30;
061    
062        /**
063         * false
064         * TODO upgrade to config via JVM param.
065         */
066        public static final boolean JGROWL_ERROR_FAILURE = false;
067    
068        /**
069         * green
070         */
071        public static final String JS_HIGHLIGHT_BACKGROUND = "green";
072    
073        /**
074         * green
075         */
076        public static final String JS_HIGHLIGHT_BOARDER = "green";
077    
078        /**
079         * 400 milliseconds
080         */
081        public static final int JS_HIGHLIGHT_MS = 400;
082    
083        /**
084         * -Dremote.driver.highlight=true to enable highlighting of elements as selenium runs
085         */
086        public static final String JS_HIGHLIGHT_PROPERTY = "remote.driver.highlight";
087    
088        /**
089         * TODO introduce SHORT_IMPLICIT_WAIT_TIME with param in WebDriverITBase
090         * TODO parametrize for JVM Arg
091         * 1 Second
092         */
093        public static int SHORT_IMPLICIT_WAIT_TIME = 1;
094    
095        /**
096         * Set -Dremote.driver.saucelabs for running on saucelabs
097         * @link https://wiki.kuali.org/display/KULRICE/How+To+Run+a+Selenium+Test for patch required
098         */
099        public static final String REMOTE_DRIVER_SAUCELABS_PROPERTY = "remote.driver.saucelabs";
100    
101        /**
102         * Selenium's webdriver.chrome.driver parameter, you can set -Dwebdriver.chrome.driver= or Rice's REMOTE_PUBLIC_CHROME
103         */
104        public static final String WEBDRIVER_CHROME_DRIVER = "webdriver.chrome.driver";
105    
106        /**
107         * Set -Dremote.jgrowl.enabled=
108         */
109        public static final String REMOTE_JGROWL_ENABLED = "remote.jgrowl.enabled";
110    
111        /**
112         * Set -Dremote.login.uif=KNS to use old login screen.  Default value = KRAD
113         */
114        public static final String REMOTE_LOGIN_UIF = "remote.login.uif";
115    
116        /**
117         * Set -Dremote.public.chrome= or WEBDRIVER_CHROME_DRIVER
118         */
119        public static final String REMOTE_PUBLIC_CHROME = "remote.public.chrome";
120    
121        /**
122         * Time to wait for the URL used in setup to load.  Sometimes this is the first hit on the app and it needs a bit
123         * longer than any other.  120 Seconds.
124         * TODO parametrize for JVM Arg
125         */
126        public static final int SETUP_URL_LOAD_WAIT_SECONDS = 120;
127    
128        /**
129         * local proxy used for running tests thru jmeter.
130         * Include host name and port number. Example: localhost:7777
131         */
132        public static final String PROXY_HOST_PROPERTY = "remote.public.proxy";
133    
134        /**
135         * Setup the WebDriver test, login, and load the given web page
136         *
137         * @param username
138         * @param url
139         * @return driver
140         * @throws Exception
141         */
142        public static WebDriver setUp(String username, String url) throws Exception {
143            return setUp(username, url, null, null);
144        }
145    
146        /**
147         * Setup the WebDriver test, login, and load the given web page
148         *
149         * @param username
150         * @param url
151         * @param className
152         * @param testName
153         * @return driver
154         * @throws Exception
155         */
156        public static WebDriver setUp(String username, String url, String className, String testName) throws Exception {
157            if ("true".equals(System.getProperty(REMOTE_JGROWL_ENABLED, "false"))) {
158                jGrowlEnabled = true;
159            }
160    
161            if ("true".equals(System.getProperty(JS_HIGHLIGHT_PROPERTY, "false"))) {
162                jsHighlightEnabled = true;
163            }
164    
165            WebDriver driver = null;
166            if (System.getProperty(REMOTE_DRIVER_SAUCELABS_PROPERTY) == null) {
167                driver = getWebDriver();
168            } else {
169                SauceLabsWebDriverHelper saucelabs = new SauceLabsWebDriverHelper();
170                saucelabs.setUp(className, testName);
171                driver = saucelabs.getDriver();
172            }
173    
174            driver.manage().timeouts().implicitlyWait(SETUP_URL_LOAD_WAIT_SECONDS, TimeUnit.SECONDS);
175    
176            if (!System.getProperty(SauceLabsWebDriverHelper.SAUCE_BROWSER_PROPERTY,"ff").equals("opera")) {
177                driver.manage().window().maximize();
178            }
179    
180            // TODO Got into the situation where the first url doesn't expect server, but all others do.  Readdress once
181            // the NavIT WDIT conversion has been completed.
182            if (!url.startsWith("http")) {
183                url = ITUtil.getBaseUrlString() + url;
184            }
185    
186            driver.get(url);
187            driver.manage().timeouts().implicitlyWait(DEFAULT_IMPLICIT_WAIT_TIME, TimeUnit.SECONDS);
188            return driver;
189        }
190    
191        /**
192         *
193         * @param passed
194         * @param sessionId
195         * @param testParam
196         * @param userParam
197         * @throws Exception
198         */
199        public static void tearDown(boolean passed, String sessionId, String testParam, String userParam) throws Exception {
200    
201            if (System.getProperty(SauceLabsWebDriverHelper.SAUCE_PROPERTY) != null) {
202                SauceLabsWebDriverHelper.tearDown(passed, sessionId, System.getProperty(SauceLabsWebDriverHelper.SAUCE_USER_PROPERTY),
203                        System.getProperty(SauceLabsWebDriverHelper.SAUCE_KEY_PROPERTY));
204            }
205    
206            if (System.getProperty(WebDriverLegacyITBase.REMOTE_PUBLIC_USERPOOL_PROPERTY) != null) {
207                ITUtil.getHTML(ITUtil.prettyHttp(System.getProperty(WebDriverLegacyITBase.REMOTE_PUBLIC_USERPOOL_PROPERTY) + "?test="
208                        + testParam + "&user=" + userParam));
209            }
210        }
211    
212        /**
213         *
214         * @param testParam
215         * @return
216         */
217        public static String determineUser(String testParam) {
218            String user = null;
219    
220            if (System.getProperty(WebDriverLegacyITBase.REMOTE_PUBLIC_USER_PROPERTY) != null) {
221                return System.getProperty(WebDriverLegacyITBase.REMOTE_PUBLIC_USER_PROPERTY);
222            } else if (System.getProperty(WebDriverLegacyITBase.REMOTE_PUBLIC_USERPOOL_PROPERTY) != null) { // deprecated
223                String userResponse = ITUtil.getHTML(ITUtil.prettyHttp(System.getProperty(
224                        WebDriverLegacyITBase.REMOTE_PUBLIC_USERPOOL_PROPERTY) + "?test=" + testParam.trim()));
225                return userResponse.substring(userResponse.lastIndexOf(":") + 2, userResponse.lastIndexOf("\""));
226            }
227    
228            return user;
229        }
230    
231        /***
232         * @link ITUtil#checkForIncidentReport
233         * @param driver
234         * @param locator
235         * @param message
236         */
237        public static void checkForIncidentReport(WebDriver driver, String locator, Failable failable,
238                String message) {
239            ITUtil.checkForIncidentReport(driver.getPageSource(), locator, failable, message);
240        }
241    
242        /**
243         * @link http://code.google.com/p/chromedriver/downloads/list
244         * @link #REMOTE_PUBLIC_CHROME
245         * @link #WEBDRIVER_CHROME_DRIVER
246         * @link ITUtil#HUB_DRIVER_PROPERTY
247         * @return chromeDriverService
248         */
249        public static ChromeDriverService chromeDriverCreateCheck() {
250            String driverParam = System.getProperty(ITUtil.HUB_DRIVER_PROPERTY);
251            // TODO can the saucelabs driver stuff be leveraged here?
252            if (driverParam != null && "chrome".equals(driverParam.toLowerCase())) {
253                if (System.getProperty(WEBDRIVER_CHROME_DRIVER) == null) {
254                    if (System.getProperty(REMOTE_PUBLIC_CHROME) != null) {
255                        System.setProperty(WEBDRIVER_CHROME_DRIVER, System.getProperty(REMOTE_PUBLIC_CHROME));
256                    }
257                }
258                try {
259                    ChromeDriverService chromeDriverService = new ChromeDriverService.Builder()
260                            .usingDriverExecutable(new File(System.getProperty(WEBDRIVER_CHROME_DRIVER)))
261                            .usingAnyFreePort()
262                            .build();
263                    return chromeDriverService;
264                } catch (Throwable t) {
265                    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)   ;
266                }
267            }
268            return null;
269        }
270    
271        public static void jGrowl(WebDriver driver, String jGrowlHeader, boolean sticky, String message, Throwable t) {
272            if (jGrowlEnabled) { // check if jGrowl is enabled to skip over the stack trace extraction if it is not.
273                jGrowl(driver, jGrowlHeader, sticky, message + " " + t.getMessage() + "\n" + ExceptionUtils.getStackTrace(t));
274            }
275        }
276    
277        public static void jGrowl(WebDriver driver, String jGrowlHeader, boolean sticky, String message) {
278            if (jGrowlEnabled) {
279                try {
280                    String javascript="jQuery.jGrowl('" + message + "' , {sticky: " + sticky + ", header : '" + jGrowlHeader + "'});";
281                    ((JavascriptExecutor) driver).executeScript(javascript);
282                } catch (Throwable t) {
283                    jGrowlException(t);
284                }
285            }
286        }
287    
288        public static void jGrowlException(Throwable t) {
289            String failMessage = t.getMessage() + "\n" + ExceptionUtils.getStackTrace(t);
290            System.out.println("jGrowl failure " + failMessage);
291            if (JGROWL_ERROR_FAILURE) {
292                SeleneseTestBase.fail(failMessage);
293            }
294        }
295    
296        public static void highlightElement(WebDriver webDriver, WebElement webElement) {
297            if (jsHighlightEnabled) {
298                try {
299                    JavascriptExecutor js = (JavascriptExecutor) webDriver;
300                    js.executeScript("element = arguments[0];\n"
301                            + "originalStyle = element.getAttribute('style');\n"
302                            + "element.setAttribute('style', originalStyle + \"; background: "
303                            + JS_HIGHLIGHT_BACKGROUND + "; border: 2px solid " + JS_HIGHLIGHT_BOARDER + ";\");\n"
304                            + "setTimeout(function(){\n"
305                            + "    element.setAttribute('style', originalStyle);\n"
306                            + "}, " + JS_HIGHLIGHT_MS + ");", webElement);
307                } catch (Throwable t) {
308                    System.out.println("Throwable during javascript highlight element");
309                    t.printStackTrace();
310                }
311            }
312        }
313    
314    
315        /**
316         * remote.public.driver set to chrome or firefox (null assumes firefox)
317         * if remote.public.hub is set a RemoteWebDriver is created (Selenium Grid)
318         * if proxy.host is set, the web driver is setup to use a proxy
319         * @return WebDriver or null if unable to create
320         */
321        public static WebDriver getWebDriver() {
322            String driverParam = System.getProperty(ITUtil.HUB_DRIVER_PROPERTY);
323            String hubParam = System.getProperty(ITUtil.HUB_PROPERTY);
324            String proxyParam = System.getProperty(PROXY_HOST_PROPERTY);
325    
326            // setup proxy if specified as VM Arg
327            DesiredCapabilities capabilities = new DesiredCapabilities();
328            WebDriver webDriver = null;
329            if (StringUtils.isNotEmpty(proxyParam)) {
330                capabilities.setCapability(CapabilityType.PROXY, new Proxy().setHttpProxy(proxyParam));
331            }
332    
333            if (hubParam == null) {
334                if (driverParam == null || "firefox".equalsIgnoreCase(driverParam)) {
335                    FirefoxProfile profile = new FirefoxProfile();
336                    profile.setEnableNativeEvents(false);
337                    capabilities.setCapability(FirefoxDriver.PROFILE, profile);
338                    return new FirefoxDriver(capabilities);
339                } else if ("chrome".equalsIgnoreCase(driverParam)) {
340                    return new ChromeDriver(capabilities);
341                } else if ("safari".equals(driverParam)) {
342                    System.out.println("SafariDriver probably won't work, if it does please contact Erik M.");
343                    return new SafariDriver(capabilities);
344                }
345            } else {
346                try {
347                    if (driverParam == null || "firefox".equalsIgnoreCase(driverParam)) {
348                        return new RemoteWebDriver(new URL(ITUtil.getHubUrlString()), DesiredCapabilities.firefox());
349                    } else if ("chrome".equalsIgnoreCase(driverParam)) {
350                        return new RemoteWebDriver(new URL(ITUtil.getHubUrlString()), DesiredCapabilities.chrome());
351                    }
352                } catch (MalformedURLException mue) {
353                    System.out.println(ITUtil.getHubUrlString() + " " + mue.getMessage());
354                    mue.printStackTrace();
355                }
356            }
357            return null;
358        }
359    
360        /**
361         * Logs in using the KRAD Login Page
362         * If the JVM arg remote.autologin is set, auto login as admin will not be done.
363         * @param driver
364         * @param userName
365         * @param failable
366         * @throws InterruptedException
367         */
368        public static void kradLogin(WebDriver driver, String userName, Failable failable) throws InterruptedException {
369                driver.findElement(By.name("login_user")).clear();
370                driver.findElement(By.name("login_user")).sendKeys(userName);
371                driver.findElement(By.id("Rice-LoginButton")).click();
372                Thread.sleep(1000);
373                String contents = driver.getPageSource();
374                ITUtil.failOnInvalidUserName(userName, contents, failable);
375                ITUtil.checkForIncidentReport(driver.getPageSource(), "Krad Login", failable, "Krad Login failure");
376        }
377    
378        /**
379         * Logs into the Rice portal using the KNS Style Login Page.
380         * @param driver
381         * @param userName
382         * @param failable
383         * @throws InterruptedException
384         */
385        public static void login(WebDriver driver, String userName, Failable failable) throws InterruptedException {
386                driver.findElement(By.name("__login_user")).clear();
387                driver.findElement(By.name("__login_user")).sendKeys(userName);
388                driver.findElement(By.cssSelector("input[type=\"submit\"]")).click();
389                Thread.sleep(1000);
390                String contents = driver.getPageSource();
391                ITUtil.failOnInvalidUserName(userName, contents, failable);
392                ITUtil.checkForIncidentReport(driver.getPageSource(), "KNS Login", failable, "KNS Login failure");
393        }
394    
395        public static void loginKradOrKns(WebDriver driver, String user, Failable failable) throws InterruptedException {// login via either KRAD or KNS login page
396            if ("true".equalsIgnoreCase(System.getProperty(ITUtil.REMOTE_AUTOLOGIN_PROPERTY, "true"))) {
397                if (isKradLogin()){
398                    WebDriverUtil.kradLogin(driver, user, failable);
399                } else {
400                    WebDriverUtil.login(driver, user, failable);
401                }
402            }
403        }
404    
405        /**
406         * Use the KRAD Login Screen or the old KNS Login Screen
407         */
408        public static boolean isKradLogin(){
409            // check system property, default to KRAD
410            String loginUif = System.getProperty(REMOTE_LOGIN_UIF);
411            if (loginUif == null) {
412                loginUif = ITUtil.REMOTE_UIF_KRAD;
413            }
414    
415            return (ITUtil.REMOTE_UIF_KRAD.equalsIgnoreCase(loginUif));
416        }
417    
418        protected static void selectFrameSafe(WebDriver driver, String locator) {
419            try {
420                driver.switchTo().frame(locator);
421            } catch (NoSuchFrameException nsfe) {
422                // don't fail
423            }
424        }
425    
426        /**
427         * Wait for the given amount of seconds, for the given by, using the given driver.  The message is displayed if the
428         * by cannot be found.  No action is performed on the by, so it is possible that the by found is not visible or enabled.
429         *
430         * @param driver WebDriver
431         * @param waitSeconds int
432         * @param by By
433         * @param message String
434         * @throws InterruptedException
435         */
436        public static void waitFor(WebDriver driver, int waitSeconds, By by, String message) throws InterruptedException {
437            driver.manage().timeouts().implicitlyWait(waitSeconds, TimeUnit.SECONDS);
438            Thread.sleep(1000);
439            driver.findElement(by);  // NOTICE just the find, no action, so by is found, but might not be visible or enabled.
440            driver.manage().timeouts().implicitlyWait(1, TimeUnit.SECONDS);
441        }
442    
443        /**
444         * Wait for the given amount of seconds, for the given by, using the given driver.  The message is displayed if the
445         * by cannot be found.  No action is performed on the by, so it is possible that the by found is not visible or enabled.
446         *
447         * @param driver WebDriver
448         * @param waitSeconds int
449         * @param by By
450         * @param message String
451         * @throws InterruptedException
452         */
453        public static void waitFors(WebDriver driver, int waitSeconds, By by, String message) throws InterruptedException {
454            driver.manage().timeouts().implicitlyWait(waitSeconds, TimeUnit.SECONDS);
455            Thread.sleep(1000);
456            driver.findElements(by);  // NOTICE just the find, no action, so by is found, but might not be visible or enabled.
457            driver.manage().timeouts().implicitlyWait(1, TimeUnit.SECONDS);
458        }
459    
460    
461        public static void failOnMatchedJira(String contents, Failable failable) {
462            JiraAwareFailureUtil.failOnMatchedJira(contents, failable);
463        }
464        
465        private static void failWithReportInfoForKim(String contents, String linkLocator, String message) {
466            final String kimIncidentReport = extractIncidentReportKim(contents, linkLocator, message);
467            SeleneseTestBase.fail(kimIncidentReport);
468        }
469        
470        private static String extractIncidentReportKim(String contents, String linkLocator, String message) {
471            String chunk =  contents.substring(contents.indexOf("id=\"headerarea\""), contents.lastIndexOf("</div>") );
472            String docIdPre = "type=\"hidden\" value=\"";
473            String docId = chunk.substring(chunk.indexOf(docIdPre) + docIdPre.length(), chunk.indexOf("\" name=\"documentId\""));
474    
475            String stackTrace = chunk.substring(chunk.lastIndexOf("name=\"displayMessage\""), chunk.length());
476            String stackTracePre = "value=\"";
477            stackTrace = stackTrace.substring(stackTrace.indexOf(stackTracePre) + stackTracePre.length(), stackTrace.indexOf("name=\"stackTrace\"") - 2);
478    
479            return "\nIncident report "+ message+ " navigating to "+ linkLocator + " Doc Id: "+ docId.trim()+ "\nStackTrace: "+ stackTrace.trim().replace(" at ","");
480        }
481        
482        private static void processIncidentReport(String contents, String linkLocator, Failable failable, String message) {
483            failOnMatchedJira(contents, failable);
484    
485            if (contents.indexOf("Incident Feedback") > -1) {
486                failWithReportInfo(contents, linkLocator, message);
487            }
488    
489            if (contents.indexOf("Incident Report") > -1) { // KIM incident report
490                failWithReportInfoForKim(contents, linkLocator, message);
491            }
492    
493            SeleneseTestBase.fail("\nIncident report detected " + message + "\n Unable to parse out details for the contents that triggered exception: " + deLinespace(
494                    contents));
495        }
496    
497        private static void failWithReportInfo(String contents, String linkLocator, String message) {
498            final String incidentReportInformation = extractIncidentReportInfo(contents, linkLocator, message);
499            SeleneseTestBase.fail(incidentReportInformation);
500        }
501        
502        private static String extractIncidentReportInfo(String contents, String linkLocator, String message) {
503            String chunk =  contents.substring(contents.indexOf("Incident Feedback"), contents.lastIndexOf("</div>") );
504            String docId = chunk.substring(chunk.lastIndexOf("Document Id"), chunk.indexOf("View Id"));
505            docId = docId.substring(0, docId.indexOf("</span>"));
506            docId = docId.substring(docId.lastIndexOf(">") + 2, docId.length());
507    
508            String viewId = chunk.substring(chunk.lastIndexOf("View Id"), chunk.indexOf("Error Message"));
509            viewId = viewId.substring(0, viewId.indexOf("</span>"));
510            viewId = viewId.substring(viewId.lastIndexOf(">") + 2, viewId.length());
511    
512            String stackTrace = chunk.substring(chunk.lastIndexOf("(only in dev mode)"), chunk.length());
513            stackTrace = stackTrace.substring(stackTrace.indexOf("<span id=\"") + 3, stackTrace.length());
514            stackTrace = stackTrace.substring(stackTrace.indexOf("\">") + 2, stackTrace.indexOf("</span>"));
515        
516            return "\nIncident report "+ message+ " navigating to "+ linkLocator+ " : View Id: "+ viewId.trim()+ " Doc Id: "+ docId.trim()+ "\nStackTrace: "+ stackTrace.trim().replace(" at ", "");
517        }
518        
519        public static String deLinespace(String contents) {
520            while (contents.contains("\n\n")) {
521                contents = contents.replaceAll("\n\n", "\n");
522            }
523            
524            return contents;
525        }
526    }