Testing@LMAX – Screenshots with Selenium/WebDriver
By Adrian Sutton
When an automated UI test fails, it can be hard to tell exactly what went wrong just from the failure message. The failure message typically just says that some element the test was looking for wasn’t found, but it doesn’t tell you what was there. Was there an error message displayed instead? Was the operation still executing? Did something completely unexpected happen instead?
To answer those questions our DSL automatically captures a screenshot when any UI operation fails and we include a link to it in the failure message. That way when someone reviews the test result they can see exactly what was on screen which typically makes it straight forward to identify what went wrong and fix it.
Until recently we’d been using the convenient and helpful looking TakesScreenshot.getScreenshotAs
method that WebDriver provides. For example:
((TakesScreenshot)webDriver).getScreenshotAs(
new SaveScreenshotOutputType(pngFilename));
As expected, this creates a PNG image in the specified location that looks for all the world like a screenshot of the browser content. Unfortunately, it’s lying.
WebDriver actually does something very clever and gets the browser to render the page content into a canvas element and then saves that as the PNG file. This is an extremely close approximation of what the page looks like with two important exceptions:
- It doesn’t respect the viewport size so body content is never scrolled off-screen.
- Any browser chrome or random other windows that have popped up aren’t shown.
Both of these things can be an issue – the scrolled-off-screen one being the most problematic. Modern WebDriver quite accurately simulates a user clicking and typing keys so if somethings not on screen it can’t be clicked. When your test fails because an element was “present but not visible” and the screenshot shows it as very clearly visible, hilarity ensues. Very frustrating hilarity.
To fix this we’ve started taking honest-to-goodness screenshots. Since all our tests get their own X session (courtesy of vncserver) their windows are completely isolated from each other and a dump of the entire screen will capture precisely what a real user would see, browser chrome and scrolling included. Linux provides an entertaining array of options for capturing screenshots from the command line but the one that happened to be already installed was import, part of the ImageMagick suite. We simply execute:
import -display :20 -window root screenshot.png
where :20 is the X display this particular test has been allocated and screenshot.png is where we want the screenshot to wind up.
Since the WebDriver screenshot can be useful as well – for example finding out an error message is displayed at the top of the screen we continue to grab that too.
Finally, for completeness we grab a dump of the DOM to a HTML file so we can later inspect what IDs, classes, attributes etc are present, including any hidden elements. webDriver.getPageSource()
makes that easy and we append an extra HTML comment that includes webDriver.getCurrentUrl() for good measure.