blob: 6a9a65c26d7f581b0d5ed11959b1ad529be03953 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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.
*/
using System;
using System.Collections.Generic;
using System.Text;
using Apache.NMS.ActiveMQ.Commands;
namespace Apache.NMS.ActiveMQ
{
public sealed class NetTxMessageConsumer : MessageConsumer
{
private readonly NetTxSession session;
private readonly NetTxTransactionContext transactionContext;
internal NetTxMessageConsumer(Session session, ConsumerId id, ActiveMQDestination destination,
string name, string selector, int prefetch, int maxPendingMessageCount,
bool noLocal, bool browser, bool dispatchAsync) :
base(session, id, destination, name, selector, prefetch,
maxPendingMessageCount, noLocal, browser, dispatchAsync)
{
this.session = session as NetTxSession;
this.transactionContext = session.TransactionContext as NetTxTransactionContext;
}
public override void Close()
{
if (this.Closed)
{
return;
}
lock (this.transactionContext.SyncRoot)
{
if (this.session.IsTransacted || this.session.TransactionContext.InTransaction)
{
Tracer.DebugFormat("Consumer {0} Registering new ConsumerCloseSynchronization",
this.ConsumerId);
this.session.TransactionContext.AddSynchronization(
new ConsumerCloseSynchronization(this));
}
else
{
Tracer.DebugFormat("Consumer {0} No Active TX closing normally.",
this.ConsumerId);
this.DoClose();
}
}
}
public override void BeforeMessageIsConsumed(MessageDispatch dispatch)
{
if (!IsAutoAcknowledgeBatch)
{
if (this.session.IsTransacted)
{
bool waitForDtcWaitHandle = false;
lock (this.transactionContext.SyncRoot)
{
// In the case where the consumer is operating in concert with a
// distributed TX manager we need to wait whenever the TX is being
// controlled by the DTC as it completes all operations async and
// we cannot start consumption again until all its tasks have completed.)
waitForDtcWaitHandle = this.transactionContext.InNetTransaction &&
this.transactionContext.NetTxState ==
NetTxTransactionContext.TxState.Pending;
}
if (waitForDtcWaitHandle)
{
this.transactionContext.DtcWaitHandle.WaitOne();
}
}
}
base.BeforeMessageIsConsumed(dispatch);
}
}
}