I posted some time ago about Browser based testing in Scala using Cucumber leveraging JUnit and Selenium. That mechanism is pretty complicated, and there seems a much better way of doing it. The FluentLenium library gives a good way to integrate browser based testing into Scala. There are still some challenges with continuous integration that have to be solved, and I'll talk about that later.
What does a FluentLenium test case look like with this system? Here's a simple example that opens a home page and clicks on a link:
class HomePageSpec extends FunSuite with ShouldMatchers {
test("Visit the links page") {
withBrowser {
startAt(baseUrl) then click("a#linksPage") then
assertTitle("A List of Awesome Links") then testFooter
}
}
}
And we can fill in a form and submit it like this:
class RegistrationSpec extends FunSuite with ShouldMatchers {
val testUser = "ciTestUser"
test("Creating a fake user account") {
withBrowser {
startAt(baseUrl) then click("a#registerMe") then
formText("#firstName", "test") then
formText("#lastName", "user") then
formText("#username", testUser) then
formText("#email", "test.user@example.com") then
formText("#password", "123") then
formText("#verify", "123") then
hangAround(500) then click("#registerSubmit")
}
}
}
Much of what you see above isn't out of the box functionality with FluentLenium. Scala gives us the power to create simple DSLs to provide very powerful functionality that is easy to read and easy write. People often don't like writing tests, and Scala is a language that is still somewhat obscure. A DSL like this makes it trivial for any developer, even one who is totally unfamiliar with Scala to construct effective browser-based tests.
Now I'm going to delve into some of the specifics of how this is constructed! (example code can be found at:
git://gitorious.org/technology-madness-examples/technology-madness-examples.git)
The first piece is the basic configuration for such a project. I'm using the play project template to start with as it offers some basic helper functionality that's pretty handy. The first thing to do is create a bare play project
play create fluentlenium-example
I personally prefer ScalaTest to the built-in test mechanism in play, and the fluentlenium dependencies are needed, so the project's Build.scala gets updated with the following changes:
val appDependencies = Seq(
"org.scalatest" %% "scalatest" % "1.6.1" % "test",
"org.fluentlenium" % "fluentlenium-core" % "0.6.0",
"org.fluentlenium" % "fluentlenium-festassert" % "0.6.0"
)
val main = PlayProject(appName, appVersion, appDependencies, mainLang = JAVA).settings(
// Add your own project settings here
testOptions in Test := Nil
)
Now for the main test constructs. A wrapper object is constructed to allow us to chain function calls, and that object is instantiated with the function startAt():
case class BrowserTestWrapper(fl: List[TestBrowser => Unit]) extends Function1[TestBrowser, Unit] {
def apply(browser: TestBrowser) {
fl.foreach(x => x(browser))
}
def then(f: TestBrowser => Unit): BrowserTestWrapper = {
BrowserTestWrapper(fl :+ f)
}
}
This object is the container if you will for a list of test predicates that will execute once the test has been constructed. It is essentially a wrapped list of functions which we can see from the type List[TestBrowser => Unit]. Each test function doesn't have a return value because it's using the test systems built-in assertion system and therefore doesn't return anything useful. When this object is executed as a function, it simply runs through it's contained list and executed the tests against the browser object that is passed in.
The special sauce here is the then() method. This method takes in a new function, and builds a new BrowserTestWrapper instance with the currently list plus the new function. Each piece of the test chain simply creates a new Wrapper object!
Now we add a few helper functions in the companion object:
object BrowserTestWrapper {
def startAt(url: String): BrowserTestWrapper = {
BrowserTestWrapper(List({browser => browser.goTo(url)}, hangAround(5000)))
}
def hangAround(t: Long)(browser: TestBrowser = null) {
println("hanging around")
Thread.sleep(t)
}
def click(selector:String, index: Int = 0)(browser:TestBrowser) {
waitForSelector(selector, browser)
browser.$(selector).get(index).click()
}
def formText(selector: String, text: String)(browser: TestBrowser) {
waitForSelector(selector, browser)
browser.$(selector).text(text)
}
def waitForSelector(selector: String, browser: TestBrowser) {
waitFor(3000, NonZeroPredicate(selector))(browser)
}
def waitFor(timeout: Long, predicate: WaitPredicate): TestBrowser => Unit = { implicit browser =>
val startTime = new Date().getTime
while(!predicate(browser) && new Date().getTime < (startTime + timeout)) {
hangAround(100)()
}
}
}
sealed trait WaitPredicate extends Function[TestBrowser, Boolean] {
}
case class NonZeroPredicate(selector: String) extends WaitPredicate {
override def apply(browser: TestBrowser) = browser.$(selector).size() !=0
}
This gives us the basic pieces for the test chain itself. Now we need to define the withBrowser function so that the test chain gets executed:
object WebDriverFactory {
def withBrowser(t: BrowserTestWrapper) {
val browser = TestBrowser(getDriver)
try {
t(browser)
}
catch {
case e: Exception => {
browser.takeScreenShot(System.getProperty("user.home")+"/fail-shot-"+("%tF".format(new Date())+".png"))
throw e
}
}
browser.quit()
}
def getDriver = {
(getDriverFromSimpleName orElse defaultDriver orElse failDriver)(System.getProperty("driverName"))
}
def baseUrl = {
Option[String](System.getProperty("baseUrl")).getOrElse("http://www.mysite.com").reverse.dropWhile(_=='/').reverse + "/"
}
val defaultDriver: PartialFunction[String, WebDriver] = {
case null => internetExplorerDriver
}
val failDriver: PartialFunction[String, WebDriver] = { case x = > throw new RuntimeException("Unknown browser driver specified: " + x) }
val getDriverFromSimpleName: PartialFunction[String, WebDriver] = {
case "Firefox" => firefoxDriver
case "InternetExplorer" => internetExplorerDriver
}
def firefoxDriver = new FirefoxDriver()
def internetExplorerDriver = new InternetExplorerDriver()
}
This gives us just about all the constructs we need to run a browser driven test. I'll leave the implementation of assertTitle() and some of the other test functions up to the reader.
Once we have this structure, we can run browser tests from our local system, but it doesn't dovetail easily with a Continuous Integration server. As I write this, my CI of choice doesn't have an SBT plugin, so, I have to go a different route. Pick your poison as you may, mine is Maven, so I create a Maven pom file for the CI to execute that looks something like this:
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>com.example</groupId>
<artifactId>fluentlenium-tests</artifactId>
<version>1.0.0</version>
<inceptionYear>2012</inceptionYear>
<packaging>war</packaging>
<properties>
<scala.version>2.9.1</scala.version>
</properties>
<repositories>
<repository>
<id>scala-tools.org</id>
<name>Scala-Tools Maven2 Repository</name>
<url>http://scala-tools.org/repo-releases</url>
</repository>
<repository>
<id>typesafe</id>
<name>typesafe-releases</name>
<url>http://repo.typesafe.com/typesafe/repo</url>
</repository>
<repository>
<id>codahale</id>
<name>Codahale Repository</name>
<url>http://repo.codahale.com</url>
</repository>
</repositories>
<pluginRepositories>
<pluginRepository>
<id>scala-tools.org</id>
<name>Scala-Tools Maven2 Repository</name>
<url>http://scala-tools.org/repo-releases</url>
</pluginRepository>
</pluginRepositories>
<dependencies>
<dependency>
<groupId>org.scala-lang</groupId>
<artifactId>scala-library</artifactId>
<version>${scala.version}</version>
</dependency>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.4</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.scalatest</groupId>
<artifactId>scalatest_${scala.version}</artifactId>
<version>1.8</version>
</dependency>
<dependency>
<groupId>org.fluentlenium</groupId>
<artifactId>fluentlenium-core</artifactId>
<version>0.7.2</version>
</dependency>
<dependency>
<groupId>org.fluentlenium</groupId>
<artifactId>fluentlenium-festassert</artifactId>
<version>0.7.2</version>
</dependency>
<dependency>
<groupId>play</groupId>
<artifactId>play_${scala.version}</artifactId>
<version>2.0.3</version>
</dependency>
<dependency>
<groupId>play</groupId>
<artifactId>play-test_${scala.version}</artifactId>
<version>2.0.3</version>
</dependency>
<dependency>
<groupId>org.scala-tools.testing</groupId>
<artifactId>specs_${scala.version}</artifactId>
<version>1.6.9</version>
</dependency>
</dependencies>
<build>
<sourceDirectory>app</sourceDirectory>
<testSourceDirectory>test</testSourceDirectory>
<plugins>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
<configuration>
<source>1.6</source>
<target>1.6</target>
</configuration>
</plugin>
<plugin>
<groupId>org.scala-tools</groupId>
<artifactId>maven-scala-plugin</artifactId>
<executions>
<execution>
<goals>
<goal>compile</goal>
<goal>testCompile</goal>
</goals>
</execution>
</executions>
<configuration>
<scalaVersion>${scala.version}</scalaVersion>
<args>
<arg>-target:jvm-1.5</arg>
</args>
</configuration>
</plugin>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-surefire-plugin</artifactId>
<configuration>
<argLine>-DdriverName=Firefox</argLine>
<includes>
<include>**/*Spec.class</include>
</includes>
</configuration>
</plugin>
</plugins>
</build>
<reporting>
<plugins>
<plugin>
<groupId>org.scala-tools</groupId>
<artifactId>maven-scala-plugin</artifactId>
<configuration>
<scalaVersion>${scala.version}</scalaVersion>
</configuration>
</plugin>
</plugins>
</reporting>
</project>
You might notice that the above Maven configuration uses JUnit to execute out Spec tests. This doesn't happen by default, as JUnit doesn't pick up those classes, so we have to add an annotation at the head of the class to signal JUnit to pick up the test:
@RunWith(classOf[JUnitRunner])
class HomePageSpec extends FunSuite with ShouldMatchers {
...
}