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    
017    package edu.samplu.common;
018    
019    import org.junit.rules.TestName;
020    import org.openqa.selenium.By;
021    import org.openqa.selenium.NoSuchFrameException;
022    import org.openqa.selenium.WebDriver;
023    import org.openqa.selenium.chrome.ChromeDriver;
024    import org.openqa.selenium.chrome.ChromeDriverService;
025    import org.openqa.selenium.firefox.FirefoxDriver;
026    import org.openqa.selenium.firefox.FirefoxProfile;
027    import org.openqa.selenium.remote.DesiredCapabilities;
028    import org.openqa.selenium.remote.RemoteWebDriver;
029    import org.openqa.selenium.safari.SafariDriver;
030    
031    import com.thoughtworks.selenium.SeleneseTestBase;
032    
033    import java.io.File;
034    import java.net.MalformedURLException;
035    import java.net.URL;
036    import java.util.HashMap;
037    import java.util.Iterator;
038    import java.util.Map;
039    import java.util.concurrent.TimeUnit;
040    
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        /**
052         * TODO apparent dup WebDriverITBase.DEFAULT_WAIT_SEC
053         * TODO parametrize for JVM Arg
054         * 30 Seconds
055         */
056        public static int DEFAULT_IMPLICIT_WAIT_TIME = 30;
057    
058        /**
059         * TODO introduce SHORT_IMPLICIT_WAIT_TIME with param in WebDriverITBase
060         * TODO parametrize for JVM Arg
061         * 1 Second
062         */
063        public static int SHORT_IMPLICIT_WAIT_TIME = 1;
064    
065        /**
066         * Set -Dremote.driver.saucelabs for running on saucelabs
067         * @link https://wiki.kuali.org/display/KULRICE/How+To+Run+a+Selenium+Test for patch required
068         */
069        public static final String REMOTE_DRIVER_SAUCELABS_PROPERTY = "remote.driver.saucelabs";
070    
071        /**
072         * Selenium's webdriver.chrome.driver parameter, you can set -Dwebdriver.chrome.driver= or Rice's REMOTE_PUBLIC_CHROME
073         */
074        public static final String WEBDRIVER_CHROME_DRIVER = "webdriver.chrome.driver";
075    
076        /**
077         * Set -Dremote.public.chrome= or WEBDRIVER_CHROME_DRIVER
078         */
079        public static final String REMOTE_PUBLIC_CHROME = "remote.public.chrome";
080    
081        /**
082         * 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
083         * longer than any other.  120 Seconds.
084         * TODO parametrize for JVM Arg
085         */
086        public static final int SETUP_URL_LOAD_WAIT_SECONDS = 120;
087        
088        /**
089         * https://jira.kuali.org/browse/
090         */
091        public static final String JIRA_BROWSE_URL = "https://jira.kuali.org/browse/";
092        static Map<String, String> jiraMatches;
093        static {
094            jiraMatches = new HashMap<String, String>();
095            jiraMatches.put("Error setting property values; nested exception is org.springframework.beans.NotWritablePropertyException: Invalid property 'refreshWhenChanged' of bean class [org.kuali.rice.krad.uif.element.Action]: Bean property 'refreshWhenChanged' is not writable or has an invalid setter method. Does the parameter type of the setter match the return type of the getter?",
096                    "KULRICE-8137 Agenda Rule edit Incident report Invalid property 'refreshWhenChanged'");
097    
098            jiraMatches.put("org.kuali.rice.kns.maintenance.rules.MaintenanceDocumentRuleBase.processAddCollectionLineBusinessRules(MaintenanceDocumentRuleBase.",
099                    "KULRICE-8142 NPE in MaintenanceDocumentRuleBase.processAddCollectionLineBusinessRules");
100    
101            jiraMatches.put("at org.kuali.rice.krad.rules.DocumentRuleBase.isDocumentOverviewValid(DocumentRuleBase.",
102                    "KULRICE-8134 NPE in DocumentRuleBase.isDocumentOverviewValid(DocumentRuleBase");
103    
104            jiraMatches.put("org.kuali.rice.krad.uif.layout.TableLayoutManager.buildLine(TableLayoutManager.",
105                    "KULRICE-8160 NPE at TableLayoutManager.buildLine(TableLayoutManager");
106    
107            jiraMatches.put("Bean property 'configFileLocations' is not writable or has an invalid setter method. Does the parameter type of the setter match the return type of the getter?",
108                    "KULRICE-8173 Bean property 'configFileLocations' is not writable or has an invalid setter method");
109    
110            jiraMatches.put("Bean property 'componentSecurity' is not readable or has an invalid getter method: Does the return type of the getter match the parameter type of the setter?",
111                    "KULRICE-8182 JDK7 Bean property 'componentSecurity' is not readable...");
112    
113            jiraMatches.put("java.sql.SQLSyntaxErrorException: ORA-00904: \"ROUTEHEADERID\": invalid identifier",
114                    "KULRICE-8277 Several ITs fail with OJB operation; bad SQL grammar []; nested exception is java.sql.SQLException: ORA-00904: \"ROUTEHEADERID\": invalid identifier");
115    
116            jiraMatches.put("By.xpath: //button[@data-loadingmessage='Adding Line...']",
117                    "KULRICE-9044 KRAD \"stacked\" collection elements are not rendering add/delete buttons ");
118    
119            jiraMatches.put("Error: on line 135, column 39 in krad/WEB-INF/ftl/lib/grid.ftl",
120                    "KULRICE-9047 Term maintenance freemarker exception ");
121        }
122        
123        /**
124         * Setup the WebDriver test, login, and load the given web page
125         *
126         * @param username
127         * @param url
128         * @return driver
129         * @throws Exception
130         */
131        public static WebDriver setUp(String username, String url) throws Exception {
132            return setUp(username, url, null, null);
133        }
134    
135        /**
136         * Setup the WebDriver test, login, and load the given web page
137         *
138         * @param username
139         * @param url
140         * @param className
141         * @param testName
142         * @return driver
143         * @throws Exception
144         */
145        public static WebDriver setUp(String username, String url, String className, TestName testName) throws Exception {
146            WebDriver driver = null;
147            if (System.getProperty(REMOTE_DRIVER_SAUCELABS_PROPERTY) == null) {
148                driver = getWebDriver();
149    //        } else {
150    //            SauceLabsWebDriverHelper saucelabs = new SauceLabsWebDriverHelper();
151    //            saucelabs.setUp(className, testName);
152    //            driver = saucelabs.getDriver();
153            }
154            driver.manage().timeouts().implicitlyWait(SETUP_URL_LOAD_WAIT_SECONDS, TimeUnit.SECONDS);
155    
156            // TODO Got into the situation where the first url doesn't expect server, but all others do.  Readdress once
157            // the NavIT WDIT conversion has been completed.
158            if (!url.startsWith("http")) {
159                url = ITUtil.getBaseUrlString() + url;
160            }
161    
162            driver.get(url);
163            driver.manage().timeouts().implicitlyWait(DEFAULT_IMPLICIT_WAIT_TIME, TimeUnit.SECONDS);
164            return driver;
165        }
166    
167        /**
168         *
169         * @param passed
170         * @param sessionId
171         * @param testParam
172         * @param userParam
173         * @throws Exception
174         */
175        public static void tearDown(boolean passed, String sessionId, String testParam, String userParam) throws Exception {
176    
177    //        if (System.getProperty(SauceLabsWebDriverHelper.SAUCE_PROPERTY) != null) {
178    //            SauceLabsWebDriverHelper.tearDown(passed, sessionId, System.getProperty(SauceLabsWebDriverHelper.SAUCE_USER_PROPERTY),
179    //                    System.getProperty(SauceLabsWebDriverHelper.SAUCE_KEY_PROPERTY));
180    //        }
181    
182            if (System.getProperty(WebDriverLegacyITBase.REMOTE_PUBLIC_USERPOOL_PROPERTY) != null) {
183                ITUtil.getHTML(ITUtil.prettyHttp(System.getProperty(WebDriverLegacyITBase.REMOTE_PUBLIC_USERPOOL_PROPERTY) + "?test="
184                        + testParam + "&user=" + userParam));
185            }
186        }
187    
188        /**
189         *
190         * @param testParam
191         * @return
192         */
193        public static String determineUser(String testParam) {
194            String user = null;
195    
196            if (System.getProperty(WebDriverLegacyITBase.REMOTE_PUBLIC_USER_PROPERTY) != null) {
197                return System.getProperty(WebDriverLegacyITBase.REMOTE_PUBLIC_USER_PROPERTY);
198            } else if (System.getProperty(WebDriverLegacyITBase.REMOTE_PUBLIC_USERPOOL_PROPERTY) != null) { // deprecated
199                String userResponse = ITUtil.getHTML(ITUtil.prettyHttp(System.getProperty(
200                        WebDriverLegacyITBase.REMOTE_PUBLIC_USERPOOL_PROPERTY) + "?test=" + testParam.trim()));
201                return userResponse.substring(userResponse.lastIndexOf(":") + 2, userResponse.lastIndexOf("\""));
202            }
203    
204            return user;
205        }
206    
207        /***
208         * @link ITUtil#checkForIncidentReport
209         * @param driver
210         * @param locator
211         * @param message
212         */
213        public static void checkForIncidentReport(WebDriver driver, String locator, Failable failable,
214                String message) {
215            ITUtil.checkForIncidentReport(driver.getPageSource(), locator, failable, message);
216        }
217    
218        /**
219         * @link http://code.google.com/p/chromedriver/downloads/list
220         * @link #REMOTE_PUBLIC_CHROME
221         * @link #WEBDRIVER_CHROME_DRIVER
222         * @link ITUtil#HUB_DRIVER_PROPERTY
223         * @return chromeDriverService
224         */
225        public static ChromeDriverService chromeDriverCreateCheck() {
226            String driverParam = System.getProperty(ITUtil.HUB_DRIVER_PROPERTY);
227            // TODO can the saucelabs driver stuff be leveraged here?
228            if (driverParam != null && "chrome".equals(driverParam.toLowerCase())) {
229                if (System.getProperty(WEBDRIVER_CHROME_DRIVER) == null) {
230                    if (System.getProperty(REMOTE_PUBLIC_CHROME) != null) {
231                        System.setProperty(WEBDRIVER_CHROME_DRIVER, System.getProperty(REMOTE_PUBLIC_CHROME));
232                    }
233                }
234                try {
235                    ChromeDriverService chromeDriverService = new ChromeDriverService.Builder()
236                            .usingChromeDriverExecutable(new File(System.getProperty(WEBDRIVER_CHROME_DRIVER)))
237                            .usingAnyFreePort()
238                            .build();
239                    return chromeDriverService;
240                } catch (Throwable t) {
241                    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)   ;
242                }
243            }
244            return null;
245        }
246    
247        /**
248         * remote.public.driver set to chrome or firefox (null assumes firefox)
249         * if remote.public.hub is set a RemoteWebDriver is created (Selenium Grid)
250         * @return WebDriver or null if unable to create
251         */
252        public static WebDriver getWebDriver() {
253            String driverParam = System.getProperty(ITUtil.HUB_DRIVER_PROPERTY);
254            String hubParam = System.getProperty(ITUtil.HUB_PROPERTY);
255            if (hubParam == null) {
256                if (driverParam == null || "firefox".equalsIgnoreCase(driverParam)) {
257                    FirefoxProfile profile = new FirefoxProfile();
258                    profile.setEnableNativeEvents(false);
259                    return new FirefoxDriver(profile);
260                } else if ("chrome".equalsIgnoreCase(driverParam)) {
261                    return new ChromeDriver();
262                } else if ("safari".equals(driverParam)) {
263                    System.out.println("SafariDriver probably won't work, if it does please contact Erik M.");
264                    return new SafariDriver();
265                }
266            } else {
267                try {
268                    if (driverParam == null || "firefox".equalsIgnoreCase(driverParam)) {
269                        return new RemoteWebDriver(new URL(ITUtil.getHubUrlString()), DesiredCapabilities.firefox());
270                    } else if ("chrome".equalsIgnoreCase(driverParam)) {
271                        return new RemoteWebDriver(new URL(ITUtil.getHubUrlString()), DesiredCapabilities.chrome());
272                    }
273                } catch (MalformedURLException mue) {
274                    System.out.println(ITUtil.getHubUrlString() + " " + mue.getMessage());
275                    mue.printStackTrace();
276                }
277            }
278            return null;
279        }
280    
281        /**
282         * If the JVM arg remote.autologin is set, auto login as admin will not be done.
283         * @param driver
284         * @param userName
285         * @param failable
286         * @throws InterruptedException
287         */
288        public static void login(WebDriver driver, String userName, Failable failable) throws InterruptedException {
289            if (System.getProperty(ITUtil.REMOTE_AUTOLOGIN_PROPERTY) == null) {
290                driver.findElement(By.name("__login_user")).clear();
291                driver.findElement(By.name("__login_user")).sendKeys(userName);
292                driver.findElement(By.cssSelector("input[type=\"submit\"]")).click();
293                Thread.sleep(1000);
294                String contents = driver.getPageSource();
295                ITUtil.failOnInvalidUserName(userName, contents, failable);
296            }
297        }
298    
299        protected static void selectFrameSafe(WebDriver driver, String locator) {
300            try {
301                driver.switchTo().frame(locator);
302            } catch (NoSuchFrameException nsfe) {
303                // don't fail
304            }
305        }
306    
307        /**
308         * Wait for the given amount of seconds, for the given by, using the given driver.  The message is displayed if the
309         * by cannot be found.  No action is performed on the by, so it is possible that the by found is not visible or enabled.
310         *
311         * @param driver WebDriver
312         * @param waitSeconds int
313         * @param by By
314         * @param message String
315         * @throws InterruptedException
316         */
317        public static void waitFor(WebDriver driver, int waitSeconds, By by, String message) throws InterruptedException {
318            driver.manage().timeouts().implicitlyWait(waitSeconds, TimeUnit.SECONDS);
319            Thread.sleep(1000);
320            driver.findElement(by);  // NOTICE just the find, no action, so by is found, but might not be visible or enabled.
321            driver.manage().timeouts().implicitlyWait(1, TimeUnit.SECONDS);
322        }
323        
324        public static void failOnMatchedJira(String contents) {
325            Iterator<String> iter = jiraMatches.keySet().iterator();
326            String key = null;
327    
328            while (iter.hasNext()) {
329                key = iter.next();
330                if (contents.contains(key)) {
331                    SeleneseTestBase.fail(JIRA_BROWSE_URL + jiraMatches.get(key));
332                }
333            }
334        }
335        
336        private static void failWithReportInfoForKim(String contents, String linkLocator, String message) {
337            final String kimIncidentReport = extractIncidentReportKim(contents, linkLocator, message);
338            SeleneseTestBase.fail(kimIncidentReport);
339        }
340        
341        private static String extractIncidentReportKim(String contents, String linkLocator, String message) {
342            String chunk =  contents.substring(contents.indexOf("id=\"headerarea\""), contents.lastIndexOf("</div>") );
343            String docIdPre = "type=\"hidden\" value=\"";
344            String docId = chunk.substring(chunk.indexOf(docIdPre) + docIdPre.length(), chunk.indexOf("\" name=\"documentId\""));
345    
346            String stackTrace = chunk.substring(chunk.lastIndexOf("name=\"displayMessage\""), chunk.length());
347            String stackTracePre = "value=\"";
348            stackTrace = stackTrace.substring(stackTrace.indexOf(stackTracePre) + stackTracePre.length(), stackTrace.indexOf("name=\"stackTrace\"") - 2);
349    
350            return "\nIncident report "+ message+ " navigating to "+ linkLocator + " Doc Id: "+ docId.trim()+ "\nStackTrace: "+ stackTrace.trim();
351        }
352        
353        private static void processIncidentReport(String contents, String linkLocator, String message) {
354            failOnMatchedJira(contents);
355    
356            if (contents.indexOf("Incident Feedback") > -1) {
357                failWithReportInfo(contents, linkLocator, message);
358            }
359    
360            if (contents.indexOf("Incident Report") > -1) { // KIM incident report
361                failWithReportInfoForKim(contents, linkLocator, message);
362            }
363    
364            SeleneseTestBase.fail("\nIncident report detected " + message + "\n Unable to parse out details for the contents that triggered exception: " + deLinespace(
365                    contents));
366        }
367    
368        private static void failWithReportInfo(String contents, String linkLocator, String message) {
369            final String incidentReportInformation = extractIncidentReportInfo(contents, linkLocator, message);
370            SeleneseTestBase.fail(incidentReportInformation);
371        }
372        
373        private static String extractIncidentReportInfo(String contents, String linkLocator, String message) {
374            String chunk =  contents.substring(contents.indexOf("Incident Feedback"), contents.lastIndexOf("</div>") );
375            String docId = chunk.substring(chunk.lastIndexOf("Document Id"), chunk.indexOf("View Id"));
376            docId = docId.substring(0, docId.indexOf("</span>"));
377            docId = docId.substring(docId.lastIndexOf(">") + 2, docId.length());
378    
379            String viewId = chunk.substring(chunk.lastIndexOf("View Id"), chunk.indexOf("Error Message"));
380            viewId = viewId.substring(0, viewId.indexOf("</span>"));
381            viewId = viewId.substring(viewId.lastIndexOf(">") + 2, viewId.length());
382    
383            String stackTrace = chunk.substring(chunk.lastIndexOf("(only in dev mode)"), chunk.length());
384            stackTrace = stackTrace.substring(stackTrace.indexOf("<span id=\"") + 3, stackTrace.length());
385            stackTrace = stackTrace.substring(stackTrace.indexOf("\">") + 2, stackTrace.indexOf("</span>"));
386        
387            return "\nIncident report "+ message+ " navigating to "+ linkLocator+ " : View Id: "+ viewId.trim()+ " Doc Id: "+ docId.trim()+ "\nStackTrace: "+ stackTrace.trim();
388        }
389        
390        public static String deLinespace(String contents) {
391            while (contents.contains("\n\n")) {
392                contents = contents.replaceAll("\n\n", "\n");
393            }
394            
395            return contents;
396        }
397    }