io.gatling.jms.JmsReqReplyAction.scala Maven / Gradle / Ivy
/**
* Copyright 2011-2014 eBusiness Information, Groupe Excilys (www.ebusinessinformation.fr)
*
* 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 io.gatling.jms
import akka.actor.ActorRef
import io.gatling.core.action.{ Failable, Interruptable }
import io.gatling.core.session.Expression
import io.gatling.core.util.TimeHelper.nowMillis
import javax.jms.Message
import java.util.concurrent.atomic.AtomicBoolean
import io.gatling.core.validation.Validation
import io.gatling.core.validation.SuccessWrapper
import io.gatling.core.session.Session
import io.gatling.jms.client.SimpleJmsClient
object JmsReqReplyAction {
val blockingReceiveReturnedNull = new Exception("Blocking receive returned null. Possibly the consumer was closed.")
}
/**
* Core JMS Action to handle Request-Reply semantics
*
* This handles the core "send"ing of messages. Gatling calls the execute method to trigger a send.
* This implementation then forwards it on to a tracking actor.
* @author [email protected]
*/
class JmsReqReplyAction(
val next: ActorRef,
attributes: JmsAttributes,
protocol: JmsProtocol,
tracker: ActorRef)
extends Interruptable
with Failable {
// Create a client to refer to
val client = new SimpleJmsClient(
protocol.connectionFactoryName,
attributes.destination,
attributes.replyDestination,
protocol.url,
protocol.credentials,
protocol.contextFactory,
protocol.deliveryMode,
protocol.messageMatcher)
val messageMatcher = protocol.messageMatcher
class ListenerThread(val continue: AtomicBoolean = new AtomicBoolean(true)) extends Thread(new Runnable {
def run(): Unit = {
val replyConsumer = client.createReplyConsumer(attributes.selector.orNull)
try {
while (continue.get) {
val m = replyConsumer.receive
m match {
case msg: Message =>
tracker ! MessageReceived(messageMatcher.responseID(msg), nowMillis, msg)
logMessage(s"Message received ${msg.getJMSMessageID}", msg)
case _ =>
logger.error(JmsReqReplyAction.blockingReceiveReturnedNull.getMessage)
throw JmsReqReplyAction.blockingReceiveReturnedNull
}
}
} catch {
// when we close, receive can throw exception
case e: Exception => logger.error(e.getMessage)
} finally {
replyConsumer.close()
}
}
}) {
def close() = {
continue.set(false)
interrupt()
join()
}
}
val listenerThreads = (1 to protocol.listenerCount).map(_ => new ListenerThread)
listenerThreads.foreach(_.start)
override def postStop(): Unit = {
listenerThreads.foreach(_.close())
client.close()
}
/**
* Framework calls the execute() method to send a single request
*
* Note this does not catch any exceptions (even JMSException) as generally these indicate a
* configuration failure that is unlikely to be addressed by retrying with another message
*/
def executeOrFail(session: Session): Validation[Unit] = {
// send the message
val start = nowMillis
val msg = resolveProperties(attributes.messageProperties, session).flatMap { messageProperties =>
attributes.message match {
case BytesJmsMessage(bytes) => bytes(session).map(bytes => client.sendBytesMessage(bytes, messageProperties))
case MapJmsMessage(map) => map(session).map(map => client.sendMapMessage(map, messageProperties))
case ObjectJmsMessage(o) => o(session).map(o => client.sendObjectMessage(o, messageProperties))
case TextJmsMessage(txt) => txt(session).map(txt => client.sendTextMessage(txt, messageProperties))
}
}
msg.map { msg =>
// notify the tracker that a message was sent
tracker ! MessageSent(messageMatcher.requestID(msg), start, nowMillis, attributes.checks, session, next, attributes.requestName)
logMessage(s"Message sent ${msg.getJMSMessageID}", msg)
}
}
def resolveProperties(properties: Map[Expression[String], Expression[Any]],
session: Session): Validation[Map[String, Any]] = {
properties.foldLeft(Map.empty[String, Any].success) {
case (resolvedProperties, (key, value)) =>
val newProperty: Validation[(String, Any)] =
for {
key <- key(session)
value <- value(session)
} yield key -> value
for {
newProperty <- newProperty
resolvedProperties <- resolvedProperties
} yield resolvedProperties + newProperty
}
}
def logMessage(text: String, msg: Message): Unit = {
logger.debug(text)
logger.trace(msg.toString)
}
}