mq5.1-source.src.share.cclient.client.Connection.cpp Maven / Gradle / Ivy
The newest version!
/*
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
*
* Copyright (c) 2000-2013 Oracle and/or its affiliates. All rights reserved.
*
* The contents of this file are subject to the terms of either the GNU
* General Public License Version 2 only ("GPL") or the Common Development
* and Distribution License("CDDL") (collectively, the "License"). You
* may not use this file except in compliance with the License. You can
* obtain a copy of the License at
* https://glassfish.dev.java.net/public/CDDL+GPL_1_1.html
* or packager/legal/LICENSE.txt. See the License for the specific
* language governing permissions and limitations under the License.
*
* When distributing the software, include this License Header Notice in each
* file and include the License file at packager/legal/LICENSE.txt.
*
* GPL Classpath Exception:
* Oracle designates this particular file as subject to the "Classpath"
* exception as provided by Oracle in the GPL Version 2 section of the License
* file that accompanied this code.
*
* Modifications:
* If applicable, add the following below the License Header, with the fields
* enclosed by brackets [] replaced by your own identifying information:
* "Portions Copyright [year] [name of copyright owner]"
*
* Contributor(s):
* If you wish your version of this file to be governed by only the CDDL or
* only the GPL Version 2, indicate your decision by adding "[Contributor]
* elects to include this software in this distribution under the [CDDL or GPL
* Version 2] license." If you don't indicate a single choice of license, a
* recipient has the option to distribute your version of this file under
* either the CDDL, the GPL Version 2 or to extend the choice of license to
* its licensees as provided above. However, if you add GPL Version 2 code
* and therefore, elected the GPL Version 2 license, then the option applies
* only if the new code is made subject to such option by the copyright
* holder.
*/
/*
* @(#)Connection.cpp 1.46 10/17/07
*/
#include "../cshim/mqconnection-props.h"
#include "Connection.hpp"
#include "ProducerFlow.hpp"
#include "../util/UtilityMacros.h"
#include "protocol/TCPProtocolHandler.hpp"
#include "protocol/SSLProtocolHandler.hpp"
#include "../util/LogUtils.hpp"
#include "../cshim/iMQCallbackUtils.hpp"
#include "NSSInitCall.h"
#include
PRBool Connection::nsprVersionChecked = PR_FALSE;
#if !defined(WIN32)
extern "C" {
#endif // !defined(WIN32)
static void startThreadHelper(void *arg);
#if !defined(WIN32)
}
#endif // !defined(WIN32)
/*
*
*/
Connection::Connection(): producerFlowTable(PR_TRUE, PR_FALSE)
{
CHECK_OBJECT_VALIDITY();
init();
return;
}
/*
*
*/
void
Connection::init()
{
CHECK_OBJECT_VALIDITY();
this->properties = NULL;
this->protocolHandler = NULL;
this->readChannel = NULL;
this->pingTimer = NULL;
this->flowControl = NULL;
this->transport = NULL;
this->username = NULL;
this->password = NULL;
this->authHandler = NULL;
this->isClosed = PR_TRUE;
this->isAborted = PR_TRUE;
this->isStopped = PR_TRUE;
this->tempDestSequence = INITIAL_TEMP_DEST_SEQUENCE;
this->clientID = NULL;
// Optional properties
this->transportConnectionType = DEFAULT_CONNECTION_TYPE;
this->ackTimeoutMicroSec = msTimeoutToMicroSeconds(DEFAULT_ACK_TIMEOUT_MILLISEC);
this->writeTimeoutMicroSec = msTimeoutToMicroSeconds(DEFAULT_WRITE_TIMEOUT_MILLISEC);
this->ackOnPersistentProduce = DEFAULT_ACK_ON_PERSISTENT_PRODUCE;
this->ackOnNonPersistentProduce = DEFAULT_ACK_ON_NON_PERSISTENT_PRODUCE;
this->ackOnAcknowledge = DEFAULT_ACK_ON_ACKNOWLEDGE;
this->flowControlIsLimited = DEFAULT_CONNECTION_FLOW_LIMIT_ENABLED;
this->flowControlWaterMark = DEFAULT_CONNECTION_FLOW_LIMIT;
this->flowControlNumMessagesBeforePausing = DEFAULT_CONNECTION_FLOW_COUNT;
this->pingIntervalSec = DEFAULT_PING_INTERVAL_SEC;
this->consumerPrefetchMaxMsgCount =
DEFAULT_CONSUMER_PREFETCH_MAX_MESSAGE_COUNT;
this->consumerPrefetchThresholdPercent =
DEFAULT_CONSUMER_PREFETCH_THRESHOLD_PERCENT;
// Callback info
this->exceptionListenerCallback = NULL;
this->exceptionListenerCallbackData = NULL;
this->createThreadCallback = NULL;
this->createThreadCallbackData = NULL;
this->connectionID = NULL_CONN_ID;
}
/*
*
*/
Connection::~Connection()
{
CHECK_OBJECT_VALIDITY();
LOG_FINE(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Connection::~Connection() called" ));
this->close();
this->deleteMemberVariables();
this->init();
}
void
Connection::setIsXA() {
CHECK_OBJECT_VALIDITY();
this->isXA = PR_TRUE;
}
PRBool
Connection::getIsXA() {
CHECK_OBJECT_VALIDITY();
return this->isXA;
}
/*
*
*/
void
Connection::deleteMemberVariables()
{
CHECK_OBJECT_VALIDITY();
DELETE( this->flowControl ); // only called from Connection
DELETE( this->readChannel ); // keeps a pointer to the protocolHandler
DELETE( this->pingTimer );
DELETE( this->protocolHandler );
DELETE( this->transport ); // cannot be deleted while protocolHandler is active
DELETE( this->username ); // can safely be deleted
DELETE( this->password ); // can safely be deleted
DELETE( this->authHandler ); // cannot be deleted while protocolHandler is active
DELETE( this->clientID ); // can safely be deleted
HANDLED_DELETE( this->properties );
}
/*
*
*/
MQError
Connection::openConnection(Properties * connectionProperties,
UTF8String * usernameArg,
UTF8String * passwordArg,
UTF8String * clientIDArg,
MQConnectionExceptionListenerFunc exceptionListener,
void * exceptionCallBackData,
const MQCreateThreadFunc createThreadFunc,
void * createThreadFuncData)
{
static const char FUNCNAME[] = "openConnection";
CHECK_OBJECT_VALIDITY();
MQError errorCode;
this->isXA = PR_FALSE;
monitor.enter();
ASSERT( this->isClosed );
CNDCHK( !this->isClosed, MQ_CONNECTION_OPEN_ERROR );
init();
this->setCreateThreadCallback(createThreadFunc, createThreadFuncData);
// We shouldn't get a null parameter, but if we do then return.
// We're responsible for freeing these objects so do that too.
ASSERT( connectionProperties != NULL );
ASSERT( usernameArg != NULL );
ASSERT( passwordArg != NULL );
CNDCHK( connectionProperties == NULL, MQ_NULL_PTR_ARG );
CNDCHK( usernameArg == NULL, MQ_NULL_PTR_ARG );
CNDCHK( passwordArg == NULL, MQ_NULL_PTR_ARG );
this->properties = connectionProperties;
this->username = usernameArg;
this->password = passwordArg;
this->clientID = clientIDArg;
// These are owned by Connection now
connectionProperties = NULL;
usernameArg = NULL;
passwordArg = NULL;
clientIDArg = NULL;
// Set the member variables based on the optional properties
ERRCHK( this->setFieldsFromProperties() );
if (STRCMP(this->transportConnectionType, TCP_CONNECTION_TYPE) == 0) {
if (PR_GetEnv("MQ_NSS_5078380_WORKAROUND") == NULL) {
NSPRCHK_TRACE( callOnceNSS_Init(NULL), "callOnceNSS_Init(NULL)", "mq" );
}
}
// Open a socket connection to the broker.
ERRCHK( this->connectToBroker() );
// Create a protocol handler.
MEMCHK( this->protocolHandler = new ProtocolHandler(this) );
// These must be set here, or ReadChannel will deadlock.
this->isClosed = PR_FALSE;
this->isAborted = PR_FALSE;
this->isStopped = PR_TRUE;
this->setExceptionListenerCallback(exceptionListener, exceptionCallBackData);
// Create a flow control handler.
MEMCHK( this->flowControl = new FlowControl(this) );
this->readChannel = new ReadChannel(this);
MEMCHK( this->readChannel );
ERRCHK( this->readChannel->getInitializationError() );
if (this->pingIntervalSec > 0) {
this->pingTimer = new PingTimer(this);
MEMCHK( this->pingTimer );
ERRCHK( this->pingTimer->getInitializationError() );
LOG_INFO(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Connection ping enabled (ping interval = %d second).", this->pingIntervalSec ));
}
// This connects to the broker at the JMS protocol layer and
// authenticates.
LOG_FINE(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Connection saying hello to the broker" ));
errorCode = this->hello();
LOG_FINE(( CODELOC, CONNECTION_LOG_MASK, this->id(), errorCode,
"Connection::openConnection() returned from this->hello(). errorCode = %d", errorCode ));
MQ_ERRCHK_TRACE( errorCode, FUNCNAME );
// Set the clientID for this connection.
if (this->clientID != NULL) {
ERRCHK( setClientID(PR_FALSE) );
}
LOG_INFO(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Connection connected to broker" ));
monitor.exit();
return MQ_SUCCESS;
Cleanup:
// Something failed so we must delete these parameters
HANDLED_DELETE( connectionProperties );
DELETE( usernameArg );
DELETE( passwordArg );
DELETE( clientIDArg );
// Close the connection
this->close();
LOG_SEVERE(( CODELOC, CONNECTION_LOG_MASK, this->id(),
MQ_COULD_NOT_CONNECT_TO_BROKER,
"Could not connect to broker "
"because '%s' (%d).",
errorStr(errorCode), errorCode ));
monitor.exit();
MQ_ERROR_TRACE( FUNCNAME, errorCode );
if (errorCode == PR_CONNECT_RESET_ERROR) {
errorCode = MQ_COULD_NOT_CONNECT_TO_BROKER;
}
return errorCode;
}
/*
*
*/
MQError
Connection::setClientID(PRBool ifnotNULL)
{
CHECK_OBJECT_VALIDITY();
MQError errorCode = MQ_SUCCESS;
if (this->clientID == NULL) {
if (ifnotNULL) {
return MQ_SUCCESS;
}
ERRCHK( MQ_INVALID_CLIENTID );
}
CNDCHK( this->clientID->length() == 0, MQ_INVALID_CLIENTID );
// Set the clientID with the broker
ERRCHK( protocolHandler->setClientID(clientID) );
return MQ_SUCCESS;
Cleanup:
MQ_ERROR_TRACE( "setClientID", errorCode );
return errorCode;
}
/*
*
*/
MQError
Connection::setFieldsFromProperties()
{
MQError errorCode = MQ_SUCCESS;
PRInt32 intProp = 0;
PRBool boolProp = PR_FALSE;
const char * stringProp = NULL;
NULLCHK( this->properties );
// We could use more error checking on these properties to make sure
// they are within bounds.
// Option: the type of the connection (i.e. TCP or SSL).
if (this->properties->getStringProperty(MQ_CONNECTION_TYPE_PROPERTY,
&stringProp) == MQ_SUCCESS)
{
this->transportConnectionType = stringProp;
}
// Option: the number of milliseconds to wait for an acknowledgement
// from the broker. A value of 0 implies no timeout.
if (this->properties->getIntegerProperty(MQ_ACK_TIMEOUT_PROPERTY,
&intProp) == MQ_SUCCESS)
{
if (intProp < 0) return MQ_UNSUPPORTED_ARGUMENT_VALUE;
this->ackTimeoutMicroSec = msTimeoutToMicroSeconds(intProp);
}
// Option: the number of milliseconds to wait for a Packet write
// to complete the broker. A value of 0 implies no timeout.
if (this->properties->getIntegerProperty(MQ_WRITE_TIMEOUT_PROPERTY,
&intProp) == MQ_SUCCESS)
{
if (intProp < 0) return MQ_UNSUPPORTED_ARGUMENT_VALUE;
this->writeTimeoutMicroSec = msTimeoutToMicroSeconds(intProp);
}
// Option: the number of milliseconds to wait for read from portmapper
// A value of 0 implies no timeout.
if (this->properties->getIntegerProperty(MQ_READ_PORTMAPPER_TIMEOUT_PROPERTY,
&intProp) == MQ_SUCCESS)
{
if (intProp < 0) return MQ_UNSUPPORTED_ARGUMENT_VALUE;
}
if (this->properties->getIntegerProperty(MQ_PING_INTERVAL_PROPERTY,
&intProp) == MQ_SUCCESS)
{
this->pingIntervalSec = intProp;
}
// Option: whether the broker must acknowledge each message that is
// sent by the client
if (this->properties->getBooleanProperty(MQ_ACK_ON_PRODUCE_PROPERTY,
&boolProp) == MQ_SUCCESS)
{
this->ackOnPersistentProduce = boolProp;
this->ackOnNonPersistentProduce = boolProp;
}
// Option: whether the broker must acknowledge each ack message that
// is sent by the client
if (this->properties->getBooleanProperty(MQ_ACK_ON_ACKNOWLEDGE_PROPERTY,
&boolProp) == MQ_SUCCESS)
{
this->ackOnAcknowledge = boolProp;
}
// Option: flow control from broker is limited
if (this->properties->getBooleanProperty(MQ_CONNECTION_FLOW_LIMIT_ENABLED_PROPERTY,
&boolProp) == MQ_SUCCESS)
{
this->flowControlIsLimited = boolProp;
}
// Option: watermark for how many undelivered messages the client will buffer.
// This is used only if flowControlIsLimited.
if (this->properties->getIntegerProperty(MQ_CONNECTION_FLOW_LIMIT_PROPERTY,
&intProp) == MQ_SUCCESS)
{
if (intProp <= 0) return MQ_UNSUPPORTED_ARGUMENT_VALUE;
this->flowControlWaterMark = intProp;
}
// Option: the number of messages that the broker sends before asking the
// client if it should continue.
if (this->properties->getIntegerProperty(MQ_CONNECTION_FLOW_COUNT_PROPERTY,
&intProp) == MQ_SUCCESS)
{
if (intProp <= 0) return MQ_UNSUPPORTED_ARGUMENT_VALUE;
this->flowControlNumMessagesBeforePausing = intProp;
}
LOG_FINE(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Connection FlowCount=%d, FlowLimitEnabled=%d and FlowLimit=%d",
this->flowControlNumMessagesBeforePausing,
this->flowControlIsLimited,
this->flowControlWaterMark ));
return MQ_SUCCESS;
Cleanup:
return errorCode;
}
/*
*
*/
MQError
Connection::getProperties(const Properties ** const props)
{
CHECK_OBJECT_VALIDITY();
RETURN_ERROR_IF_NULL( props );
*props = NULL;
*props = this->properties;
return MQ_SUCCESS;
}
/*
*
*/
MQError
Connection::createTransportHandler()
{
CHECK_OBJECT_VALIDITY();
ASSERT( this->transportConnectionType != NULL );
ASSERT( this->transport == NULL );
if (this->transportConnectionType == NULL) {
this->transportConnectionType = DEFAULT_CONNECTION_TYPE;
}
// Switch statement
if (STRCMP(this->transportConnectionType, TCP_CONNECTION_TYPE) == 0) {
RETURN_IF_OUT_OF_MEMORY( this->transport = new TCPProtocolHandler() );
}
else if (STRCMP(this->transportConnectionType, SSL_CONNECTION_TYPE) == 0 ||
STRCMP(this->transportConnectionType, TLS_CONNECTION_TYPE) == 0) {
RETURN_IF_OUT_OF_MEMORY( this->transport = new SSLProtocolHandler() );
}
else {
return MQ_CONNECTION_UNSUPPORTED_TRANSPORT;
}
return MQ_SUCCESS;
}
/*
*
*/
MQError
Connection::connectToBroker()
{
CHECK_OBJECT_VALIDITY();
// Create a transport handler (either TCP or SSL).
RETURN_IF_ERROR( createTransportHandler() );
// Connect to the broker at the transport layer
MQError errorCode = transport->connect(this->properties);
if (errorCode != MQ_SUCCESS) {
DELETE(this->transport);
LOG_SEVERE(( CODELOC, CONNECTION_LOG_MASK, this->id(),
MQ_COULD_NOT_CONNECT_TO_BROKER,
"Transport protocol could not connect to the broker because '%s' (%d).",
errorStr(errorCode), errorCode ));
MQ_ERROR_TRACE( "connectToBroker", errorCode );
return MQ_COULD_NOT_CONNECT_TO_BROKER;
}
return MQ_SUCCESS;
}
/*
*
*/
TransportProtocolHandler *
Connection::getTransport() const
{
CHECK_OBJECT_VALIDITY();
return transport;
}
/*
*
*/
AuthenticationProtocolHandler *
Connection::getAuthenticationHandler() const
{
CHECK_OBJECT_VALIDITY();
return authHandler;
}
/*
*
*/
HandledObjectType
Connection::getObjectType() const
{
CHECK_OBJECT_VALIDITY();
return CONNECTION_OBJECT;
}
/*
*
*/
void
Connection::setAuthenticationHandler(
AuthenticationProtocolHandler * const handler)
{
CHECK_OBJECT_VALIDITY();
DELETE( this->authHandler );
this->authHandler = handler;
}
/*
*
*/
MQError
Connection::deleteDestination(Destination * const dest)
{
CHECK_OBJECT_VALIDITY();
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
RETURN_IF_ERROR( this->protocolHandler->deleteDestination(dest) );
return MQ_SUCCESS;
}
MQError
Connection::createDestination(Destination * const dest)
{
CHECK_OBJECT_VALIDITY();
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
RETURN_IF_ERROR( this->protocolHandler->createDestination(dest) );
return MQ_SUCCESS;
}
/*
*
*/
MQError
Connection::addProducerFlow(PRInt64 producerID, const ProducerFlow * const producerFlow)
{
CHECK_OBJECT_VALIDITY();
MQError errorCode = MQ_SUCCESS;
RETURN_ERROR_IF_NULL( producerFlow );
Long * producerIDLong = new Long(producerID);
if (producerIDLong != NULL) {
producerFlowTableMonitor.enter();
errorCode = this->producerFlowTable.addEntry(producerIDLong, (Object * const)producerFlow);
producerFlowTableMonitor.exit();
if (errorCode != MQ_SUCCESS) {
DELETE( producerIDLong );
}
}
else {
errorCode = MQ_OUT_OF_MEMORY;
}
if (errorCode == MQ_SUCCESS) {
LOG_FINE(( CODELOC, PRODUCER_FLOWCONTROL_LOG_MASK, this->id(), MQ_SUCCESS,
"Added producerID=%s (0x%p) to producerFlowTable",
producerIDLong->toString(), producerFlow ));
} else {
LOG_FINE(( CODELOC, PRODUCER_FLOWCONTROL_LOG_MASK, this->id(), errorCode,
"Failed to add producerID=%s (0x%p) to producerFlowTable because '%s' (%d)",
(producerIDLong == NULL) ? "NULL":producerIDLong->toString(), producerFlow,
errorStr(errorCode), errorCode ));
}
return errorCode;
}
MQError
Connection::getProducerFlow(PRInt64 producerID, ProducerFlow ** const producerFlow)
{
CHECK_OBJECT_VALIDITY();
MQError errorCode = IMQ_SUCCESS;
RETURN_ERROR_IF_NULL(producerFlow);
Long producerIDLong(producerID);
producerFlowTableMonitor.enter();
errorCode = this->producerFlowTable.getValueFromKey(&producerIDLong,
(const Object** const)producerFlow);
if (errorCode == MQ_SUCCESS) {
errorCode = (*producerFlow)->acquireReference();
}
producerFlowTableMonitor.exit();
if (errorCode != MQ_SUCCESS) {
LOG_FINER(( CODELOC, PRODUCER_FLOWCONTROL_LOG_MASK, this->id(), errorCode,
"Failed to get producerID=%s from producerFlowTable because '%s' (%d)",
producerIDLong.toString(), errorStr(errorCode), errorCode ));
} else {
LOG_FINE(( CODELOC, PRODUCER_FLOWCONTROL_LOG_MASK, this->id(), MQ_SUCCESS,
"Get producerID=%s (0x%p) from producerFlowTable success",
producerIDLong.toString(), *producerFlow ));
}
return errorCode;
}
MQError
Connection::removeProducerFlow(PRInt64 producerID)
{
CHECK_OBJECT_VALIDITY();
MQError errorCode = MQ_SUCCESS;
ProducerFlow * producerFlow = NULL;
Long producerIDLong(producerID);
producerFlowTableMonitor.enter();
errorCode = this->getProducerFlow(producerID, &producerFlow);
if (errorCode == MQ_SUCCESS) {
errorCode = this->producerFlowTable.removeEntry(&producerIDLong);
producerFlow->close(MQ_PRODUCER_CLOSED);
this->releaseProducerFlow(&producerFlow);
}
producerFlowTableMonitor.exit();
if (errorCode == MQ_SUCCESS) {
LOG_FINE(( CODELOC, PRODUCER_FLOWCONTROL_LOG_MASK, this->id(), IMQ_SUCCESS,
"Removed producerID=%s (0x%p) from producerFlowTable",
producerIDLong.toString(), producerFlow ));
} else {
LOG_FINE(( CODELOC, PRODUCER_FLOWCONTROL_LOG_MASK, this->id(), errorCode,
"Failed to remove producerID=%s from producerFlowTable because '%s' (%d)",
producerIDLong.toString(), errorStr(errorCode), errorCode ));
}
return errorCode;
}
void
Connection::releaseProducerFlow(ProducerFlow ** producerFlow)
{
ASSERT( producerFlow != NULL && (*producerFlow) != NULL );
producerFlowTableMonitor.enter();
if ((*producerFlow)->releaseReference() == PR_TRUE) {
DELETE( (*producerFlow) );
}
producerFlowTableMonitor.exit();
}
MQError
Connection::closeAllProducerFlow()
{
CHECK_OBJECT_VALIDITY();
MQError errorCode = MQ_SUCCESS;
Long * producerIDLong = NULL;
ProducerFlow * producerFlow = NULL;
LOG_FINEST(( CODELOC, PRODUCER_FLOWCONTROL_LOG_MASK, this->id(), MQ_SUCCESS,
"In closeAllProducerFlow" ));
producerFlowTableMonitor.enter();
errorCode = this->producerFlowTable.keyIterationStart();
if (errorCode == MQ_SUCCESS) {
while (this->producerFlowTable.keyIterationHasNext()) {
errorCode = this->producerFlowTable.keyIterationGetNext((const BasicType**)&producerIDLong);
if (errorCode == MQ_SUCCESS) {
errorCode = this->producerFlowTable.getValueFromKey(producerIDLong, (const Object** const)&producerFlow);
if (errorCode == MQ_SUCCESS) {
LOG_FINE(( CODELOC, PRODUCER_FLOWCONTROL_LOG_MASK, this->id(), MQ_SUCCESS,
"Closing ProducerFlow(producerID=%s)", producerIDLong->toString() ));
producerFlow->close(MQ_BROKER_CONNECTION_CLOSED);
} else {
LOG_WARNING(( CODELOC, PRODUCER_FLOWCONTROL_LOG_MASK, this->id(), errorCode,
"Unable to get ProducerFlow(producerID=%s) from ProducerFlowTable because '%s' (%d)",
producerIDLong->toString(), errorStr(errorCode), errorCode ));
}
} else {
LOG_WARNING(( CODELOC, PRODUCER_FLOWCONTROL_LOG_MASK, this->id(), errorCode,
"Unable to get next ProducerFlow in ProducerFlowTable because '%s' (%d)",
errorStr(errorCode), errorCode ));
}
} //while
} else {
LOG_WARNING(( CODELOC, PRODUCER_FLOWCONTROL_LOG_MASK, this->id(), errorCode,
"Unable to start iterating ProducerFlowTable because '%s' (%d)",
errorStr(errorCode), errorCode ));
}
producerFlowTableMonitor.exit();
LOG_FINEST(( CODELOC, PRODUCER_FLOWCONTROL_LOG_MASK, this->id(), MQ_SUCCESS,
"CloseAllProducerFlow return"));
return errorCode;
}
/*
*
*/
MQError
Connection::addToReceiveQTable(PRInt64 consumerIDArg, ReceiveQueue * const receiveQ)
{
CHECK_OBJECT_VALIDITY();
MQError errorCode = MQ_SUCCESS;
RETURN_ERROR_IF_NULL( receiveQ );
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
errorCode = receiveQTable.add(consumerIDArg, receiveQ);
if (errorCode == MQ_SUCCESS) {
LOG_FINEST(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Connection::addToReceiveQTable(%lld, 0x%p) succeeded",
consumerIDArg, receiveQ ));
} else {
LOG_FINE(( CODELOC, CONNECTION_LOG_MASK, this->id(), errorCode,
"Connection::addToReceiveQTable(%lld, 0x%p) failed because '%s' (%d)",
consumerIDArg, receiveQ, errorStr(errorCode), errorCode ));
}
return errorCode;
}
/*
*
*/
MQError
Connection::addToAckQTable(PRInt64 * ackIDArg, ReceiveQueue * const receiveQ)
{
CHECK_OBJECT_VALIDITY();
MQError errorCode = MQ_SUCCESS;
RETURN_ERROR_IF_NULL( receiveQ );
RETURN_ERROR_IF_NULL( ackIDArg );
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
errorCode = ackQTable.add(ackIDArg, receiveQ);
if (errorCode == MQ_SUCCESS) {
LOG_FINEST(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Connection::addToAckQTable(%lld, 0x%p) succeeded",
*ackIDArg, receiveQ ));
} else {
LOG_FINE(( CODELOC, CONNECTION_LOG_MASK, this->id(), errorCode,
"Connection::addToAckQTable(%lld, 0x%p) failed because '%s' (%d)",
*ackIDArg, receiveQ, errorStr(errorCode), errorCode ));
}
return errorCode;
}
/*
*
*/
MQError
Connection::addToPendingConsumerTable(PRInt64 ackIDArg, MessageConsumer * const consumer)
{
CHECK_OBJECT_VALIDITY();
MQError errorCode = MQ_SUCCESS;
RETURN_ERROR_IF_NULL( consumer );
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
errorCode = pendingConsumerTable.add(ackIDArg, consumer);
if (errorCode == MQ_SUCCESS) {
LOG_FINEST(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Connection::addToPendingConsumerTable(%lld, 0x%p) succeeded",
ackIDArg, consumer ));
} else {
LOG_FINE(( CODELOC, CONNECTION_LOG_MASK, this->id(), errorCode,
"Connection::addToPendingConsumerTable(%lld, 0x%p) failed because '%s' (%d)",
ackIDArg, consumer, errorStr(errorCode), errorCode ));
}
return errorCode;
}
/*
*
*/
MQError
Connection::removeFromReceiveQTable(const PRInt64 consumerIDArg)
{
CHECK_OBJECT_VALIDITY();
MQError errorCode = receiveQTable.remove(consumerIDArg);
if (errorCode == MQ_SUCCESS) {
LOG_FINEST(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Connection::removed FromReceiveQTable(%lld) ", consumerIDArg ));
} else {
LOG_FINE(( CODELOC, CONNECTION_LOG_MASK, this->id(), errorCode,
"Connection::removed FromReceiveQTable(%lld) failed because '%s' (%d)",
consumerIDArg, errorStr(errorCode), errorCode ));
}
return errorCode;
}
/*
*
*/
MQError
Connection::removeFromAckQTable(const PRInt64 ackIDArg)
{
CHECK_OBJECT_VALIDITY();
MQError errorCode = ackQTable.remove(ackIDArg);
if (errorCode == MQ_SUCCESS) {
LOG_FINEST(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Connection::removed FromAckQTable(%lld) ", ackIDArg ));
} else {
LOG_FINE(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Connection::removed FromAckQTable(%lld) failed because '%s' (%d)",
ackIDArg, errorStr(errorCode), errorCode ));
}
return errorCode;
}
/*
*
*/
MQError
Connection::removeFromPendingConsumerTable(const PRInt64 ackIDArg, MessageConsumer ** const consumer)
{
CHECK_OBJECT_VALIDITY();
MQError errorCode = pendingConsumerTable.remove(ackIDArg, consumer);
if (errorCode == MQ_SUCCESS) {
LOG_FINEST(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Connection::removed FromPendingConsumerTable(%lld) ", ackIDArg ));
} else {
LOG_FINE(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Connection::removed FromPendingConsumerTable(%lld) failed because '%s' (%d)",
ackIDArg, errorStr(errorCode), errorCode ));
}
return errorCode;
}
/*
*
*/
MQError
Connection::enqueueReceiveQPacket(const PRInt64 consumerIDArg, Packet * const packet)
{
CHECK_OBJECT_VALIDITY();
RETURN_ERROR_IF_NULL( packet );
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
return ( receiveQTable.enqueue(consumerIDArg, packet) );
}
/*
*
*/
MQError
Connection::enqueueAckQPacket(const PRInt64 ackIDArg, Packet * const packet)
{
CHECK_OBJECT_VALIDITY();
RETURN_ERROR_IF_NULL( packet );
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
return ( ackQTable.enqueue(ackIDArg, packet) );
}
/*
*
*/
ProtocolHandler *
Connection::getProtocolHandler() const
{
CHECK_OBJECT_VALIDITY();
return protocolHandler;
}
/*
*
*/
MQError
Connection::hello()
{
CHECK_OBJECT_VALIDITY();
RETURN_ERROR_IF_NULL( protocolHandler );
return protocolHandler->hello(this->username, this->password);
}
/*
*
*/
const IPAddress *
Connection::getLocalIP() const
{
CHECK_OBJECT_VALIDITY();
const IPAddress * ipAddr = NULL;
if (transport != NULL) {
transport->getLocalIP(&ipAddr);
}
return ipAddr;
}
/*
*
*/
PRUint16
Connection::getLocalPort() const
{
CHECK_OBJECT_VALIDITY();
PRUint16 localPort = 0;
if (this->transport!= NULL) {
this->transport->getLocalPort(&localPort);
}
return localPort;
}
/*
*
*/
const UTF8String *
Connection::getClientID()
{
CHECK_OBJECT_VALIDITY();
return this->clientID;
}
/*
*
*/
PRInt32
Connection::getTemporaryDestinationSequence()
{
CHECK_OBJECT_VALIDITY();
PRInt32 seq;
monitor.enter();
// This is what the Java code does. But it seems like we would
// worry about over-flow. They might just assume that there won't
// be that many temporary destinations created.
seq = tempDestSequence;
tempDestSequence++;
monitor.exit();
return seq;
}
/*
* Caller is responsible to free the return string
*/
char *
Connection::getTemporaryDestinationPrefix(PRBool isQueue)
{
MQError errorCode = MQ_SUCCESS;
char * tempDestPrefix = NULL;
UTF8String * destinationName = NULL;
const char * cidCharStr = NULL;
const UTF8String * cid = NULL;
tempDestPrefix = new char[MAX_DESTINATION_NAME_LEN];
CNDCHK( tempDestPrefix == NULL, MQ_OUT_OF_MEMORY );
cid = this->getClientID();
if (cid != NULL) {
cidCharStr = cid->getCharStr();
} else {
const IPAddress * ipaddr = this->getLocalIP();
cidCharStr = (ipaddr != NULL) ? ipaddr->toString() : "unknown";
}
SNPRINTF( tempDestPrefix, MAX_DESTINATION_NAME_LEN, "%s%s%s/%d",
TEMPORARY_DESTINATION_URI_PREFIX,
isQueue ? TEMPORARY_QUEUE_URI_NAME : TEMPORARY_TOPIC_URI_NAME,
cidCharStr, this->getLocalPort() );
tempDestPrefix[MAX_DESTINATION_NAME_LEN-1] = '\0'; // just to be safe
return tempDestPrefix;
Cleanup:
return NULL;
}
/*
*
*/
PRBool
Connection::isAdminKeyUsed() const
{
CHECK_OBJECT_VALIDITY();
// Whenever adminKey is used, then this shouldn't be hardcoded.
return PR_FALSE;
}
/*
*
*/
const char *
Connection::getTransportConnectionType() const
{
CHECK_OBJECT_VALIDITY();
return this->transportConnectionType;
}
/*
*
*/
PRInt32
Connection::getAckTimeoutMicroSec() const
{
CHECK_OBJECT_VALIDITY();
return this->ackTimeoutMicroSec;
}
PRInt32
Connection::getWriteTimeoutMicroSec() const
{
CHECK_OBJECT_VALIDITY();
return this->writeTimeoutMicroSec;
}
/*
*
*/
PRInt32
Connection::getPingIntervalSec() const
{
CHECK_OBJECT_VALIDITY();
return this->pingIntervalSec;
}
/*
*
*/
PRBool
Connection::getAckOnPersistentProduce() const
{
CHECK_OBJECT_VALIDITY();
return this->ackOnPersistentProduce;
}
/*
*
*/
PRBool
Connection::getAckOnNonPersistentProduce() const
{
CHECK_OBJECT_VALIDITY();
return this->ackOnNonPersistentProduce;
}
/*
*
*/
PRBool
Connection::getAckOnAcknowledge() const
{
CHECK_OBJECT_VALIDITY();
return this->ackOnAcknowledge;
}
/*
*
*/
PRBool
Connection::getFlowControlIsLimited() const
{
CHECK_OBJECT_VALIDITY();
return this->flowControlIsLimited;
}
/*
*
*/
PRInt32
Connection::getFlowControlWaterMark() const
{
CHECK_OBJECT_VALIDITY();
return this->flowControlWaterMark;
}
/*
*
*/
PRInt32
Connection::getNumMessagesBeforePausing() const
{
CHECK_OBJECT_VALIDITY();
return this->flowControlNumMessagesBeforePausing;
}
/*
*
*/
PRInt32
Connection::getConsumerPrefetchMaxMsgCount() const
{
CHECK_OBJECT_VALIDITY();
return this->consumerPrefetchMaxMsgCount;
}
/*
*
*/
PRFloat64
Connection::getConsumerPrefetchThresholdPercent() const
{
CHECK_OBJECT_VALIDITY();
return this->consumerPrefetchThresholdPercent;
}
PRBool
Connection::getIsConnectionClosed() const
{
CHECK_OBJECT_VALIDITY();
return isClosed;
}
/*
*
*/
PRBool
Connection::getIsClosed() const
{
CHECK_OBJECT_VALIDITY();
return isClosed ||
isAborted ||
(transport == NULL) ||
transport->isClosed() ||
(protocolHandler == NULL);
}
PRInt64
Connection::id() const
{
CHECK_OBJECT_VALIDITY();
return this->connectionID;
}
void
Connection::setid(PRInt64 idArg)
{
CHECK_OBJECT_VALIDITY();
this->connectionID = idArg;
}
// exitConnection is called whenever the connection should be aborted.
// This does a hard connection shutdown. This does not send a GOODBYE
// message to the MQ Broker. If an exception occurs at the broker,
// this can also be called from ReadChannel::run (see ReadChannel for
// more information).
void
Connection::exitConnection(const MQError errorCode,
const PRBool calledFromReadChannel,
const PRBool abortConnection)
{
CHECK_OBJECT_VALIDITY();
LOG_FINE(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Connection::exitConnection(error=%d, fromReadChannel=%d, abort=%d) waitting for exitMonitor",
errorCode, calledFromReadChannel, abortConnection ));
exitMonitor.enter();
LOG_FINE(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Connection::exitConnection() acquired exitMonitor " ));
if (isAborted != PR_TRUE) {
isAborted = PR_TRUE;
if (this->pingTimer != NULL) {
this->pingTimer->terminate();
}
/** This shuts down the TCP connection (but doesn't close the
* socket). This causes ProtocolHandler::readPacket to return,
* which will cause ReadChannel::run to abort.
*/
transport->shutdown();
ackQTable.closeAll();
}
exitMonitor.exit();
if (calledFromReadChannel) {
receiveQTable.closeAll();
this->closeAllProducerFlow();
if (abortConnection == PR_TRUE) {
this->notifyExceptionListener(errorCode);
}
} else {
if (readChannel != NULL) {
// Singal and wait the ReadChannel thread to exit
readChannel->exitConnection();
}
}
LOG_FINE(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Connection::exitConnection finished" ));
}
MQError
Connection::ping()
{
CHECK_OBJECT_VALIDITY();
MQError errorCode = MQ_SUCCESS;
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
if (protocolHandler->getHasActivity()) {
ERRCHK( MQ_SUCCESS );
}
ERRCHK( protocolHandler->ping() );
Cleanup:
protocolHandler->clearHasActivity();
return errorCode;
}
/*
*
*/
MQError
Connection::stop()
{
CHECK_OBJECT_VALIDITY();
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
MQError errorCode = MQ_SUCCESS;
monitor.enter();
if (isStopped) {
LOG_FINE(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Stopping a stopped connection" ));
monitor.exit();
return MQ_SUCCESS;
} else {
LOG_FINE(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Stopping the connection" ));
}
ERRCHK( protocolHandler->stop(PR_FALSE, NULL) );
ERRCHK( this->stopSessions() );
isStopped = PR_TRUE;
Cleanup:
monitor.exit();
if (errorCode != MQ_SUCCESS) {
LOG_WARNING(( CODELOC, CONNECTION_LOG_MASK, this->id(),
MQ_CONNECTION_START_ERROR,
"Stopping the connection failed because '%s' (%d).",
errorStr(errorCode), errorCode ));
} else {
LOG_FINE(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Connection stopped" ));
}
return errorCode;
}
PRBool
Connection::getIsStopped() const
{
CHECK_OBJECT_VALIDITY();
return this->isStopped;
}
/*
*
*/
MQError
Connection::start()
{
CHECK_OBJECT_VALIDITY();
MQError errorCode = MQ_SUCCESS;
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
monitor.enter();
if (!isStopped) {
LOG_FINE(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Starting a started connection" ));
monitor.exit();
return MQ_SUCCESS;
} else {
LOG_FINE(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Starting the connection" ));
}
ERRCHK( protocolHandler->start(PR_FALSE, NULL) );
ERRCHK( this->startSessions() );
isStopped = PR_FALSE;
Cleanup:
monitor.exit();
if (errorCode != MQ_SUCCESS) {
LOG_WARNING(( CODELOC, CONNECTION_LOG_MASK, this->id(),
MQ_CONNECTION_START_ERROR,
"Starting the connection failed because '%s' (%d).",
errorStr(errorCode), errorCode ));
} else {
LOG_FINE(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Connection started" ));
}
return errorCode;
}
/*
*
*/
MQError
Connection::close()
{
CHECK_OBJECT_VALIDITY();
MQError errorCode = MQ_SUCCESS;
monitor.enter();
if (isClosed) {
LOG_FINE(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Skipping closing a closed connection" ));
monitor.exit();
return MQ_SUCCESS;
}
if (readChannel != NULL) {
PRThread * readerThread = readChannel->getReaderThread();
if (readerThread != NULL && PR_GetCurrentThread() == readerThread) {
monitor.exit();
return MQ_CONCURRENT_DEADLOCK;
}
}
errorCode = this->stop();
if (errorCode != MQ_SUCCESS && errorCode != MQ_BROKER_CONNECTION_CLOSED) {
cleanupConnection();
monitor.exit();
return errorCode;
}
// Close all the sessions (this will close all producers and consumers too)
errorCode = this->closeSessions();
if (errorCode != MQ_SUCCESS && errorCode != MQ_BROKER_CONNECTION_CLOSED) {
cleanupConnection();
monitor.exit();
return errorCode;
}
// Send GOODBYE to broker if we can
if ((protocolHandler != NULL) &&
(readChannel != NULL) &&
(readChannel->getInitializationError() == MQ_SUCCESS))
{
this->protocolHandler->goodBye(PR_TRUE);
}
// Shutdown the reader thread and the socket
this->exitConnection(MQ_SUCCESS, PR_FALSE, PR_FALSE);
// Close the transport connection to the broker. This closes the socket.
if (transport != NULL) {
transport->close();
}
LOG_FINE(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Connection closed" ));
this->isClosed = PR_TRUE;
monitor.exit();
return MQ_SUCCESS;
}
//must only be called in monitor
void
Connection::cleanupConnection()
{
// Shutdown the reader thread and the socket
this->exitConnection(MQ_SUCCESS, PR_FALSE, PR_FALSE);
// Close the transport connection to the broker. This closes the socket.
if (transport != NULL) {
transport->close();
}
}
MQError
Connection::getMetaData(Properties ** const metaProperties)
{
CHECK_OBJECT_VALIDITY();
MQError errorCode = MQ_SUCCESS;
Properties * props = NULL;
NULLCHK(metaProperties);
*metaProperties = NULL;
MEMCHK( props = new Properties );
ERRCHK( props->getInitializationError() );
ERRCHK( props->setStringProperty( MQ_NAME_PROPERTY, PRODUCT_NAME) );
ERRCHK( props->setStringProperty( MQ_VERSION_PROPERTY, PRODUCT_VERSION) );
ERRCHK( props->setIntegerProperty( MQ_MAJOR_VERSION_PROPERTY, PRODUCT_MAJOR_VERSION) );
ERRCHK( props->setIntegerProperty( MQ_MINOR_VERSION_PROPERTY, PRODUCT_MINOR_VERSION) );
ERRCHK( props->setIntegerProperty( MQ_MICRO_VERSION_PROPERTY, PRODUCT_MICRO_VERSION) );
ERRCHK( props->setIntegerProperty( MQ_SERVICE_PACK_PROPERTY, PRODUCT_SERVICE_PACK) );
ERRCHK( props->setIntegerProperty( MQ_UPDATE_RELEASE_PROPERTY, PRODUCT_UPDATE_RELEASE) );
*metaProperties = props;
return MQ_SUCCESS;
Cleanup:
DELETE( props );
MQ_ERROR_TRACE( "getMetaData", errorCode );
return errorCode;
}
/*
*
*/
MQError
Connection::setExceptionListenerCallback(
const MQConnectionExceptionListenerFunc exceptionListenerFunc,
void * exceptionListenerFuncData)
{
CHECK_OBJECT_VALIDITY();
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
this->exceptionListenerCallback = exceptionListenerFunc;
this->exceptionListenerCallbackData = exceptionListenerFuncData;
return MQ_SUCCESS;
}
/** Set the callback that allows the user to create the threads allocated by
* this connection. */
MQError
Connection::setCreateThreadCallback(const MQCreateThreadFunc createThreadFunc,
void * createThreadFuncData)
{
CHECK_OBJECT_VALIDITY();
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
this->createThreadCallback = createThreadFunc;
this->createThreadCallbackData = createThreadFuncData;
return MQ_SUCCESS;
}
/*
*
*/
MQError
Connection::addSession(const Session * const session)
{
CHECK_OBJECT_VALIDITY();
MQError errorCode = MQ_SUCCESS;
sessionsMonitor.enter();
errorCode = sessionVector.add((void*)session);
sessionsMonitor.exit();
return errorCode;
}
/*
*
*/
MQError
Connection::removeSession(const Session * const session)
{
CHECK_OBJECT_VALIDITY();
MQError errorCode = MQ_SUCCESS;
sessionsMonitor.enter();
errorCode = sessionVector.remove((void*)session);
sessionsMonitor.exit();
return errorCode;
}
/*
*
*/
MQError
Connection::getSession(const PRInt32 index, Session ** const session)
{
CHECK_OBJECT_VALIDITY();
MQError errorCode = MQ_SUCCESS;
sessionsMonitor.enter();
errorCode = sessionVector.get(index, (void**)session);
sessionsMonitor.exit();
return errorCode;
}
/*
*
*/
PRInt32
Connection::numSessions()
{
CHECK_OBJECT_VALIDITY();
PRInt32 size = 0;
sessionsMonitor.enter();
size = (PRInt32)sessionVector.size();
sessionsMonitor.exit();
return size;
}
/*
*
*/
MQError
Connection::createSession(const PRBool isTransacted,
const AckMode ackMode,
const ReceiveMode receiveMode,
const PRBool isXASession,
MQMessageListenerBAFunc beforeMessageListener,
MQMessageListenerBAFunc afterMessageListener,
void * callbackData,
Session ** const session)
{
CHECK_OBJECT_VALIDITY();
MQError errorCode = MQ_SUCCESS;
if (isXASession == PR_TRUE && this->getIsXA() == PR_FALSE) {
return MQ_NOT_XA_CONNECTION;
}
monitor.enter();
CNDCHK( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
NULLCHK( session );
*session = NULL;
// Create the session
if (isXASession == PR_FALSE) {
MEMCHK( *session = new Session(this, isTransacted, ackMode, receiveMode) );
} else {
MEMCHK( *session = new XASession(this, receiveMode,
beforeMessageListener, afterMessageListener, callbackData) );
}
ERRCHK( (*session)->getInitializationError() );
// Add it to the list of sessions
ERRCHK( addSession(*session) );
monitor.exit();
return MQ_SUCCESS;
Cleanup:
monitor.exit();
HANDLED_DELETE(*session);
LOG_WARNING(( CODELOC, CONNECTION_LOG_MASK, this->id(),
MQ_CONNECTION_CREATE_SESSION_ERROR,
"Creating a session failed because '%s' (%d).",
errorStr(errorCode), errorCode ));
return errorCode;
}
/*
*
*/
MQError
Connection::startSessions()
{
CHECK_OBJECT_VALIDITY();
MQError errorCode = MQ_SUCCESS, error = MQ_SUCCESS;
Session * session = NULL;
sessionsMonitor.enter();
for (int i = 0; i < this->numSessions(); i++) {
session = NULL;
this->getSession(i, &session);
ASSERT( session != NULL );
if (session != NULL) {
error = session->start();
if (error != MQ_SUCCESS) errorCode = error;
}
}
sessionsMonitor.exit();
return errorCode;
}
MQError
Connection::stopSessions()
{
CHECK_OBJECT_VALIDITY();
MQError errorCode = MQ_SUCCESS, error = MQ_SUCCESS;
Session * session = NULL;
sessionsMonitor.enter();
for (int i = 0; i < this->numSessions(); i++) {
session = NULL;
this->getSession(i, &session);
ASSERT( session != NULL );
if (session != NULL) {
error = session->stop();
if (error != MQ_SUCCESS) errorCode = error;
}
}
sessionsMonitor.exit();
return errorCode;
}
/*
*
*/
MQError
Connection::closeSessions()
{
CHECK_OBJECT_VALIDITY();
MQError errorCode = MQ_SUCCESS, error = MQ_SUCCESS;
Session * session = NULL;
sessionsMonitor.enter();
for (int i = 0; i < this->numSessions(); i++) {
session = NULL;
this->getSession(i, &session);
ASSERT( session != NULL );
if (session != NULL) {
error = session->close(PR_FALSE);
if (error != MQ_SUCCESS) errorCode = error;
}
}
sessionsMonitor.exit();
return errorCode;
}
/*
* This function has not been tested.
*/
void
Connection::notifyExceptionListener(const MQError error) const
{
CHECK_OBJECT_VALIDITY();
if (exceptionListenerCallback != NULL) {
invokeExceptionListenerCallback(this, error, this->exceptionListenerCallback,
this->exceptionListenerCallbackData);
}
}
/*
*
*/
MQError
Connection::startThread(Runnable * const threadToRun)
{
CHECK_OBJECT_VALIDITY();
RETURN_ERROR_IF_NULL( threadToRun );
if (createThreadCallback == NULL) {
PRThread * thread = PR_CreateThread(PR_SYSTEM_THREAD,
startThreadHelper,
threadToRun,
PR_PRIORITY_NORMAL,
PR_GLOBAL_THREAD,
PR_UNJOINABLE_THREAD,
0);
if (thread == NULL) {
LOG_SEVERE(( CODELOC, CONNECTION_LOG_MASK, this->id(),
MQ_COULD_NOT_CREATE_THREAD,
"Creating a thread failed, error=%d, oserror=%d", PR_GetError(), PR_GetOSError() ));
} else {
PRThreadScope scope = PR_GetThreadScope(thread);
if (scope == PR_GLOBAL_THREAD) {
LOG_FINE(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Connection started a thread with thread scope %s", "GLOBAL" ));
} else if (scope == PR_GLOBAL_BOUND_THREAD) {
LOG_FINE(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Connection started a thread with thread scope %s", "GLOBAL_BOUND" ));
} else {
LOG_FINE(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Connection started a thread with thread scope %s", "LOCAL" ));
}
}
RETURN_ERROR_IF( thread == NULL, MQ_COULD_NOT_CREATE_THREAD );
} else {
PRBool success;
success = invokeCreateThreadCallback(startThreadHelper, threadToRun,
createThreadCallback,
createThreadCallbackData);
RETURN_ERROR_IF( !success, MQ_COULD_NOT_CREATE_THREAD );
}
return MQ_SUCCESS;
}
/*
*
*/
static void
startThreadHelper(void *arg)
{
Runnable * runnable = (Runnable*)arg;
if (runnable != NULL) {
runnable->run();
}
}
/*
* static method
* Caller is responsible to free the return string
*/
char *
Connection::getUserAgent()
{
MQError errorCode = MQ_SUCCESS;
char * userAgent = NULL;
userAgent = new char[SYS_INFO_BUFFER_LENGTH*4+4];
CNDCHK( userAgent == NULL, MQ_OUT_OF_MEMORY );
SNPRINTF(userAgent, SYS_INFO_BUFFER_LENGTH, "%s/%s (C; ", PRODUCT_NAME, PRODUCT_VERSION);
NSPRCHK( PR_GetSystemInfo(PR_SI_SYSNAME, &userAgent[STRLEN(userAgent)], SYS_INFO_BUFFER_LENGTH) );
STRCAT( userAgent, " " );
NSPRCHK( PR_GetSystemInfo(PR_SI_RELEASE, &userAgent[STRLEN(userAgent)], SYS_INFO_BUFFER_LENGTH) );
STRCAT( userAgent, " " );
NSPRCHK( PR_GetSystemInfo(PR_SI_ARCHITECTURE, &userAgent[STRLEN(userAgent)], SYS_INFO_BUFFER_LENGTH) );
STRCAT( userAgent, ")" );
return userAgent;
Cleanup:
//XXX logging errorCode
return NULL;
}
MQError
Connection::versionCheck(PRBool mq)
{
MQError errorCode = MQ_SUCCESS;
if (Connection::nsprVersionChecked) return MQ_SUCCESS;
if (PR_VersionCheck(PR_VERSION) == PR_FALSE) {
LOG_SEVERE(( CODELOC, CONNECTION_LOG_MASK, NULL_CONN_ID, MQ_INCOMPATIBLE_LIBRARY,
"The version of the NSPR library linked to by this application is not compatible with the version supported by the MQ API (NSPR %s)", PR_VERSION ));
return MQ_INCOMPATIBLE_LIBRARY;
}
Connection::nsprVersionChecked = PR_TRUE;
return errorCode;
}
/*
*
*/
MQError
Connection::registerMessageProducer(const Session * const session,
const Destination * const destination,
PRInt64 * producerID)
{
CHECK_OBJECT_VALIDITY();
MQError errorCode = MQ_SUCCESS;
ProducerFlow * producerFlow = NULL;
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
NULLCHK( session );
NULLCHK( producerID );
MEMCHK( producerFlow = new ProducerFlow() );
ERRCHK( protocolHandler->registerMessageProducer(session, destination, producerFlow) );
*producerID = producerFlow->getProducerID();
ERRCHK( this->addProducerFlow(*producerID, producerFlow) );
return MQ_SUCCESS;
Cleanup:
DELETE( producerFlow );
return errorCode;
}
/*
*
*/
MQError
Connection::unregisterMessageProducer(PRInt64 producerID)
{
CHECK_OBJECT_VALIDITY();
MQError errorCode = MQ_SUCCESS;
errorCode = protocolHandler->unregisterMessageProducer(producerID);
if (errorCode == MQ_SUCCESS || errorCode == MQ_BROKER_CONNECTION_CLOSED) {
errorCode = this->removeProducerFlow(producerID);
}
ERRCHK( errorCode );
return MQ_SUCCESS;
Cleanup:
return errorCode;
}
/*
*
*/
MQError
Connection::registerMessageConsumer(MessageConsumer * const messageConsumer)
{
CHECK_OBJECT_VALIDITY();
MQError errorCode = MQ_SUCCESS;
//PRBool addedToTable = PR_FALSE;
CNDCHK( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
NULLCHK( messageConsumer );
/*
ERRCHK( this->addToReadQTable(messageConsumer->getConsumerID(),
messageConsumer->getReceiveQueue()) );
addedToTable = PR_TRUE;
LOG_FINER(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Connection::registerMessageConsumer(%d, 0x%p) added consumer to readQueue",
messageConsumer->getConsumerID(), messageConsumer->getReceiveQueue() ));
*/
ERRCHK( protocolHandler->registerMessageConsumer(messageConsumer) );
{
Long consumerIDLong(messageConsumer->getConsumerID());
LOG_FINER(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Connection::registerMessageConsumer(%s, 0x%p) successful",
consumerIDLong.toString() ));
}
return MQ_SUCCESS;
Cleanup:
/*
if (addedToTable) {
this->removeFromReadQTable(messageConsumer->getConsumerID());
LOG_FINER(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Connection::registerMessageConsumer(%d, 0x%p) failed",
messageConsumer->getConsumerID(), messageConsumer->getReceiveQueue() ));
}
*/
LOG_FINER(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Connection::registerMessageConsumer(0x%p) failed", messageConsumer ));
return errorCode;
}
/*
*
*/
MQError
Connection::unregisterMessageConsumer(MessageConsumer * const messageConsumer)
{
CHECK_OBJECT_VALIDITY();
MQError errorCode = MQ_SUCCESS;
MQError ecode = MQ_SUCCESS;
Long consumerIDLong;
NULLCHK( messageConsumer );
consumerIDLong.setValue(messageConsumer->getConsumerID());
errorCode = protocolHandler->unregisterMessageConsumer(messageConsumer);
ecode = this->removeFromReceiveQTable(messageConsumer->getConsumerID());
ERRCHK( errorCode );
LOG_FINER(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Connection::unregisterMessageConsumer(0x%p) succeeded. consumerID=%s",
messageConsumer, consumerIDLong.toString() ));
return MQ_SUCCESS;
Cleanup:
LOG_FINE(( CODELOC, CONNECTION_LOG_MASK, this->id(), MQ_SUCCESS,
"Connection::unregisterMessageConsumer(0x%p) failed because %d. consumerID=%s",
messageConsumer,
errorCode,
(messageConsumer == NULL) ? "-1" : consumerIDLong.toString() ));
return errorCode;
}
/*
*
*/
MQError
Connection::unsubscribeDurableConsumer(const UTF8String * const durableName)
{
CHECK_OBJECT_VALIDITY();
MQError errorCode = MQ_SUCCESS;
CNDCHK( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
NULLCHK( durableName );
ERRCHK( protocolHandler->unsubscribeDurableConsumer(durableName) );
return MQ_SUCCESS;
Cleanup:
return errorCode;
}
/*
*
*/
MQError
Connection::writeJMSMessage(const Session * const session,
Message * const message, PRInt64 producerID)
{
CHECK_OBJECT_VALIDITY();
MQError errorCode = MQ_SUCCESS;
ProducerFlow * producerFlow = NULL;
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
ERRCHK( this->getProducerFlow(producerID, &producerFlow) );
errorCode = producerFlow->checkFlowControl(message);
this->releaseProducerFlow(&producerFlow);
ERRCHK( errorCode );
ERRCHK( protocolHandler->writeJMSMessage(session, message) );
Cleanup:
return errorCode;
}
/*
*
*/
MQError
Connection::acknowledge(const Session * const session,
const PRUint8 * const ackBlock,
const PRInt32 ackBlockSize)
{
CHECK_OBJECT_VALIDITY();
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
return protocolHandler->acknowledge(session,
ackBlock, ackBlockSize);
}
/*
*
*/
MQError
Connection::acknowledgeExpired(const PRUint8 * const ackBlock,
const PRInt32 ackBlockSize)
{
CHECK_OBJECT_VALIDITY();
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
return protocolHandler->acknowledgeExpired(ackBlock, ackBlockSize);
}
MQError
Connection::redeliver(const Session * const session, PRBool setRedelivered,
const PRUint8 * const redeliverBlock,
const PRInt32 redeliverBlockSize)
{
CHECK_OBJECT_VALIDITY();
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
return protocolHandler->redeliver(session, setRedelivered,
redeliverBlock, redeliverBlockSize);
}
MQError
Connection::registerSession(Session * session)
{
CHECK_OBJECT_VALIDITY();
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
return protocolHandler->registerSession(session);
}
MQError
Connection::unregisterSession(PRInt64 sessionID)
{
CHECK_OBJECT_VALIDITY();
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
return protocolHandler->unregisterSession(sessionID);
}
MQError
Connection::startSession(const Session * session)
{
CHECK_OBJECT_VALIDITY();
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
return protocolHandler->start(PR_TRUE, session);
}
MQError
Connection::stopSession(const Session * session)
{
CHECK_OBJECT_VALIDITY();
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
return protocolHandler->stop(PR_TRUE, session);
}
MQError
Connection::startTransaction(PRInt64 sessionID, PRInt64 * transactionID)
{
CHECK_OBJECT_VALIDITY();
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
return protocolHandler->startTransaction(sessionID, PR_TRUE, NULL, 0L, transactionID);
}
MQError
Connection::startTransaction(XID *xid, long xaflags, PRInt64 * transactionID)
{
CHECK_OBJECT_VALIDITY();
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
return protocolHandler->startTransaction((PRInt64)0, PR_FALSE, xid, xaflags, transactionID);
}
MQError
Connection::endTransaction(PRInt64 transactionID, XID *xid, long xaflags)
{
CHECK_OBJECT_VALIDITY();
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
return protocolHandler->endTransaction(transactionID, xid, xaflags);
}
MQError
Connection::prepareTransaction(PRInt64 transactionID, XID *xid)
{
CHECK_OBJECT_VALIDITY();
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
return protocolHandler->prepareTransaction(transactionID, xid);
}
MQError
Connection::commitTransaction(PRInt64 transactionID, PRInt32 * const replyStatus)
{
CHECK_OBJECT_VALIDITY();
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
return protocolHandler->commitTransaction(transactionID, NULL, 0L, replyStatus);
}
MQError
Connection::commitTransaction(PRInt64 transactionID, XID *xid, long xaflags,
PRInt32 * const replyStatus)
{
CHECK_OBJECT_VALIDITY();
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
return protocolHandler->commitTransaction(transactionID, xid, xaflags, replyStatus);
}
MQError
Connection::rollbackTransaction(PRInt64 transactionID)
{
CHECK_OBJECT_VALIDITY();
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
return protocolHandler->rollbackTransaction(transactionID, NULL, PR_FALSE);
}
MQError
Connection::rollbackTransaction(PRInt64 transactionID, XID *xid)
{
CHECK_OBJECT_VALIDITY();
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
return protocolHandler->rollbackTransaction(transactionID, xid, PR_TRUE);
}
MQError
Connection::recoverTransaction(long xaflags, ObjectVector ** const xidv)
{
CHECK_OBJECT_VALIDITY();
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
return protocolHandler->recoverTransaction(xaflags, xidv);
}
/*
*
*/
MQError
Connection::resumeFlow()
{
CHECK_OBJECT_VALIDITY();
RETURN_ERROR_IF( this->getIsClosed(), MQ_BROKER_CONNECTION_CLOSED );
return protocolHandler->resumeFlow(PR_FALSE, 0);
}
/*
*
*/
void
Connection::messageReceived()
{
CHECK_OBJECT_VALIDITY();
if (this->getIsClosed()) {
return;
}
flowControl->messageReceived();
}
/*
*
*/
void
Connection::messageDelivered()
{
CHECK_OBJECT_VALIDITY();
if (this->getIsClosed()) {
return;
}
flowControl->messageDelivered();
}
/*
*
*/
void
Connection::requestResume()
{
CHECK_OBJECT_VALIDITY();
if (this->getIsClosed()) {
return;
}
flowControl->requestResume();
}
/*
*
*/
MQError
Connection::requestResumeConsumer(PRInt64 consumerID)
{
CHECK_OBJECT_VALIDITY();
if (this->getIsClosed()) { //XXX amy
return MQ_SUCCESS;
}
return protocolHandler->resumeFlow(PR_TRUE, consumerID);
}
/**
*
*/
PRInt32
Connection::msTimeoutToMicroSeconds(const PRInt32 timeoutMS)
{
if (timeoutMS == 0) {
return TRANSPORT_NO_TIMEOUT;
} else {
return 1000 * timeoutMS;
}
}
/*
*
*/
static const char * CONFIG_FILE = INPUT_FILE_DIR "config/connection.properties";
MQError
Connection::test(const PRInt32 simultaneousConnections)
{
MQError errorCode = MQ_SUCCESS;
Connection * connections = new Connection[simultaneousConnections];
Properties * connectionProperties = NULL;
PRBool deleteProperties = PR_TRUE;
UTF8String * usernameStr = NULL;
UTF8String * passwordStr = NULL;
int i;
MEMCHK( connections );
for (i = 0; i < simultaneousConnections; i++) {
MEMCHK( connectionProperties = new Properties );
ERRCHK( connectionProperties->readFromFile(CONFIG_FILE) );
static const char * username = "guest";
static const char * password = "guest";
usernameStr = new UTF8String(username);
passwordStr = new UTF8String(password);
if ((usernameStr == NULL) || (passwordStr == NULL)) {
DELETE( usernameStr );
DELETE( passwordStr );
ERRCHK( MQ_OUT_OF_MEMORY );
}
deleteProperties = PR_FALSE; // connectionProperties is owned by connections[i] now
ERRCHK( connections[i].openConnection(connectionProperties,
usernameStr,
passwordStr,
NULL, NULL, NULL,
NULL,
NULL) );
Session * session = NULL;
ERRCHK( connections[i].createSession(PR_FALSE,
CLIENT_ACKNOWLEDGE,
SESSION_SYNC_RECEIVE,
PR_FALSE,
NULL, NULL, NULL,
&session) );
// Test the session
Session::test(session);
}
for (i = 0; i < simultaneousConnections; i++) {
ERRCHK( connections[i].close() );
}
Cleanup:
if (deleteProperties) {
HANDLED_DELETE( connectionProperties );
}
DELETE_ARR( connections );
return errorCode;
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy