blended.itestsupport.BlendedTestContextManager.scala Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of blended.itestsupport Show documentation
Show all versions of blended.itestsupport Show documentation
Define an integration test API for collaborating blended container(s) using docker as a runtime
for the container(s) under test and an Akka based Camel framework to perform the integration tests
as pure blackbox tests. Container(s) may be prestarted and discovered (for execution speed) or
started by the integration test (for reproducability).
The newest version!
package blended.itestsupport
import akka.actor.{Actor, ActorLogging, ActorRef, Props}
import akka.camel.CamelExtension
import akka.pattern._
import blended.akka.MemoryStash
import blended.itestsupport.condition.{Condition, ConditionActor, ConditionProvider}
import blended.itestsupport.docker.protocol._
import blended.itestsupport.docker.{ContainerManager, DockerClientFactory, DockerClientProvider}
import blended.itestsupport.protocol.{TestContextRequest, _}
import com.github.dockerjava.api.DockerClient
import org.apache.camel.CamelContext
class BlendedTestContextManager extends Actor with ActorLogging with MemoryStash { this : TestContextConfigurator =>
val camel = CamelExtension(context.system)
def initializing : Receive = {
case req : TestContextRequest =>
log.debug("Configuring Camel Extension for the test...")
val containerMgr = context.actorOf(Props(new ContainerManager with DockerClientProvider {
override def getClient : DockerClient = {
implicit val logger = context.system.log
DockerClientFactory(context.system.settings.config)
}
}), "ContainerMgr")
containerMgr ! StartContainerManager(req.cuts)
context.become(starting(List(sender()), containerMgr) orElse stashing)
}
def starting(requestors: List[ActorRef], containerMgr: ActorRef) : Receive = {
case ContainerManagerStarted(result) =>
result match {
case Right(cuts) =>
val camelCtxt = configure(cuts, camel.context)
context.become(working(cuts, camelCtxt, containerMgr))
requestors.foreach(_ ! camelCtxt)
case m => requestors.foreach(_ ! m)
}
case req : TestContextRequest => context.become(starting(sender :: requestors, containerMgr))
}
def working(cuts: Map[String, ContainerUnderTest], testContext: CamelContext, containerMgr: ActorRef) : Receive = {
case req : TestContextRequest => sender ! testContext
case ContainerReady_? =>
implicit val eCtxt = context.system.dispatcher
val requestor = sender()
val condition = containerReady(cuts)
log.info(s"Waiting for container condition(s) [$condition}]")
val checker = context.system.actorOf(Props(ConditionActor(condition)))
(checker ? CheckCondition)(condition.timeout).map {
case cr: ConditionCheckResult => ContainerReady(cr.allSatisfied)
case _ => ContainerReady(false)
}.pipeTo(requestor)
case ConfiguredContainers_? => sender ! ConfiguredContainers(cuts)
case cc : ConfiguredContainer_? => sender() ! ConfiguredContainer(cuts.get(cc.ctName))
case gcd : GetContainerDirectory =>
containerMgr.tell(gcd, sender())
case wcd : WriteContainerDirectory =>
containerMgr.tell(wcd, sender())
case exec : ExecuteContainerCommand =>
containerMgr.tell(exec, sender())
case scm : StopContainerManager =>
camel.context.stop()
containerMgr.forward(scm)
}
def containerReady(cuts: Map[String, ContainerUnderTest]) : Condition = ConditionProvider.alwaysTrue()
def receive = initializing orElse stashing
}