All Downloads are FREE. Search and download functionalities are using the official Maven repository.

play.api.test.WSTestClient.scala Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (C) from 2022 The Play Framework Contributors , 2011-2021 Lightbend Inc. 
 */

package play.api.test

import org.apache.pekko.stream.Materializer
import play.api.libs.ws._
import play.api.mvc.Call

/**
 * A standalone test client that is useful for running standalone integration tests.
 */
trait WsTestClient {
  type Port = Int

  private val clientProducer: (Port, String) => WSClient = { (port, scheme) =>
    new WsTestClient.InternalWSClient(scheme, port)
  }

  /**
   * Constructs a WS request for the given reverse route.  Optionally takes a WSClient producing function.  Note that the WS client used
   * by default requires a running Play application (use WithApplication for tests).
   *
   * For example:
   * {{{
   * "work" in new WithApplication() { implicit app =>
   *   wsCall(controllers.routes.Application.index()).get()
   * }
   * }}}
   */
  def wsCall(
      call: Call
  )(implicit port: Port, client: (Port, String) => WSClient = clientProducer, scheme: String = "http"): WSRequest = {
    wsUrl(call.url)
  }

  /**
   * Constructs a WS request holder for the given relative URL.  Optionally takes a scheme, a port, or a client producing function.  Note that the WS client used
   * by default requires a running Play application (use WithApplication for tests).
   */
  def wsUrl(
      url: String
  )(implicit port: Port, client: (Port, String) => WSClient = clientProducer, scheme: String = "http"): WSRequest = {
    client(port, scheme).url(s"$scheme://localhost:" + port + url)
  }

  /**
   * Run the given block of code with a client.
   *
   * The client passed to the block of code supports absolute path relative URLs passed to the url method.  If an
   * absolute path relative URL is used, the protocol is assumed to be http, the host localhost, and the port is the
   * implicit port parameter passed to this method.  This is designed to work smoothly with the Server.with* methods,
   * for example:
   *
   * {{{
   * Server.withRouter() {
   *   case GET(p"/hello/\$who") => Action(Ok("Hello " + who))
   * } { implicit port =>
   *   withClient { ws =>
   *     await(ws.url("/hello/world").get()).body must_== "Hello world"
   *   }
   * }
   * }}}
   *
   * @param block The block of code to run
   * @param port The port
   * @return The result of the block of code
   */
  def withClient[T](
      block: WSClient => T
  )(implicit port: play.api.http.Port = new play.api.http.Port(-1), scheme: String = "http"): T = {
    val client = clientProducer(port.value, scheme)
    try {
      block(client)
    } finally {
      client.close()
    }
  }
}

object WsTestClient extends WsTestClient {
  private val singletonClient = new SingletonWSClient()

  /**
   * Creates a standalone WSClient, using its own ActorSystem and Netty thread pool.
   *
   * This client has no dependencies at all on the underlying system, but is wasteful of resources.
   *
   * @param port   the port to connect to the server on.
   * @param scheme the scheme to connect on ("http" or "https")
   */
  class InternalWSClient(scheme: String, port: Port) extends WSClient {
    singletonClient.addReference(this)

    def underlying[T] = singletonClient.underlying.asInstanceOf[T]

    def url(url: String): WSRequest = {
      if (url.startsWith("/") && port != -1) {
        val httpPort = new play.api.http.Port(port)
        singletonClient.url(s"$scheme://localhost:$httpPort$url")
      } else {
        singletonClient.url(url)
      }
    }

    /** Closes this client, and releases underlying resources. */
    override def close(): Unit = {
      singletonClient.removeReference(this)
    }
  }

  /**
   * A singleton ws client that keeps an ActorSystem / AHC client around only when
   * it is needed.
   */
  private class SingletonWSClient extends WSClient {
    import java.util.concurrent._
    import java.util.concurrent.atomic._

    import scala.annotation.tailrec
    import scala.concurrent.duration._
    import scala.concurrent.Future

    import org.apache.pekko.actor.ActorSystem
    import org.apache.pekko.actor.Cancellable
    import org.apache.pekko.actor.Terminated
    import play.api.libs.ws.ahc.AhcWSClient
    import play.api.libs.ws.ahc.AhcWSClientConfig

    private val logger = org.slf4j.LoggerFactory.getLogger(this.getClass)

    private val ref = new AtomicReference[WSClient]()

    private val count = new AtomicInteger(1)

    private val references = new ConcurrentLinkedQueue[WSClient]()

    private val idleDuration = 5.seconds

    private var idleCheckTask: Option[Cancellable] = None

    def removeReference(client: InternalWSClient): Boolean = {
      references.remove(client)
    }

    def addReference(client: InternalWSClient): Boolean = {
      references.add(client)
    }

    private def closeIdleResources(client: WSClient, system: ActorSystem): Future[Terminated] = {
      ref.compareAndSet(client, null)
      client.close()
      system.terminate()
    }

    @tailrec private def wsClientInstance: WSClient = ref.get match {
      case null =>
        val (newInstance, system) = createNewClient()
        if (ref.compareAndSet(null, newInstance)) {
          // We successfully created a client and set the reference; schedule an idle check on the ActorSystem
          scheduleIdleCheck(newInstance, system)
          newInstance
        } else {
          // Another thread got there first; close the resources and try again
          closeIdleResources(newInstance, system)
          wsClientInstance // recurse
        }
      case client => client
    }

    private def createNewClient(): (WSClient, ActorSystem) = {
      val name = "ws-test-client-" + count.getAndIncrement()
      logger.info(s"createNewClient: name = $name")
      val system       = ActorSystem(name)
      val materializer = Materializer.matFromSystem(system)
      val config       = AhcWSClientConfig(maxRequestRetry = 0) // Don't retry for tests
      val client       = AhcWSClient(config)(materializer)
      (client, system)
    }

    private def scheduleIdleCheck(client: WSClient, system: ActorSystem) = {
      val scheduler = system.scheduler
      idleCheckTask match {
        case Some(cancellable) =>
          // Something else got here first...
          logger.error(s"scheduleIdleCheck: looks like a race condition of WsTestClient...")
          // This way we immediately see the error on the closed client
          closeIdleResources(client, system)
        case None =>
          //
          idleCheckTask = Option {
            scheduler.scheduleAtFixedRate(initialDelay = idleDuration, interval = idleDuration)(() =>
              if (references.size() == 0) {
                logger.debug(s"check: no references found on client $client, system $system")
                idleCheckTask.map(_.cancel())
                idleCheckTask = None
                closeIdleResources(client, system)
              } else {
                logger.debug(s"check: client references = ${references.toArray.toSeq}")
              }
            )(system.dispatcher)
          }
      }
    }

    /**
     * The underlying implementation of the client, if any.  You must cast explicitly to the type you want.
     *
     * @tparam T the type you are expecting (i.e. isInstanceOf)
     * @return the backing class.
     */
    override def underlying[T]: T = wsClientInstance.underlying

    /**
     * Generates a request holder which can be used to build requests.
     *
     * @param url The base URL to make HTTP requests to.
     * @return a WSRequestHolder
     */
    override def url(url: String): WSRequest = wsClientInstance.url(url)

    override def close(): Unit = {}
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy