
org.scalatestplus.play.OneServerPerSuite.scala Maven / Gradle / Ivy
The newest version!
/*
* Copyright 2001-2014 Artima, Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.scalatestplus.play
import play.api.test._
import org.scalatest._
/**
* Trait that provides a new `FakeApplication` and running `TestServer` instance per ScalaTest `Suite`.
*
* By default, this trait creates a new `FakeApplication` for the `Suite` using default parameter values, which
* is made available via the `app` field defined in this trait and a new `TestServer` for the `Suite` using the port number provided by
* its `port` field and the `FakeApplication` provided by its `app` field. If your `Suite` needs a
* `FakeApplication` with non-default parameters, override `app`. If it needs a different port number,
* override `port`.
*
*
* This `SuiteMixin` trait's overridden `run` method calls `start` on the `TestServer`
* before executing the `Suite` via a call to `super.run`.
* In addition, it places a reference to the `FakeApplication` provided by `app` into the `ConfigMap`
* under the key `org.scalatestplus.play.app` and to the port number provided by `port` under the key
* `org.scalatestplus.play.port`. This allows any nested `Suite`s to access the `Suite`'s
* `FakeApplication` and port number as well, most easily by having the nested `Suite`s mix in the
* [[org.scalatestplus.play.ConfiguredServer ConfiguredServer]] trait. On the status returned by `super.run`, this
* trait's overridden `run` method registers a call to `stop` on the `TestServer` to be executed when the `Status`
* completes, and returns the same `Status`. This ensure the `TestServer` will continue to execute until
* all nested suites have completed, after which the `TestServer` will be stopped.
*
* Here's an example that demonstrates some of the services provided by this trait:
*
*
* package org.scalatestplus.play.examples.oneserverpersuite
*
* import play.api.test._
* import org.scalatest._
* import org.scalatestplus.play._
* import play.api.{Play, Application}
*
* class ExampleSpec extends PlaySpec with OneServerPerSuite {
*
* // Override app if you need a FakeApplication with other than non-default parameters.
* implicit override lazy val app: FakeApplication =
* FakeApplication(additionalConfiguration = Map("ehcacheplugin" -> "disabled"))
*
* "The OneServerPerSuite trait" must {
* "provide a FakeApplication" in {
* app.configuration.getString("ehcacheplugin") mustBe Some("disabled")
* }
* "make the FakeApplication available implicitly" in {
* def getConfig(key: String)(implicit app: Application) = app.configuration.getString(key)
* getConfig("ehcacheplugin") mustBe Some("disabled")
* }
* "start the FakeApplication" in {
* Play.maybeApplication mustBe Some(app)
* }
* "provide the port number" in {
* port mustBe Helpers.testServerPort
* }
* "provide an actual running server" in {
* import Helpers._
* import java.net._
* val url = new URL("http://localhost:" + port + "/boum")
* val con = url.openConnection().asInstanceOf[HttpURLConnection]
* try con.getResponseCode mustBe 404
* finally con.disconnect()
* }
* }
* }
*
*
* If you have many tests that can share the same `FakeApplication` and `TestServer`, and you don't want to put them all into one
* test class, you can place them into different `Suite` classes.
* These will be your nested suites. Create a master suite that extends `OneServerPerSuite` and declares the nested
* `Suite`s. Annotate the nested suites with `@DoNotDiscover` and have them extend `ConfiguredServer`. Here's an example:
*
*
* package org.scalatestplus.play.examples.oneserverpersuite
*
* import play.api.test._
* import org.scalatest._
* import org.scalatestplus.play._
* import play.api.{Play, Application}
*
* // This is the "master" suite
* class NestedExampleSpec extends Suites(
* new OneSpec,
* new TwoSpec,
* new RedSpec,
* new BlueSpec
* ) with OneServerPerSuite {
* // Override app if you need a FakeApplication with other than non-default parameters.
* implicit override lazy val app: FakeApplication =
* FakeApplication(additionalConfiguration = Map("ehcacheplugin" -> "disabled"))
* }
*
* // These are the nested suites
* @DoNotDiscover class OneSpec extends PlaySpec with ConfiguredServer
* @DoNotDiscover class TwoSpec extends PlaySpec with ConfiguredServer
* @DoNotDiscover class RedSpec extends PlaySpec with ConfiguredServer
*
* @DoNotDiscover
* class BlueSpec extends PlaySpec with ConfiguredServer {
*
* "The OneAppPerSuite trait" must {
* "provide a FakeApplication" in {
* app.configuration.getString("ehcacheplugin") mustBe Some("disabled")
* }
* "make the FakeApplication available implicitly" in {
* def getConfig(key: String)(implicit app: Application) = app.configuration.getString(key)
* getConfig("ehcacheplugin") mustBe Some("disabled")
* }
* "start the FakeApplication" in {
* Play.maybeApplication mustBe Some(app)
* }
* "provide the port number" in {
* port mustBe Helpers.testServerPort
* }
* "provide an actual running server" in {
* import Helpers._
* import java.net._
* val url = new URL("http://localhost:" + port + "/boum")
* val con = url.openConnection().asInstanceOf[HttpURLConnection]
* try con.getResponseCode mustBe 404
* finally con.disconnect()
* }
* }
* }
*
*/
trait OneServerPerSuite extends SuiteMixin with ServerProvider { this: Suite =>
/**
* An implicit instance of `FakeApplication`.
*
* This trait's implementation initializes this `lazy` `val` with a new instance of `FakeApplication` with
* parameters set to their defaults. Override this `lazy` `val` if you need a `FakeApplication` created with non-default parameter values.
*/
implicit lazy val app: FakeApplication = new FakeApplication()
/**
* The port used by the `TestServer`. By default this will be set to the result returned from
* `Helpers.testServerPort`. You can override this to provide a different port number.
*/
lazy val port: Int = Helpers.testServerPort
/**
* Invokes `start` on a new `TestServer` created with the `FakeApplication` provided by `app` and the
* port number defined by `port`, places the `FakeApplication` and port number into the `ConfigMap` under the keys
* `org.scalatestplus.play.app` and `org.scalatestplus.play.port`, respectively, to make
* them available to nested suites; calls `super.run`; and lastly ensures the `FakeApplication and test server are stopped after
* all tests and nested suites have completed.
*
* @param testName an optional name of one test to run. If `None`, all relevant tests should be run.
* I.e., `None` acts like a wildcard that means run all relevant tests in this `Suite`.
* @param args the `Args` for this run
* @return a `Status` object that indicates when all tests and nested suites started by this method have completed, and whether or not a failure occurred.
*/
abstract override def run(testName: Option[String], args: Args): Status = {
val testServer = TestServer(port, app)
testServer.start()
try {
val newConfigMap = args.configMap + ("org.scalatestplus.play.app" -> app) + ("org.scalatestplus.play.port" -> port)
val newArgs = args.copy(configMap = newConfigMap)
val status = super.run(testName, newArgs)
status.whenCompleted { _ => testServer.stop() }
status
}
catch { // In case the suite aborts, ensure the server is stopped
case ex: Throwable =>
testServer.stop()
throw ex
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy