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

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) } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy